mirror of
				https://github.com/python/cpython.git
				synced 2025-10-31 05:31:20 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			221 lines
		
	
	
	
		
			5.2 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			221 lines
		
	
	
	
		
			5.2 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /***********************************************************
 | |
| Copyright 1991-1995 by Stichting Mathematisch Centrum, Amsterdam,
 | |
| The Netherlands.
 | |
| 
 | |
|                         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.
 | |
| 
 | |
| ******************************************************************/
 | |
| 
 | |
| /* Function object implementation */
 | |
| 
 | |
| #include "allobjects.h"
 | |
| #include "compile.h"
 | |
| #include "structmember.h"
 | |
| 
 | |
| object *
 | |
| newfuncobject(code, globals)
 | |
| 	object *code;
 | |
| 	object *globals;
 | |
| {
 | |
| 	funcobject *op = NEWOBJ(funcobject, &Functype);
 | |
| 	if (op != NULL) {
 | |
| 		object *doc;
 | |
| 		object *consts;
 | |
| 		INCREF(code);
 | |
| 		op->func_code = code;
 | |
| 		INCREF(globals);
 | |
| 		op->func_globals = globals;
 | |
| 		op->func_name = ((codeobject *)code)->co_name;
 | |
| 		INCREF(op->func_name);
 | |
| 		op->func_argcount = -1; /* Unknown */
 | |
| 		op->func_argdefs = NULL; /* No default arguments */
 | |
| 		consts = ((codeobject *)code)->co_consts;
 | |
| 		if (gettuplesize(consts) >= 1) {
 | |
| 			doc = gettupleitem(consts, 0);
 | |
| 			if (!is_stringobject(doc))
 | |
| 				doc = None;
 | |
| 		}
 | |
| 		else
 | |
| 			doc = None;
 | |
| 		INCREF(doc);
 | |
| 		op->func_doc = doc;
 | |
| 	}
 | |
| 	return (object *)op;
 | |
| }
 | |
| 
 | |
| object *
 | |
| getfunccode(op)
 | |
| 	object *op;
 | |
| {
 | |
| 	if (!is_funcobject(op)) {
 | |
| 		err_badcall();
 | |
| 		return NULL;
 | |
| 	}
 | |
| 	return ((funcobject *) op) -> func_code;
 | |
| }
 | |
| 
 | |
| object *
 | |
| getfuncglobals(op)
 | |
| 	object *op;
 | |
| {
 | |
| 	if (!is_funcobject(op)) {
 | |
| 		err_badcall();
 | |
| 		return NULL;
 | |
| 	}
 | |
| 	return ((funcobject *) op) -> func_globals;
 | |
| }
 | |
| 
 | |
| object *
 | |
| getfuncargstuff(op, argcount_return)
 | |
| 	object *op;
 | |
| 	int *argcount_return;
 | |
| {
 | |
| 	if (!is_funcobject(op)) {
 | |
| 		err_badcall();
 | |
| 		return NULL;
 | |
| 	}
 | |
| 	*argcount_return = ((funcobject *) op) -> func_argcount;
 | |
| 	return ((funcobject *) op) -> func_argdefs;
 | |
| }
 | |
| 
 | |
| int
 | |
| setfuncargstuff(op, argcount, argdefs)
 | |
| 	object *op;
 | |
| 	int argcount;
 | |
| 	object *argdefs;
 | |
