ref: b9df40911c422cfbfddcda16321dab9736e9d1c1
dir: /cc1/stmt.c/
#include <stddef.h> #include <stdint.h> #include <stdio.h> #include <cc.h> #include "cc1.h" struct scase { Symbol *label; Node *expr; struct scase *next; }; struct caselist { short nr; Symbol *deflabel; struct scase *head; }; Symbol *curfun; extern Node *convert(Node *np, Type *tp1, char iscast); extern Node *iszero(Node *np), *eval(Node *np); static void stmt(Symbol *lbreak, Symbol *lcont, Caselist *lswitch); static Symbol * label(char *s, char define) { Symbol *sym; if (s) { if ((sym = lookup(s, NS_LABEL)) != NULL) { if (define && sym->s.isdefined) error("label '%s' already defined", s); else sym->s.isdefined = 1; return sym; } } else { s = ""; } sym = install(s, NS_LABEL); sym->s.isdefined = define; return sym; } static void stmtexp(void) { if (accept(';')) return; emitexp(expr()); expect(';'); } static Node * condition(void) { Node *np; expect('('); np = iszero(expr()); expect(')'); return np; } static void While(Caselist *lswitch) { Symbol *begin, *cond, *end; Node *np; begin = label(NULL, 1); end = label(NULL, 1); cond = label(NULL, 1); expect(WHILE); np = condition(); emitjump(cond, NULL); emitbloop(); emitlabel(begin); stmt(end, begin, lswitch); emitlabel(cond); emitjump(begin, np); emiteloop(); emitlabel(end); } static void For(Caselist *lswitch) { Symbol *begin, *cond, *end; Node *econd = NULL, *einc = NULL; begin = label(NULL, 1); end = label(NULL, 1); cond = label(NULL, 1); expect(FOR); expect('('); stmtexp(); if (yytoken != ';') econd = expr(); expect(';'); if (yytoken != ')') einc = expr(); expect(')'); emitjump(cond, NULL); emitbloop(); emitlabel(begin); stmt(end, begin, lswitch); if (einc) emitexp(einc); emitlabel(cond); emitjump(begin, econd); emiteloop(); emitlabel(end); } static void Dowhile(Caselist *lswitch) { Symbol *begin= label(NULL, 1), *end = label(NULL, 1); expect(DO); emitbloop(); emitlabel(begin); stmt(end, begin, lswitch); expect(WHILE); emitjump(begin, condition()); emiteloop(); emitlabel(end); } static void Return(void) { Node *np; Type *tp = curfun->type->type; expect(RETURN); np = (yytoken == ';') ? NULL : eval(expr()); expect(';'); if (!np) { if (tp != voidtype) warn(1, "function returning non void returns no value"); tp = voidtype; } else if (np->type != tp) { if (tp == voidtype) warn(1, "function returning void returns a value"); else if ((np = convert(np, tp, 0)) == NULL) error("incorrect type in return"); } emitret(tp); emitexp(np); } static void Break(Symbol *lbreak) { expect(BREAK); if (!lbreak) error("break statement not within loop or switch"); emitjump(lbreak, NULL); expect(';'); } static void Label(void) { emitlabel(label(yytext, 1)); expect(IDEN); expect(':'); } static void Continue(Symbol *lcont) { expect(CONTINUE); if (!lcont) error("continue statement not within loop"); emitjump(lcont, NULL); expect(';'); } static void Goto(void) { expect(GOTO); if (yytoken != IDEN) error("unexpected '%s'", yytext); emitjump(label(yytext, 0), NULL); next(); expect(';'); } static void Switch(Symbol *lcont) { Caselist lcase = {.nr = 0, .head = NULL, .deflabel = NULL}; struct scase *p; Symbol *lbreak = label(NULL, 1), *lcond = label(NULL, 1); Node *cond; expect(SWITCH); expect ('('); cond = eval(expr()); expect (')'); /* TODO: check integer type */ emitjump(lcond, NULL); stmt(lbreak, lcont, &lcase); emitlabel(lcond); emitswitch(lcase.nr, cond); for (p = lcase.head; p; p = p->next) emitcase(p->label, p->expr); if (lcase.deflabel) emitdefault(lcase.deflabel); emitlabel(lbreak); /* TODO: free memory */ } static void Case(Symbol *lbreak, Symbol *lcont, Caselist *lswitch) { Node *np; Symbol *lcase = label(NULL, 1); struct scase *pcase; if (!lswitch) error("case label not within a switch statement"); expect(CASE); if (yytoken == ':') error("expected expression before ':'"); np = eval(expr()); /* TODO: check integer type */ expect(':'); emitlabel(lcase); pcase = xmalloc(sizeof(*pcase)); pcase->expr = np; pcase->label = lcase; pcase->next = lswitch->head; lswitch->head = pcase; ++lswitch->nr; } static void Default(Symbol *lbreak, Symbol *lcont, Caselist *lswitch) { Symbol *ldefault = label(NULL, 1); expect(DEFAULT); expect(':'); emitlabel(ldefault); lswitch->deflabel = ldefault; } static void If(Symbol *lbreak, Symbol *lcont, Caselist *lswitch) { Symbol *end, *lelse = label(NULL, 1); Node *np; expect(IF); np = condition(); NEGATE(np, 1); emitjump(lelse, np); stmt(lbreak, lcont, lswitch); if (accept(ELSE)) { end = label(NULL, 1); emitjump(end, NULL); emitlabel(lelse); stmt(lbreak, lcont, lswitch); emitlabel(end); } else { emitlabel(lelse); } } void compound(Symbol *lbreak, Symbol *lcont, Caselist *lswitch) { expect('{'); for (;;) { switch (yytoken) { case '}': next(); return; case TYPE: case SCLASS: case TQUALIFIER: decl(); break; default: stmt(lbreak, lcont, lswitch); } } } static void stmt(Symbol *lbreak, Symbol *lcont, Caselist *lswitch) { repeat: switch (yytoken) { case '{': context(compound, lbreak, lcont, lswitch); break; case RETURN: Return(); break; case WHILE: While(lswitch); break; case FOR: For(lswitch); break; case DO: Dowhile(lswitch); break; case IF: If(lbreak, lcont, lswitch); break; case BREAK: Break(lbreak); break; case CONTINUE: Continue(lcont); break; case GOTO: Goto(); break; case SWITCH: Switch(lcont); break; case CASE: Case(lbreak, lcont, lswitch); break; case DEFAULT: Default(lbreak, lcont, lswitch); break; case IDEN: if (ahead() == ':') { Label(); goto repeat; } default: stmtexp(); break; } }