mirror of
				https://github.com/python/cpython.git
				synced 2025-11-04 11:49:12 +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*/
 | 
						|
};
 |