summaryrefslogtreecommitdiffstats
path: root/Python
diff options
context:
space:
mode:
authorGuido van Rossum <guido@python.org>1992-08-12 15:35:34 (GMT)
committerGuido van Rossum <guido@python.org>1992-08-12 15:35:34 (GMT)
commit04691fc1c1bb737c0db772f5c1ea697a351a01d9 (patch)
tree80d559ec0adb25dd1774638ebb0f2bc8aa8afc67 /Python
parent423d6c6bcae783f8ebfcb0b692059eb294ab14dd (diff)
downloadcpython-04691fc1c1bb737c0db772f5c1ea697a351a01d9.zip
cpython-04691fc1c1bb737c0db772f5c1ea697a351a01d9.tar.gz
cpython-04691fc1c1bb737c0db772f5c1ea697a351a01d9.tar.bz2
Changes so that user-defined classes can implement operations invoked
by special syntax: you can now define your own numbers, sequences and mappings.
Diffstat (limited to 'Python')
-rw-r--r--Python/bltinmodule.c35
-rw-r--r--Python/ceval.c106
2 files changed, 109 insertions, 32 deletions
diff --git a/Python/bltinmodule.c b/Python/bltinmodule.c
index ca5043d..cce19e1 100644
--- a/Python/bltinmodule.c
+++ b/Python/bltinmodule.c
@@ -85,6 +85,24 @@ builtin_chr(self, v)
}
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 *
builtin_dir(self, v)
object *self;
object *v;
@@ -250,6 +268,9 @@ builtin_float(self, v)
INCREF(v);
return v;
}
+ else if (is_instanceobject(v)) {
+ return instance_convert(v, "__float__");
+ }
err_setstr(TypeError, "float() argument must be int, long or float");
return NULL;
}
@@ -359,6 +380,9 @@ builtin_int(self, v)
/* XXX should check for overflow */
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");
return NULL;
}
@@ -385,7 +409,10 @@ builtin_len(self, v)
err_setstr(TypeError, "len() of unsized object");
return NULL;
}
- return newintobject(len);
+ if (len < 0)
+ return NULL;
+ else
+ return newintobject(len);
}
static object *
@@ -407,6 +434,9 @@ builtin_long(self, v)
double x = getfloatvalue(v);
return dnewlongobject(x);
}
+ else if (is_instanceobject(v)) {
+ return instance_convert(v, "__long__");
+ }
err_setstr(TypeError, "long() argument must be int, long or float");
return NULL;
}
@@ -648,6 +678,7 @@ static struct methodlist builtin_methods[] = {
{"abs", builtin_abs},
{"apply", builtin_apply},
{"chr", builtin_chr},
+ {"coerce", builtin_coerce},
{"dir", builtin_dir},
{"divmod", builtin_divmod},
{"eval", builtin_eval},
@@ -766,6 +797,8 @@ coerce(pv, pw)
INCREF(w);
return 0;
}
+ if (is_instanceobject(v) || is_instanceobject(w))
+ return instance_coerce(pv, pw);
if (v->ob_type->tp_as_number == NULL ||
w->ob_type->tp_as_number == NULL) {
err_setstr(TypeError, "mixing number and non-number");
diff --git a/Python/ceval.c b/Python/ceval.c
index 252b1dc..ad679a6 100644
--- a/Python/ceval.c
+++ b/Python/ceval.c
@@ -36,6 +36,9 @@ OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
#include "bltinmodule.h"
#include "traceback.h"
+/* Turn this on if your compiler chokes on the big switch: */
+/* #define CASE_TOO_BIG 1 /**/
+
#ifndef NDEBUG
/* For debugging the interpreter: */
#define LLTRACE 1 /* Low-level trace feature */
@@ -106,13 +109,13 @@ init_save_thread()
dynamically loaded modules needn't be compiled separately for use
with and without threads: */
-void *
+object *
save_thread()
{
#ifdef USE_THREAD
if (interpreter_lock) {
- void *res;
- res = (void *)current_frame;
+ object *res;
+ res = (object *)current_frame;
current_frame = NULL;
release_lock(interpreter_lock);
return res;
@@ -124,7 +127,7 @@ save_thread()
void
restore_thread(x)
- void *x;
+ object *x;
{
#ifdef USE_THREAD
if (interpreter_lock) {
@@ -722,6 +725,10 @@ eval_code(co, globals, locals, arg)
if ((err = dict2remove(f->f_locals, w)) != 0)
err_setstr(NameError, getstringvalue(w));
break;
+
+#ifdef CASE_TOO_BIG
+ default: switch (opcode) {
+#endif
case UNPACK_VARARG:
if (EMPTY()) {
@@ -1023,13 +1030,19 @@ eval_code(co, globals, locals, arg)
break;
case JUMP_IF_FALSE:
- if (!testbool(TOP()))
+ err = testbool(TOP());
+ if (err > 0)
+ err = 0;
+ else if (err == 0)
JUMPBY(oparg);
break;
case JUMP_IF_TRUE:
- if (testbool(TOP()))
+ err = testbool(TOP());
+ if (err > 0) {
+ err = 0;
JUMPBY(oparg);
+ }
break;
case JUMP_ABSOLUTE:
@@ -1092,7 +1105,11 @@ eval_code(co, globals, locals, arg)
err_setstr(SystemError, "eval_code: unknown opcode");
why = WHY_EXCEPTION;
break;
-
+
+#ifdef CASE_TOO_BIG
+ }
+#endif
+
} /* switch */
on_error:
@@ -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
testbool(v)
object *v;
{
+ int res;
if (v == None)
- return 0;
- if (v->ob_type->tp_as_number != NULL)
- return (*v->ob_type->tp_as_number->nb_nonzero)(v);
- if (v->ob_type->tp_as_sequence != NULL)
- return (*v->ob_type->tp_as_sequence->sq_length)(v) != 0;
- if (v->ob_type->tp_as_mapping != NULL)
- return (*v->ob_type->tp_as_mapping->mp_length)(v) != 0;
- /* All other objects are 'true' */
- return 1;
+ res = 0;
+ else if (v->ob_type->tp_as_number != NULL)
+ res = (*v->ob_type->tp_as_number->nb_nonzero)(v);
+ else if (v->ob_type->tp_as_mapping != NULL)
+ res = (*v->ob_type->tp_as_mapping->mp_length)(v);
+ else if (v->ob_type->tp_as_sequence != NULL)
+ res = (*v->ob_type->tp_as_sequence->sq_length)(v);
+ else
+ res = 0;
+ if (res > 0)
+ res = 1;
+ return res;
}
static object *
@@ -1649,7 +1671,13 @@ not(v)
object *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);
return w;
}
@@ -1780,18 +1808,24 @@ apply_subscript(v, w)
err_setstr(TypeError, "unsubscriptable object");
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;
if (!is_intobject(w)) {
err_setstr(TypeError, "sequence subscript not int");
return NULL;
}
i = getintvalue(w);
- if (i < 0)
- i += (*tp->tp_as_sequence->sq_length)(v);
+ if (i < 0) {
+ 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_mapping->mp_subscript)(v, w);
}
static object *
@@ -1841,6 +1875,8 @@ apply_slice(u, v, w) /* return u[v:w] */
}
ilow = 0;
isize = ihigh = (*tp->tp_as_sequence->sq_length)(u);
+ if (isize < 0)
+ return NULL;
if (slice_index(v, isize, &ilow) != 0)
return NULL;
if (slice_index(w, isize, &ihigh) != 0)
@@ -1858,7 +1894,11 @@ assign_subscript(w, key, v) /* w[key] = v */
sequence_methods *sq;
mapping_methods *mp;
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) {
if (!is_intobject(key)) {
err_setstr(TypeError,
@@ -1867,15 +1907,15 @@ assign_subscript(w, key, v) /* w[key] = v */
}
else {
int i = getintvalue(key);
- if (i < 0)
- i += (*sq->sq_length)(w);
+ if (i < 0) {
+ int len = (*sq->sq_length)(w);
+ if (len < 0)
+ return -1;
+ i += len;
+ }
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 {
err_setstr(TypeError,
"can't assign to this subscripted object");
@@ -1899,6 +1939,8 @@ assign_slice(u, v, w, x) /* u[v:w] = x */
}
ilow = 0;
isize = ihigh = (*sq->sq_length)(u);
+ if (isize < 0)
+ return -1;
if (slice_index(v, isize, &ilow) != 0)
return -1;
if (slice_index(w, isize, &ihigh) != 0)
@@ -1955,6 +1997,8 @@ cmp_member(v, w)
return -1;
}
n = (*sq->sq_length)(w);
+ if (n < 0)
+ return -1;
for (i = 0; i < n; i++) {
x = (*sq->sq_item)(w, i);
cmp = cmpobject(v, x);
@@ -1977,7 +2021,7 @@ cmp_outcome(op, v, w)
case IS:
case IS_NOT:
res = (v == w);
- if (op == IS_NOT)
+ if (op == (int) IS_NOT)
res = !res;
break;
case IN:
@@ -1985,7 +2029,7 @@ cmp_outcome(op, v, w)
res = cmp_member(v, w);
if (res < 0)
return NULL;
- if (op == NOT_IN)
+ if (op == (int) NOT_IN)
res = !res;
break;
case EXC_MATCH: