summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorGravatar Adam Chlipala <adam@chlipala.net>2010-08-21 10:58:13 -0400
committerGravatar Adam Chlipala <adam@chlipala.net>2010-08-21 10:58:13 -0400
commit1bd7feb39add77e14df852d2087902049f2b5df3 (patch)
treef8f49a6d9be3065b6863399d7108b99a354762a1
parent458bed0042d7bb206138ba6dce98686b056ae5c0 (diff)
Some post-type-checking support for polymorphic variants
-rw-r--r--src/monoize.sml51
-rw-r--r--src/settings.sml3
2 files changed, 33 insertions, 21 deletions
diff --git a/src/monoize.sml b/src/monoize.sml
index a7f790c3..85ff1905 100644
--- a/src/monoize.sml
+++ b/src/monoize.sml
@@ -53,6 +53,7 @@ structure RM = BinaryMapFn(struct
val nextPvar = ref 0
val pvars = ref (RM.empty : (int * (string * int * L'.typ) list) RM.map)
val pvarDefs = ref ([] : L'.decl list)
+val pvarOldDefs = ref ([] : (int * (string * int * L.con option) list) list)
fun choosePvar () =
let
@@ -62,17 +63,20 @@ fun choosePvar () =
n
end
-fun pvar (r, loc) =
- case RM.find (!pvars, r) of
+fun pvar (r, r', loc) =
+ case RM.find (!pvars, r') of
NONE =>
let
val n = choosePvar ()
- val fs = map (fn (x, t) => (x, choosePvar (), t)) r
- val fs' = foldl (fn ((x, n, _), fs') => SM.insert (fs', x, n)) SM.empty fs
+ val fs = map (fn (x, t) => (x, choosePvar (), t)) r'
+ val (r, fs') = ListPair.foldr (fn ((_, t), (x, n, _), (r, fs')) =>
+ ((x, n, SOME t) :: r,
+ SM.insert (fs', x, n))) ([], SM.empty) (r, fs)
in
- pvars := RM.insert (!pvars, r, (n, fs));
+ pvars := RM.insert (!pvars, r', (n, fs));
pvarDefs := (L'.DDatatype [("$poly" ^ Int.toString n, n, map (fn (x, n, t) => (x, n, SOME t)) fs)], loc)
:: !pvarDefs;
+ pvarOldDefs := (n, r) :: !pvarOldDefs;
(n, fs)
end
| SOME v => v
@@ -158,9 +162,9 @@ fun monoType env =
| L.CApp ((L.CFfi ("Basis", "variant"), _), (L.CRecord ((L.KType, _), xts), _)) =>
let
- val xts = map (fn (x, t) => (monoName env x, mt env dtmap t)) xts
- val xts = ListMergeSort.sort (fn ((x, _), (y, _)) => String.compare (x, y) = GREATER) xts
- val (n, cs) = pvar (xts, loc)
+ val xts' = map (fn (x, t) => (monoName env x, mt env dtmap t)) xts
+ val xts' = ListMergeSort.sort (fn ((x, _), (y, _)) => String.compare (x, y) = GREATER) xts'
+ val (n, cs) = pvar (xts, xts', loc)
val cs = map (fn (x, n, t) => (x, n, SOME t)) cs
in
(L'.TDatatype (n, ref (ElabUtil.classifyDatatype cs, cs)), loc)
@@ -816,21 +820,21 @@ fun monoExp (env, st, fm) (all as (e, loc)) =
| L.ECApp (
(L.ECApp (
- (L.ECApp ((L.EFfi ("Basis", "make"), _), (L.CName nm, _)), _),
+ (L.ECApp ((L.EFfi ("Basis", "make"), _), nmC as (L.CName nm, _)), _),
t), _),
(L.CRecord (_, xts), _)) =>
let
- val t = monoType env t
- val xts = map (fn (x, t) => (monoName env x, monoType env t)) xts
- val xts = (nm, t) :: xts
- val xts = ListMergeSort.sort (fn ((x, _), (y, _)) => String.compare (x, y) = GREATER) xts
- val (n, cs) = pvar (xts, loc)
+ val t' = monoType env t
+ val xts' = map (fn (x, t) => (monoName env x, monoType env t)) xts
+ val xts' = (nm, t') :: xts'
+ val xts' = ListMergeSort.sort (fn ((x, _), (y, _)) => String.compare (x, y) = GREATER) xts'
+ val (n, cs) = pvar ((nmC, t) :: xts, xts', loc)
val cs' = map (fn (x, n, t) => (x, n, SOME t)) cs
val cl = ElabUtil.classifyDatatype cs'
in
case List.find (fn (nm', _, _) => nm' = nm) cs of
NONE => raise Fail "Monoize: Polymorphic variant tag mismatch for 'make'"
- | SOME (_, n', _) => ((L'.EAbs ("x", t, (L'.TDatatype (n, ref (cl, cs')), loc),
+ | SOME (_, n', _) => ((L'.EAbs ("x", t', (L'.TDatatype (n, ref (cl, cs')), loc),
(L'.ECon (cl, L'.PConVar n', SOME (L'.ERel 0, loc)), loc)), loc),
fm)
end
@@ -840,12 +844,12 @@ fun monoExp (env, st, fm) (all as (e, loc)) =
t) =>
let
val t = monoType env t
- val xts = map (fn (x, t) => (monoName env x, monoType env t)) xts
- val xts = ListMergeSort.sort (fn ((x, _), (y, _)) => String.compare (x, y) = GREATER) xts
- val (n, cs) = pvar (xts, loc)
+ val xts' = map (fn (x, t) => (monoName env x, monoType env t)) xts
+ val xts' = ListMergeSort.sort (fn ((x, _), (y, _)) => String.compare (x, y) = GREATER) xts'
+ val (n, cs) = pvar (xts, xts', loc)
val cs' = map (fn (x, n, t) => (x, n, SOME t)) cs
val cl = ElabUtil.classifyDatatype cs'
- val fs = (L'.TRecord (map (fn (x, t') => (x, (L'.TFun (t', t), loc))) xts), loc)
+ val fs = (L'.TRecord (map (fn (x, t') => (x, (L'.TFun (t', t), loc))) xts'), loc)
val dt = (L'.TDatatype (n, ref (cl, cs')), loc)
in
((L'.EAbs ("v",
@@ -4099,16 +4103,23 @@ fun monoize env file =
end
| _ =>
(pvarDefs := [];
+ pvarOldDefs := [];
case monoDecl (env, fm) d of
NONE => (env, fm, ds)
| SOME (env, fm, ds') =>
- (env,
+ (foldr (fn ((n, cs), env) =>
+ Env.declBinds env (L.DDatatype [("$poly" ^ Int.toString n,
+ n,
+ [],
+ cs)], loc))
+ env (!pvarOldDefs),
Fm.enter fm,
ds' @ Fm.decls fm @ !pvarDefs @ ds)))
(env, Fm.empty mname, []) file
in
pvars := RM.empty;
pvarDefs := [];
+ pvarOldDefs := [];
rev ds
end
diff --git a/src/settings.sml b/src/settings.sml
index 967efe07..40f2ff20 100644
--- a/src/settings.sml
+++ b/src/settings.sml
@@ -72,7 +72,8 @@ val clientToServerBase = basis ["int",
"unit",
"option",
"list",
- "bool"]
+ "bool",
+ "variant"]
val clientToServer = ref clientToServerBase
fun setClientToServer ls = clientToServer := S.addList (clientToServerBase, ls)
fun mayClientToServer x = S.member (!clientToServer, x)