shithub: mc

Download patch

ref: cc5182284db2880dcfc3b0bd61dbc6ed319bacc3
parent: 32abfabfaf95942558a590a307432b499f392ea9
author: Ori Bernstein <[email protected]>
date: Sun Feb 10 18:03:36 EST 2019

Pass the type envs through properly in delayed unification.

--- a/parse/infer.c
+++ b/parse/infer.c
@@ -16,7 +16,7 @@
 #include "parse.h"
 
 typedef struct Traitmap Traitmap;
-typedef struct Enttype Enttype;
+typedef struct Postcheck Postcheck;
 
 struct Traitmap {
 	Bitset *traits;
@@ -29,6 +29,13 @@
 	size_t nfiltertr;
 };
 
+struct Postcheck {
+	Node *node;
+	Stab *stab;
+	Tyenv *env;
+};
+
+
 int allowhidden;
 
 static void infernode(Node **np, Type *ret, int *sawret);
@@ -53,10 +60,8 @@
 
 /* post-inference checking/unification */
 static Htab *delayed;
-static Node **postcheck;
+static Postcheck **postcheck;
 static size_t npostcheck;
-static Stab **postcheckscope;
-static size_t npostcheckscope;
 
 /* generic declarations to be specialized */
 static Node **genericdecls;
@@ -298,10 +303,15 @@
 }
 
 static void
-delayedcheck(Node *n, Stab *s)
+delayedcheck(Node *n)
 {
-	lappend(&postcheck, &npostcheck, n);
-	lappend(&postcheckscope, &npostcheckscope, s);
+	Postcheck *pc;
+
+	pc = xalloc(sizeof(Postcheck));
+	pc->node = n;
+	pc->stab = curstab();
+	pc->env = curenv();
+	lappend(&postcheck, &npostcheck, pc);
 }
 
 static void
