shithub: mc

Download patch

ref: 404b1885e3cec9eeb8a2fa338174f38cbfd168ad
parent: fb11c9ffe92e56016771a9001cdf09b0e2295f7d
author: Ori Bernstein <[email protected]>
date: Mon Dec 21 19:42:44 EST 2015

Make traits work across files.

	This bumps ABI, and contains a hack.

--- a/mbld/deps.myr
+++ b/mbld/deps.myr
@@ -11,7 +11,7 @@
 	const myrdeps	: (b : build#, mt : myrtarg#, doclean : bool, addsrc : bool -> depgraph#)
 ;;
 
-const Abiversion = 8
+const Abiversion = 9
 
 var usepat	: regex.regex#
 var cflagpat	: regex.regex#
--- a/parse/infer.c
+++ b/parse/infer.c
@@ -423,9 +423,9 @@
 		t->sub[i] = tf(st, t->sub[i]);
 	base = tybase(t);
 	/* no-ops if base == t */
-	if (t->traits)
+	if (t->traits && base->traits)
 		bsunion(t->traits, base->traits);
-	else
+	else if (base->traits)
 		t->traits = bsdup(base->traits);
 	if (tyinfinite(st, t, NULL))
 		lfatal(t->loc, "type %s includes itself", tystr(t));
@@ -2448,18 +2448,16 @@
 
 	pushstab(f->file.globls);
 	/* for now, traits can only be declared globally */
-	for (i = 0; i < f->file.nstmts; i++) {
-		if (f->file.stmts[i]->type == Nimpl) {
-			n = f->file.stmts[i];
-			tr = gettrait(f->file.globls, n->impl.traitname);
-			if (!tr)
-				fatal(n, "trait %s does not exist near %s",
+	for (i = 0; i < nimpltab; i++) {
+		n = impltab[i];
+		tr = gettrait(f->file.globls, n->impl.traitname);
+		if (!tr)
+			fatal(n, "trait %s does not exist near %s",
 					namestr(n->impl.traitname), ctxstr(st, n));
-			ty = tf(st, n->impl.type);
-			settrait(ty, tr);
-			if (tr->uid == Tciter) {
-				htput(st->seqbase, tf(st, n->impl.type), tf(st, n->impl.aux[0]));
-			}
+		ty = tf(st, n->impl.type);
+		settrait(ty, tr);
+		if (tr->uid == Tciter) {
+			htput(st->seqbase, tf(st, n->impl.type), tf(st, n->impl.aux[0]));
 		}
 	}
 	popstab();
--- a/parse/node.c
+++ b/parse/node.c
@@ -220,6 +220,7 @@
 	n->impl.naux = naux;
 	n->impl.decls = decls;
 	n->impl.ndecls = ndecls;
+	lappend(&impltab, &nimpltab, n);
 	return n;
 }
 
--- a/parse/parse.h
+++ b/parse/parse.h
@@ -4,7 +4,7 @@
 #define FATAL
 #endif
 
-#define Abiversion 8
+#define Abiversion 9
 
 typedef uint8_t byte;
 typedef unsigned int uint;
@@ -399,6 +399,8 @@
 extern size_t ntypes;
 extern Trait **traittab; /* int -> trait map */
 extern size_t ntraittab;
+extern Node **impltab; /* int -> impl map */
+extern size_t nimpltab;
 extern Node **decls; /* decl id -> decl map */
 extern size_t nnodes;
 extern Node **nodes; /* node id -> node map */
--- a/parse/stab.c
+++ b/parse/stab.c
@@ -412,6 +412,14 @@
 
 static int mergeimpl(Node *old, Node *new)
 {
+	Vis vis;
+
+	if (old->impl.vis > new->impl.vis)
+		vis = old->impl.vis;
+	else
+		vis = new->impl.vis;
+	old->impl.vis = vis;
+	new->impl.vis = vis;
 	if (old->impl.isproto && !new->impl.isproto)
 		*old = *new;
 	else if (new->impl.isproto && !old->impl.isproto)
--- a/parse/type.c
+++ b/parse/type.c
@@ -24,6 +24,8 @@
 size_t ntypes;
 Trait **traittab;
 size_t ntraittab;
+Node **impltab;
+size_t nimpltab;
 
 /* Built in type constraints */
 static Trait *traits[Ntypes + 1][4];
--- a/parse/use.c
+++ b/parse/use.c
@@ -40,6 +40,9 @@
 static intptr_t *traitfixid;	/* list of traits we need to replace */
 static size_t ntraitfixid;	/* size of replacement list */
 
+static Node **implfix;	/* list of impls we need to fix up */
+static size_t nimplfix;		/* size of replacement list */
+
 void addextlibs(Node *file, char **libs, size_t nlibs)
 {
 	size_t i, j;
@@ -276,6 +279,8 @@
 	pickle(fd, tr->name);
 	typickle(fd, tr->param);
 	wrint(fd, tr->nmemb);
+	for (i = 0; i < tr->naux; i++)
+		wrtype(fd, tr->aux[i]);
 	for (i = 0; i < tr->nmemb; i++)
 		wrsym(fd, tr->memb[i]);
 	wrint(fd, tr->nfuncs);
@@ -411,6 +416,11 @@
 	tr->name = unpickle(fd);
 	tr->param = tyunpickle(fd);
 	n = rdint(fd);
+	for (i = 0; i < n; i++) {
+		lappend(&tr->aux, &tr->naux, NULL);
+		rdtype(fd, &tr->aux[i]);
+	}
+	n = rdint(fd);
 	for (i = 0; i < n; i++)
 		lappend(&tr->memb, &tr->nmemb, rdsym(fd, tr));
 	n = rdint(fd);
@@ -542,6 +552,9 @@
 		pickle(fd, n->impl.traitname);
 		wrint(fd, n->impl.trait->uid);
 		wrtype(fd, n->impl.type);
+		wrint(fd, n->impl.naux);
+		for (i = 0; i < n->impl.naux; i++)
+			wrtype(fd, n->impl.aux[i]);
 		wrint(fd, n->impl.ndecls);
 		for (i = 0; i < n->impl.ndecls; i++)
 			wrsym(fd, n->impl.decls[i]);
@@ -681,13 +694,17 @@
 		break;
 	case Nimpl:
 		n->impl.traitname = unpickle(fd);
-		i = rdint(fd);
 		rdtrait(fd, &n->impl.trait, NULL);
 		rdtype(fd, &n->impl.type);
+		n->impl.naux = rdint(fd);
+		n->impl.aux = zalloc(sizeof(Node *) * n->impl.naux);
+		for (i = 0; i < n->impl.naux; i++)
+			rdtype(fd, &n->impl.aux[i]);
 		n->impl.ndecls = rdint(fd);
 		n->impl.decls = zalloc(sizeof(Node *) * n->impl.ndecls);
 		for (i = 0; i < n->impl.ndecls; i++)
 			n->impl.decls[i] = rdsym(fd, n->impl.trait);
+		lappend(&impltab, &nimpltab, n);
 		break;
 	case Nnone:
 		die("Nnone should not be seen as node type!");
@@ -795,6 +812,38 @@
 	lfree(&traitfixid, &ntraitfixid);
 }
 
+static void protomap(Trait *tr, Type *ty, Node *dcl)
+{
+	size_t i, len;
+	char *protoname, *dclname;
+
+	dclname = declname(dcl);
+	for (i = 0; i < tr->nfuncs; i++) {
+		protoname = declname(tr->funcs[i]);
+		len = strlen(protoname);
+		if (strstr(dclname, protoname) == dclname && dclname[len] == '$')
+			htput(tr->funcs[i]->decl.impls, ty, dcl);
+	}
+}
+
+static void fiximplmappings(Stab *st)
+{
+	Node *impl;
+	Trait *tr;
+	size_t i;
+	for (i = 0; i < nimplfix; i++) {
+		impl = implfix[i];
+		tr = impl->impl.trait;
+
+		putimpl(st, impl);
+		settrait(impl->impl.type, tr);
+		for (i = 0; i < impl->impl.ndecls; i++) {
+			putdcl(file->file.globls, impl->impl.decls[i]);
+			protomap(tr, impl->impl.type, impl->impl.decls[i]);
+		}
+	}
+}
+
 /* Usefile format:
  *     U<pkgname>
  *     T<pickled-type>
@@ -898,8 +947,10 @@
 			  tr = traitunpickle(f);
 			  tr->vis = vis;
 			  puttrait(s, tr->name, tr);
-			  for (i = 0; i < tr->nfuncs; i++)
+			  for (i = 0; i < tr->nfuncs; i++) {
 				  putdcl(s, tr->funcs[i]);
+				  tr->funcs[i]->decl.impls = mkht(tyhash, tyeq);
+			  }
 			  break;
 		case 'T':
 			  tid = rdint(f);
@@ -925,10 +976,12 @@
 			  break;
 		case 'I':
 			  impl = unpickle(f);
-			  putimpl(s, impl);
 			  /* specialized declarations always go into the global stab */
-			  for (i = 0; i < impl->impl.ndecls; i++)
+			  for (i = 0; i < impl->impl.ndecls; i++) {
+				  impl->impl.decls[i]->decl.isglobl = 1;
 				  putdcl(file->file.globls, impl->impl.decls[i]);
+			  }
+			  lappend(&implfix, &nimplfix, impl);
 			  break;
 		case EOF: break;
 		}
@@ -935,6 +988,7 @@
 	}
 	fixtypemappings(s);
 	fixtraitmappings(s);
+	fiximplmappings(s);
 	htfree(tidmap);
 	popstab();
 	return 1;