ref: 91ad7689b0645c5bf0b860412c96d10016d80293
parent: f0f1654937b0bd8944189c15c76169ca02ff88e9
author: Ori Bernstein <[email protected]>
date: Sat Feb 6 10:41:43 EST 2016
Work towards generic iterables. Keep track of base types.
--- a/lib/std/htab.myr
+++ b/lib/std/htab.myr
@@ -25,6 +25,13 @@
generic htgetv : (ht : htab(@k, @v)#, k : @k, fallback : @v-> @v)
generic hthas : (ht : htab(@k, @v)#, k : @k -> bool)
generic htkeys : (ht : htab(@k, @v)# -> @k[:])
+
+ generic htbykeyvals : (ht : htab(@k, @v)# -> htkviter(@k, @v))
+ impl iterable htkviter(@k, @v) -> (@k, @v)
+ type htkviter(@k, @v) = struct
+ ht : std.htab(@k, @v)#
+ idx : size
+ ;;
;;
const Initsz = 32
@@ -202,3 +209,29 @@
-> keys
}
+generic htbykeyvals = {ht
+ -> [.ht = ht, .idx = 0]
+}
+
+impl iterable htkviter(@k, @v) -> (@k, @v) =
+ __iternext__ = {itp, valp
+ var i, ht
+
+ ht = itp.ht
+ for i = itp.idx; i < ht.keys.len; i++
+ if ht.hashes[i] != 0 && !ht.dead[i]
+ break
+ ;;
+ ;;
+ itp.idx = i
+ if i < ht.hashes.len
+ valp# = (ht.keys[i], ht.vals[i])
+ -> true
+ else
+ -> false
+ ;;
+ }
+
+ __iterfin__ = {itp, valp
+ }
+;;
--- a/parse/infer.c
+++ b/parse/infer.c
@@ -359,10 +359,10 @@
tybind(st, t);
if (!subst) {
subst = mksubst();
- t = tyspecialize(t, subst, st->delayed);
+ t = tyspecialize(t, subst, st->delayed, st->seqbase);
substfree(subst);
} else {
- t = tyspecialize(t, subst, st->delayed);
+ t = tyspecialize(t, subst, st->delayed, st->seqbase);
}
tyunbind(st, t);
if (debugopt['u']) {
@@ -1695,7 +1695,7 @@
substput(subst, t->param, n->impl.type);
for (j = 0; j < t->naux; j++)
substput(subst, t->aux[j], n->impl.aux[j]);
- ty = tyspecialize(type(st, proto), subst, st->delayed);
+ ty = tyspecialize(type(st, proto), subst, st->delayed, NULL);
substfree(subst);
inferdecl(st, dcl);
--- a/parse/parse.h
+++ b/parse/parse.h
@@ -628,7 +628,7 @@
void substpush(Tysubst *subst);
void substpop(Tysubst *subst);
Node *specializedcl(Node *n, Type *to, Node **name);
-Type *tyspecialize(Type *t, Tysubst *tymap, Htab *delayed);
+Type *tyspecialize(Type *t, Tysubst *tymap, Htab *delayed, Htab *tybase);
Node *genericname(Node *n, Type *t);
void geninit(Node *file);
--- a/parse/specialize.c
+++ b/parse/specialize.c
@@ -82,9 +82,10 @@
* parameters (type schemes in most literature)
* replaced with type variables that we can unify
* against */
-Type *tyspecialize(Type *orig, Tysubst *tsmap, Htab *delayed)
+Type *tyspecialize(Type *orig, Tysubst *tsmap, Htab *delayed, Htab *trbase)
{
- Type *t, *ret, *tmp, **arg, *var;
+ Type *t, *ret, *tmp, *var, *base;
+ Type **arg;
size_t i, narg;
t = tysearch(orig);
@@ -108,8 +109,8 @@
substput(tsmap, t->gparam[i], orig->arg[i]);
}
for (i = 0; i < t->ngparam; i++)
- lappend(&arg, &narg, tyspecialize(t->gparam[i], tsmap, delayed));
- ret = mktyname(t->loc, t->name, tyspecialize(t->sub[0], tsmap, delayed));
+ lappend(&arg, &narg, tyspecialize(t->gparam[i], tsmap, delayed, trbase));
+ ret = mktyname(t->loc, t->name, tyspecialize(t->sub[0], tsmap, delayed, trbase));
if (orig->type == Tyunres)
substpop(tsmap);
ret->issynth = 1;
@@ -123,11 +124,15 @@
var = mktyvar(t->loc);
substput(tsmap, t, var);
for (i = 0; i < t->narg; i++)
- lappend(&arg, &narg, tyspecialize(t->arg[i], tsmap, delayed));
- ret = mktyname(t->loc, t->name, tyspecialize(t->sub[0], tsmap, delayed));
+ lappend(&arg, &narg, tyspecialize(t->arg[i], tsmap, delayed, trbase));
+ ret = mktyname(t->loc, t->name, tyspecialize(t->sub[0], tsmap, delayed, trbase));
ret->traits = bsdup(t->traits);
ret->arg = arg;
ret->narg = narg;
+ if (trbase && hthas(trbase, orig) && !hthas(trbase, ret)) {
+ base = htget(trbase, orig);
+ htput(trbase, ret, tyspecialize(base, tsmap, delayed, trbase));
+ }
tytab[var->tid] = ret;
break;
case Tystruct:
@@ -144,7 +149,7 @@
for (i = 0; i < t->nmemb; i++) {
tmp = NULL;
if (ret->udecls[i]->etype)
- tmp = tyspecialize(t->udecls[i]->etype, tsmap, delayed);
+ tmp = tyspecialize(t->udecls[i]->etype, tsmap, delayed, trbase);
ret->udecls[i] = mkucon(t->loc, t->udecls[i]->name, ret, tmp);
ret->udecls[i]->utype = ret;
ret->udecls[i]->id = i;
@@ -155,7 +160,7 @@
ret = t;
if (delayed && hthas(delayed, t)) {
tmp = htget(delayed, t);
- htput(delayed, ret, tyspecialize(tmp, tsmap, delayed));
+ htput(delayed, ret, tyspecialize(tmp, tsmap, delayed, trbase));
}
break;
default:
@@ -163,7 +168,7 @@
ret = tydup(t);
substput(tsmap, t, ret);
for (i = 0; i < t->nsub; i++)
- ret->sub[i] = tyspecialize(t->sub[i], tsmap, delayed);
+ ret->sub[i] = tyspecialize(t->sub[i], tsmap, delayed, trbase);
} else {
ret = t;
}
@@ -178,7 +183,7 @@
static Type *tysubst(Type *t, Tysubst *tsmap)
{
if (hasparams(t))
- return tyspecialize(t, tsmap, NULL);
+ return tyspecialize(t, tsmap, NULL, NULL);
else
return t;
}