summaryrefslogtreecommitdiffstats
path: root/Python/compile.c
diff options
context:
space:
mode:
Diffstat (limited to 'Python/compile.c')
-rw-r--r--Python/compile.c177
1 files changed, 113 insertions, 64 deletions
diff --git a/Python/compile.c b/Python/compile.c
index 3299ad9..f20036b 100644
--- a/Python/compile.c
+++ b/Python/compile.c
@@ -47,6 +47,10 @@ OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
#include <ctype.h>
#include <errno.h>
+#define OP_DELETE 0
+#define OP_ASSIGN 1
+#define OP_APPLY 2
+
#define OFF(x) offsetof(codeobject, x)
static struct memberlist code_memberlist[] = {
@@ -821,31 +825,6 @@ com_slice(c, n, op)
}
}
-static void
-com_apply_subscript(c, n)
- struct compiling *c;
- node *n;
-{
- REQ(n, subscript);
- if (TYPE(CHILD(n, 0)) == COLON || (NCH(n) > 1 && TYPE(CHILD(n, 1)) == COLON)) {
- /* It's a slice: [expr] ':' [expr] */
- com_slice(c, n, SLICE);
- }
- else {
- /* It's a list of subscripts */
- if (NCH(n) == 1)
- com_node(c, CHILD(n, 0));
- else {
- int i;
- int len = (NCH(n)+1)/2;
- for (i = 0; i < NCH(n); i += 2)
- com_node(c, CHILD(n, i));
- com_addoparg(c, BUILD_TUPLE, len);
- }
- com_addbyte(c, BINARY_SUBSCR);
- }
-}
-
static int
com_argument(c, n, inkeywords)
struct compiling *c;
@@ -924,6 +903,107 @@ com_select_member(c, n)
}
static void
+com_sliceobj(c, n)
+ struct compiling *c;
+ node *n;
+{
+ int i=0;
+ int ns=2; /* number of slice arguments */
+ int first_missing=0;
+ node *ch;
+
+ /* first argument */
+ if (TYPE(CHILD(n,i)) == COLON) {
+ com_addoparg(c, LOAD_CONST, com_addconst(c, None));
+ i++;
+ }
+ else {
+ com_node(c, CHILD(n,i));
+ i++;
+ REQ(CHILD(n,i),COLON);
+ i++;
+ }
+ /* second argument */
+ if (i < NCH(n) && TYPE(CHILD(n,i)) == test) {
+ com_node(c, CHILD(n,i));
+ i++;
+ }
+ else com_addoparg(c, LOAD_CONST, com_addconst(c, None));
+ /* remaining arguments */
+ for (; i < NCH(n); i++) {
+ ns++;
+ ch=CHILD(n,i);
+ REQ(ch, sliceop);
+ if (NCH(ch) == 1) {
+ /* right argument of ':' missing */
+ com_addoparg(c, LOAD_CONST, com_addconst(c, None));
+ }
+ else
+ com_node(c, CHILD(ch,1));
+ }
+ com_addoparg(c, BUILD_SLICE, ns);
+}
+
+static void
+com_subscript(c, n)
+ struct compiling *c;
+ node *n;
+{
+ node *ch;
+ REQ(n, subscript);
+ ch = CHILD(n,0);
+ /* check for rubber index */
+ if (TYPE(ch) == DOT && TYPE(CHILD(n,1)) == DOT)
+ com_addoparg(c, LOAD_CONST, com_addconst(c, Py_Ellipses));
+ else {
+ /* check for slice */
+ if ((TYPE(ch) == COLON || NCH(n) > 1))
+ com_sliceobj(c, n);
+ else {
+ REQ(ch, test);
+ com_node(c, ch);
+ }
+ }
+}
+
+static void
+com_subscriptlist(c, n, assigning)
+ struct compiling *c;
+ node *n;
+ int assigning;
+{
+ int i, op;
+ REQ(n, subscriptlist);
+ /* Check to make backward compatible slice behavior for '[i:j]' */
+ if (NCH(n) == 1) {
+ node *sub = CHILD(n, 0); /* subscript */
+ /* Make it is a simple slice.
+ should have exactly one colon. */
+ if ((TYPE(CHILD(sub, 0)) == COLON
+ || (NCH(sub) > 1 && TYPE(CHILD(sub, 1)) == COLON))
+ && (TYPE(CHILD(sub,NCH(sub)-1)) != sliceop)) {
+ if (assigning == OP_APPLY)
+ op = SLICE;
+ else
+ op = ((assigning == OP_ASSIGN) ? STORE_SLICE : DELETE_SLICE);
+ com_slice(c, sub, op);
+ return;
+ }
+ }
+ /* Else normal subscriptlist. Compile each subscript. */
+ for (i = 0; i < NCH(n); i += 2)
+ com_subscript(c, CHILD(n, i));
+ /* Put multiple subscripts into a tuple */
+ if (NCH(n) > 1)
+ com_addoparg(c, BUILD_TUPLE, (NCH(n)+1) / 2);
+ if (assigning == OP_APPLY)
+ op = BINARY_SUBSCR;
+ else
+ op = ((assigning == OP_ASSIGN) ? STORE_SUBSCR : DELETE_SUBSCR);
+ com_addbyte(c, op);
+}
+
+static void
com_apply_trailer(c, n)
struct compiling *c;
node *n;
@@ -937,7 +1017,7 @@ com_apply_trailer(c, n)
com_select_member(c, CHILD(n, 1));
break;
case LSQB:
- com_apply_subscript(c, CHILD(n, 1));
+ com_subscriptlist(c, CHILD(n, 1), OP_APPLY);
break;
default:
err_setstr(SystemError,
@@ -970,6 +1050,7 @@ com_factor(c, n)
struct compiling *c;
node *n;
{
+ int i;
REQ(n, factor);
if (TYPE(CHILD(n, 0)) == PLUS) {
com_factor(c, CHILD(n, 1));
@@ -1365,33 +1446,6 @@ com_assign_attr(c, n, assigning)
}
static void
-com_assign_slice(c, n, assigning)
- struct compiling *c;
- node *n;
- int assigning;
-{
- com_slice(c, n, assigning ? STORE_SLICE : DELETE_SLICE);
-}
-
-static void
-com_assign_subscript(c, n, assigning)
- struct compiling *c;
- node *n;
- int assigning;
-{
- if (NCH(n) == 1)
- com_node(c, CHILD(n, 0));
- else {
- int i;
- int len = (NCH(n)+1)/2;
- for (i = 0; i < NCH(n); i += 2)
- com_node(c, CHILD(n, i));
- com_addoparg(c, BUILD_TUPLE, len);
- }
- com_addbyte(c, assigning ? STORE_SUBSCR : DELETE_SUBSCR);
-}
-
-static void
com_assign_trailer(c, n, assigning)
struct compiling *c;
node *n;
@@ -1406,13 +1460,8 @@ com_assign_trailer(c, n, assigning)
case DOT: /* '.' NAME */
com_assign_attr(c, CHILD(n, 1), assigning);
break;
- case LSQB: /* '[' subscript ']' */
- n = CHILD(n, 1);
- REQ(n, subscript); /* subscript: expr (',' expr)* | [expr] ':' [expr] */
- if (TYPE(CHILD(n, 0)) == COLON || (NCH(n) > 1 && TYPE(CHILD(n, 1)) == COLON))
- com_assign_slice(c, n, assigning);
- else
- com_assign_subscript(c, n, assigning);
+ case LSQB: /* '[' subscriptlist ']' */
+ com_subscriptlist(c, CHILD(n, 1), assigning);
break;
default:
err_setstr(SystemError, "unknown trailer type");
@@ -1585,7 +1634,7 @@ com_expr_stmt(c, n)
for (i = 0; i < NCH(n)-2; i+=2) {
if (i+2 < NCH(n)-2)
com_addbyte(c, DUP_TOP);
- com_assign(c, CHILD(n, i), 1/*assign*/);
+ com_assign(c, CHILD(n, i), OP_ASSIGN);
}
}
}
@@ -1896,7 +1945,7 @@ com_for_stmt(c, n)
c->c_begin = c->c_nexti;
com_addoparg(c, SET_LINENO, n->n_lineno);
com_addfwref(c, FOR_LOOP, &anchor);
- com_assign(c, CHILD(n, 1), 1/*assigning*/);
+ com_assign(c, CHILD(n, 1), OP_ASSIGN);
c->c_loops++;
com_node(c, CHILD(n, 5));
c->c_loops--;
@@ -2015,7 +2064,7 @@ com_try_except(c, n)
}
com_addbyte(c, POP_TOP);
if (NCH(ch) > 3)
- com_assign(c, CHILD(ch, 3), 1/*assigning*/);
+ com_assign(c, CHILD(ch, 3), OP_ASSIGN);
else
com_addbyte(c, POP_TOP);
com_addbyte(c, POP_TOP);
@@ -2342,7 +2391,7 @@ com_node(c, n)
com_print_stmt(c, n);
break;
case del_stmt: /* 'del' exprlist */
- com_assign(c, CHILD(n, 1), 0/*delete*/);
+ com_assign(c, CHILD(n, 1), OP_DELETE);
break;
case pass_stmt:
break;