1
1
mirror of https://github.com/kanaka/mal.git synced 2024-08-17 01:30:26 +03:00

Add step 3

This commit is contained in:
sogaiu 2020-11-24 22:22:37 +09:00
parent 73da0bdbd9
commit c916bde23d
2 changed files with 190 additions and 0 deletions

75
impls/fennel/env.fnl Normal file
View File

@ -0,0 +1,75 @@
(local t (require :types))
(local u (require :utils))
(fn make-env
[outer]
{:outer outer
:data {}})
(fn env-set
[env sym-ast val-ast]
(tset (. env :data)
(t.get-value sym-ast)
val-ast)
env)
(fn env-find
[env sym-ast]
(let [inner-env (. env :data)
val-ast (. inner-env (t.get-value sym-ast))]
(if val-ast
env
(let [outer (. env :outer)]
(when outer
(env-find outer sym-ast))))))
(fn env-get
[env sym-ast]
(let [target-env (env-find env sym-ast)]
(if target-env
(. (. target-env :data)
(t.get-value sym-ast))
(u.throw*
(t.make-string (.. "'" (t.get-value sym-ast) "'"
" not found"))))))
(comment
(local test-env (make-env {}))
(env-set test-env
(t.make-symbol "fun")
(t.make-number 1))
(env-find test-env (t.make-symbol "fun"))
(env-get test-env (t.make-symbol "fun"))
(local test-env-2 (make-env nil))
(env-set test-env-2
(t.make-symbol "smile")
(t.make-keyword ":yay"))
(env-find test-env-2 (t.make-symbol "smile"))
(env-get test-env-2 (t.make-symbol "smile"))
(local test-env-3 (make-env nil))
(env-set test-env-3
(t.make-symbol "+")
(fn [ast-1 ast-2]
(t.make-number (+ (t.get-value ast-1)
(t.get-value ast-2)))))
(env-find test-env-3 (t.make-symbol "+"))
(env-get test-env-3 (t.make-symbol "+"))
)
{:make-env make-env
:env-set env-set
:env-find env-find
:env-get env-get}

115
impls/fennel/step3_env.fnl Normal file
View File

@ -0,0 +1,115 @@
(local printer (require :printer))
(local reader (require :reader))
(local t (require :types))
(local e (require :env))
(local u (require :utils))
(local repl_env
(-> (e.make-env nil)
(e.env-set (t.make-symbol "+")
(fn [ast-1 ast-2]
(t.make-number (+ (t.get-value ast-1)
(t.get-value ast-2)))))
(e.env-set (t.make-symbol "-")
(fn [ast-1 ast-2]
(t.make-number (- (t.get-value ast-1)
(t.get-value ast-2)))))
(e.env-set (t.make-symbol "*")
(fn [ast-1 ast-2]
(t.make-number (* (t.get-value ast-1)
(t.get-value ast-2)))))
(e.env-set (t.make-symbol "/")
(fn [ast-1 ast-2]
(t.make-number (/ (t.get-value ast-1)
(t.get-value ast-2)))))))
(fn READ
[arg]
(reader.read_str arg))
;; forward declaration
(var EVAL 1)
(fn eval_ast
[ast env]
(if (t.symbol?* ast)
(e.env-get env ast)
;;
(t.list?* ast)
(t.make-list (u.map (fn [elt-ast]
(EVAL elt-ast env))
(t.get-value ast)))
;;
(t.vector?* ast)
(t.make-vector (u.map (fn [elt-ast]
(EVAL elt-ast env))
(t.get-value ast)))
;;
(t.hash-map?* ast)
(t.make-hash-map (u.map (fn [elt-ast]
(EVAL elt-ast env))
(t.get-value ast)))
;;
ast))
(set EVAL
(fn [ast env]
(if (not (t.list?* ast))
(eval_ast ast env)
;;
(t.empty?* ast)
ast
;;
(let [ast-elts (t.get-value ast)
head-name (t.get-value (. ast-elts 1))]
;; XXX: want to check for symbol, but that screws up logic below
(if (= "def!" head-name)
(let [def-name (. ast-elts 2)
def-val (EVAL (. ast-elts 3) env)]
(e.env-set env
def-name def-val)
def-val)
;;
(= "let*" head-name)
(let [new-env (e.make-env env)
bindings (t.get-value (. ast-elts 2))
stop (/ (length bindings) 2)]
(for [idx 1 stop]
(let [b-name (. bindings (- (* 2 idx) 1))
b-val (EVAL (. bindings (* 2 idx)) new-env)]
(e.env-set new-env
b-name b-val)))
(EVAL (. ast-elts 3) new-env))
;;
(let [eval-list (t.get-value (eval_ast ast env))
f (. eval-list 1)
args (u.slice eval-list 2 -1)]
(f (table.unpack args))))))))
(fn PRINT
[ast]
(printer.pr_str ast true))
(fn rep
[code-str]
(PRINT (EVAL (READ code-str) repl_env)))
(fn handle-error
[err]
(if (t.nil?* err)
(print)
(= "string" (type err))
(print err)
(print (.. "Error: " (PRINT err)))))
(var done false)
(while (not done)
(io.write "user> ")
(io.flush)
(let [input (io.read)]
(if (not input)
(set done true)
(xpcall (fn []
(print (rep input)))
handle-error))))