ref: f991cf4a837be41c1e4972e06066815cdd67b140
parent: d2a28cc46d5101760733f1b831287f6b9fcf1be1
author: Ori Bernstein <[email protected]>
date: Mon May 9 11:39:26 EDT 2016
Avoid infinite recursion when priting bad types. We sometimes error on bad types. Let's not crash when we do that.
--- a/parse/type.c
+++ b/parse/type.c
@@ -31,7 +31,7 @@
static Htab *tydeduptab;
/* Built in type constraints */
static Trait *traits[Ntypes + 1][4];
-static int tybfmt(char *buf, size_t len, Type *t);
+static int tybfmt(char *buf, size_t len, Bitset *visted, Type *t);
char stackness[] = {
#define Ty(t, n, stk) stk,
@@ -465,11 +465,12 @@
return p - buf;
}
-static int fmtstruct(char *buf, size_t len, Type *t)
+static int fmtstruct(char *buf, size_t len, Bitset *visited, Type *t)
{
size_t i;
char *end, *p;
- char *name, *ty;
+ char *name;
+ char subbuf[512];
p = buf;
end = p + len;
@@ -476,19 +477,19 @@
p += bprintf(p, end - p, "struct\n");
for (i = 0; i < t->nmemb; i++) {
name = declname(t->sdecls[i]);
- ty = tystr(decltype(t->sdecls[i]));
- p += bprintf(p, end - p, "\t%s:%s\n ", name, ty);
- free(ty);
+ tybfmt(subbuf, sizeof subbuf, visited, decltype(t->sdecls[i]));
+ p += bprintf(p, end - p, "\t%s:%s\n ", name, subbuf);
}
p += bprintf(p, end - p, ";;");
return p - buf;
}
-static int fmtunion(char *buf, size_t len, Type *t)
+static int fmtunion(char *buf, size_t len, Bitset *visited, Type *t)
{
size_t i;
char *end, *p;
- char *name, *ty;
+ char *name;
+ char subbuf[512];
p = buf;
end = p + len;
@@ -496,9 +497,8 @@
for (i = 0; i < t->nmemb; i++) {
name = namestr(t->udecls[i]->name);
if (t->udecls[i]->etype) {
- ty = tystr(t->udecls[i]->etype);
- p += bprintf(p, end - p, "\t`%s %s\n", name, ty);
- free(ty);
+ tybfmt(subbuf, sizeof subbuf, visited, t->udecls[i]->etype);
+ p += bprintf(p, end - p, "\t`%s %s\n", name, subbuf);
} else {
p += bprintf(p, end - p, "\t`%s\n", name);
}
@@ -507,7 +507,7 @@
return p - buf;
}
-static int fmtlist(char *buf, size_t len, Type **arg, size_t narg)
+static int fmtlist(char *buf, size_t len, Bitset *visited, Type **arg, size_t narg)
{
char *end, *p, *sep;
size_t i;
@@ -518,7 +518,7 @@
p += bprintf(p, end - p, "(");
for (i = 0; i < narg; i++) {
p += bprintf(p, end - p, "%s", sep);
- p += tybfmt(p, end - p, arg[i]);
+ p += tybfmt(p, end - p, visited, arg[i]);
sep = ", ";
}
p += bprintf(p, end - p, ")");
@@ -525,7 +525,7 @@
return p - buf;
}
-static int tybfmt(char *buf, size_t len, Type *t)
+static int tybfmt(char *buf, size_t len, Bitset *visited, Type *t)
{
size_t i;
char *p;
@@ -534,13 +534,21 @@
if (t)
t = tysearch(t);
+
sep = "";
p = buf;
end = p + len;
+
if (!t) {
p += bprintf(p, end - p, "tynil");
return len - (end - p);
}
+
+ if (bshas(visited, t->tid)) {
+ p += bprintf(buf, len, "<...>");
+ return len - (end - p);
+ }
+ bsput(visited, t->tid);
switch (t->type) {
case Tybad: p += bprintf(p, end - p, "BAD"); break;
case Tyvoid: p += bprintf(p, end - p, "void"); break;
@@ -563,15 +571,15 @@
case Tyvar: p += bprintf(p, end - p, "$%d", t->tid); break;
case Typtr:
- p += tybfmt(p, end - p, t->sub[0]);
+ p += tybfmt(p, end - p, visited, t->sub[0]);
p += bprintf(p, end - p, "#");
break;
case Tyslice:
- p += tybfmt(p, end - p, t->sub[0]);
+ p += tybfmt(p, end - p, visited, t->sub[0]);
p += bprintf(p, end - p, "[:]");
break;
case Tyarray:
- p += tybfmt(p, end - p, t->sub[0]);
+ p += tybfmt(p, end - p, visited, t->sub[0]);
if (t->asize) {
i = t->asize->expr.args[0]->lit.intval;
p += bprintf(p, end - p, "[%zd]", i);
@@ -584,11 +592,11 @@
p += bprintf(p, end - p, "(");
for (i = 1; i < t->nsub; i++) {
p += bprintf(p, end - p, "%s", sep);
- p += tybfmt(p, end - p, t->sub[i]);
+ p += tybfmt(p, end - p, visited, t->sub[i]);
sep = ", ";
}
p += bprintf(p, end - p, " -> ");
- p += tybfmt(p, end - p, t->sub[0]);
+ p += tybfmt(p, end - p, visited, t->sub[0]);
p += bprintf(p, end - p, ")");
break;
case Tytuple:
@@ -595,7 +603,7 @@
p += bprintf(p, end - p, "(");
for (i = 0; i < t->nsub; i++) {
p += bprintf(p, end - p, "%s", sep);
- p += tybfmt(p, end - p, t->sub[i]);
+ p += tybfmt(p, end - p, visited, t->sub[i]);
sep = ",";
}
p += bprintf(p, end - p, ")");
@@ -604,7 +612,7 @@
case Tyunres:
p += namefmt(p, end - p, t->name);
if (t->narg)
- p += fmtlist(p, end - p, t->arg, t->narg);
+ p += fmtlist(p, end - p, visited, t->arg, t->narg);
break;
case Tyname:
if (t->name->name.ns)
@@ -611,7 +619,7 @@
p += bprintf(p, end - p, "%s.", t->name->name.ns);
p += bprintf(p, end - p, "%s", namestr(t->name));
if (t->narg)
- p += fmtlist(p, end - p, t->arg, t->narg);
+ p += fmtlist(p, end - p, visited, t->arg, t->narg);
break;
case Tygeneric:
if (t->name->name.ns)
@@ -618,10 +626,10 @@
p += bprintf(p, end - p, "%s.", t->name->name.ns);
p += bprintf(p, end - p, "%s", namestr(t->name));
if (t->ngparam)
- p += fmtlist(p, end - p, t->gparam, t->ngparam);
+ p += fmtlist(p, end - p, visited, t->gparam, t->ngparam);
break;
- case Tystruct: p += fmtstruct(p, end - p, t); break;
- case Tyunion: p += fmtunion(p, end - p, t); break;
+ case Tystruct: p += fmtstruct(p, end - p, visited, t); break;
+ case Tyunion: p += fmtunion(p, end - p, visited, t); break;
case Ntypes: die("Ntypes is not a type"); break;
}
@@ -634,7 +642,11 @@
char *tyfmt(char *buf, size_t len, Type *t)
{
- tybfmt(buf, len, t);
+ Bitset *bs;
+
+ bs = mkbs();
+ tybfmt(buf, len, bs, t);
+ bsfree(bs);
return buf;
}