ref: 1b52ea9d5660f5b21a465f59448941212b2f921e
dir: /6/typeinfo.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 "util.h" #include "parse.h" #include "mi.h" #include "asm.h" #include "../config.h" #define Tdindirect 0x80 Blob *tydescsub(Type *ty); size_t blobsz(Blob *b) { size_t n; size_t i; switch (b->type) { case Bti8: return 1; break; case Bti16: return 2; break; case Bti32: return 4; break; case Bti64: return 8; break; case Btref: return 8; break; case Btbytes: return b->bytes.len; break; case Btimin: if (b->ival >= 1ULL << 56) die("packed int too big"); for (i = 1; i < 8; i++) if (b->ival < 1ULL << (7*i)) return i; die("impossible blob size"); break; case Btseq: n = 0; for (i = 0; i < b->seq.nsub; i++) n += blobsz(b->seq.sub[i]); return n; break; default: die("unknown blob type"); } return 0; } void namevec(Blob ***sub, size_t *nsub, Node *n) { char *buf; size_t len; if (n->name.ns) { len = strlen(n->name.name) + strlen(n->name.ns) + 1; buf = xalloc(len + 1); bprintf(buf, len + 1, "%s.%s", n->name.ns, n->name.name); } else { len = strlen(n->name.name); buf = xalloc(len + 1); bprintf(buf, len + 1, "%s", n->name.name); } lappend(sub, nsub, mkblobi(Btimin, len)); lappend(sub, nsub, mkblobbytes(buf, len)); } static void structmemb(Blob ***sub, size_t *nsub, Node *sdecl) { Blob *b; namevec(sub, nsub, sdecl->decl.name); b = tydescsub(sdecl->decl.type); lappend(sub, nsub, b); } static void unionmemb(Blob ***sub, size_t *nsub, Ucon *ucon) { namevec(sub, nsub, ucon->name); if (ucon->etype) { lappend(sub, nsub, tydescsub(ucon->etype)); } else { lappend(sub, nsub, mkblobi(Btimin, 1)); lappend(sub, nsub, mkblobi(Bti8, Tybad)); } } static void encodetypeinfo(Blob ***sub, size_t *nsub, Type *t) { lappend(sub, nsub, mkblobi(Btimin, tysize(t))); lappend(sub, nsub, mkblobi(Btimin, tyalign(t))); } Blob * tydescsub(Type *ty) { Blob **sub, *sz, *bt, *b; size_t i, nsub; char buf[512]; uint8_t tt; Node *len; int isextern; sub = NULL; nsub = 0; /* names are pulled out of line */ tt = ty->type; /* tyvars can get tagged, but aren't desired */ if (ty->type == Tyvar) return NULL; ty = tydedup(ty); if (ty->type == Tyname) tt |= Tdindirect; sz = mkblobi(Btimin, 0); bt = mkblobi(Bti8, tt); lappend(&sub, &nsub, bt); switch (ty->type) { case Ntypes: case Tyvar: case Tybad: case Typaram: case Tygeneric: case Tycode: case Tyunres: die("invalid type in tydesc"); break; /* atomic types -- nothing else to do */ case Tyvoid: case Tychar: case Tybool: case Tyint8: case Tyint16: case Tyint: case Tyint32: case Tyint64: case Tybyte: case Tyuint8: case Tyuint16: case Tyuint: case Tyuint32: case Tyuint64: case Tyflt32: case Tyflt64: case Tyvalist: break; case Typtr: lappend(&sub, &nsub, tydescsub(ty->sub[0])); break; case Tyslice: lappend(&sub, &nsub, tydescsub(ty->sub[0])); break; case Tyarray: encodetypeinfo(&sub, &nsub, ty); ty->asize = fold(ty->asize, 1); len = ty->asize; if (len) { assert(len->type == Nexpr); len = len->expr.args[0]; assert(len->type == Nlit && len->lit.littype == Lint); lappend(&sub, &nsub, mkblobi(Btimin, len->lit.intval)); } else { lappend(&sub, &nsub, mkblobi(Btimin, 0)); } lappend(&sub, &nsub, tydescsub(ty->sub[0])); break; case Tyfunc: lappend(&sub, &nsub, mkblobi(Btimin, ty->nsub)); for (i = 0; i < ty->nsub; i++) lappend(&sub, &nsub, tydescsub(ty->sub[i])); break; case Tytuple: encodetypeinfo(&sub, &nsub, ty); lappend(&sub, &nsub, mkblobi(Btimin, ty->nsub)); for (i = 0; i < ty->nsub; i++) lappend(&sub, &nsub, tydescsub(ty->sub[i])); break; case Tystruct: encodetypeinfo(&sub, &nsub, ty); lappend(&sub, &nsub, mkblobi(Btimin, ty->nmemb)); for (i = 0; i < ty->nmemb; i++) structmemb(&sub, &nsub, ty->sdecls[i]); break; case Tyunion: encodetypeinfo(&sub, &nsub, ty); lappend(&sub, &nsub, mkblobi(Btimin, ty->nmemb)); for (i = 0; i < ty->nmemb; i++) unionmemb(&sub, &nsub, ty->udecls[i]); break; case Tyname: i = bprintf(buf, sizeof buf, "%s", Symprefix); tydescid(buf + i, sizeof buf - i, ty); isextern = ty->isimport || ty->vis != Visintern; lappend(&sub, &nsub, mkblobref(buf, 0, isextern)); break; } b = mkblobseq(sub, nsub); sz->ival = blobsz(b); linsert(&b->seq.sub, &b->seq.nsub, 0, sz); return b; } Blob * namedesc(Type *ty) { Blob **sub; size_t nsub; sub = NULL; nsub = 0; lappend(&sub, &nsub, mkblobi(Bti8, Tyname)); namevec(&sub, &nsub, ty->name); lappend(&sub, &nsub, tydescsub(ty->sub[0])); return mkblobseq(sub, nsub); } Blob * tydescblob(Type *ty) { char buf[512]; Blob *b, *sz, *sub; if (ty->type == Tyname && hasparams(ty)) return NULL; if (ty->type == Tyname) { b = mkblobseq(NULL, 0); sz = mkblobi(Btimin, 0); sub = namedesc(ty); sz->ival = blobsz(sub); lappend(&b->seq.sub, &b->seq.nsub, sz); lappend(&b->seq.sub, &b->seq.nsub, sub); if (ty->vis != Visintern) b->isglobl = 1; } else { b = tydescsub(ty); } tydescid(buf, sizeof buf, ty); b->lbl = strdup(buf); return b; } size_t tysize(Type *t) { size_t sz; size_t i; sz = 0; if (!t) die("size of empty type => bailing."); switch (t->type) { case Tyvoid: return 0; case Tybool: case Tyint8: case Tybyte: case Tyuint8: return 1; case Tyint16: case Tyuint16: return 2; case Tyint: case Tyint32: case Tyuint: case Tyuint32: case Tychar: /* utf32 */ return 4; case Typtr: case Tyvalist: /* ptr to first element of valist */ return Ptrsz; case Tyint64: case Tyuint64: return 8; /*end integer types*/ case Tyflt32: return 4; case Tyflt64: return 8; case Tycode: return Ptrsz; case Tyfunc: return 2*Ptrsz; case Tyslice: return 2*Ptrsz; /* len; ptr */ case Tyname: return tysize(t->sub[0]); case Tyarray: if (!t->asize) return 0; t->asize = fold(t->asize, 1); assert(exprop(t->asize) == Olit); return t->asize->expr.args[0]->lit.intval * tysize(t->sub[0]); case Tytuple: for (i = 0; i < t->nsub; i++) { sz = alignto(sz, t->sub[i]); sz += tysize(t->sub[i]); } sz = alignto(sz, t); return sz; break; case Tystruct: for (i = 0; i < t->nmemb; i++) { sz = alignto(sz, decltype(t->sdecls[i])); sz += size(t->sdecls[i]); } sz = alignto(sz, t); return sz; break; case Tyunion: sz = Wordsz; for (i = 0; i < t->nmemb; i++) if (t->udecls[i]->etype) sz = max(sz, tysize(t->udecls[i]->etype) + Wordsz); return align(sz, tyalign(t)); break; case Tygeneric: case Tybad: case Tyvar: case Typaram: case Tyunres: case Ntypes: die("Type %s does not have size; why did it get down to here?", tystr(t)); break; } return -1; } size_t tyalign(Type *ty) { size_t align, i; align = 1; ty = tybase(ty); switch (ty->type) { case Tyarray: align = tyalign(ty->sub[0]); break; case Tytuple: for (i = 0; i < ty->nsub; i++) align = max(align, tyalign(ty->sub[i])); break; case Tyunion: align = 4; for (i = 0; i < ty->nmemb; i++) if (ty->udecls[i]->etype) align = max(align, tyalign(ty->udecls[i]->etype)); break; case Tystruct: for (i = 0; i < ty->nmemb; i++) align = max(align, tyalign(decltype(ty->sdecls[i]))); break; case Tyslice: align = 8; break; default: align = max(align, tysize(ty)); } return min(align, Ptrsz); } /* gets the byte offset of 'memb' within the aggregate type 'ty' */ ssize_t tyoffset(Type *ty, Node *memb) { size_t i; size_t off; ty = tybase(ty); if (ty->type == Typtr) ty = tybase(ty->sub[0]); switch (memb->type) { case Nname: assert(ty->type == Tystruct); off = 0; for (i = 0; i < ty->nmemb; i++) { off = alignto(off, decltype(ty->sdecls[i])); if (!strcmp(namestr(memb), declname(ty->sdecls[i]))) return off; off += size(ty->sdecls[i]); } die("bad offset"); return 0; case Nlit: assert(ty->type == Tytuple); assert(memb->lit.intval < ty->nsub); off = 0; for (i = 0; i < memb->lit.intval; i++) { off += tysize(ty->sub[i]); off = alignto(off, ty->sub[i+1]); } return off; default: die("bad offset node type"); return 0; } } size_t size(Node *n) { Type *t; if (n->type == Nexpr) t = n->expr.type; else t = n->decl.type; return tysize(t); } ssize_t offset(Node *aggr, Node *memb) { return tyoffset(exprtype(aggr), memb); } size_t countargs(Type *t) { size_t nargs; t = tybase(t); nargs = t->nsub - 1; if (classify(t->sub[0]) == ArgBig) nargs++; /* valists are replaced with hidden type parameter, * which we want on the stack for ease of ABI */ if (tybase(t->sub[t->nsub - 1])->type == Tyvalist) nargs--; return nargs; } static void join_classification(PassIn *current, PassIn new) { if (*current == PassInNoPref) { *current = new; } else if ((*current == PassInInt) || (new == PassInInt)) { *current = PassInInt; } else if (*current != new) { *current = PassInMemory; } } static void classify_recursive(Type *t, PassIn *p, size_t *total_offset) { size_t i = 0, sz = tysize(t); size_t cur_offset = *total_offset; PassIn *cur = 0; if (!t) die("cannot pass empty type."); if (cur_offset + sz > 16) { p[0] = PassInMemory; p[1] = PassInMemory; return; } cur = &p[cur_offset / 8]; switch(t->type) { case Tyvoid: break; case Tybool: case Tybyte: case Tychar: case Tyint: case Tyint16: case Tyint32: case Tyint64: case Tyint8: case Typtr: case Tyuint: case Tyuint16: case Tyuint32: case Tyuint64: case Tyuint8: join_classification(cur, PassInInt); break; case Tyslice: /* Slices are too myrddin-specific, they go on the stack. */ join_classification(&p[0], PassInMemory); join_classification(&p[1], PassInMemory); break; case Tyflt32: case Tyflt64: join_classification(cur, PassInSSE); break; case Tyname: classify_recursive(t->sub[0], p, total_offset); break; case Tybad: case Tycode: case Tyfunc: case Tygeneric: case Typaram: case Tyunres: case Tyvalist: case Tyvar: case Ntypes: /* We shouldn't even be in this function */ join_classification(cur, PassInMemory); break; case Tytuple: for (i = 0; i < t->nsub; ++i) { *total_offset = alignto(*total_offset, t->sub[i]); classify_recursive(t->sub[i], p, total_offset); } *total_offset = alignto(*total_offset, t); break; case Tystruct: for (i = 0; i < t->nmemb; ++i) { Type *fieldt = decltype(t->sdecls[i]); *total_offset = alignto(*total_offset, fieldt); classify_recursive(fieldt, p, total_offset); } *total_offset = alignto(*total_offset, t); break; case Tyunion: /* * General enums are too complicated to interop with C, which is the only * reason for anything other than PassInMemory. */ if (isenum(t)) join_classification(cur, PassInInt); else join_classification(cur, PassInMemory); break; case Tyarray: if (t->asize) { t->asize = fold(t->asize, 1); assert(exprop(t->asize) == Olit); for (i = 0; i < t->asize->expr.args[0]->lit.intval; ++i) { classify_recursive(t->sub[0], p, total_offset); } } } *total_offset = align(cur_offset + sz, tyalign(t)); } int isaggregate(Type *t) { t = tybase(t); return (t->type == Tystruct || t->type == Tyarray || t->type == Tytuple || (t->type == Tyunion && !isenum(t))); } ArgType classify(Type *t) { size_t sz = tysize(t); size_t total_offset = 0; /* p must be of length exactly 2 */ PassIn pi[2] = { PassInNoPref, PassInNoPref }; if (tybase(t)->type == Tyvoid) { return ArgVoid; } else if (isstacktype(t)) { if (isaggregate(t) && sz <= 16) { classify_recursive(t, pi, &total_offset); if (pi[0] == PassInMemory || pi[1] == PassInMemory) { return ArgBig; } switch(pi[0]) { case PassInInt: if (sz <= 8) { return ArgAggrI; } switch(pi[1]) { case PassInInt: return ArgAggrII; case PassInSSE: return ArgAggrIF; default: die("Impossible return from classify_recursive"); break; } break; case PassInSSE: if (sz <= 8) { return ArgAggrF; } switch(pi[1]) { case PassInInt: return ArgAggrFI; case PassInSSE: return ArgAggrFF; default: die("Impossible return from classify_recursive"); break; } break; default: die("Impossible return from classify_recursive"); break; } } return ArgBig; } return ArgReg; }