summaryrefslogtreecommitdiffstats
path: root/Python/compile.c
diff options
context:
space:
mode:
authorJelle Zijlstra <jelle.zijlstra@gmail.com>2023-05-18 05:22:17 (GMT)
committerGitHub <noreply@github.com>2023-05-18 05:22:17 (GMT)
commit662aede68b0ea222cf3db4715b310e91c51b665f (patch)
tree3b560e61fa420eb18de21cfc68a6c196a156a214 /Python/compile.c
parent152227b569c3a9b87fe0483706f704762ced6d75 (diff)
downloadcpython-662aede68b0ea222cf3db4715b310e91c51b665f.zip
cpython-662aede68b0ea222cf3db4715b310e91c51b665f.tar.gz
cpython-662aede68b0ea222cf3db4715b310e91c51b665f.tar.bz2
gh-104374: Remove access to class scopes for inlined comprehensions (#104528)
Co-authored-by: Carl Meyer <carl@oddbird.net>
Diffstat (limited to 'Python/compile.c')
-rw-r--r--Python/compile.c20
1 files changed, 15 insertions, 5 deletions
diff --git a/Python/compile.c b/Python/compile.c
index 60c845a..07f8d66 100644
--- a/Python/compile.c
+++ b/Python/compile.c
@@ -388,6 +388,8 @@ struct compiler_unit {
instr_sequence u_instr_sequence; /* codegen output */
int u_nfblocks;
+ int u_in_inlined_comp;
+
struct fblockinfo u_fblock[CO_MAXBLOCKS];
_PyCompile_CodeUnitMetadata u_metadata;
@@ -1290,6 +1292,7 @@ compiler_enter_scope(struct compiler *c, identifier name,
}
u->u_nfblocks = 0;
+ u->u_in_inlined_comp = 0;
u->u_metadata.u_firstlineno = lineno;
u->u_metadata.u_consts = PyDict_New();
if (!u->u_metadata.u_consts) {
@@ -4137,7 +4140,7 @@ compiler_nameop(struct compiler *c, location loc,
case OP_DEREF:
switch (ctx) {
case Load:
- if (c->u->u_ste->ste_type == ClassBlock) {
+ if (c->u->u_ste->ste_type == ClassBlock && !c->u->u_in_inlined_comp) {
op = LOAD_FROM_DICT_OR_DEREF;
// First load the locals
if (codegen_addop_noarg(INSTR_SEQUENCE(c), LOAD_LOCALS, loc) < 0) {
@@ -4188,7 +4191,12 @@ compiler_nameop(struct compiler *c, location loc,
break;
case OP_NAME:
switch (ctx) {
- case Load: op = LOAD_NAME; break;
+ case Load:
+ op = (c->u->u_ste->ste_type == ClassBlock
+ && c->u->u_in_inlined_comp)
+ ? LOAD_GLOBAL
+ : LOAD_NAME;
+ break;
case Store: op = STORE_NAME; break;
case Del: op = DELETE_NAME; break;
}
@@ -5415,6 +5423,8 @@ push_inlined_comprehension_state(struct compiler *c, location loc,
PySTEntryObject *entry,
inlined_comprehension_state *state)
{
+ int in_class_block = (c->u->u_ste->ste_type == ClassBlock) && !c->u->u_in_inlined_comp;
+ c->u->u_in_inlined_comp++;
// iterate over names bound in the comprehension and ensure we isolate
// them from the outer scope as needed
PyObject *k, *v;
@@ -5426,7 +5436,7 @@ push_inlined_comprehension_state(struct compiler *c, location loc,
// at all; DEF_LOCAL | DEF_NONLOCAL can occur in the case of an
// assignment expression to a nonlocal in the comprehension, these don't
// need handling here since they shouldn't be isolated
- if (symbol & DEF_LOCAL && !(symbol & DEF_NONLOCAL)) {
+ if ((symbol & DEF_LOCAL && !(symbol & DEF_NONLOCAL)) || in_class_block) {
if (!_PyST_IsFunctionLike(c->u->u_ste)) {
// non-function scope: override this name to use fast locals
PyObject *orig = PyDict_GetItem(c->u->u_metadata.u_fasthidden, k);
@@ -5448,8 +5458,7 @@ push_inlined_comprehension_state(struct compiler *c, location loc,
long scope = (symbol >> SCOPE_OFFSET) & SCOPE_MASK;
PyObject *outv = PyDict_GetItemWithError(c->u->u_ste->ste_symbols, k);
if (outv == NULL) {
- assert(PyErr_Occurred());
- return ERROR;
+ outv = _PyLong_GetZero();
}
assert(PyLong_Check(outv));
long outsc = (PyLong_AS_LONG(outv) >> SCOPE_OFFSET) & SCOPE_MASK;
@@ -5523,6 +5532,7 @@ static int
pop_inlined_comprehension_state(struct compiler *c, location loc,
inlined_comprehension_state state)
{
+ c->u->u_in_inlined_comp--;
PyObject *k, *v;
Py_ssize_t pos = 0;
if (state.temp_symbols) {