From fc501ceff7fefc6ea2fdd956e2c769fcd11d9fdd Mon Sep 17 00:00:00 2001 From: Michael Forney Date: Sun, 14 Apr 2024 23:34:15 -0700 Subject: decl: Save and re-open parameter scope for body --- decl.c | 49 ++++++++++++++++++++++++++++++++--------------- qbe.c | 10 ++++------ test/func-param-scope.c | 9 +++++++++ test/func-param-scope.qbe | 13 +++++++++++++ 4 files changed, 60 insertions(+), 21 deletions(-) create mode 100644 test/func-param-scope.c create mode 100644 test/func-param-scope.qbe diff --git a/decl.c b/decl.c index 57c4a4d..96428ba 100644 --- a/decl.c +++ b/decl.c @@ -537,11 +537,11 @@ is used for the qualifiers of the base type). This is corrected in declarator(). */ static void -declaratortypes(struct scope *s, struct list *result, char **name, bool allowabstract) +declaratortypes(struct scope *s, struct list *result, char **name, struct scope **funcscope, bool allowabstract) { - struct list *ptr; + struct list *ptr, *prev; struct type *t; - struct decl **p; + struct decl *d, **paramend; struct expr *e; enum typequal tq; bool allowattr; @@ -557,6 +557,7 @@ declaratortypes(struct scope *s, struct list *result, char **name, bool allowabs if (name) *name = NULL; ptr = result->next; + prev = ptr->prev; switch (tok.kind) { case TLPAREN: next(); @@ -574,7 +575,7 @@ declaratortypes(struct scope *s, struct list *result, char **name, bool allowabs goto func; } } - declaratortypes(s, result, name, allowabstract); + declaratortypes(s, result, name, funcscope, allowabstract); expect(TRPAREN, "after parenthesized declarator"); allowattr = false; break; @@ -601,10 +602,14 @@ declaratortypes(struct scope *s, struct list *result, char **name, bool allowabs t->u.func.isnoreturn = false; t->u.func.params = NULL; t->u.func.nparam = 0; - p = &t->u.func.params; + paramend = &t->u.func.params; + s = mkscope(s); while (tok.kind != TRPAREN) { - *p = parameter(s); - p = &(*p)->next; + d = parameter(s); + if (d->name) + scopeputdecl(s, d); + *paramend = d; + paramend = &d->next; ++t->u.func.nparam; if (!consume(TCOMMA)) break; @@ -613,7 +618,14 @@ declaratortypes(struct scope *s, struct list *result, char **name, bool allowabs break; } } - if (t->u.func.nparam == 1 && !t->u.func.isvararg && t->u.func.params->type->kind == TYPEVOID && !t->u.func.params->name) { + if (funcscope && ptr->prev == prev) { + /* we may need to re-open the scope later if this is a function definition */ + *funcscope = s; + s = s->parent; + } else { + s = delscope(s); + } + if (t->u.func.nparam == 1 && !t->u.func.isvararg && d->type->kind == TYPEVOID && !d->name) { t->u.func.params = NULL; t->u.func.nparam = 0; } @@ -655,14 +667,16 @@ declaratortypes(struct scope *s, struct list *result, char **name, bool allowabs } static struct qualtype -declarator(struct scope *s, struct qualtype base, char **name, bool allowabstract) +declarator(struct scope *s, struct qualtype base, char **name, struct scope **funcscope, bool allowabstract) { struct type *t; enum typequal tq; struct expr *e; struct list result = {&result, &result}, *l, *prev; - declaratortypes(s, &result, name, allowabstract); + if (funcscope) + *funcscope = NULL; + declaratortypes(s, &result, name, funcscope, allowabstract); for (l = result.prev; l != &result; l = prev) { prev = l->prev; t = listelement(l, struct type, link); @@ -715,7 +729,7 @@ parameter(struct scope *s) error(&tok.loc, "no type in parameter declaration"); if (sc && sc != SCREGISTER) error(&tok.loc, "parameter declaration has invalid storage-class specifier"); - t = declarator(s, t, &name, true); + t = declarator(s, t, &name, NULL, true); t.type = typeadjust(t.type, &t.qual); return mkdecl(name, DECLOBJECT, t.type, t.qual, LINKNONE); } @@ -855,7 +869,7 @@ structdecl(struct scope *s, struct structbuilder *b) width = intconstexpr(s, false); addmember(b, base, NULL, 0, width); } else { - mt = declarator(s, base, &name, false); + mt = declarator(s, base, &name, NULL, false); width = consume(TCOLON) ? intconstexpr(s, false) : -1; addmember(b, mt, name, align, width); } @@ -874,7 +888,7 @@ typename(struct scope *s, enum typequal *tq) t = declspecs(s, NULL, NULL, NULL); if (t.type) { - t = declarator(s, t, NULL, true); + t = declarator(s, t, NULL, NULL, true); if (tq) *tq |= t.qual; } @@ -957,6 +971,7 @@ decl(struct scope *s, struct func *f) int allowfunc = !f; struct decl *d, *prior; enum declkind kind; + struct scope *funcscope; int align; if (staticassert(s)) @@ -983,7 +998,7 @@ decl(struct scope *s, struct func *f) return true; } for (;;) { - qt = declarator(s, base, &name, false); + qt = declarator(s, base, &name, &funcscope, false); t = qt.type; tq = qt.qual; if (consume(T__ASM__)) { @@ -1051,7 +1066,9 @@ decl(struct scope *s, struct func *f) error(&tok.loc, "function definition not allowed"); if (d->defined) error(&tok.loc, "function '%s' redefined", name); - s = mkscope(&filescope); + /* re-open scope from function declarator */ + assert(funcscope); + s = funcscope; f = mkfunc(d, name, t, s); stmt(f, s); /* XXX: need to keep track of function in case a later declaration specifies extern */ @@ -1061,6 +1078,8 @@ decl(struct scope *s, struct func *f) delfunc(f); d->defined = true; return true; + } else if (funcscope) { + delscope(funcscope); } break; } diff --git a/qbe.c b/qbe.c index f3f50b3..13643dc 100644 --- a/qbe.c +++ b/qbe.c @@ -473,7 +473,7 @@ mkfunc(struct decl *decl, char *name, struct type *t, struct scope *s) { struct func *f; struct decl *d; - struct value *v, *pv; + struct value *v; f = xmalloc(sizeof(*f)); f->decl = decl; @@ -486,19 +486,17 @@ mkfunc(struct decl *decl, char *name, struct type *t, struct scope *s) /* allocate space for parameters */ f->paramtemps = xreallocarray(NULL, t->u.func.nparam, sizeof *f->paramtemps); - for (d = t->u.func.params, pv = f->paramtemps; d; d = d->next, ++pv) { + for (d = t->u.func.params, v = f->paramtemps; d; d = d->next, ++v) { emittype(d->type); - functemp(f, pv); + functemp(f, v); if(!d->name) continue; if (d->type->value) { - d->value = pv; + d->value = v; } else { - v = pv; funcalloc(f, d); funcstore(f, d->type, QUALNONE, (struct lvalue){d->value}, v); } - scopeputdecl(s, d); } t = mkarraytype(&typechar, QUALCONST, strlen(name) + 1); diff --git a/test/func-param-scope.c b/test/func-param-scope.c new file mode 100644 index 0000000..f5e51f3 --- /dev/null +++ b/test/func-param-scope.c @@ -0,0 +1,9 @@ +enum {A = 1}; +char (*f(enum {A = 2} *p, int (*a)[A], double (*b)[sizeof **a]))[A] { + static_assert(A == 2); + static_assert(sizeof *b == sizeof(int) * sizeof(double)); + static_assert(sizeof *a == 2 * sizeof(int)); + return 0; +} +static_assert(A == 1); +static_assert(sizeof *f(0, 0, 0) == 1); diff --git a/test/func-param-scope.qbe b/test/func-param-scope.qbe new file mode 100644 index 0000000..4272fc7 --- /dev/null +++ b/test/func-param-scope.qbe @@ -0,0 +1,13 @@ +export +function l $f(l %.1, l %.3, l %.5) { +@start.1 + %.2 =l alloc8 8 + storel %.1, %.2 + %.4 =l alloc8 8 + storel %.3, %.4 + %.6 =l alloc8 8 + storel %.5, %.6 +@body.2 + %.7 =l extsw 0 + ret %.7 +} -- cgit v1.2.3