push
This commit is contained in:
commit
ef7ff34b3e
4 changed files with 181 additions and 0 deletions
11
brainfuck.ipkg
Normal file
11
brainfuck.ipkg
Normal file
|
|
@ -0,0 +1,11 @@
|
|||
package brainfuck
|
||||
|
||||
authors = "you"
|
||||
version = 0.1.0
|
||||
|
||||
sourcedir = "src"
|
||||
main = Main
|
||||
executable = "brainfuck"
|
||||
|
||||
depends = contrib
|
||||
|
||||
51
flake.nix
Normal file
51
flake.nix
Normal file
|
|
@ -0,0 +1,51 @@
|
|||
{
|
||||
description = "idris2 brainfuck interpreter";
|
||||
inputs = {
|
||||
nixpkgs.url = "github:NixOS/nixpkgs/nixos-unstable";
|
||||
flake-utils.url = "github:numtide/flake-utils";
|
||||
};
|
||||
outputs = { self, nixpkgs, flake-utils }:
|
||||
flake-utils.lib.eachDefaultSystem (system:
|
||||
let
|
||||
pkgs = import nixpkgs { inherit system; };
|
||||
brainfuck = pkgs.stdenv.mkDerivation {
|
||||
pname = "brainfuck";
|
||||
version = "0.1.0";
|
||||
src = ./.;
|
||||
nativeBuildInputs = [ pkgs.idris2 ];
|
||||
buildPhase = ''
|
||||
idris2 --build brainfuck.ipkg
|
||||
'';
|
||||
installPhase = ''
|
||||
mkdir -p $out/bin
|
||||
cp -r build/exec/brainfuck_app $out/bin/brainfuck_app
|
||||
cp build/exec/brainfuck $out/bin/brainfuck
|
||||
'';
|
||||
meta = with pkgs.lib; {
|
||||
description = "A Brainfuck interpreter in Idris2";
|
||||
license = licenses.mit;
|
||||
platforms = platforms.unix;
|
||||
};
|
||||
};
|
||||
in {
|
||||
packages.default = brainfuck;
|
||||
packages.brainfuck = brainfuck;
|
||||
apps.default = {
|
||||
type = "app";
|
||||
program = "${brainfuck}/bin/brainfuck";
|
||||
};
|
||||
devShells.default = pkgs.mkShell {
|
||||
packages = [
|
||||
pkgs.idris2
|
||||
pkgs.rlwrap
|
||||
pkgs.nodePackages.nodemon
|
||||
];
|
||||
shellHook = ''
|
||||
echo "Idris2 brainfuck dev shell"
|
||||
echo " Build : idris2 --build brainfuck.ipkg"
|
||||
echo " Run : ./build/exec/brainfuck <file.bf>"
|
||||
'';
|
||||
};
|
||||
}
|
||||
);
|
||||
}
|
||||
1
hello.bf
Normal file
1
hello.bf
Normal file
|
|
@ -0,0 +1 @@
|
|||
++++++++[>++++[>++>+++>+++>+<<<<-]>+>+>->>+[<]<-]>>.>---.+++++++..+++.>>.<-.<.+++.------.--------.>>+.>++.
|
||||
118
src/Main.idr
Normal file
118
src/Main.idr
Normal file
|
|
@ -0,0 +1,118 @@
|
|||
module Main
|
||||
|
||||
import Data.List
|
||||
import Data.SortedMap
|
||||
import System
|
||||
import System.File
|
||||
|
||||
record BFState where
|
||||
constructor MkBFState
|
||||
leftCells : List Int
|
||||
currentCell : Int
|
||||
rightCells : List Int
|
||||
|
||||
BraceMap : Type
|
||||
BraceMap = SortedMap Int Int
|
||||
|
||||
inits : BFState
|
||||
inits = MkBFState [] 0 (replicate 29999 0)
|
||||
|
||||
mr : BFState -> BFState
|
||||
mr (MkBFState left cur right) =
|
||||
case right of
|
||||
[] => MkBFState (cur :: left) 0 []
|
||||
(r::rs) => MkBFState (cur :: left) r rs
|
||||
|
||||
moveLeft : BFState -> BFState
|
||||
moveLeft (MkBFState left cur right) =
|
||||
case left of
|
||||
[] => MkBFState [] 0 (cur :: right)
|
||||
(l::ls) => MkBFState ls l (cur :: right)
|
||||
|
||||
increment : BFState -> BFState
|
||||
increment st = { currentCell $= (+1) } st
|
||||
|
||||
decrement : BFState -> BFState
|
||||
decrement st = { currentCell $= (\x => x - 1) } st
|
||||
|
||||
buildBraceMap : List (Int, Char) -> List Int -> BraceMap -> BraceMap
|
||||
buildBraceMap [] _ bmap = bmap
|
||||
buildBraceMap ((i, c) :: rest) stack bmap =
|
||||
case c of
|
||||
'[' => buildBraceMap rest (i :: stack) bmap
|
||||
']' => case stack of
|
||||
[] => buildBraceMap rest [] bmap
|
||||
(s :: ss) =>
|
||||
let bmap' = insert s i (insert i s bmap)
|
||||
in buildBraceMap rest ss bmap'
|
||||
_ => buildBraceMap rest stack bmap
|
||||
|
||||
mbm : List Char -> BraceMap
|
||||
mbm code =
|
||||
buildBraceMap (zip [0..(cast (length code) - 1)] code) [] empty
|
||||
|
||||
cleanup : String -> List Char
|
||||
cleanup s = filter (\c => c `elem` (unpack ".,[]<>+-")) (unpack s)
|
||||
|
||||
safeIndex : Int -> List a -> Maybe a
|
||||
safeIndex _ [] = Nothing
|
||||
safeIndex 0 (x :: xs) = Just x
|
||||
safeIndex n (x :: xs) =
|
||||
if n < 0 then Nothing else safeIndex (n - 1) xs
|
||||
|
||||
|
||||
runStep : List Char -> BraceMap -> Int -> BFState -> IO (BFState, Int)
|
||||
runStep code bmap cp st =
|
||||
case safeIndex cp code of
|
||||
Nothing => pure (st, cp)
|
||||
Just op =>
|
||||
case op of
|
||||
'>' => pure (mr st, cp + 1)
|
||||
'<' => pure (moveLeft st, cp + 1)
|
||||
'+' => pure (increment st, cp + 1)
|
||||
'-' => pure (decrement st, cp + 1)
|
||||
|
||||
'.' => do
|
||||
putChar (cast (currentCell st))
|
||||
pure (st, cp + 1)
|
||||
|
||||
',' => do
|
||||
c <- getChar
|
||||
pure ({ currentCell := cast c } st, cp + 1)
|
||||
|
||||
'[' => if currentCell st == 0
|
||||
then case lookup cp bmap of
|
||||
Just target => pure (st, target + 1)
|
||||
Nothing => pure (st, cp + 1)
|
||||
else pure (st, cp + 1)
|
||||
|
||||
']' => if currentCell st /= 0
|
||||
then case lookup cp bmap of
|
||||
Just target => pure (st, target + 1)
|
||||
Nothing => pure (st, cp + 1)
|
||||
else pure (st, cp + 1)
|
||||
|
||||
_ => pure (st, cp + 1)
|
||||
|
||||
interpret : List Char -> BraceMap -> Int -> BFState -> IO ()
|
||||
interpret code bmap cp st =
|
||||
if cp >= cast (length code)
|
||||
then pure ()
|
||||
else do
|
||||
(st', cp') <- runStep code bmap cp st
|
||||
interpret code bmap cp' st'
|
||||
|
||||
|
||||
main : IO ()
|
||||
main = do
|
||||
args <- getArgs
|
||||
case args of
|
||||
(_ :: filename :: _) => do
|
||||
result <- readFile filename
|
||||
case result of
|
||||
Left err => putStrLn ("Error reading file: " ++ show err)
|
||||
Right src => do
|
||||
let code = cleanup src
|
||||
bmap = mbm code
|
||||
interpret code bmap 0 inits
|
||||
_ => putStrLn "Usage: brainfuck <filename.bf>"
|
||||
Loading…
Add table
Add a link
Reference in a new issue