-
Notifications
You must be signed in to change notification settings - Fork 0
/
Interpreter3.hs
103 lines (84 loc) · 2.91 KB
/
Interpreter3.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
--------------------------------------------------------------------------------
-- Pepe Gallardo, January 2004
--
-- A monadic interpreter for Dijkstra's Guarded Command Language:
-- https://en.wikipedia.org/wiki/Guarded_Command_Language
--
-- Second version. Uses state transformer monad to propagate environments
--
-- Drawbacks: * Language is deterministic
-- * Errors abort interpreter execution
--------------------------------------------------------------------------------
import qualified Environment as Env
import Expression
import Command
import Examples
import Data.List(intercalate)
import Control.Exception(catch, ErrorCall)
import Control.Monad
--------------------------------------------------------------------------------
-- The (state transformer) monad
--------------------------------------------------------------------------------
newtype Mon a = M (Environment -> [(a, Environment)])
instance Monad Mon where
-- return :: a -> Mon a
return x = M (\s -> [(x,s)])
-- fail :: String -> Mon a
fail str = M (\_ -> error str)
-- (>>=) :: Mon a -> (a -> Mon b) -> Mon b
M st0 >>= f = M (\s0 -> concat [ st1 s1
| (x, s1) <- st0 s0
, let M st1 = f x
])
instance MonadPlus Mon where
-- mzero :: Mon a
mzero = M (\_ -> [])
-- mplus :: Mon a -> Mon a -> Mon a
M st0 `mplus` M st1 = M (\s -> st0 s ++ st1 s)
readMon :: Mon Environment
readMon = M (\s -> [(s, s)])
writeMon :: Environment -> Mon ()
writeMon s = M (\_ -> [((), s)])
--------------------------------------------------------------------------------
-- Sematics of commands
--------------------------------------------------------------------------------
sem :: Command -> Mon ()
sem Skip =
return ()
sem Abort =
fail "Program aborted"
sem (var := e) = do
rho <- readMon
val <- eval rho e
rho' <- Env.set rho (var, val)
writeMon rho'
sem (s1 :$ s2) = do
sem s1
sem s2
sem (If gs) = do
rho <- readMon
ss <- select rho gs
case ss of
[] -> fail "Program aborted: all guards are false"
ss -> foldr1 mplus . map sem $ ss
sem (Do gs) = do
rho <- readMon
ss <- select rho gs
case ss of
[] -> return ()
ss -> foldr1 mplus . map (sem . (:$ Do gs)) $ ss
-- Running a program. Prints out final environment
run :: Command -> IO ()
run prog = sequence_ [ printEnv rho `catch` \error -> print (error :: ErrorCall)
| rho <- rhos
]
where
M st = sem prog
rhos = [ rho | (_, rho) <- st Env.empty ]
printEnv = putStrLn
. intercalate ", "
. Env.fold (\var val -> ((var++"->"++show val):)) []
--------------------------------------------------------------------------------
-- Run all examples
--------------------------------------------------------------------------------
main = sequence_ [ do run e; putStr "\n" | e <- examples ]