ack/util/ncgg/output.c

1054 lines
21 KiB
C

/*
* (c) copyright 1987 by the Vrije Universiteit, Amsterdam, The Netherlands.
* See the copyright notice in the ACK home directory, in the file "Copyright".
*/
/* #define CODEDEBUG *//* print readable code */
#ifdef CODEDEBUG
int code_in_c=0; /* put readable code in "code" */
int tabledebug=1; /* generate code for table debugging */
#else
int code_in_c = 1; /* put code in "tables.c" */
int tabledebug = 0; /* do not generate code for table debugging */
#endif
int verbose = 0; /* print all statistics */
int use_tes; /* use top element size information */
char *c_file = "tables.c";
char *h_file = "tables.H";
char *cd_file = "code";
#ifndef NORCSID
static char rcsid[] = "$Id$";
#endif
#include <assert.h>
#include <stdio.h>
#include <ctype.h>
#include "varinfo.h"
#include "param.h"
#include "reg.h"
#include "property.h"
#include "token.h"
#include "set.h"
#include "subr.h"
#include "instruct.h"
#include "lookup.h"
#include <cgg_cg.h>
#include "pseudo.h"
#include "regvar.h"
#include "extern.h"
#define BMASK 0xFF
#define BSHIFT 8
FILE *ctable, *htable;
FILE *code;
short *lineset;
int maxline;
extern int nstrings;
extern char *l_strings[];
extern int ninstances;
extern inst_t l_instances[];
extern int nmoves;
extern move_t l_moves[];
extern int ntests;
extern test_t l_tests[];
extern int nstacks;
extern c1_t l_stacks[];
extern int ncoercs;
extern c3_t l_coercs[];
extern int nsplit, maxsplit;
extern c2_t l_split[];
extern set_t l_sets[];
int maxallreg = 0;
int maxregvars = 0;
int setsize;
int prevind = 0;
int npatbytes = -1;
char pattern[MAXPATBYTES];
int pathash[256];
extern int npatterns;
extern int patindex[];
extern int empatlen;
extern int emmnem[];
extern int empatexpr;
extern node_t nodes[];
extern int nnodes;
extern char * filename;
/* Forward declarations */
static void patbyte(int);
static void patshort(int);
static void pat(int);
void opnfile(FILE **f, char *s)
{
if ((*f = fopen(s, "w")) == NULL)
fatal("Can't create %s", s);
}
void unlfile(FILE *f, char *s)
{
if (f)
fclose(f);
if (remove(s) < 0)
error("%s incorrect, must be removed!!", s);
}
void initio(void)
{
opnfile(&ctable, c_file);
opnfile(&htable, h_file);
if (code_in_c)
fprintf(ctable, "char coderules[] = {");
else
opnfile(&code, cd_file);
patbyte(0);
if (tabledebug)
lineset = (short *) myalloc(SZOFSET(MAXSOURCELINES) * sizeof(short));
}
void finishcode(void)
{
if (code_in_c)
fprintf(ctable, "\n};\n\n");
fprintf(ctable, "int allsetno = %d;\n", allsetno);
if (tabledebug)
{
FILE *fd;
int sz;
fd = fopen("lineset","wb");
if (fd != NULL)
{
sz = SZOFSET(maxline) * 2;
fwrite(&sz,1,sizeof(int),fd);
fwrite(lineset,1,sz,fd);
fclose(fd);
}
else
error("Can't create lineset");
}
}
void errorexit(void)
{
unlfile(ctable, c_file);
unlfile(htable, h_file);
if (!code_in_c)
unlfile(code, cd_file);
}
#ifdef CODEDEBUG
#define code8(x) fprintf(code,"%s","x")
#define code8nl(x) fprintf(code,"%s\n","x")
#define code53(x,y) fprintf(code,"%s-%d","x",y)
#define codeint(x) fprintf(code," %d",x)
#define codenl() fprintf(code,"\n")
#else
#define codenl()
#define code8nl(x) code8(x)
void code8(int x)
{
codeindex++;
if (code_in_c)
fprintf(ctable, "%d,", x & 0377);
else
putc(x, code);
}
void code53(int x, int y)
{
code8(x + (y << 5));
}
void codeint(int x)
{
assert(x >= 0 && x <= 32767);
if (x < 128)
{
code8(x);
}
else
{
code8(x / 256 + 128);
code8(x % 256);
}
}
#endif
void outpatterns(void)
{
register int i;
if (!inproc)
{
patbyte(0);
patshort(prevind);
prevind = npatbytes - 2;
patbyte(empatlen);
for (i = 0; i < empatlen; i++)
patbyte(emmnem[i]);
pat(empatexpr);
}
if (callproc == 0)
{
patbyte(npatterns);
for (i = 0; i < npatterns; i++)
pat(patindex[i]);
}
else
{
patbyte(0);
pat(callproc);
pat(nprocargs);
for (i = 0; i < nprocargs; i++)
pat(procarg[i]);
}
}
static void pat(int n)
{
assert(n >= 0);
if (n < 128)
patbyte(n);
else
{
patbyte(n / 256 + 128);
patbyte(n % 256);
}
}
static void patshort(int n)
{
patbyte(n % 256);
patbyte(n / 256);
}
static void patbyte(int n)
{
NEXT(npatbytes, MAXPATBYTES, "Pattern bytes");
pattern[npatbytes] = n;
}
void hashpatterns(void)
{
short index;
register char *bp, *tp;
register short i;
unsigned short hashvalue;
int patlen;
index = prevind;
while (index != 0)
{
bp = &pattern[index];
tp = &bp[PO_MATCH];
i = *tp++ & BMASK;
if (i == BMASK)
{
i = *tp++ & BMASK;
i |= (*tp++ & BMASK) << BSHIFT;
}
patlen = i;
hashvalue = 0;
switch (patlen)
{
default: /* 3 or more */
hashvalue = (hashvalue << 4) ^ (*tp++ & BMASK);
case 2:
hashvalue = (hashvalue << 4) ^ (*tp++ & BMASK);
case 1:
hashvalue = (hashvalue << 4) ^ (*tp++ & BMASK);
}
assert(hashvalue!= ILLHASH);
i = index;
index = (bp[PO_NEXT] & BMASK) | (bp[PO_NEXT + 1] << BSHIFT);
bp[PO_HASH] = hashvalue >> BSHIFT;
hashvalue &= BMASK;
bp[PO_NEXT] = pathash[hashvalue] & BMASK;
bp[PO_NEXT + 1] = pathash[hashvalue] >> BSHIFT;
pathash[hashvalue] = i;
}
}
void outincludes(void)
{
fprintf(ctable, "#include \"param.h\"\n");
fprintf(ctable, "#include \"tables.h\"\n");
fprintf(ctable, "#include \"types.h\"\n");
fprintf(ctable, "#include <cgg_cg.h>\n");
fprintf(ctable, "#include \"data.h\"\n");
}
void outregs(void)
{
register int i, j, k;
short rset[SZOFSET(MAXREGS)];
short clashlist[MAXREGS * MAXREGS];
int iclashlist = 0;
int t, ready;
fprintf(ctable, "char stregclass[] = {\n");
for (i = 0; i < nregs; i++)
fprintf(ctable, "\t%d,\n", l_regs[i].ri_class);
fprintf(ctable, "};\n\nstruct reginfo machregs[] = {\n{0},\n");
for (i = 1; i < nregs; i++)
{
fprintf(ctable, "{%d,%d", strlookup(l_regs[i].ri_repr),
l_regs[i].ri_size);
if (maxmembers != 0)
{
fprintf(ctable, ",{");
for (j = 0; j < maxmembers; j++)
fprintf(ctable, "%d,", l_regs[i].ri_memb[j]);
/* now compute and print set of registers
* that clashes with this register.
* A register clashes with al its children (and theirs)
* and with all their parents.
*/
for (j = 0; j < SZOFSET(MAXREGS); j++)
rset[j] = 0;
BIS(rset, i);
do
{
ready = 1;
for (j = 1; j < nregs; j++)
if (BIT(rset, j))
for (k = 0; k < 2; k++)
if ((t = l_regs[j].ri_memb[k]) != 0)
{
if (BIT(rset,t) == 0)
ready = 0;
BIS(rset, t);
}
} while (!ready);
do
{
ready = 1;
for (j = 1; j < nregs; j++)
for (k = 0; k < 2; k++)
if ((t = l_regs[j].ri_memb[k]) != 0)
if (BIT(rset, t))
{
if (BIT(rset,j) == 0)
ready = 0;
BIS(rset, j);
}
} while (!ready);
fprintf(ctable, "},{");
for (j = 0; j < SZOFSET(nregs); j++)
fprintf(ctable, "0%o,", rset[j] & 0xFFFF);
fprintf(ctable, "}, %d", iclashlist);
for (j = 1; j < nregs; j++)
{
if (BIT(rset, j))
clashlist[iclashlist++] = j;
}
clashlist[iclashlist++] = 0;
}
/*
* Write .r_refcount = 1 for register variables or 0
* for other registers.
*
* reglap: Write .r_refcount = 0 if the register
* variable has a subregister.
*/
fprintf(ctable, ",%d",
l_regs[i].ri_rregvar >= 0 && l_regs[i].ri_memb[0] == 0);
fprintf(ctable, "},\n");
}
fprintf(ctable, "};\n\n short clashlist[] = {\n\t");
for (i = 0; i < iclashlist; i++)
{
fprintf(ctable, "%d, ", clashlist[i]);
if (clashlist[i] == 0)
fprintf(ctable, "\n\t");
}
fprintf(ctable, "0};\n\n");
}
void outregvars(void)
{
register int i, j;
fprintf(htable, "#define REGVARS\n");
if (reglap != 0)
fprintf(htable, "#define REGLAP\n");
fprintf(ctable, "#include \"regvar.h\"\n");
fprintf(ctable, "int nregvar[4] = { ");
for (i = 0; i < 4; i++)
{
fprintf(ctable, "%d, ", nregvar[i]);
if (nregvar[i] > maxregvars)
maxregvars = nregvar[i];
}
fprintf(ctable, "};\n");
for (i = 0; i < 4; i++)
if (nregvar[i] > 0)
fprintf(ctable, "struct regassigned ratar%d[%d];\n", i, nregvar[i]);
for (i = 0; i < 4; i++)
if (nregvar[i] > 0)
{
fprintf(ctable, "int rvtar%d[] = {", i);
for (j = 0; j < nregvar[i]; j++)
fprintf(ctable, "%d,", rvnumbers[i][j]);
fprintf(ctable, "};\n");
}
fprintf(ctable, "\nint *rvnumbers[] = {\n");
for (i = 0; i < 4; i++)
if (nregvar[i] > 0)
fprintf(ctable, "\trvtar%d,\n", i);
else
fprintf(ctable, "\t0,\n");
fprintf(ctable, "};\n\nstruct regassigned *regassigned[] = {\n");
for (i = 0; i < 4; i++)
if (nregvar[i] > 0)
fprintf(ctable, "\tratar%d,\n", i);
else
fprintf(ctable, "\t0,\n");
fprintf(ctable, "};\n");
}
int typeconv(int n)
{
if (n >= 0)
return (2);
if (n == -1)
return (1);
if (n == -2)
return (3);
assert(n == -3);
return (0);
}
static int is_ascii(int c)
{
if ((c >= 0) && (c <= 127))
return 1;
return 0;
}
void outfmt(register char *p)
{
register int c;
fprintf(ctable, "\"");
while ((c = (*p++ & 0377)) != 0)
{
if (!is_ascii(c) || iscntrl(c))
{
fprintf(ctable, "\\%c%c%c", ((c & ~0300) >> 6) + '0',
((c & 070) >> 3) + '0', (c & 07) + '0');
}
else
fprintf(ctable, "%c", c);
}
fprintf(ctable, "\"");
}
void outtokens(void)
{
register int tokno, i;
register token_p tp;
fprintf(ctable, "tkdef_t tokens[] = {{0},\n");
for (tokno = 1; tokno < ntokens; tokno++)
{
tp = l_tokens[tokno];
fprintf(ctable, "/* %3d */{%d,{%d,%d},{", tokno, tp->tk_size,
tp->tk_cost.ct_space, tp->tk_cost.ct_time);
for (i = 0; i < maxtokensize; i++)
fprintf(ctable, "%d,", typeconv(tp->tk_att[i].ta_type));
fprintf(ctable, "},%d},\t/* ", tp->tk_format);
if (tp->tk_format >= 0)
outfmt(l_strings[tp->tk_format]);
else
fprintf(ctable, "(no format)");
fprintf(ctable, " */\n");
}
fprintf(ctable, "{0}};\n\n");
}
void outenodes(void)
{
register node_p np;
fprintf(ctable, "node_t enodes[] = {\n");
for (np = nodes; np < &nodes[nnodes]; np++)
fprintf(ctable, "{%d,%d,%d},\n", np->ex_operator, np->ex_lnode,
np->ex_rnode);
fprintf(ctable, "};\n\n");
}
void outstrings(void)
{
register int i;
#if 0
register char *p;
register int c;
#endif
if (tabledebug)
fprintf(ctable, "char *tablename = \"%s\";\n", filename);
fprintf(ctable, "string codestrings[] = {\n");
for (i = 0; i < nstrings; i++)
{
fprintf(ctable, "\t");
outfmt(l_strings[i]);
#if 0
while ((c= (*p++&0377))!=0)
{
if (! isascii(c) || iscntrl(c))
{
fprintf(ctable,"\\%c%c%c",
((c&~0300)>>6) + '0', ((c&070)>>3)+'0',
(c&07)+'0');
}
else fprintf(ctable, "%c",c);
}
fprintf(ctable,"\",\n");
#endif
fprintf(ctable, ",\n");
}
fprintf(ctable, "};\n\n");
}
extern set_t unstackset;
void outsets(void)
{
register int i;
register set_p sp;
fprintf(ctable, "set_t machsets[] = {\n");
for (sp = l_sets; sp < &l_sets[nsets]; sp++)
{
fprintf(ctable, "/* %3ld */ {%3d,{", (long) (sp - l_sets),
sp->set_size);
for (i = 0; i < setsize; i++)
fprintf(ctable, "0x%x,", sp->set_val[i] & 0xFFFF);
fprintf(ctable, "}},\n");
}
fprintf(ctable, "};\n\n");
fprintf(ctable, "set_t unstackset = { %3d,{\n", unstackset.set_size);
for (i = 0; i < setsize; i++)
fprintf(ctable, "0x%x,", unstackset.set_val[i] & 0xFFFF);
fprintf(ctable, "}};\n\n");
}
void outinstances(void)
{
register inst_p ip;
register int i;
fprintf(ctable, "inst_t tokeninstances[] = {\n");
for (ip = l_instances; ip < &l_instances[ninstances]; ip++)
{
fprintf(ctable, "{ %d, {", ip->in_which);
for (i = 0; i <= maxtokensize; i++)
fprintf(ctable, "%d,", ip->in_info[i]);
fprintf(ctable, "}},\n");
}
fprintf(ctable, "};\n\n");
}
void outmoves(void)
{
register move_p mp;
fprintf(ctable, "move_t moves[] = {\n");
for (mp = l_moves; mp < &l_moves[nmoves]; mp++)
fprintf(ctable, "{%d,%d,%d,%d,%d},\n", mp->m_set1, mp->m_expr1,
mp->m_set2, mp->m_expr2, mp->m_cindex);
fprintf(ctable, "{-1}\n};\n\n");
}
void outtests(void)
{
register test_p tp;
fprintf(ctable, "test_t tests[] = {\n");
for (tp = l_tests; tp < &l_tests[ntests]; tp++)
fprintf(ctable, "{%d,%d,%d},\n", tp->t_set, tp->t_expr, tp->t_cindex);
fprintf(ctable, "{-1}\n};\n\n");
}
void outstacks(void)
{
register c1_p cp;
fprintf(ctable, "c1_t c1coercs[] = {\n");
for (cp = l_stacks; cp < &l_stacks[nstacks]; cp++)
fprintf(ctable, "{%d,%d,%d,%d},\n", cp->c1_texpno, cp->c1_expr,
cp->c1_prop, cp->c1_codep);
fprintf(ctable, "{-1}\n};\n\n");
}
void outsplits(void)
{
register c2_p cp;
register int i;
fprintf(ctable, "c2_t c2coercs[] = {\n");
for (cp = l_split; cp < &l_split[nsplit]; cp++)
{
fprintf(ctable, "{%d,%d,%d,{", cp->c2_texpno, cp->c2_expr,
cp->c2_nsplit);
for (i = 0; i < maxsplit; i++)
fprintf(ctable, "%d,", cp->c2_repl[i]);
fprintf(ctable, "},%d},\n", cp->c2_codep);
}
fprintf(ctable, "{-1}\n};\n\n");
}
void outcoercs(void)
{
register c3_p cp;
fprintf(ctable, "c3_t c3coercs[] = {\n");
for (cp = l_coercs; cp < &l_coercs[ncoercs]; cp++)
fprintf(ctable, "{%d,%d,%d,%d,%d},\n", cp->c3_texpno, cp->c3_expr,
cp->c3_prop, cp->c3_repl, cp->c3_codep);
fprintf(ctable, "{-1}\n};\n\n");
}
void outproplists(void)
{
register int propno;
register int regno;
for (propno = 0; propno < nprops; propno++)
{
fprintf(ctable, "struct reginfo *rlist%d[] = {\n", propno);
for (regno = 1; regno < nregs; regno++)
if (BIT(l_props[propno].pr_regset, regno))
fprintf(ctable, "&machregs[%d],\n", regno);
fprintf(ctable, "0\n};\n");
}
fprintf(ctable, "struct reginfo **reglist[] = {\n");
for (propno = 0; propno < nprops; propno++)
fprintf(ctable, "rlist%d,\n", propno);
fprintf(ctable, "};\n\n");
}
void outconsts(void)
{
fprintf(ctable, "unsigned cc1 = %u;\n", fc1);
fprintf(ctable, "unsigned cc2 = %u;\n", fc2);
fprintf(ctable, "unsigned cc3 = %u;\n", fc3);
fprintf(ctable, "unsigned cc4 = %u;\n", fc4);
}
void cdef(char *s, int n)
{
fprintf(htable, "#define %s %d\n", s, n);
}
void passon(char *s)
{
char buf[32];
sprintf(buf, "T%s", s);
cdef(buf, cmustbeset(s));
}
void outdefs(void)
{
register symbol *sy_p;
extern int maxempatlen, maxrule;
char *wrdfmt;
passon("EM_WSIZE");
passon("EM_PSIZE");
passon("EM_BSIZE");
if ((sy_p = lookup("FORMAT", symsconst, justlooking)) != 0)
{
wrdfmt = l_strings[sy_p->sy_value.syv_stringno];
fprintf(htable, "#define WRD_FMT \"%s\"\n", wrdfmt);
}
cdef("MAXALLREG", maxallreg);
cdef("SETSIZE", setsize);
cdef("NREGS", nregs);
cdef("REGSETSIZE", SZOFSET(nregs));
cdef("TOKENSIZE", maxtokensize);
cdef("MAXMEMBERS", maxmembers);
cdef("LONGESTPATTERN", maxempatlen);
cdef("MAXPATLEN", maxtokpatlen);
cdef("MAXREPLLEN", maxtokrepllen);
cdef("MAXEMREPLLEN", maxemrepllen);
cdef("MAXPROCARG", maxprocargs);
cdef("MAXRULE", maxrule < 16 ? 16 : maxrule);
if (nsplit > 0)
{
cdef("MAXSPLIT", maxsplit);
}
if (tabledebug)
cdef("TABLEDEBUG", 1);
if (use_tes)
cdef("USE_TES", 1);
}
void outars(void)
{
register int i;
if (code_in_c)
fprintf(htable, "#define CODEINC 1\n");
else
{
fprintf(ctable, "char coderules[%d];\n", codeindex);
fprintf(ctable, "int ncodebytes=%d;\n", codeindex);
}
fprintf(ctable, "char pattern[%d]={\n", npatbytes + 1);
for (i = 0; i <= npatbytes; i++)
{
fprintf(ctable, "%d,%c", pattern[i] & BMASK, i % 16 == 15 ? '\n' : ' ');
}
fprintf(ctable, "};\n\n");
fprintf(ctable, "int pathash[256]={\n");
for (i = 0; i < 256; i++)
{
fprintf(ctable, "%d,%c", pathash[i] & 0xFFFF, i % 10 == 9 ? '\n' : ' ');
}
fprintf(ctable, "};\n");
}
void finishio(void)
{
extern int nregs;
finishcode();
hashpatterns();
setsize = SZOFSET(nregs + ntokens);
outdefs();
outincludes();
outregs();
outtokens();
outenodes();
outstrings();
outsets();
outinstances();
outmoves();
outtests();
outstacks();
if (nsplit > 0)
outsplits();
outcoercs();
outproplists();
outconsts();
if (rvused)
outregvars();
outars();
}
void codecoco(int cocono)
{
if (cocono == -1)
return;
code8(DO_SETCC);
codeint(cocono);
codenl();
}
void dopattern(int stackcoerc, varinfo *kills, varinfo *allocates,
varinfo *generates, varinfo *yields, varinfo *leaving)
{
register int i;
int n, nops;
register struct varinfo *vp, *vivp;
register instr_p instp;
int al, deal;
int vil;
int cocono = -1;
cost_t totcost;
int nremoves;
int removelist[100];
static char tlab[] = "0:";
extern int optexact, optstack, startline;
extern char *filename;
extern int lineno;
#ifdef CODEDEBUG
fprintf(code,"Code(%d) at \"%s\", line %d\n",stackcoerc,filename,lineno);
#endif
if (code_in_c)
fprintf(ctable, "\n/* \"%s\", line %d */ ", filename, lineno);
if (tabledebug)
{
code8(DO_DLINE);
codeint(startline);
codenl();
if (startline < MAXSOURCELINES)
{
if (startline > maxline)
maxline = startline;
BIS(lineset, startline);
}
else
{
static int beenhere = 0;
if (!beenhere)
{
beenhere++;
error("Too many source lines for table debug");
}
}
}
/* MATCH part */
if (tokpatlen)
{
if (optexact)
if (optstack)
code53(DO_XXMATCH, tokpatlen);
else
code53(DO_XMATCH, tokpatlen);
else
code53(DO_MATCH, tokpatlen);
for (i = 0; i < tokpatlen; i++)
codeint(tokpatset[i]);
codenl();
}
else if (stackcoerc)
code8nl(DO_COERC);
if (optstack)
{
code53(DO_TOSTACK, 0);
codeint(allsetno);
codenl();
}
/* The kills */
for (vp = kills; vp != 0; vp = vp->vi_next)
{
if (vp->vi_int[1] != 0)
{
code53(DO_REMOVE, 1);
codeint(vp->vi_int[0]);
codeint(vp->vi_int[1]);
}
else if (vp->vi_int[0] >= 0)
{
code53(DO_REMOVE, 0);
codeint(vp->vi_int[0]);
}
else
{
code8(DO_KILLREG);
codeint(-vp->vi_int[0] - 1);
} codenl();
}
nremoves = 0;
for (vp = generates; vp != 0; vp = vp->vi_next)
{
if (vp->vi_int[0] != INSREMOVE)
continue;
for (i = 0; i < nremoves; i++)
if (vp->vi_int[1] == removelist[i])
break;
if (i == nremoves)
{
assert(nremoves < (sizeof(removelist) / sizeof(int)));
removelist[nremoves++] = vp->vi_int[1];
}
}
for (i = 0; i < nremoves; i++)
{
code8(DO_RREMOVE);
codeint(removelist[i]);
codenl();
}
/* allocate part */
deal = 0;
al = 0;
for (vp = allocates; vp != 0; vp = vp->vi_next)
{
if (vp->vi_int[0] == -1)
{ /* Deallocate */
deal++;
code8(DO_DEALLOCATE);
codeint(vp->vi_int[1]);
codenl();
}
else
{
if (vp->vi_int[1] == 0)
{
code53(DO_ALLOCATE, 0);
codeint(vp->vi_int[0]);
codenl();
}
else
{
code53(DO_ALLOCATE, 1);
codeint(vp->vi_int[0]);
codeint(vp->vi_int[1]);
codenl();
}
al++;
}
}
if (deal)
code8nl(DO_REALLOCATE);
if (al > maxallreg)
maxallreg = al;
totcost.ct_space = 0;
totcost.ct_time = 0;
for (vp = generates; vp != 0; vp = vp->vi_next)
{
n = vp->vi_int[0];
switch (n)
{
default:
assert(n >= 0);
instp = &l_instr[n];
nops = instp->i_nops;
code53(DO_INSTR, nops);
if (vp->vi_int[1] == 0)
{
codeint(instp->i_asname);
}
else
{
codeint(10000 + vp->vi_int[1]);
}
vivp = vp->vi_vi;
for (i = 0; i < nops; i++)
{
codeint(vivp->vi_int[0]);
vivp = vivp->vi_vi;
}
codenl();
totcost.ct_space += instp->i_cost.ct_space;
totcost.ct_time += instp->i_cost.ct_time;
break;
case INSREMOVE:
break;
case INSMOVE:
codecoco(cocono);
code8(DO_MOVE);
codeint(vp->vi_int[1]);
codeint(vp->vi_int[2]);
codenl();
break;
case INSTEST:
codecoco(cocono);
code8(DO_TEST);
codeint(vp->vi_int[1]);
codenl();
break;
case INSPRETURN:
code8(DO_PRETURN);
codenl();
break;
case INSTLAB:
cocono = 0;
tlab[0] = vp->vi_int[1] + '0';
code53(DO_INSTR, 0);
codeint(strlookup(tlab));
codenl();
break;
case INSSETCC:
cocono = vp->vi_int[1];
break;
case INSERASE:
code8(DO_ERASE);
codeint(vp->vi_int[1]);
codenl();
break;
case INSLABDEF:
cocono = 0;
code8(DO_LABDEF);
codeint(vp->vi_int[1]);
codenl();
break;
}
}
codecoco(cocono);
vil = vilength(yields);
if (vil != 0 || tokpatlen != 0 || allocates != 0)
{
code53(DO_TOKREPLACE, vilength(yields));
for (vp = yields; vp != 0; vp = vp->vi_next)
{
codeint(vp->vi_int[0]);
} codenl();
}
if (leaving != 0)
{
code53(DO_EMREPLACE, vilength(leaving));
while (leaving != 0)
{
codeint(leaving->vi_int[0]);
codeint(leaving->vi_int[1]);
leaving = leaving->vi_next;
} codenl();
}
if (totcost.ct_space != 0 || totcost.ct_time != 0)
{
code8(DO_COST);
codeint(totcost.ct_space);
codeint(totcost.ct_time);
codenl();
}
if (empatlen == 0 && !inproc)
code8nl(DO_RETURN);
else
code8nl(DO_NEXTEM);
}
void used(char *resource, int use, int max)
{
if (verbose || 4 * use > 3 * max)
fprintf(stderr, "%s %d(%d)\n", resource, use, max);
}
void statistics(void)
{
extern int nnodes, maxempatlen, maxrule;
used("Registers", nregs, MAXREGS);
used("Properties", nprops, MAXPROPS);
used("Tokens", ntokens, MAXTOKENS);
used("Tokensize", maxtokensize, MAXATT);
used("Sets", nsets, MAXSETS);
used("Instructions", ninstr, MAXINSTR);
used("Strings", nstrings, MAXSTRINGS);
used("Exp-nodes", nnodes, MAXNODES);
used("EM-pat length", maxempatlen, EMPATMAX);
used("rules/EM-pattern", maxrule, MAXPATTERNS);
used("Allocates/rule", maxallreg, MAXALLREG);
used("Instances", ninstances, MAXINSTANCES);
used("Moves", nmoves, MAXMOVES);
used("Tests", ntests, MAXTESTS);
used("Stacks", nstacks, MAXSTACKS);
used("1->1 Coercions", ncoercs, MAXCOERCS);
used("Splitting coercions", nsplit, MAXSPLCOERC);
used("Register variables", maxregvars, MAXREGVAR);
used("Pat bytes", npatbytes + 1, MAXPATBYTES);
if (tabledebug)
used("Source lines", maxline, MAXSOURCELINES);
}