ref: 128470e404d843b3d71c5837cac05d4d4203bd66
parent: fb8753c942a8d06926977d221d5c89c3a867fa29
author: Ori Bernstein <[email protected]>
date: Sun Jan 31 17:18:17 EST 2016
Add support for generic impls. You can now implement generic shit like iterators.
--- a/6/simp.c
+++ b/6/simp.c
@@ -507,7 +507,7 @@
name = declname(tr->funcs[i]);
if (!strcmp(fn, name)) {
proto = tr->funcs[i];
- dcl = htget(proto->decl.impls, ty);
+ dcl = htget(proto->decl.__impls, ty);
var = mkexpr(loc, Ovar, dcl->decl.name, NULL);
var->expr.type = codetype(dcl->decl.type);
var->expr.did = dcl->decl.did;
--- a/parse/gram.y
+++ b/parse/gram.y
@@ -399,7 +399,7 @@
0);
for (i = 0; i < $6.nn; i++) {
$6.nl[i]->decl.trait = $$;
- $6.nl[i]->decl.impls = mkht(tyhash, tyeq);
+ $6.nl[i]->decl.__impls = mkht(tyhash, tyeq);
$6.nl[i]->decl.isgeneric = 1;
}
}
--- a/parse/infer.c
+++ b/parse/infer.c
@@ -809,8 +809,6 @@
return 2;
}
-static int hasparam(Type *t) { return t->type == Tyname && t->narg > 0; }
-
static void unionunify(Inferstate *st, Node *ctx, Type *u, Type *v)
{
size_t i, j;
@@ -914,6 +912,10 @@
tystr(a), tystr(b), ctxstr(st, ctx));
}
+static int hasargs(Type *t)
+{
+ return t->type == Tyname && t->narg > 0;
+}
/* Unifies two types, or errors if the types are not unifiable. */
static Type *unify(Inferstate *st, Node *ctx, Type *u, Type *v)
@@ -972,7 +974,7 @@
/* if the tyrank of a is 0 (ie, a raw tyvar), just unify.
* Otherwise, match up subtypes. */
if (a->type == b->type && a->type != Tyvar) {
- if (hasparam(a) && hasparam(b)) {
+ if (hasargs(a) && hasargs(b)) {
/* Only Tygeneric and Tyname should be able to unify. And they
* should have the same names for this to be true. */
if (!nameeq(a->name, b->name))
@@ -1684,9 +1686,6 @@
namestr(t->name), ctxstr(st, n));
/* infer and unify types */
- if (n->impl.type->type == Tygeneric || n->impl.type->type == Typaram)
- fatal(n, "trait specialization requires concrete type, got %s",
- tystr(n->impl.type));
verifytraits(st, n, t->param, n->impl.type);
subst = mksubst();
substput(subst, t->param, n->impl.type);
@@ -1707,7 +1706,13 @@
fname(sym->loc), lnum(sym->loc));
dcl->decl.name = name;
putdcl(file->file.globls, dcl);
- htput(proto->decl.impls, n->impl.type, dcl);
+ htput(proto->decl.__impls, n->impl.type, dcl);
+ dcl->decl.isconst = 1;
+ if (n->impl.type->type == Tygeneric || hasparams(n->impl.type)) {
+ dcl->decl.isgeneric = 1;
+ lappend(&proto->decl.gimpl, &proto->decl.ngimpl, dcl);
+ lappend(&proto->decl.gtype, &proto->decl.ngtype, ty);
+ }
if (debugopt['S'])
printf("specializing trait [%d]%s:%s => %s:%s\n", n->loc.line,
namestr(proto->decl.name), tystr(type(st, proto)), namestr(name),
@@ -2071,7 +2076,7 @@
ty = NULL;
dcl = NULL;
if (n->expr.param)
- dcl = htget(proto->decl.impls, tf(st, n->expr.param));
+ dcl = htget(proto->decl.__impls, tf(st, n->expr.param));
if (dcl)
ty = dcl->decl.type;
if (!ty)
--- a/parse/parse.h
+++ b/parse/parse.h
@@ -328,9 +328,13 @@
impl.
*/
Trait *trait;
- Htab *impls;
- char vis;
+ Htab *__impls;
+ Node **gimpl; /* generic impls of this trait */
+ size_t ngimpl;
+ Node **gtype; /* generic impls of this trait */
+ size_t ngtype;
+ char vis;
/* flags */
char isglobl;
char isconst;
--- a/parse/specialize.c
+++ b/parse/specialize.c
@@ -317,7 +317,9 @@
r = mknode(n->loc, n->type);
switch (n->type) {
case Nfile:
- case Nuse: die("Node %s not allowed here\n", nodestr[n->type]); break;
+ case Nuse:
+ die("Node %s not allowed here\n", nodestr[n->type]);
+ break;
case Nexpr:
r->expr.op = n->expr.op;
r->expr.type = tysubst(n->expr.type, tsmap);
@@ -447,6 +449,107 @@
return name;
}
+/* this doesn't walk through named types, so it can't recurse infinitely. */
+int matchquality(Type *pat, Type *to)
+{
+ int match, q;
+ size_t i;
+ Ucon *puc, *tuc;
+
+ if (pat->type == Typaram)
+ return 0;
+ else if (pat->type != to->type)
+ return -1;
+
+ match = 0;
+ switch (pat->type) {
+ case Tystruct:
+ if (pat->nmemb != to->nmemb)
+ return -1;
+ for (i = 0; i < pat->nmemb; i++) {
+ if (!streq(declname(pat->sdecls[i]),declname( to->sdecls[i])))
+ return -1;
+ q = matchquality(decltype(pat->sdecls[i]), decltype(to->sdecls[i]));
+ if (q < 0)
+ return -1;
+ match += q;
+ }
+ break;
+ case Tyunion:
+ if (pat->nmemb != to->nmemb)
+ return -1;
+ for (i = 0; i < pat->nmemb; i++) {
+ if (!nameeq(pat->udecls[i], to->udecls[i]))
+ return -1;
+ puc = pat->udecls[i];
+ tuc = to->udecls[i];
+ if (puc->etype && tuc->etype) {
+ q = matchquality(puc->etype, tuc->etype);
+ if (q < 0)
+ return -1;
+ match += q;
+ } else if (puc->etype != tuc->etype) {
+ return -1;
+ }
+ }
+ break;
+ case Tyname:
+ if (!nameeq(pat->name, to->name))
+ return -1;
+ if (pat->narg != to->narg)
+ return -1;
+ for (i = 0; i < pat->narg; i++) {
+ q = matchquality(pat->sub[i], to->sub[i]);
+ if (q < 0)
+ return -1;
+ match += q;
+ }
+ break;
+ default:
+ if (pat->nsub != to->nsub)
+ break;
+ if (pat->type == to->type)
+ match = 1;
+ for (i = 0; i < pat->nsub; i++) {
+ q = matchquality(pat->sub[i], to->sub[i]);
+ if (q < 0)
+ return -1;
+ match += q;
+ }
+ break;
+ }
+ return match;
+}
+
+Node *bestimpl(Node *n, Type *to)
+{
+ Node **gimpl, **ambig, *match;
+ size_t ngimpl, nambig, i;
+ int score;
+ int best;
+
+ ambig = NULL;
+ nambig = 0;
+ best = -1;
+ gimpl = n->decl.gimpl;
+ ngimpl = n->decl.ngimpl;
+ for (i = 0; i < ngimpl; i++) {
+ score = matchquality(decltype(gimpl[i]), to);
+ if (score > best) {
+ lfree(&ambig, &nambig);
+ best = score;
+ match = gimpl[i];
+ } else if (score == best) {
+ lappend(&ambig, &nambig, gimpl[i]);
+ }
+ }
+ if (best < 0)
+ return NULL;
+ else if (nambig > 0)
+ fatal(n, "ambiguous implementation for %s", tystr(to));
+ return match;
+}
+
/*
* Takes a generic declaration, and creates a specialized
* duplicate of it with type 'to'. It also generates
@@ -471,15 +574,16 @@
if (!st)
fatal(n, "Can't find symbol table for %s.%s", n->name.ns, n->name.name);
d = getdcl(st, n);
- if (debugopt['S'])
- printf("depth[%d] specializing [%d]%s => %s\n", stabstkoff, g->loc.line,
- namestr(g->decl.name), namestr(n));
if (d)
return d;
if (g->decl.trait) {
- printf("%s\n", namestr(n));
- fatal(g, "no trait implemented for for %s:%s", namestr(g->decl.name), tystr(to));
+ g = bestimpl(g, to);
+ if (!g)
+ fatal(g, "no trait implemented for for %s:%s", namestr(g->decl.name), tystr(to));
}
+ if (debugopt['S'])
+ printf("depth[%d] specializing [%d]%s => %s\n", stabstkoff, g->loc.line,
+ namestr(g->decl.name), namestr(n));
/* namespaced names need to be looked up in their correct
* context. */
if (n->name.ns)
--- a/parse/type.c
+++ b/parse/type.c
@@ -856,7 +856,7 @@
func = mkdecl(Zloc, mkname(Zloc, "__iternext__"), ty);
func->decl.trait = tr;
- func->decl.impls = mkht(tyhash, tyeq);
+ func->decl.__impls = mkht(tyhash, tyeq);
func->decl.isgeneric = 1;
func->decl.isconst = 1;
func->decl.isglobl = 1;
@@ -876,7 +876,7 @@
func = mkdecl(Zloc, mkname(Zloc, "__iterfin__"), ty);
func->decl.trait = tr;
- func->decl.impls = mkht(tyhash, tyeq);
+ func->decl.__impls = mkht(tyhash, tyeq);
func->decl.isgeneric = 1;
func->decl.isconst = 1;
func->decl.isglobl = 1;
--- a/parse/use.c
+++ b/parse/use.c
@@ -813,14 +813,20 @@
{
size_t i, len;
char *protoname, *dclname, *p;
+ Node *proto;
dclname = declname(dcl);
for (i = 0; i < tr->nfuncs; i++) {
- protoname = declname(tr->funcs[i]);
+ proto = tr->funcs[i];
+ protoname = declname(proto);
len = strlen(protoname);
p = strstr(dclname, protoname);
if (p && p[len] == '$')
- htput(tr->funcs[i]->decl.impls, ty, dcl);
+ htput(proto->decl.__impls, ty, dcl);
+ if (ty->type == Tygeneric || hasparams(ty)) {
+ lappend(&proto->decl.gimpl, &proto->decl.ngimpl, dcl);
+ lappend(&proto->decl.gtype, &proto->decl.ngtype, ty);
+ }
}
}
@@ -948,7 +954,7 @@
puttrait(s, tr->name, tr);
for (i = 0; i < tr->nfuncs; i++) {
putdcl(s, tr->funcs[i]);
- tr->funcs[i]->decl.impls = mkht(tyhash, tyeq);
+ tr->funcs[i]->decl.__impls = mkht(tyhash, tyeq);
}
break;
case 'T':
--- /dev/null
+++ b/test/genericimpl.myr
@@ -1,0 +1,33 @@
+use std
+
+trait gimpl @a =
+ desc : (x : @a -> byte[:])
+;;
+
+impl gimpl int =
+ desc = {x
+ -> "int"
+ }
+;;
+
+impl gimpl byte[:] =
+ desc = {x
+ -> "string"
+ }
+;;
+
+impl gimpl @a =
+ desc = {x
+ -> "@a"
+ }
+;;
+
+impl gimpl @b[:] =
+ desc = {x
+ -> "@a[:]"
+ }
+;;
+
+const main = {
+ std.put("{} {} {} {}\n", desc(123), desc("foo"), desc('x'), desc([true][:]))
+}
--- a/test/tests
+++ b/test/tests
@@ -93,6 +93,7 @@
B genericret E 42
B genericmatch E 15
B genericrec E 0
+B genericimpl P 'int string @a @a[:]'
B recgeneric P 'built'
B bigtyblob P '`U100 '
# B genericchain P 'val = 123' ## BUGGERED