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
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
|
module Verify.WithSMT
where
import Control.Monad ( unless, when )
import Data.Either ( partitionEithers, rights )
import Data.IORef
import Data.List ( (\\), find, init, isPrefixOf, last, maximum, nub
, partition, union )
import Data.Maybe ( catMaybes, fromMaybe, isJust )
import Numeric ( readHex )
import System.CPUTime ( getCPUTime )
import System.Directory ( doesFileExist )
import Debug.Trace
import Control.Monad.Trans.State
import Data.Tuple.Extra ( both )
import FlatCurry.AddTypes
import FlatCurry.Build
import FlatCurry.Goodies
import FlatCurry.Names
import FlatCurry.Print
import FlatCurry.Simplify
import FlatCurry.Types as FC
import System.FilePath ( (</>) )
import System.IOExts ( evalCmd )
import Text.Pretty ( pPrint, pretty, text )
import Verify.ESMT as SMT
import Verify.Helpers
import Verify.Options
import Verify.ProgInfo
import PackageConfig ( getPackagePath )
checkUnsatisfiabilityWithSMT :: Options -> QName -> String -> IORef ProgInfo
-> [(QName,ConsInfo)] -> [(Int,TypeExpr)] -> Expr -> IO (Maybe Bool)
checkUnsatisfiabilityWithSMT opts qf scripttitle pistore consinfos
vartypes assertionexp0 = do
let assertionexp = wrapCaseWithId assertionexp0
(cnames,fnames) = both nub
(partitionEithers (allQNamesInExp assertionexp))
cnamesreq = filter (`notElem` [pre "False", pre "True", anonCons]) cnames
loadModulesForQNames opts pistore (cnamesreq ++ fnames)
pinfos <- fmap progInfos $ readIORef pistore
let typedassertion = addTypes2VarExp pinfos vartypes assertionexp fcBool
let foassertexp = replaceHigherOrderInExp (simpExpr typedassertion)
foassertfuncs = rights (allQNamesInExp foassertexp)
maybe
(transExpError typedassertion)
(\ (assertion,varsorts) ->
catch (checkUnsatWithSMT opts qf scripttitle pistore consinfos vartypes
varsorts assertion cnamesreq foassertfuncs)
(\e -> print e >> return Nothing))
(exp2SMTWithVars (maximum (0 : map fst vartypes))
Nothing foassertexp)
where
transExpError e = do putStrLn $ "Cannot translate expression:\n" ++ showExp e
return Nothing
wrapCaseWithId =
updCases (\ct e brs -> Case ct (Comb FuncCall (pre "id") [e]) brs)
checkUnsatWithSMT :: Options -> QName -> String -> IORef ProgInfo
-> [(QName,ConsInfo)]
-> [(Int,TypeExpr)] -> [(Int,Sort)]
-> Term -> [QName] -> [QName] -> IO (Maybe Bool)
checkUnsatWithSMT opts qf title pistore consinfos vartypes
extravars
assertion
assertioncons
assertfuncs =
flip catch (\e -> print e >> return Nothing) $ do
unless (null assertfuncs) $ printWhenDetails opts $
"Translating operations into SMT: " ++ unwords (map showQName assertfuncs)
fdecls0 <- getAllFunctions opts pistore assertfuncs
pinfos <- fmap progInfos $ readIORef pistore
let fdecls1 = addTypes2FuncDecls pinfos
(map (completeBranchesInFunc consinfos True) fdecls0)
fdecls = usedFunctions assertfuncs
(map (updFuncBody replaceHigherOrderInExp) fdecls1)
let smtfuncs = maybe (Comment $ "ERROR translating " ++
show (map funcName fdecls))
(\fds -> DefineSigsRec fds)
(mapM fun2SMT fdecls)
fdecltvars = nub (concatMap allTVarsInFuncDecl fdecls)
let vartypestcons = foldr union [] (map (tconsOfTypeExpr . snd) vartypes)
funcstcons = foldr union [] (map (tconsOfTypeExpr . funcType) fdecls)
asserttcons = map (\(ConsType _ tc _) -> tc)
(map (snd3 . infoOfCons consinfos) assertioncons)
(primtypes,usertypes) =
partition ((== "Prelude") . fst)
(union funcstcons (union vartypestcons asserttcons))
decls <- mapM (getTypeDeclOf pistore) usertypes
let tvarsInVars = foldr union []
(map (typeParamsOfSort . type2sort)
(map snd vartypes ++ map TVar fdecltvars))
varsorts = map (\(i,te) -> (i, type2sort te)) vartypes ++ extravars
let smt = concatMap preludeType2SMT (map snd primtypes) ++
[ EmptyLine ] ++
(if null decls
then []
else [ Comment "User-defined datatypes:" ] ++
map tdecl2SMT decls) ++
(if null tvarsInVars
then []
else [ EmptyLine, Comment "Polymorphic sorts:" ] ++
map (\tv -> DeclareSort tv 0) tvarsInVars) ++
[ EmptyLine, smtfuncs, EmptyLine
, Comment "Free variables:" ] ++
map (\(i,s) -> DeclareVar (SV i s)) varsorts ++
[ EmptyLine
, Comment "Boolean formula of assertion (known properties):"
, sAssert assertion, EmptyLine
, Comment "check satisfiability:"
, CheckSat
]
let preludesmt = if all ((`elem` defaultSMTTypes) . snd) primtypes
then "Prelude_min.smt"
else "Prelude.smt"
smtprelude <- readIncludeFile preludesmt
let scripttitle = unlines (map ("; "++) (lines title))
printWhenAll opts $
"RAW SMT SCRIPT:\n" ++ scripttitle ++ "\n\n" ++ showSMTRaw smt
let smtinput = scripttitle ++ "\n" ++ smtprelude ++ showSMT smt
printWhenDetails opts $ "SMT SCRIPT:\n" ++ showWithLineNums smtinput
let z3opts = ["-smt2", "-T:2"]
when (optStoreSMT opts) (storeSMT "SMT-" z3opts smtinput >> return ())
printWhenDetails opts $
"CALLING Z3 (with options: " ++ unwords z3opts ++ ")..."
(ecode,out,err) <- evalCmd "z3" ("-in" : z3opts) smtinput
when (ecode > 0) $ do
printWhenStatus opts $ "EXIT CODE: " ++ show ecode
outfile <- storeSMT "smterror-" z3opts smtinput
when (optVerb opts < 3) $ printWhenStatus opts $
"ERROR in SMT script (saved in '" ++ outfile ++ "'):\n" ++
out ++ "\n" ++ err
printWhenDetails opts $ "RESULT:\n" ++ out
unless (null err) $ printWhenDetails opts $ "ERROR:\n" ++ err
let pcvalid = let ls = lines out in not (null ls) && head ls == "unsat"
return (if ecode>0 then Nothing else Just pcvalid)
where
defaultSMTTypes = ["Int","Float","Bool", "Char", "[]"]
storeSMT fileprefix z3opts script = do
ctime <- getCPUTime
let outfile = fileprefix ++ transOpName qf ++ "-" ++ show ctime ++ ".smt"
execcmt = unwords $ ["; Run with: z3"] ++ z3opts ++ [outfile]
writeFile outfile (execcmt ++ "\n\n" ++ script)
return outfile
readIncludeFile :: String -> IO String
readIncludeFile incfile = do
ppinclude <- fmap (</> "include" </> incfile) getPackagePath
exppinclude <- doesFileExist ppinclude
if exppinclude
then readFile ppinclude
else do
let localinclude = "include" </> incfile
exlocalinclude <- doesFileExist localinclude
if exlocalinclude
then readFile localinclude
else do putStrLn $ "Warning: " ++ localinclude ++ " not found!\n" ++
"SMT script might be incomplete!"
return ""
typedVar2SMT :: (Int,TypeExpr) -> Command
typedVar2SMT (i,te) = DeclareVar (SV i (type2sort te))
tconsOfTypeExpr :: TypeExpr -> [QName]
tconsOfTypeExpr (TVar _) = []
tconsOfTypeExpr (FuncType a b) = union (tconsOfTypeExpr a) (tconsOfTypeExpr b)
tconsOfTypeExpr (TCons qName texps) =
foldr union [qName] (map tconsOfTypeExpr texps)
tconsOfTypeExpr (ForallType _ te) = tconsOfTypeExpr te
allQNamesInExp :: Expr -> [Either QName QName]
allQNamesInExp e =
trExpr (const id) (const id) comb lt fr (.) cas branch const e []
where
comb ct qn exp = ((classifyName ct) qn:) . foldr (.) id exp
where classifyName FuncCall = Right
classifyName (FuncPartCall _) = Right
classifyName ConsCall = Left
classifyName (ConsPartCall _) = Left
lt bs exp = exp . foldr (.) id (map snd bs)
fr _ exp = exp
cas _ exp bs = exp . foldr (.) id bs
branch pat exp = ((args pat)++) . exp
args (Pattern qc _) = [Left qc]
args (LPattern _) = []
showWithLineNums :: String -> String
showWithLineNums txt =
let txtlines = lines txt
maxlog = ilog (length txtlines + 1)
showNum n = (take (maxlog - ilog n) (repeat ' ')) ++ show n ++ ": "
in unlines . map (uncurry (++)) . zip (map showNum [1..]) $ txtlines
ilog :: Int -> Int
ilog n | n>0 = if n<10 then 0 else 1 + ilog (n `div` 10)
tdecl2SMT :: TypeDecl -> Command
tdecl2SMT (TypeSyn tc _ _ _) =
error $ "Cannot translate type synonym '" ++ showQName tc ++ "' into SMT!"
tdecl2SMT (TypeNew tc _ _ _) =
error $ "Cannot translate newtype '" ++ showQName tc ++ "' into SMT!"
tdecl2SMT (Type tc _ tvars consdecls) =
DeclareDatatypes
[(tcons2SMT tc, length tvars,
DT (map (\ (v,_) -> 'T' : show v) tvars) (map tconsdecl consdecls))]
where
tconsdecl (Cons qn _ _ texps) =
let cname = transOpName qn
in DCons cname (map (texp2sel qn) (zip [1..] texps))
texp2sel cname (i,texp) = (genSelName cname i, type2sortD True texp)
genSelName :: QName -> Int -> String
genSelName qc@(mn,fn) i
| mn == "Prelude" && take 3 fn == "(,,"
= transOpName qc ++ "_" ++ show i
| otherwise
= "sel" ++ show i ++ '-' : transOpName qc
preludeType2SMT :: String -> [Command]
preludeType2SMT tn
| take 3 tn == "(,,"
= let arity = length tn -1
in [ Comment "Declaration of tuple type:"
, DeclareDatatypes
[(tcons2SMT (pre tn), arity,
DT (map (\v -> 'T' : show v) [1 .. arity])
[DCons (transOpName (pre tn)) (map texp2sel [1 .. arity])])]]
| otherwise
= []
where
texp2sel i = (genSelName (pre tn) i, SComb ('T' : show i) [])
fun2SMT :: FuncDecl -> Maybe ([Ident],FunSig,Term)
fun2SMT (Func qn _ _ texp rule) = do
let fsig = FunSig (transOpName qn)
(map type2sort (argTypes texp))
(type2sort (resultType texp))
srule <- rule2SMT rule
let tpars = union (typeParamsOfFunSig fsig) (typeParamsOfTerm srule)
return (tpars, fsig, srule)
where
rule2SMT (External s) = return $
tEqu (tComb (transOpName qn) []) (tComb ("External:" ++ s) [])
rule2SMT (Rule vs exp) = do
let isnd = ndExpr exp
lhs = tComb (transOpName qn) (map TSVar vs)
(rhs,varsorts) <- exp2SMTWithVars (maximum (0:vs))
(if isnd then Just lhs else Nothing)
(simpExpr exp)
return $
Forall (map (\ (v,t) -> SV v (type2sort t))
(funcType2TypedVars vs texp))
(Exists (map (\ (v,s) -> SV v s) varsorts)
(if isnd then rhs else tEqu lhs rhs))
ndExpr :: Expr -> Bool
ndExpr = trExpr (\_ -> False)
(\_ -> False)
(\_ _ nds -> or nds)
(\bs nd -> nd || any snd bs)
(\_ _ -> True)
(\_ _ -> True)
(\_ nd bs -> nd || or bs)
(\_ -> id)
(\nd _ -> nd)
replaceHigherOrderInExp :: Expr -> Expr
replaceHigherOrderInExp =
trExpr Var Lit updComb FC.Let Free Or Case Branch Typed
where
updComb ct qn args = case ct of
FuncPartCall _ -> fcFailed
ConsPartCall _ -> fcFailed
FuncCall | qn == pre "apply" -> fcFailed
_ -> Comb ct qn args
exp2SMTWithVars :: Int -> Maybe Term -> Expr -> Maybe (Term, [(Int,Sort)])
exp2SMTWithVars maxusedvar lhs exp =
maybe Nothing
(\t -> Just $ elimFailed maxusedvar t)
(exp2SMT lhs (replaceHigherOrderInExp exp))
exp2SMT :: Maybe Term -> Expr -> Maybe Term
exp2SMT lhs exp = case exp of
Var i -> Just $ makeRHS (TSVar i)
Lit l -> Just $ makeRHS (lit2SMT l)
Comb _ qf args -> mapM (exp2SMT Nothing) args >>= comb2SMT qf
Case _ e bs -> do t <- exp2SMT Nothing e
bts <- mapM branch2SMT bs
return $ makeRHS (Match t bts)
FC.Let bs e -> do tbs <- mapM (\(v,be) -> do t <- exp2SMT Nothing be
return (v,t))
bs
t <- exp2SMT lhs e
return $ tLet tbs t
Free _ _ -> Nothing
Typed e te -> case e of
Comb _ qf args | all isTyped args
-> mapM (exp2SMT Nothing) args >>=
return . makeRHS .
TComb (As (transOpName qf)
(type2sort (foldr FuncType te (map exprType args))))
_ -> exp2SMT lhs e
Or e1 e2 -> do t1 <- exp2SMT lhs e1
t2 <- exp2SMT lhs e2
return $ tDisj [t1,t2]
where
comb2SMT qf targs
| qf `elem` map pre ["chr", "ord"] && length targs == 1
= return $ makeRHS (head targs)
| otherwise
= return $ makeRHS (tComb (transOpName qf) targs)
makeRHS rhs = maybe rhs (\l -> tEqu l rhs) lhs
branch2SMT (Branch (LPattern _) _) = Nothing
branch2SMT (Branch (Pattern qf vs) e) = do
t <- exp2SMT lhs e
return (PComb (Id (transOpName qf)) vs, t)
isTyped :: Expr -> Bool
isTyped e = case e of
Typed _ _ -> True
_ -> False
exprType :: Expr -> TypeExpr
exprType e = case e of
Typed _ te -> te
_ -> error "FlatCurry.Goodies.exprType: not a typed expression"
lit2SMT :: Literal -> Term
lit2SMT (Intc i) = TConst (TInt i)
lit2SMT (Floatc f) = TConst (TFloat f)
lit2SMT (Charc c) = TConst (TInt (ord c))
data TransApplyState = TransApplyState
{ tsFreshVarIndex :: Int
, tsNewVars :: [(Int,Sort)]
}
type TAState a = State TransApplyState a
elimFailed :: Int -> Term -> (Term, [(Int,Sort)])
elimFailed maxusedvar trm =
let st0 = TransApplyState (maximum (maxusedvar : allVarsInTerm trm) + 1) []
(ntrm,st1) = runState (elimFailedInTerm trm) st0
in (ntrm, reverse (tsNewVars st1))
elimFailedInTerm :: Term -> TAState Term
elimFailedInTerm t = case t of
TConst _ -> return t
TSVar _ -> return t
TComb (As n srt) [] | n == "Prelude_failed"
-> do st <- get
let fv = tsFreshVarIndex st
fvt = (fv, srt)
put st { tsFreshVarIndex = tsFreshVarIndex st + 1
, tsNewVars = fvt : tsNewVars st }
return (TSVar fv)
TComb qid ts -> mapM elimFailedInTerm ts >>= return . TComb qid
SMT.Let bs bt -> do trbs <- mapM (elimFailedInTerm . snd) bs
trbt <- elimFailedInTerm bt
return $ SMT.Let (zip (map fst bs) trbs) trbt
Forall vs te -> elimFailedInTerm te >>= return . Forall vs
Exists vs te -> elimFailedInTerm te >>= return . Exists vs
Match mt brs -> do trmt <- elimFailedInTerm mt
trbs <- mapM (elimFailedInTerm . snd) brs
return $ Match trmt (zip (map fst brs) trbs)
allVarsInTerm :: Term -> [SVar]
allVarsInTerm (TConst _) = []
allVarsInTerm (TSVar v) = [v]
allVarsInTerm (TComb _ args) = foldr union [] (map allVarsInTerm args)
allVarsInTerm (Forall vs arg) = union (map varOfSV vs) (allVarsInTerm arg)
allVarsInTerm (Exists vs arg) = union (map varOfSV vs) (allVarsInTerm arg)
allVarsInTerm (SMT.Let bs e) =
foldr union (map fst bs) (map allVarsInTerm (e : map snd bs))
allVarsInTerm (Match e ps) =
foldr union [] (map allVarsInTerm (e : map snd ps) ++ map (patVars . fst) ps)
where
patVars (PComb _ vs) = vs
varOfSV :: SortedVar -> SVar
varOfSV (SV v _) = v
type2sort :: TypeExpr -> Sort
type2sort = type2sortD False
type2sortD :: Bool -> TypeExpr -> Sort
type2sortD dt = t2s
where
t2s texp = case texp of
TVar i -> SComb ((if dt then "T" else "TVar") ++ show i) []
TCons qc targs -> SComb (tcons2SMT qc) (map t2s targs)
FuncType dom ran -> SComb "Func" (map t2s [dom,ran])
ForallType _ te -> t2s te
tcons2SMT :: QName -> String
tcons2SMT (mn,tc)
| "_Dict#" `isPrefixOf` tc
= "Dict"
| mn == "Prelude" && take 3 tc == "(,,"
= "Tuple" ++ show (length tc - 1)
| mn == "Prelude"
= maybe (encodeSpecialChars tc) id (lookup tc transPrimTCons)
| otherwise
= mn ++ "_" ++ encodeSpecialChars tc
transPrimTCons :: [(String,String)]
transPrimTCons =
[("Int","Int")
,("Float","Real")
,("Char","Int")
,("[]","List")
,("()","Unit")
,("(,)","Pair")
,("Maybe","Maybe")
,("Either","Either")
,("Ordering","Ordering")
]
encodeSpecialChars :: String -> String
encodeSpecialChars = concatMap encChar
where
encChar c | c `elem` "#$%[]()!,"
= let oc = ord c
in ['%', int2hex (oc `div` 16), int2hex(oc `mod` 16)]
| otherwise = [c]
int2hex i = if i<10 then chr (ord '0' + i)
else chr (ord 'A' + i - 10)
decodeSpecialChars :: String -> String
decodeSpecialChars [] = []
decodeSpecialChars (c:cs)
| c == '%' = let n = case readHex (take 2 cs) of
[(h,"")] -> h
_ -> 0
in chr n : decodeSpecialChars (drop 2 cs)
| otherwise = c : decodeSpecialChars cs
transOpName :: QName -> String
transOpName (mn,fn)
| mn=="Prelude" = fromMaybe tname (lookup fn primNames)
| otherwise = tname
where
tname = mn ++ "_" ++ encodeSpecialChars fn
untransOpName :: String -> Maybe QName
untransOpName s =
let (mn,ufn) = break (=='_') s
in if null ufn
then Nothing
else Just (mn, decodeSpecialChars (tail ufn))
primNames :: [(String,String)]
primNames =
[
("&&","and")
,("||","or")
,("not","not")
,("==","=")
,("/=","/=")
,("===","=")
,("/==","/=")
,("=","=")
,("<","<")
,("<=","<=")
,(">",">")
,(">=",">=")
,("+","+")
,("-","-")
,("*","*")
,("True","true")
,("False","false")
,("[]","nil")
,(":","insert")
,("()","unit")
,("(,)","mk-pair")
,("LT","LT")
,("EQ","EQ")
,("GT","GT")
,("Nothing","Nothing")
,("Just","Just")
,("Left","Left")
,("Right","Right")
,("_","_")
] ++
map (\i -> ('(' : take (i-1) (repeat ',') ++ ")", "Tuple" ++ show i)) [3..15]
getAllFunctions :: Options -> IORef ProgInfo -> [QName] -> IO [FuncDecl]
getAllFunctions opts pistore newfuns = do
mods <- fmap (map (miProg . snd) . progInfos) $ readIORef pistore
getAllFuncs mods preloadedFuncDecls newfuns
where
getAllFuncs _ currfuncs [] = return (reverse currfuncs)
getAllFuncs allmods currfuncs (newfun:newfuncs)
| newfun `elem` excludedCurryOperations ||
newfun `elem` map (pre . fst) primNames ||
newfun `elem` map funcName currfuncs || isPrimOp newfun
= getAllFuncs allmods currfuncs newfuncs
| fst newfun `elem` map progName allmods
= maybe (error $ "getAllFunctions: " ++ show newfun ++ " not found!")
(\fdecl ->
getAllFuncs allmods (fdecl : currfuncs)
(newfuncs ++
filter (`notElem` excludedCurryOperations)
(funcsOfFuncDecl fdecl)))
(find (\fd -> funcName fd == newfun)
(maybe []
progFuncs
(find (\m -> progName m == fst newfun) allmods)))
| otherwise
= do let mname = fst newfun
printWhenStatus opts $
"Loading module '" ++ mname ++ "' for '"++ snd newfun ++"'"
addModInfoFor pistore mname
newmod <- fmap miProg $ getModInfoFor pistore mname
getAllFuncs (newmod : allmods) currfuncs (newfun:newfuncs)
usedFunctions :: [QName] -> [FuncDecl] -> [FuncDecl]
usedFunctions _ [] = []
usedFunctions usedfns (fdecl : fdecls)
| funcName fdecl `elem` usedfns
= fdecl : usedFunctions (union (funcsOfFuncDecl fdecl) usedfns) fdecls
| otherwise
= usedFunctions usedfns fdecls
loadModulesForQNames :: Options -> IORef ProgInfo -> [QName] -> IO ()
loadModulesForQNames opts pistore qns = mapM_ loadMod (nub (map fst qns))
where
loadMod m = do
mloaded <- hasModInfoFor pistore m
unless mloaded $ do
printWhenStatus opts $ "Loading module '" ++ m ++ "'..."
addModInfoFor pistore m
preloadedFuncDecls :: [FuncDecl]
preloadedFuncDecls =
[Func (pre "id") 1 Public
(FuncType (TVar 0) (TVar 0))
(Rule [1] (Var 1)),
Func (pre "not") 1 Public
(FuncType fcBool fcBool)
(Rule [1] (Case Flex (Var 1)
[Branch (Pattern (pre "True") [] ) fcFalse,
Branch (Pattern (pre "False") []) fcTrue])),
Func (pre "null") 1 Public
(FuncType (fcList (TVar 0)) fcBool)
(Rule [1] (Case Flex (Var 1)
[Branch (Pattern (pre "[]") []) fcTrue,
Branch (Pattern (pre ":") [2,3]) fcFalse]))
]
excludedCurryOperations :: [QName]
excludedCurryOperations =
map pre ["apply", "failed", "chr", "ord"]
funcsOfFuncDecl :: FuncDecl -> [QName]
funcsOfFuncDecl fd =
nub (trRule (\_ e -> funcsInExpr e) (\_ -> []) (funcRule fd))
|