ref: 8022f78f34e2b0532f880d2d64d363bb1e20531d
dir: /cc1/decl.c/
#include <inttypes.h> #include <setjmp.h> #include <stdio.h> #include <string.h> #include "../inc/sizes.h" #include "../inc/cc.h" #include "cc1.h" #define ID_EXPECTED 1 #define ID_ACCEPTED 2 #define ID_FORBIDDEN 3 /* TODO: check identifiers in enum declaration */ struct dcldata { unsigned char op; unsigned short nelem; unsigned ndcl; void *data; }; static struct dcldata * queue(struct dcldata *dp, unsigned op, short nelem, void *data) { unsigned n; if ((n = dp->ndcl) == NR_DECLARATORS) error("too much declarators"); dp->op = op; dp->nelem = nelem; dp->data = data; ++dp; dp->ndcl = n+1; return dp; } static struct dcldata * arydcl(struct dcldata *dp) { expect('['); expect(']'); return queue(dp, ARY, 0, NULL); } static Symbol *parameter(void); static struct dcldata * fundcl(struct dcldata *dp) { size_t siz; unsigned n, i, noname; Type *pars[NR_FUNPARAM], **tp = pars; Symbol *syms[NR_FUNPARAM], **sp = syms, *sym; pushctx(); expect('('); n = noname = 0; do { if ((sym = parameter()) == NULL) { if (n == 0) break; error("incorrect void parameter"); } if (n++ == NR_FUNPARAM) error("too much parameters in function definition"); *sp++ = sym; *tp++ = sym->type; noname |= sym->name[0] == '\0'; } while (accept(',')); expect(')'); if (n != 0) { siz = sizeof(*tp) * n; tp = memcpy(xmalloc(siz), pars, siz); } else { tp = NULL; } if (yytoken != '{') { /* it is only a prototype */ popctx(); } else { /* it is a function definition */ if (noname) error("parameter name omitted"); sp = syms; for (i = 0; i < n; ++i) emit(ODECL, *sp++); } return queue(dp, FTN, n, tp); } static struct dcldata *declarator0(struct dcldata *dp, unsigned ns); static struct dcldata * directdcl(struct dcldata *dp, unsigned ns) { Symbol *sym; if (accept('(')) { dp = declarator0(dp, ns); expect(')'); } else { if (yytoken == IDEN || yytoken == TYPEIDEN) { if ((sym = install(ns)) == NULL) error("redeclaration of '%s'", yytext); next(); } else { sym = newsym(ns); } dp = queue(dp, IDEN, 0, sym); } for (;;) { switch (yytoken) { case '(': dp = fundcl(dp); break; case '[': dp = arydcl(dp); break; default: return dp; } } } static struct dcldata* declarator0(struct dcldata *dp, unsigned ns) { unsigned n; for (n = 0; accept('*'); ++n) { while (accept(TQUALIFIER)) /* nothing */; } dp = directdcl(dp, ns); while (n--) dp = queue(dp, PTR, 0, NULL); return dp; } static Symbol * declarator(Type *tp, int flags, unsigned ns) { struct dcldata data[NR_DECLARATORS+1]; struct dcldata *bp; Symbol *sym; data[0].ndcl = 0; for (bp = declarator0(data, ns); bp > data; ) { --bp; if (bp->op != IDEN) { tp = mktype(tp, bp->op, bp->nelem, bp->data); } else { sym = bp->data; if (flags == ID_EXPECTED && *sym->name == '\0') error("missed identifier in declaration"); if (flags == ID_FORBIDDEN && *sym->name != '\0') error("unexpected identifier in type name"); break; } } /* TODO: deal with external array declarations of [] */ if (!tp->defined && *sym->name) error("declared variable '%s' of incomplete type", sym->name); sym->type = tp; return sym; } static Type *structdcl(void), *enumdcl(void); static Type * specifier(unsigned *sclass) { Type *tp = NULL; unsigned qlf, sign, type, cls, size; qlf = sign = type = cls = size = 0; for (;;) { unsigned *p; Type *(*dcl)(void) = NULL; switch (yytoken) { case SCLASS: p = &cls; break; case TQUALIFIER: if ((qlf |= yylval.token) & RESTRICT) goto invalid_type; next(); continue; case TYPEIDEN: if (type) goto return_type; tp = yylval.sym->type; p = &type; break; case TYPE: switch (yylval.token) { case ENUM: dcl = enumdcl; p = &type; break; case STRUCT: case UNION: dcl = structdcl; p = &type; break; case VOID: case BOOL: case CHAR: case INT: case FLOAT: case DOUBLE: p = &type; break; case SIGNED: case UNSIGNED: p = &sign; break; case LONG: if (size == LONG) { size = LLONG; break; } case SHORT: p = &size; break; } break; default: goto return_type; } if (*p) goto invalid_type; *p = yylval.token; if (dcl) { if (size || sign) goto invalid_type; tp = (*dcl)(); } else { next(); } } return_type: if (sclass) *sclass = cls; if (!tp) tp = ctype(type, sign, size); return tp; invalid_type: error("invalid type specification"); } /* TODO: check correctness of the initializator */ /* TODO: emit initializer */ static struct node * initializer(Symbol *sym) { if (!(sym->flags & ISEXTERN)) error("'%s' initialized and declared extern", sym->name); if (accept('{')) { initializer(sym); expect('}'); } else { do { expr(); } while (accept(',')); } return NULL; } static Symbol * newtag(void) { Symbol *sym; unsigned tag = yylval.token; static unsigned ns = NS_STRUCTS; setnamespace(NS_TAG); next(); switch (yytoken) { case IDEN: case TYPEIDEN: sym = yylval.sym; next(); break; default: sym = newsym(NS_TAG); break; } if (!sym->type) { if (ns == NS_STRUCTS + NR_MAXSTRUCTS) error("too much tags declared"); sym->type = mktype(NULL, tag, 0, NULL); sym->type->ns = ns++; } sym->flags |= ISDEFINED; if (sym->type->op != tag) error("'%s' defined as wrong kind of tag", yytext); return sym; } /* TODO: bitfields */ static Type * structdcl(void) { Type *tagtype, *buff[NR_MAXSTRUCTS], **bp = &buff[0]; Symbol *tagsym, *sym; unsigned n; size_t siz; tagsym = newtag(); tagtype = tagsym->type; if (!accept('{')) return tagtype; if (tagtype->defined) error("redefinition of struct/union '%s'", yytext); tagtype->defined = 1; emit(OSTRUCT, tagsym); while (!accept('}')) { Type *base, *tp; switch (yytoken) { case SCLASS: error("storage class '%s' in struct/union field", yytext); case IDEN: case TYPE: case TYPEIDEN: case TQUALIFIER: base = specifier(NULL); break; case ';': next(); continue; default: unexpected(); } if (accept(';')) error("identifier expected"); do { sym = declarator(base, ID_EXPECTED, tagtype->ns); sym->flags |= ISFIELD; tp = sym->type; if (tp->op == FTN) error("invalid type in struct/union"); if (bp == &buff[NR_MAXSTRUCTS]) error("too much fields in struct/union"); *bp++ = sym->type; emit(ODECL, sym); } while (accept(',')); expect(';'); } emit(OESTRUCT, NULL); if ((n = bp - buff) != 0) { siz = sizeof(Type *) * n; tagtype->n.elem = n; tagtype->pars = memcpy(xmalloc(siz), buff, siz); } return tagtype; } static Type * enumdcl(void) { Type *tp; Symbol *sym; int val = 0; tp = newtag()->type; if (yytoken == ';') return tp; expect('{'); if (tp->defined) error("redefinition of enumeration '%s'", yytext); tp->defined = 1; while (yytoken != '}') { if (yytoken != IDEN) unexpected(); if ((sym = install(NS_IDEN)) == NULL) error("duplicated member '%s'", yytext); next(); sym->type = inttype; if (accept('=')) initializer(sym); sym->u.i = val++; if (!accept(',')) break; } expect('}'); return tp; } static Symbol * parameter(void) { Symbol *sym; unsigned sclass; Type *tp; if ((tp = specifier(&sclass)) == voidtype) return NULL; sym = declarator(tp, ID_ACCEPTED, NS_IDEN); sym->flags |= ISPARAM; tp = sym->type; if (tp->op == FTN) error("incorrect function type for a function parameter"); if (tp->op == ARY) tp = mktype(tp->type, PTR, 0, NULL); switch (sclass) { case REGISTER: sym->flags |= ISREGISTER; break; case 0: sym->flags |= ISAUTO; break; default: error("bad storage class in function parameter"); } return sym; } void decl(void) { Type *tp; Symbol *sym; unsigned sclass, isfun; extern jmp_buf recover; setsafe(END_DECL); if (setjmp(recover)) return; tp = specifier(&sclass); if (accept(';')) return; do { setsafe(END_LDECL); setjmp(recover); sym = declarator(tp, ID_EXPECTED, NS_IDEN); isfun = sym->type->op == FTN; switch (sclass) { case TYPEDEF: sym->token = TYPEIDEN; continue; case STATIC: sym->flags |= ISSTATIC; break; case EXTERN: sym->flags |= ISEXTERN; break; case REGISTER: sym->flags = ISREGISTER; if (isfun) goto bad_function; break; case AUTO: if (isfun) goto bad_function; /* passtrough */ default: sym->flags |= ISAUTO; break; } if (accept('=')) initializer(sym); emit(ODECL, sym); } while (accept(',')); expect(';'); return; bad_function: error("invalid storage class for function '%s'", sym->name); } Type * typename(void) { unsigned sclass; Type *tp; Symbol *sym; tp = specifier(&sclass); if (sclass) error("class storage in type name"); sym = declarator(tp, ID_FORBIDDEN, NS_IDEN); return sym->type; } void extdecl(void) { Type *base, *tp; unsigned sclass; Symbol *sym; extern Symbol *curfun; extern jmp_buf recover; setsafe(END_DECL); if (setjmp(recover)) return; switch (yytoken) { case IDEN: case TYPE: case TYPEIDEN: case SCLASS: case TQUALIFIER: base = specifier(&sclass); if (accept(';')) return; do { /* FIX: we cannot put a setjmp here because base was already assigned, and we were having problems with EOF */ sym = declarator(base, ID_EXPECTED, NS_IDEN); tp = sym->type; sym->flags |= ISSTATIC; sym->flags |= ISGLOBAL; switch (sclass) { case REGISTER: case AUTO: error("incorrect storage class for file-scope declaration"); case STATIC: sym->flags |= ISSTATIC; break; case EXTERN: sym->flags |= ISEXTERN; break; case TYPEDEF: sym->token = TYPEIDEN; continue; } if (tp->op != FTN) { if (accept('=')) initializer(sym); emit(ODECL, sym); } else if (yytoken == '{') { curfun = sym; emit(OFUN, sym); compound(NULL, NULL, NULL); emit(OEFUN, NULL); popctx(); return; } } while (accept(',')); /* PASSTHROUGH */ case ';': expect(';'); return; default: unexpected(); } }