Commit 04691fc1 authored by Guido van Rossum's avatar Guido van Rossum

Changes so that user-defined classes can implement operations invoked

by special syntax: you can now define your own numbers, sequences and
mappings.
parent 423d6c6b
...@@ -77,13 +77,13 @@ void flushline PROTO((void)); ...@@ -77,13 +77,13 @@ void flushline PROTO((void));
*/ */
extern void init_save_thread PROTO((void)); extern void init_save_thread PROTO((void));
extern void *save_thread PROTO((void)); extern object *save_thread PROTO((void));
extern void restore_thread PROTO((void *)); extern void restore_thread PROTO((object *));
#ifdef USE_THREAD #ifdef USE_THREAD
#define BGN_SAVE { \ #define BGN_SAVE { \
void *_save; \ object *_save; \
_save = save_thread(); _save = save_thread();
#define RET_SAVE restore_thread(_save); #define RET_SAVE restore_thread(_save);
#define RES_SAVE _save = save_thread(); #define RES_SAVE _save = save_thread();
......
...@@ -42,3 +42,6 @@ extern object *newinstancemethodobject PROTO((object *, object *)); ...@@ -42,3 +42,6 @@ extern object *newinstancemethodobject PROTO((object *, object *));
extern object *instancemethodgetfunc PROTO((object *)); extern object *instancemethodgetfunc PROTO((object *));
extern object *instancemethodgetself PROTO((object *)); extern object *instancemethodgetself PROTO((object *));
extern int instance_coerce PROTO((object **, object **));
extern object *instance_convert PROTO((object *, char *));
This diff is collapsed.
...@@ -84,6 +84,24 @@ builtin_chr(self, v) ...@@ -84,6 +84,24 @@ builtin_chr(self, v)
return newsizedstringobject(s, 1); return newsizedstringobject(s, 1);
} }
static object *
builtin_coerce(self, args)
object *self;
object *args;
{
object *v, *w;
object *res;
if (!getargs(args, "(OO)", &v, &w))
return NULL;
if (coerce(&v, &w) < 0)
return NULL;
res = mkvalue("(OO)", v, w);
DECREF(v);
DECREF(w);
return res;
}
static object * static object *
builtin_dir(self, v) builtin_dir(self, v)
object *self; object *self;
...@@ -250,6 +268,9 @@ builtin_float(self, v) ...@@ -250,6 +268,9 @@ builtin_float(self, v)
INCREF(v); INCREF(v);
return v; return v;
} }
else if (is_instanceobject(v)) {
return instance_convert(v, "__float__");
}
err_setstr(TypeError, "float() argument must be int, long or float"); err_setstr(TypeError, "float() argument must be int, long or float");
return NULL; return NULL;
} }
...@@ -359,6 +380,9 @@ builtin_int(self, v) ...@@ -359,6 +380,9 @@ builtin_int(self, v)
/* XXX should check for overflow */ /* XXX should check for overflow */
return newintobject((long)x); return newintobject((long)x);
} }
else if (is_instanceobject(v)) {
return instance_convert(v, "__int__");
}
err_setstr(TypeError, "int() argument must be int, long or float"); err_setstr(TypeError, "int() argument must be int, long or float");
return NULL; return NULL;
} }
...@@ -385,7 +409,10 @@ builtin_len(self, v) ...@@ -385,7 +409,10 @@ builtin_len(self, v)
err_setstr(TypeError, "len() of unsized object"); err_setstr(TypeError, "len() of unsized object");
return NULL; return NULL;
} }
return newintobject(len); if (len < 0)
return NULL;
else
return newintobject(len);
} }
static object * static object *
...@@ -407,6 +434,9 @@ builtin_long(self, v) ...@@ -407,6 +434,9 @@ builtin_long(self, v)
double x = getfloatvalue(v); double x = getfloatvalue(v);
return dnewlongobject(x); return dnewlongobject(x);
} }
else if (is_instanceobject(v)) {
return instance_convert(v, "__long__");
}
err_setstr(TypeError, "long() argument must be int, long or float"); err_setstr(TypeError, "long() argument must be int, long or float");
return NULL; return NULL;
} }
...@@ -648,6 +678,7 @@ static struct methodlist builtin_methods[] = { ...@@ -648,6 +678,7 @@ static struct methodlist builtin_methods[] = {
{"abs", builtin_abs}, {"abs", builtin_abs},
{"apply", builtin_apply}, {"apply", builtin_apply},
{"chr", builtin_chr}, {"chr", builtin_chr},
{"coerce", builtin_coerce},
{"dir", builtin_dir}, {"dir", builtin_dir},
{"divmod", builtin_divmod}, {"divmod", builtin_divmod},
{"eval", builtin_eval}, {"eval", builtin_eval},
...@@ -766,6 +797,8 @@ coerce(pv, pw) ...@@ -766,6 +797,8 @@ coerce(pv, pw)
INCREF(w); INCREF(w);
return 0; return 0;
} }
if (is_instanceobject(v) || is_instanceobject(w))
return instance_coerce(pv, pw);
if (v->ob_type->tp_as_number == NULL || if (v->ob_type->tp_as_number == NULL ||
w->ob_type->tp_as_number == NULL) { w->ob_type->tp_as_number == NULL) {
err_setstr(TypeError, "mixing number and non-number"); err_setstr(TypeError, "mixing number and non-number");
......
...@@ -36,6 +36,9 @@ OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. ...@@ -36,6 +36,9 @@ OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
#include "bltinmodule.h" #include "bltinmodule.h"
#include "traceback.h" #include "traceback.h"
/* Turn this on if your compiler chokes on the big switch: */
/* #define CASE_TOO_BIG 1 /**/
#ifndef NDEBUG #ifndef NDEBUG
/* For debugging the interpreter: */ /* For debugging the interpreter: */
#define LLTRACE 1 /* Low-level trace feature */ #define LLTRACE 1 /* Low-level trace feature */
...@@ -106,13 +109,13 @@ init_save_thread() ...@@ -106,13 +109,13 @@ init_save_thread()
dynamically loaded modules needn't be compiled separately for use dynamically loaded modules needn't be compiled separately for use
with and without threads: */ with and without threads: */
void * object *
save_thread() save_thread()
{ {
#ifdef USE_THREAD #ifdef USE_THREAD
if (interpreter_lock) { if (interpreter_lock) {
void *res; object *res;
res = (void *)current_frame; res = (object *)current_frame;
current_frame = NULL; current_frame = NULL;
release_lock(interpreter_lock); release_lock(interpreter_lock);
return res; return res;
...@@ -124,7 +127,7 @@ save_thread() ...@@ -124,7 +127,7 @@ save_thread()
void void
restore_thread(x) restore_thread(x)
void *x; object *x;
{ {
#ifdef USE_THREAD #ifdef USE_THREAD
if (interpreter_lock) { if (interpreter_lock) {
...@@ -722,6 +725,10 @@ eval_code(co, globals, locals, arg) ...@@ -722,6 +725,10 @@ eval_code(co, globals, locals, arg)
if ((err = dict2remove(f->f_locals, w)) != 0) if ((err = dict2remove(f->f_locals, w)) != 0)
err_setstr(NameError, getstringvalue(w)); err_setstr(NameError, getstringvalue(w));
break; break;
#ifdef CASE_TOO_BIG
default: switch (opcode) {
#endif
case UNPACK_VARARG: case UNPACK_VARARG:
if (EMPTY()) { if (EMPTY()) {
...@@ -1023,13 +1030,19 @@ eval_code(co, globals, locals, arg) ...@@ -1023,13 +1030,19 @@ eval_code(co, globals, locals, arg)
break; break;
case JUMP_IF_FALSE: case JUMP_IF_FALSE:
if (!testbool(TOP())) err = testbool(TOP());
if (err > 0)
err = 0;
else if (err == 0)
JUMPBY(oparg); JUMPBY(oparg);
break; break;
case JUMP_IF_TRUE: case JUMP_IF_TRUE:
if (testbool(TOP())) err = testbool(TOP());
if (err > 0) {
err = 0;
JUMPBY(oparg); JUMPBY(oparg);
}
break; break;
case JUMP_ABSOLUTE: case JUMP_ABSOLUTE:
...@@ -1092,7 +1105,11 @@ eval_code(co, globals, locals, arg) ...@@ -1092,7 +1105,11 @@ eval_code(co, globals, locals, arg)
err_setstr(SystemError, "eval_code: unknown opcode"); err_setstr(SystemError, "eval_code: unknown opcode");
why = WHY_EXCEPTION; why = WHY_EXCEPTION;
break; break;
#ifdef CASE_TOO_BIG
}
#endif
} /* switch */ } /* switch */
on_error: on_error:
...@@ -1388,22 +1405,27 @@ flushline() ...@@ -1388,22 +1405,27 @@ flushline()
} }
/* Test a value used as condition, e.g., in a for or if statement */ /* Test a value used as condition, e.g., in a for or if statement.
Return -1 if an error occurred */
static int static int
testbool(v) testbool(v)
object *v; object *v;
{ {
int res;
if (v == None) if (v == None)
return 0; res = 0;
if (v->ob_type->tp_as_number != NULL) else if (v->ob_type->tp_as_number != NULL)
return (*v->ob_type->tp_as_number->nb_nonzero)(v); res = (*v->ob_type->tp_as_number->nb_nonzero)(v);
if (v->ob_type->tp_as_sequence != NULL) else if (v->ob_type->tp_as_mapping != NULL)
return (*v->ob_type->tp_as_sequence->sq_length)(v) != 0; res = (*v->ob_type->tp_as_mapping->mp_length)(v);
if (v->ob_type->tp_as_mapping != NULL) else if (v->ob_type->tp_as_sequence != NULL)
return (*v->ob_type->tp_as_mapping->mp_length)(v) != 0; res = (*v->ob_type->tp_as_sequence->sq_length)(v);
/* All other objects are 'true' */ else
return 1; res = 0;
if (res > 0)
res = 1;
return res;
} }
static object * static object *
...@@ -1649,7 +1671,13 @@ not(v) ...@@ -1649,7 +1671,13 @@ not(v)
object *v; object *v;
{ {
int outcome = testbool(v); int outcome = testbool(v);
object *w = outcome == 0 ? True : False; object *w;
if (outcome < 0)
return NULL;
if (outcome == 0)
w = True;
else
w = False;
INCREF(w); INCREF(w);
return w; return w;
} }
...@@ -1780,18 +1808,24 @@ apply_subscript(v, w) ...@@ -1780,18 +1808,24 @@ apply_subscript(v, w)
err_setstr(TypeError, "unsubscriptable object"); err_setstr(TypeError, "unsubscriptable object");
return NULL; return NULL;
} }
if (tp->tp_as_sequence != NULL) { if (tp->tp_as_mapping != NULL) {
return (*tp->tp_as_mapping->mp_subscript)(v, w);
}
else {
int i; int i;
if (!is_intobject(w)) { if (!is_intobject(w)) {
err_setstr(TypeError, "sequence subscript not int"); err_setstr(TypeError, "sequence subscript not int");
return NULL; return NULL;
} }
i = getintvalue(w); i = getintvalue(w);
if (i < 0) if (i < 0) {
i += (*tp->tp_as_sequence->sq_length)(v); int len = (*tp->tp_as_sequence->sq_length)(v);
if (len < 0)
return NULL;
i += len;
}
return (*tp->tp_as_sequence->sq_item)(v, i); return (*tp->tp_as_sequence->sq_item)(v, i);
} }
return (*tp->tp_as_mapping->mp_subscript)(v, w);
} }
static object * static object *
...@@ -1841,6 +1875,8 @@ apply_slice(u, v, w) /* return u[v:w] */ ...@@ -1841,6 +1875,8 @@ apply_slice(u, v, w) /* return u[v:w] */
} }
ilow = 0; ilow = 0;
isize = ihigh = (*tp->tp_as_sequence->sq_length)(u); isize = ihigh = (*tp->tp_as_sequence->sq_length)(u);
if (isize < 0)
return NULL;
if (slice_index(v, isize, &ilow) != 0) if (slice_index(v, isize, &ilow) != 0)
return NULL; return NULL;
if (slice_index(w, isize, &ihigh) != 0) if (slice_index(w, isize, &ihigh) != 0)
...@@ -1858,7 +1894,11 @@ assign_subscript(w, key, v) /* w[key] = v */ ...@@ -1858,7 +1894,11 @@ assign_subscript(w, key, v) /* w[key] = v */
sequence_methods *sq; sequence_methods *sq;
mapping_methods *mp; mapping_methods *mp;
int (*func)(); int (*func)();
if ((sq = tp->tp_as_sequence) != NULL && if ((mp = tp->tp_as_mapping) != NULL &&
(func = mp->mp_ass_subscript) != NULL) {
return (*func)(w, key, v);
}
else if ((sq = tp->tp_as_sequence) != NULL &&
(func = sq->sq_ass_item) != NULL) { (func = sq->sq_ass_item) != NULL) {
if (!is_intobject(key)) { if (!is_intobject(key)) {
err_setstr(TypeError, err_setstr(TypeError,
...@@ -1867,15 +1907,15 @@ assign_subscript(w, key, v) /* w[key] = v */ ...@@ -1867,15 +1907,15 @@ assign_subscript(w, key, v) /* w[key] = v */
} }
else { else {
int i = getintvalue(key); int i = getintvalue(key);
if (i < 0) if (i < 0) {
i += (*sq->sq_length)(w); int len = (*sq->sq_length)(w);
if (len < 0)
return -1;
i += len;
}
return (*func)(w, i, v); return (*func)(w, i, v);
} }
} }
else if ((mp = tp->tp_as_mapping) != NULL &&
(func = mp->mp_ass_subscript) != NULL) {
return (*func)(w, key, v);
}
else { else {
err_setstr(TypeError, err_setstr(TypeError,
"can't assign to this subscripted object"); "can't assign to this subscripted object");
...@@ -1899,6 +1939,8 @@ assign_slice(u, v, w, x) /* u[v:w] = x */ ...@@ -1899,6 +1939,8 @@ assign_slice(u, v, w, x) /* u[v:w] = x */
} }
ilow = 0; ilow = 0;
isize = ihigh = (*sq->sq_length)(u); isize = ihigh = (*sq->sq_length)(u);
if (isize < 0)
return -1;
if (slice_index(v, isize, &ilow) != 0) if (slice_index(v, isize, &ilow) != 0)
return -1; return -1;
if (slice_index(w, isize, &ihigh) != 0) if (slice_index(w, isize, &ihigh) != 0)
...@@ -1955,6 +1997,8 @@ cmp_member(v, w) ...@@ -1955,6 +1997,8 @@ cmp_member(v, w)
return -1; return -1;
} }
n = (*sq->sq_length)(w); n = (*sq->sq_length)(w);
if (n < 0)
return -1;
for (i = 0; i < n; i++) { for (i = 0; i < n; i++) {
x = (*sq->sq_item)(w, i); x = (*sq->sq_item)(w, i);
cmp = cmpobject(v, x); cmp = cmpobject(v, x);
...@@ -1977,7 +2021,7 @@ cmp_outcome(op, v, w) ...@@ -1977,7 +2021,7 @@ cmp_outcome(op, v, w)
case IS: case IS:
case IS_NOT: case IS_NOT:
res = (v == w); res = (v == w);
if (op == IS_NOT) if (op == (int) IS_NOT)
res = !res; res = !res;
break; break;
case IN: case IN:
...@@ -1985,7 +2029,7 @@ cmp_outcome(op, v, w) ...@@ -1985,7 +2029,7 @@ cmp_outcome(op, v, w)
res = cmp_member(v, w); res = cmp_member(v, w);
if (res < 0) if (res < 0)
return NULL; return NULL;
if (op == NOT_IN) if (op == (int) NOT_IN)
res = !res; res = !res;
break; break;
case EXC_MATCH: case EXC_MATCH:
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment