ref: 2a5f32b8f028d8cfbc221bbccd6624f323de6825
dir: /cc1/code.c/
/* See LICENSE file for copyright and license details. */ static char sccsid[] = "@(#) ./cc1/code.c"; #include <ctype.h> #include <stdio.h> #include <stdlib.h> #include <stdarg.h> #include "../inc/cc.h" #include "cc1.h" static void emitbin(unsigned, void *), emitcast(unsigned, void *), emitsym(unsigned, void *), emitexp(unsigned, void *), emitsymid(unsigned, void *), emittext(unsigned, void *), emitfun(unsigned, void *), emitdcl(unsigned, void *), emitinit(unsigned, void *), emittype(unsigned, void *), emitbuilt(unsigned, void *); char *optxt[] = { [OADD] = "+", [OSUB] = "-", [OMUL] = "*", [OINC] = ":i", [ODEC] = ":d", [OPTR] = "@", [OMOD] = "%", [ODIV] = "/", [OSHL] = "l", [OSHR] = "r", [OLT] = "<", [OGT] = ">", [OGE] = "]", [OLE] = "[", [OEQ] = "=", [ONE] = "!", [OBAND] = "&", [OBXOR] = "^", [OBOR] = "|", [OASSIGN] = ":", [OA_MUL] = ":*", [OA_DIV] = ":/", [OA_MOD] = ":%", [OA_ADD] = ":+", [OA_SUB] = ":-", [OA_SHL] = ":l", [OA_SHR] = ":r", [OA_AND] = ":&", [OA_XOR] = ":^", [OA_OR] = ":|", [OADDR] = "'", [OSNEG] = "_", [ONEG] = "n", [OCPL] = "~", [OAND] = "a", [OOR] = "o", [OASK] = "?", [OCOMMA] = ",", [OLABEL] = "L%d\n", [ODEFAULT] = "\tf\tL%d\n", [OBSWITCH] = "\ts", [OESWITCH] = "\tt\tL%d\n", [OCASE] = "\tv\tL%d", [OJUMP] = "\tj\tL%d\n", [OBRANCH] = "\ty\tL%d", [OEFUN] = "}\n", [OELOOP] = "\tb\n", [OBLOOP] = "\te\n", [ORET] = "\th", [OPAR] = "p", [OCALL] = "c", [OCALLE] = "z", [OFIELD] = "." }; void (*opcode[])(unsigned, void *) = { [OADD] = emitbin, [OSUB] = emitbin, [OMUL] = emitbin, [OINC] = emitbin, [ODEC] = emitbin, [OPTR] = emitbin, [OMOD] = emitbin, [ODIV] = emitbin, [OSHL] = emitbin, [OSHR] = emitbin, [OLT] = emitbin, [OGT] = emitbin, [OGE] = emitbin, [OLE] = emitbin, [OEQ] = emitbin, [ONE] = emitbin, [OBAND] = emitbin, [OBXOR] = emitbin, [OBOR] = emitbin, [OASSIGN] = emitbin, [OA_MUL] = emitbin, [OA_DIV] = emitbin, [OA_MOD] = emitbin, [OA_ADD] = emitbin, [OA_SUB] = emitbin, [OA_SHL] = emitbin, [OA_SHR] = emitbin, [OA_AND] = emitbin, [OA_XOR] = emitbin, [OA_OR] = emitbin, [OADDR] = emitbin, [OSNEG] = emitbin, [ONEG] = emitbin, [OCPL] = emitbin, [OAND] = emitbin, [OOR] = emitbin, [OCOMMA] = emitbin, [OCAST] = emitcast, [OSYM] = emitsym, [OASK] = emitbin, [OCOLON] = emitbin, [OFIELD]= emitbin, [OEXPR] = emitexp, [OLABEL] = emitsymid, [ODEFAULT] = emitsymid, [OCASE] = emitsymid, [OJUMP] = emitsymid, [OBRANCH] = emitsymid, [OEFUN] = emittext, [OELOOP] = emittext, [OBLOOP] = emittext, [OFUN] = emitfun, [ORET] = emittext, [ODECL] = emitdcl, [OBSWITCH] = emittext, [OESWITCH] = emitsymid, [OPAR] = emitbin, [OCALL] = emitbin, [OCALLE] = emitbin, [OINIT] = emitinit, [OBUILTIN] = emitbuilt, [OTYP] = emittype, }; static FILE *outfp; void icode(void) { outfp = stdout; } void freetree(Node *np) { if (!np) return; freetree(np->left); freetree(np->right); free(np); } static void emitnode(Node *np) { if (np) (*opcode[np->op])(np->op, np); } void prtree(Node *np) { outfp = stderr; fputs("DBG prtree", outfp); emitnode(np); putc('\n', outfp); outfp = stdout; } void emit(unsigned op, void *arg) { extern int failure; if (failure || onlycpp || onlyheader) return; (*opcode[op])(op, arg); } static void emitvar(Symbol *sym) { char c; short flags = sym->flags; if (flags & SLOCAL) c = 'T'; else if (flags & SPRIVATE) c = 'Y'; else if (flags & SGLOBAL) c = 'G'; else if (flags & SREGISTER) c = 'R'; else if (flags & SFIELD) c = 'M'; else if (flags & SEXTERN) c = 'X'; else c = 'A'; fprintf(outfp, "%c%u", c, sym->id); } static void emitconst(Node *np) { Symbol *sym = np->sym; Type *tp = np->type; TUINT u; switch (tp->op) { case PTR: case INT: case ENUM: u = (tp->prop & TSIGNED) ? (TUINT) sym->u.i : sym->u.u; fprintf(outfp, "#%c%llX", np->type->letter, (long long) u & ones(tp->size)); break; default: abort(); } } static void emitsym(unsigned op, void *arg) { Node *np = arg; if ((np->sym->flags & SINITLST) == 0) { /* * When we have a compound literal we are going * to call to emitnode for every element of it, * and it means that we will have two '\t' * for the first element */ putc('\t', outfp); } (np->flags & NCONST) ? emitconst(np) : emitvar(np->sym); } static void emitletter(Type *tp) { int letter; letter = (tp->prop&TELLIPSIS) ? 'E' : tp->letter; putc(letter, outfp); switch (tp->op) { case ARY: case STRUCT: case UNION: fprintf(outfp, "%u", tp->id); } } static void emittype(unsigned op, void *arg) { TINT n; Symbol **sp; char *tag; Type *tp = arg; if (!(tp->prop & TDEFINED)) return; switch (tp->op) { case ARY: emitletter(tp); putc('\t', outfp); emitletter(tp->type); fprintf(outfp, "\t#%c%llX\n", sizettype->letter, (long long) tp->n.elem); return; case UNION: case STRUCT: emitletter(tp); tag = tp->tag->name; fprintf(outfp, "\t\"%s\t#%c%lX\t#%c%X\n", (tag) ? tag : "", sizettype->letter, tp->size, sizettype->letter, tp->align); n = tp->n.elem; for (sp = tp->p.fields; n-- > 0; ++sp) emit(ODECL, *sp); break; case PTR: case FTN: case ENUM: return; default: abort(); } } static void emitstring(Symbol *sym, Type *tp) { char *bp, *s, *lim; int n; bp = sym->u.s; lim = &sym->u.s[tp->n.elem]; while (bp < lim) { s = bp; while (bp < lim && isprint(*bp)) ++bp; if ((n = bp - s) > 1) fprintf(outfp, "\t#\"%.*s\n", n, s); else bp = s; if (bp == lim) break; do { fprintf(outfp, "\t#%c%02X\n", chartype->letter, (*bp++) & 0xFF); } while (bp < lim && !isprint(*bp)); } } static void emitdesig(Node *np, Type *tp) { Symbol *sym; size_t n; Node *aux; Type *p; if (!np) { sym = NULL; } else { if (!np->sym) goto emit_expression; sym = np->sym; if (sym->flags & SSTRING) { emitstring(sym, tp); return; } if ((sym->flags & SINITLST) == 0) goto emit_expression; } switch (tp->op) { case PTR: case INT: case ENUM: aux = (sym) ? *sym->u.init : convert(constnode(zero), tp, 0); emitexp(OEXPR, aux); break; /* TODO: case UNION: */ case STRUCT: case ARY: for (n = 0; n < tp->n.elem; ++n) { aux = (sym) ? sym->u.init[n] : NULL; p = (tp->op == ARY) ? tp->type : tp->p.fields[n]->type; emitdesig(aux, p); } break; default: abort(); } if (sym) { free(sym->u.init); sym->u.init = NULL; } freetree(np); return; emit_expression: emitexp(OEXPR, np); } static void emitinit(unsigned op, void *arg) { Node *np = arg; fputs("\t(\n", outfp); emitdesig(np, np->type); fputs(")\n", outfp); } static void emitdcl(unsigned op, void *arg) { Symbol *sym = arg; if (sym->flags & SEMITTED) return; emitvar(sym); putc('\t', outfp); if (sym->type->op == FTN) { emitletter(sym->type->type); putc('\t', outfp); } emitletter(sym->type); fprintf(outfp, "\t\"%s", (sym->name) ? sym->name : ""); if (sym->flags & SFIELD) fprintf(outfp, "\t#%c%llX", sizettype->letter, sym->u.i); sym->flags |= SEMITTED; if ((sym->flags & SHASINIT) == 0) putc('\n', outfp); } static void emitcast(unsigned op, void *arg) { Node *np = arg, *lp = np->left; emitnode(lp); if (np->type != voidtype) fprintf(outfp, "\tg%c", np->type->letter); } static void emitbin(unsigned op, void *arg) { Node *np = arg; char *s; emitnode(np->left); emitnode(np->right); if ((s = optxt[op]) != NULL) { /* do not print in OCOLON case */ fprintf(outfp, "\t%s", s); emitletter(np->type); } } static void emitbuilt(unsigned op, void *arg) { Node *np = arg; emitnode(np->left); emitnode(np->right); fprintf(outfp, "\t\"%s\tm", np->sym->name); emitletter(np->type); } static void emitexp(unsigned op, void *arg) { Node *np = arg; emitnode(np); putc('\n', outfp); freetree(np); } static void emitfun(unsigned op, void *arg) { Symbol *sym = arg, **sp; emitdcl(op, arg); fputs("{\n", outfp); for (sp = sym->u.pars; sp && *sp; ++sp) emit(ODECL, *sp); fputs("\\\n", outfp); } static void emittext(unsigned op, void *arg) { fputs(optxt[op], outfp); } static void emitsymid(unsigned op, void *arg) { Symbol *sym = arg; fprintf(outfp, optxt[op], sym->id); } Node * node(unsigned op, Type *tp, Node *lp, Node *rp) { Node *np; np = xmalloc(sizeof(*np)); np->op = op; np->type = tp; np->sym = NULL; np->flags = 0; np->left = lp; np->right = rp; if (lp) np->flags |= lp->flags & NEFFECT; if (rp) np->flags |= rp->flags & NEFFECT; return np; } Node * varnode(Symbol *sym) { Node *np; Type *tp = sym->type; np = node(OSYM, sym->type, NULL, NULL); np->type = sym->type; np->flags = (tp->op != FTN && tp->op != ARY) ? NLVAL : 0; np->sym = sym; return np; } Node * constnode(Symbol *sym) { Node *np; np = node(OSYM, sym->type, NULL, NULL); np->type = sym->type; np->flags = NCONST; np->sym = sym; return np; } Node * sizeofnode(Type *tp) { Symbol *sym; sym = newsym(NS_IDEN, NULL); sym->type = sizettype; sym->u.i = tp->size; return constnode(sym); }