diff options
author | Guido van Rossum <guido@python.org> | 2022-11-23 00:04:57 (GMT) |
---|---|---|
committer | GitHub <noreply@github.com> | 2022-11-23 00:04:57 (GMT) |
commit | 8f18ac04d32515eab841172c956a8cb14bcee9c3 (patch) | |
tree | b2c54230bb6a336b40e18c42c0e3cc5f1da41e24 /Python | |
parent | f1a4a6a58736196f766d51f048d19a2b0a0a155a (diff) | |
download | cpython-8f18ac04d32515eab841172c956a8cb14bcee9c3.zip cpython-8f18ac04d32515eab841172c956a8cb14bcee9c3.tar.gz cpython-8f18ac04d32515eab841172c956a8cb14bcee9c3.tar.bz2 |
GH-98831: Add `macro` and `op` and their implementation to DSL (#99495)
Newly supported interpreter definition syntax:
- `op(NAME, (input_stack_effects -- output_stack_effects)) { ... }`
- `macro(NAME) = OP1 + OP2;`
Also some other random improvements:
- Convert `WITH_EXCEPT_START` to use stack effects
- Fix lexer to balk at unrecognized characters, e.g. `@`
- Fix moved output names; support object pointers in cache
- Introduce `error()` method to print errors
- Introduce read_uint16(p) as equivalent to `*p`
Co-authored-by: Brandt Bucher <brandtbucher@gmail.com>
Diffstat (limited to 'Python')
-rw-r--r-- | Python/bytecodes.c | 32 | ||||
-rw-r--r-- | Python/generated_cases.c.h | 103 |
2 files changed, 73 insertions, 62 deletions
diff --git a/Python/bytecodes.c b/Python/bytecodes.c index 78f7d4a..a1f910d 100644 --- a/Python/bytecodes.c +++ b/Python/bytecodes.c @@ -70,6 +70,8 @@ do { \ #define DISPATCH_SAME_OPARG() ((void)0) #define inst(name, ...) case name: +#define op(name, ...) /* NAME is ignored */ +#define macro(name) static int MACRO_##name #define super(name) static int SUPER_##name #define family(name, ...) static int family_##name @@ -80,6 +82,7 @@ do { \ static PyObject *value, *value1, *value2, *left, *right, *res, *sum, *prod, *sub; static PyObject *container, *start, *stop, *v, *lhs, *rhs; static PyObject *list, *tuple, *dict; +static PyObject *exit_func, *lasti, *val; static PyObject * dummy_func( @@ -156,10 +159,7 @@ dummy_func( res = NULL; } - inst(END_FOR, (value1, value2 --)) { - Py_DECREF(value1); - Py_DECREF(value2); - } + macro(END_FOR) = POP_TOP + POP_TOP; inst(UNARY_POSITIVE, (value -- res)) { res = PyNumber_Positive(value); @@ -2725,33 +2725,27 @@ dummy_func( PUSH(res); } - // stack effect: ( -- __0) - inst(WITH_EXCEPT_START) { + inst(WITH_EXCEPT_START, (exit_func, lasti, unused, val -- exit_func, lasti, unused, val, res)) { /* At the top of the stack are 4 values: - - TOP = exc_info() - - SECOND = previous exception - - THIRD: lasti of exception in exc_info() - - FOURTH: the context.__exit__ bound method + - val: TOP = exc_info() + - unused: SECOND = previous exception + - lasti: THIRD = lasti of exception in exc_info() + - exit_func: FOURTH = the context.__exit__ bound method We call FOURTH(type(TOP), TOP, GetTraceback(TOP)). Then we push the __exit__ return value. */ - PyObject *exit_func; - PyObject *exc, *val, *tb, *res; + PyObject *exc, *tb; - val = TOP(); assert(val && PyExceptionInstance_Check(val)); exc = PyExceptionInstance_Class(val); tb = PyException_GetTraceback(val); Py_XDECREF(tb); - assert(PyLong_Check(PEEK(3))); - exit_func = PEEK(4); + assert(PyLong_Check(lasti)); + (void)lasti; // Shut up compiler warning if asserts are off PyObject *stack[4] = {NULL, exc, val, tb}; res = PyObject_Vectorcall(exit_func, stack + 1, 3 | PY_VECTORCALL_ARGUMENTS_OFFSET, NULL); - if (res == NULL) - goto error; - - PUSH(res); + ERROR_IF(res == NULL, error); } // stack effect: ( -- __0) diff --git a/Python/generated_cases.c.h b/Python/generated_cases.c.h index 2c6333f..ae8fdd5 100644 --- a/Python/generated_cases.c.h +++ b/Python/generated_cases.c.h @@ -78,15 +78,6 @@ DISPATCH(); } - TARGET(END_FOR) { - PyObject *value2 = PEEK(1); - PyObject *value1 = PEEK(2); - Py_DECREF(value1); - Py_DECREF(value2); - STACK_SHRINK(2); - DISPATCH(); - } - TARGET(UNARY_POSITIVE) { PyObject *value = PEEK(1); PyObject *res; @@ -446,7 +437,7 @@ TARGET(BINARY_SUBSCR_GETITEM) { uint32_t type_version = read_u32(next_instr + 1); - uint16_t func_version = *(next_instr + 3); + uint16_t func_version = read_u16(next_instr + 3); PyObject *sub = PEEK(1); PyObject *container = PEEK(2); PyTypeObject *tp = Py_TYPE(container); @@ -2754,31 +2745,32 @@ } TARGET(WITH_EXCEPT_START) { + PyObject *val = PEEK(1); + PyObject *lasti = PEEK(3); + PyObject *exit_func = PEEK(4); + PyObject *res; /* At the top of the stack are 4 values: - - TOP = exc_info() - - SECOND = previous exception - - THIRD: lasti of exception in exc_info() - - FOURTH: the context.__exit__ bound method + - val: TOP = exc_info() + - unused: SECOND = previous exception + - lasti: THIRD = lasti of exception in exc_info() + - exit_func: FOURTH = the context.__exit__ bound method We call FOURTH(type(TOP), TOP, GetTraceback(TOP)). Then we push the __exit__ return value. */ - PyObject *exit_func; - PyObject *exc, *val, *tb, *res; + PyObject *exc, *tb; - val = TOP(); assert(val && PyExceptionInstance_Check(val)); exc = PyExceptionInstance_Class(val); tb = PyException_GetTraceback(val); Py_XDECREF(tb); - assert(PyLong_Check(PEEK(3))); - exit_func = PEEK(4); + assert(PyLong_Check(lasti)); + (void)lasti; // Shut up compiler warning if asserts are off PyObject *stack[4] = {NULL, exc, val, tb}; res = PyObject_Vectorcall(exit_func, stack + 1, 3 | PY_VECTORCALL_ARGUMENTS_OFFSET, NULL); - if (res == NULL) - goto error; - - PUSH(res); + if (res == NULL) goto error; + STACK_GROW(1); + POKE(1, res); DISPATCH(); } @@ -3711,13 +3703,14 @@ } TARGET(LOAD_FAST__LOAD_FAST) { + PyObject *_tmp_1; + PyObject *_tmp_2; { PyObject *value; value = GETLOCAL(oparg); assert(value != NULL); Py_INCREF(value); - STACK_GROW(1); - POKE(1, value); + _tmp_1 = value; } NEXTOPARG(); next_instr++; @@ -3726,20 +3719,23 @@ value = GETLOCAL(oparg); assert(value != NULL); Py_INCREF(value); - STACK_GROW(1); - POKE(1, value); + _tmp_2 = value; } + STACK_GROW(2); + POKE(1, _tmp_2); + POKE(2, _tmp_1); DISPATCH(); } TARGET(LOAD_FAST__LOAD_CONST) { + PyObject *_tmp_1; + PyObject *_tmp_2; { PyObject *value; value = GETLOCAL(oparg); assert(value != NULL); Py_INCREF(value); - STACK_GROW(1); - POKE(1, value); + _tmp_1 = value; } NEXTOPARG(); next_instr++; @@ -3747,17 +3743,19 @@ PyObject *value; value = GETITEM(consts, oparg); Py_INCREF(value); - STACK_GROW(1); - POKE(1, value); + _tmp_2 = value; } + STACK_GROW(2); + POKE(1, _tmp_2); + POKE(2, _tmp_1); DISPATCH(); } TARGET(STORE_FAST__LOAD_FAST) { + PyObject *_tmp_1 = PEEK(1); { - PyObject *value = PEEK(1); + PyObject *value = _tmp_1; SETLOCAL(oparg, value); - STACK_SHRINK(1); } NEXTOPARG(); next_instr++; @@ -3766,35 +3764,37 @@ value = GETLOCAL(oparg); assert(value != NULL); Py_INCREF(value); - STACK_GROW(1); - POKE(1, value); + _tmp_1 = value; } + POKE(1, _tmp_1); DISPATCH(); } TARGET(STORE_FAST__STORE_FAST) { + PyObject *_tmp_1 = PEEK(2); + PyObject *_tmp_2 = PEEK(1); { - PyObject *value = PEEK(1); + PyObject *value = _tmp_2; SETLOCAL(oparg, value); - STACK_SHRINK(1); } NEXTOPARG(); next_instr++; { - PyObject *value = PEEK(1); + PyObject *value = _tmp_1; SETLOCAL(oparg, value); - STACK_SHRINK(1); } + STACK_SHRINK(2); DISPATCH(); } TARGET(LOAD_CONST__LOAD_FAST) { + PyObject *_tmp_1; + PyObject *_tmp_2; { PyObject *value; value = GETITEM(consts, oparg); Py_INCREF(value); - STACK_GROW(1); - POKE(1, value); + _tmp_1 = value; } NEXTOPARG(); next_instr++; @@ -3803,8 +3803,25 @@ value = GETLOCAL(oparg); assert(value != NULL); Py_INCREF(value); - STACK_GROW(1); - POKE(1, value); + _tmp_2 = value; } + STACK_GROW(2); + POKE(1, _tmp_2); + POKE(2, _tmp_1); + DISPATCH(); + } + + TARGET(END_FOR) { + PyObject *_tmp_1 = PEEK(2); + PyObject *_tmp_2 = PEEK(1); + { + PyObject *value = _tmp_2; + Py_DECREF(value); + } + { + PyObject *value = _tmp_1; + Py_DECREF(value); + } + STACK_SHRINK(2); DISPATCH(); } |