Prompted by +Mike Begley I just used the peg parser I showed in previous blog posts to make a simple implementation of
Tiny Basic
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
module tinybasic | |
open System | |
type Expression = | |
| Number of int | |
| Variable of string | |
| String of string | |
| Binary of Expression * string * Expression | |
| Unary of string * Expression | |
| VarList of string list | |
| ExprList of Expression list | |
| PrintSeparator of string | |
type Statement = | |
| Command of string | |
| Gosub of Expression | |
| Let of string * Expression | |
| Input of string list | |
| Goto of Expression | |
| If of Expression * Statement | |
| Print of Expression list | |
| Rem | |
type ParseResult = | |
| Line of int * Statement | |
| Immediate of Statement | |
| Parsed of Expression | |
| Unmatched | |
| TerminalSymbol of string | |
| Production of ParseResult list | |
| EmptyMatch | |
let syntaxError s = failwithf "Syntax Error %A" s | |
let parseString (s:string) _ = | |
Parsed <| (Expression.String (s.[1..s.Length - 2].Replace("\"\"","\""))) | |
let parseFactor s = function | |
| Parsed _ as x -> x | |
| Production [TerminalSymbol "("; _; x ; _; TerminalSymbol ")"] -> x | |
| _ -> syntaxError s | |
let parseBinary s = function | |
| Production [x; EmptyMatch] -> x | |
| Production [Parsed x; Production y] -> | |
Parsed <| List.fold (fun l -> function | Production [_; TerminalSymbol op; _; Parsed r] -> Binary (l, op, r) | _ -> syntaxError s) x y | |
| _ -> syntaxError s | |
let parseUnary s = function | |
| Production [EmptyMatch; _; x] -> x | |
| Production [TerminalSymbol op; _; Parsed x] -> Parsed <| Unary (op, x) | |
| _ -> syntaxError s | |
let parseVarList s = function | |
| Production [Parsed (Variable x); EmptyMatch] -> Parsed <| VarList [x] | |
| Production [Parsed (Variable x); Production y] -> | |
Parsed (VarList (x :: List.map (function | Production [_; TerminalSymbol ","; _; Parsed (Variable y)] -> y | _ -> syntaxError s) y)) | |
| _ -> syntaxError s | |
let parseExprList s = function | |
| Production [Parsed x; EmptyMatch; _; TerminalSymbol y] -> Parsed <| ExprList [x; PrintSeparator y] | |
| Production [Parsed x; EmptyMatch; _; EmptyMatch] -> Parsed <| ExprList [x] | |
| Production [Parsed x; Production y; _; z] -> | |
let right = List.collect (function | Production [_; TerminalSymbol s; _; Parsed y] -> [PrintSeparator s; y] | _ -> syntaxError s) y | |
let term = match z with | TerminalSymbol t -> [PrintSeparator t] | EmptyMatch -> [] | _ -> syntaxError z | |
Parsed (ExprList (x :: List.append right term)) | |
| _ -> syntaxError s | |
let parsePredicate s = function | |
| Production [Parsed l; _; TerminalSymbol op; _; Parsed r] -> Parsed <| Binary (l, op, r) | |
| _ -> syntaxError s | |
let parseStatement s = function | |
| Production [TerminalSymbol "PRINT"; _; Parsed (ExprList x)] -> Immediate <| Print x | |
| Production [TerminalSymbol "IF"; _; Parsed x; _; TerminalSymbol "THEN"; _; Immediate y] -> Immediate <| If (x, y) | |
| Production [TerminalSymbol "GOTO"; _; Parsed x] -> Immediate <| Goto x | |
| Production [TerminalSymbol "INPUT"; _; Parsed (VarList x)] -> Immediate <| Input x | |
| Production [TerminalSymbol "LET"; _; Parsed (Variable x); _; TerminalSymbol "="; _; Parsed y] -> Immediate <| Let (x, y) | |
| Production [TerminalSymbol "GOSUB"; _; Parsed x] -> Immediate <| Gosub x | |
| Production (TerminalSymbol "REM" :: _) -> Immediate <| Rem | |
| Immediate (Command _) as x -> x | |
| _ -> syntaxError s | |
let parseLine s = function | |
| Production [EmptyMatch; _; x; _; EmptyMatch] -> x | |
| Production [Parsed (Number x); _; Immediate y; _; EmptyMatch] -> Line (x, y) | |
| _ -> syntaxError s | |
(*%% | |
line <- number? space statement space <epsilon> { parseLine } | |
statement <- print / if / goto / input / let / gosub / rem / command { parseStatement } | |
print <- printkey space expr-list | |
if <- ifkey space predicate space "THEN" space statement | |
goto <- gotokey space expression | |
input <- inputkey space var-list | |
let <- letkey space var space '=' space expression | |
gosub <- gosubkey space expression | |
rem <- remkey <anychar>* | |
command <- ({Lu} / {Ll})+ { (fun (s:string) _ -> Immediate <| Command (s.ToUpper())) } | |
printkey <- [Pp] [Rr] ([Ii] [Nn] [Tt])? { (fun _ _ -> TerminalSymbol "PRINT") } | |
ifkey <- [Ii] [Ff] { (fun _ _ -> TerminalSymbol "IF") } | |
gotokey <- [Gg] [Oo] [Tt] [Oo] { (fun _ _ -> TerminalSymbol "GOTO") } | |
inputkey <- [Ii] [Nn] ([Pp] [Uu] [Tt])? { (fun _ _ -> TerminalSymbol "INPUT") } | |
letkey <- ([Ll] [Ee] [Tt])? { (fun _ _ -> TerminalSymbol "LET") } | |
gosubkey <- [Gg] [Oo] [Ss] [Uu] [Bb] { (fun _ _ -> TerminalSymbol "GOSUB") } | |
remkey <- ([Rr] [Ee] [Mm]) / "'" { (fun _ _ -> TerminalSymbol "REM") } | |
expr-list <- (string / expression) (space [,;] space (string / expression))* space [,;]? { parseExprList } | |
var-list <- var (space ',' space var)* { parseVarList } | |
predicate <- expression space relop space expression { parsePredicate } | |
expression <- term (space [+-] space term)* { parseBinary } | |
term <- unary (space [*/] space unary)* { parseBinary } | |
unary <- [+-]? space factor { parseUnary } | |
factor <- var / number / ('(' space expression space ')') { parseFactor } | |
var <- {Lu} / {Ll} { (fun (s:string) _ -> Parsed <| (Variable (s.ToUpperInvariant()))) } | |
number <- {Nd}+ { (fun s _ -> Parsed <| Number (Int32.Parse(s))) } | |
relop <- "<>" / "<=" / '<' / "><" / ">=" / '>' / '=' | |
string <- '\"' ((!'\"' <anychar>) / "\"\"")* '\"' { parseString } | |
space <- [ \t]* { (fun _ _ -> EmptyMatch) } | |
%%*) | |
type Context (program:(int * Statement * string) list, variables:Map<string,Expression>, next:int, stack:int list) = | |
member this.Program = program | |
member this.Variables = variables | |
member this.Next = next | |
member this.Stack = stack | |
let comparison op = | |
let fn = match op with | ">" -> (>) | ">=" -> (>=) | "<>" | "><" -> (<>) | "<" -> (<) | "<=" -> (<=) | "=" -> (=) | x -> failwithf "Unexpected operator %A" x | |
(fun a b -> if fn a b then -1 else 0) | |
let rec evalAsNumber (context:Context) x = | |
match evalExpression context x with | |
| Number a -> a | |
| a -> failwithf "Expecting number not %A" a | |
and evalExpression (context:Context) = function | |
| Number _ as x -> x | |
| Variable x -> match Map.tryFind x context.Variables with | Some y -> y | None -> Number 0 | |
| String _ as x -> x | |
| Binary (x, op, y) -> | |
let left = evalAsNumber context x | |
let right = evalAsNumber context y | |
let fn = match op with | "+" -> (+) | "-" -> (-) | "*" -> (*) | "/" -> (/) | a -> comparison a | |
Number <| fn left right | |
| Unary ("+", x) -> evalExpression context x | |
| Unary ("-", x) -> Number <| -evalAsNumber context x | |
| PrintSeparator _ as x -> x | |
| x -> failwith "Internal Error" | |
let listProgram (context:Context) = for (_,_,o) in context.Program do printfn "%s" o | |
let setVariable (context:Context) name value = Context(context.Program, Map.add name value context.Variables, context.Next, context.Stack) | |
let setLine (context:Context) line = Context(context.Program, context.Variables, line, context.Stack) | |
let setGosub (context:Context) line = Context(context.Program, context.Variables, line, (context.Next :: context.Stack)) | |
let gosubReturn (context:Context) = | |
match context.Stack with | |
| [] -> failwith "RETURN without GOSUB" | |
| (head :: tail) -> Context(context.Program, context.Variables, head, tail) | |
let findLine (context:Context) = List.tryFind (fun (l, _, _) -> l >= context.Next) context.Program | |
let rec runProgram (context:Context) = | |
if context.Next < 0 then context | |
else | |
match findLine context with | |
| Some (n, s, _) -> evalImmediate (setLine context (n + 1)) s |> runProgram | |
| None -> setLine context -1 | |
and evalImmediate context = function | |
| Command "CLEAR" -> Context([], Map.empty, 0, []) | |
| Command "LIST" -> | |
listProgram context | |
context | |
| Command "RUN" -> runProgram <| setLine context 0 | |
| Command "END" -> Context(context.Program, context.Variables, -1, []) | |
| Command "RETURN" -> gosubReturn context | |
| Command x -> failwithf "Unknown command %A" x | |
| Goto x -> | |
let cp = setLine context <| evalAsNumber context x | |
if context.Next < 0 then runProgram cp else cp | |
| Gosub x -> | |
let cp = setGosub context <| evalAsNumber context x | |
if context.Next < 0 then runProgram cp else cp | |
| Let (x,y) -> setVariable context x <| evalExpression context y | |
| Print x -> | |
let rec doPrint = function | |
| [] -> printfn "" | |
| [PrintSeparator ";"] -> () | |
| [PrintSeparator ","] -> printf "\t" | |
| (head :: tail) -> | |
match head with | |
| Number n -> printf "%d" n | |
| String s -> printf "%s" s | |
| PrintSeparator "," -> printf "\t" | |
| PrintSeparator ";" -> () | |
| _ -> failwithf "Eval error %A" head | |
doPrint tail | |
doPrint <| List.map (evalExpression context) x | |
context | |
| If (x, y) -> | |
match evalExpression context x with | |
| Number 0 -> context | |
| Number _ -> evalImmediate context y | |
| a -> failwithf "Eval error %A" a | |
| Input x -> | |
let rec doInput context = function | |
| [] -> context | |
| (head :: tail) -> | |
printf "?" | |
match Int32.TryParse(Console.ReadLine()) with | |
| (true, i) -> doInput (setVariable context head (Number i)) tail | |
| (false, _) -> failwith "Input error" | |
doInput context x | |
| Rem -> context | |
let addLine (context:Context) l s o = | |
let rec insertLine = function | |
| [] -> [(l,s,o)] | |
| ((currLine, _, _) as curr :: tail) -> | |
if currLine = l then ((l,s,o) :: tail) | |
else if currLine < l then (curr :: insertLine tail) | |
else ((l,s,o) :: curr :: tail) | |
Context(insertLine context.Program, context.Variables, context.Next, context.Stack) | |
do | |
let rec runBasic context = | |
printf "?" | |
let line = Console.ReadLine() | |
if line.StartsWith("!") then | |
match line.[1..].Split([|' '|], 2) with | |
| [|"load"; file|] -> | |
let lines = Seq.ofArray <| System.IO.File.ReadAllLines(file) | |
let program = Seq.fold (fun c l -> match parse l with | (Line (x, y), _) -> ((x, y, l) :: c) | _ -> failwith "Error reading file") [] lines | |
runBasic <| Context(List.rev program, Map.empty, 0, []) | |
| [|"save"; file|] -> | |
System.IO.File.WriteAllLines(file, List.map (fun (_,_,l) -> l) context.Program) | |
runBasic context | |
| [|"quit"; file|] | [|"exit"; file|] -> () | |
| _ -> printfn "Did not understand %A" line | |
runBasic context | |
else | |
try | |
match parse line with | |
| (Immediate x, _) -> | |
runBasic <| evalImmediate context x | |
| (Line (x, y), _) -> | |
runBasic <| addLine context x y line | |
| (Unmatched, _) -> | |
printfn "Syntax error" | |
runBasic context | |
| x -> printfn "%A" x | |
runBasic context | |
with | |
| ex -> printfn "%s" ex.Message | |
runBasic context | |
printfn "Running Tiny Basic (F# edition)" | |
runBasic <| Context([], Map.empty, 0, []) | |
This allows simple Tiny Basic code to run, !load and !save do the obvious things
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
10 GOSUB 100 | |
20 GOSUB 200 | |
30 IF A <> 0 THEN GOTO 10 | |
40 PRINT "Goodbye!" | |
50 END | |
100 '---------------------- | |
101 ' read a number into A | |
102 '---------------------- | |
110 PRINT "Enter a number"; | |
120 INPUT A | |
130 RETURN | |
200 '----------------- | |
201 ' print A A^2 A^3 | |
202 '----------------- | |
210 PRINT A, A*A, A*A*A | |
220 RETURN |
Hmmmmmmm
ReplyDelete