-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathbfck.hs
128 lines (106 loc) · 3.98 KB
/
bfck.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
-- A simple interpreter of the Brainf*ck programming language
{-# LANGUAGE RecordWildCards, FlexibleContexts #-}
import Text.Parsec.Pos
import Text.Parsec
import Data.List
import Data.Char
import Safe
import System.Environment
import Control.Applicative
-- |Tokens of the language represented as types
data Token = LeftBracket
| RightBracket
| Plus
| Minus
| LeftShift
| RightShift
| Print
| Input
deriving (Show, Eq)
-- |Parse tree types
data Tree = Op Token Tree
| Loop Tree Tree
| End
deriving (Show, Eq)
-- |Memory representing the current state during the computation
data Memory = Memory { memRight :: [Int], memLeft :: [Int], current :: Int }
newMem :: Memory
newMem = let memRight = [0,0 ..]
memLeft = [0,0 ..]
current = 0
in Memory {..}
memValue :: Memory -> Int
memValue Memory{..} = current
moveLeft :: Memory -> Memory
moveLeft Memory{..} = Memory (current : memRight) (tail memLeft) (head memLeft)
moveRight :: Memory -> Memory
moveRight Memory{..} = Memory (tail memRight) (current : memLeft) (head memRight)
modifyMem :: Int -> Memory -> Memory
modifyMem i Memory{..} = Memory memRight memLeft (current + i)
incMem :: Memory -> Memory
incMem = modifyMem 1
decMem :: Memory -> Memory
decMem = modifyMem (-1)
printMem :: Memory -> IO ()
printMem Memory{..} = print . chr $ current
readMem :: Memory -> IO Memory
readMem Memory{..} = do
c <- getChar
return Memory { current = digitToInt c, .. }
main :: IO ()
main = do
bfckSource <- head <$> getArgs
let (Right x) = parse (many1 tokenizer) "" bfckSource
_ <- eval (buildTree x) newMem
return ()
buildTree :: [Token] -> Tree
buildTree [] = End
buildTree ts@(LeftBracket : _) = let (firstPart, restPart) = loopSplit
in Loop (buildTree $ tailSafe firstPart) (buildTree $ tailSafe restPart)
where
loopSplit = splitAt (snd bracketTuple) ts -- the idea is to split at position 'X' in a sequence '[......]X......'
bracketTuple = head . filter (\x -> fst x == 0) $ bracketIndices ts
buildTree (x : rest) = Op x $ buildTree rest
-- |This is really ugly, please skip this function
matchIndices :: [Int] -> [Int] -> [(Int, Int)]
matchIndices ls = foldl match []
where
match acc r = (leftBracketIndex acc r, r) : acc
leftBracketIndex acc r = index r (map fst acc)
index r used = maximum $ filter (\x -> x < r && x `notElem` used) ls
-- |The idea is that for each bracket pair [ ] we get a tuple (i,j) where i is the index of [ and j is the index of ]
bracketIndices :: [Token] -> [(Int,Int)]
bracketIndices ts
| length leftIndices == length rightIndices = matchIndices leftIndices rightIndices
| otherwise = error "mismatch in brackets [ ]"
where
leftIndices = elemIndices LeftBracket ts
rightIndices = elemIndices RightBracket ts
recognise :: Char -> Token
recognise c
| c == '+' = Plus
| c == '-' = Minus
| c == '<' = LeftShift
| c == '>' = RightShift
| c == '.' = Print
| c == ',' = Input
| c == '[' = LeftBracket
| c == ']' = RightBracket
| otherwise = error "unsupported character"
tokenSatisfy :: (Stream s m Char) => (Char -> Maybe Token) -> ParsecT s u m Token
tokenSatisfy = tokenPrim
(\c -> show [c])
(\pos c _cs -> updatePosChar pos c)
tokenizer :: (Stream s m Char) => ParsecT s u m Token
tokenizer = tokenSatisfy $ Just . recognise
eval :: Tree -> Memory -> IO Memory
eval End m = return m
eval (Op Plus x) m = eval x (incMem m)
eval (Op Minus x) m = eval x (decMem m)
eval (Op Print x) m = printMem m >> eval x m
eval (Op Input x) m = readMem m >>= eval x
eval (Op LeftShift x) m = eval x (moveLeft m)
eval (Op RightShift x) m = eval x (moveRight m)
eval loop@(Loop l r) m = if memValue m == 0 then eval r m
else eval l m >>= eval loop
eval _ _ = error "unsupported node in parsing tree"