blob: 82de067c8d619d20698b785590572743ad9776fe (
plain) (
blame)
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
|
-- MIT License, Copyright (c) 2023 Marvin Borner
-- based on the RKNL abstract machine
module Reducer
( reduce
) where
import Data.IORef
import Data.List ( elemIndex )
import Data.Map ( Map )
import qualified Data.Map as Map
import Helper
import System.Random ( randomIO )
type Store = Map Int Box
type Stack = [Redex]
data BoxValue = Todo Redex | Done Redex | Empty
data Box = Box (IORef BoxValue)
data Rvar = Num Int | Hole
data Redex = Rabs Int Redex | Rapp Redex Redex | Rvar Rvar | Rclosure Redex Store | Rcache Box Redex
data Conf = Econf Redex Store Stack | Cconf Stack Redex | End
toRedex :: Expression -> IO Redex
toRedex = convertWorker []
where
convertWorker ns (Abstraction e) = do
v <- randomIO :: IO Int
t <- convertWorker (v : ns) e
pure $ Rabs v t
convertWorker ns (Application l r) = do
lhs <- convertWorker ns l
rhs <- convertWorker ns r
pure $ Rapp lhs rhs
convertWorker ns (Bruijn i) =
pure $ Rvar $ Num (if i < 0 || i >= length ns then i else ns !! i)
convertWorker _ _ = invalidProgramState
fromRedex :: Redex -> IO Expression
fromRedex = convertWorker []
where
convertWorker es (Rabs n e) = Abstraction <$> convertWorker (n : es) e
convertWorker es (Rapp l r) = do
lhs <- convertWorker es l
rhs <- convertWorker es r
pure $ Application lhs rhs
convertWorker es (Rvar (Num n)) = pure $ Bruijn $ maybe n id (elemIndex n es)
convertWorker _ _ = invalidProgramState
transition :: Conf -> IO Conf
transition (Econf (Rapp u v) e s) =
pure $ Econf u e ((Rapp (Rvar Hole) (Rclosure v e)) : s)
transition (Econf (Rabs x t) e s) = do
box <- newIORef Empty
pure $ Cconf s (Rcache (Box box) (Rclosure (Rabs x t) e))
transition (Econf (Rvar (Num x)) e s) = do
def <- newIORef $ Done $ Rvar $ Num x
let b@(Box m) = Map.findWithDefault (Box def) x e
rd <- readIORef m
case rd of
Todo (Rclosure v e') -> pure $ Econf v e' ((Rcache b (Rvar Hole)) : s)
Done t -> pure $ Cconf s t
_ -> invalidProgramState
transition (Cconf ((Rcache (Box m) (Rvar Hole)) : s) t) = do
writeIORef m (Done t)
pure $ Cconf s t
transition (Cconf ((Rapp (Rvar Hole) ve) : s) (Rcache _ (Rclosure (Rabs x t) e)))
= do
box <- newIORef (Todo ve)
pure $ Econf t (Map.insert x (Box box) e) s
transition (Cconf s (Rcache (Box m) (Rclosure (Rabs x t) e))) = do
rd <- readIORef m
case rd of
Done v -> pure $ Cconf s v
Empty -> do
x1 <- randomIO :: IO Int
box <- newIORef $ Done $ Rvar $ Num x1
pure $ Econf t
(Map.insert x (Box box) e)
((Rabs x1 (Rvar Hole)) : (Rcache (Box m) (Rvar Hole)) : s)
Todo _ -> invalidProgramState
transition (Cconf ((Rapp (Rvar Hole) (Rclosure v e)) : s) t) =
pure $ Econf v e ((Rapp t (Rvar Hole)) : s)
transition (Cconf ((Rapp t (Rvar Hole)) : s) v) = pure $ Cconf s (Rapp t v)
transition (Cconf ((Rabs x1 (Rvar Hole)) : s) v) = pure $ Cconf s (Rabs x1 v)
transition (Cconf [] _) = pure End
transition _ = invalidProgramState
forEachState :: Conf -> (Conf -> IO Conf) -> IO Conf
forEachState conf trans = trans conf >>= \case
End -> pure conf
next -> forEachState next trans
loadTerm :: Redex -> Conf
loadTerm t = Econf t Map.empty []
reduce :: Expression -> IO Expression
reduce e = do
redex <- toRedex e
forEachState (loadTerm redex) transition >>= \case
Cconf [] v -> fromRedex v
_ -> invalidProgramState
|