208 lines
4.3 KiB
C
208 lines
4.3 KiB
C
/***********************************************************
|
|
Copyright 1991, 1992, 1993, 1994 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.
|
|
|
|
******************************************************************/
|
|
|
|
/* Method object implementation */
|
|
|
|
#include "allobjects.h"
|
|
|
|
#include "token.h"
|
|
|
|
typedef struct {
|
|
OB_HEAD
|
|
char *m_name;
|
|
method m_meth;
|
|
object *m_self;
|
|
int m_varargs;
|
|
} methodobject;
|
|
|
|
object *
|
|
newmethodobject(name, meth, self, varargs)
|
|
char *name; /* static string */
|
|
method meth;
|
|
object *self;
|
|
int varargs;
|
|
{
|
|
methodobject *op = NEWOBJ(methodobject, &Methodtype);
|
|
if (op != NULL) {
|
|
op->m_name = name;
|
|
op->m_meth = meth;
|
|
if (self != NULL)
|
|
INCREF(self);
|
|
op->m_self = self;
|
|
op->m_varargs = varargs;
|
|
}
|
|
return (object *)op;
|
|
}
|
|
|
|
method
|
|
getmethod(op)
|
|
object *op;
|
|
{
|
|
if (!is_methodobject(op)) {
|
|
err_badcall();
|
|
return NULL;
|
|
}
|
|
return ((methodobject *)op) -> m_meth;
|
|
}
|
|
|
|
object *
|
|
getself(op)
|
|
object *op;
|
|
{
|
|
if (!is_methodobject(op)) {
|
|
err_badcall();
|
|
return NULL;
|
|
}
|
|
return ((methodobject *)op) -> m_self;
|
|
}
|
|
|
|
int
|
|
getvarargs(op)
|
|
object *op;
|
|
{
|
|
if (!is_methodobject(op)) {
|
|
err_badcall();
|
|
return -1;
|
|
}
|
|
return ((methodobject *)op) -> m_varargs;
|
|
}
|
|
|
|
/* Methods (the standard built-in methods, that is) */
|
|
|
|
static void
|
|
meth_dealloc(m)
|
|
methodobject *m;
|
|
{
|
|
if (m->m_self != NULL)
|
|
DECREF(m->m_self);
|
|
free((char *)m);
|
|
}
|
|
|
|
static object *
|
|
meth_repr(m)
|
|
methodobject *m;
|
|
{
|
|
char buf[200];
|
|
if (m->m_self == NULL)
|
|
sprintf(buf, "<built-in function %.80s>", m->m_name);
|
|
else
|
|
sprintf(buf,
|
|
"<built-in method %.80s of %.80s object at %lx>",
|
|
m->m_name, m->m_self->ob_type->tp_name,
|
|
(long)m->m_self);
|
|
return newstringobject(buf);
|
|
}
|
|
|
|
static int
|
|
meth_compare(a, b)
|
|
methodobject *a, *b;
|
|
{
|
|
if (a->m_self != b->m_self)
|
|
return cmpobject(a->m_self, b->m_self);
|
|
if (a->m_meth == b->m_meth)
|
|
return 0;
|
|
if (strcmp(a->m_name, b->m_name) < 0)
|
|
return -1;
|
|
else
|
|
return 1;
|
|
}
|
|
|
|
static long
|
|
meth_hash(a)
|
|
methodobject *a;
|
|
{
|
|
long x, y;
|
|
if (a->m_self == NULL)
|
|
x = 0;
|
|
else {
|
|
x = hashobject(a->m_self);
|
|
if (x == -1)
|
|
return -1;
|
|
}
|
|
return x ^ (long) a->m_meth;
|
|
}
|
|
|
|
typeobject Methodtype = {
|
|
OB_HEAD_INIT(&Typetype)
|
|
0,
|
|
"builtin_function_or_method",
|
|
sizeof(methodobject),
|
|
0,
|
|
(destructor)meth_dealloc, /*tp_dealloc*/
|
|
0, /*tp_print*/
|
|
0, /*tp_getattr*/
|
|
0, /*tp_setattr*/
|
|
(cmpfunc)meth_compare, /*tp_compare*/
|
|
(reprfunc)meth_repr, /*tp_repr*/
|
|
0, /*tp_as_number*/
|
|
0, /*tp_as_sequence*/
|
|
0, /*tp_as_mapping*/
|
|
(hashfunc)meth_hash, /*tp_hash*/
|
|
};
|
|
|
|
static object *listmethods PROTO((struct methodlist *)); /* Forward */
|
|
|
|
static object *
|
|
listmethods(ml)
|
|
struct methodlist *ml;
|
|
{
|
|
int i, n;
|
|
object *v;
|
|
for (n = 0; ml[n].ml_name != NULL; n++)
|
|
;
|
|
v = newlistobject(n);
|
|
if (v != NULL) {
|
|
for (i = 0; i < n; i++)
|
|
setlistitem(v, i, newstringobject(ml[i].ml_name));
|
|
if (err_occurred()) {
|
|
DECREF(v);
|
|
v = NULL;
|
|
}
|
|
else {
|
|
sortlist(v);
|
|
}
|
|
}
|
|
return v;
|
|
}
|
|
|
|
/* Find a method in a module's method table.
|
|
Usually called from an object's getattr method. */
|
|
|
|
object *
|
|
findmethod(ml, op, name)
|
|
struct methodlist *ml;
|
|
object *op;
|
|
char *name;
|
|
{
|
|
if (strcmp(name, "__methods__") == 0)
|
|
return listmethods(ml);
|
|
for (; ml->ml_name != NULL; ml++) {
|
|
if (strcmp(name, ml->ml_name) == 0)
|
|
return newmethodobject(ml->ml_name, ml->ml_meth,
|
|
op, ml->ml_varargs);
|
|
}
|
|
err_setstr(AttributeError, name);
|
|
return NULL;
|
|
}
|