diff options
Diffstat (limited to 'Python')
-rw-r--r-- | Python/compile.c | 203 |
1 files changed, 176 insertions, 27 deletions
diff --git a/Python/compile.c b/Python/compile.c index 03a1e32..2f5fd69 100644 --- a/Python/compile.c +++ b/Python/compile.c @@ -1,4 +1,3 @@ -#define DEBUG /* Compile an expression node to intermediate code */ #include <stdio.h> @@ -174,7 +173,8 @@ com_addint(c, x) struct compiling *c; int x; { - com_addbyte(c, x); + com_addbyte(c, x & 0xff); + com_addbyte(c, x >> 8); /* XXX x should be positive */ } static void @@ -215,22 +215,10 @@ com_backpatch(c, anchor) int prev; for (;;) { /* Make the JUMP instruction at anchor point to target */ - prev = code[anchor]; - dist = target - (anchor+1); - if (dist > 255 && lastanchor && - code[lastanchor-1] == code[anchor-1]) { - /* Attempt to jump to a similar jump closer by */ -/* XXX Show that it works */ -fprintf(stderr, "com_backpatch: huge jump rescued (?)\n"); - target = lastanchor-1; - dist = target - (anchor+1); - lastanchor = 0; - } - if (dist > 255) { - err_setstr(SystemError, "relative jump > 255 bytes"); - c->c_errors++; - } - code[anchor] = dist; + prev = code[anchor] + (code[anchor+1] << 8); + dist = target - (anchor+2); + code[anchor] = dist & 0xff; + code[anchor+1] = dist >> 8; if (!prev) break; lastanchor = anchor; @@ -246,11 +234,16 @@ com_add(c, list, v) object *list; object *v; { - /* XXX Should look through list for object with same value */ - int i = getlistsize(list); + int n = getlistsize(list); + int i; + for (i = n; --i >= 0; ) { + object *w = getlistitem(list, i); + if (cmpobject(v, w) == 0) + return i; + } if (addlistitem(list, v) != 0) c->c_errors++; - return i; + return n; } static int @@ -926,6 +919,7 @@ com_assign(c, n, assigning) /* Loop to avoid trivial recursion */ for (;;) { switch (TYPE(n)) { + case exprlist: case testlist: if (NCH(n) > 1) { @@ -934,6 +928,7 @@ com_assign(c, n, assigning) } n = CHILD(n, 0); break; + case test: case and_test: case not_test: @@ -945,6 +940,7 @@ com_assign(c, n, assigning) } n = CHILD(n, 0); break; + case comparison: if (NCH(n) > 1) { err_setstr(TypeError, @@ -954,6 +950,7 @@ com_assign(c, n, assigning) } n = CHILD(n, 0); break; + case expr: if (NCH(n) > 1) { err_setstr(TypeError, @@ -963,6 +960,7 @@ com_assign(c, n, assigning) } n = CHILD(n, 0); break; + case term: if (NCH(n) > 1) { err_setstr(TypeError, @@ -972,6 +970,7 @@ com_assign(c, n, assigning) } n = CHILD(n, 0); break; + case factor: /* ('+'|'-') factor | atom trailer* */ if (TYPE(CHILD(n, 0)) != atom) { /* '+' | '-' */ err_setstr(TypeError, @@ -991,6 +990,7 @@ com_assign(c, n, assigning) } n = CHILD(n, 0); break; + case atom: switch (TYPE(CHILD(n, 0))) { case LPAR: @@ -1023,11 +1023,13 @@ com_assign(c, n, assigning) return; } break; + default: fprintf(stderr, "node type %d\n", TYPE(n)); err_setstr(SystemError, "com_assign: bad node"); c->c_errors++; return; + } } } @@ -1065,6 +1067,7 @@ com_print_stmt(c, n) } if (TYPE(CHILD(n, NCH(n)-2)) != COMMA) com_addbyte(c, PRINT_NEWLINE); + /* XXX Alternatively, LOAD_CONST '\n' and then PRINT_ITEM */ } static void @@ -1112,6 +1115,7 @@ com_import_stmt(c, n) com_addbyte(c, POP_TOP); } else { + /* 'import' ... */ for (i = 1; i < NCH(n); i += 2) { com_addopname(c, IMPORT_NAME, CHILD(n, i)); com_addopname(c, STORE_NAME, CHILD(n, i)); @@ -1197,6 +1201,91 @@ com_for_stmt(c, n) com_backpatch(c, break_anchor); } +/* Although 'execpt' and 'finally' clauses can be combined + syntactically, they are compiled separately. In fact, + try: S + except E1: S1 + except E2: S2 + ... + finally: Sf + is equivalent to + try: + try: S + except E1: S1 + except E2: S2 + ... + finally: Sf + meaning that the 'finally' clause is entered even if things + go wrong again in an exception handler. Note that this is + not the case for exception handlers: at most one is entered. + + Code generated for "try: S finally: Sf" is as follows: + + SETUP_FINALLY L + <code for S> + POP_BLOCK + LOAD_CONST <nil> + LOAD_CONST <nil> + L: <code for Sf> + END_FINALLY + + The special instructions use the block stack. Each block + stack entry contains the instruction that created it (here + SETUP_FINALLY), the level of the value stack at the time the + block stack entry was created, and a label (here L). + + SETUP_FINALLY: + Pushes the current value stack level and the label + onto the block stack. + POP_BLOCK: + Pops en entry from the block stack, and pops the value + stack until its level is the same as indicated on the + block stack. (The label is ignored.) + END_FINALLY: + Pops two entries from the *value* stack and re-raises + the exception they specify. If the top entry is nil, + no exception is raised. If it is a number, a pseudo + exception is raised ('return' or 'break'). Otherwise, + a real exception (specified by a string) is raised. + The second entry popped from the is the value that goes + with the exception (or the return value). + + The block stack is unwound when an exception is raised: + when a SETUP_FINALLY entry is found, the exception is pushed + onto the value stack (and the exception condition is cleared), + and the interpreter jumps to the label gotten from the block + stack. + + Code generated for "try: S except E1, V1: S1 except E2, V2: S2 ...": + + Value stack Label Instruction Argument + [] SETUP_EXCEPT L1 + [] <code for S> + [] POP_BLOCK + [] JUMP_FORWARD L0 + + [val, exc] L1: DUP + [val, exc, exc] <evaluate E1> + [val, exc, exc, E1] COMPARE_OP EXC_MATCH + [val, exc, 1-or-0] JUMP_IF_FALSE L2 + [val, exc, 1] POP + [val, exc] POP + [val] <assign to V1> + [] <code for S1> + JUMP_FORWARD L0 + + [val, exc, 0] L2: POP + [val, exc] DUP + .............................etc....................... + + [val, exc, 0] Ln+1: POP + [val, exc] END_FINALLY # re-raise exception + + [] L0: <next statement> + + Of course, parts are not generated if Vi or Ei is not present. +*/ + static void com_try_stmt(c, n) struct compiling *c; @@ -1206,6 +1295,7 @@ com_try_stmt(c, n) int except_anchor = 0; REQ(n, try_stmt); /* 'try' ':' suite (except_clause ':' suite)* ['finally' ':' suite] */ + if (NCH(n) > 3 && TYPE(CHILD(n, NCH(n)-3)) != except_clause) { /* Have a 'finally' clause */ com_addfwref(c, SETUP_FINALLY, &finally_anchor); @@ -1226,12 +1316,18 @@ com_try_stmt(c, n) i < NCH(n) && TYPE(ch = CHILD(n, i)) == except_clause; i += 3) { /* except_clause: 'except' [expr [',' expr]] */ - int next_anchor = 0; + if (except_anchor == 0) { + err_setstr(TypeError, + "default 'except:' must be last"); + c->c_errors++; + break; + } + except_anchor = 0; if (NCH(ch) > 1) { com_addbyte(c, DUP_TOP); com_node(c, CHILD(ch, 1)); com_addoparg(c, COMPARE_OP, EXC_MATCH); - com_addfwref(c, JUMP_IF_FALSE, &next_anchor); + com_addfwref(c, JUMP_IF_FALSE, &except_anchor); com_addbyte(c, POP_TOP); } com_addbyte(c, POP_TOP); @@ -1241,8 +1337,10 @@ com_try_stmt(c, n) com_addbyte(c, POP_TOP); com_node(c, CHILD(n, i+2)); com_addfwref(c, JUMP_FORWARD, &end_anchor); - if (next_anchor) - com_backpatch(c, next_anchor); + if (except_anchor) { + com_backpatch(c, except_anchor); + com_addbyte(c, POP_TOP); + } } com_addbyte(c, END_FINALLY); com_backpatch(c, end_anchor); @@ -1297,18 +1395,49 @@ com_funcdef(c, n) } static void +com_bases(c, n) + struct compiling *c; + node *n; +{ + int i, nbases; + REQ(n, baselist); + /* + baselist: atom arguments (',' atom arguments)* + arguments: '(' [testlist] ')' + */ + for (i = 0; i < NCH(n); i += 3) + com_node(c, CHILD(n, i)); + com_addoparg(c, BUILD_TUPLE, (NCH(n)+1) / 3); +} + +static void com_classdef(c, n) struct compiling *c; node *n; { + object *v; REQ(n, classdef); /* classdef: 'class' NAME parameters ['=' baselist] ':' suite baselist: atom arguments (',' atom arguments)* arguments: '(' [testlist] ')' */ - err_setstr(SystemError, "can't compile 'class' yet"); - c->c_errors++; + if (NCH(n) == 7) + com_bases(c, CHILD(n, 4)); + else + com_addoparg(c, LOAD_CONST, com_addconst(c, None)); + v = (object *)compile(n); + if (v == NULL) + c->c_errors++; + else { + int i = com_addconst(c, v); + com_addoparg(c, LOAD_CONST, i); + com_addbyte(c, BUILD_FUNCTION); + com_addbyte(c, UNARY_CALL); + com_addbyte(c, BUILD_CLASS); + com_addopname(c, STORE_NAME, CHILD(n, 1)); + DECREF(v); + } } static void @@ -1485,6 +1614,22 @@ compile_funcdef(c, n) } static void +compile_classdef(c, n) + struct compiling *c; + node *n; +{ + node *ch; + REQ(n, classdef); + /* + classdef: 'class' NAME parameters ['=' baselist] ':' suite + */ + com_addbyte(c, REFUSE_ARGS); + com_node(c, CHILD(n, NCH(n)-1)); + com_addbyte(c, LOAD_LOCALS); + com_addbyte(c, RETURN_VALUE); +} + +static void compile_node(c, n) struct compiling *c; node *n; @@ -1511,6 +1656,10 @@ compile_node(c, n) compile_funcdef(c, n); break; + case classdef: + compile_classdef(c, n); + break; + default: fprintf(stderr, "node type %d\n", TYPE(n)); err_setstr(SystemError, "compile_node: unexpected node type"); |