| 
									
										
										
										
											1991-02-19 12:39:46 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | /* Method object implementation */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1997-05-02 03:12:38 +00:00
										 |  |  | #include "Python.h"
 | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1997-08-05 02:11:41 +00:00
										 |  |  | static PyCFunctionObject *free_list = NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1997-05-02 03:12:38 +00:00
										 |  |  | PyObject * | 
					
						
							| 
									
										
										
										
											2000-07-09 06:03:25 +00:00
										 |  |  | PyCFunction_New(PyMethodDef *ml, PyObject *self) | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											1997-08-05 02:11:41 +00:00
										 |  |  | 	PyCFunctionObject *op; | 
					
						
							|  |  |  | 	op = free_list; | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | 	if (op != NULL) { | 
					
						
							| 
									
										
										
										
											1997-08-05 02:11:41 +00:00
										 |  |  | 		free_list = (PyCFunctionObject *)(op->m_self); | 
					
						
							| 
									
										
										
										
											2000-05-03 23:44:39 +00:00
										 |  |  | 		PyObject_INIT(op, &PyCFunction_Type); | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											1997-08-05 02:11:41 +00:00
										 |  |  | 	else { | 
					
						
							| 
									
										
										
										
											2002-03-18 20:44:53 +00:00
										 |  |  | 		op = PyObject_GC_New(PyCFunctionObject, &PyCFunction_Type); | 
					
						
							| 
									
										
										
										
											1997-08-05 02:11:41 +00:00
										 |  |  | 		if (op == NULL) | 
					
						
							|  |  |  | 			return NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	op->m_ml = ml; | 
					
						
							|  |  |  | 	Py_XINCREF(self); | 
					
						
							|  |  |  | 	op->m_self = self; | 
					
						
							| 
									
										
										
										
											2002-03-18 20:44:53 +00:00
										 |  |  | 	_PyObject_GC_TRACK(op); | 
					
						
							| 
									
										
										
										
											1997-05-02 03:12:38 +00:00
										 |  |  | 	return (PyObject *)op; | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1997-05-02 03:12:38 +00:00
										 |  |  | PyCFunction | 
					
						
							| 
									
										
										
										
											2000-07-09 06:03:25 +00:00
										 |  |  | PyCFunction_GetFunction(PyObject *op) | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											1997-05-02 03:12:38 +00:00
										 |  |  | 	if (!PyCFunction_Check(op)) { | 
					
						
							|  |  |  | 		PyErr_BadInternalCall(); | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | 		return NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											1997-05-02 03:12:38 +00:00
										 |  |  | 	return ((PyCFunctionObject *)op) -> m_ml -> ml_meth; | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1997-05-02 03:12:38 +00:00
										 |  |  | PyObject * | 
					
						
							| 
									
										
										
										
											2000-07-09 06:03:25 +00:00
										 |  |  | PyCFunction_GetSelf(PyObject *op) | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											1997-05-02 03:12:38 +00:00
										 |  |  | 	if (!PyCFunction_Check(op)) { | 
					
						
							|  |  |  | 		PyErr_BadInternalCall(); | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | 		return NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											1997-05-02 03:12:38 +00:00
										 |  |  | 	return ((PyCFunctionObject *)op) -> m_self; | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1991-12-16 13:07:24 +00:00
										 |  |  | int | 
					
						
							| 
									
										
										
										
											2000-07-09 06:03:25 +00:00
										 |  |  | PyCFunction_GetFlags(PyObject *op) | 
					
						
							| 
									
										
										
										
											1991-12-16 13:07:24 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											1997-05-02 03:12:38 +00:00
										 |  |  | 	if (!PyCFunction_Check(op)) { | 
					
						
							|  |  |  | 		PyErr_BadInternalCall(); | 
					
						
							| 
									
										
										
										
											1991-12-16 13:07:24 +00:00
										 |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											1997-05-02 03:12:38 +00:00
										 |  |  | 	return ((PyCFunctionObject *)op) -> m_ml -> ml_flags; | 
					
						
							| 
									
										
										
										
											1991-12-16 13:07:24 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-08-12 21:52:24 +00:00
										 |  |  | PyObject * | 
					
						
							|  |  |  | PyCFunction_Call(PyObject *func, PyObject *arg, PyObject *kw) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	PyCFunctionObject* f = (PyCFunctionObject*)func; | 
					
						
							|  |  |  | 	PyCFunction meth = PyCFunction_GET_FUNCTION(func); | 
					
						
							|  |  |  | 	PyObject *self = PyCFunction_GET_SELF(func); | 
					
						
							| 
									
										
										
										
											2002-03-28 05:33:33 +00:00
										 |  |  | 	int flags = PyCFunction_GET_FLAGS(func) & ~(METH_CLASS | METH_STATIC); | 
					
						
							| 
									
										
										
										
											2001-08-16 13:15:00 +00:00
										 |  |  | 	int size = PyTuple_GET_SIZE(arg); | 
					
						
							| 
									
										
										
										
											2001-08-12 21:52:24 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (flags & METH_KEYWORDS) { | 
					
						
							|  |  |  | 		return (*(PyCFunctionWithKeywords)meth)(self, arg, kw); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if (kw != NULL && PyDict_Size(kw) != 0) { | 
					
						
							|  |  |  | 		PyErr_Format(PyExc_TypeError, | 
					
						
							|  |  |  | 			     "%.200s() takes no keyword arguments", | 
					
						
							|  |  |  | 			     f->m_ml->ml_name); | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2001-08-16 13:15:00 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	switch (flags) { | 
					
						
							|  |  |  | 	case METH_VARARGS: | 
					
						
							| 
									
										
										
										
											2001-08-12 21:52:24 +00:00
										 |  |  | 		return (*meth)(self, arg); | 
					
						
							| 
									
										
										
										
											2001-08-16 13:15:00 +00:00
										 |  |  | 	case METH_NOARGS: | 
					
						
							|  |  |  | 		if (size == 0) | 
					
						
							|  |  |  | 			return (*meth)(self, NULL); | 
					
						
							|  |  |  | 		PyErr_Format(PyExc_TypeError, | 
					
						
							|  |  |  | 			     "%.200s() takes no arguments (%d given)", | 
					
						
							|  |  |  | 			     f->m_ml->ml_name, size); | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							|  |  |  | 	case METH_O: | 
					
						
							|  |  |  | 		if (size == 1) | 
					
						
							|  |  |  | 			return (*meth)(self, PyTuple_GET_ITEM(arg, 0)); | 
					
						
							|  |  |  | 		PyErr_Format(PyExc_TypeError, | 
					
						
							|  |  |  | 			     "%.200s() takes exactly one argument (%d given)", | 
					
						
							|  |  |  | 			     f->m_ml->ml_name, size); | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							|  |  |  | 	case METH_OLDARGS: | 
					
						
							| 
									
										
										
										
											2001-08-12 21:52:24 +00:00
										 |  |  | 		/* the really old style */ | 
					
						
							|  |  |  | 		if (size == 1) | 
					
						
							|  |  |  | 			arg = PyTuple_GET_ITEM(arg, 0); | 
					
						
							|  |  |  | 		else if (size == 0) | 
					
						
							|  |  |  | 			arg = NULL; | 
					
						
							|  |  |  | 		return (*meth)(self, arg); | 
					
						
							| 
									
										
										
										
											2001-08-16 13:15:00 +00:00
										 |  |  | 	default: | 
					
						
							|  |  |  | 		/* should never get here ??? */ | 
					
						
							|  |  |  | 		PyErr_BadInternalCall(); | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							| 
									
										
										
										
											2001-08-12 21:52:24 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | /* Methods (the standard built-in methods, that is) */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							| 
									
										
										
										
											2000-07-09 06:03:25 +00:00
										 |  |  | meth_dealloc(PyCFunctionObject *m) | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2002-03-18 20:44:53 +00:00
										 |  |  | 	_PyObject_GC_UNTRACK(m); | 
					
						
							| 
									
										
										
										
											1997-05-02 03:12:38 +00:00
										 |  |  | 	Py_XDECREF(m->m_self); | 
					
						
							| 
									
										
										
										
											1997-08-05 02:11:41 +00:00
										 |  |  | 	m->m_self = (PyObject *)free_list; | 
					
						
							|  |  |  | 	free_list = m; | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-08-02 04:15:00 +00:00
										 |  |  | static PyObject * | 
					
						
							|  |  |  | meth_get__doc__(PyCFunctionObject *m, void *closure) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	char *doc = m->m_ml->ml_doc; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (doc != NULL) | 
					
						
							|  |  |  | 		return PyString_FromString(doc); | 
					
						
							|  |  |  | 	Py_INCREF(Py_None); | 
					
						
							|  |  |  | 	return Py_None; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyObject * | 
					
						
							|  |  |  | meth_get__name__(PyCFunctionObject *m, void *closure) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return PyString_FromString(m->m_ml->ml_name); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-07-12 13:27:35 +00:00
										 |  |  | static int | 
					
						
							|  |  |  | meth_traverse(PyCFunctionObject *m, visitproc visit, void *arg) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if (m->m_self != NULL) | 
					
						
							|  |  |  | 		return visit(m->m_self, arg); | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 		return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1997-05-02 03:12:38 +00:00
										 |  |  | static PyObject * | 
					
						
							| 
									
										
										
										
											2001-08-02 04:15:00 +00:00
										 |  |  | meth_get__self__(PyCFunctionObject *m, void *closure) | 
					
						
							| 
									
										
										
										
											1995-01-07 12:34:58 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2001-08-02 04:15:00 +00:00
										 |  |  | 	PyObject *self; | 
					
						
							|  |  |  | 	if (PyEval_GetRestricted()) { | 
					
						
							|  |  |  | 		PyErr_SetString(PyExc_RuntimeError, | 
					
						
							|  |  |  | 			"method.__self__ not accessible in restricted mode"); | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							| 
									
										
										
										
											1995-01-07 12:34:58 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2001-08-02 04:15:00 +00:00
										 |  |  | 	self = m->m_self; | 
					
						
							|  |  |  | 	if (self == NULL) | 
					
						
							|  |  |  | 		self = Py_None; | 
					
						
							|  |  |  | 	Py_INCREF(self); | 
					
						
							|  |  |  | 	return self; | 
					
						
							| 
									
										
										
										
											1995-01-07 12:34:58 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-09-20 21:45:26 +00:00
										 |  |  | static PyGetSetDef meth_getsets [] = { | 
					
						
							| 
									
										
										
										
											2001-08-02 04:15:00 +00:00
										 |  |  | 	{"__doc__",  (getter)meth_get__doc__,  NULL, NULL}, | 
					
						
							|  |  |  | 	{"__name__", (getter)meth_get__name__, NULL, NULL}, | 
					
						
							|  |  |  | 	{"__self__", (getter)meth_get__self__, NULL, NULL}, | 
					
						
							|  |  |  | 	{0} | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1997-05-02 03:12:38 +00:00
										 |  |  | static PyObject * | 
					
						
							| 
									
										
										
										
											2000-07-09 06:03:25 +00:00
										 |  |  | meth_repr(PyCFunctionObject *m) | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	if (m->m_self == NULL) | 
					
						
							| 
									
										
										
										
											2001-08-24 18:34:26 +00:00
										 |  |  | 		return PyString_FromFormat("<built-in function %s>", | 
					
						
							|  |  |  | 					   m->m_ml->ml_name); | 
					
						
							|  |  |  | 	return PyString_FromFormat("<built-in method %s of %s object at %p>", | 
					
						
							|  |  |  | 				   m->m_ml->ml_name, | 
					
						
							|  |  |  | 				   m->m_self->ob_type->tp_name, | 
					
						
							|  |  |  | 				   m->m_self); | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1993-03-29 10:43:31 +00:00
										 |  |  | static int | 
					
						
							| 
									
										
										
										
											2000-07-09 06:03:25 +00:00
										 |  |  | meth_compare(PyCFunctionObject *a, PyCFunctionObject *b) | 
					
						
							| 
									
										
										
										
											1993-03-29 10:43:31 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	if (a->m_self != b->m_self) | 
					
						
							| 
									
										
										
										
											1998-12-04 18:52:55 +00:00
										 |  |  | 		return (a->m_self < b->m_self) ? -1 : 1; | 
					
						
							| 
									
										
										
										
											1995-01-07 12:34:58 +00:00
										 |  |  | 	if (a->m_ml->ml_meth == b->m_ml->ml_meth) | 
					
						
							| 
									
										
										
										
											1993-03-29 10:43:31 +00:00
										 |  |  | 		return 0; | 
					
						
							| 
									
										
										
										
											1995-01-07 12:34:58 +00:00
										 |  |  | 	if (strcmp(a->m_ml->ml_name, b->m_ml->ml_name) < 0) | 
					
						
							| 
									
										
										
										
											1993-03-29 10:43:31 +00:00
										 |  |  | 		return -1; | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 		return 1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static long | 
					
						
							| 
									
										
										
										
											2000-07-09 06:03:25 +00:00
										 |  |  | meth_hash(PyCFunctionObject *a) | 
					
						
							| 
									
										
										
										
											1993-03-29 10:43:31 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2000-06-29 19:17:04 +00:00
										 |  |  | 	long x,y; | 
					
						
							| 
									
										
										
										
											1993-03-29 10:43:31 +00:00
										 |  |  | 	if (a->m_self == NULL) | 
					
						
							|  |  |  | 		x = 0; | 
					
						
							|  |  |  | 	else { | 
					
						
							| 
									
										
										
										
											1997-05-02 03:12:38 +00:00
										 |  |  | 		x = PyObject_Hash(a->m_self); | 
					
						
							| 
									
										
										
										
											1993-03-29 10:43:31 +00:00
										 |  |  | 		if (x == -1) | 
					
						
							|  |  |  | 			return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2000-06-30 22:46:04 +00:00
										 |  |  | 	y = _Py_HashPointer((void*)(a->m_ml->ml_meth)); | 
					
						
							| 
									
										
										
										
											2000-06-29 19:17:04 +00:00
										 |  |  | 	if (y == -1) | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	x ^= y; | 
					
						
							|  |  |  | 	if (x == -1) | 
					
						
							|  |  |  | 		x = -2; | 
					
						
							|  |  |  | 	return x; | 
					
						
							| 
									
										
										
										
											1993-03-29 10:43:31 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-08-02 04:15:00 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1997-05-02 03:12:38 +00:00
										 |  |  | PyTypeObject PyCFunction_Type = { | 
					
						
							|  |  |  | 	PyObject_HEAD_INIT(&PyType_Type) | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | 	0, | 
					
						
							| 
									
										
										
										
											1992-09-17 17:54:56 +00:00
										 |  |  | 	"builtin_function_or_method", | 
					
						
							| 
									
										
										
										
											2002-03-18 20:44:53 +00:00
										 |  |  | 	sizeof(PyCFunctionObject), | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | 	0, | 
					
						
							| 
									
										
										
										
											2001-07-12 13:27:35 +00:00
										 |  |  | 	(destructor)meth_dealloc, 		/* tp_dealloc */ | 
					
						
							|  |  |  | 	0,					/* tp_print */ | 
					
						
							| 
									
										
										
										
											2001-08-02 04:15:00 +00:00
										 |  |  | 	0,					/* tp_getattr */ | 
					
						
							| 
									
										
										
										
											2001-07-12 13:27:35 +00:00
										 |  |  | 	0,					/* tp_setattr */ | 
					
						
							|  |  |  | 	(cmpfunc)meth_compare,			/* tp_compare */ | 
					
						
							|  |  |  | 	(reprfunc)meth_repr,			/* tp_repr */ | 
					
						
							|  |  |  | 	0,					/* tp_as_number */ | 
					
						
							|  |  |  | 	0,					/* tp_as_sequence */ | 
					
						
							|  |  |  | 	0,					/* tp_as_mapping */ | 
					
						
							|  |  |  | 	(hashfunc)meth_hash,			/* tp_hash */ | 
					
						
							| 
									
										
										
										
											2001-08-12 21:52:24 +00:00
										 |  |  | 	PyCFunction_Call,			/* tp_call */ | 
					
						
							| 
									
										
										
										
											2001-07-12 13:27:35 +00:00
										 |  |  | 	0,					/* tp_str */ | 
					
						
							| 
									
										
										
										
											2001-08-02 04:15:00 +00:00
										 |  |  | 	PyObject_GenericGetAttr,		/* tp_getattro */ | 
					
						
							| 
									
										
										
										
											2001-07-12 13:27:35 +00:00
										 |  |  | 	0,					/* tp_setattro */ | 
					
						
							|  |  |  | 	0,					/* tp_as_buffer */ | 
					
						
							| 
									
										
										
										
											2002-03-18 20:44:53 +00:00
										 |  |  | 	Py_TPFLAGS_DEFAULT | Py_TPFLAGS_HAVE_GC,/* tp_flags */ | 
					
						
							| 
									
										
										
										
											2001-07-12 13:27:35 +00:00
										 |  |  |  	0,					/* tp_doc */ | 
					
						
							|  |  |  |  	(traverseproc)meth_traverse,		/* tp_traverse */ | 
					
						
							| 
									
										
										
										
											2001-08-02 04:15:00 +00:00
										 |  |  | 	0,					/* tp_clear */ | 
					
						
							|  |  |  | 	0,					/* tp_richcompare */ | 
					
						
							|  |  |  | 	0,					/* tp_weaklistoffset */ | 
					
						
							|  |  |  | 	0,					/* tp_iter */ | 
					
						
							|  |  |  | 	0,					/* tp_iternext */ | 
					
						
							|  |  |  | 	0,					/* tp_methods */ | 
					
						
							|  |  |  | 	0,					/* tp_members */ | 
					
						
							|  |  |  | 	meth_getsets,				/* tp_getset */ | 
					
						
							|  |  |  | 	0,					/* tp_base */ | 
					
						
							|  |  |  | 	0,					/* tp_dict */ | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											1990-12-20 15:06:42 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1995-01-26 22:58:48 +00:00
										 |  |  | /* List all methods in a chain -- helper for findmethodinchain */ | 
					
						
							| 
									
										
										
										
											1991-10-20 20:21:15 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1997-05-02 03:12:38 +00:00
										 |  |  | static PyObject * | 
					
						
							| 
									
										
										
										
											2000-07-09 06:03:25 +00:00
										 |  |  | listmethodchain(PyMethodChain *chain) | 
					
						
							| 
									
										
										
										
											1991-10-20 20:21:15 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											1997-05-02 03:12:38 +00:00
										 |  |  | 	PyMethodChain *c; | 
					
						
							|  |  |  | 	PyMethodDef *ml; | 
					
						
							| 
									
										
										
										
											1991-10-20 20:21:15 +00:00
										 |  |  | 	int i, n; | 
					
						
							| 
									
										
										
										
											1997-05-02 03:12:38 +00:00
										 |  |  | 	PyObject *v; | 
					
						
							| 
									
										
										
										
											1995-01-26 22:58:48 +00:00
										 |  |  | 	 | 
					
						
							|  |  |  | 	n = 0; | 
					
						
							|  |  |  | 	for (c = chain; c != NULL; c = c->link) { | 
					
						
							|  |  |  | 		for (ml = c->methods; ml->ml_name != NULL; ml++) | 
					
						
							|  |  |  | 			n++; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											1997-05-02 03:12:38 +00:00
										 |  |  | 	v = PyList_New(n); | 
					
						
							| 
									
										
										
										
											1995-01-26 22:58:48 +00:00
										 |  |  | 	if (v == NULL) | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							|  |  |  | 	i = 0; | 
					
						
							|  |  |  | 	for (c = chain; c != NULL; c = c->link) { | 
					
						
							|  |  |  | 		for (ml = c->methods; ml->ml_name != NULL; ml++) { | 
					
						
							| 
									
										
										
										
											1997-05-02 03:12:38 +00:00
										 |  |  | 			PyList_SetItem(v, i, PyString_FromString(ml->ml_name)); | 
					
						
							| 
									
										
										
										
											1995-01-26 22:58:48 +00:00
										 |  |  | 			i++; | 
					
						
							| 
									
										
										
										
											1991-10-20 20:21:15 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											1997-05-02 03:12:38 +00:00
										 |  |  | 	if (PyErr_Occurred()) { | 
					
						
							|  |  |  | 		Py_DECREF(v); | 
					
						
							| 
									
										
										
										
											1995-01-26 22:58:48 +00:00
										 |  |  | 		return NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											1997-05-02 03:12:38 +00:00
										 |  |  | 	PyList_Sort(v); | 
					
						
							| 
									
										
										
										
											1991-10-20 20:21:15 +00:00
										 |  |  | 	return v; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1995-01-26 22:58:48 +00:00
										 |  |  | /* Find a method in a method chain */ | 
					
						
							| 
									
										
										
										
											1990-12-20 15:06:42 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1997-05-02 03:12:38 +00:00
										 |  |  | PyObject * | 
					
						
							| 
									
										
										
										
											2000-07-09 06:03:25 +00:00
										 |  |  | Py_FindMethodInChain(PyMethodChain *chain, PyObject *self, char *name) | 
					
						
							| 
									
										
										
										
											1990-12-20 15:06:42 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											1998-06-27 18:28:59 +00:00
										 |  |  | 	if (name[0] == '_' && name[1] == '_') { | 
					
						
							|  |  |  | 		if (strcmp(name, "__methods__") == 0) | 
					
						
							|  |  |  | 			return listmethodchain(chain); | 
					
						
							|  |  |  | 		if (strcmp(name, "__doc__") == 0) { | 
					
						
							|  |  |  | 			char *doc = self->ob_type->tp_doc; | 
					
						
							|  |  |  | 			if (doc != NULL) | 
					
						
							|  |  |  | 				return PyString_FromString(doc); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											1995-01-26 22:58:48 +00:00
										 |  |  | 	while (chain != NULL) { | 
					
						
							| 
									
										
										
										
											1997-05-02 03:12:38 +00:00
										 |  |  | 		PyMethodDef *ml = chain->methods; | 
					
						
							| 
									
										
										
										
											1995-01-26 22:58:48 +00:00
										 |  |  | 		for (; ml->ml_name != NULL; ml++) { | 
					
						
							|  |  |  | 			if (name[0] == ml->ml_name[0] && | 
					
						
							|  |  |  | 			    strcmp(name+1, ml->ml_name+1) == 0) | 
					
						
							| 
									
										
										
										
											1997-05-02 03:12:38 +00:00
										 |  |  | 				return PyCFunction_New(ml, self); | 
					
						
							| 
									
										
										
										
											1995-01-26 22:58:48 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 		chain = chain->link; | 
					
						
							| 
									
										
										
										
											1990-12-20 15:06:42 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											1997-05-02 03:12:38 +00:00
										 |  |  | 	PyErr_SetString(PyExc_AttributeError, name); | 
					
						
							| 
									
										
										
										
											1990-12-20 15:06:42 +00:00
										 |  |  | 	return NULL; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											1995-01-26 22:58:48 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* Find a method in a single method list */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1997-05-02 03:12:38 +00:00
										 |  |  | PyObject * | 
					
						
							| 
									
										
										
										
											2000-07-09 06:03:25 +00:00
										 |  |  | Py_FindMethod(PyMethodDef *methods, PyObject *self, char *name) | 
					
						
							| 
									
										
										
										
											1995-01-26 22:58:48 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											1997-05-02 03:12:38 +00:00
										 |  |  | 	PyMethodChain chain; | 
					
						
							| 
									
										
										
										
											1995-01-26 22:58:48 +00:00
										 |  |  | 	chain.methods = methods; | 
					
						
							|  |  |  | 	chain.link = NULL; | 
					
						
							| 
									
										
										
										
											1997-05-02 03:12:38 +00:00
										 |  |  | 	return Py_FindMethodInChain(&chain, self, name); | 
					
						
							| 
									
										
										
										
											1995-01-26 22:58:48 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											1997-08-05 02:11:41 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* Clear out the free list */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void | 
					
						
							| 
									
										
										
										
											2000-07-09 06:03:25 +00:00
										 |  |  | PyCFunction_Fini(void) | 
					
						
							| 
									
										
										
										
											1997-08-05 02:11:41 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	while (free_list) { | 
					
						
							|  |  |  | 		PyCFunctionObject *v = free_list; | 
					
						
							|  |  |  | 		free_list = (PyCFunctionObject *)(v->m_self); | 
					
						
							| 
									
										
										
										
											2002-03-18 20:44:53 +00:00
										 |  |  | 		PyObject_GC_Del(v); | 
					
						
							| 
									
										
										
										
											1997-08-05 02:11:41 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | } |