module Control.Monad.BrainFuck where
import qualified Control.Monad as M
newtype BrainFuck a = BrainFuck (DataPointer -> ([Char], DataPointer, a))
type DataPointer = Integer
func :: BrainFuck f -> DataPointer -> ([Char], DataPointer, f)
func (BrainFuck f) = f
brainfuck :: BrainFuck f -> String
brainfuck (BrainFuck f) = bytes where (bytes, _, _) = f 0
instance Monad BrainFuck where
return ret = BrainFuck $ \loc -> ([], loc, ret)
a >>= b = BrainFuck $ \start -> let
(left, mid, val) = func a start
(right, end, ret) = func (b val) mid
in (left ++ right, end, ret)
next, prev, incr, decr, output, input, open, close :: BrainFuck ()
next = cmd' succ '>'
prev = cmd' pred '<'
incr = cmd '+'
decr = cmd '-'
output = cmd '.'
input = cmd ','
open = cmd '['
close = cmd ']'
add, sub :: Int -> BrainFuck ()
add = multi incr
sub = multi decr
cmd :: Char -> BrainFuck ()
cmd = cmd' id
cmd' :: (DataPointer -> DataPointer) -> Char -> BrainFuck ()
cmd' f x = BrainFuck $ \loc -> ([x], f loc, ())
multi :: BrainFuck () -> Int -> BrainFuck ()
multi c n = do
_ <- sequence (replicate n c)
return ()
addr :: BrainFuck DataPointer
addr = BrainFuck $ \loc -> ([], loc, loc)
setAddr :: Integer -> BrainFuck ()
setAddr n = do
a <- addr
if a > n
then prev >> setAddr n
else if a < n
then next >> setAddr n
else return ()
loopUnless0 :: BrainFuck () -> BrainFuck ()
loopUnless0 a = do
open
a
close
data Constants
brainfuckConstants :: (Constants -> BrainFuck a) -> String
brainfuckConstants a = brainfuck $ do
M.forM_ [0..8] $ \n -> do
add n
next
a (undefined :: Constants)
at :: Constants -> Integer -> BrainFuck ()
at _ = setAddr
loop :: Constants -> BrainFuck () -> BrainFuck ()
loop constants a = do
here <- addr
at constants 1
loopUnless0 $ do
setAddr here
a
forever :: Constants -> BrainFuck () -> BrainFuck ()
forever constants a = loop constants $ do
a
at constants 1
demo :: String
demo = brainfuckConstants $ \constants -> do
add 32
forever constants $ do
add 1
output