#include #include #include #include #include #include #include #include "util.h" #include "cc.h" static struct list tentativedefns = {&tentativedefns, &tentativedefns}; struct qualtype { struct type *type; enum typequal qual; }; enum storageclass { SCNONE, SCTYPEDEF = 1<<1, SCEXTERN = 1<<2, SCSTATIC = 1<<3, SCAUTO = 1<<4, SCREGISTER = 1<<5, SCTHREADLOCAL = 1<<6, }; enum typespec { SPECNONE, SPECVOID = 1<<1, SPECCHAR = 1<<2, SPECBOOL = 1<<3, SPECINT = 1<<4, SPECFLOAT = 1<<5, SPECDOUBLE = 1<<6, SPECSHORT = 1<<7, SPECLONG = 1<<8, SPECLONG2 = 1<<9, SPECSIGNED = 1<<10, SPECUNSIGNED = 1<<11, SPECCOMPLEX = 1<<12, SPECLONGLONG = SPECLONG|SPECLONG2, }; enum funcspec { FUNCNONE, FUNCINLINE = 1<<1, FUNCNORETURN = 1<<2, }; struct structbuilder { struct type *type; struct member **last; unsigned bits; /* number of bits remaining in the last byte */ }; struct decl * mkdecl(enum declkind k, struct type *t, enum typequal tq, enum linkage linkage) { struct decl *d; d = xmalloc(sizeof(*d)); memset(d, 0, sizeof(*d)); d->kind = k; d->linkage = linkage; d->type = t; d->qual = tq; if (k == DECLOBJECT) d->u.obj.align = t->align; return d; } /* 6.7.1 Storage-class specifiers */ static int storageclass(enum storageclass *sc) { enum storageclass allowed, new; switch (tok.kind) { case TTYPEDEF: new = SCTYPEDEF; break; case TEXTERN: new = SCEXTERN; break; case TSTATIC: new = SCSTATIC; break; case TTHREAD_LOCAL: new = SCTHREADLOCAL; break; case TAUTO: new = SCAUTO; break; case TREGISTER: new = SCREGISTER; break; default: return 0; } if (!sc) error(&tok.loc, "storage class not allowed in this declaration"); switch (*sc) { case SCNONE: allowed = ~SCNONE; break; case SCTHREADLOCAL: allowed = SCSTATIC|SCEXTERN; break; case SCSTATIC: case SCEXTERN: allowed = SCTHREADLOCAL; break; default: allowed = SCNONE; break; } if (new & ~allowed) error(&tok.loc, "invalid combination of storage class specifiers"); *sc |= new; next(); return 1; } /* 6.7.3 Type qualifiers */ static int typequal(enum typequal *tq) { switch (tok.kind) { case TCONST: *tq |= QUALCONST; break; case TVOLATILE: *tq |= QUALVOLATILE; break; case TRESTRICT: *tq |= QUALRESTRICT; break; case T_ATOMIC: error(&tok.loc, "_Atomic type qualifier is not yet supported"); default: return 0; } next(); return 1; } /* 6.7.4 Function specifiers */ static int funcspec(enum funcspec *fs) { enum funcspec new; switch (tok.kind) { case TINLINE: new = FUNCINLINE; break; case T_NORETURN: new = FUNCNORETURN; break; default: return 0; } if (!fs) error(&tok.loc, "function specifier not allowed in this declaration"); *fs |= new; next(); return 1; } static void structdecl(struct scope *, struct structbuilder *); static struct qualtype declspecs(struct scope *, enum storageclass *, enum funcspec *, int *); static struct type * tagspec(struct scope *s) { static struct type *const inttypes[][2] = { {&typeuint, &typeint}, {&typeulong, &typelong}, {&typeullong, &typellong}, }; struct type *t, *et; char *tag, *name; enum typekind kind; struct decl *d, *enumconsts; struct expr *e; struct structbuilder b; unsigned long long value, max, min; bool sign; int i; switch (tok.kind) { case TSTRUCT: kind = TYPESTRUCT; break; case TUNION: kind = TYPEUNION; break; case TENUM: kind = TYPEENUM; break; default: fatal("internal error: unknown tag kind"); } next(); tag = NULL; t = NULL; et = NULL; if (tok.kind == TIDENT) { tag = tok.lit; next(); } if (kind == TYPEENUM && consume(TCOLON)) { et = declspecs(s, NULL, NULL, NULL).type; if (!et) error(&tok.loc, "no type in enum type specifier"); } if (tag) t = scopegettag(s, tag, tok.kind != TLBRACE && tok.kind != TSEMICOLON); if (t) { if (t->kind != kind) error(&tok.loc, "redeclaration of tag '%s' with different kind", tag); } else { if (kind == TYPEENUM) { t = mktype(kind, PROPSCALAR|PROPARITH|PROPREAL|PROPINT); t->base = et; } else { t = mktype(kind, 0); t->size = 0; t->align = 0; t->u.structunion.tag = tag; t->u.structunion.members = NULL; } t->incomplete = true; if (tag) scopeputtag(s, tag, t); } if (tok.kind != TLBRACE) return t; if (!t->incomplete) error(&tok.loc, "redefinition of tag '%s'", tag); next(); switch (t->kind) { case TYPESTRUCT: case TYPEUNION: b.type = t; b.last = &t->u.structunion.members; b.bits = 0; do structdecl(s, &b); while (tok.kind != TRBRACE); if (!t->u.structunion.members) error(&tok.loc, "struct/union has no members"); next(); t->size = ALIGNUP(t->size, t->align); break; case TYPEENUM: enumconsts = NULL; if (et) { t->size = t->base->size; t->align = t->base->align; t->u.basic.issigned = t->base->u.basic.issigned; t->incomplete = false; et = t; } else { et = &typeint; } max = 0; min = 0; for (value = 0; tok.kind == TIDENT; ++value) { name = tok.lit; next(); if (consume(TASSIGN)) { e = evalexpr(s); if (e->kind != EXPRCONST || !(e->type->prop & PROPINT)) error(&tok.loc, "expected integer constant expression"); value = e->u.constant.u; if (!t->base) et = typehasint(&typeint, value, e->type->u.basic.issigned) ? &typeint : e->type; else if (!typehasint(et, value, e->type->u.basic.issigned)) goto invalid; } else if (value == 0 && !et->u.basic.issigned || value == 1ull << 63 && et->u.basic.issigned) { error(&tok.loc, "no %ssigned integer type can represent enumerator value", et->u.basic.issigned ? "" : "un"); } else if (!typehasint(et, value, et->u.basic.issigned)) { if (t->base) { invalid: /* fixed underlying type */ error(&tok.loc, "enumerator '%s' value cannot be represented in underlying type", name); } sign = et->u.basic.issigned; for (i = 0; i < LEN(inttypes); ++i) { et = inttypes[i][sign]; if (typehasint(et, value, sign)) break; } assert(i < LEN(inttypes)); } d = mkdecl(DECLCONST, et, QUALNONE, LINKNONE); d->value = mkintconst(value); d->u.enumconst.next = enumconsts; enumconsts = d; if (et->u.basic.issigned && value >= 1ull << 63) { if (-value > min) min = -value; } else if (value > max) { max = value; } scopeputdecl(s, name, d); if (!consume(TCOMMA)) break; } expect(TRBRACE, "to close enum specifier"); if (!t->base) { if (min <= 0x80000000 && max <= 0x7fffffff) { t->base = min ? &typeint : &typeuint; } else { sign = min > 0; for (i = 0; i < LEN(inttypes); ++i) { et = inttypes[i][sign]; if (typehasint(et, max, false) && typehasint(et, -min, true)) break; } if (i == LEN(inttypes)) error(&tok.loc, "no integer type can represent all enumerator values"); t->base = et; for (d = enumconsts; d; d = d->u.enumconst.next) d->type = t; } t->size = t->base->size; t->align = t->base->align; t->u.basic.issigned = t->base->u.basic.issigned; } } t->incomplete = false; return t; } /* 6.7 Declarations */ static struct qualtype declspecs(struct scope *s, enum storageclass *sc, enum funcspec *fs, int *align) { struct type *t, *other; struct decl *d; struct expr *e; enum typespec ts = SPECNONE; enum typequal tq = QUALNONE; int ntypes = 0; unsigned long long i; t = NULL; if (sc) *sc = SCNONE; if (fs) *fs = FUNCNONE; if (align) *align = 0; for (;;) { if (typequal(&tq) || storageclass(sc) || funcspec(fs)) continue; switch (tok.kind) { /* 6.7.2 Type specifiers */ case TVOID: t = &typevoid; ++ntypes; next(); break; case TCHAR: ts |= SPECCHAR; ++ntypes; next(); break; case TSHORT: if (ts & SPECSHORT) error(&tok.loc, "duplicate 'short'"); ts |= SPECSHORT; next(); break; case TINT: ts |= SPECINT; ++ntypes; next(); break; case TLONG: if (ts & SPECLONG2) error(&tok.loc, "too many 'long'"); if (ts & SPECLONG) ts |= SPECLONG2; ts |= SPECLONG; next(); break; case TFLOAT: ts |= SPECFLOAT; ++ntypes; next(); break; case TDOUBLE: ts |= SPECDOUBLE; ++ntypes; next(); break; case TSIGNED: if (ts & SPECSIGNED) error(&tok.loc, "duplicate 'signed'"); ts |= SPECSIGNED; next(); break; case TUNSIGNED: if (ts & SPECUNSIGNED) error(&tok.loc, "duplicate 'unsigned'"); ts |= SPECUNSIGNED; next(); break; case TBOOL: t = &typebool; ++ntypes; next(); break; case T_COMPLEX: error(&tok.loc, "_Complex is not yet supported"); break; case T_ATOMIC: error(&tok.loc, "_Atomic is not yet supported"); break; case TSTRUCT: case TUNION: case TENUM: t = tagspec(s); ++ntypes; break; case TIDENT: if (t || ts) goto done; d = scopegetdecl(s, tok.lit, 1); if (!d || d->kind != DECLTYPE) goto done; t = d->type; tq |= d->qual; ++ntypes; next(); break; case TTYPEOF: next(); expect(TLPAREN, "after 'typeof'"); t = typename(s, &tq); if (!t) { e = expr(s); if (e->decayed) e = e->base; t = e->type; tq |= e->qual; delexpr(e); } ++ntypes; expect(TRPAREN, "to close 'typeof'"); break; /* 6.7.5 Alignment specifier */ case TALIGNAS: if (!align) error(&tok.loc, "alignment specifier not allowed in this declaration"); next(); expect(TLPAREN, "after 'alignas'"); other = typename(s, NULL); i = other ? other->align : intconstexpr(s, false); if (i & (i - 1)) error(&tok.loc, "invalid alignment: %d", i); if (i > *align) *align = i; expect(TRPAREN, "to close 'alignas' specifier"); break; default: goto done; } if (ntypes > 1 || (t && ts)) error(&tok.loc, "multiple types in declaration specifiers"); } done: switch ((int)ts) { case SPECNONE: break; case SPECCHAR: t = &typechar; break; case SPECSIGNED|SPECCHAR: t = &typeschar; break; case SPECUNSIGNED|SPECCHAR: t = &typeuchar; break; case SPECSHORT: case SPECSHORT|SPECINT: case SPECSIGNED|SPECSHORT: case SPECSIGNED|SPECSHORT|SPECINT: t = &typeshort; break; case SPECUNSIGNED|SPECSHORT: case SPECUNSIGNED|SPECSHORT|SPECINT: t = &typeushort; break; case SPECINT: case SPECSIGNED: case SPECSIGNED|SPECINT: t = &typeint; break; case SPECUNSIGNED: case SPECUNSIGNED|SPECINT: t = &typeuint; break; case SPECLONG: case SPECLONG|SPECINT: case SPECSIGNED|SPECLONG: case SPECSIGNED|SPECLONG|SPECINT: t = &typelong; break; case SPECUNSIGNED|SPECLONG: case SPECUNSIGNED|SPECLONG|SPECINT: t = &typeulong; break; case SPECLONGLONG: case SPECLONGLONG|SPECINT: case SPECSIGNED|SPECLONGLONG: case SPECSIGNED|SPECLONGLONG|SPECINT: t = &typellong; break; case SPECUNSIGNED|SPECLONGLONG: case SPECUNSIGNED|SPECLONGLONG|SPECINT: t = &typeullong; break; case SPECFLOAT: t = &typefloat; break; case SPECDOUBLE: t = &typedouble; break; case SPECLONG|SPECDOUBLE: t = &typeldouble; break; default: error(&tok.loc, "invalid combination of type specifiers"); } if (!t && (tq || sc && *sc || fs && *fs)) error(&tok.loc, "declaration has no type specifier"); if (t && tq && t->kind == TYPEARRAY) { t = mkarraytype(t->base, t->qual | tq, t->u.array.length); tq = QUALNONE; } return (struct qualtype){t, tq}; } /* 6.7.6 Declarators */ static struct param *parameter(struct scope *); static bool istypename(struct scope *s, const char *name) { struct decl *d; d = scopegetdecl(s, name, 1); return d && d->kind == DECLTYPE; } /* When parsing a declarator, qualifiers for derived types are temporarily stored in the `qual` field of the type itself (elsewhere this field 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) { struct list *ptr; struct type *t; struct param **p; struct expr *e; unsigned long long i; enum typequal tq; while (consume(TMUL)) { tq = QUALNONE; while (typequal(&tq)) ; t = mkpointertype(NULL, tq); listinsert(result, &t->link); } if (name) *name = NULL; ptr = result->next; switch (tok.kind) { case TLPAREN: next(); if (allowabstract) { switch (tok.kind) { case TMUL: case TLPAREN: break; case TIDENT: if (!istypename(s, tok.lit)) break; /* fallthrough */ default: goto func; } } declaratortypes(s, result, name, allowabstract); expect(TRPAREN, "after parenthesized declarator"); break; case TIDENT: if (!name) error(&tok.loc, "identifier not allowed in abstract declarator"); *name = tok.lit; next(); break; default: if (!allowabstract) error(&tok.loc, "expected '(' or identifier"); } for (;;) { switch (tok.kind) { case TLPAREN: /* function declarator */ next(); func: t = mktype(TYPEFUNC, 0); t->qual = QUALNONE; t->u.func.isprototype = false; t->u.func.isvararg = false; t->u.func.isnoreturn = false; t->u.func.params = NULL; t->u.func.nparam = 0; p = &t->u.func.params; switch (tok.kind) { case TIDENT: if (!istypename(s, tok.lit)) { /* identifier-list (K&R declaration) */ do { *p = mkparam(tok.lit, NULL, QUALNONE); p = &(*p)->next; next(); } while (consume(TCOMMA) && tok.kind == TIDENT); break; } /* fallthrough */ default: t->u.func.isprototype = true; for (;;) { *p = parameter(s); p = &(*p)->next; ++t->u.func.nparam; if (!consume(TCOMMA)) break; if (consume(TELLIPSIS)) { t->u.func.isvararg = true; break; } } if (t->u.func.params->type->kind == TYPEVOID && !t->u.func.params->next) t->u.func.params = NULL; break; case TRPAREN: break; } expect(TRPAREN, "to close function declarator"); t->u.func.paraminfo = t->u.func.isprototype || t->u.func.params || tok.kind == TLBRACE; listinsert(ptr->prev, &t->link); break; case TLBRACK: /* array declarator */ next(); tq = QUALNONE; while (consume(TSTATIC) || typequal(&tq)) ; if (tok.kind == TMUL) error(&tok.loc, "VLAs are not yet supported"); t = mkarraytype(NULL, tq, 0); if (tok.kind != TRBRACK) { e = eval(assignexpr(s), EVALARITH); if (e->kind != EXPRCONST || !(e->type->prop & PROPINT)) error(&tok.loc, "VLAs are not yet supported"); i = e->u.constant.u; if (e->type->u.basic.issigned && i >> 63) error(&tok.loc, "array length must be non-negative"); delexpr(e); t->u.array.length = i; t->incomplete = false; } expect(TRBRACK, "after array length"); listinsert(ptr->prev, &t->link); break; default: return; } } } static struct qualtype declarator(struct scope *s, struct qualtype base, char **name, bool allowabstract) { struct type *t; enum typequal tq; struct list result = {&result, &result}, *l, *prev; declaratortypes(s, &result, name, allowabstract); for (l = result.prev; l != &result; l = prev) { prev = l->prev; t = listelement(l, struct type, link); tq = t->qual; t->base = base.type; t->qual = base.qual; switch (t->kind) { case TYPEFUNC: if (base.type->kind == TYPEFUNC) error(&tok.loc, "function declarator specifies function return type"); if (base.type->kind == TYPEARRAY) error(&tok.loc, "function declarator specifies array return type"); break; case TYPEARRAY: if (base.type->incomplete) error(&tok.loc, "array element has incomplete type"); if (base.type->kind == TYPEFUNC) error(&tok.loc, "array element has function type"); t->align = base.type->align; t->size = base.type->size * t->u.array.length; /* XXX: overflow? */ break; } base.type = t; base.qual = tq; } return base; } static struct param * parameter(struct scope *s) { char *name; struct qualtype t; enum storageclass sc; t = declspecs(s, &sc, NULL, NULL); if (!t.type) 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); return mkparam(name, typeadjust(t.type), t.qual); } static bool paramdecl(struct scope *s, struct param *params) { struct param *p; struct qualtype t, base; enum storageclass sc; char *name; base = declspecs(s, &sc, NULL, NULL); if (!base.type) return false; if (sc && sc != SCREGISTER) error(&tok.loc, "parameter declaration has invalid storage-class specifier"); for (;;) { t = declarator(s, base, &name, false); for (p = params; p && strcmp(name, p->name) != 0; p = p->next) ; if (!p) error(&tok.loc, "old-style function declarator has no parameter named '%s'", name); p->type = typeadjust(t.type); p->qual = t.qual; if (tok.kind == TSEMICOLON) break; expect(TCOMMA, "or ';' after parameter declarator"); } next(); return true; } static void addmember(struct structbuilder *b, struct qualtype mt, char *name, int align, unsigned long long width) { struct type *t = b->type; struct member *m; size_t end; if (t->flexible) error(&tok.loc, "struct has member after flexible array member"); if (mt.type->incomplete) { if (mt.type->kind != TYPEARRAY) error(&tok.loc, "struct member has incomplete type"); t->flexible = true; } if (mt.type->kind == TYPEFUNC) error(&tok.loc, "struct member has function type"); if (mt.type->flexible) error(&tok.loc, "struct member contains flexible array member"); assert(mt.type->align > 0); if (name || width == -1) { m = xmalloc(sizeof(*m)); m->type = mt.type; m->qual = mt.qual; m->name = name; m->next = NULL; *b->last = m; b->last = &m->next; } else { m = NULL; } if (width == -1) { m->bits.before = 0; m->bits.after = 0; if (align < mt.type->align) { if (align) error(&tok.loc, "specified alignment of struct member is less strict than is required by type"); align = mt.type->align; } if (t->kind == TYPESTRUCT) { m->offset = ALIGNUP(t->size, align); t->size = m->offset + mt.type->size; } else { m->offset = 0; if (t->size < mt.type->size) t->size = mt.type->size; } b->bits = 0; } else { /* bit-field */ if (!(mt.type->prop & PROPINT)) error(&tok.loc, "bit-field has invalid type"); if (align) error(&tok.loc, "alignment specified for bit-field"); if (!width && name) error(&tok.loc, "bit-field with zero width must not have declarator"); if (width > mt.type->size * 8) error(&tok.loc, "bit-field exceeds width of underlying type"); align = mt.type->align; if (t->kind == TYPESTRUCT) { /* calculate end of the storage-unit for this bit-field */ end = ALIGNUP(t->size, mt.type->size); if (!width || width > (end - t->size) * 8 + b->bits) { /* no room, allocate a new storage-unit */ t->size = end; b->bits = 0; } if (m) { m->offset = ALIGNDOWN(t->size - !!b->bits, mt.type->size); m->bits.before = (t->size - m->offset) * 8 - b->bits; m->bits.after = mt.type->size * 8 - width - m->bits.before; } t->size += (width - b->bits + 7) / 8; b->bits = (b->bits - width) % 8; } else if (m) { m->offset = 0; m->bits.before = 0; m->bits.after = mt.type->size * 8 - width; if (t->size < mt.type->size) t->size = mt.type->size; } } if (m && t->align < align) t->align = align; } static bool staticassert(struct scope *s) { struct stringlit msg; unsigned long long c; if (!consume(TSTATIC_ASSERT)) return false; expect(TLPAREN, "after static_assert"); c = intconstexpr(s, true); if (consume(TCOMMA)) { tokencheck(&tok, TSTRINGLIT, "after static assertion expression"); stringconcat(&msg, true); if (!c) error(&tok.loc, "static assertion failed: %.*s", (int)(msg.size - 1), msg.data); } else if (!c) { error(&tok.loc, "static assertion failed"); } expect(TRPAREN, "after static assertion"); expect(TSEMICOLON, "after static assertion"); return true; } static void structdecl(struct scope *s, struct structbuilder *b) { struct qualtype base, mt; char *name; unsigned long long width; int align; if (staticassert(s)) return; base = declspecs(s, NULL, NULL, &align); if (!base.type) error(&tok.loc, "no type in struct member declaration"); if (tok.kind == TSEMICOLON) { if ((base.type->kind != TYPESTRUCT && base.type->kind != TYPEUNION) || base.type->u.structunion.tag) error(&tok.loc, "struct declaration must declare at least one member"); next(); addmember(b, base, NULL, align, -1); return; } for (;;) { if (consume(TCOLON)) { width = intconstexpr(s, false); addmember(b, base, NULL, 0, width); } else { mt = declarator(s, base, &name, false); width = consume(TCOLON) ? intconstexpr(s, false) : -1; addmember(b, mt, name, align, width); } if (tok.kind == TSEMICOLON) break; expect(TCOMMA, "or ';' after declarator"); } next(); } /* 6.7.7 Type names */ struct type * typename(struct scope *s, enum typequal *tq) { struct qualtype t; t = declspecs(s, NULL, NULL, NULL); if (t.type) { t = declarator(s, t, NULL, true); if (tq) *tq |= t.qual; } return t.type; } static enum linkage getlinkage(enum declkind kind, enum storageclass sc, struct decl *prior, bool filescope) { if (sc & SCSTATIC) return filescope ? LINKINTERN : LINKNONE; if (sc & SCEXTERN || kind == DECLFUNC) return prior ? prior->linkage : LINKEXTERN; return filescope ? LINKEXTERN : LINKNONE; } static struct decl * declcommon(struct scope *s, enum declkind kind, char *name, char *asmname, struct type *t, enum typequal tq, enum storageclass sc, struct decl *prior) { struct decl *d; enum linkage linkage; const char *kindstr = kind == DECLFUNC ? "function" : "object"; if (prior) { if (prior->linkage == LINKNONE) error(&tok.loc, "%s '%s' with no linkage redeclared", kindstr, name); linkage = getlinkage(kind, sc, prior, s == &filescope); if (prior->linkage != linkage) error(&tok.loc, "%s '%s' redeclared with different linkage", kindstr, name); if (!typecompatible(t, prior->type) || tq != prior->qual) error(&tok.loc, "%s '%s' redeclared with incompatible type", kindstr, name); if (asmname && (!prior->asmname || strcmp(prior->asmname, asmname) != 0)) error(&tok.loc, "%s '%s' redeclared with different assembler name", kindstr, name); prior->type = typecomposite(t, prior->type); return prior; } if (s->parent) prior = scopegetdecl(s->parent, name, true); linkage = getlinkage(kind, sc, prior, s == &filescope); if (linkage != LINKNONE && s->parent) { /* XXX: should maintain map of identifiers with linkage to their declaration, and use that */ if (s->parent != &filescope) prior = scopegetdecl(&filescope, name, false); if (prior && prior->linkage != LINKNONE) { if (prior->kind != kind) error(&tok.loc, "'%s' redeclared with different kind", name); if (prior->linkage != linkage) error(&tok.loc, "%s '%s' redeclared with different linkage", kindstr, name); if (!typecompatible(t, prior->type) || tq != prior->qual) error(&tok.loc, "%s '%s' redeclared with incompatible type", kindstr, name); if (!asmname) asmname = prior->asmname; else if (!prior->asmname || strcmp(prior->asmname, asmname) != 0) error(&tok.loc, "%s '%s' redeclared with different assembler name", kindstr, name); t = typecomposite(t, prior->type); } } d = mkdecl(kind, t, tq, linkage); scopeputdecl(s, name, d); if (kind == DECLFUNC || linkage != LINKNONE || sc & SCSTATIC) { if (asmname) name = asmname; d->value = mkglobal(name, linkage == LINKNONE && !asmname); d->asmname = asmname; } return d; } bool decl(struct scope *s, struct func *f) { struct qualtype base, qt; struct type *t; enum typequal tq; enum storageclass sc; enum funcspec fs; struct init *init; bool hasinit; struct param *p; char *name, *asmname; int allowfunc = !f; struct decl *d, *prior; enum declkind kind; int align; if (staticassert(s)) return true; base = declspecs(s, &sc, &fs, &align); if (!base.type) return false; if (f) { if (sc == SCTHREADLOCAL) error(&tok.loc, "block scope declaration containing 'thread_local' must contain 'static' or 'extern'"); } else { /* 6.9p2 */ if (sc & SCAUTO) error(&tok.loc, "external declaration must not contain 'auto'"); if (sc & SCREGISTER) error(&tok.loc, "external declaration must not contain 'register'"); } if (sc & SCTHREADLOCAL) error(&tok.loc, "'_Thread_local' is not yet supported"); if (consume(TSEMICOLON)) { /* XXX 6.7p2 error unless in function parameter/struct/union, or tag/enum members are declared */ return true; } for (;;) { qt = declarator(s, base, &name, false); t = qt.type; tq = qt.qual; if (consume(T__ASM__)) { expect(TLPAREN, "after __asm__"); asmname = expect(TSTRINGLIT, "for assembler name"); expect(TRPAREN, "after assembler name"); allowfunc = 0; } else { asmname = NULL; } kind = sc & SCTYPEDEF ? DECLTYPE : t->kind == TYPEFUNC ? DECLFUNC : DECLOBJECT; prior = scopegetdecl(s, name, false); if (prior && prior->kind != kind) error(&tok.loc, "'%s' redeclared with different kind", name); switch (kind) { case DECLTYPE: if (align) error(&tok.loc, "typedef '%s' declared with alignment specifier", name); if (asmname) error(&tok.loc, "typedef '%s' declared with assembler label", name); if (!prior) scopeputdecl(s, name, mkdecl(DECLTYPE, t, tq, LINKNONE)); else if (!typesame(prior->type, t) || prior->qual != tq) error(&tok.loc, "typedef '%s' redefined with different type", name); break; case DECLOBJECT: if (align && align < t->align) error(&tok.loc, "object '%s' requires alignment %d, which is stricter than specified alignment %d", name, t->align, align); d = declcommon(s, kind, name, asmname, t, tq, sc, prior); if (d->u.obj.align < align) d->u.obj.align = align; init = NULL; hasinit = false; if (consume(TASSIGN)) { if (f && d->linkage != LINKNONE) error(&tok.loc, "object '%s' with block scope and %s linkage cannot have initializer", name, d->linkage == LINKEXTERN ? "external" : "internal"); if (d->defined) error(&tok.loc, "object '%s' redefined", name); init = parseinit(s, d->type); hasinit = true; } else if (d->linkage != LINKNONE) { if (!(sc & SCEXTERN) && !d->defined && !d->u.obj.tentative.next) listinsert(tentativedefns.prev, &d->u.obj.tentative); break; } if (d->linkage != LINKNONE || sc & SCSTATIC) emitdata(d, init); else funcinit(f, d, init, hasinit); d->defined = true; if (d->u.obj.tentative.next) listremove(&d->u.obj.tentative); break; case DECLFUNC: if (align) error(&tok.loc, "function '%s' declared with alignment specifier", name); t->u.func.isnoreturn |= fs & FUNCNORETURN; if (f && sc && sc != SCEXTERN) /* 6.7.1p7 */ error(&tok.loc, "function '%s' with block scope may only have storage class 'extern'", name); if (!t->u.func.isprototype && t->u.func.params) { if (!allowfunc) error(&tok.loc, "function definition not allowed"); /* collect type information for parameters before we check compatibility */ while (paramdecl(s, t->u.func.params)) ; if (tok.kind != TLBRACE) error(&tok.loc, "function declaration with identifier list is not part of definition"); for (p = t->u.func.params; p; p = p->next) { if (!p->type) error(&tok.loc, "old-style function definition does not declare '%s'", p->name); } } d = declcommon(s, kind, name, asmname, t, tq, sc, prior); d->u.func.inlinedefn = d->linkage == LINKEXTERN && fs & FUNCINLINE && !(sc & SCEXTERN) && (!prior || prior->u.func.inlinedefn); if (tok.kind == TLBRACE) { if (!allowfunc) error(&tok.loc, "function definition not allowed"); if (d->defined) error(&tok.loc, "function '%s' redefined", name); s = mkscope(&filescope); f = mkfunc(d, name, t, s); stmt(f, s); /* XXX: need to keep track of function in case a later declaration specifies extern */ if (!d->u.func.inlinedefn) emitfunc(f, d->linkage == LINKEXTERN); s = delscope(s); delfunc(f); d->defined = true; return true; } break; } if (consume(TSEMICOLON)) return true; expect(TCOMMA, "or ';' after declarator"); allowfunc = 0; } } struct decl * stringdecl(struct expr *expr) { static struct map *strings; struct mapkey key; void **entry; struct decl *d; if (!strings) strings = mkmap(64); assert(expr->kind == EXPRSTRING); mapkey(&key, expr->u.string.data, expr->u.string.size); entry = mapput(strings, &key); d = *entry; if (!d) { d = mkdecl(DECLOBJECT, expr->type, QUALNONE, LINKNONE); d->value = mkglobal("string", true); emitdata(d, mkinit(0, expr->type->size, (struct bitfield){0}, expr)); *entry = d; } return d; } void emittentativedefns(void) { struct list *l; for (l = tentativedefns.next; l != &tentativedefns; l = l->next) emitdata(listelement(l, struct decl, u.obj.tentative), NULL); }