ref: c96b770a0aa93c1e7b99e5874808af85ddacc6a7
parent: 468265b7b2719d783afc46ea5aa7c9ffa3142970
author: Ori Bernstein <[email protected]>
date: Tue Jan 6 08:29:51 EST 2015
Improve error messages. This involved lots of table changes.
--- a/6/gengas.c
+++ b/6/gengas.c
@@ -514,7 +514,7 @@
simpglobl(n, globls, &fn, &nfn, &blob, &nblob);
break;
default:
- die("Bad node %s in toplevel", nodestr(n->type));
+ die("Bad node %s in toplevel", nodestr[n->type]);
break;
}
}
--- a/6/genp9.c
+++ b/6/genp9.c
@@ -533,7 +533,7 @@
simpglobl(n, globls, &fn, &nfn, &blob, &nblob);
break;
default:
- die("Bad node %s in toplevel", nodestr(n->type));
+ die("Bad node %s in toplevel", nodestr[n->type]);
break;
}
}
--- a/6/isel.c
+++ b/6/isel.c
@@ -102,11 +102,11 @@
case Lbool: l = loclit(v->lit.boolval, mode(n)); break;
case Lint: l = loclit(v->lit.intval, mode(n)); break;
default:
- die("Literal type %s should be blob", litstr(v->lit.littype));
+ die("Literal type %s should be blob", litstr[v->lit.littype]);
}
break;
default:
- die("Node %s not leaf in loc()", opstr(exprop(n)));
+ die("Node %s not leaf in loc()", opstr[exprop(n)]);
break;
}
return l;
@@ -648,7 +648,7 @@
return r;
case Oasn: /* relabel */
- die("Unimplemented op %s", opstr(exprop(n)));
+ die("Unimplemented op %s", opstr[exprop(n)]);
break;
case Oset:
assert(exprop(args[0]) == Ovar || exprop(args[0]) == Oderef);
@@ -756,7 +756,7 @@
case Obreak: case Ocontinue:
case Numops:
dump(n, stdout);
- die("Should not see %s in isel", opstr(exprop(n)));
+ die("Should not see %s in isel", opstr[exprop(n)]);
break;
}
return r;
--- a/6/simp.c
+++ b/6/simp.c
@@ -277,7 +277,7 @@
static int ispure(Node *n)
{
- return ispureop[exprop(n)];
+ return opispure[exprop(n)];
}
static int isconstfn(Node *s)
@@ -822,7 +822,7 @@
case Oderef: r = deref(rval(s, args[0], NULL), NULL); break;
case Omemb: r = deref(membaddr(s, n), NULL); break;
default:
- fatal(n, "%s cannot be an lvalue", opstr(exprop(n)));
+ fatal(n, "%s cannot be an lvalue", opstr[exprop(n)]);
break;
}
return r;
--- a/mi/cfg.c
+++ b/mi/cfg.c
@@ -211,7 +211,7 @@
case Ndecl:
break;
default:
- die("Invalid node type %s in mkcfg", nodestr(nl[i]->type));
+ die("Invalid node type %s in mkcfg", nodestr[nl[i]->type]);
}
}
post = mkbb(cfg);
--- a/mi/match.c
+++ b/mi/match.c
@@ -224,7 +224,7 @@
break;
default:
ret = NULL;
- fatal(pat, "unsupported pattern %s of type %s", opstr(exprop(pat)), tystr(exprtype(pat)));
+ fatal(pat, "unsupported pattern %s of type %s", opstr[exprop(pat)], tystr(exprtype(pat)));
break;
}
return ret;
@@ -332,7 +332,7 @@
case Ovar:
return namestr(n->expr.args[0]);
case Olit:
- return litstr(n->expr.args[0]->lit.littype);
+ return litstr[n->expr.args[0]->lit.littype];
case Oucon:
return namestr(n->expr.args[0]);
case Otup:
@@ -354,7 +354,7 @@
size_t i;
if (dt->patexpr) {
e = dt->patexpr;
- indentf(depth, "%s%s %s : %s\n", iswild ? "WILDCARD " : "", opstr(exprop(e)), dtnodestr(e), tystr(exprtype(e)));
+ indentf(depth, "%s%s %s : %s\n", iswild ? "WILDCARD " : "", opstr[exprop(e)], dtnodestr(e), tystr(exprtype(e)));
}
if (dt->cap)
for (i = 0; i < dt->ncap; i++)
--- a/parse/dump.c
+++ b/parse/dump.c
@@ -113,7 +113,7 @@
findentf(fd, depth, "Nil\n");
return;
}
- findentf(fd, depth, "%s.%zd@%i", nodestr(n->type), n->nid, lnum(n->loc));
+ findentf(fd, depth, "%s.%zd@%i", nodestr[n->type], n->nid, lnum(n->loc));
switch(n->type) {
case Nfile:
fprintf(fd, "(name = %s)\n", n->file.files[0]);
@@ -191,7 +191,7 @@
else
tid = -1;
fprintf(fd, " (type = %s [tid %d], op = %s, isconst = %d, did=%zd)\n",
- ty, tid, opstr(n->expr.op), n->expr.isconst, n->expr.did);
+ ty, tid, opstr[n->expr.op], n->expr.isconst, n->expr.did);
free(ty);
outnode(n->expr.idx, fd, depth + 1);
for (i = 0; i < n->expr.nargs; i++)
--- a/parse/infer.c
+++ b/parse/infer.c
@@ -56,6 +56,42 @@
static Type *unify(Inferstate *st, Node *ctx, Type *a, Type *b);
static Type *tf(Inferstate *st, Type *t);
+static void ctxstrcall(char *buf, size_t sz, Inferstate *st, Node *n)
+{
+ char *p, *end, *sep, *t;
+ size_t nargs, i;
+ Node **args;
+ Type *et;
+
+ args = n->expr.args;
+ nargs = n->expr.nargs;
+ p = buf;
+ end = buf + sz;
+ sep = "";
+
+ if (exprop(args[0]) == Ovar)
+ p += snprintf(p, end - p, "%s(", namestr(args[0]->expr.args[0]));
+ else
+ p += snprintf(p, end - p, "<e>(");
+ for (i = 1; i < nargs; i++) {
+ et = exprtype(args[i]);
+ if (et != NULL)
+ t = tystr(et);
+ else
+ t = "?";
+
+ if (exprop(args[i]) == Ovar)
+ p += snprintf(p, end - p, "%s%s:%s", sep, namestr(args[0]->expr.args[0]), t);
+ else
+ p += snprintf(p, end - p, "%s<e%zd>:%s", sep, i, t);
+ sep = ", ";
+ free(t);
+ }
+ t = tystr(exprtype(args[0])->sub[0]);
+ p += snprintf(p, end - p, "): %s", t);
+ free(t);
+}
+
/* Tries to give a good string describing the context
* for the sake of error messages. */
static char *ctxstr(Inferstate *st, Node *n)
@@ -69,7 +105,7 @@
idx = NULL;
switch (n->type) {
default:
- s = strdup(nodestr(n->type));
+ s = strdup(nodestr[n->type]);
break;
case Ndecl:
u = declname(n);
@@ -82,21 +118,34 @@
s = strdup(namestr(n));
break;
case Nexpr:
- if (n->expr.idx)
- idx = ctxstr(st, n->expr.idx);
if (exprop(n) == Ovar)
u = namestr(n->expr.args[0]);
else
- u = opstr(exprop(n));
+ u = opstr[exprop(n)];
if (exprtype(n))
t = tystr(tf(st, exprtype(n)));
else
- t = strdup("unknown");
- if (idx)
- snprintf(buf, sizeof buf, ".%s=%s:%s", idx, u, t);
- else
- snprintf(buf, sizeof buf, "%s:%s", u, t);
- free(idx);
+ t = strdup("unknown type");
+ switch (opclass[exprop(n)]) {
+ case OTbin:
+ snprintf(buf, sizeof buf, "<e1> %s <e2>", oppretty[exprop(n)]);
+ break;
+ case OTpre:
+ snprintf(buf, sizeof buf, "%s<e>", oppretty[exprop(n)]);
+ break;
+ case OTpost:
+ snprintf(buf, sizeof buf, "<e>%s", oppretty[exprop(n)]);
+ break;
+ case OTzarg:
+ snprintf(buf, sizeof buf, "%s", oppretty[exprop(n)]);
+ case OTmisc:
+ if (exprop(n) == Ovar)
+ snprintf(buf, sizeof buf, "%s:%s", namestr(n->expr.args[0]), t);
+ else if (exprop(n) == Ocall)
+ ctxstrcall(buf, sizeof buf, st, n);
+ else
+ snprintf(buf, sizeof buf, "%s:%s", u, t);
+ }
free(t);
s = strdup(buf);
break;
@@ -368,7 +417,7 @@
case Nlit: n->lit.type = t; break;
case Nfunc: n->func.type = t; break;
default:
- die("untypable node %s", nodestr(n->type));
+ die("untypable node %s", nodestr[n->type]);
break;
}
}
@@ -422,7 +471,7 @@
case Nfunc: t = n->func.type; break;
default:
t = NULL;
- die("untypeable node %s", nodestr(n->type));
+ die("untypeable node %s", nodestr[n->type]);
break;
};
return tf(st, t);
@@ -1073,7 +1122,7 @@
}
if (exprop(n) == Ovar)
n->expr.isconst = decls[n->expr.did]->decl.isconst;
- else if (ispureop[exprop(n)])
+ else if (opispure[exprop(n)])
n->expr.isconst = isconst;
*exprconst = n->expr.isconst;
}
@@ -1288,7 +1337,7 @@
case Ofeq: case Ofne: case Ofgt: case Ofge: case Oflt: case Ofle:
case Oueq: case Oune: case Ougt: case Ouge: case Oult: case Oule:
case Oudata:
- die("Should not see %s in fe", opstr(exprop(n)));
+ die("Should not see %s in fe", opstr[exprop(n)]);
break;
}
}
--- a/parse/names.c
+++ b/parse/names.c
@@ -11,52 +11,45 @@
#include "parse.h"
-static char *optab[] = {
-#define O(op, pure) #op,
+char *opstr[] = {
+#define O(op, pure, class, pretty) #op,
#include "ops.def"
#undef O
};
-int ispureop[] = {
-#define O(op, pure) pure,
+char * oppretty[] = {
+#define O(op, pure, class, pretty) pretty,
#include "ops.def"
#undef O
};
-static char *nodetab[] = {
+
+int opispure[] = {
+#define O(op, pure, class, pretty) pure,
+#include "ops.def"
+#undef O
+};
+
+int opclass[] = {
+#define O(op, pure, class, pretty) class,
+#include "ops.def"
+#undef O
+};
+
+char *nodestr[] = {
#define N(nt) #nt,
#include "nodes.def"
#undef N
};
-static char *littab[] = {
+char *litstr[] = {
#define L(lt) #lt,
#include "lits.def"
#undef L
};
-static char *tidtab[] = {
+char *tidstr[] = {
#define Ty(t, n) n,
#include "types.def"
#undef Ty
};
-
-char *opstr(Op o)
-{
- return optab[o];
-}
-
-char *nodestr(Ntype nt)
-{
- return nodetab[nt];
-}
-
-char *litstr(Littype lt)
-{
- return littab[lt];
-}
-
-char *tidstr(Ty tid)
-{
- return tidtab[tid];
-}
--- a/parse/node.c
+++ b/parse/node.c
@@ -374,7 +374,7 @@
case Ndecl: return n->decl.type; break;
case Nexpr: return n->expr.type; break;
case Nlit: return n->lit.type; break;
- default: die("Node %s has no type", nodestr(n->type)); break;
+ default: die("Node %s has no type", nodestr[n->type]); break;
}
return NULL;
}
--- a/parse/ops.def
+++ b/parse/ops.def
@@ -1,99 +1,100 @@
-/* operator name, is it pure */
-O(Obad, 1)
-O(Oadd, 1)
-O(Osub, 1)
-O(Omul, 1)
-O(Odiv, 1)
-O(Omod, 1)
-O(Oneg, 1)
-O(Obor, 1)
-O(Oband, 1)
-O(Obxor, 1)
-O(Obsl, 1)
-O(Obsr, 1)
-O(Obnot, 1)
-O(Opreinc, 1)
-O(Opostinc, 1)
-O(Opredec, 1)
-O(Opostdec, 1)
-O(Oaddr, 1)
-O(Oderef, 1)
-O(Olor, 1)
-O(Oland, 1)
-O(Olnot, 1)
-O(Oeq, 1)
-O(One, 1)
-O(Ogt, 1)
-O(Oge, 1)
-O(Olt, 1)
-O(Ole, 1)
-O(Oasn, 1)
-O(Oaddeq, 1)
-O(Osubeq, 1)
-O(Omuleq, 1)
-O(Odiveq, 1)
-O(Omodeq, 1)
-O(Oboreq, 1)
-O(Obandeq, 1)
-O(Obxoreq, 1)
-O(Obsleq, 1)
-O(Obsreq, 1)
-O(Oidx, 1)
-O(Oslice, 1)
-O(Omemb, 1)
-O(Osize, 1)
-O(Ocall, 0)
-O(Ocast, 1)
-O(Oret, 1)
-O(Ojmp, 0)
-O(Obreak, 0)
-O(Ocontinue, 0)
-O(Ovar, 1)
-O(Olit, 1)
-O(Oucon, 1)
-O(Otup, 1)
-O(Ostruct, 1)
-O(Oarr, 1)
+/* operator name, is it pure, pretty name */
+O(Obad, 1, OTmisc, "BAD")
+O(Oadd, 1, OTbin, "+")
+O(Osub, 1, OTbin, "-")
+O(Omul, 1, OTbin, "*")
+O(Odiv, 1, OTbin, "/")
+O(Omod, 1, OTbin, "%")
+O(Oneg, 1, OTpre, "-")
+O(Obor, 1, OTbin, "|")
+O(Oband, 1, OTbin, "&")
+O(Obxor, 1, OTbin, "^")
+O(Obsl, 1, OTbin, "<<")
+O(Obsr, 1, OTbin, ">>")
+O(Obnot, 1, OTpre, "~")
+O(Opreinc, 1, OTpre, "++")
+O(Opostinc, 1, OTpost, "++")
+O(Opredec, 1, OTpre, "--")
+O(Opostdec, 1, OTpost, "--")
+O(Oaddr, 1, OTpre, "&")
+O(Oderef, 1, OTpost, "#")
+O(Olor, 1, OTbin, "||")
+O(Oland, 1, OTbin, "&&")
+O(Olnot, 1, OTpre, "!")
+O(Oeq, 1, OTbin, "==")
+O(One, 1, OTbin, "!=")
+O(Ogt, 1, OTbin, ">")
+O(Oge, 1, OTbin, ">=")
+O(Olt, 1, OTbin, "<")
+O(Ole, 1, OTbin, "<=")
+O(Oasn, 1, OTbin, "==")
+O(Oaddeq, 1, OTbin, "+=")
+O(Osubeq, 1, OTbin, "-=")
+O(Omuleq, 1, OTbin, "*=")
+O(Odiveq, 1, OTbin, "/=")
+O(Omodeq, 1, OTbin, "%=")
+O(Oboreq, 1, OTbin, "|=")
+O(Obandeq, 1, OTbin, "&=")
+O(Obxoreq, 1, OTbin, "^=")
+O(Obsleq, 1, OTbin, "<<=")
+O(Obsreq, 1, OTbin, ">>=")
+O(Oidx, 1, OTmisc, NULL)
+O(Oslice, 1, OTmisc, NULL)
+O(Omemb, 1, OTmisc, NULL)
+O(Osize, 1, OTmisc, NULL)
+O(Ocall, 0, OTmisc, NULL)
+O(Ocast, 1, OTmisc, NULL)
+O(Oret, 1, OTpre, "->")
+O(Ojmp, 0, OTpre, "goto")
+O(Obreak, 0, OTzarg, "break")
+O(Ocontinue, 0, OTzarg, "continue")
+O(Ovar, 1, OTmisc, NULL)
+O(Olit, 1, OTmisc, NULL)
+O(Oucon, 1, OTmisc, "`")
+O(Otup, 1, OTmisc, NULL)
+O(Ostruct, 1, OTmisc, NULL)
+O(Oarr, 1, OTmisc, NULL)
/* all below this point are backend-only */
-O(Ocjmp, 1) /* conditional jump */
-O(Ojtab, 1) /* jump table */
-O(Oset, 1) /* store to var */
-O(Osllen, 1) /* size of slice */
-O(Oslbase, 1) /* base of sice */
-O(Outag, 1) /* tag of union */
-O(Oudata, 1) /* pointer to contents of union */
-O(Oblit, 1) /* blit memory */
+O(Ocjmp, 1, OTmisc, NULL) /* conditional jump */
+O(Ojtab, 1, OTmisc, NULL) /* jump table */
+O(Oset, 1, OTbin, "=") /* store to var */
+O(Osllen, 1, OTpre, "SLLEN") /* size of slice */
+O(Oslbase, 1, OTpre, "SLBASE") /* base of sice */
+O(Outag, 1, OTpre, "UTAG") /* tag of union */
+O(Oudata, 1, OTpre, "UDATA") /* pointer to contents of union */
+O(Oblit, 1, OTbin, "BLIT") /* blit memory */
/* integer conversions */
-O(Otrunc, 1) /* truncating cast */
-O(Ozwiden, 1) /* zero-extending widening cast */
-O(Oswiden, 1) /* sign-extending widening cast */
+O(Otrunc, 1, OTmisc, NULL) /* truncating cast */
+O(Ozwiden, 1, OTmisc, NULL) /* zero-extending widening cast */
+O(Oswiden, 1, OTmisc, NULL) /* sign-extending widening cast */
/* float conversions */
-O(Oflt2int, 1) /* float to int conversion */
-O(Oint2flt, 1) /* int to float conversion */
-O(Oflt2flt, 1) /* flt32<->flt64 conversion */
+O(Oflt2int, 1, OTmisc, NULL) /* float to int conversion */
+O(Oint2flt, 1, OTmisc, NULL) /* int to float conversion */
+O(Oflt2flt, 1, OTmisc, NULL) /* flt32<->flt64 conversion */
/* floating arithmetic */
-O(Ofadd, 1)
-O(Ofsub, 1)
-O(Ofmul, 1)
-O(Ofdiv, 1)
-O(Ofneg, 1)
+O(Ofadd, 1, OTmisc, NULL)
+O(Ofsub, 1, OTmisc, NULL)
+O(Ofmul, 1, OTmisc, NULL)
+O(Ofdiv, 1, OTmisc, NULL)
+O(Ofneg, 1, OTmisc, NULL)
/* floating point comparisons */
-O(Ofeq, 1)
-O(Ofne, 1)
-O(Ofgt, 1)
-O(Ofge, 1)
-O(Oflt, 1)
-O(Ofle, 1)
+O(Ofeq, 1, OTmisc, NULL)
+O(Ofne, 1, OTmisc, NULL)
+O(Ofgt, 1, OTmisc, NULL)
+O(Ofge, 1, OTmisc, NULL)
+O(Oflt, 1, OTmisc, NULL)
+O(Ofle, 1, OTmisc, NULL)
/* unsigned comparisons */
-O(Oueq, 1)
-O(Oune, 1)
-O(Ougt, 1)
-O(Ouge, 1)
-O(Oult, 1)
-O(Oule, 1)
+O(Oueq, 1, OTmisc, NULL)
+O(Oune, 1, OTmisc, NULL)
+O(Ougt, 1, OTmisc, NULL)
+O(Ouge, 1, OTmisc, NULL)
+O(Oult, 1, OTmisc, NULL)
+O(Oule, 1, OTmisc, NULL)
+
--- a/parse/parse.h
+++ b/parse/parse.h
@@ -26,7 +26,15 @@
typedef struct Trait Trait;
typedef enum {
-#define O(op, pure) op,
+ OTmisc,
+ OTpre,
+ OTpost,
+ OTbin,
+ OTzarg,
+} Optype;
+
+typedef enum {
+#define O(op, pure, type, pretty) op,
#include "ops.def"
Numops,
#undef O
@@ -367,7 +375,14 @@
extern size_t nexportimpls;
extern size_t maxnid; /* the maximum node id generated so far */
-extern int ispureop[];
+/* property tables */
+extern int opispure[];
+extern char *opstr[];
+extern char *oppretty[];
+extern int opclass[];
+extern char *nodestr[];
+extern char *litstr[];
+extern char *tidstr[];
/* data structures */
Bitset *mkbs(void);
@@ -563,10 +578,6 @@
void dump(Node *t, FILE *fd);
void dumpsym(Node *s, FILE *fd);
void dumpstab(Stab *st, FILE *fd);
-char *opstr(Op o);
-char *nodestr(Ntype nt);
-char *litstr(Littype lt);
-char *tidstr(Ty tid);
/* option parsing */
void optinit(Optctx *ctx, char *optstr, char **optargs, size_t noptargs);
--- a/parse/specialize.c
+++ b/parse/specialize.c
@@ -158,7 +158,7 @@
switch (n->type) {
case Nfile:
case Nuse:
- die("Node %s not allowed here\n", nodestr(n->type));
+ die("Node %s not allowed here\n", nodestr[n->type]);
break;
case Nexpr:
fixup(n->expr.idx);
@@ -253,7 +253,7 @@
switch (n->type) {
case Nfile:
case Nuse:
- die("Node %s not allowed here\n", nodestr(n->type));
+ die("Node %s not allowed here\n", nodestr[n->type]);
break;
case Nexpr:
r->expr.op = n->expr.op;