| {
 | |
| 	if (!is_funcobject(op) ||
 | |
| 	    argdefs != NULL && !is_tupleobject(argdefs)) {
 | |
| 		err_badcall();
 | |
| 		return -1;
 | |
| 	}
 | |
| 	if (argdefs == None)
 | |
| 		argdefs = NULL;
 | |
| 	else if (is_tupleobject(argdefs))
 | |
| 		XINCREF(argdefs);
 | |
| 	else {
 | |
| 		err_setstr(SystemError, "non-tuple default args");
 | |
| 		return -1;
 | |
| 	}
 | |
| 	((funcobject *) op) -> func_argcount = argcount;
 | |
| 	XDECREF(((funcobject *) op) -> func_argdefs);
 | |
| 	((funcobject *) op) -> func_argdefs = argdefs;
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| /* Methods */
 | |
| 
 | |
| #define OFF(x) offsetof(funcobject, x)
 | |
| 
 | |
| static struct memberlist func_memberlist[] = {
 | |
| 	{"func_code",	T_OBJECT,	OFF(func_code),		READONLY},
 | |
| 	{"func_globals",T_OBJECT,	OFF(func_globals),	READONLY},
 | |
| 	{"func_name",	T_OBJECT,	OFF(func_name),		READONLY},
 | |
| 	{"__name__",	T_OBJECT,	OFF(func_name),		READONLY},
 | |
| 	{"func_argcount",T_INT,		OFF(func_argcount),	READONLY},
 | |
| 	{"func_argdefs",T_OBJECT,	OFF(func_argdefs),	READONLY},
 | |
| 	{"func_doc",	T_OBJECT,	OFF(func_doc)},
 | |
| 	{"__doc__",	T_OBJECT,	OFF(func_doc)},
 | |
| 	{NULL}	/* Sentinel */
 | |
| };
 | |
| 
 | |
| static object *
 | |
| func_getattr(op, name)
 | |
| 	funcobject *op;
 | |
| 	char *name;
 | |
| {
 | |
| 	if (name[0] != '_' && getrestricted()) {
 | |
| 		err_setstr(RuntimeError,
 | |
| 		  "function attributes not accessible in restricted mode");
 | |
| 		return NULL;
 | |
| 	}
 | |
| 	return getmember((char *)op, func_memberlist, name);
 | |
| }
 | |
| 
 | |
| static void
 | |
| func_dealloc(op)
 | |
| 	funcobject *op;
 | |
| {
 | |
| 	DECREF(op->func_code);
 | |
| 	DECREF(op->func_globals);
 | |
| 	DECREF(op->func_name);
 | |
| 	XDECREF(op->func_argdefs);
 | |
| 	XDECREF(op->func_doc);
 | |
| 	DEL(op);
 | |
| }
 | |
| 
 | |
| static object*
 | |
| func_repr(op)
 | |
| 	funcobject *op;
 | |
| {
 | |
| 	char buf[140];
 | |
| 	if (op->func_name == None)
 | |
| 		sprintf(buf, "<anonymous function at %lx>", (long)op);
 | |
| 	else
 | |
| 		sprintf(buf, "<function %.100s at %lx>",
 | |
| 			getstringvalue(op->func_name),
 | |
| 			(long)op);
 | |
| 	return newstringobject(buf);
 | |
| }
 | |
| 
 | |
| static int
 | |
| func_compare(f, g)
 | |
| 	funcobject *f, *g;
 | |
| {
 | |
| 	int c;
 | |
| 	if (f->func_globals != g->func_globals)
 | |
| 		return (f->func_globals < g->func_globals) ? -1 : 1;
 | |
| 	c = f->func_argcount < g->func_argcount;
 | |
| 	if (c != 0)
 | |
| 		return c < 0 ? -1 : 1;
 | |
| 	c = cmpobject(f->func_argdefs, g->func_argdefs);
 | |
| 	if (c != 0)
 | |
| 		return c;
 | |
| 	return cmpobject(f->func_code, g->func_code);
 | |
| }
 | |
| 
 | |
| static long
 | |
| func_hash(f)
 | |
| 	funcobject *f;
 | |
| {
 | |
| 	long h;
 | |
| 	h = hashobject(f->func_code);
 | |
| 	if (h == -1) return h;
 | |
| 	h = h ^ (long)f->func_globals;
 | |
| 	if (h == -1) h = -2;
 | |
| 	return h;
 | |
| }
 | |
| 
 | |
| typeobject Functype = {
 | |
| 	OB_HEAD_INIT(&Typetype)
 | |
| 	0,
 | |
| 	"function",
 | |
| 	sizeof(funcobject),
 | |
| 	0,
 | |
| 	(destructor)func_dealloc, /*tp_dealloc*/
 | |
| 	0,		/*tp_print*/
 | |
| 	(getattrfunc)func_getattr, /*tp_getattr*/
 | |
| 	0,		/*tp_setattr*/
 | |
| 	(cmpfunc)func_compare, /*tp_compare*/
 | |
| 	(reprfunc)func_repr, /*tp_repr*/
 | |
| 	0,		/*tp_as_number*/
 | |
| 	0,		/*tp_as_sequence*/
 | |
| 	0,		/*tp_as_mapping*/
 | |
| 	(hashfunc)func_hash, /*tp_hash*/
 | |
| };
 | 
