forked from xanxys/hs2bf
-
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathCore.hs
391 lines (304 loc) · 12.6 KB
/
Core.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
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
-- | parametric variable:
-- Partially type-annotated
-- * kind-inference -> possible kind error
-- Fully-kind-annotated -> throw away kind
-- * type-inference -> possible type error
-- Fully-type-annotated
--
-- * type-inference
--
-- * uniquify
--
-- * dependency-analysis(convert letrec to let)
--
-- * MFE-detection
--
-- * lambda lifting
--
-- are done in Core language
module Core where
import Control.Arrow
import Control.Monad.Writer
import Data.Ord
import Data.Char
import Data.List
import qualified Data.Map as M
import qualified Data.Set as S
import qualified Data.Foldable as F
import Data.Sequence((><),(|>),(<|))
import qualified Data.Sequence as Q
import Util as U hiding(Pack)
import qualified Util as U
import GMachine
type LocHint=String
data Core=Core [CrData] [CrProc]
data CrData=CrData CrName [CrName] [(CrName,[(Bool,CrType)])] deriving(Show)
data CrProc=CrProc CrName [CrName] CrExpr deriving(Show)
library=M.fromList
[("undefined",[UError "undefined"])
,("addByteRaw",f AAdd)
,("subByteRaw",f ASub)
,("cmpByteRaw",f CCmp)
,("seq",[PushArg 1,Push 0,Reduce RAny,Update 1,Pop 1,PushArg 2,Slide 3])
]
where f op=[PushArg 2,PushArg 2,Arith op,Slide 3]
-- | Rename all variables to be unique in each 'CrProc'.
uniquify :: Core -> Core
uniquify (Core ds ps)=Core ds $ map (uniquifyP m0 r0) ps
where
r0=""
m0=M.fromList $ zip gs gs
gs=concatMap (\(CrData _ _ cons)->map fst cons) ds++map (\(CrProc n _ _)->n) ps
uniquifyP :: M.Map CrName CrName -> String -> CrProc -> CrProc
uniquifyP m r (CrProc n as e)=CrProc n (map (m' M.!) as) $ uniquifyE m' r e
where m'=bind r m as
uniquifyE :: M.Map CrName CrName -> String -> CrExpr -> CrExpr
uniquifyE m r (CrVar v)=CrVar $ M.findWithDefault (error $ "uniquifyE:"++v) v m
uniquifyE m r (CrApp e0 e1)=CrApp (uniquifyE m n1 e0) (uniquifyE m n2 e1)
where [n1,n2]=branch 2 r
uniquifyE m r (CrCstr t es)=CrCstr t $ zipWith (uniquifyE m) ns es
where ns=branch (length es) r
uniquifyE m r (CrCase e cs)=CrCase (uniquifyE m n e) (zipWith f cs ns)
where
n:ns=branch (length cs+1) r
f (tag,vs,e) n=let m'=bind r m vs in (tag,map (m' M.!) vs,uniquifyE m' n e)
uniquifyE m r (CrLet flag bs e)=CrLet flag (zipWith f bs ns) (uniquifyE m' n e)
where
m'=bind r m $ map fst bs
n:ns=branch (length bs+1) r
f (v,e) n=(m' M.! v,uniquifyE (if flag then m' else m) n e)
uniquifyE m r (CrLm vs e)=CrLm (map (m' M.!) vs) $ uniquifyE m' r e
where m'=bind r m vs
uniquifyE m r e@(CrByte _)=e
branch :: Int -> String -> [String]
branch n r=map ((r++) . (:[])) ss
where ss=take n $ iterate succ 'a'
bind :: String -> M.Map CrName CrName -> [CrName] -> M.Map CrName CrName
bind r m vs=M.union (M.fromList $ zip vs vs') m
where vs'=map ((++r) . (++"_")) vs
liftLambdaW :: Core -> Core
liftLambdaW (Core ds ps)=Core ds $ concatMap liftLambda ps
liftLambda :: CrProc -> [CrProc]
liftLambda (CrProc n args e)=CrProc n args e':ps
where (e',ps)=runWriter (liftl ("_l_"++n) e)
liftl :: String -> CrExpr -> Writer [CrProc] CrExpr
liftl n e0@(CrLm as e)=do
liftl (n++"_") e >>= post . CrProc n (fvs++as)
return $! multiApp (CrVar n) $ map CrVar fvs
where fvs=S.toList $ S.filter (not . isUpper . head) $ freeVar e0
liftl n (CrLet flag bs e)=liftM2 (CrLet flag) (mapM f bs) (liftl (n++"_") e)
where f (v,e)=liftM (\x->(v,x)) $ liftl (n++"_"++v) e
liftl n (CrCase e cs)=liftM2 CrCase (liftl (n++"_") e) (mapM f cs)
where f (t,vs,e)=liftM (\x->(t,vs,x)) $ liftl (n++"_"++t) e
liftl n (CrApp e0 e1)=liftM2 CrApp (liftl n e0) (liftl (n++"_") e1)
liftl n (CrCstr tag es)=liftM (CrCstr tag) (zipWithM f es [0..])
where f e k=liftl (n++"_"++show k) e
liftl n e=return e
post :: a -> Writer [a] ()
post=tell . (:[])
freeVar :: CrExpr -> S.Set CrName
freeVar e=collectV e `S.difference` collectB e
collectB :: CrExpr -> S.Set CrName
collectB (CrApp e0 e1)=collectB e0 `S.union` collectB e1
collectB (CrLet _ bs e)=S.fromList (map fst bs) `S.union` (S.unions $ map collectB $ e:map snd bs)
collectB (CrCase e cs)=S.fromList (concatMap snd3 cs) `S.union` (S.unions $ map collectB $ e:map thr3 cs)
collectB (CrLm as e)=S.fromList as `S.union` collectB e
collectB _=S.empty
collectV :: CrExpr -> S.Set CrName
collectV (CrVar x)=S.singleton x
collectV (CrApp e0 e1)=collectV e0 `S.union` collectV e1
collectV (CrLet _ bs e)=S.unions $ map collectV $ e:map snd bs
collectV (CrCase e cs)=S.unions $ map collectV $ e:map thr3 cs
collectV (CrLm as e)=collectV e
collectV (CrByte _)=S.empty
collectV e=error $ "collectV: "++show e
multiApp :: CrExpr -> [CrExpr] -> CrExpr
multiApp=foldl CrApp
optimize (Core ds ps)=Core ds (map (\(CrProc n as e)->CrProc n as $ optLetVar e) ps)
-- | If rhs of let binder is a variable, remove it from let.
optLetVar (CrLet False bs e)
|null bsN = e'
|otherwise = CrLet False bsN e'
where
e'=optLetVar $ replaceVar t e
t=M.fromList $ map (second $ \(CrVar x)->x) bsS
isVar (CrVar _)=True
isVar _=False
(bsS,bsN)=partition (isVar . snd) bs
optLetVar (CrLet True bs e)=CrLet True bs $ optLetVar e
optLetVar (CrApp e0 e1)=CrApp (optLetVar e0) (optLetVar e1)
optLetVar (CrCase e cs)=CrCase (optLetVar e) (map (\(tag,vs,e)->(tag,vs,optLetVar e)) cs)
optLetVar e=e
replaceVar :: M.Map CrName CrName -> CrExpr -> CrExpr
replaceVar t (CrVar x)=CrVar $ M.findWithDefault x x t
replaceVar t (CrApp e0 e1)=CrApp (replaceVar t e0) (replaceVar t e1)
replaceVar t (CrCase e cs)=CrCase (replaceVar t e) (map (\(tag,vs,e)->(tag,vs,replaceVar t e)) cs)
replaceVar t (CrLet f bs e)=CrLet f (map (second $ replaceVar t) bs) $ replaceVar t e
replaceVar t e=e
compile :: Core -> Process (M.Map String [GMCode])
compile (Core ds ps)=return $ M.union library $ M.fromList (map (compileP m) (ps++pds))
where
m=M.fromList cons
(pds,cons)=unzip $ concatMap convertData ds
-- | Convert one data declaration to procs and cons.
convertData :: CrData -> [(CrProc,(String,Int))]
convertData (CrData _ _ cs)=zipWith convertDataCon [0..] cs
-- | Int argument is a tag, not an arity
convertDataCon :: Int -> (CrName,[(Bool,CrType)]) -> (CrProc,(String,Int))
convertDataCon t (name,xs)=(CrProc name (map snd args) exp,(name,t))
where
exp=foldr (\v e->multiApp (CrVar "seq") [v,e]) con $ map (CrVar . snd) sarg
con=CrCstr t $ map (CrVar . snd) args
sarg=filter (fst . fst) args
args=zip xs $ stringSeq "#d"
-- | Resolve default clause in 'Case' and 'uniquify'.
simplify :: Core -> Process Core
simplify (Core ds ps)=return $ liftLambdaW $ optimize $ uniquify $ Core ds $ map (smplP table) ps
where
table=M.fromList $ concatMap (mkP . map snd) $ groupBy (equaling fst) $ concatMap conCT ds
mkP xs=map (\x->(fst x,S.fromList xs)) xs
conCT :: CrData -> [(CrName,(CrName,Int))]
conCT (CrData n _ xs)=zip (repeat n) (map (second length) xs)
smplP :: M.Map String (S.Set (String,Int)) -> CrProc -> CrProc
smplP t (CrProc name args expr)=CrProc name args $ smplE t expr
smplE :: M.Map String (S.Set (String,Int)) -> CrExpr -> CrExpr
smplE t (CrApp e0 e1)=CrApp (smplE t e0) (smplE t e1)
smplE t (CrCstr tag es)=CrCstr tag $ map (smplE t) es
smplE t (CrLet f bs e)=CrLet f (map (second $ smplE t) bs) $ smplE t e
smplE t (CrLm vs e)=CrLm vs $ smplE t e
smplE t (CrCase ec cs)
|null cocs = CrCase (smplE t ec) $ nrmcons
|length cocs==1 = CrCase (smplE t ec) $ cocons (thr3 $ head cocs)++nrmcons
|otherwise = error "smplE: more than 2 defaults!"
where
(cocs,nrmcs)=partition (null . fst3) cs
nrmcons=map (\(x,y,z)->(x,y,smplE t z)) nrmcs
cocons x=map (\(c,n)->(c,replicate n "",smplE t x)) $ F.toList s
s=S.difference (M.findWithDefault (error "smplE") (fst $ head cons) t) (S.fromList cons)
cons=filter (not . null . fst) $ map (\(x,y,_)->(x,length y)) cs
smplE t x=x
-- | Compile one super combinator to 'GMCode'
--
-- requirement:
--
-- * must not contain lambda
--
compileP :: M.Map String Int -> CrProc -> (String,[GMCode])
compileP mc (CrProc name args expr)=
(name,F.toList $ compileE mc mv expr><Q.fromList [Update $ n+1,Pop n])
where
n=length args
mv=M.fromList $ zip args (map PushArg [1..])
compileE :: M.Map String Int -> M.Map String GMCode -> CrExpr -> Q.Seq GMCode
compileE mc mv (CrApp e0 e1)=(compileE mc mv e1 >< compileE mc (shift mv 1) e0) |> MkApp
compileE mc mv (CrVar v)=Q.singleton $ maybe (PushSC v) id $ M.lookup v mv
compileE mc mv (CrByte x)=Q.singleton $ PushByte x
compileE mc mv (CrCstr t es)=
concatS (zipWith (compileE mc) (map (shift mv) [0..]) (reverse es)) |> Pack t (length es)
compileE mc mv (CrCase ec cs)=compileE mc mv ec |> Reduce RAny |> Case (map f cs)
where
f (con,vs,e)=(M.findWithDefault (error $ "cE:not found:"++con) con mc
,F.toList $
(UnPack (length vs) <|
compileE mc (insMV $ reverse vs) e) |>
Slide (length vs)
)
insMV vs=M.union (M.fromList $ zip vs (map Push [0..])) $ shift mv $ length vs
compileE mc mv (CrLet False bs e)=
concatS (zipWith (compileE mc) (map (shift mv) [0..]) (map snd $ reverse bs)) ><
compileE mc mv' e ><
Q.fromList [Slide n]
where
n=length bs
mv'=M.union (M.fromList $ zip (map fst bs) (map Push [0..])) $ shift mv n
compileE mc mv (CrLet True bs e)=
Q.fromList [Alloc n] ><
concatS (map (compileE mc mv' . snd) $ reverse bs) ><
compileE mc mv' e ><
Q.fromList [Slide n]
where
n=length bs
mv'=M.union (M.fromList $ zip (map fst bs) (map Push [0..])) $ shift mv n
compileE mc mv (CrLm _ _)=error "compileE: lambda must be lifted beforehand"
concatS :: [Q.Seq a] -> Q.Seq a
concatS=foldr (><) Q.empty
shift :: M.Map String GMCode -> Int -> M.Map String GMCode
shift m d=M.map f m
where
f (Push n)=Push $ n+d
f (PushArg n)=PushArg $ n+d
-- | Pretty printer for 'Core'
pprint :: Core -> String
pprint (Core ds ps)=compileSB $ Group $ intersperse EmptyLine $ map pprintData ds++map pprintProc ps
pprintData :: CrData -> SBlock
pprintData (CrData name xs cons)=Group
[Line $ Span [Prim "data",Prim name]
,Indent $ Group $ zipWith cv cons ("=":repeat "|")]
where cv (name,xs) eq=Line $ Span [U.Pack [Prim eq,Prim name],Prim $ show $ length xs]
pprintProc :: CrProc -> SBlock
pprintProc (CrProc n as e)=Group
[Line $ U.Pack [Span $ map Prim $ n:as,Prim "="]
,Indent $ pprintExpr e]
pprintExpr :: CrExpr -> SBlock
pprintExpr (CrCase e as)=Group
[Line $ Span [Prim "case",pprintExprI e,Prim "of"]
,Indent $ Group $ map cv as]
where
cv (con,vs,e)=Group [Line $ Span $ Prim con:map Prim vs++[Prim "->"],Indent $ pprintExpr e]
pprintExpr (CrLet flag binds e)=Group
[Line $ Prim $ if flag then "letrec" else "let"
,Indent $ Group $ map (\(v,e)->Line $ Span [Prim v,Prim "=",pprintExprI e]) binds
,Line $ Prim "in"
,Indent $ pprintExpr e]
pprintExpr x=Line $ pprintExprI x
pprintExprI :: CrExpr -> IBlock
pprintExprI (CrLm ns e)=U.Pack $
[U.Pack [Prim "\\",Span $ map Prim ns]
,U.Pack [Prim "->",pprintExprI e]]
pprintExprI (CrVar x)=Prim x
pprintExprI (CrCase e as)=Span $
[Span [Prim "case",pprintExprI e,Prim "of"],Span $ map cv as]
where
cv (con,vs,e)=Span [Span $ Prim con:map Prim vs,Prim "->",pprintExprI e,Prim ";"]
pprintExprI (CrLet flag binds e)=Span $
[Span $ (Prim $ if flag then "letrec" else "let"):map cv binds
,Prim "in"
,pprintExprI e]
where cv (v,e)=U.Pack [Prim v,Prim "=",pprintExprI e,Prim ";"]
pprintExprI (CrApp e0 e1)=U.Pack [Prim "(",Span [pprintExprI e0,pprintExprI e1],Prim ")"]
pprintExprI (CrByte n)=Prim $ show n
-- pprintExpr f (Cr
pprintExprI e=error $ "pprintExprI:"++show e
{-
checkKind :: [CrData CrKind] -> Maybe [(CrName,CrKind)]
checkKind []=Just []
checkKind (CrData name vars cons)=Nothing
-}
-- | kind
data CrKind
=CrKiApp CrKind CrKind -- ^ left associative application of types
|CrKiX -- ^ the kind of proper types, /*/
instance Show CrKind where
show (CrKiApp k0 k1)="("++show k0++") -> ("++show k1++")"
show CrKiX="*"
-- | type
data CrType
=CrTyApp CrType CrType
|CrTyVar CrName -- ex.: x,y,z
|CrTyCon CrName -- ex.: #A,#L,#T,#Byte,Integer
instance Show CrType where
show (CrTyApp t0 t1)="("++show t0++") -> ("++show t1++")"
show (CrTyVar x)=x
show (CrTyCon x)=x
-- | expression
data CrExpr
=CrLm [CrName] CrExpr
|CrApp CrExpr CrExpr
|CrLet Bool [(CrName,CrExpr)] CrExpr -- ^ rec?
|CrCstr Int [CrExpr]
|CrCase CrExpr [(String,[CrName],CrExpr)]
|CrVar CrName
|CrByte Int
deriving(Show)
-- | identifier
type CrName=String