ref: 67e077fe5811a7bc218cb58de5a32e337417268c
parent: 8bcba75efd9dd84da2ba9d9634d456d960580f68
author: Ori Bernstein <[email protected]>
date: Sun Dec 20 19:14:33 EST 2015
Iterable traits pass the frontend. The backend still needs to lower them, but this is relatively easy.
--- a/6/simp.c
+++ b/6/simp.c
@@ -449,7 +449,7 @@
* cjmp (match) :body :step
* :end
*/
-static void simpiter(Simp *s, Node *n)
+static void simpidxiter(Simp *s, Node *n)
{
Node *lbody, *lload, *lstep, *lcond, *lmatch, *lend;
Node *idx, *len, *dcl, *seq, *val, *done;
@@ -508,6 +508,36 @@
s->nloopstep--;
s->nloopexit--;
+}
+
+/* for pat in seq
+ * body;;
+ * =>
+ * .seq = seq
+ * .elt = elt
+ * :body
+ * ..body..
+ * __iterfin__(&seq, &elt)
+ * :step
+ * cond = __iternext__(&seq, &eltout)
+ * cjmp (cond) :match :end
+ * :match
+ * ...match...
+ * cjmp (match) :body :step
+ * :end
+ */
+static void simptraititer(Simp *s, Node *n)
+{
+ die("unimplemented");
+}
+
+static void simpiter(Simp *s, Node *n)
+{
+ switch (tybase(exprtype(n->iterstmt.seq))->type) {
+ case Tyarray: simpidxiter(s, n); break;
+ case Tyslice: simpidxiter(s, n); break;
+ default: simptraititer(s, n); break;
+ }
}
static Node *uconid(Simp *s, Node *n)
--- a/parse/infer.c
+++ b/parse/infer.c
@@ -40,6 +40,8 @@
/* delayed unification -- we fall back to these types in a post pass if we
* haven't unifed to something more specific */
Htab *delayed;
+ /* mappings from iterator type to element type */
+ Htab *seqbase;
/* the nodes that we've specialized them to, and the scopes they
* appear in */
Node **specializations;
@@ -786,15 +788,6 @@
}
/* Tells us if we have an index hack on the type */
-static int idxhacked(Type *a, Type *b)
-{
- if (a->type == Tyvar && a->nsub > 0)
- return 1;
- if (a->type == Tyarray || a->type == Tyslice)
- return a->type == b->type;
- return 0;
-}
-
/* prevents types that contain themselves in the unification;
* eg @a U (@a -> foo) */
static int occurs(Type *a, Type *b)
@@ -813,14 +806,15 @@
* direction should we unify. A lower ranked type
* should be mapped to the higher ranked (ie, more
* specific) type. */
-static int tyrank(Type *t)
+static int tyrank(Inferstate *st, Type *t)
{
/* plain tyvar */
- if (t->type == Tyvar && t->nsub == 0)
- return 0;
- /* parameterized tyvar */
- if (t->type == Tyvar && t->nsub > 0)
- return 1;
+ if (t->type == Tyvar) {
+ if (hthas(st->seqbase, t))
+ return 1;
+ else
+ return 0;
+ }
/* concrete type */
return 2;
}
@@ -892,11 +886,25 @@
structunify(st, ctx, u, v);
}
+static Type *basetype(Inferstate *st, Type *a)
+{
+ Type *t;
+
+ if (a->nsub == 1)
+ t = a->sub[0];
+ else
+ t = htget(st->seqbase, a);
+ if (t)
+ t = tf(st, t);
+ return t;
+}
+
/* Unifies two types, or errors if the types are not unifiable. */
static Type *unify(Inferstate *st, Node *ctx, Type *u, Type *v)
{
Type *t, *r;
Type *a, *b;
+ Type *ea, *eb;
char *from, *to;
size_t i;
@@ -907,7 +915,7 @@
return a;
/* we unify from lower to higher ranked types */
- if (tyrank(b) < tyrank(a)) {
+ if (tyrank(st, b) < tyrank(st, a)) {
t = a;
a = b;
b = t;
@@ -917,6 +925,8 @@
from = tystr(a);
to = tystr(b);
indentf(st->indentdepth, "Unify %s => %s\n", from, to);
+ indentf(st->indentdepth + 1, "indexes: %s => %s\n",
+ tystr(htget(st->seqbase, a)), tystr(htget(st->seqbase, b)));
free(from);
free(to);
}
@@ -924,6 +934,10 @@
r = NULL;
if (a->type == Tyvar) {
tytab[a->tid] = b;
+ ea = basetype(st, a);
+ eb = basetype(st, b);
+ if (ea && eb)
+ unify(st, ctx, ea, eb);
r = b;
}
@@ -935,7 +949,7 @@
/* if the tyrank of a is 0 (ie, a raw tyvar), just unify.
* Otherwise, match up subtypes. */
- if ((a->type == b->type || idxhacked(a, b)) && tyrank(a) != 0) {
+ if (a->type == b->type && a->type != Tyvar) {
if (hasparam(a) && hasparam(b)) {
/* Only Tygeneric and Tyname should be able to unify. And they
* should have the same names for this to be true. */
@@ -957,8 +971,7 @@
for (i = 0; i < b->nsub; i++)
unify(st, ctx, a->sub[i], b->sub[i]);
r = b;
- }
- else if (a->type != Tyvar) {
+ } else if (a->type != Tyvar) {
typeerror(st, a, b, ctx, NULL);
}
mergetraits(st, ctx, a, b);
@@ -1312,7 +1325,7 @@
Node **args;
size_t i, nargs;
Node *s, *n;
- Type *t;
+ Type *t, *b;
int isconst;
n = *np;
@@ -1416,19 +1429,23 @@
break;
case Oidx: /* @a[@b::tcint] -> @a */
infersub(st, n, ret, sawret, &isconst);
- t = mktyidxhack(n->loc, mktyvar(n->loc));
+ b = mktyvar(n->loc);
+ t = mktyvar(n->loc);
+ htput(st->seqbase, t, b);
unify(st, n, type(st, args[0]), t);
constrain(st, n, type(st, args[0]), traittab[Tcidx]);
constrain(st, n, type(st, args[1]), traittab[Tcint]);
- settype(st, n, t->sub[0]);
+ settype(st, n, b);
break;
case Oslice: /* @a[@b::tcint,@b::tcint] -> @a[,] */
infersub(st, n, ret, sawret, &isconst);
- t = mktyidxhack(n->loc, mktyvar(n->loc));
+ b = mktyvar(n->loc);
+ t = mktyvar(n->loc);
+ htput(st->seqbase, t, b);
unify(st, n, type(st, args[0]), t);
constrain(st, n, type(st, args[1]), traittab[Tcint]);
constrain(st, n, type(st, args[2]), traittab[Tcint]);
- settype(st, n, mktyslice(n->loc, t->sub[0]));
+ settype(st, n, mktyslice(n->loc, b));
break;
/* special cases */
@@ -1693,7 +1710,7 @@
{
size_t i, nbound;
Node **bound, *n, *pat;
- Type *t;
+ Type *t, *b;
n = *np;
if (!n)
@@ -1755,10 +1772,12 @@
infernode(st, &n->iterstmt.seq, NULL, sawret);
infernode(st, &n->iterstmt.body, ret, sawret);
- t = mktyidxhack(n->loc, mktyvar(n->loc));
+ b = mktyvar(n->loc);
+ t = mktyvar(n->loc);
+ htput(st->seqbase, t, b);
constrain(st, n, type(st, n->iterstmt.seq), traittab[Tciter]);
unify(st, n, type(st, n->iterstmt.seq), t);
- unify(st, n, type(st, n->iterstmt.elt), t->sub[0]);
+ unify(st, n, type(st, n->iterstmt.elt), b);
break;
case Nmatchstmt:
infernode(st, &n->matchstmt.val, NULL, sawret);
@@ -2422,7 +2441,7 @@
{
size_t i;
Node *n;
- Trait *trait;
+ Trait *tr;
Type *ty;
pushstab(f->file.globls);
@@ -2430,12 +2449,15 @@
for (i = 0; i < f->file.nstmts; i++) {
if (f->file.stmts[i]->type == Nimpl) {
n = f->file.stmts[i];
- trait = gettrait(f->file.globls, n->impl.traitname);
- if (!trait)
+ 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, trait);
+ settrait(ty, tr);
+ if (tr->uid == Tciter) {
+ htput(st->seqbase, tf(st, n->impl.type), tf(st, n->impl.aux[0]));
+ }
}
}
popstab();
@@ -2467,6 +2489,7 @@
assert(file->type == Nfile);
st.delayed = mkht(tyhash, tyeq);
+ st.seqbase = mkht(tyhash, tyeq);
/* set up the symtabs */
loaduses(file);
// mergeexports(&st, file);
--- a/parse/parse.h
+++ b/parse/parse.h
@@ -522,7 +522,6 @@
Type *mktyunres(Srcloc l, Node *name, Type **params, size_t nparams);
Type *mktyarray(Srcloc l, Type *base, Node *sz);
Type *mktyslice(Srcloc l, Type *base);
-Type *mktyidxhack(Srcloc l, Type *base);
Type *mktyptr(Srcloc l, Type *base);
Type *mktytuple(Srcloc l, Type **sub, size_t nsub);
Type *mktyfunc(Srcloc l, Node **args, size_t nargs, Type *ret);
--- a/parse/type.c
+++ b/parse/type.c
@@ -237,17 +237,6 @@
return t;
}
-Type *mktyidxhack(Srcloc loc, Type *base)
-{
- Type *t;
-
- t = mktype(loc, Tyvar);
- t->nsub = 1;
- t->sub = xalloc(sizeof(Type *));
- t->sub[0] = base;
- return t;
-}
-
Type *mktyptr(Srcloc loc, Type *base)
{
Type *t;