ref: 55f902cc327c45e15fba898fcc004f01a2f187e3
dir: /6/gen.c/
#include <stdlib.h> #include <stdio.h> #include <stdarg.h> #include <inttypes.h> #include <ctype.h> #include <string.h> #include <assert.h> #include <sys/types.h> #include <sys/stat.h> #include <fcntl.h> #include <unistd.h> #include "parse.h" #include "mi.h" #include "asm.h" #include "../config.h" void fillglobls(Stab *st, Htab *globls) { size_t i, j, nk, nns; void **k, **ns; Stab *stab; Node *s; k = htkeys(st->dcl, &nk); for (i = 0; i < nk; i++) { s = htget(st->dcl, k[i]); if (isconstfn(s)) s->decl.type = codetype(s->decl.type); htput(globls, s, asmname(s)); } free(k); ns = htkeys(file->file.ns, &nns); for (j = 0; j < nns; j++) { stab = htget(file->file.ns, ns[j]); k = htkeys(stab->dcl, &nk); for (i = 0; i < nk; i++) { s = htget(stab->dcl, k[i]); htput(globls, s, asmname(s)); } free(k); } free(ns); } Type *codetype(Type *ft) { ft = tybase(ft); if (ft->type == Tycode) return ft; assert(ft->type == Tyfunc); ft = tydup(ft); ft->type = Tycode; return ft; } Type *closuretype(Type *ft) { ft = tybase(ft); if (ft->type == Tyfunc) return ft; assert(ft->type == Tycode); ft = tydup(ft); ft->type = Tyfunc; return ft; } static int islocal(Node *dcl) { if (dcl->decl.vis != Visintern) return 0; if (dcl->decl.isimport || dcl->decl.isextern) return 0; return 1; } static int nextlbl; char *genlocallblstr(char *buf, size_t sz) { if (asmsyntax == Plan9) bprintf(buf, 128, ".L%d<>", nextlbl++); else bprintf(buf, 128, ".L%d", nextlbl++); return buf; } char *genlblstr(char *buf, size_t sz) { bprintf(buf, 128, ".L%d", nextlbl++); return buf; } int isconstfn(Node *n) { Node *d; Type *t; if (n->type == Nexpr) { if (exprop(n) != Ovar) return 0; d = decls[n->expr.did]; } else { d = n; } t = tybase(decltype(d)); if (d && d->decl.isconst && d->decl.isglobl) return t->type == Tyfunc || t->type == Tycode; return 0; } /* * For x86, the assembly names are generated as follows: * local symbols: .name * un-namespaced symbols: <symprefix>name * namespaced symbols: <symprefix>namespace$name * local symbols on plan9 have the file-unique suffix '<>' appended */ char *asmname(Node *dcl) { char buf[1024]; char *vis, *pf, *ns, *name, *sep; Node *n; n = dcl->decl.name; pf = Symprefix; ns = n->name.ns; name = n->name.name; vis = ""; sep = ""; if (asmsyntax == Plan9) if (islocal(dcl)) vis = "<>"; if (!ns || !ns[0]) ns = ""; else sep = "$"; if (name[0] == '.') pf = ""; bprintf(buf, sizeof buf, "%s%s%s%s%s", pf, ns, sep, name, vis); return strdup(buf); } char *tydescid(char *buf, size_t bufsz, Type *ty) { char *sep, *ns; sep = ""; ns = ""; if (ty->type == Tyname) { if (ty->name->name.ns) { ns = ty->name->name.ns; sep = "$"; } if (ty->vis == Visexport || ty->isimport) bprintf(buf, bufsz, "_tydesc$%s%s%s", ns, sep, ty->name->name.name); else bprintf(buf, bufsz, "_tydesc$%s%s%s$%d", ns, sep, ty->name->name.name, ty->tid); } else { if (file->file.globls->name) { ns = file->file.globls->name; sep = "$"; } bprintf(buf, bufsz, "_tydesc%s%s$%d",sep, ns, ty->tid); } return buf; } void gen(Node *file, char *out) { switch (asmsyntax) { case Plan9: genp9(file, out); break; case Gnugas: gengas(file, out); break; default: die("unknown target"); break; } }