mirror of
https://github.com/kanaka/mal.git
synced 2024-11-10 12:47:45 +03:00
130 lines
4.7 KiB
Forth
130 lines
4.7 KiB
Forth
module Eval
|
|
|
|
open Types
|
|
|
|
type Env = Map<string, Node>
|
|
|
|
let errExpected tok = EvalError(sprintf "expected %s" tok)
|
|
|
|
let iterPairs f (source : seq<_>) =
|
|
use iter = source.GetEnumerator()
|
|
let rec loop () =
|
|
if iter.MoveNext() then
|
|
let first = iter.Current
|
|
if not (iter.MoveNext()) then raise <| errExpected "even node count"
|
|
let second = iter.Current
|
|
f first second
|
|
loop ()
|
|
loop ()
|
|
|
|
let quasiquoteForm nodes =
|
|
let identity n = n
|
|
let singleNodeTransform f = function
|
|
| [node] -> f node
|
|
| _ -> raise <| Core.errArity ()
|
|
let rec quasiquote node =
|
|
match node with
|
|
| Vector(seg) when seg.Count > 0 -> quasiquote (List(List.ofSeq seg))
|
|
| List(Symbol("unquote")::rest) -> rest |> singleNodeTransform identity
|
|
| List(List(Symbol("splice-unquote")::spliceRest)::rest) ->
|
|
List([
|
|
Symbol("concat")
|
|
(singleNodeTransform identity spliceRest)
|
|
(quasiquote (List(rest)))])
|
|
| List(node::rest) ->
|
|
List([
|
|
Symbol("cons")
|
|
(quasiquote node)
|
|
(quasiquote (List(rest)))])
|
|
| n -> List([Symbol("quote"); n])
|
|
nodes |> singleNodeTransform quasiquote
|
|
|
|
let rec eval_ast env = function
|
|
| Symbol(sym) -> Env.get env sym
|
|
| List(lst) -> lst |> List.map (eval env) |> List
|
|
| Vector(seg) -> seg |> Seq.map (eval env) |> Array.ofSeq |> makeVector
|
|
| Map(map) -> map |> Map.map (fun k v -> eval env v) |> Map
|
|
| node -> node
|
|
|
|
and defBangForm env = function
|
|
| [sym; form] ->
|
|
match sym with
|
|
| Symbol(sym) ->
|
|
let node = eval env form
|
|
Env.set env sym node
|
|
node
|
|
| _ -> raise <| errExpected "symbol"
|
|
| _ -> raise <| Core.errArity ()
|
|
|
|
and setBinding env first second =
|
|
let s = match first with
|
|
| Symbol(s) -> s
|
|
| _ -> raise <| errExpected "symbol"
|
|
let form = eval env second
|
|
Env.set env s form
|
|
|
|
and letStarForm outer = function
|
|
| [bindings; form] ->
|
|
let inner = Env.makeNew outer [] []
|
|
let binder = setBinding inner
|
|
match bindings with
|
|
| List(lst) -> lst |> iterPairs binder
|
|
| Vector(seg) -> seg |> iterPairs binder
|
|
| _ -> raise <| errExpected "list or vector"
|
|
inner, form
|
|
| _ -> raise <| Core.errArity ()
|
|
|
|
and ifForm env = function
|
|
| [condForm; trueForm; falseForm] -> ifForm3 env condForm trueForm falseForm
|
|
| [condForm; trueForm] -> ifForm3 env condForm trueForm Nil
|
|
| _ -> raise <| Core.errArity ()
|
|
|
|
and ifForm3 env condForm trueForm falseForm =
|
|
match eval env condForm with
|
|
| Bool(false) | Nil -> falseForm
|
|
| _ -> trueForm
|
|
|
|
and doForm env = function
|
|
| [a] -> a
|
|
| a::rest ->
|
|
eval env a |> ignore
|
|
doForm env rest
|
|
| _ -> raise <| Core.errArity ()
|
|
|
|
and fnStarForm outer nodes =
|
|
let makeFunc binds body =
|
|
let f = fun nodes ->
|
|
let inner = Env.makeNew outer binds nodes
|
|
eval inner body
|
|
Env.makeFunc f body binds outer
|
|
|
|
match nodes with
|
|
| [List(binds); body] -> makeFunc binds body
|
|
| [Vector(seg); body] -> makeFunc (List.ofSeq seg) body
|
|
| [_; _] -> raise <| errExpected "bindings of list or vector"
|
|
| _ -> raise <| Core.errArity ()
|
|
|
|
and quoteForm = function
|
|
| [node] -> node
|
|
| _ -> raise <| Core.errArity ()
|
|
|
|
and eval env = function
|
|
| List(Symbol("def!")::rest) -> defBangForm env rest
|
|
| List(Symbol("let*")::rest) ->
|
|
let inner, form = letStarForm env rest
|
|
form |> eval inner
|
|
| List(Symbol("if")::rest) -> ifForm env rest |> eval env
|
|
| List(Symbol("do")::rest) -> doForm env rest |> eval env
|
|
| List(Symbol("fn*")::rest) -> fnStarForm env rest
|
|
| List(Symbol("quote")::rest) -> quoteForm rest
|
|
| List(Symbol("quasiquote")::rest) -> quasiquoteForm rest |> eval env
|
|
| List(_) as node ->
|
|
let resolved = node |> eval_ast env
|
|
match resolved with
|
|
| List(Func(_, f, _, _, [])::rest) -> f rest
|
|
| List(Func(_, _, body, binds, outer)::rest) ->
|
|
let inner = Env.makeNew outer binds rest
|
|
body |> eval inner
|
|
| _ -> raise <| errExpected "function"
|
|
| node -> node |> eval_ast env
|