-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathSimpleCalc.hs
246 lines (233 loc) · 8.61 KB
/
SimpleCalc.hs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
{-# LANGUAGE NoMonomorphismRestriction #-}
module SimpleCalc where
import qualified Data.Map as M
import qualified Data.Set as S
import Data.Maybe
import Control.Monad.State
import Control.Monad.Error
import Control.Monad.Identity
-- TODO
-- let generalization -> fully polymorphic arguments
-- normal forms for quantifiers
-- subsumption vs. unification
-- constructors/case-elimination -> records and variants
-- effects
-- friendlier printing of fixpoints
-- garbage collecting closures could help
-- won't truly stop spewing until recursion is broken by env/store split
-- implicits
-- generalize the concept of constraint propagation in abstract interpretation
type Name = Int
nm_next name = name + 1
type Env = [Value]
env_empty = []
env_lookup env name = env !! name
env_extend env value = value : env
type OrderedSet a = (Int, M.Map Int a, M.Map a Int)
ords_empty = (0, M.empty, M.empty) :: OrderedSet a
ords_count (count, _, _) = count
ords_isEmpty (0, _, _) = True
ords_isEmpty _ = False
ords_insert os@(count, index, table) el = case M.lookup el table of
Nothing -> (count + 1, M.insert count el index, M.insert el count table)
Just _ -> os
ords_concat oa ob = ords_fromList $ ords_toList oa ++ ords_toList ob
ords_lookup (_, _, table) name = fromMaybe
(error $ "out of bounds: " ++ show (name, table))
(M.lookup name table)
ords_toSet (_, _, table) = M.keysSet table
ords_toList (count, index, _) =
[fromJust (M.lookup ix index) | ix <- take count [0 ..]]
ords_fromList xs = (length xs, index, table)
where index = M.fromList $ zip [0 ..] xs
table = M.fromList $ zip xs [0 ..]
ords_diffMeet oa ob = (ords_fromList ldiff, ords_fromList lmeet)
where shared = S.intersection (ords_toSet oa) (ords_toSet ob)
la = ords_toList oa
ldiff = filter (not . (`S.member` shared)) la
lmeet = filter (`S.member` shared) la
data Term =
Ann Term Term |
Type | Arrow Term Term |
Var Name | App Term Term | Lam Term | Fix Term | Let Term Term
deriving (Show, Eq)
-- TODO: unifiable neutral terms/types
type Closure = (Env, Term)
data Value = VClo Closure | VCVar Name | VType | VArrow Value Value
deriving (Show, Eq)
bigeval (Ann term _) env = bigeval term env
bigeval Type _ = VType
bigeval (Arrow ta tb) env = VArrow (bigeval ta env) (bigeval tb env)
bigeval (Var name) env = env_lookup env name
bigeval (Lam body) env = VClo (env, body)
bigeval (Fix body) env = result
where result = bigeval body env'
env' = env_extend env result
bigeval (App proc arg) env = bigapply clo varg
where
VClo clo = bigeval proc env
varg = bigeval arg env
bigeval (Let arg body) env = bigeval body nenv
where
varg = bigeval arg env
nenv = env_extend env varg
bigapply (env, body) val = bigeval body nenv
where nenv = env_extend env val
type Typing = (Value, Constraint)
type Constraint = M.Map Name Term
cstr_empty = M.empty
cstr_assign name ty = modify $ M.insert name ty
cstr_find (VCVar name) = do
cstr <- get
case M.lookup name cstr of
Nothing -> return $ VCVar name
Just ty -> do
ty' <- cstr_find ty
cstr_assign name ty'
return ty'
cstr_find ty = return ty
cstr_occurs name ty = cstr_find ty >>= cstr_occurs' name
where
cstr_occurs' na (VCVar nb) = return $ na == nb
cstr_occurs' name (VArrow ta tb) = liftM2 (||) (occ ta) (occ tb)
where occ = cstr_occurs name
cstr_occurs' name _ = return False
cstr_unify ta tb = do
ta' <- cstr_find ta
tb' <- cstr_find tb
cstr_unify' ta' tb'
where
cstr_unify' (VCVar na) var@(VCVar nb) | na == nb = return var
cstr_unify' (VCVar name) ty = do
result <- cstr_occurs name ty
if result then do
cstr <- get
throwError $ "Occurs check: " ++ show (name, ty, cstr)
else cstr_assign name ty >> return ty
cstr_unify' other var@(VCVar _) = cstr_unify' var other
cstr_unify' VType VType = return VType
cstr_unify' (VArrow la lb) (VArrow ra rb) = do
ta <- cstr_unify la ra
tb <- cstr_unify lb rb
return $ VArrow ta tb
cstr_unify' ta tb = throwError $ "Type mismatch: " ++ show (ta, tb)
cstr_join ca cb = do
((), cstr') <- runStateT (merge cbshared) cstr
return cstr'
where
cbshared = M.toList $ M.intersection cb ca
cstr = M.union ca $ M.difference cb ca
merge = mapM_ (\(name, bval) -> cstr_unify (VCVar name) bval)
cstr_freeInType = cstr_freeInType' ords_empty
cstr_freeInType' acc ty = cstr_find ty >>= cft acc
where
cft acc var@(VCVar name) = return $ ords_insert acc name
cft acc VType = return acc
cft acc (VArrow ta tb) = inner acc ta >>= (`inner` tb)
where inner = cstr_freeInType'
cft acc (VClo (env, _)) = cstr_freeInEnv' acc env
cstr_freeInEnv = cstr_freeInEnv' ords_empty
cstr_freeInEnv' = foldM cstr_freeInType'
cstr_generalize ty env = do
(oquant, ofree) <-
liftM2 ords_diffMeet (cstr_freeInType ty) (cstr_freeInEnv env)
if ords_isEmpty oquant then return ty
else
let oall = ords_concat oquant ofree
termify ty = cstr_find ty >>= tfy
tfy (VCVar name) = return . Var $ ords_lookup oall name
tfy VType = return Type
tfy (VArrow ta tb) = liftM2 Arrow (termify ta) (termify tb)
tfy (VClo (env, body)) = error "TODO: will this ever happen?"
in do
term <- termify ty
let free = map VCVar (ords_toList ofree)
return $ VClo (free, iterate Lam term !! (ords_count oquant - 1))
data Context = Context{cxt_nextName :: Name}
deriving Show
cxt_empty = Context{cxt_nextName = 0}
cxt_freshName = do
fresh <- gets cxt_nextName
modify (\cxt -> cxt{cxt_nextName = nm_next fresh})
return . VCVar $ fresh
instantiate (VClo clo) = do
fresh <- cxt_freshName
instantiate $ bigapply clo fresh
instantiate ty = return ty
bigtype Type _ = return (VType, cstr_empty)
bigtype (Arrow ta tb) env = do
(vta, ca) <- bigtype ta env
(vtb, cb) <- bigtype tb env
cstr <- cstr_join ca cb
(_, cstr') <- runStateT (cstr_unify vta VType >> cstr_unify vtb VType) cstr
return (VType, cstr')
bigtype (Var name) env = return (env_lookup env name, cstr_empty)
bigtype (Lam body) env = do
fresh <- cxt_freshName
(tbody, cstr) <- bigtype body (fresh : env)
return (VArrow fresh tbody, cstr)
bigtype (Fix body) env = do
fresh <- cxt_freshName
(tbody, cstr) <- bigtype body (fresh : env)
runStateT (cstr_unify fresh tbody) cstr
bigtype (App proc arg) env = do
(gtproc, ca) <- bigtype proc env
(gtarg, cb) <- bigtype arg env
tproc <- instantiate gtproc
targ <- instantiate gtarg
cstr <- cstr_join ca cb
na <- cxt_freshName
nb <- cxt_freshName
(_, cstr') <- runStateT
(cstr_unify tproc (VArrow na nb) >> cstr_unify targ na) cstr
return (nb, cstr')
bigtype (Let arg body) env = do
(targ, ca) <- bigtype arg env
(gtarg, cstr) <- runStateT (cstr_generalize targ env) ca
(tbody, cb) <- bigtype body $ gtarg : env
cstr' <- cstr_join cstr cb
return (tbody, cstr')
bigtype (Ann term tyterm) env = do
(tt, ca) <- bigtype tyterm env
(tty, cb) <- bigtype term env
cstr <- cstr_join ca cb
let ty = bigeval tyterm env
runStateT (cstr_unify tt VType >> cstr_unify tty ty) cstr
-- TODO: garbage collect constraints based on env reachability
-- testing
test_id = Lam $ Var 0
test_k = Lam . Lam $ Var 1
test_false = Lam . Lam $ Var 0
-- (10 -> 10 -> 10) -> 10 -> 10 -> 10
test_fix = Fix . Lam . Lam . Lam $ Var 2 `App` (Var 3 `App` test_false `App` Var 0 `App` Var 1) `App` Var 0
-- should fail occurs check: 20 = 24 -> 23, 23 = 20
test_fixapp0 = test_fix `App` test_k `App` test_id `App` test_k
-- 21 -> 21 -> 21
test_fixapp1 = test_fix `App` test_k `App` test_false `App` test_k
fakelet val body = Lam body `App` val
-- should fail occurs check: 4 = 5 -> 4
test_fakelet = fakelet test_id $ Var 0 `App` test_id `App` (Var 0 `App` test_k)
-- 5 -> 6 -> 5
test_let = Let test_id $ Var 0 `App` test_id `App` (Var 0 `App` test_k)
test_app0 = test_id `App` test_id
test_app1 = test_k `App` test_id
test_app2 = test_k `App` test_k
test_app3 = test_app1 `App` test_k
test_app4 = test_app2 `App` test_k
tests = [test_id, test_k, test_app0, test_app1, test_app2, test_app3, test_app4]
testsf = [test_fix, test_fixapp0, test_fixapp1]
testsl = [test_fakelet, test_let]
test_evals = map (`bigeval` env_empty) tests
runInit = runIdentity . runErrorT . flip runStateT cxt_empty
runType = runInit . flip bigtype env_empty
test_types = map runType tests
test_typesf = map runType testsf
test_typesl = map runType testsl
test = forM_ test_types print
testf = forM_ test_typesf print
testl = forM_ test_typesl print
runGeneralize term = do
((ty, cstr), _) <- runType term
return $ runIdentity $ runStateT (cstr_generalize ty env_empty) cstr
test_generalize = map runGeneralize tests
testg = forM_ test_generalize print