@@ -1356,7 +1366,7 @@
 	n->expr.isconst = s->decl.isconst;
 	if (param) {
 		n->expr.param = param;
-		delayedcheck(n, curstab());
+		delayedcheck(n);
 	}
 	if (s->decl.isgeneric && !ingeneric) {
 		t = tyfreshen(NULL, t);
@@ -1363,7 +1373,7 @@
 		addspecialization(n, curstab());
 		if (t->type == Tyvar) {
 			settype(n, mktyvar(n->loc));
-			delayedcheck(n, curstab());
+			delayedcheck(n);
 		} else {
 			settype(n, t);
 		}
@@ -1414,7 +1424,7 @@
 
 	*isconst = 1;
 	/* we want to check outer nodes before inner nodes when unifying nested structs */
-	delayedcheck(n, curstab());
+	delayedcheck(n);
 	for (i = 0; i < n->expr.nargs; i++) {
 		infernode(&n->expr.args[i], NULL, NULL);
 		if (!n->expr.args[i]->expr.isconst)
@@ -1778,7 +1788,7 @@
 	case Omemb:	/* @a.Ident -> @b, verify type(@a.Ident)==@b later */
 		infersub(n, ret, sawret, &isconst);
 		settype(n, mktyvar(n->loc));
-		delayedcheck(n, curstab());
+		delayedcheck(n);
 		break;
 	case Osize:	/* sizeof(@a) -> size */
 		infersub(n, ret, sawret, &isconst);
@@ -1790,7 +1800,7 @@
 		break;
 	case Ocast:	/* (@a : @b) -> @b */
 		infersub(n, ret, sawret, &isconst);
-		delayedcheck(n, curstab());
+		delayedcheck(n);
 		break;
 	case Oret:	/* -> @a -> void */
 		infersub(n, ret, sawret, &isconst);
@@ -1827,7 +1837,9 @@
 			fatal(n, "undeclared var %s", ctxstr(args[0]));
 		if (n->expr.param && !s->decl.trait)
 			fatal(n, "var %s must refer to a trait decl", ctxstr(args[0]));
+		pushenv(s->decl.env);
 		initvar(n, s);
+		popenv(s->decl.env);
 		break;
 	case Ogap:	/* _ -> @a */
 		if (n->expr.type)
@@ -2123,7 +2135,7 @@
 			unify(n, e, b);
 		else
 			t->seqaux = e;
-		delayedcheck(n, curstab());
+		delayedcheck(n);
 		break;
 	case Nmatchstmt:
 		infernode(&n->matchstmt.val, NULL, sawret);
@@ -2267,7 +2279,7 @@
 }
 
 static void
-checkcast(Node *n, Node ***rem, size_t *nrem, Stab ***remscope, size_t *nremscope)
+checkcast(Node *n, Postcheck ***pc, size_t *npc)
 {
 	/* FIXME: actually verify the casts. Right now, it's ok to leave thi
 	 * unimplemented because bad casts get caught by the backend. */
@@ -2274,16 +2286,14 @@
 }
 
 static void
-infercompn(Node *n, Node ***rem, size_t *nrem, Stab ***remscope, size_t *nremscope)
+infercompn(Node *n, Postcheck ***post, size_t *npost)
 {
-	Node *aggr;
-	Node *memb;
-	Node **nl;
-	Type *t;
-	size_t i;
-	int found;
-	int ismemb;
+	Node *aggr, *memb, **nl;
+	int found, ismemb;
+	Postcheck *pc;
 	uvlong idx;
+	size_t i;
+	Type *t;
 
 	aggr = n->expr.args[0];
 	memb = n->expr.args[1];
@@ -2302,11 +2312,11 @@
 		if (tybase(t)->type == Typtr)
 			t = tybase(tf(t->sub[0]));
 		if (tybase(t)->type == Tyvar) {
-			if (!rem)
-				fatal(n, "underspecified type defined on %s:%d used near %s",
-						fname(t->loc), lnum(t->loc), ctxstr(n));
-			lappend(rem, nrem, n);
-			lappend(remscope, nremscope, curstab());
+			pc = xalloc(sizeof(Postcheck));
+			pc->node = n;
+			pc->stab = curstab();
+			pc->env = curenv();
+			lappend(post, npost, pc);
 			return;
 		}
 		if (ismemb) {
@@ -2349,11 +2359,12 @@
 }
 
 static void
-checkstruct(Node *n, Node ***rem, size_t *nrem, Stab ***remscope, size_t *nremscope)
+checkstruct(Node *n, Postcheck ***post, size_t *npost)
 {
 	Type *t, *et;
 	Node *val, *name;
 	size_t i, j;
+	Postcheck *pc;
 
 	t = tybase(tf(n->lit.type));
 	/*
@@ -2379,23 +2390,22 @@
 			}
 		}
 
-		if (!et) {
-			if (rem) {
-				lappend(rem, nrem, n);
-				lappend(remscope, nremscope, curstab());
-				return;
-			} else {
-				fatal(n, "could not find member %s in struct %s, near %s",
-						namestr(name), tystr(t), ctxstr(n));
-			}
+		if (et) {
+			unify(val, et, type(val));
+		} else {
+			assert(post != NULL);
+			pc = xalloc(sizeof(Postcheck));
+			pc->node = n;
+			pc->stab = curstab();
+			pc->env = curenv();
+			lappend(post, npost, pc);
+			return;
 		}
-
-		unify(val, et, type(val));
 	}
 }
 
 static void
-checkvar(Node *n, Node ***rem, size_t *nrem, Stab ***remscope, size_t *nremscope)
+checkvar(Node *n, Postcheck ***post, size_t *npost)
 {
 	Node *proto, *dcl;
 	Type *ty;
@@ -2403,6 +2413,7 @@
 	proto = decls[n->expr.did];
 	ty = NULL;
 	dcl = NULL;
+	pushenv(proto->decl.env);
 	if (n->expr.param)
 		dcl = htget(proto->decl.impls, tf(n->expr.param));
 	if (dcl)
@@ -2410,6 +2421,7 @@
 	if (!ty)
 		ty = tyfreshen(NULL, type(proto));
 	unify(n, type(n), ty);
+	popenv(proto->decl.env);
 }
 
 static void
@@ -2450,26 +2462,30 @@
 }
 
 static void
-postcheckpass(Node ***rem, size_t *nrem, Stab ***remscope, size_t *nremscope)
+postcheckpass(Postcheck ***post, size_t *npost)
 {
 	size_t i;
 	Node *n;
 
 	for (i = 0; i < npostcheck; i++) {
-		n = postcheck[i];
-		pushstab(postcheckscope[i]);
+		n = postcheck[i]->node;
+		pushstab(postcheck[i]->stab);
+		pushenv(postcheck[i]->env);
 		if (n->type == Nexpr) {
 			switch (exprop(n)) {
 			case Otupmemb:
-			case Omemb:	infercompn(n, rem, nrem, remscope, nremscope);	break;
-			case Ocast:	checkcast(n, rem, nrem, remscope, nremscope);	break;
-			case Ostruct:	checkstruct(n, rem, nrem, remscope, nremscope);	break;
-			case Ovar:	checkvar(n, rem, nrem, remscope, nremscope);	break;
-			default:	die("should not see %s in postcheck\n", opstr[exprop(n)]);
+			case Omemb:	infercompn(n, post, npost);	break;
+			case Ocast:	checkcast(n, post, npost);	break;
+			case Ostruct:	checkstruct(n, post, npost);	break;
+			case Ovar:	checkvar(n, post, npost);	break;
+			default:
+				die("should not see %s in postcheck\n", opstr[exprop(n)]);
+				break;
 			}
 		} else if (n->type == Niterstmt) {
 			fixiter(n, type(n->iterstmt.seq), type(n->iterstmt.elt));
 		}
+		popenv(postcheck[i]->env);
 		popstab();
 	}
 }
@@ -2477,23 +2493,20 @@
 static void
 postinfer(void)
 {
-	size_t nrem, nremscope;
-	Stab **remscope;
-	Node **rem;
+	Postcheck **post, **last;
+	size_t npost, nlast;
 
+	/* Iterate until we reach a fixpoint. */
+	nlast = 0;
 	while (1) {
-		remscope = NULL;
-		nremscope = 0;
-		rem = NULL;
-		nrem = 0;
-		postcheckpass(&rem, &nrem, &remscope, &nremscope);
-		if (nrem == npostcheck) {
+		post = NULL;
+		npost = 0;
+		postcheckpass(&post, &npost);
+		if (npost == nlast) {
 			break;
 		}
-		postcheck = rem;
-		npostcheck = nrem;
-		postcheckscope = remscope;
-		npostcheckscope = nremscope;
+		last = post;
+		nlast = npost;
 	}
 }
 
--- a/parse/specialize.c
+++ b/parse/specialize.c
@@ -430,6 +430,7 @@
 	case Ndecl:
 		r->decl.did = ndecls;
 		/* sym */
+		r->decl.env = n->decl.env;
 		r->decl.name = specializenode(n->decl.name, tsmap);
 		r->decl.type = tysubst(n->decl.type, tsmap);
 
@@ -442,13 +443,17 @@
 			putdcl(curstab(), r);
 
 		/* init */
+		pushenv(r->decl.env);
 		r->decl.init = specializenode(n->decl.init, tsmap);
+		popenv(r->decl.env);
 		lappend(&decls, &ndecls, r);
 		break;
 	case Nfunc:
 		r->func.scope = mkstab(1);
 		r->func.scope->super = curstab();
+		r->func.env = n->func.env;
 		pushstab(r->func.scope);
+		pushenv(r->func.env);
 		r->func.type = tysubst(n->func.type, tsmap);
 		r->func.nargs = n->func.nargs;
 		r->func.args = xalloc(sizeof(Node *) * n->func.nargs);
@@ -455,6 +460,7 @@
 		for (i = 0; i < n->func.nargs; i++)
 			r->func.args[i] = specializenode(n->func.args[i], tsmap);
 		r->func.body = specializenode(n->func.body, tsmap);
+		popenv(r->func.env);
 		popstab();
 		break;
 	case Nimpl:
--- a/parse/type.c
+++ b/parse/type.c
@@ -834,7 +834,6 @@
 	switch (a->type) {
 	case Typaram:
 		ret = (a == b);
-		/* FIXME: this streq check needs to go */
 		ret = ret || streq(a->pname, b->pname);
 		break;
 	case Tyvar: