ref: 3f65d0e80c706fb06d946512d69421209071694e
dir: /parse/use.c/
#include <stdlib.h> #include <stdio.h> #include <stdint.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" static void wrtype(FILE *fd, Type *val); static void rdtype(FILE *fd, Type **dest); static void wrstab(FILE *fd, Stab *val); static Stab *rdstab(FILE *fd); static void wrsym(FILE *fd, Node *val); static Node *rdsym(FILE *fd); static void pickle(Node *n, FILE *fd); static Node *unpickle(FILE *fd); /* type fixup list */ static Htab *dedup; /* map from name -> type, contains all Tynames loaded ever */ static Htab *tidmap; /* map from tid -> type */ static Type ***typefixdest; /* list of types we need to replace */ static size_t ntypefixdest; /* size of replacement list */ static intptr_t *typefixid; /* list of types we need to replace */ static size_t ntypefixid; /* size of replacement list */ #define Builtinmask (1 << 30) /* Outputs a symbol table to file in a way that can be * read back usefully. Only writes declarations, types * and sub-namespaces. Captured variables are ommitted. */ static void wrstab(FILE *fd, Stab *val) { size_t n, i; void **keys; pickle(val->name, fd); /* write decls */ keys = htkeys(val->dcl, &n); wrint(fd, n); for (i = 0; i < n; i++) wrsym(fd, getdcl(val, keys[i])); free(keys); /* write types */ keys = htkeys(val->ty, &n); wrint(fd, n); for (i = 0; i < n; i++) { pickle(keys[i], fd); /* name */ wrtype(fd, gettype(val, keys[i])); /* type */ } free(keys); /* write stabs */ keys = htkeys(val->ns, &n); wrint(fd, n); for (i = 0; i < n; i++) wrstab(fd, getns(val, keys[i])); free(keys); } /* Reads a symbol table from file. The converse * of wrstab. */ static Stab *rdstab(FILE *fd) { Stab *st; Type *ty; Node *nm; int n; int i; /* read dcls */ st = mkstab(); st->name = unpickle(fd); n = rdint(fd); for (i = 0; i < n; i++) putdcl(st, rdsym(fd)); /* read types */ n = rdint(fd); for (i = 0; i < n; i++) { nm = unpickle(fd); rdtype(fd, &ty); puttype(st, nm, ty); } /* read stabs */ n = rdint(fd); for (i = 0; i < n; i++) putns(st, rdstab(fd)); return st; } static void wrucon(FILE *fd, Ucon *uc) { wrint(fd, uc->line); wrint(fd, uc->id); wrbool(fd, uc->synth); pickle(uc->name, fd); wrbool(fd, uc->etype != NULL); if (uc->etype) wrtype(fd, uc->etype); } static Ucon *rducon(FILE *fd, Type *ut) { Type *et; Node *name; Ucon *uc; size_t id; int line; int synth; et = NULL; line = rdint(fd); id = rdint(fd); synth = rdbool(fd); name = unpickle(fd); uc = mkucon(line, name, ut, et); if (rdbool(fd)) rdtype(fd, &uc->etype); uc->id = id; uc->synth = synth; return uc; } /* Writes the name and type of a variable, * but only writes its intializer for things * we want to inline cross-file (currently, * the only cross-file inline is generics) */ static void wrsym(FILE *fd, Node *val) { /* sym */ wrint(fd, val->line); pickle(val->decl.name, fd); wrtype(fd, val->decl.type); /* symflags */ wrint(fd, val->decl.vis); wrbool(fd, val->decl.isconst); wrbool(fd, val->decl.isgeneric); wrbool(fd, val->decl.isextern); if (val->decl.isgeneric) pickle(val->decl.init, fd); } static Node *rdsym(FILE *fd) { int line; Node *name; Node *n; line = rdint(fd); name = unpickle(fd); n = mkdecl(line, name, NULL); rdtype(fd, &n->decl.type); if (rdint(fd) == Vishidden) n->decl.ishidden = 1; n->decl.isconst = rdbool(fd); n->decl.isgeneric = rdbool(fd); n->decl.isextern = rdbool(fd); if (n->decl.isgeneric) n->decl.init = unpickle(fd); return n; } /* Writes types to a file. Errors on * internal only types like Tyvar that * will not be meaningful in another file*/ static void typickle(FILE *fd, Type *ty) { size_t i; if (!ty) { die("trying to pickle null type\n"); return; } wrbyte(fd, ty->type); wrbyte(fd, ty->vis); /* tid is generated; don't write */ /* FIXME: since we only support hardcoded cstrs, we just write * out the set of them. we should write out the cstr list as * well */ if (!ty->cstrs) { wrint(fd, 0); } else { wrint(fd, bscount(ty->cstrs)); for (i = 0; bsiter(ty->cstrs, &i); i++) wrint(fd, i); } wrint(fd, ty->nsub); switch (ty->type) { case Tyunres: pickle(ty->name, fd); break; case Typaram: wrstr(fd, ty->pname); break; case Tystruct: wrint(fd, ty->nmemb); for (i = 0; i < ty->nmemb; i++) pickle(ty->sdecls[i], fd); break; case Tyunion: wrint(fd, ty->nmemb); for (i = 0; i < ty->nmemb; i++) wrucon(fd, ty->udecls[i]); break; case Tyarray: wrtype(fd, ty->sub[0]); pickle(ty->asize, fd); break; case Tyslice: wrtype(fd, ty->sub[0]); break; case Tyvar: die("Attempting to pickle %s. This will not work.\n", tystr(ty)); break; case Tyname: pickle(ty->name, fd); wrbool(fd, ty->issynth); wrint(fd, ty->nparam); for (i = 0; i < ty->nparam; i++) wrtype(fd, ty->param[i]); wrint(fd, ty->narg); for (i = 0; i < ty->narg; i++) wrtype(fd, ty->arg[i]); wrtype(fd, ty->sub[0]); break; default: for (i = 0; i < ty->nsub; i++) wrtype(fd, ty->sub[i]); break; } } static void wrtype(FILE *fd, Type *ty) { if (ty->tid >= Builtinmask) die("Type id %d for %s too big", ty->tid, tystr(ty)); if (ty->vis == Visbuiltin) wrint(fd, ty->type | Builtinmask); else wrint(fd, ty->tid); } static void rdtype(FILE *fd, Type **dest) { intptr_t tid; tid = rdint(fd); if (tid & Builtinmask) { *dest = mktype(-1, tid & ~Builtinmask); } else { lappend(&typefixdest, &ntypefixdest, dest); lappend(&typefixid, &ntypefixid, (void*)tid); } } /* Writes types to a file. Errors on * internal only types like Tyvar that * will not be meaningful in another file */ static Type *tyunpickle(FILE *fd) { size_t i, n; size_t v; Type *ty; Ty t; t = rdbyte(fd); ty = mktype(-1, t); if (rdbyte(fd) == Vishidden) ty->ishidden = 1; /* tid is generated; don't write */ n = rdint(fd); if (n > 0) { ty->cstrs = mkbs(); for (i = 0; i < n; i++) { v = rdint(fd); setcstr(ty, cstrtab[v]); } } ty->nsub = rdint(fd); if (ty->nsub > 0) ty->sub = zalloc(ty->nsub * sizeof(Type*)); switch (ty->type) { case Tyunres: ty->name = unpickle(fd); break; case Typaram: ty->pname = rdstr(fd); break; case Tystruct: ty->nmemb = rdint(fd); ty->sdecls = zalloc(ty->nmemb * sizeof(Node*)); for (i = 0; i < ty->nmemb; i++) ty->sdecls[i] = unpickle(fd); break; case Tyunion: ty->nmemb = rdint(fd); ty->udecls = zalloc(ty->nmemb * sizeof(Node*)); for (i = 0; i < ty->nmemb; i++) ty->udecls[i] = rducon(fd, ty); break; case Tyarray: rdtype(fd, &ty->sub[0]); ty->asize = unpickle(fd); break; case Tyslice: rdtype(fd, &ty->sub[0]); break; case Tyname: ty->name = unpickle(fd); ty->issynth = rdbool(fd); ty->nparam = rdint(fd); ty->param = zalloc(ty->nparam * sizeof(Type *)); for (i = 0; i < ty->nparam; i++) rdtype(fd, &ty->param[i]); ty->narg = rdint(fd); ty->arg = zalloc(ty->narg * sizeof(Type *)); for (i = 0; i < ty->narg; i++) rdtype(fd, &ty->arg[i]); rdtype(fd, &ty->sub[0]); break; default: for (i = 0; i < ty->nsub; i++) rdtype(fd, &ty->sub[i]); break; } return ty; } /* Pickles a node to a file. The format * is more or less equivalen to to * simplest serialization of the * in-memory representation. Minimal * checking is done, so a bad type can * crash the compiler */ static void pickle(Node *n, FILE *fd) { size_t i; if (!n) { wrbyte(fd, Nnone); return; } wrbyte(fd, n->type); wrint(fd, n->line); switch (n->type) { case Nfile: wrstr(fd, n->file.name); wrint(fd, n->file.nuses); for (i = 0; i < n->file.nuses; i++) pickle(n->file.uses[i], fd); wrint(fd, n->file.nstmts); for (i = 0; i < n->file.nstmts; i++) pickle(n->file.stmts[i], fd); wrstab(fd, n->file.globls); wrstab(fd, n->file.exports); break; case Nexpr: wrbyte(fd, n->expr.op); wrtype(fd, n->expr.type); wrbool(fd, n->expr.isconst); pickle(n->expr.idx, fd); wrint(fd, n->expr.nargs); for (i = 0; i < n->expr.nargs; i++) pickle(n->expr.args[i], fd); break; case Nname: wrbool(fd, n->name.ns != NULL); if (n->name.ns) { wrstr(fd, n->name.ns); } wrstr(fd, n->name.name); break; case Nuse: wrbool(fd, n->use.islocal); wrstr(fd, n->use.name); break; case Nlit: wrbyte(fd, n->lit.littype); wrtype(fd, n->lit.type); wrint(fd, n->lit.nelt); switch (n->lit.littype) { case Lchr: wrint(fd, n->lit.chrval); break; case Lint: wrint(fd, n->lit.intval); break; case Lflt: wrflt(fd, n->lit.fltval); break; case Lstr: wrstr(fd, n->lit.strval); break; case Llbl: wrstr(fd, n->lit.lblval); break; case Lbool: wrbool(fd, n->lit.boolval); break; case Lfunc: pickle(n->lit.fnval, fd); break; } break; case Nloopstmt: pickle(n->loopstmt.init, fd); pickle(n->loopstmt.cond, fd); pickle(n->loopstmt.step, fd); pickle(n->loopstmt.body, fd); break; case Nmatchstmt: pickle(n->matchstmt.val, fd); wrint(fd, n->matchstmt.nmatches); for (i = 0; i < n->matchstmt.nmatches; i++) pickle(n->matchstmt.matches[i], fd); break; case Nmatch: pickle(n->match.pat, fd); pickle(n->match.block, fd); break; case Nifstmt: pickle(n->ifstmt.cond, fd); pickle(n->ifstmt.iftrue, fd); pickle(n->ifstmt.iffalse, fd); break; case Nblock: wrstab(fd, n->block.scope); wrint(fd, n->block.nstmts); for (i = 0; i < n->block.nstmts; i++) pickle(n->block.stmts[i], fd); break; case Ndecl: /* sym */ pickle(n->decl.name, fd); wrtype(fd, n->decl.type); /* symflags */ wrint(fd, n->decl.isconst); wrint(fd, n->decl.isgeneric); wrint(fd, n->decl.isextern); /* init */ pickle(n->decl.init, fd); break; case Nfunc: wrtype(fd, n->func.type); wrstab(fd, n->func.scope); wrint(fd, n->func.nargs); for (i = 0; i < n->func.nargs; i++) pickle(n->func.args[i], fd); pickle(n->func.body, fd); break; case Nnone: die("Nnone should not be seen as node type!"); break; } } /* Unpickles a node from a file. Minimal checking * is done. Specifically, no checks are done for * sane arities, a bad file can crash the compiler */ static Node *unpickle(FILE *fd) { size_t i; Ntype type; Node *n; type = rdbyte(fd); if (type == Nnone) return NULL; n = mknode(-1, type); n->line = rdint(fd); switch (n->type) { case Nfile: n->file.name = rdstr(fd); n->file.nuses = rdint(fd); n->file.uses = zalloc(sizeof(Node*)*n->file.nuses); for (i = 0; i < n->file.nuses; i++) n->file.uses[i] = unpickle(fd); n->file.nstmts = rdint(fd); n->file.stmts = zalloc(sizeof(Node*)*n->file.nstmts); for (i = 0; i < n->file.nstmts; i++) n->file.stmts[i] = unpickle(fd); n->file.globls = rdstab(fd); n->file.exports = rdstab(fd); break; case Nexpr: n->expr.op = rdbyte(fd); rdtype(fd, &n->expr.type); n->expr.isconst = rdbool(fd); n->expr.idx = unpickle(fd); n->expr.nargs = rdint(fd); n->expr.args = zalloc(sizeof(Node *)*n->expr.nargs); for (i = 0; i < n->expr.nargs; i++) n->expr.args[i] = unpickle(fd); break; case Nname: if (rdbool(fd)) n->name.ns = rdstr(fd); n->name.name = rdstr(fd); break; case Nuse: n->use.islocal = rdbool(fd); n->use.name = rdstr(fd); break; case Nlit: n->lit.littype = rdbyte(fd); rdtype(fd, &n->lit.type); n->lit.nelt = rdint(fd); switch (n->lit.littype) { case Lchr: n->lit.chrval = rdint(fd); break; case Lint: n->lit.intval = rdint(fd); break; case Lflt: n->lit.fltval = rdflt(fd); break; case Lstr: n->lit.strval = rdstr(fd); break; case Llbl: n->lit.lblval = rdstr(fd); break; case Lbool: n->lit.boolval = rdbool(fd); break; case Lfunc: n->lit.fnval = unpickle(fd); break; } break; case Nloopstmt: n->loopstmt.init = unpickle(fd); n->loopstmt.cond = unpickle(fd); n->loopstmt.step = unpickle(fd); n->loopstmt.body = unpickle(fd); break; case Nmatchstmt: n->matchstmt.val = unpickle(fd); n->matchstmt.nmatches = rdint(fd); n->matchstmt.matches = zalloc(sizeof(Node *)*n->matchstmt.nmatches); for (i = 0; i < n->matchstmt.nmatches; i++) n->matchstmt.matches[i] = unpickle(fd); break; case Nmatch: n->match.pat = unpickle(fd); n->match.block = unpickle(fd); break; case Nifstmt: n->ifstmt.cond = unpickle(fd); n->ifstmt.iftrue = unpickle(fd); n->ifstmt.iffalse = unpickle(fd); break; case Nblock: n->block.scope = rdstab(fd); n->block.nstmts = rdint(fd); n->block.stmts = zalloc(sizeof(Node *)*n->block.nstmts); n->block.scope->super = curstab(); pushstab(n->func.scope->super); for (i = 0; i < n->block.nstmts; i++) n->block.stmts[i] = unpickle(fd); popstab(); break; case Ndecl: n->decl.did = ndecls; /* unique within file */ /* sym */ n->decl.name = unpickle(fd); rdtype(fd, &n->decl.type); /* symflags */ n->decl.isconst = rdint(fd); n->decl.isgeneric = rdint(fd); n->decl.isextern = rdint(fd); /* init */ n->decl.init = unpickle(fd); lappend(&decls, &ndecls, n); break; case Nfunc: rdtype(fd, &n->func.type); n->func.scope = rdstab(fd); n->func.nargs = rdint(fd); n->func.args = zalloc(sizeof(Node *)*n->func.nargs); n->func.scope->super = curstab(); pushstab(n->func.scope->super); for (i = 0; i < n->func.nargs; i++) n->func.args[i] = unpickle(fd); n->func.body = unpickle(fd); popstab(); break; case Nnone: die("Nnone should not be seen as node type!"); break; } return n; } static Stab *findstab(Stab *st, char *pkg) { Node *n; Stab *s; if (!pkg) { if (!st->name) return st; else return NULL; } n = mkname(-1, pkg); if (getns(st, n)) { s = getns(st, n); } else { s = mkstab(); s->name = n; putns(st, s); } return s; } static void fixmappings(Stab *st) { size_t i; Type *t, *old; /* * merge duplicate definitions. * This allows us to compare named types by id, instead * of doing a deep walk through the type. This ability is * depended on when we do type inference. */ for (i = 0; i < ntypefixdest; i++) { t = htget(tidmap, (void*)typefixid[i]); if (t->type == Tyname && !t->issynth) { old = htget(dedup, t->name); if (old != t) if (t != old) t = old; } *typefixdest[i] = t; if (!*typefixdest[i]) die("Couldn't find type %d\n", (int)typefixid[i]); } /* check for duplicate type definitions */ for (i = 0; i < ntypefixdest; i++) { t = htget(tidmap, (void*)typefixid[i]); if (t->type != Tyname || t->issynth) continue; old = htget(dedup, t->name); if (old && !tyeq(t, old)) fatal(-1, "Duplicate definition of type %s", tystr(old)); } lfree(&typefixdest, &ntypefixdest); lfree(&typefixid, &ntypefixid); } /* Usefile format: * U<pkgname> * T<pickled-type> * D<picled-decl> * G<pickled-decl><pickled-initializer> */ int loaduse(FILE *f, Stab *st) { intptr_t tid; size_t i; char *pkg; Node *dcl; Stab *s; Type *t; int c; pushstab(file->file.globls); if (!dedup) dedup = mkht(namehash, nameeq); if (fgetc(f) != 'U') return 0; pkg = rdstr(f); /* if the package names match up, or the usefile has no declared * package, then we simply add to the current stab. Otherwise, * we add a new stab under the current one */ if (st->name) { if (pkg && !strcmp(pkg, namestr(st->name))) { s = st; } else { s = findstab(st, pkg); } } else { if (pkg) { s = findstab(st, pkg); } else { s = st; } } tidmap = mkht(ptrhash, ptreq); while ((c = fgetc(f)) != EOF) { switch(c) { case 'G': case 'D': dcl = rdsym(f); putdcl(s, dcl); break; case 'T': tid = rdint(f); t = tyunpickle(f); htput(tidmap, (void*)tid, t); /* fix up types */ if (t->type == Tyname) { if (t->issynth) break; if (!gettype(st, t->name) && !t->ishidden) puttype(s, t->name, t); if (!hthas(dedup, t->name)) htput(dedup, t->name, t); } else if (t->type == Tyunion) { for (i = 0; i < t->nmemb; i++) if (!getucon(s, t->udecls[i]->name) && !t->udecls[i]->synth) putucon(s, t->udecls[i]); } break; case EOF: break; } } fixmappings(s); htfree(tidmap); popstab(); return 1; } void readuse(Node *use, Stab *st) { size_t i; FILE *fd; char *p, *q; /* local (quoted) uses are always relative to the cwd */ fd = NULL; if (use->use.islocal) { fd = fopen(use->use.name, "r"); /* nonlocal (barename) uses are always searched on the include path */ } else { for (i = 0; i < nincpaths; i++) { p = strjoin(incpaths[i], "/"); q = strjoin(p, use->use.name); fd = fopen(q, "r"); if (fd) { free(p); free(q); break; } } } if (!fd) fatal(use->line, "Could not open %s", use->use.name); if (!loaduse(fd, st)) die("Could not load usefile %s", use->use.name); } /* Usefile format: * U<pkgname> * T<pickled-type> * D<picled-decl> * G<pickled-decl><pickled-initializer> * Z */ void writeuse(FILE *f, Node *file) { Stab *st; void **k; Node *s; size_t i, n; st = file->file.exports; wrbyte(f, 'U'); if (st->name) wrstr(f, namestr(st->name)); else wrstr(f, NULL); for (i = 0; i < ntypes; i++) { if (types[i]->vis == Visexport || types[i]->vis == Vishidden) { wrbyte(f, 'T'); wrint(f, types[i]->tid); typickle(f, types[i]); } } k = htkeys(st->dcl, &n); for (i = 0; i < n; i++) { s = getdcl(st, k[i]); if (s && s->decl.isgeneric) wrbyte(f, 'G'); else wrbyte(f, 'D'); wrsym(f, s); } free(k); }