| 
									
										
										
										
											2006-08-17 05:42:55 +00:00
										 |  |  | /* Class object implementation (dead now except for methods) */ | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1997-05-02 03:12:38 +00:00
										 |  |  | #include "Python.h"
 | 
					
						
							| 
									
										
										
										
											2021-10-14 21:53:04 +02:00
										 |  |  | #include "pycore_call.h"          // _PyObject_VectorcallTstate()
 | 
					
						
							| 
									
										
										
										
											2023-08-24 20:25:22 +02:00
										 |  |  | #include "pycore_ceval.h"         // _PyEval_GetBuiltin()
 | 
					
						
							| 
									
										
										
										
											2025-01-12 14:01:49 +01:00
										 |  |  | #include "pycore_freelist.h"
 | 
					
						
							| 
									
										
										
										
											2018-11-21 22:27:47 +01:00
										 |  |  | #include "pycore_object.h"
 | 
					
						
							| 
									
										
										
										
											2019-11-08 10:05:17 +01:00
										 |  |  | #include "pycore_pyerrors.h"
 | 
					
						
							| 
									
										
										
										
											2020-04-15 02:35:41 +02:00
										 |  |  | #include "pycore_pystate.h"       // _PyThreadState_GET()
 | 
					
						
							| 
									
										
										
										
											2023-07-25 15:28:30 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1992-08-12 15:35:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-19 05:56:53 +03:00
										 |  |  | #include "clinic/classobject.c.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-04 12:00:00 +02:00
										 |  |  | #define _PyMethodObject_CAST(op) _Py_CAST(PyMethodObject*, (op))
 | 
					
						
							| 
									
										
										
										
											2006-07-27 21:53:35 +00:00
										 |  |  | #define TP_DESCR_GET(t) ((t)->tp_descr_get)
 | 
					
						
							| 
									
										
										
										
											2001-10-17 20:26:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-19 05:56:53 +03:00
										 |  |  | /*[clinic input]
 | 
					
						
							|  |  |  | class method "PyMethodObject *" "&PyMethod_Type" | 
					
						
							|  |  |  | [clinic start generated code]*/ | 
					
						
							|  |  |  | /*[clinic end generated code: output=da39a3ee5e6b4b0d input=b16e47edf6107c23]*/ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-10-10 18:11:30 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-09-05 22:52:50 +00:00
										 |  |  | PyObject * | 
					
						
							|  |  |  | PyMethod_Function(PyObject *im) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-05-09 15:52:27 +00:00
										 |  |  |     if (!PyMethod_Check(im)) { | 
					
						
							|  |  |  |         PyErr_BadInternalCall(); | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return ((PyMethodObject *)im)->im_func; | 
					
						
							| 
									
										
										
										
											2001-09-05 22:52:50 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | PyObject * | 
					
						
							|  |  |  | PyMethod_Self(PyObject *im) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-05-09 15:52:27 +00:00
										 |  |  |     if (!PyMethod_Check(im)) { | 
					
						
							|  |  |  |         PyErr_BadInternalCall(); | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return ((PyMethodObject *)im)->im_self; | 
					
						
							| 
									
										
										
										
											2001-09-05 22:52:50 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-29 20:31:52 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | static PyObject * | 
					
						
							|  |  |  | method_vectorcall(PyObject *method, PyObject *const *args, | 
					
						
							|  |  |  |                   size_t nargsf, PyObject *kwnames) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-02-17 19:09:15 +09:00
										 |  |  |     assert(Py_IS_TYPE(method, &PyMethod_Type)); | 
					
						
							| 
									
										
										
										
											2019-11-08 10:05:17 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     PyThreadState *tstate = _PyThreadState_GET(); | 
					
						
							|  |  |  |     PyObject *self = PyMethod_GET_SELF(method); | 
					
						
							|  |  |  |     PyObject *func = PyMethod_GET_FUNCTION(method); | 
					
						
							| 
									
										
										
										
											2019-05-29 20:31:52 +02:00
										 |  |  |     Py_ssize_t nargs = PyVectorcall_NARGS(nargsf); | 
					
						
							| 
									
										
										
										
											2023-07-27 15:27:11 +01:00
										 |  |  |     assert(nargs == 0 || args[nargs-1]); | 
					
						
							| 
									
										
										
										
											2019-05-29 20:31:52 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-08 10:05:17 +01:00
										 |  |  |     PyObject *result; | 
					
						
							| 
									
										
										
										
											2019-05-29 20:31:52 +02:00
										 |  |  |     if (nargsf & PY_VECTORCALL_ARGUMENTS_OFFSET) { | 
					
						
							|  |  |  |         /* PY_VECTORCALL_ARGUMENTS_OFFSET is set, so we are allowed to mutate the vector */ | 
					
						
							|  |  |  |         PyObject **newargs = (PyObject**)args - 1; | 
					
						
							|  |  |  |         nargs += 1; | 
					
						
							|  |  |  |         PyObject *tmp = newargs[0]; | 
					
						
							|  |  |  |         newargs[0] = self; | 
					
						
							| 
									
										
										
										
											2023-07-27 15:27:11 +01:00
										 |  |  |         assert(newargs[nargs-1]); | 
					
						
							| 
									
										
										
										
											2019-11-08 10:05:17 +01:00
										 |  |  |         result = _PyObject_VectorcallTstate(tstate, func, newargs, | 
					
						
							|  |  |  |                                             nargs, kwnames); | 
					
						
							| 
									
										
										
										
											2019-05-29 20:31:52 +02:00
										 |  |  |         newargs[0] = tmp; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							|  |  |  |         Py_ssize_t nkwargs = (kwnames == NULL) ? 0 : PyTuple_GET_SIZE(kwnames); | 
					
						
							|  |  |  |         Py_ssize_t totalargs = nargs + nkwargs; | 
					
						
							| 
									
										
										
										
											2019-07-03 12:54:00 +02:00
										 |  |  |         if (totalargs == 0) { | 
					
						
							| 
									
										
										
										
											2019-11-08 10:05:17 +01:00
										 |  |  |             return _PyObject_VectorcallTstate(tstate, func, &self, 1, NULL); | 
					
						
							| 
									
										
										
										
											2019-07-03 12:54:00 +02:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-18 10:56:53 +02:00
										 |  |  |         PyObject *newargs_stack[_PY_FASTCALL_SMALL_STACK]; | 
					
						
							| 
									
										
										
										
											2019-07-03 12:54:00 +02:00
										 |  |  |         PyObject **newargs; | 
					
						
							| 
									
										
										
										
											2019-06-18 10:56:53 +02:00
										 |  |  |         if (totalargs <= (Py_ssize_t)Py_ARRAY_LENGTH(newargs_stack) - 1) { | 
					
						
							|  |  |  |             newargs = newargs_stack; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							|  |  |  |             newargs = PyMem_Malloc((totalargs+1) * sizeof(PyObject *)); | 
					
						
							|  |  |  |             if (newargs == NULL) { | 
					
						
							| 
									
										
										
										
											2019-11-08 10:05:17 +01:00
										 |  |  |                 _PyErr_NoMemory(tstate); | 
					
						
							| 
									
										
										
										
											2019-06-18 10:56:53 +02:00
										 |  |  |                 return NULL; | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2019-05-29 20:31:52 +02:00
										 |  |  |         } | 
					
						
							|  |  |  |         /* use borrowed references */ | 
					
						
							|  |  |  |         newargs[0] = self; | 
					
						
							| 
									
										
										
										
											2019-07-03 12:54:00 +02:00
										 |  |  |         /* bpo-37138: since totalargs > 0, it's impossible that args is NULL.
 | 
					
						
							|  |  |  |          * We need this, since calling memcpy() with a NULL pointer is | 
					
						
							|  |  |  |          * undefined behaviour. */ | 
					
						
							|  |  |  |         assert(args != NULL); | 
					
						
							|  |  |  |         memcpy(newargs + 1, args, totalargs * sizeof(PyObject *)); | 
					
						
							| 
									
										
										
										
											2019-11-08 10:05:17 +01:00
										 |  |  |         result = _PyObject_VectorcallTstate(tstate, func, | 
					
						
							|  |  |  |                                             newargs, nargs+1, kwnames); | 
					
						
							| 
									
										
										
										
											2019-06-18 10:56:53 +02:00
										 |  |  |         if (newargs != newargs_stack) { | 
					
						
							|  |  |  |             PyMem_Free(newargs); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-05-29 20:31:52 +02:00
										 |  |  |     } | 
					
						
							|  |  |  |     return result; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-11-27 10:40:20 +00:00
										 |  |  | /* Method objects are used for bound instance methods returned by
 | 
					
						
							|  |  |  |    instancename.methodname. ClassName.methodname returns an ordinary | 
					
						
							|  |  |  |    function. | 
					
						
							| 
									
										
										
										
											1993-05-20 14:24:46 +00:00
										 |  |  | */ | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1997-05-02 03:12:38 +00:00
										 |  |  | PyObject * | 
					
						
							| 
									
										
										
										
											2007-11-27 10:40:20 +00:00
										 |  |  | PyMethod_New(PyObject *func, PyObject *self) | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2010-05-09 15:52:27 +00:00
										 |  |  |     if (self == NULL) { | 
					
						
							|  |  |  |         PyErr_BadInternalCall(); | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2025-01-12 14:01:49 +01:00
										 |  |  |     PyMethodObject *im = _Py_FREELIST_POP(PyMethodObject, pymethodobjects); | 
					
						
							| 
									
										
										
										
											2019-07-26 15:05:50 +09:00
										 |  |  |     if (im == NULL) { | 
					
						
							| 
									
										
										
										
											2025-01-12 14:01:49 +01:00
										 |  |  |         im = PyObject_GC_New(PyMethodObject, &PyMethod_Type); | 
					
						
							|  |  |  |         if (im == NULL) { | 
					
						
							|  |  |  |             return NULL; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2010-05-09 15:52:27 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |     im->im_weakreflist = NULL; | 
					
						
							| 
									
										
										
										
											2022-11-10 16:27:32 +01:00
										 |  |  |     im->im_func = Py_NewRef(func); | 
					
						
							|  |  |  |     im->im_self = Py_NewRef(self); | 
					
						
							| 
									
										
										
										
											2019-05-29 20:31:52 +02:00
										 |  |  |     im->vectorcall = method_vectorcall; | 
					
						
							| 
									
										
										
										
											2010-05-09 15:52:27 +00:00
										 |  |  |     _PyObject_GC_TRACK(im); | 
					
						
							|  |  |  |     return (PyObject *)im; | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-19 05:56:53 +03:00
										 |  |  | /*[clinic input]
 | 
					
						
							|  |  |  | method.__reduce__ | 
					
						
							|  |  |  | [clinic start generated code]*/ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-23 18:59:12 +01:00
										 |  |  | static PyObject * | 
					
						
							| 
									
										
										
										
											2022-04-19 05:56:53 +03:00
										 |  |  | method___reduce___impl(PyMethodObject *self) | 
					
						
							|  |  |  | /*[clinic end generated code: output=6c04506d0fa6fdcb input=143a0bf5e96de6e8]*/ | 
					
						
							| 
									
										
										
										
											2013-11-23 18:59:12 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2022-04-19 05:56:53 +03:00
										 |  |  |     PyObject *funcself = PyMethod_GET_SELF(self); | 
					
						
							|  |  |  |     PyObject *func = PyMethod_GET_FUNCTION(self); | 
					
						
							|  |  |  |     PyObject *funcname = PyObject_GetAttr(func, &_Py_ID(__name__)); | 
					
						
							| 
									
										
										
										
											2013-11-23 18:59:12 +01:00
										 |  |  |     if (funcname == NULL) { | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-02-08 13:39:07 -07:00
										 |  |  |     return Py_BuildValue( | 
					
						
							| 
									
										
										
										
											2022-04-19 05:56:53 +03:00
										 |  |  |             "N(ON)", _PyEval_GetBuiltin(&_Py_ID(getattr)), funcself, funcname); | 
					
						
							| 
									
										
										
										
											2013-11-23 18:59:12 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyMethodDef method_methods[] = { | 
					
						
							| 
									
										
										
										
											2022-04-19 05:56:53 +03:00
										 |  |  |     METHOD___REDUCE___METHODDEF | 
					
						
							| 
									
										
										
										
											2013-11-23 18:59:12 +01:00
										 |  |  |     {NULL, NULL} | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-09-18 03:53:24 +00:00
										 |  |  | /* Descriptors for PyMethod attributes */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-11-27 10:40:20 +00:00
										 |  |  | /* im_func and im_self are stored in the PyMethod object */ | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-12-11 19:56:40 +00:00
										 |  |  | #define MO_OFF(x) offsetof(PyMethodObject, x)
 | 
					
						
							| 
									
										
										
										
											1990-12-20 15:06:42 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-08-24 00:41:19 +00:00
										 |  |  | static PyMemberDef method_memberlist[] = { | 
					
						
							| 
									
										
										
										
											2023-07-25 15:28:30 +02:00
										 |  |  |     {"__func__", _Py_T_OBJECT, MO_OFF(im_func), Py_READONLY, | 
					
						
							| 
									
										
										
										
											2010-05-09 15:52:27 +00:00
										 |  |  |      "the function (or other callable) implementing a method"}, | 
					
						
							| 
									
										
										
										
											2023-07-25 15:28:30 +02:00
										 |  |  |     {"__self__", _Py_T_OBJECT, MO_OFF(im_self), Py_READONLY, | 
					
						
							| 
									
										
										
										
											2010-05-09 15:52:27 +00:00
										 |  |  |      "the instance to which a method is bound"}, | 
					
						
							|  |  |  |     {NULL}      /* Sentinel */ | 
					
						
							| 
									
										
										
										
											1990-12-20 15:06:42 +00:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-11-22 23:55:50 +00:00
										 |  |  | /* Christian Tismer argued convincingly that method attributes should
 | 
					
						
							|  |  |  |    (nearly) always override function attributes. | 
					
						
							|  |  |  |    The one exception is __doc__; there's a default __doc__ which | 
					
						
							|  |  |  |    should only be used for the class, not for instances */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyObject * | 
					
						
							| 
									
										
										
										
											2024-10-04 12:00:00 +02:00
										 |  |  | method_get_doc(PyObject *self, void *context) | 
					
						
							| 
									
										
										
										
											2003-11-22 23:55:50 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2024-10-04 12:00:00 +02:00
										 |  |  |     PyMethodObject *im = _PyMethodObject_CAST(self); | 
					
						
							| 
									
										
										
										
											2022-02-22 17:23:51 -07:00
										 |  |  |     return PyObject_GetAttr(im->im_func, &_Py_ID(__doc__)); | 
					
						
							| 
									
										
										
										
											2003-11-22 23:55:50 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-08-24 00:41:19 +00:00
										 |  |  | static PyGetSetDef method_getset[] = { | 
					
						
							| 
									
										
										
										
											2024-10-04 12:00:00 +02:00
										 |  |  |     {"__doc__", method_get_doc, NULL, NULL}, | 
					
						
							| 
									
										
										
										
											2010-05-09 15:52:27 +00:00
										 |  |  |     {0} | 
					
						
							| 
									
										
										
										
											2003-11-22 23:55:50 +00:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2001-01-15 20:40:19 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1997-05-02 03:12:38 +00:00
										 |  |  | static PyObject * | 
					
						
							| 
									
										
										
										
											2006-08-24 00:41:19 +00:00
										 |  |  | method_getattro(PyObject *obj, PyObject *name) | 
					
						
							| 
									
										
										
										
											1990-12-20 15:06:42 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2010-05-09 15:52:27 +00:00
										 |  |  |     PyMethodObject *im = (PyMethodObject *)obj; | 
					
						
							| 
									
										
										
										
											2020-02-07 03:04:21 +01:00
										 |  |  |     PyTypeObject *tp = Py_TYPE(obj); | 
					
						
							| 
									
										
										
										
											2010-05-09 15:52:27 +00:00
										 |  |  |     PyObject *descr = NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2023-04-27 16:19:43 -06:00
										 |  |  |         if (!_PyType_IsReady(tp)) { | 
					
						
							| 
									
										
										
										
											2010-05-09 15:52:27 +00:00
										 |  |  |             if (PyType_Ready(tp) < 0) | 
					
						
							|  |  |  |                 return NULL; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2024-05-06 10:50:35 -07:00
										 |  |  |         descr = _PyType_LookupRef(tp, name); | 
					
						
							| 
									
										
										
										
											2010-05-09 15:52:27 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (descr != NULL) { | 
					
						
							| 
									
										
										
										
											2020-02-07 03:04:21 +01:00
										 |  |  |         descrgetfunc f = TP_DESCR_GET(Py_TYPE(descr)); | 
					
						
							| 
									
										
										
										
											2024-05-06 10:50:35 -07:00
										 |  |  |         if (f != NULL) { | 
					
						
							|  |  |  |             PyObject *res = f(descr, obj, (PyObject *)Py_TYPE(obj)); | 
					
						
							|  |  |  |             Py_DECREF(descr); | 
					
						
							|  |  |  |             return res; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2010-05-09 15:52:27 +00:00
										 |  |  |         else { | 
					
						
							| 
									
										
										
										
											2024-05-06 10:50:35 -07:00
										 |  |  |             return descr; | 
					
						
							| 
									
										
										
										
											2010-05-09 15:52:27 +00:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return PyObject_GetAttr(im->im_func, name); | 
					
						
							| 
									
										
										
										
											1990-12-20 15:06:42 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-19 05:56:53 +03:00
										 |  |  | /*[clinic input]
 | 
					
						
							|  |  |  | @classmethod | 
					
						
							|  |  |  | method.__new__ as method_new | 
					
						
							|  |  |  |     function: object | 
					
						
							|  |  |  |     instance: object | 
					
						
							|  |  |  |     / | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Create a bound instance method object. | 
					
						
							|  |  |  | [clinic start generated code]*/ | 
					
						
							| 
									
										
										
										
											2002-06-14 20:41:17 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | static PyObject * | 
					
						
							| 
									
										
										
										
											2022-04-19 05:56:53 +03:00
										 |  |  | method_new_impl(PyTypeObject *type, PyObject *function, PyObject *instance) | 
					
						
							|  |  |  | /*[clinic end generated code: output=d33ef4ebf702e1f7 input=4e32facc3c3108ae]*/ | 
					
						
							| 
									
										
										
										
											2002-06-14 20:41:17 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2022-04-19 05:56:53 +03:00
										 |  |  |     if (!PyCallable_Check(function)) { | 
					
						
							| 
									
										
										
										
											2010-05-09 15:52:27 +00:00
										 |  |  |         PyErr_SetString(PyExc_TypeError, | 
					
						
							|  |  |  |                         "first argument must be callable"); | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-04-19 05:56:53 +03:00
										 |  |  |     if (instance == NULL || instance == Py_None) { | 
					
						
							| 
									
										
										
										
											2010-05-09 15:52:27 +00:00
										 |  |  |         PyErr_SetString(PyExc_TypeError, | 
					
						
							| 
									
										
										
										
											2022-04-19 05:56:53 +03:00
										 |  |  |             "instance must not be None"); | 
					
						
							| 
									
										
										
										
											2010-05-09 15:52:27 +00:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-19 05:56:53 +03:00
										 |  |  |     return PyMethod_New(function, instance); | 
					
						
							| 
									
										
										
										
											2002-06-14 20:41:17 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | static void | 
					
						
							| 
									
										
										
										
											2024-10-04 12:00:00 +02:00
										 |  |  | method_dealloc(PyObject *self) | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2024-10-04 12:00:00 +02:00
										 |  |  |     PyMethodObject *im = _PyMethodObject_CAST(self); | 
					
						
							| 
									
										
										
										
											2010-05-09 15:52:27 +00:00
										 |  |  |     _PyObject_GC_UNTRACK(im); | 
					
						
							|  |  |  |     if (im->im_weakreflist != NULL) | 
					
						
							|  |  |  |         PyObject_ClearWeakRefs((PyObject *)im); | 
					
						
							|  |  |  |     Py_DECREF(im->im_func); | 
					
						
							|  |  |  |     Py_XDECREF(im->im_self); | 
					
						
							| 
									
										
										
										
											2025-01-12 14:01:49 +01:00
										 |  |  |     assert(Py_IS_TYPE(self, &PyMethod_Type)); | 
					
						
							|  |  |  |     _Py_FREELIST_FREE(pymethodobjects, (PyObject *)im, PyObject_GC_Del); | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-08-24 00:41:19 +00:00
										 |  |  | static PyObject * | 
					
						
							|  |  |  | method_richcompare(PyObject *self, PyObject *other, int op) | 
					
						
							| 
									
										
										
										
											1992-09-03 20:39:51 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2010-05-09 15:52:27 +00:00
										 |  |  |     PyMethodObject *a, *b; | 
					
						
							|  |  |  |     PyObject *res; | 
					
						
							|  |  |  |     int eq; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if ((op != Py_EQ && op != Py_NE) || | 
					
						
							|  |  |  |         !PyMethod_Check(self) || | 
					
						
							|  |  |  |         !PyMethod_Check(other)) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2011-08-10 20:28:54 -05:00
										 |  |  |         Py_RETURN_NOTIMPLEMENTED; | 
					
						
							| 
									
										
										
										
											2010-05-09 15:52:27 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |     a = (PyMethodObject *)self; | 
					
						
							|  |  |  |     b = (PyMethodObject *)other; | 
					
						
							|  |  |  |     eq = PyObject_RichCompareBool(a->im_func, b->im_func, Py_EQ); | 
					
						
							|  |  |  |     if (eq == 1) { | 
					
						
							| 
									
										
										
										
											2018-07-31 09:18:24 +03:00
										 |  |  |         eq = (a->im_self == b->im_self); | 
					
						
							| 
									
										
										
										
											2010-05-09 15:52:27 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-07-31 09:18:24 +03:00
										 |  |  |     else if (eq < 0) | 
					
						
							| 
									
										
										
										
											2010-05-09 15:52:27 +00:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     if (op == Py_EQ) | 
					
						
							|  |  |  |         res = eq ? Py_True : Py_False; | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |         res = eq ? Py_False : Py_True; | 
					
						
							| 
									
										
										
										
											2022-11-10 16:27:32 +01:00
										 |  |  |     return Py_NewRef(res); | 
					
						
							| 
									
										
										
										
											1992-09-03 20:39:51 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1997-05-02 03:12:38 +00:00
										 |  |  | static PyObject * | 
					
						
							| 
									
										
										
										
											2024-10-04 12:00:00 +02:00
										 |  |  | method_repr(PyObject *op) | 
					
						
							| 
									
										
										
										
											1993-05-19 14:50:45 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2024-10-04 12:00:00 +02:00
										 |  |  |     PyMethodObject *a = _PyMethodObject_CAST(op); | 
					
						
							| 
									
										
										
										
											2010-05-09 15:52:27 +00:00
										 |  |  |     PyObject *self = a->im_self; | 
					
						
							|  |  |  |     PyObject *func = a->im_func; | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |     PyObject *funcname, *result; | 
					
						
							| 
									
										
										
										
											2014-08-20 18:41:57 -05:00
										 |  |  |     const char *defname = "?"; | 
					
						
							| 
									
										
										
										
											2010-05-09 15:52:27 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-12 08:57:10 +03:00
										 |  |  |     if (PyObject_GetOptionalAttr(func, &_Py_ID(__qualname__), &funcname) < 0 || | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         (funcname == NULL && | 
					
						
							| 
									
										
										
										
											2023-07-12 08:57:10 +03:00
										 |  |  |          PyObject_GetOptionalAttr(func, &_Py_ID(__name__), &funcname) < 0)) | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |     { | 
					
						
							|  |  |  |         return NULL; | 
					
						
							| 
									
										
										
										
											2014-08-20 18:41:57 -05:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2015-03-18 21:53:15 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-08-20 18:41:57 -05:00
										 |  |  |     if (funcname != NULL && !PyUnicode_Check(funcname)) { | 
					
						
							| 
									
										
										
										
											2022-11-23 14:57:50 +01:00
										 |  |  |         Py_SETREF(funcname, NULL); | 
					
						
							| 
									
										
										
										
											2010-05-09 15:52:27 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* XXX Shouldn't use repr()/%R here! */ | 
					
						
							| 
									
										
										
										
											2014-08-20 18:41:57 -05:00
										 |  |  |     result = PyUnicode_FromFormat("<bound method %V of %R>", | 
					
						
							| 
									
										
										
										
											2010-05-09 15:52:27 +00:00
										 |  |  |                                   funcname, defname, self); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Py_XDECREF(funcname); | 
					
						
							|  |  |  |     return result; | 
					
						
							| 
									
										
										
										
											1993-05-19 14:50:45 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-10-17 20:54:53 +00:00
										 |  |  | static Py_hash_t | 
					
						
							| 
									
										
										
										
											2024-10-04 12:00:00 +02:00
										 |  |  | method_hash(PyObject *self) | 
					
						
							| 
									
										
										
										
											1993-03-29 10:43:31 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2024-10-04 12:00:00 +02:00
										 |  |  |     PyMethodObject *a = _PyMethodObject_CAST(self); | 
					
						
							|  |  |  |     Py_hash_t x = PyObject_GenericHash(a->im_self); | 
					
						
							|  |  |  |     Py_hash_t y = PyObject_Hash(a->im_func); | 
					
						
							|  |  |  |     if (y == -1) { | 
					
						
							| 
									
										
										
										
											2010-05-09 15:52:27 +00:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2024-10-04 12:00:00 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-09 15:52:27 +00:00
										 |  |  |     x = x ^ y; | 
					
						
							| 
									
										
										
										
											2024-10-04 12:00:00 +02:00
										 |  |  |     if (x == -1) { | 
					
						
							| 
									
										
										
										
											2010-05-09 15:52:27 +00:00
										 |  |  |         x = -2; | 
					
						
							| 
									
										
										
										
											2024-10-04 12:00:00 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-05-09 15:52:27 +00:00
										 |  |  |     return x; | 
					
						
							| 
									
										
										
										
											1993-03-29 10:43:31 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-06-23 14:18:11 +00:00
										 |  |  | static int | 
					
						
							| 
									
										
										
										
											2024-10-04 12:00:00 +02:00
										 |  |  | method_traverse(PyObject *self, visitproc visit, void *arg) | 
					
						
							| 
									
										
										
										
											2000-06-23 14:18:11 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2024-10-04 12:00:00 +02:00
										 |  |  |     PyMethodObject *im = _PyMethodObject_CAST(self); | 
					
						
							| 
									
										
										
										
											2010-05-09 15:52:27 +00:00
										 |  |  |     Py_VISIT(im->im_func); | 
					
						
							|  |  |  |     Py_VISIT(im->im_self); | 
					
						
							|  |  |  |     return 0; | 
					
						
							| 
									
										
										
										
											2000-06-23 14:18:11 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-12-21 16:08:35 -06:00
										 |  |  | static PyObject * | 
					
						
							|  |  |  | method_descr_get(PyObject *meth, PyObject *obj, PyObject *cls) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     Py_INCREF(meth); | 
					
						
							|  |  |  |     return meth; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1997-05-02 03:12:38 +00:00
										 |  |  | PyTypeObject PyMethod_Type = { | 
					
						
							| 
									
										
										
										
											2010-05-09 15:52:27 +00:00
										 |  |  |     PyVarObject_HEAD_INIT(&PyType_Type, 0) | 
					
						
							| 
									
										
										
										
											2022-04-19 05:56:53 +03:00
										 |  |  |     .tp_name = "method", | 
					
						
							|  |  |  |     .tp_basicsize = sizeof(PyMethodObject), | 
					
						
							| 
									
										
										
										
											2024-10-04 12:00:00 +02:00
										 |  |  |     .tp_dealloc = method_dealloc, | 
					
						
							| 
									
										
										
										
											2022-04-19 05:56:53 +03:00
										 |  |  |     .tp_vectorcall_offset = offsetof(PyMethodObject, vectorcall), | 
					
						
							| 
									
										
										
										
											2024-10-04 12:00:00 +02:00
										 |  |  |     .tp_repr = method_repr, | 
					
						
							|  |  |  |     .tp_hash = method_hash, | 
					
						
							| 
									
										
										
										
											2022-04-19 05:56:53 +03:00
										 |  |  |     .tp_call = PyVectorcall_Call, | 
					
						
							|  |  |  |     .tp_getattro = method_getattro, | 
					
						
							|  |  |  |     .tp_setattro = PyObject_GenericSetAttr, | 
					
						
							|  |  |  |     .tp_flags = Py_TPFLAGS_DEFAULT | Py_TPFLAGS_HAVE_GC | | 
					
						
							|  |  |  |                 Py_TPFLAGS_HAVE_VECTORCALL, | 
					
						
							|  |  |  |     .tp_doc = method_new__doc__, | 
					
						
							| 
									
										
										
										
											2024-10-04 12:00:00 +02:00
										 |  |  |     .tp_traverse = method_traverse, | 
					
						
							| 
									
										
										
										
											2022-04-19 05:56:53 +03:00
										 |  |  |     .tp_richcompare = method_richcompare, | 
					
						
							|  |  |  |     .tp_weaklistoffset = offsetof(PyMethodObject, im_weakreflist), | 
					
						
							|  |  |  |     .tp_methods = method_methods, | 
					
						
							|  |  |  |     .tp_members = method_memberlist, | 
					
						
							|  |  |  |     .tp_getset = method_getset, | 
					
						
							| 
									
										
										
										
											2023-12-21 16:08:35 -06:00
										 |  |  |     .tp_descr_get = method_descr_get, | 
					
						
							| 
									
										
										
										
											2022-04-19 05:56:53 +03:00
										 |  |  |     .tp_new = method_new, | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											1997-08-05 02:06:53 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-12-11 19:56:40 +00:00
										 |  |  | /* ------------------------------------------------------------------------
 | 
					
						
							|  |  |  |  * instance method | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-19 05:56:53 +03:00
										 |  |  | /*[clinic input]
 | 
					
						
							|  |  |  | class instancemethod "PyInstanceMethodObject *" "&PyInstanceMethod_Type" | 
					
						
							|  |  |  | [clinic start generated code]*/ | 
					
						
							|  |  |  | /*[clinic end generated code: output=da39a3ee5e6b4b0d input=28c9762a9016f4d2]*/ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-12-11 19:56:40 +00:00
										 |  |  | PyObject * | 
					
						
							|  |  |  | PyInstanceMethod_New(PyObject *func) { | 
					
						
							| 
									
										
										
										
											2010-05-09 15:52:27 +00:00
										 |  |  |     PyInstanceMethodObject *method; | 
					
						
							|  |  |  |     method = PyObject_GC_New(PyInstanceMethodObject, | 
					
						
							|  |  |  |                              &PyInstanceMethod_Type); | 
					
						
							|  |  |  |     if (method == NULL) return NULL; | 
					
						
							| 
									
										
										
										
											2022-11-10 16:27:32 +01:00
										 |  |  |     method->func = Py_NewRef(func); | 
					
						
							| 
									
										
										
										
											2010-05-09 15:52:27 +00:00
										 |  |  |     _PyObject_GC_TRACK(method); | 
					
						
							|  |  |  |     return (PyObject *)method; | 
					
						
							| 
									
										
										
										
											2007-12-11 19:56:40 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | PyObject * | 
					
						
							|  |  |  | PyInstanceMethod_Function(PyObject *im) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-05-09 15:52:27 +00:00
										 |  |  |     if (!PyInstanceMethod_Check(im)) { | 
					
						
							|  |  |  |         PyErr_BadInternalCall(); | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return PyInstanceMethod_GET_FUNCTION(im); | 
					
						
							| 
									
										
										
										
											2007-12-11 19:56:40 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define IMO_OFF(x) offsetof(PyInstanceMethodObject, x)
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyMemberDef instancemethod_memberlist[] = { | 
					
						
							| 
									
										
										
										
											2023-07-25 15:28:30 +02:00
										 |  |  |     {"__func__", _Py_T_OBJECT, IMO_OFF(func), Py_READONLY, | 
					
						
							| 
									
										
										
										
											2010-05-09 15:52:27 +00:00
										 |  |  |      "the function (or other callable) implementing a method"}, | 
					
						
							|  |  |  |     {NULL}      /* Sentinel */ | 
					
						
							| 
									
										
										
										
											2007-12-11 19:56:40 +00:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyObject * | 
					
						
							|  |  |  | instancemethod_get_doc(PyObject *self, void *context) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2022-02-22 17:23:51 -07:00
										 |  |  |     return PyObject_GetAttr(PyInstanceMethod_GET_FUNCTION(self), | 
					
						
							|  |  |  |                             &_Py_ID(__doc__)); | 
					
						
							| 
									
										
										
										
											2007-12-11 19:56:40 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyGetSetDef instancemethod_getset[] = { | 
					
						
							| 
									
										
										
										
											2024-10-04 12:00:00 +02:00
										 |  |  |     {"__doc__", instancemethod_get_doc, NULL, NULL}, | 
					
						
							| 
									
										
										
										
											2010-05-09 15:52:27 +00:00
										 |  |  |     {0} | 
					
						
							| 
									
										
										
										
											2007-12-11 19:56:40 +00:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyObject * | 
					
						
							|  |  |  | instancemethod_getattro(PyObject *self, PyObject *name) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-02-07 03:04:21 +01:00
										 |  |  |     PyTypeObject *tp = Py_TYPE(self); | 
					
						
							| 
									
										
										
										
											2010-05-09 15:52:27 +00:00
										 |  |  |     PyObject *descr = NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-27 16:19:43 -06:00
										 |  |  |     if (!_PyType_IsReady(tp)) { | 
					
						
							| 
									
										
										
										
											2010-05-09 15:52:27 +00:00
										 |  |  |         if (PyType_Ready(tp) < 0) | 
					
						
							|  |  |  |             return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-05-06 10:50:35 -07:00
										 |  |  |     descr = _PyType_LookupRef(tp, name); | 
					
						
							| 
									
										
										
										
											2010-05-09 15:52:27 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (descr != NULL) { | 
					
						
							| 
									
										
										
										
											2020-02-07 03:04:21 +01:00
										 |  |  |         descrgetfunc f = TP_DESCR_GET(Py_TYPE(descr)); | 
					
						
							| 
									
										
										
										
											2024-05-06 10:50:35 -07:00
										 |  |  |         if (f != NULL) { | 
					
						
							|  |  |  |             PyObject *res = f(descr, self, (PyObject *)Py_TYPE(self)); | 
					
						
							|  |  |  |             Py_DECREF(descr); | 
					
						
							|  |  |  |             return res; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2010-05-09 15:52:27 +00:00
										 |  |  |         else { | 
					
						
							| 
									
										
										
										
											2024-05-06 10:50:35 -07:00
										 |  |  |             return descr; | 
					
						
							| 
									
										
										
										
											2010-05-09 15:52:27 +00:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return PyObject_GetAttr(PyInstanceMethod_GET_FUNCTION(self), name); | 
					
						
							| 
									
										
										
										
											2007-12-11 19:56:40 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							|  |  |  | instancemethod_dealloc(PyObject *self) { | 
					
						
							| 
									
										
										
										
											2010-05-09 15:52:27 +00:00
										 |  |  |     _PyObject_GC_UNTRACK(self); | 
					
						
							|  |  |  |     Py_DECREF(PyInstanceMethod_GET_FUNCTION(self)); | 
					
						
							|  |  |  |     PyObject_GC_Del(self); | 
					
						
							| 
									
										
										
										
											2007-12-11 19:56:40 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | instancemethod_traverse(PyObject *self, visitproc visit, void *arg) { | 
					
						
							| 
									
										
										
										
											2010-05-09 15:52:27 +00:00
										 |  |  |     Py_VISIT(PyInstanceMethod_GET_FUNCTION(self)); | 
					
						
							|  |  |  |     return 0; | 
					
						
							| 
									
										
										
										
											2007-12-11 19:56:40 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyObject * | 
					
						
							|  |  |  | instancemethod_call(PyObject *self, PyObject *arg, PyObject *kw) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2021-07-28 13:10:36 +00:00
										 |  |  |     return PyObject_Call(PyInstanceMethod_GET_FUNCTION(self), arg, kw); | 
					
						
							| 
									
										
										
										
											2007-12-11 19:56:40 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyObject * | 
					
						
							|  |  |  | instancemethod_descr_get(PyObject *descr, PyObject *obj, PyObject *type) { | 
					
						
							| 
									
										
										
										
											2013-08-13 20:18:52 +02:00
										 |  |  |     PyObject *func = PyInstanceMethod_GET_FUNCTION(descr); | 
					
						
							| 
									
										
										
										
											2010-05-09 15:52:27 +00:00
										 |  |  |     if (obj == NULL) { | 
					
						
							| 
									
										
										
										
											2022-11-10 16:27:32 +01:00
										 |  |  |         return Py_NewRef(func); | 
					
						
							| 
									
										
										
										
											2010-05-09 15:52:27 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |         return PyMethod_New(func, obj); | 
					
						
							| 
									
										
										
										
											2007-12-11 19:56:40 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyObject * | 
					
						
							|  |  |  | instancemethod_richcompare(PyObject *self, PyObject *other, int op) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-05-09 15:52:27 +00:00
										 |  |  |     PyInstanceMethodObject *a, *b; | 
					
						
							|  |  |  |     PyObject *res; | 
					
						
							|  |  |  |     int eq; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if ((op != Py_EQ && op != Py_NE) || | 
					
						
							|  |  |  |         !PyInstanceMethod_Check(self) || | 
					
						
							|  |  |  |         !PyInstanceMethod_Check(other)) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2011-08-10 20:28:54 -05:00
										 |  |  |         Py_RETURN_NOTIMPLEMENTED; | 
					
						
							| 
									
										
										
										
											2010-05-09 15:52:27 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |     a = (PyInstanceMethodObject *)self; | 
					
						
							|  |  |  |     b = (PyInstanceMethodObject *)other; | 
					
						
							|  |  |  |     eq = PyObject_RichCompareBool(a->func, b->func, Py_EQ); | 
					
						
							|  |  |  |     if (eq < 0) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     if (op == Py_EQ) | 
					
						
							|  |  |  |         res = eq ? Py_True : Py_False; | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |         res = eq ? Py_False : Py_True; | 
					
						
							| 
									
										
										
										
											2022-11-10 16:27:32 +01:00
										 |  |  |     return Py_NewRef(res); | 
					
						
							| 
									
										
										
										
											2007-12-11 19:56:40 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyObject * | 
					
						
							|  |  |  | instancemethod_repr(PyObject *self) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-05-09 15:52:27 +00:00
										 |  |  |     PyObject *func = PyInstanceMethod_Function(self); | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |     PyObject *funcname, *result; | 
					
						
							| 
									
										
										
										
											2017-11-11 13:06:26 +02:00
										 |  |  |     const char *defname = "?"; | 
					
						
							| 
									
										
										
										
											2010-05-09 15:52:27 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (func == NULL) { | 
					
						
							|  |  |  |         PyErr_BadInternalCall(); | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-12 08:57:10 +03:00
										 |  |  |     if (PyObject_GetOptionalAttr(func, &_Py_ID(__name__), &funcname) < 0) { | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |         return NULL; | 
					
						
							| 
									
										
										
										
											2010-05-09 15:52:27 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-01-25 10:49:40 +02:00
										 |  |  |     if (funcname != NULL && !PyUnicode_Check(funcname)) { | 
					
						
							| 
									
										
										
										
											2022-11-23 14:57:50 +01:00
										 |  |  |         Py_SETREF(funcname, NULL); | 
					
						
							| 
									
										
										
										
											2010-05-09 15:52:27 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     result = PyUnicode_FromFormat("<instancemethod %V at %p>", | 
					
						
							|  |  |  |                                   funcname, defname, self); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Py_XDECREF(funcname); | 
					
						
							|  |  |  |     return result; | 
					
						
							| 
									
										
										
										
											2007-12-11 19:56:40 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-19 05:56:53 +03:00
										 |  |  | /*[clinic input]
 | 
					
						
							|  |  |  | @classmethod | 
					
						
							|  |  |  | instancemethod.__new__ as instancemethod_new | 
					
						
							|  |  |  |     function: object | 
					
						
							|  |  |  |     / | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Bind a function to a class. | 
					
						
							|  |  |  | [clinic start generated code]*/ | 
					
						
							| 
									
										
										
										
											2007-12-11 19:56:40 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | static PyObject * | 
					
						
							| 
									
										
										
										
											2022-04-19 05:56:53 +03:00
										 |  |  | instancemethod_new_impl(PyTypeObject *type, PyObject *function) | 
					
						
							|  |  |  | /*[clinic end generated code: output=5e0397b2bdb750be input=cfc54e8b973664a8]*/ | 
					
						
							| 
									
										
										
										
											2007-12-11 19:56:40 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2022-04-19 05:56:53 +03:00
										 |  |  |     if (!PyCallable_Check(function)) { | 
					
						
							| 
									
										
										
										
											2010-05-09 15:52:27 +00:00
										 |  |  |         PyErr_SetString(PyExc_TypeError, | 
					
						
							|  |  |  |                         "first argument must be callable"); | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-19 05:56:53 +03:00
										 |  |  |     return PyInstanceMethod_New(function); | 
					
						
							| 
									
										
										
										
											2007-12-11 19:56:40 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | PyTypeObject PyInstanceMethod_Type = { | 
					
						
							| 
									
										
										
										
											2010-05-09 15:52:27 +00:00
										 |  |  |     PyVarObject_HEAD_INIT(&PyType_Type, 0) | 
					
						
							| 
									
										
										
										
											2022-04-19 05:56:53 +03:00
										 |  |  |     .tp_name = "instancemethod", | 
					
						
							|  |  |  |     .tp_basicsize = sizeof(PyInstanceMethodObject), | 
					
						
							|  |  |  |     .tp_dealloc = instancemethod_dealloc, | 
					
						
							| 
									
										
										
										
											2024-10-04 12:00:00 +02:00
										 |  |  |     .tp_repr = instancemethod_repr, | 
					
						
							| 
									
										
										
										
											2022-04-19 05:56:53 +03:00
										 |  |  |     .tp_call = instancemethod_call, | 
					
						
							|  |  |  |     .tp_getattro = instancemethod_getattro, | 
					
						
							|  |  |  |     .tp_setattro = PyObject_GenericSetAttr, | 
					
						
							|  |  |  |     .tp_flags = Py_TPFLAGS_DEFAULT | Py_TPFLAGS_HAVE_GC, | 
					
						
							|  |  |  |     .tp_doc = instancemethod_new__doc__, | 
					
						
							|  |  |  |     .tp_traverse = instancemethod_traverse, | 
					
						
							|  |  |  |     .tp_richcompare = instancemethod_richcompare, | 
					
						
							|  |  |  |     .tp_members = instancemethod_memberlist, | 
					
						
							|  |  |  |     .tp_getset = instancemethod_getset, | 
					
						
							|  |  |  |     .tp_descr_get = instancemethod_descr_get, | 
					
						
							|  |  |  |     .tp_new = instancemethod_new, | 
					
						
							| 
									
										
										
										
											2007-12-11 19:56:40 +00:00
										 |  |  | }; |