diff options
author | Mark Shannon <mark@hotpy.org> | 2022-01-06 13:09:25 (GMT) |
---|---|---|
committer | GitHub <noreply@github.com> | 2022-01-06 13:09:25 (GMT) |
commit | e028ae99ecee671c0e8a3eabb829b5b2acfc4441 (patch) | |
tree | 497e68f1caeb92104bf3a464977bb0e024131f69 /Python | |
parent | 3e43fac2503afe219336742b150b3ef6e470686f (diff) | |
download | cpython-e028ae99ecee671c0e8a3eabb829b5b2acfc4441.zip cpython-e028ae99ecee671c0e8a3eabb829b5b2acfc4441.tar.gz cpython-e028ae99ecee671c0e8a3eabb829b5b2acfc4441.tar.bz2 |
bpo-45923: Handle call events in bytecode (GH-30364)
* Add a RESUME instruction to handle "call" events.
Diffstat (limited to 'Python')
-rw-r--r-- | Python/ceval.c | 117 | ||||
-rw-r--r-- | Python/compile.c | 70 | ||||
-rw-r--r-- | Python/opcode_targets.h | 2 |
3 files changed, 117 insertions, 72 deletions
diff --git a/Python/ceval.c b/Python/ceval.c index 86d834c..be26ffd 100644 --- a/Python/ceval.c +++ b/Python/ceval.c @@ -1546,6 +1546,17 @@ eval_frame_handle_pending(PyThreadState *tstate) #define TRACE_FUNCTION_ENTRY() \ if (cframe.use_tracing) { \ + _PyFrame_SetStackPointer(frame, stack_pointer); \ + int err = trace_function_entry(tstate, frame); \ + stack_pointer = _PyFrame_GetStackPointer(frame); \ + if (err) { \ + goto error; \ + } \ + } + +#define TRACE_FUNCTION_THROW_ENTRY() \ + if (cframe.use_tracing) { \ + assert(frame->stacktop >= 0); \ if (trace_function_entry(tstate, frame)) { \ goto exit_unwind; \ } \ @@ -1694,7 +1705,7 @@ _PyEval_EvalFrameDefault(PyThreadState *tstate, InterpreterFrame *frame, int thr tstate->recursion_remaining--; goto exit_unwind; } - TRACE_FUNCTION_ENTRY(); + TRACE_FUNCTION_THROW_ENTRY(); DTRACE_FUNCTION_ENTRY(); goto resume_with_error; } @@ -1734,17 +1745,6 @@ start_frame: goto exit_unwind; } - assert(tstate->cframe == &cframe); - assert(frame == cframe.current_frame); - - TRACE_FUNCTION_ENTRY(); - DTRACE_FUNCTION_ENTRY(); - - if (_Py_IncrementCountAndMaybeQuicken(frame->f_code) < 0) { - goto exit_unwind; - } - frame->f_state = FRAME_EXECUTING; - resume_frame: SET_LOCALS_FROM_FRAME(); @@ -1825,6 +1825,24 @@ check_eval_breaker: DISPATCH(); } + TARGET(RESUME) { + assert(tstate->cframe == &cframe); + assert(frame == cframe.current_frame); + + int err = _Py_IncrementCountAndMaybeQuicken(frame->f_code); + if (err) { + if (err < 0) { + goto error; + } + /* Update first_instr and next_instr to point to newly quickened code */ + int nexti = INSTR_OFFSET(); + first_instr = frame->f_code->co_firstinstr; + next_instr = first_instr + nexti; + } + frame->f_state = FRAME_EXECUTING; + DISPATCH(); + } + TARGET(LOAD_CLOSURE) { /* We keep LOAD_CLOSURE so that the bytecode stays more readable. */ PyObject *value = GETLOCAL(oparg); @@ -3134,7 +3152,7 @@ check_eval_breaker: PyObject *initial = GETLOCAL(oparg); PyObject *cell = PyCell_New(initial); if (cell == NULL) { - goto error; + goto resume_with_error; } SETLOCAL(oparg, cell); DISPATCH(); @@ -5209,33 +5227,40 @@ check_eval_breaker: int instr_prev = skip_backwards_over_extended_args(frame->f_code, frame->f_lasti); frame->f_lasti = INSTR_OFFSET(); TRACING_NEXTOPARG(); - if (PyDTrace_LINE_ENABLED()) { - maybe_dtrace_line(frame, &tstate->trace_info, instr_prev); + if (opcode == RESUME) { + /* Call tracing */ + TRACE_FUNCTION_ENTRY(); + DTRACE_FUNCTION_ENTRY(); } - /* line-by-line tracing support */ - - if (cframe.use_tracing && - tstate->c_tracefunc != NULL && !tstate->tracing) { - int err; - /* see maybe_call_line_trace() - for expository comments */ - _PyFrame_SetStackPointer(frame, stack_pointer); - - err = maybe_call_line_trace(tstate->c_tracefunc, - tstate->c_traceobj, - tstate, frame, instr_prev); - if (err) { - /* trace function raised an exception */ - next_instr++; - goto error; + else { + /* line-by-line tracing support */ + if (PyDTrace_LINE_ENABLED()) { + maybe_dtrace_line(frame, &tstate->trace_info, instr_prev); } - /* Reload possibly changed frame fields */ - JUMPTO(frame->f_lasti); - stack_pointer = _PyFrame_GetStackPointer(frame); - frame->stacktop = -1; - TRACING_NEXTOPARG(); + if (cframe.use_tracing && + tstate->c_tracefunc != NULL && !tstate->tracing) { + int err; + /* see maybe_call_line_trace() + for expository comments */ + _PyFrame_SetStackPointer(frame, stack_pointer); + + err = maybe_call_line_trace(tstate->c_tracefunc, + tstate->c_traceobj, + tstate, frame, instr_prev); + if (err) { + /* trace function raised an exception */ + next_instr++; + goto error; + } + /* Reload possibly changed frame fields */ + JUMPTO(frame->f_lasti); + + stack_pointer = _PyFrame_GetStackPointer(frame); + frame->stacktop = -1; + } } + TRACING_NEXTOPARG(); PRE_DISPATCH_GOTO(); DISPATCH_GOTO(); } @@ -6046,6 +6071,7 @@ _PyEval_Vector(PyThreadState *tstate, PyFunctionObject *func, return NULL; } PyObject *retval = _PyEval_EvalFrame(tstate, frame, 0); + assert(frame->stacktop >= 0); assert(_PyFrame_GetStackPointer(frame) == _PyFrame_Stackbase(frame)); _PyEvalFrameClearAndPop(tstate, frame); return retval; @@ -6492,13 +6518,9 @@ call_trace(Py_tracefunc func, PyObject *obj, if (f == NULL) { return -1; } - if (frame->f_lasti < 0) { - f->f_lineno = frame->f_code->co_firstlineno; - } - else { - initialize_trace_info(&tstate->trace_info, frame); - f->f_lineno = _PyCode_CheckLineNumber(frame->f_lasti*sizeof(_Py_CODEUNIT), &tstate->trace_info.bounds); - } + assert (frame->f_lasti >= 0); + initialize_trace_info(&tstate->trace_info, frame); + f->f_lineno = _PyCode_CheckLineNumber(frame->f_lasti*sizeof(_Py_CODEUNIT), &tstate->trace_info.bounds); result = func(obj, f, what, arg); f->f_lineno = 0; _PyThreadState_ResumeTracing(tstate); @@ -6534,7 +6556,14 @@ maybe_call_line_trace(Py_tracefunc func, PyObject *obj, then call the trace function if we're tracing source lines. */ initialize_trace_info(&tstate->trace_info, frame); - int lastline = _PyCode_CheckLineNumber(instr_prev*sizeof(_Py_CODEUNIT), &tstate->trace_info.bounds); + _Py_CODEUNIT prev = ((_Py_CODEUNIT *)PyBytes_AS_STRING(frame->f_code->co_code))[instr_prev]; + int lastline; + if (_Py_OPCODE(prev) == RESUME && _Py_OPARG(prev) == 0) { + lastline = -1; + } + else { + lastline = _PyCode_CheckLineNumber(instr_prev*sizeof(_Py_CODEUNIT), &tstate->trace_info.bounds); + } int line = _PyCode_CheckLineNumber(frame->f_lasti*sizeof(_Py_CODEUNIT), &tstate->trace_info.bounds); PyFrameObject *f = _PyFrame_GetFrameObject(frame); if (f == NULL) { diff --git a/Python/compile.c b/Python/compile.c index 625a07b..62f37ca 100644 --- a/Python/compile.c +++ b/Python/compile.c @@ -689,9 +689,9 @@ compiler_enter_scope(struct compiler *c, identifier name, u->u_blocks = NULL; u->u_nfblocks = 0; u->u_firstlineno = lineno; - u->u_lineno = 0; + u->u_lineno = lineno; u->u_col_offset = 0; - u->u_end_lineno = 0; + u->u_end_lineno = lineno; u->u_end_col_offset = 0; u->u_consts = PyDict_New(); if (!u->u_consts) { @@ -995,6 +995,7 @@ stack_effect(int opcode, int oparg, int jump) switch (opcode) { case NOP: case EXTENDED_ARG: + case RESUME: return 0; /* Stack manipulation */ @@ -1664,8 +1665,8 @@ compiler_addop_j_noline(struct compiler *c, int opcode, basicblock *b) the ASDL name to synthesize the name of the C type and the visit function. */ -#define ADD_YIELD_FROM(C) \ - RETURN_IF_FALSE(compiler_add_yield_from((C))) +#define ADD_YIELD_FROM(C, await) \ + RETURN_IF_FALSE(compiler_add_yield_from((C), (await))) #define POP_EXCEPT_AND_RERAISE(C) \ RETURN_IF_FALSE(compiler_pop_except_and_reraise((C))) @@ -1823,18 +1824,19 @@ compiler_call_exit_with_nones(struct compiler *c) { } static int -compiler_add_yield_from(struct compiler *c) +compiler_add_yield_from(struct compiler *c, int await) { - basicblock *start, *jump, *exit; + basicblock *start, *resume, *exit; start = compiler_new_block(c); - jump = compiler_new_block(c); + resume = compiler_new_block(c); exit = compiler_new_block(c); - if (start == NULL || jump == NULL || exit == NULL) { + if (start == NULL || resume == NULL || exit == NULL) { return 0; } compiler_use_next_block(c, start); ADDOP_JUMP(c, SEND, exit); - compiler_use_next_block(c, jump); + compiler_use_next_block(c, resume); + ADDOP_I(c, RESUME, await ? 3 : 2); ADDOP_JUMP(c, JUMP_ABSOLUTE, start); compiler_use_next_block(c, exit); return 1; @@ -1928,7 +1930,7 @@ compiler_unwind_fblock(struct compiler *c, struct fblockinfo *info, if (info->fb_type == ASYNC_WITH) { ADDOP(c, GET_AWAITABLE); ADDOP_LOAD_CONST(c, Py_None); - ADD_YIELD_FROM(c); + ADD_YIELD_FROM(c, 1); } ADDOP(c, POP_TOP); /* The exit block should appear to execute after the @@ -2047,9 +2049,11 @@ compiler_mod(struct compiler *c, mod_ty mod) if (module == NULL) { return 0; } - /* Use 0 for firstlineno initially, will fixup in assemble(). */ if (!compiler_enter_scope(c, module, COMPILER_SCOPE_MODULE, mod, 1)) return NULL; + c->u->u_lineno = -1; + ADDOP_I(c, RESUME, 0); + c->u->u_lineno = 1; switch (mod->kind) { case Module_kind: if (!compiler_body(c, mod->v.Module.body)) { @@ -2504,6 +2508,7 @@ compiler_function(struct compiler *c, stmt_ty s, int is_async) if (!compiler_enter_scope(c, name, scope_type, (void *)s, firstlineno)) { return 0; } + ADDOP_I(c, RESUME, 0); /* if not -OO mode, add docstring */ if (c->c_optimize < 2) { @@ -2573,8 +2578,10 @@ compiler_class(struct compiler *c, stmt_ty s) /* 1. compile the class body into a code object */ if (!compiler_enter_scope(c, s->v.ClassDef.name, - COMPILER_SCOPE_CLASS, (void *)s, firstlineno)) + COMPILER_SCOPE_CLASS, (void *)s, firstlineno)) { return 0; + } + ADDOP_I(c, RESUME, 0); /* this block represents what we do in the new scope */ { /* use the class name for name mangling */ @@ -2907,11 +2914,13 @@ compiler_lambda(struct compiler *c, expr_ty e) if (funcflags == -1) { return 0; } + ADDOP_I(c, RESUME, 0); if (!compiler_enter_scope(c, name, COMPILER_SCOPE_LAMBDA, (void *)e, e->lineno)) return 0; + ADDOP_I(c, RESUME, 0); /* Make None the first constant, so the lambda can't have a docstring. */ if (compiler_add_const(c, Py_None) < 0) @@ -3041,7 +3050,7 @@ compiler_async_for(struct compiler *c, stmt_ty s) ADDOP_JUMP(c, SETUP_FINALLY, except); ADDOP(c, GET_ANEXT); ADDOP_LOAD_CONST(c, Py_None); - ADD_YIELD_FROM(c); + ADD_YIELD_FROM(c, 1); ADDOP(c, POP_BLOCK); /* for SETUP_FINALLY */ /* Success block for __anext__ */ @@ -5135,6 +5144,7 @@ compiler_sync_comprehension_generator(struct compiler *c, case COMP_GENEXP: VISIT(c, expr, elt); ADDOP(c, YIELD_VALUE); + ADDOP_I(c, RESUME, 1); ADDOP(c, POP_TOP); break; case COMP_LISTCOMP: @@ -5207,7 +5217,7 @@ compiler_async_comprehension_generator(struct compiler *c, ADDOP_JUMP(c, SETUP_FINALLY, except); ADDOP(c, GET_ANEXT); ADDOP_LOAD_CONST(c, Py_None); - ADD_YIELD_FROM(c); + ADD_YIELD_FROM(c, 1); ADDOP(c, POP_BLOCK); VISIT(c, expr, gen->target); @@ -5233,6 +5243,7 @@ compiler_async_comprehension_generator(struct compiler *c, case COMP_GENEXP: VISIT(c, expr, elt); ADDOP(c, YIELD_VALUE); + ADDOP_I(c, RESUME, 1); ADDOP(c, POP_TOP); break; case COMP_LISTCOMP: @@ -5285,6 +5296,7 @@ compiler_comprehension(struct compiler *c, expr_ty e, int type, { goto error; } + ADDOP_I(c, RESUME, 0); SET_LOC(c, e); is_async_generator = c->u->u_ste->ste_coroutine; @@ -5357,7 +5369,7 @@ compiler_comprehension(struct compiler *c, expr_ty e, int type, if (is_async_generator && type != COMP_GENEXP) { ADDOP(c, GET_AWAITABLE); ADDOP_LOAD_CONST(c, Py_None); - ADD_YIELD_FROM(c); + ADD_YIELD_FROM(c, 1); } return 1; @@ -5506,7 +5518,7 @@ compiler_async_with(struct compiler *c, stmt_ty s, int pos) ADDOP(c, BEFORE_ASYNC_WITH); ADDOP(c, GET_AWAITABLE); ADDOP_LOAD_CONST(c, Py_None); - ADD_YIELD_FROM(c); + ADD_YIELD_FROM(c, 1); ADDOP_JUMP(c, SETUP_WITH, final); @@ -5543,7 +5555,7 @@ compiler_async_with(struct compiler *c, stmt_ty s, int pos) return 0; ADDOP(c, GET_AWAITABLE); ADDOP_LOAD_CONST(c, Py_None); - ADD_YIELD_FROM(c); + ADD_YIELD_FROM(c, 1); ADDOP(c, POP_TOP); @@ -5557,7 +5569,7 @@ compiler_async_with(struct compiler *c, stmt_ty s, int pos) ADDOP(c, WITH_EXCEPT_START); ADDOP(c, GET_AWAITABLE); ADDOP_LOAD_CONST(c, Py_None); - ADD_YIELD_FROM(c); + ADD_YIELD_FROM(c, 1); compiler_with_except_finish(c, cleanup); compiler_use_next_block(c, exit); @@ -5703,6 +5715,7 @@ compiler_visit_expr1(struct compiler *c, expr_ty e) ADDOP_LOAD_CONST(c, Py_None); } ADDOP(c, YIELD_VALUE); + ADDOP_I(c, RESUME, 1); break; case YieldFrom_kind: if (c->u->u_ste->ste_type != FunctionBlock) @@ -5714,7 +5727,7 @@ compiler_visit_expr1(struct compiler *c, expr_ty e) VISIT(c, expr, e->v.YieldFrom.value); ADDOP(c, GET_YIELD_FROM_ITER); ADDOP_LOAD_CONST(c, Py_None); - ADD_YIELD_FROM(c); + ADD_YIELD_FROM(c, 0); break; case Await_kind: if (!IS_TOP_LEVEL_AWAIT(c)){ @@ -5731,7 +5744,7 @@ compiler_visit_expr1(struct compiler *c, expr_ty e) VISIT(c, expr, e->v.Await.value); ADDOP(c, GET_AWAITABLE); ADDOP_LOAD_CONST(c, Py_None); - ADD_YIELD_FROM(c); + ADD_YIELD_FROM(c, 1); break; case Compare_kind: return compiler_compare(c, e); @@ -7987,6 +8000,7 @@ insert_prefix_instructions(struct compiler *c, basicblock *entryblock, if (flags < 0) { return -1; } + assert(c->u->u_firstlineno > 0); /* Set up cells for any variable that escapes, to be put in a closure. */ const int ncellvars = (int)PyDict_GET_SIZE(c->u->u_cellvars); @@ -8191,17 +8205,19 @@ assemble(struct compiler *c, int addNone) goto error; } - // This must be called before fix_cell_offsets(). - if (insert_prefix_instructions(c, entryblock, cellfixedoffsets, nfreevars)) { - goto error; - } - /* Set firstlineno if it wasn't explicitly set. */ if (!c->u->u_firstlineno) { - if (entryblock->b_instr && entryblock->b_instr->i_lineno) + if (entryblock->b_instr && entryblock->b_instr->i_lineno) { c->u->u_firstlineno = entryblock->b_instr->i_lineno; - else + } + else { c->u->u_firstlineno = 1; + } + } + + // This must be called before fix_cell_offsets(). + if (insert_prefix_instructions(c, entryblock, cellfixedoffsets, nfreevars)) { + goto error; } if (!assemble_init(&a, nblocks, c->u->u_firstlineno)) diff --git a/Python/opcode_targets.h b/Python/opcode_targets.h index 7ba4566..c78425f 100644 --- a/Python/opcode_targets.h +++ b/Python/opcode_targets.h @@ -150,7 +150,7 @@ static void *opcode_targets[256] = { &&TARGET_LOAD_CLASSDEREF, &&TARGET_COPY_FREE_VARS, &&_unknown_opcode, - &&_unknown_opcode, + &&TARGET_RESUME, &&TARGET_MATCH_CLASS, &&_unknown_opcode, &&_unknown_opcode, |