home *** CD-ROM | disk | FTP | other *** search
- /***********************************************************
- 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 or Corporation for National Research Initiatives or
- CNRI not be used in advertising or publicity pertaining to
- distribution of the software without specific, written prior
- permission.
-
- While CWI is the initial source for this software, a modified version
- is made available by the Corporation for National Research Initiatives
- (CNRI) at the Internet address ftp://ftp.python.org.
-
- STICHTING MATHEMATISCH CENTRUM AND CNRI DISCLAIM ALL WARRANTIES WITH
- REGARD TO THIS SOFTWARE, INCLUDING ALL IMPLIED WARRANTIES OF
- MERCHANTABILITY AND FITNESS, IN NO EVENT SHALL STICHTING MATHEMATISCH
- CENTRUM OR CNRI 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 "Python.h"
- #include "compile.h"
- #include "structmember.h"
-
- #include "protos/funcobject.h"
-
- PyObject *
- PyFunction_New(code, globals)
- PyObject *code;
- PyObject *globals;
- {
- PyFunctionObject *op = PyObject_NEW(PyFunctionObject,
- &PyFunction_Type);
- if (op != NULL) {
- PyObject *doc;
- PyObject *consts;
- Py_INCREF(code);
- op->func_code = code;
- Py_INCREF(globals);
- op->func_globals = globals;
- op->func_name = ((PyCodeObject *)code)->co_name;
- Py_INCREF(op->func_name);
- op->func_defaults = NULL; /* No default arguments */
- consts = ((PyCodeObject *)code)->co_consts;
- if (PyTuple_Size(consts) >= 1) {
- doc = PyTuple_GetItem(consts, 0);
- if (!PyString_Check(doc))
- doc = Py_None;
- }
- else
- doc = Py_None;
- Py_INCREF(doc);
- op->func_doc = doc;
- }
- return (PyObject *)op;
- }
-
- PyObject *
- PyFunction_GetCode(op)
- PyObject *op;
- {
- if (!PyFunction_Check(op)) {
- PyErr_BadInternalCall();
- return NULL;
- }
- return ((PyFunctionObject *) op) -> func_code;
- }
-
- PyObject *
- PyFunction_GetGlobals(op)
- PyObject *op;
- {
- if (!PyFunction_Check(op)) {
- PyErr_BadInternalCall();
- return NULL;
- }
- return ((PyFunctionObject *) op) -> func_globals;
- }
-
- PyObject *
- PyFunction_GetDefaults(op)
- PyObject *op;
- {
- if (!PyFunction_Check(op)) {
- PyErr_BadInternalCall();
- return NULL;
- }
- return ((PyFunctionObject *) op) -> func_defaults;
- }
-
- int
- PyFunction_SetDefaults(op, defaults)
- PyObject *op;
- PyObject *defaults;
- {
- if (!PyFunction_Check(op)) {
- PyErr_BadInternalCall();
- return -1;
- }
- if (defaults == Py_None)
- defaults = NULL;
- else if (PyTuple_Check(defaults)) {
- Py_XINCREF(defaults);
- }
- else {
- PyErr_SetString(PyExc_SystemError, "non-tuple default args");
- return -1;
- }
- Py_XDECREF(((PyFunctionObject *) op) -> func_defaults);
- ((PyFunctionObject *) op) -> func_defaults = defaults;
- return 0;
- }
-
- /* Methods */
-
- #define OFF(x) offsetof(PyFunctionObject, x)
-
- static struct memberlist func_memberlist[] = {
- {"func_code", T_OBJECT, OFF(func_code)},
- {"func_globals",T_OBJECT, OFF(func_globals), READONLY},
- {"func_name", T_OBJECT, OFF(func_name), READONLY},
- {"__name__", T_OBJECT, OFF(func_name), READONLY},
- {"func_defaults",T_OBJECT, OFF(func_defaults)},
- {"func_doc", T_OBJECT, OFF(func_doc)},
- {"__doc__", T_OBJECT, OFF(func_doc)},
- {NULL} /* Sentinel */
- };
-
- static PyObject *
- func_getattr(op, name)
- PyFunctionObject *op;
- char *name;
- {
- if (name[0] != '_' && PyEval_GetRestricted()) {
- PyErr_SetString(PyExc_RuntimeError,
- "function attributes not accessible in restricted mode");
- return NULL;
- }
- return PyMember_Get((char *)op, func_memberlist, name);
- }
-
- static int
- func_setattr(op, name, value)
- PyFunctionObject *op;
- char *name;
- PyObject *value;
- {
- if (PyEval_GetRestricted()) {
- PyErr_SetString(PyExc_RuntimeError,
- "function attributes not settable in restricted mode");
- return -1;
- }
- if (strcmp(name, "func_code") == 0) {
- if (value == NULL || !PyCode_Check(value)) {
- PyErr_SetString(
- PyExc_TypeError,
- "func_code must be set to a code object");
- return -1;
- }
- }
- else if (strcmp(name, "func_defaults") == 0) {
- if (value != Py_None && !PyTuple_Check(value)) {
- PyErr_SetString(
- PyExc_TypeError,
- "func_defaults must be set to a tuple object");
- return -1;
- }
- if (value == Py_None)
- value = NULL;
- }
- return PyMember_Set((char *)op, func_memberlist, name, value);
- }
-
- static void
- func_dealloc(op)
- PyFunctionObject *op;
- {
- Py_DECREF(op->func_code);
- Py_DECREF(op->func_globals);
- Py_DECREF(op->func_name);
- Py_XDECREF(op->func_defaults);
- Py_XDECREF(op->func_doc);
- PyMem_DEL(op);
- }
-
- static PyObject*
- func_repr(op)
- PyFunctionObject *op;
- {
- char buf[140];
- if (op->func_name == Py_None)
- sprintf(buf, "<anonymous function at %lx>", (long)op);
- else
- sprintf(buf, "<function %.100s at %lx>",
- PyString_AsString(op->func_name),
- (long)op);
- return PyString_FromString(buf);
- }
-
- static int
- func_compare(f, g)
- PyFunctionObject *f, *g;
- {
- int c;
- if (f->func_globals != g->func_globals)
- return (f->func_globals < g->func_globals) ? -1 : 1;
- if (f->func_defaults != g->func_defaults) {
- if (f->func_defaults == NULL)
- return -1;
- if (g->func_defaults == NULL)
- return 1;
- c = PyObject_Compare(f->func_defaults, g->func_defaults);
- if (c != 0)
- return c;
- }
- return PyObject_Compare(f->func_code, g->func_code);
- }
-
- static long
- func_hash(f)
- PyFunctionObject *f;
- {
- long h;
- h = PyObject_Hash(f->func_code);
- if (h == -1) return h;
- h = h ^ (long)f->func_globals;
- if (h == -1) h = -2;
- return h;
- }
-
- PyTypeObject PyFunction_Type = {
- PyObject_HEAD_INIT(&PyType_Type)
- 0,
- "function",
- sizeof(PyFunctionObject),
- 0,
- (destructor)func_dealloc, /*tp_dealloc*/
- 0, /*tp_print*/
- (getattrfunc)func_getattr, /*tp_getattr*/
- (setattrfunc)func_setattr, /*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*/
- };
-