ref: 3de952510eb2a23350d24ed926f19c0cf72a12f2
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]);
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);
}
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;
}
/*
* 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;
}
}