|
|
|
@ -146,8 +146,6 @@ static void format_exc_unbound(PyCodeObject *co, int oparg); |
|
|
|
static PyObject * unicode_concatenate(PyObject *, PyObject *, |
|
|
|
PyFrameObject *, unsigned char *); |
|
|
|
static PyObject * special_lookup(PyObject *, _Py_Identifier *); |
|
|
|
static PyObject * apply_coroutine_wrapper(PyObject *); |
|
|
|
|
|
|
|
|
|
|
|
#define NAME_ERROR_MSG \ |
|
|
|
"name '%.200s' is not defined" |
|
|
|
@ -3923,6 +3921,18 @@ _PyEval_EvalCodeWithName(PyObject *_co, PyObject *globals, PyObject *locals, |
|
|
|
|
|
|
|
if (co->co_flags & CO_GENERATOR) { |
|
|
|
PyObject *gen; |
|
|
|
PyObject *coro_wrapper = tstate->coroutine_wrapper; |
|
|
|
int is_coro = co->co_flags & (CO_COROUTINE | CO_ITERABLE_COROUTINE); |
|
|
|
|
|
|
|
if (is_coro && tstate->in_coroutine_wrapper) { |
|
|
|
assert(coro_wrapper != NULL); |
|
|
|
PyErr_Format(PyExc_RuntimeError, |
|
|
|
"coroutine wrapper %.200R attempted " |
|
|
|
"to recursively wrap %.200R", |
|
|
|
coro_wrapper, |
|
|
|
co); |
|
|
|
goto fail; |
|
|
|
} |
|
|
|
|
|
|
|
/* Don't need to keep the reference to f_back, it will be set |
|
|
|
* when the generator is resumed. */ |
|
|
|
@ -3936,8 +3946,13 @@ _PyEval_EvalCodeWithName(PyObject *_co, PyObject *globals, PyObject *locals, |
|
|
|
if (gen == NULL) |
|
|
|
return NULL; |
|
|
|
|
|
|
|
if (co->co_flags & (CO_COROUTINE | CO_ITERABLE_COROUTINE)) |
|
|
|
return apply_coroutine_wrapper(gen); |
|
|
|
if (is_coro && coro_wrapper != NULL) { |
|
|
|
PyObject *wrapped; |
|
|
|
tstate->in_coroutine_wrapper = 1; |
|
|
|
wrapped = PyObject_CallFunction(coro_wrapper, "N", gen); |
|
|
|
tstate->in_coroutine_wrapper = 0; |
|
|
|
return wrapped; |
|
|
|
} |
|
|
|
|
|
|
|
return gen; |
|
|
|
} |
|
|
|
@ -5232,33 +5247,6 @@ unicode_concatenate(PyObject *v, PyObject *w, |
|
|
|
return res; |
|
|
|
} |
|
|
|
|
|
|
|
static PyObject * |
|
|
|
apply_coroutine_wrapper(PyObject *gen) |
|
|
|
{ |
|
|
|
PyObject *wrapped; |
|
|
|
PyThreadState *tstate = PyThreadState_GET(); |
|
|
|
PyObject *wrapper = tstate->coroutine_wrapper; |
|
|
|
|
|
|
|
if (tstate->in_coroutine_wrapper) { |
|
|
|
assert(wrapper != NULL); |
|
|
|
PyErr_Format(PyExc_RuntimeError, |
|
|
|
"coroutine wrapper %.200R attempted " |
|
|
|
"to recursively wrap %.200R", |
|
|
|
wrapper, |
|
|
|
gen); |
|
|
|
return NULL; |
|
|
|
} |
|
|
|
|
|
|
|
if (wrapper == NULL) { |
|
|
|
return gen; |
|
|
|
} |
|
|
|
|
|
|
|
tstate->in_coroutine_wrapper = 1; |
|
|
|
wrapped = PyObject_CallFunction(wrapper, "N", gen); |
|
|
|
tstate->in_coroutine_wrapper = 0; |
|
|
|
return wrapped; |
|
|
|
} |
|
|
|
|
|
|
|
#ifdef DYNAMIC_EXECUTION_PROFILE |
|
|
|
|
|
|
|
static PyObject * |
|
|
|
|