| 
									
										
										
										
											1991-02-19 12:39:46 +00:00
										 |  |  | /***********************************************************
 | 
					
						
							| 
									
										
										
										
											1993-03-16 12:15:04 +00:00
										 |  |  | Copyright 1991, 1992, 1993 by Stichting Mathematisch Centrum, | 
					
						
							|  |  |  | Amsterdam, The Netherlands. | 
					
						
							| 
									
										
										
										
											1991-02-19 12:39:46 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |                         All Rights Reserved | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Permission to use, copy, modify, and distribute this software and its  | 
					
						
							|  |  |  | documentation for any purpose and without fee is hereby granted,  | 
					
						
							|  |  |  | provided that the above copyright notice appear in all copies and that | 
					
						
							|  |  |  | both that copyright notice and this permission notice appear in  | 
					
						
							|  |  |  | supporting documentation, and that the names of Stichting Mathematisch | 
					
						
							|  |  |  | Centrum or CWI not be used in advertising or publicity pertaining to | 
					
						
							|  |  |  | distribution of the software without specific, written prior permission. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | STICHTING MATHEMATISCH CENTRUM DISCLAIMS ALL WARRANTIES WITH REGARD TO | 
					
						
							|  |  |  | THIS SOFTWARE, INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND | 
					
						
							|  |  |  | FITNESS, IN NO EVENT SHALL STICHTING MATHEMATISCH CENTRUM BE LIABLE | 
					
						
							|  |  |  | FOR ANY SPECIAL, INDIRECT OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES | 
					
						
							|  |  |  | WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN | 
					
						
							|  |  |  | ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT | 
					
						
							|  |  |  | OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ******************************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | /* Class object implementation */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1990-12-20 15:06:42 +00:00
										 |  |  | #include "allobjects.h"
 | 
					
						
							| 
									
										
										
										
											1992-08-12 15:35:34 +00:00
										 |  |  | #include "modsupport.h"
 | 
					
						
							| 
									
										
										
										
											1990-12-20 15:06:42 +00:00
										 |  |  | #include "structmember.h"
 | 
					
						
							| 
									
										
										
										
											1992-08-12 15:35:34 +00:00
										 |  |  | #include "ceval.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | object * | 
					
						
							| 
									
										
										
										
											1993-05-20 14:24:46 +00:00
										 |  |  | newclassobject(bases, dict, name) | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | 	object *bases; /* NULL or tuple of classobjects! */ | 
					
						
							| 
									
										
										
										
											1993-05-20 14:24:46 +00:00
										 |  |  | 	object *dict; | 
					
						
							| 
									
										
										
										
											1991-10-20 20:11:48 +00:00
										 |  |  | 	object *name; /* String; NULL if unknown */ | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											1993-05-20 14:24:46 +00:00
										 |  |  | 	int pos; | 
					
						
							|  |  |  | 	object *key, *value; | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | 	classobject *op; | 
					
						
							| 
									
										
										
										
											1991-12-10 13:53:23 +00:00
										 |  |  | 	if (bases == NULL) { | 
					
						
							|  |  |  | 		bases = newtupleobject(0); | 
					
						
							|  |  |  | 		if (bases == NULL) | 
					
						
							| 
									
										
										
										
											1993-03-16 12:15:04 +00:00
										 |  |  | 			return NULL; | 
					
						
							| 
									
										
										
										
											1991-12-10 13:53:23 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 		INCREF(bases); | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | 	op = NEWOBJ(classobject, &Classtype); | 
					
						
							| 
									
										
										
										
											1991-12-10 13:53:23 +00:00
										 |  |  | 	if (op == NULL) { | 
					
						
							|  |  |  | 		DECREF(bases); | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | 		return NULL; | 
					
						
							| 
									
										
										
										
											1991-12-10 13:53:23 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | 	op->cl_bases = bases; | 
					
						
							| 
									
										
										
										
											1993-05-20 14:24:46 +00:00
										 |  |  | 	INCREF(dict); | 
					
						
							|  |  |  | 	op->cl_dict = dict; | 
					
						
							| 
									
										
										
										
											1991-10-20 20:11:48 +00:00
										 |  |  | 	XINCREF(name); | 
					
						
							|  |  |  | 	op->cl_name = name; | 
					
						
							| 
									
										
										
										
											1993-05-20 14:24:46 +00:00
										 |  |  | 	pos = 0; | 
					
						
							|  |  |  | 	while (mappinggetnext(dict, &pos, &key, &value)) { | 
					
						
							|  |  |  | 		if (is_accessobject(value)) | 
					
						
							|  |  |  | 			setaccessowner(value, (object *)op); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | 	return (object *) op; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Class methods */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							|  |  |  | class_dealloc(op) | 
					
						
							|  |  |  | 	classobject *op; | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											1991-12-10 13:53:23 +00:00
										 |  |  | 	DECREF(op->cl_bases); | 
					
						
							| 
									
										
										
										
											1993-05-20 14:24:46 +00:00
										 |  |  | 	DECREF(op->cl_dict); | 
					
						
							| 
									
										
										
										
											1991-10-20 20:11:48 +00:00
										 |  |  | 	XDECREF(op->cl_name); | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | 	free((ANY *)op); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1993-05-20 14:24:46 +00:00
										 |  |  | static object * | 
					
						
							|  |  |  | class_lookup(cp, name, pclass) | 
					
						
							|  |  |  | 	classobject *cp; | 
					
						
							|  |  |  | 	char *name; | 
					
						
							|  |  |  | 	classobject **pclass; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int i, n; | 
					
						
							|  |  |  | 	object *value = dictlookup(cp->cl_dict, name); | 
					
						
							|  |  |  | 	if (value != NULL) { | 
					
						
							|  |  |  | 		*pclass = cp; | 
					
						
							|  |  |  | 		return value; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	n = gettuplesize(cp->cl_bases); | 
					
						
							|  |  |  | 	for (i = 0; i < n; i++) { | 
					
						
							|  |  |  | 		object *v = class_lookup((classobject *) | 
					
						
							|  |  |  | 				 gettupleitem(cp->cl_bases, i), name, pclass); | 
					
						
							|  |  |  | 		if (v != NULL) | 
					
						
							|  |  |  | 			return v; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | static object * | 
					
						
							|  |  |  | class_getattr(op, name) | 
					
						
							|  |  |  | 	register classobject *op; | 
					
						
							|  |  |  | 	register char *name; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	register object *v; | 
					
						
							| 
									
										
										
										
											1993-05-20 14:24:46 +00:00
										 |  |  | 	object *class; | 
					
						
							| 
									
										
										
										
											1991-10-20 20:11:48 +00:00
										 |  |  | 	if (strcmp(name, "__dict__") == 0) { | 
					
						
							| 
									
										
										
										
											1993-05-20 14:24:46 +00:00
										 |  |  | 		INCREF(op->cl_dict); | 
					
						
							|  |  |  | 		return op->cl_dict; | 
					
						
							| 
									
										
										
										
											1991-10-20 20:11:48 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	if (strcmp(name, "__bases__") == 0) { | 
					
						
							|  |  |  | 		INCREF(op->cl_bases); | 
					
						
							|  |  |  | 		return op->cl_bases; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if (strcmp(name, "__name__") == 0) { | 
					
						
							|  |  |  | 		if (op->cl_name == NULL) | 
					
						
							|  |  |  | 			v = None; | 
					
						
							|  |  |  | 		else | 
					
						
							|  |  |  | 			v = op->cl_name; | 
					
						
							|  |  |  | 		INCREF(v); | 
					
						
							|  |  |  | 		return v; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											1993-05-20 14:24:46 +00:00
										 |  |  | 	v = class_lookup(op, name, &class); | 
					
						
							| 
									
										
										
										
											1993-05-21 19:56:10 +00:00
										 |  |  | 	if (v == NULL) { | 
					
						
							|  |  |  | 		err_setstr(AttributeError, name); | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if (is_accessobject(v)) { | 
					
						
							| 
									
										
										
										
											1993-05-25 09:38:27 +00:00
										 |  |  | 		v = getaccessvalue(v, getowner()); | 
					
						
							| 
									
										
										
										
											1993-05-21 19:56:10 +00:00
										 |  |  | 		if (v == NULL) | 
					
						
							|  |  |  | 			return NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 		INCREF(v); | 
					
						
							|  |  |  | 	if (is_funcobject(v)) { | 
					
						
							|  |  |  | 		object *w = newinstancemethodobject(v, (object *)NULL, | 
					
						
							| 
									
										
										
										
											1993-05-20 14:24:46 +00:00
										 |  |  | 						    (object *)class); | 
					
						
							| 
									
										
										
										
											1993-05-21 19:56:10 +00:00
										 |  |  | 		DECREF(v); | 
					
						
							|  |  |  | 		v = w; | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											1993-05-21 19:56:10 +00:00
										 |  |  | 	return v; | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1991-10-20 20:11:48 +00:00
										 |  |  | static int | 
					
						
							|  |  |  | class_setattr(op, name, v) | 
					
						
							|  |  |  | 	classobject *op; | 
					
						
							|  |  |  | 	char *name; | 
					
						
							|  |  |  | 	object *v; | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											1993-05-19 14:50:45 +00:00
										 |  |  | 	object *ac; | 
					
						
							| 
									
										
										
										
											1992-04-05 14:24:32 +00:00
										 |  |  | 	if (name[0] == '_' && name[1] == '_') { | 
					
						
							|  |  |  | 		int n = strlen(name); | 
					
						
							|  |  |  | 		if (name[n-1] == '_' && name[n-2] == '_') { | 
					
						
							|  |  |  | 			err_setstr(TypeError, "read-only special attribute"); | 
					
						
							|  |  |  | 			return -1; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											1993-05-20 14:24:46 +00:00
										 |  |  | 	ac = dictlookup(op->cl_dict, name); | 
					
						
							| 
									
										
										
										
											1993-05-19 14:50:45 +00:00
										 |  |  | 	if (ac != NULL && is_accessobject(ac)) | 
					
						
							| 
									
										
										
										
											1993-05-25 09:38:27 +00:00
										 |  |  | 		return setaccessvalue(ac, getowner(), v); | 
					
						
							| 
									
										
										
										
											1992-09-04 09:45:18 +00:00
										 |  |  | 	if (v == NULL) { | 
					
						
							| 
									
										
										
										
											1993-05-20 14:24:46 +00:00
										 |  |  | 		int rv = dictremove(op->cl_dict, name); | 
					
						
							| 
									
										
										
										
											1992-09-04 09:45:18 +00:00
										 |  |  | 		if (rv < 0) | 
					
						
							|  |  |  | 			err_setstr(AttributeError, | 
					
						
							|  |  |  | 				   "delete non-existing class attribute"); | 
					
						
							|  |  |  | 		return rv; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											1991-10-20 20:11:48 +00:00
										 |  |  | 	else | 
					
						
							| 
									
										
										
										
											1993-05-20 14:24:46 +00:00
										 |  |  | 		return dictinsert(op->cl_dict, name, v); | 
					
						
							| 
									
										
										
										
											1991-10-20 20:11:48 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1993-05-19 14:50:45 +00:00
										 |  |  | static object * | 
					
						
							|  |  |  | class_repr(op) | 
					
						
							|  |  |  | 	classobject *op; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	char buf[140]; | 
					
						
							|  |  |  | 	char *name; | 
					
						
							|  |  |  | 	if (op->cl_name == NULL || !is_stringobject(op->cl_name)) | 
					
						
							|  |  |  | 		name = "?"; | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 		name = getstringvalue(op->cl_name); | 
					
						
							|  |  |  | 	sprintf(buf, "<class %.100s at %lx>", name, (long)op); | 
					
						
							|  |  |  | 	return newstringobject(buf); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | typeobject Classtype = { | 
					
						
							|  |  |  | 	OB_HEAD_INIT(&Typetype) | 
					
						
							|  |  |  | 	0, | 
					
						
							|  |  |  | 	"class", | 
					
						
							|  |  |  | 	sizeof(classobject), | 
					
						
							|  |  |  | 	0, | 
					
						
							|  |  |  | 	class_dealloc,	/*tp_dealloc*/ | 
					
						
							|  |  |  | 	0,		/*tp_print*/ | 
					
						
							|  |  |  | 	class_getattr,	/*tp_getattr*/ | 
					
						
							| 
									
										
										
										
											1991-10-20 20:11:48 +00:00
										 |  |  | 	class_setattr,	/*tp_setattr*/ | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | 	0,		/*tp_compare*/ | 
					
						
							| 
									
										
										
										
											1993-05-19 14:50:45 +00:00
										 |  |  | 	class_repr,	/*tp_repr*/ | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | 	0,		/*tp_as_number*/ | 
					
						
							|  |  |  | 	0,		/*tp_as_sequence*/ | 
					
						
							|  |  |  | 	0,		/*tp_as_mapping*/ | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1993-05-20 14:24:46 +00:00
										 |  |  | int | 
					
						
							|  |  |  | issubclass(class, base) | 
					
						
							|  |  |  | 	object *class; | 
					
						
							|  |  |  | 	object *base; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int i, n; | 
					
						
							|  |  |  | 	classobject *cp; | 
					
						
							|  |  |  | 	if (class == base) | 
					
						
							|  |  |  | 		return 1; | 
					
						
							| 
									
										
										
										
											1993-05-25 09:38:27 +00:00
										 |  |  | 	if (class == NULL || !is_classobject(class)) | 
					
						
							|  |  |  | 		return 0; | 
					
						
							| 
									
										
										
										
											1993-05-20 14:24:46 +00:00
										 |  |  | 	cp = (classobject *)class; | 
					
						
							|  |  |  | 	n = gettuplesize(cp->cl_bases); | 
					
						
							|  |  |  | 	for (i = 0; i < n; i++) { | 
					
						
							|  |  |  | 		if (issubclass(gettupleitem(cp->cl_bases, i), base)) | 
					
						
							|  |  |  | 			return 1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1993-05-20 14:24:46 +00:00
										 |  |  | /* Instance objects */ | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1993-05-19 14:50:45 +00:00
										 |  |  | static object *instance_getattr PROTO((instanceobject *, char *)); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1993-05-20 14:24:46 +00:00
										 |  |  | static int | 
					
						
							|  |  |  | addaccess(class, inst) | 
					
						
							|  |  |  | 	classobject *class; | 
					
						
							|  |  |  | 	instanceobject *inst; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int i, n, pos, ret; | 
					
						
							|  |  |  | 	object *key, *value, *ac; | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	n = gettuplesize(class->cl_bases); | 
					
						
							|  |  |  | 	for (i = 0; i < n; i++) { | 
					
						
							|  |  |  | 		if (addaccess(gettupleitem(class->cl_bases, i), inst) < 0) | 
					
						
							|  |  |  | 			return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	pos = 0; | 
					
						
							|  |  |  | 	while (mappinggetnext(class->cl_dict, &pos, &key, &value)) { | 
					
						
							| 
									
										
										
										
											1993-05-21 19:56:10 +00:00
										 |  |  | 		object *v; | 
					
						
							| 
									
										
										
										
											1993-05-20 14:24:46 +00:00
										 |  |  | 		if (!is_accessobject(value)) | 
					
						
							|  |  |  | 			continue; | 
					
						
							| 
									
										
										
										
											1993-05-21 19:56:10 +00:00
										 |  |  | 		if (hasaccessvalue(value)) | 
					
						
							|  |  |  | 			continue; | 
					
						
							| 
									
										
										
										
											1993-05-20 14:24:46 +00:00
										 |  |  | 		ac = dict2lookup(inst->in_dict, key); | 
					
						
							|  |  |  | 		if (ac != NULL && is_accessobject(ac)) { | 
					
						
							|  |  |  | 			err_setval(ConflictError, key); | 
					
						
							|  |  |  | 			return -1; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		ac = cloneaccessobject(value); | 
					
						
							|  |  |  | 		if (ac == NULL) | 
					
						
							|  |  |  | 			return -1; | 
					
						
							|  |  |  | 		ret = dict2insert(inst->in_dict, key, ac); | 
					
						
							|  |  |  | 		DECREF(ac); | 
					
						
							|  |  |  | 		if (ret != 0) | 
					
						
							|  |  |  | 			return -1; | 
					
						
							|  |  |  |     	} | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | object * | 
					
						
							| 
									
										
										
										
											1993-05-19 14:50:45 +00:00
										 |  |  | newinstanceobject(class, arg) | 
					
						
							|  |  |  | 	object *class; | 
					
						
							|  |  |  | 	object *arg; | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											1991-05-05 20:03:07 +00:00
										 |  |  | 	register instanceobject *inst; | 
					
						
							| 
									
										
										
										
											1992-08-12 15:35:34 +00:00
										 |  |  | 	object *v; | 
					
						
							| 
									
										
										
										
											1993-05-19 14:50:45 +00:00
										 |  |  | 	object *init; | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | 	if (!is_classobject(class)) { | 
					
						
							| 
									
										
										
										
											1990-10-21 22:15:08 +00:00
										 |  |  | 		err_badcall(); | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | 		return NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											1991-05-05 20:03:07 +00:00
										 |  |  | 	inst = NEWOBJ(instanceobject, &Instancetype); | 
					
						
							|  |  |  | 	if (inst == NULL) | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | 		return NULL; | 
					
						
							|  |  |  | 	INCREF(class); | 
					
						
							| 
									
										
										
										
											1991-05-05 20:03:07 +00:00
										 |  |  | 	inst->in_class = (classobject *)class; | 
					
						
							| 
									
										
										
										
											1993-05-20 14:24:46 +00:00
										 |  |  | 	inst->in_dict = newdictobject(); | 
					
						
							|  |  |  | 	if (inst->in_dict == NULL || | 
					
						
							|  |  |  | 	    addaccess((classobject *)class, inst) != 0) { | 
					
						
							| 
									
										
										
										
											1991-05-05 20:03:07 +00:00
										 |  |  | 		DECREF(inst); | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | 		return NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											1993-05-19 14:50:45 +00:00
										 |  |  | 	init = instance_getattr(inst, "__init__"); | 
					
						
							|  |  |  | 	if (init == NULL) { | 
					
						
							|  |  |  | 		err_clear(); | 
					
						
							|  |  |  | 		if (arg != NULL && !(is_tupleobject(arg) && | 
					
						
							|  |  |  | 				     gettuplesize(arg) == 0)) { | 
					
						
							|  |  |  | 			err_setstr(TypeError, | 
					
						
							|  |  |  | 				"this classobject() takes no arguments"); | 
					
						
							|  |  |  | 			DECREF(inst); | 
					
						
							|  |  |  | 			inst = NULL; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							|  |  |  | 		object *res = call_object(init, arg); | 
					
						
							|  |  |  | 		DECREF(init); | 
					
						
							|  |  |  | 		if (res == NULL) { | 
					
						
							|  |  |  | 			DECREF(inst); | 
					
						
							|  |  |  | 			inst = NULL; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		else { | 
					
						
							|  |  |  | 			if (res != None) { | 
					
						
							|  |  |  | 				err_setstr(TypeError, | 
					
						
							|  |  |  | 					   "__init__() should return None"); | 
					
						
							|  |  |  | 				DECREF(inst); | 
					
						
							|  |  |  | 				inst = NULL; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			DECREF(res); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											1991-05-05 20:03:07 +00:00
										 |  |  | 	return (object *)inst; | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1991-04-04 10:42:10 +00:00
										 |  |  | /* Instance methods */ | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							| 
									
										
										
										
											1991-05-05 20:03:07 +00:00
										 |  |  | instance_dealloc(inst) | 
					
						
							|  |  |  | 	register instanceobject *inst; | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											1993-05-19 14:50:45 +00:00
										 |  |  | 	object *error_type, *error_value; | 
					
						
							|  |  |  | 	object *del; | 
					
						
							|  |  |  | 	/* Call the __del__ method if it exists.  First temporarily
 | 
					
						
							|  |  |  | 	   revive the object and save the current exception, if any. */ | 
					
						
							|  |  |  | 	INCREF(inst); | 
					
						
							|  |  |  | 	err_get(&error_type, &error_value); | 
					
						
							|  |  |  | 	if ((del = instance_getattr(inst, "__del__")) != NULL) { | 
					
						
							|  |  |  | 		object *args = newtupleobject(0); | 
					
						
							|  |  |  | 		object *res = args; | 
					
						
							|  |  |  | 		if (res != NULL) | 
					
						
							|  |  |  | 			res = call_object(del, args); | 
					
						
							|  |  |  | 		XDECREF(args); | 
					
						
							|  |  |  | 		DECREF(del); | 
					
						
							|  |  |  | 		XDECREF(res); | 
					
						
							|  |  |  | 		/* XXX If __del__ raised an exception, it is ignored! */ | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	/* Restore the saved exception and undo the temporary revival */ | 
					
						
							|  |  |  | 	err_setval(error_type, error_value); | 
					
						
							|  |  |  | 	/* Can't use DECREF here, it would cause a recursive call */ | 
					
						
							|  |  |  | 	if (--inst->ob_refcnt > 0) | 
					
						
							|  |  |  | 		return; /* __del__ added a reference; don't delete now */ | 
					
						
							| 
									
										
										
										
											1991-05-05 20:03:07 +00:00
										 |  |  | 	DECREF(inst->in_class); | 
					
						
							| 
									
										
										
										
											1993-05-20 14:24:46 +00:00
										 |  |  | 	XDECREF(inst->in_dict); | 
					
						
							| 
									
										
										
										
											1991-05-05 20:03:07 +00:00
										 |  |  | 	free((ANY *)inst); | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static object * | 
					
						
							| 
									
										
										
										
											1991-05-05 20:03:07 +00:00
										 |  |  | instance_getattr(inst, name) | 
					
						
							|  |  |  | 	register instanceobject *inst; | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | 	register char *name; | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											1991-10-20 20:11:48 +00:00
										 |  |  | 	register object *v; | 
					
						
							| 
									
										
										
										
											1993-05-20 14:24:46 +00:00
										 |  |  | 	classobject *class; | 
					
						
							| 
									
										
										
										
											1993-10-22 12:04:32 +00:00
										 |  |  | 	if (name[0] == '_' && name[1] == '_') { | 
					
						
							|  |  |  | 		if (strcmp(name, "__dict__") == 0) { | 
					
						
							|  |  |  | 			INCREF(inst->in_dict); | 
					
						
							|  |  |  | 			return inst->in_dict; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if (strcmp(name, "__class__") == 0) { | 
					
						
							|  |  |  | 			INCREF(inst->in_class); | 
					
						
							|  |  |  | 			return (object *)inst->in_class; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											1991-10-20 20:11:48 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											1993-05-25 09:38:27 +00:00
										 |  |  | 	class = NULL; | 
					
						
							| 
									
										
										
										
											1993-05-20 14:24:46 +00:00
										 |  |  | 	v = dictlookup(inst->in_dict, name); | 
					
						
							| 
									
										
										
										
											1993-05-21 19:56:10 +00:00
										 |  |  | 	if (v == NULL) { | 
					
						
							|  |  |  | 		v = class_lookup(inst->in_class, name, &class); | 
					
						
							|  |  |  | 		if (v == NULL) { | 
					
						
							|  |  |  | 			err_setstr(AttributeError, name); | 
					
						
							|  |  |  | 			return NULL; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											1993-05-21 19:56:10 +00:00
										 |  |  | 	if (is_accessobject(v)) { | 
					
						
							| 
									
										
										
										
											1993-05-25 09:38:27 +00:00
										 |  |  | 		v = getaccessvalue(v, getowner()); | 
					
						
							| 
									
										
										
										
											1993-05-21 19:56:10 +00:00
										 |  |  | 		if (v == NULL) | 
					
						
							|  |  |  | 			return NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 		INCREF(v); | 
					
						
							| 
									
										
										
										
											1993-10-15 13:01:11 +00:00
										 |  |  | 	if (class != NULL) { | 
					
						
							|  |  |  | 		if (is_funcobject(v)) { | 
					
						
							|  |  |  | 			object *w = newinstancemethodobject(v, (object *)inst, | 
					
						
							|  |  |  | 							    (object *)class); | 
					
						
							|  |  |  | 			DECREF(v); | 
					
						
							|  |  |  | 			v = w; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		else if (is_instancemethodobject(v)) { | 
					
						
							|  |  |  | 			object *im_class = instancemethodgetclass(v); | 
					
						
							|  |  |  | 			/* Only if classes are compatible */ | 
					
						
							|  |  |  | 			if (issubclass((object *)class, im_class)) { | 
					
						
							|  |  |  | 				object *im_func = instancemethodgetfunc(v); | 
					
						
							|  |  |  | 				object *w = newinstancemethodobject(im_func, | 
					
						
							|  |  |  | 						(object *)inst, im_class); | 
					
						
							|  |  |  | 				DECREF(v); | 
					
						
							|  |  |  | 				v = w; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											1993-05-21 19:56:10 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	return v; | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							| 
									
										
										
										
											1991-05-05 20:03:07 +00:00
										 |  |  | instance_setattr(inst, name, v) | 
					
						
							|  |  |  | 	instanceobject *inst; | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | 	char *name; | 
					
						
							|  |  |  | 	object *v; | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											1993-05-19 14:50:45 +00:00
										 |  |  | 	object *ac; | 
					
						
							| 
									
										
										
										
											1992-04-05 14:24:32 +00:00
										 |  |  | 	if (name[0] == '_' && name[1] == '_') { | 
					
						
							|  |  |  | 		int n = strlen(name); | 
					
						
							|  |  |  | 		if (name[n-1] == '_' && name[n-2] == '_') { | 
					
						
							|  |  |  | 			err_setstr(TypeError, "read-only special attribute"); | 
					
						
							|  |  |  | 			return -1; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											1993-05-20 14:24:46 +00:00
										 |  |  | 	ac = dictlookup(inst->in_dict, name); | 
					
						
							| 
									
										
										
										
											1993-05-19 14:50:45 +00:00
										 |  |  | 	if (ac != NULL && is_accessobject(ac)) | 
					
						
							| 
									
										
										
										
											1993-05-25 09:38:27 +00:00
										 |  |  | 		return setaccessvalue(ac, getowner(), v); | 
					
						
							| 
									
										
										
										
											1992-09-04 09:45:18 +00:00
										 |  |  | 	if (v == NULL) { | 
					
						
							| 
									
										
										
										
											1993-05-20 14:24:46 +00:00
										 |  |  | 		int rv = dictremove(inst->in_dict, name); | 
					
						
							| 
									
										
										
										
											1992-09-04 09:45:18 +00:00
										 |  |  | 		if (rv < 0) | 
					
						
							|  |  |  | 			err_setstr(AttributeError, | 
					
						
							|  |  |  | 				   "delete non-existing instance attribute"); | 
					
						
							|  |  |  | 		return rv; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | 	else | 
					
						
							| 
									
										
										
										
											1993-05-20 14:24:46 +00:00
										 |  |  | 		return dictinsert(inst->in_dict, name, v); | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1993-03-29 10:43:31 +00:00
										 |  |  | static object * | 
					
						
							| 
									
										
										
										
											1992-08-12 15:35:34 +00:00
										 |  |  | instance_repr(inst) | 
					
						
							|  |  |  | 	instanceobject *inst; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	object *func; | 
					
						
							|  |  |  | 	object *res; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	func = instance_getattr(inst, "__repr__"); | 
					
						
							|  |  |  | 	if (func == NULL) { | 
					
						
							| 
									
										
										
										
											1993-05-19 14:50:45 +00:00
										 |  |  | 		char buf[140]; | 
					
						
							|  |  |  | 		object *classname = inst->in_class->cl_name; | 
					
						
							|  |  |  | 		char *cname; | 
					
						
							|  |  |  | 		if (classname != NULL && is_stringobject(classname)) | 
					
						
							|  |  |  | 			cname = getstringvalue(classname); | 
					
						
							|  |  |  | 		else | 
					
						
							|  |  |  | 			cname = "?"; | 
					
						
							| 
									
										
										
										
											1992-08-12 15:35:34 +00:00
										 |  |  | 		err_clear(); | 
					
						
							| 
									
										
										
										
											1993-05-19 14:50:45 +00:00
										 |  |  | 		sprintf(buf, "<%.100s instance at %lx>", cname, (long)inst); | 
					
						
							| 
									
										
										
										
											1992-08-12 15:35:34 +00:00
										 |  |  | 		return newstringobject(buf); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	res = call_object(func, (object *)NULL); | 
					
						
							|  |  |  | 	DECREF(func); | 
					
						
							|  |  |  | 	return res; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1993-03-29 10:43:31 +00:00
										 |  |  | static int | 
					
						
							| 
									
										
										
										
											1992-08-12 15:35:34 +00:00
										 |  |  | instance_compare(inst, other) | 
					
						
							|  |  |  | 	instanceobject *inst, *other; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	object *func; | 
					
						
							|  |  |  | 	object *res; | 
					
						
							|  |  |  | 	int outcome; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	func = instance_getattr(inst, "__cmp__"); | 
					
						
							|  |  |  | 	if (func == NULL) { | 
					
						
							|  |  |  | 		err_clear(); | 
					
						
							|  |  |  | 		if (inst < other) | 
					
						
							|  |  |  | 			return -1; | 
					
						
							|  |  |  | 		if (inst > other) | 
					
						
							|  |  |  | 			return 1; | 
					
						
							|  |  |  | 		return 0; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	res = call_object(func, (object *)other); | 
					
						
							|  |  |  | 	DECREF(func); | 
					
						
							|  |  |  | 	if (res == NULL) { | 
					
						
							|  |  |  | 		err_clear(); /* XXX Should report the error, bot how...??? */ | 
					
						
							|  |  |  | 		return 0; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if (is_intobject(res)) | 
					
						
							|  |  |  | 		outcome = getintvalue(res); | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 		outcome = 0; /* XXX Should report the error, bot how...??? */ | 
					
						
							|  |  |  | 	DECREF(res); | 
					
						
							|  |  |  | 	return outcome; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1993-03-29 10:43:31 +00:00
										 |  |  | static long | 
					
						
							|  |  |  | instance_hash(inst) | 
					
						
							|  |  |  | 	instanceobject *inst; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	object *func; | 
					
						
							|  |  |  | 	object *res; | 
					
						
							| 
									
										
										
										
											1993-04-08 12:56:19 +00:00
										 |  |  | 	long outcome; | 
					
						
							| 
									
										
										
										
											1993-03-29 10:43:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	func = instance_getattr(inst, "__hash__"); | 
					
						
							|  |  |  | 	if (func == NULL) { | 
					
						
							|  |  |  | 		/* If there is no __cmp__ method, we hash on the address.
 | 
					
						
							|  |  |  | 		   If a __cmp__ method exists, there must be a __hash__. */ | 
					
						
							|  |  |  | 		err_clear(); | 
					
						
							|  |  |  | 		func = instance_getattr(inst, "__cmp__"); | 
					
						
							| 
									
										
										
										
											1993-04-08 12:56:19 +00:00
										 |  |  | 		if (func == NULL) { | 
					
						
							|  |  |  | 			err_clear(); | 
					
						
							|  |  |  | 			outcome = (long)inst; | 
					
						
							|  |  |  | 			if (outcome == -1) | 
					
						
							|  |  |  | 				outcome = -2; | 
					
						
							|  |  |  | 			return outcome; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											1993-03-29 10:43:31 +00:00
										 |  |  | 		err_setstr(TypeError, "unhashable instance"); | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	res = call_object(func, (object *)NULL); | 
					
						
							|  |  |  | 	DECREF(func); | 
					
						
							|  |  |  | 	if (res == NULL) | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	if (is_intobject(res)) { | 
					
						
							|  |  |  | 		outcome = getintvalue(res); | 
					
						
							|  |  |  | 		if (outcome == -1) | 
					
						
							|  |  |  | 			outcome = -2; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							|  |  |  | 		err_setstr(TypeError, "__hash__() should return an int"); | 
					
						
							|  |  |  | 		outcome = -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	DECREF(res); | 
					
						
							|  |  |  | 	return outcome; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							| 
									
										
										
										
											1992-08-12 15:35:34 +00:00
										 |  |  | instance_length(inst) | 
					
						
							|  |  |  | 	instanceobject *inst; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	object *func; | 
					
						
							|  |  |  | 	object *res; | 
					
						
							|  |  |  | 	int outcome; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	func = instance_getattr(inst, "__len__"); | 
					
						
							|  |  |  | 	if (func == NULL) | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	res = call_object(func, (object *)NULL); | 
					
						
							|  |  |  | 	DECREF(func); | 
					
						
							| 
									
										
										
										
											1992-11-26 10:30:26 +00:00
										 |  |  | 	if (res == NULL) | 
					
						
							|  |  |  | 		return -1; | 
					
						
							| 
									
										
										
										
											1992-08-12 15:35:34 +00:00
										 |  |  | 	if (is_intobject(res)) { | 
					
						
							|  |  |  | 		outcome = getintvalue(res); | 
					
						
							|  |  |  | 		if (outcome < 0) | 
					
						
							|  |  |  | 			err_setstr(ValueError, "__len__() should return >= 0"); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							|  |  |  | 		err_setstr(TypeError, "__len__() should return an int"); | 
					
						
							|  |  |  | 		outcome = -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	DECREF(res); | 
					
						
							|  |  |  | 	return outcome; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1993-03-29 10:43:31 +00:00
										 |  |  | static object * | 
					
						
							| 
									
										
										
										
											1992-08-12 15:35:34 +00:00
										 |  |  | instance_subscript(inst, key) | 
					
						
							|  |  |  | 	instanceobject *inst; | 
					
						
							|  |  |  | 	object *key; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	object *func; | 
					
						
							|  |  |  | 	object *arg; | 
					
						
							|  |  |  | 	object *res; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	func = instance_getattr(inst, "__getitem__"); | 
					
						
							|  |  |  | 	if (func == NULL) | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							|  |  |  | 	arg = mkvalue("(O)", key); | 
					
						
							|  |  |  | 	if (arg == NULL) { | 
					
						
							|  |  |  | 		DECREF(func); | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	res = call_object(func, arg); | 
					
						
							|  |  |  | 	DECREF(func); | 
					
						
							|  |  |  | 	DECREF(arg); | 
					
						
							|  |  |  | 	return res; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1993-03-29 10:43:31 +00:00
										 |  |  | static int | 
					
						
							| 
									
										
										
										
											1992-08-12 15:35:34 +00:00
										 |  |  | instance_ass_subscript(inst, key, value) | 
					
						
							|  |  |  | 	instanceobject*inst; | 
					
						
							|  |  |  | 	object *key; | 
					
						
							|  |  |  | 	object *value; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	object *func; | 
					
						
							|  |  |  | 	object *arg; | 
					
						
							|  |  |  | 	object *res; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (value == NULL) | 
					
						
							|  |  |  | 		func = instance_getattr(inst, "__delitem__"); | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 		func = instance_getattr(inst, "__setitem__"); | 
					
						
							|  |  |  | 	if (func == NULL) | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	if (value == NULL) | 
					
						
							|  |  |  | 		arg = mkvalue("(O)", key); | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 		arg = mkvalue("(OO)", key, value); | 
					
						
							|  |  |  | 	if (arg == NULL) { | 
					
						
							|  |  |  | 		DECREF(func); | 
					
						
							| 
									
										
										
										
											1992-08-14 13:49:30 +00:00
										 |  |  | 		return -1; | 
					
						
							| 
									
										
										
										
											1992-08-12 15:35:34 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	res = call_object(func, arg); | 
					
						
							|  |  |  | 	DECREF(func); | 
					
						
							|  |  |  | 	DECREF(arg); | 
					
						
							|  |  |  | 	if (res == NULL) | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	DECREF(res); | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1993-03-29 10:43:31 +00:00
										 |  |  | static mapping_methods instance_as_mapping = { | 
					
						
							| 
									
										
										
										
											1992-08-12 15:35:34 +00:00
										 |  |  | 	instance_length,	/*mp_length*/ | 
					
						
							|  |  |  | 	instance_subscript,	/*mp_subscript*/ | 
					
						
							|  |  |  | 	instance_ass_subscript,	/*mp_ass_subscript*/ | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static object * | 
					
						
							|  |  |  | instance_concat(inst, other) | 
					
						
							|  |  |  | 	instanceobject *inst, *other; | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											1992-08-14 12:06:52 +00:00
										 |  |  | 	object *func, *arg, *res; | 
					
						
							| 
									
										
										
										
											1992-08-12 15:35:34 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	func = instance_getattr(inst, "__add__"); | 
					
						
							|  |  |  | 	if (func == NULL) | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							| 
									
										
										
										
											1992-08-14 12:06:52 +00:00
										 |  |  | 	arg = mkvalue("(O)", other); | 
					
						
							|  |  |  | 	if (arg == NULL) { | 
					
						
							|  |  |  | 		DECREF(func); | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	res = call_object(func, arg); | 
					
						
							| 
									
										
										
										
											1992-08-12 15:35:34 +00:00
										 |  |  | 	DECREF(func); | 
					
						
							| 
									
										
										
										
											1992-08-14 12:06:52 +00:00
										 |  |  | 	DECREF(arg); | 
					
						
							| 
									
										
										
										
											1992-08-12 15:35:34 +00:00
										 |  |  | 	return res; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static object * | 
					
						
							|  |  |  | instance_repeat(inst, count) | 
					
						
							|  |  |  | 	instanceobject *inst; | 
					
						
							|  |  |  | 	int count; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	object *func, *arg, *res; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	func = instance_getattr(inst, "__mul__"); | 
					
						
							|  |  |  | 	if (func == NULL) | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							|  |  |  | 	arg = newintobject((long)count); | 
					
						
							|  |  |  | 	if (arg == NULL) { | 
					
						
							|  |  |  | 		DECREF(func); | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	res = call_object(func, arg); | 
					
						
							|  |  |  | 	DECREF(func); | 
					
						
							|  |  |  | 	DECREF(arg); | 
					
						
							|  |  |  | 	return res; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static object * | 
					
						
							|  |  |  | instance_item(inst, i) | 
					
						
							|  |  |  | 	instanceobject *inst; | 
					
						
							|  |  |  | 	int i; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	object *func, *arg, *res; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	func = instance_getattr(inst, "__getitem__"); | 
					
						
							|  |  |  | 	if (func == NULL) | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							|  |  |  | 	arg = newintobject((long)i); | 
					
						
							|  |  |  | 	if (arg == NULL) { | 
					
						
							|  |  |  | 		DECREF(func); | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	res = call_object(func, arg); | 
					
						
							|  |  |  | 	DECREF(func); | 
					
						
							|  |  |  | 	DECREF(arg); | 
					
						
							|  |  |  | 	return res; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static object * | 
					
						
							|  |  |  | instance_slice(inst, i, j) | 
					
						
							|  |  |  | 	instanceobject *inst; | 
					
						
							|  |  |  | 	int i, j; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	object *func, *arg, *res; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	func = instance_getattr(inst, "__getslice__"); | 
					
						
							|  |  |  | 	if (func == NULL) | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							|  |  |  | 	arg = mkvalue("(ii)", i, j); | 
					
						
							|  |  |  | 	if (arg == NULL) { | 
					
						
							|  |  |  | 		DECREF(func); | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	res = call_object(func, arg); | 
					
						
							|  |  |  | 	DECREF(func); | 
					
						
							|  |  |  | 	DECREF(arg); | 
					
						
							|  |  |  | 	return res; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | instance_ass_item(inst, i, item) | 
					
						
							|  |  |  | 	instanceobject *inst; | 
					
						
							|  |  |  | 	int i; | 
					
						
							|  |  |  | 	object *item; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	object *func, *arg, *res; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (item == NULL) | 
					
						
							|  |  |  | 		func = instance_getattr(inst, "__delitem__"); | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 		func = instance_getattr(inst, "__setitem__"); | 
					
						
							|  |  |  | 	if (func == NULL) | 
					
						
							| 
									
										
										
										
											1992-08-14 13:49:30 +00:00
										 |  |  | 		return -1; | 
					
						
							| 
									
										
										
										
											1992-08-12 15:35:34 +00:00
										 |  |  | 	if (item == NULL) | 
					
						
							|  |  |  | 		arg = mkvalue("i", i); | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 		arg = mkvalue("(iO)", i, item); | 
					
						
							|  |  |  | 	if (arg == NULL) { | 
					
						
							|  |  |  | 		DECREF(func); | 
					
						
							| 
									
										
										
										
											1992-08-14 13:49:30 +00:00
										 |  |  | 		return -1; | 
					
						
							| 
									
										
										
										
											1992-08-12 15:35:34 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	res = call_object(func, arg); | 
					
						
							|  |  |  | 	DECREF(func); | 
					
						
							|  |  |  | 	DECREF(arg); | 
					
						
							|  |  |  | 	if (res == NULL) | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	DECREF(res); | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | instance_ass_slice(inst, i, j, value) | 
					
						
							|  |  |  | 	instanceobject *inst; | 
					
						
							|  |  |  | 	int i, j; | 
					
						
							|  |  |  | 	object *value; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	object *func, *arg, *res; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (value == NULL) | 
					
						
							|  |  |  | 		func = instance_getattr(inst, "__delslice__"); | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 		func = instance_getattr(inst, "__setslice__"); | 
					
						
							|  |  |  | 	if (func == NULL) | 
					
						
							| 
									
										
										
										
											1992-08-14 13:49:30 +00:00
										 |  |  | 		return -1; | 
					
						
							| 
									
										
										
										
											1992-08-12 15:35:34 +00:00
										 |  |  | 	if (value == NULL) | 
					
						
							|  |  |  | 		arg = mkvalue("(ii)", i, j); | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 		arg = mkvalue("(iiO)", i, j, value); | 
					
						
							|  |  |  | 	if (arg == NULL) { | 
					
						
							|  |  |  | 		DECREF(func); | 
					
						
							| 
									
										
										
										
											1992-08-14 13:49:30 +00:00
										 |  |  | 		return -1; | 
					
						
							| 
									
										
										
										
											1992-08-12 15:35:34 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	res = call_object(func, arg); | 
					
						
							|  |  |  | 	DECREF(func); | 
					
						
							|  |  |  | 	DECREF(arg); | 
					
						
							|  |  |  | 	if (res == NULL) | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	DECREF(res); | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static sequence_methods instance_as_sequence = { | 
					
						
							|  |  |  | 	instance_length,	/*sq_length*/ | 
					
						
							|  |  |  | 	instance_concat,	/*sq_concat*/ | 
					
						
							|  |  |  | 	instance_repeat,	/*sq_repeat*/ | 
					
						
							|  |  |  | 	instance_item,		/*sq_item*/ | 
					
						
							|  |  |  | 	instance_slice,		/*sq_slice*/ | 
					
						
							|  |  |  | 	instance_ass_item,	/*sq_ass_item*/ | 
					
						
							|  |  |  | 	instance_ass_slice,	/*sq_ass_slice*/ | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static object * | 
					
						
							|  |  |  | generic_binary_op(self, other, methodname) | 
					
						
							|  |  |  | 	instanceobject *self; | 
					
						
							|  |  |  | 	object *other; | 
					
						
							|  |  |  | 	char *methodname; | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											1992-08-14 12:06:52 +00:00
										 |  |  | 	object *func, *arg, *res; | 
					
						
							| 
									
										
										
										
											1992-08-12 15:35:34 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if ((func = instance_getattr(self, methodname)) == NULL) | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							| 
									
										
										
										
											1992-08-14 12:06:52 +00:00
										 |  |  | 	arg = mkvalue("O", other); | 
					
						
							|  |  |  | 	if (arg == NULL) { | 
					
						
							|  |  |  | 		DECREF(func); | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	res = call_object(func, arg); | 
					
						
							| 
									
										
										
										
											1992-08-12 15:35:34 +00:00
										 |  |  | 	DECREF(func); | 
					
						
							| 
									
										
										
										
											1992-08-14 12:06:52 +00:00
										 |  |  | 	DECREF(arg); | 
					
						
							| 
									
										
										
										
											1992-08-12 15:35:34 +00:00
										 |  |  | 	return res; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static object * | 
					
						
							|  |  |  | generic_unary_op(self, methodname) | 
					
						
							|  |  |  | 	instanceobject *self; | 
					
						
							|  |  |  | 	char *methodname; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	object *func, *res; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if ((func = instance_getattr(self, methodname)) == NULL) | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							|  |  |  | 	res = call_object(func, (object *)NULL); | 
					
						
							|  |  |  | 	DECREF(func); | 
					
						
							|  |  |  | 	return res; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define BINARY(funcname, methodname) \
 | 
					
						
							|  |  |  | static object * funcname(self, other) instanceobject *self; object *other; { \ | 
					
						
							|  |  |  | 	return generic_binary_op(self, other, methodname); \ | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define UNARY(funcname, methodname) \
 | 
					
						
							|  |  |  | static object *funcname(self) instanceobject *self; { \ | 
					
						
							|  |  |  | 	return generic_unary_op(self, methodname); \ | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | BINARY(instance_add, "__add__") | 
					
						
							|  |  |  | BINARY(instance_sub, "__sub__") | 
					
						
							|  |  |  | BINARY(instance_mul, "__mul__") | 
					
						
							|  |  |  | BINARY(instance_div, "__div__") | 
					
						
							|  |  |  | BINARY(instance_mod, "__mod__") | 
					
						
							|  |  |  | BINARY(instance_divmod, "__divmod__") | 
					
						
							|  |  |  | BINARY(instance_pow, "__pow__") | 
					
						
							|  |  |  | UNARY(instance_neg, "__neg__") | 
					
						
							|  |  |  | UNARY(instance_pos, "__pos__") | 
					
						
							|  |  |  | UNARY(instance_abs, "__abs__") | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1993-03-29 10:43:31 +00:00
										 |  |  | static int | 
					
						
							| 
									
										
										
										
											1992-08-12 15:35:34 +00:00
										 |  |  | instance_nonzero(self) | 
					
						
							|  |  |  | 	instanceobject *self; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	object *func, *res; | 
					
						
							|  |  |  | 	long outcome; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1993-02-12 16:29:05 +00:00
										 |  |  | 	if ((func = instance_getattr(self, "__nonzero__")) == NULL) { | 
					
						
							| 
									
										
										
										
											1992-08-12 15:35:34 +00:00
										 |  |  | 		err_clear(); | 
					
						
							| 
									
										
										
										
											1993-02-12 16:29:05 +00:00
										 |  |  | 		if ((func = instance_getattr(self, "__len__")) == NULL) { | 
					
						
							| 
									
										
										
										
											1992-08-12 15:35:34 +00:00
										 |  |  | 			err_clear(); | 
					
						
							|  |  |  | 			/* Fall back to the default behavior:
 | 
					
						
							|  |  |  | 			   all instances are nonzero */ | 
					
						
							|  |  |  | 			return 1; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	res = call_object(func, (object *)NULL); | 
					
						
							|  |  |  | 	DECREF(func); | 
					
						
							|  |  |  | 	if (res == NULL) | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	if (!is_intobject(res)) { | 
					
						
							|  |  |  | 		DECREF(res); | 
					
						
							|  |  |  | 		err_setstr(TypeError, "__nonzero__ should return an int"); | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	outcome = getintvalue(res); | 
					
						
							|  |  |  | 	DECREF(res); | 
					
						
							|  |  |  | 	if (outcome < 0) { | 
					
						
							|  |  |  | 		err_setstr(ValueError, "__nonzero__ should return >= 0"); | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return outcome > 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | UNARY(instance_invert, "__invert__") | 
					
						
							|  |  |  | BINARY(instance_lshift, "__lshift__") | 
					
						
							|  |  |  | BINARY(instance_rshift, "__rshift__") | 
					
						
							|  |  |  | BINARY(instance_and, "__and__") | 
					
						
							|  |  |  | BINARY(instance_xor, "__xor__") | 
					
						
							|  |  |  | BINARY(instance_or, "__or__") | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1992-08-14 12:06:52 +00:00
										 |  |  | static int | 
					
						
							|  |  |  | instance_coerce(pv, pw) | 
					
						
							|  |  |  | 	object **pv, **pw; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	object *v =  *pv; | 
					
						
							|  |  |  | 	object *w = *pw; | 
					
						
							|  |  |  | 	object *func; | 
					
						
							|  |  |  | 	object *res; | 
					
						
							|  |  |  | 	int outcome; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!is_instanceobject(v)) | 
					
						
							|  |  |  | 		return 1; /* XXX shouldn't be possible */ | 
					
						
							|  |  |  | 	func = instance_getattr((instanceobject *)v, "__coerce__"); | 
					
						
							|  |  |  | 	if (func == NULL) { | 
					
						
							|  |  |  | 		err_clear(); | 
					
						
							|  |  |  | 		return 1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	res = call_object(func, w); | 
					
						
							|  |  |  | 	if (res == NULL) | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	if (res == None) { | 
					
						
							|  |  |  | 		DECREF(res); | 
					
						
							|  |  |  | 		return 1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	outcome = getargs(res, "(OO)", &v, &w); | 
					
						
							|  |  |  | 	if (!outcome || v->ob_type != w->ob_type || | 
					
						
							|  |  |  | 			v->ob_type->tp_as_number == NULL) { | 
					
						
							|  |  |  | 		DECREF(res); | 
					
						
							|  |  |  | 		err_setstr(TypeError, "bad __coerce__ result"); | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	INCREF(v); | 
					
						
							|  |  |  | 	INCREF(w); | 
					
						
							|  |  |  | 	DECREF(res); | 
					
						
							|  |  |  | 	*pv = v; | 
					
						
							|  |  |  | 	*pw = w; | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1992-09-12 11:09:23 +00:00
										 |  |  | UNARY(instance_int, "__int__") | 
					
						
							|  |  |  | UNARY(instance_long, "__long__") | 
					
						
							|  |  |  | UNARY(instance_float, "__float__") | 
					
						
							|  |  |  | UNARY(instance_oct, "__oct__") | 
					
						
							|  |  |  | UNARY(instance_hex, "__hex__") | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1992-08-12 15:35:34 +00:00
										 |  |  | static number_methods instance_as_number = { | 
					
						
							|  |  |  | 	instance_add,		/*nb_add*/ | 
					
						
							|  |  |  | 	instance_sub,		/*nb_subtract*/ | 
					
						
							|  |  |  | 	instance_mul,		/*nb_multiply*/ | 
					
						
							|  |  |  | 	instance_div,		/*nb_divide*/ | 
					
						
							|  |  |  | 	instance_mod,		/*nb_remainder*/ | 
					
						
							|  |  |  | 	instance_divmod,	/*nb_divmod*/ | 
					
						
							|  |  |  | 	instance_pow,		/*nb_power*/ | 
					
						
							|  |  |  | 	instance_neg,		/*nb_negative*/ | 
					
						
							|  |  |  | 	instance_pos,		/*nb_positive*/ | 
					
						
							|  |  |  | 	instance_abs,		/*nb_absolute*/ | 
					
						
							|  |  |  | 	instance_nonzero,	/*nb_nonzero*/ | 
					
						
							|  |  |  | 	instance_invert,	/*nb_invert*/ | 
					
						
							|  |  |  | 	instance_lshift,	/*nb_lshift*/ | 
					
						
							|  |  |  | 	instance_rshift,	/*nb_rshift*/ | 
					
						
							|  |  |  | 	instance_and,		/*nb_and*/ | 
					
						
							|  |  |  | 	instance_xor,		/*nb_xor*/ | 
					
						
							|  |  |  | 	instance_or,		/*nb_or*/ | 
					
						
							| 
									
										
										
										
											1992-08-14 12:06:52 +00:00
										 |  |  | 	instance_coerce,	/*nb_coerce*/ | 
					
						
							| 
									
										
										
										
											1992-09-12 11:09:23 +00:00
										 |  |  | 	instance_int,		/*nb_int*/ | 
					
						
							|  |  |  | 	instance_long,		/*nb_long*/ | 
					
						
							|  |  |  | 	instance_float,		/*nb_float*/ | 
					
						
							|  |  |  | 	instance_oct,		/*nb_oct*/ | 
					
						
							|  |  |  | 	instance_hex,		/*nb_hex*/ | 
					
						
							| 
									
										
										
										
											1992-08-12 15:35:34 +00:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1991-04-04 10:42:10 +00:00
										 |  |  | typeobject Instancetype = { | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | 	OB_HEAD_INIT(&Typetype) | 
					
						
							|  |  |  | 	0, | 
					
						
							| 
									
										
										
										
											1991-04-04 10:42:10 +00:00
										 |  |  | 	"instance", | 
					
						
							|  |  |  | 	sizeof(instanceobject), | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | 	0, | 
					
						
							| 
									
										
										
										
											1991-04-04 10:42:10 +00:00
										 |  |  | 	instance_dealloc,	/*tp_dealloc*/ | 
					
						
							| 
									
										
										
										
											1992-09-17 17:54:56 +00:00
										 |  |  | 	0,			/*tp_print*/ | 
					
						
							| 
									
										
										
										
											1993-05-19 14:50:45 +00:00
										 |  |  | 	(object * (*) FPROTO((object *, char *))) | 
					
						
							| 
									
										
										
										
											1991-04-04 10:42:10 +00:00
										 |  |  | 	instance_getattr,	/*tp_getattr*/ | 
					
						
							|  |  |  | 	instance_setattr,	/*tp_setattr*/ | 
					
						
							| 
									
										
										
										
											1992-08-12 15:35:34 +00:00
										 |  |  | 	instance_compare,	/*tp_compare*/ | 
					
						
							|  |  |  | 	instance_repr,		/*tp_repr*/ | 
					
						
							|  |  |  | 	&instance_as_number,	/*tp_as_number*/ | 
					
						
							|  |  |  | 	&instance_as_sequence,	/*tp_as_sequence*/ | 
					
						
							|  |  |  | 	&instance_as_mapping,	/*tp_as_mapping*/ | 
					
						
							| 
									
										
										
										
											1993-03-29 10:43:31 +00:00
										 |  |  | 	instance_hash,		/*tp_hash*/ | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1993-05-20 14:24:46 +00:00
										 |  |  | /* Instance method objects are used for two purposes:
 | 
					
						
							|  |  |  |    (a) as bound instance methods (returned by instancename.methodname) | 
					
						
							|  |  |  |    (b) as unbound methods (returned by ClassName.methodname) | 
					
						
							|  |  |  |    In case (b), im_self is NULL | 
					
						
							|  |  |  | */ | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | typedef struct { | 
					
						
							|  |  |  | 	OB_HEAD | 
					
						
							| 
									
										
										
										
											1993-05-20 14:24:46 +00:00
										 |  |  | 	object	*im_func;	/* The function implementing the method */ | 
					
						
							|  |  |  | 	object	*im_self;	/* The instance it is bound to, or NULL */ | 
					
						
							|  |  |  | 	object	*im_class;	/* The class that defined the method */ | 
					
						
							| 
									
										
										
										
											1991-05-05 20:03:07 +00:00
										 |  |  | } instancemethodobject; | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | object * | 
					
						
							| 
									
										
										
										
											1993-05-20 14:24:46 +00:00
										 |  |  | newinstancemethodobject(func, self, class) | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | 	object *func; | 
					
						
							|  |  |  | 	object *self; | 
					
						
							| 
									
										
										
										
											1993-05-20 14:24:46 +00:00
										 |  |  | 	object *class; | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											1991-05-05 20:03:07 +00:00
										 |  |  | 	register instancemethodobject *im; | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | 	if (!is_funcobject(func)) { | 
					
						
							| 
									
										
										
										
											1990-10-21 22:15:08 +00:00
										 |  |  | 		err_badcall(); | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | 		return NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											1991-05-05 20:03:07 +00:00
										 |  |  | 	im = NEWOBJ(instancemethodobject, &Instancemethodtype); | 
					
						
							|  |  |  | 	if (im == NULL) | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | 		return NULL; | 
					
						
							|  |  |  | 	INCREF(func); | 
					
						
							| 
									
										
										
										
											1991-05-05 20:03:07 +00:00
										 |  |  | 	im->im_func = func; | 
					
						
							| 
									
										
										
										
											1993-05-20 14:24:46 +00:00
										 |  |  | 	XINCREF(self); | 
					
						
							| 
									
										
										
										
											1991-05-05 20:03:07 +00:00
										 |  |  | 	im->im_self = self; | 
					
						
							| 
									
										
										
										
											1993-05-20 14:24:46 +00:00
										 |  |  | 	INCREF(class); | 
					
						
							|  |  |  | 	im->im_class = class; | 
					
						
							| 
									
										
										
										
											1991-05-05 20:03:07 +00:00
										 |  |  | 	return (object *)im; | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | object * | 
					
						
							| 
									
										
										
										
											1991-05-05 20:03:07 +00:00
										 |  |  | instancemethodgetfunc(im) | 
					
						
							|  |  |  | 	register object *im; | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											1991-05-05 20:03:07 +00:00
										 |  |  | 	if (!is_instancemethodobject(im)) { | 
					
						
							| 
									
										
										
										
											1990-10-21 22:15:08 +00:00
										 |  |  | 		err_badcall(); | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | 		return NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											1991-05-05 20:03:07 +00:00
										 |  |  | 	return ((instancemethodobject *)im)->im_func; | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | object * | 
					
						
							| 
									
										
										
										
											1991-05-05 20:03:07 +00:00
										 |  |  | instancemethodgetself(im) | 
					
						
							|  |  |  | 	register object *im; | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											1991-05-05 20:03:07 +00:00
										 |  |  | 	if (!is_instancemethodobject(im)) { | 
					
						
							| 
									
										
										
										
											1990-10-21 22:15:08 +00:00
										 |  |  | 		err_badcall(); | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | 		return NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											1991-05-05 20:03:07 +00:00
										 |  |  | 	return ((instancemethodobject *)im)->im_self; | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1993-05-20 14:24:46 +00:00
										 |  |  | object * | 
					
						
							|  |  |  | instancemethodgetclass(im) | 
					
						
							|  |  |  | 	register object *im; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if (!is_instancemethodobject(im)) { | 
					
						
							|  |  |  | 		err_badcall(); | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return ((instancemethodobject *)im)->im_class; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | /* Class method methods */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1991-05-05 20:03:07 +00:00
										 |  |  | #define OFF(x) offsetof(instancemethodobject, x)
 | 
					
						
							| 
									
										
										
										
											1990-12-20 15:06:42 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1991-05-05 20:03:07 +00:00
										 |  |  | static struct memberlist instancemethod_memberlist[] = { | 
					
						
							|  |  |  | 	{"im_func",	T_OBJECT,	OFF(im_func)}, | 
					
						
							|  |  |  | 	{"im_self",	T_OBJECT,	OFF(im_self)}, | 
					
						
							| 
									
										
										
										
											1993-05-20 14:24:46 +00:00
										 |  |  | 	{"im_class",	T_OBJECT,	OFF(im_class)}, | 
					
						
							| 
									
										
										
										
											1990-12-20 15:06:42 +00:00
										 |  |  | 	{NULL}	/* Sentinel */ | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static object * | 
					
						
							| 
									
										
										
										
											1991-05-05 20:03:07 +00:00
										 |  |  | instancemethod_getattr(im, name) | 
					
						
							|  |  |  | 	register instancemethodobject *im; | 
					
						
							| 
									
										
										
										
											1990-12-20 15:06:42 +00:00
										 |  |  | 	char *name; | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											1991-05-05 20:03:07 +00:00
										 |  |  | 	return getmember((char *)im, instancemethod_memberlist, name); | 
					
						
							| 
									
										
										
										
											1990-12-20 15:06:42 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | static void | 
					
						
							| 
									
										
										
										
											1991-05-05 20:03:07 +00:00
										 |  |  | instancemethod_dealloc(im) | 
					
						
							|  |  |  | 	register instancemethodobject *im; | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											1991-05-05 20:03:07 +00:00
										 |  |  | 	DECREF(im->im_func); | 
					
						
							| 
									
										
										
										
											1993-05-20 14:24:46 +00:00
										 |  |  | 	XDECREF(im->im_self); | 
					
						
							|  |  |  | 	DECREF(im->im_class); | 
					
						
							| 
									
										
										
										
											1991-05-05 20:03:07 +00:00
										 |  |  | 	free((ANY *)im); | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1992-09-03 20:39:51 +00:00
										 |  |  | static int | 
					
						
							|  |  |  | instancemethod_compare(a, b) | 
					
						
							|  |  |  | 	instancemethodobject *a, *b; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int cmp = cmpobject(a->im_self, b->im_self); | 
					
						
							|  |  |  | 	if (cmp == 0) | 
					
						
							|  |  |  | 		cmp = cmpobject(a->im_func, b->im_func); | 
					
						
							|  |  |  | 	return cmp; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1993-05-19 14:50:45 +00:00
										 |  |  | static object * | 
					
						
							|  |  |  | instancemethod_repr(a) | 
					
						
							|  |  |  | 	instancemethodobject *a; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	char buf[240]; | 
					
						
							| 
									
										
										
										
											1993-05-20 14:24:46 +00:00
										 |  |  | 	instanceobject *self = (instanceobject *)(a->im_self); | 
					
						
							|  |  |  | 	funcobject *func = (funcobject *)(a->im_func); | 
					
						
							|  |  |  | 	classobject *class = (classobject *)(a->im_class); | 
					
						
							|  |  |  | 	object *fclassname, *iclassname, *funcname; | 
					
						
							|  |  |  | 	char *fcname, *icname, *fname; | 
					
						
							|  |  |  | 	fclassname = class->cl_name; | 
					
						
							|  |  |  | 	funcname = func->func_name; | 
					
						
							|  |  |  | 	if (fclassname != NULL && is_stringobject(fclassname)) | 
					
						
							|  |  |  | 		fcname = getstringvalue(fclassname); | 
					
						
							| 
									
										
										
										
											1993-05-19 14:50:45 +00:00
										 |  |  | 	else | 
					
						
							| 
									
										
										
										
											1993-05-20 14:24:46 +00:00
										 |  |  | 		fcname = "?"; | 
					
						
							| 
									
										
										
										
											1993-05-19 14:50:45 +00:00
										 |  |  | 	if (funcname != NULL && is_stringobject(funcname)) | 
					
						
							|  |  |  | 		fname = getstringvalue(funcname); | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 		fname = "?"; | 
					
						
							| 
									
										
										
										
											1993-05-20 14:24:46 +00:00
										 |  |  | 	if (self == NULL) | 
					
						
							|  |  |  | 		sprintf(buf, "<unbound method %.100s.%.100s>", fcname, fname); | 
					
						
							|  |  |  | 	else { | 
					
						
							|  |  |  | 		iclassname = self->in_class->cl_name; | 
					
						
							|  |  |  | 		if (iclassname != NULL && is_stringobject(iclassname)) | 
					
						
							|  |  |  | 			icname = getstringvalue(iclassname); | 
					
						
							|  |  |  | 		else | 
					
						
							|  |  |  | 			icname = "?"; | 
					
						
							|  |  |  | 		sprintf(buf, "<method %.60s.%.60s of %.60s instance at %lx>", | 
					
						
							|  |  |  | 			fcname, fname, icname, (long)self); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											1993-05-19 14:50:45 +00:00
										 |  |  | 	return newstringobject(buf); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1993-03-29 10:43:31 +00:00
										 |  |  | static long | 
					
						
							|  |  |  | instancemethod_hash(a) | 
					
						
							|  |  |  | 	instancemethodobject *a; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	long x, y; | 
					
						
							| 
									
										
										
										
											1993-05-20 14:24:46 +00:00
										 |  |  | 	if (a->im_self == NULL) | 
					
						
							|  |  |  | 		x = hashobject(None); | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 		x = hashobject(a->im_self); | 
					
						
							| 
									
										
										
										
											1993-03-29 10:43:31 +00:00
										 |  |  | 	if (x == -1) | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	y = hashobject(a->im_func); | 
					
						
							|  |  |  | 	if (y == -1) | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	return x ^ y; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1991-05-05 20:03:07 +00:00
										 |  |  | typeobject Instancemethodtype = { | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | 	OB_HEAD_INIT(&Typetype) | 
					
						
							|  |  |  | 	0, | 
					
						
							| 
									
										
										
										
											1991-04-16 08:38:43 +00:00
										 |  |  | 	"instance method", | 
					
						
							| 
									
										
										
										
											1991-05-05 20:03:07 +00:00
										 |  |  | 	sizeof(instancemethodobject), | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | 	0, | 
					
						
							| 
									
										
										
										
											1991-05-05 20:03:07 +00:00
										 |  |  | 	instancemethod_dealloc,	/*tp_dealloc*/ | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | 	0,			/*tp_print*/ | 
					
						
							| 
									
										
										
										
											1991-05-05 20:03:07 +00:00
										 |  |  | 	instancemethod_getattr,	/*tp_getattr*/ | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | 	0,			/*tp_setattr*/ | 
					
						
							| 
									
										
										
										
											1992-09-03 20:39:51 +00:00
										 |  |  | 	instancemethod_compare,	/*tp_compare*/ | 
					
						
							| 
									
										
										
										
											1993-05-19 14:50:45 +00:00
										 |  |  | 	instancemethod_repr,	/*tp_repr*/ | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | 	0,			/*tp_as_number*/ | 
					
						
							|  |  |  | 	0,			/*tp_as_sequence*/ | 
					
						
							|  |  |  | 	0,			/*tp_as_mapping*/ | 
					
						
							| 
									
										
										
										
											1993-03-29 10:43:31 +00:00
										 |  |  | 	instancemethod_hash,	/*tp_hash*/ | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | }; |