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

Add step 5

This commit is contained in:
sogaiu 2020-11-24 22:33:26 +09:00
parent 9d4c05e0eb
commit e667abecf3
2 changed files with 177 additions and 2 deletions

150
impls/fennel/step5_tco.fnl Normal file
View File

@ -0,0 +1,150 @@
(local printer (require :printer))
(local reader (require :reader))
(local t (require :types))
(local e (require :env))
(local core (require :core))
(local u (require :utils))
(local repl_env
(let [env (e.make-env)]
(each [name func (pairs core)]
(e.env-set env
(t.make-symbol name)
func))
env))
(fn READ
[code-str]
(reader.read_str code-str))
;; 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-param env-param]
(var ast ast-param)
(var env env-param)
(var result nil)
(while (not result)
(if (not (t.list?* ast))
(set result (eval_ast ast env))
;;
(t.empty?* ast)
(set result 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)
(set result 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)))
;; tco
(set ast (. ast-elts 3))
(set env new-env))
;;
(= "do" head-name)
(let [most-forms (u.slice ast-elts 2 -2) ;; XXX
last-body-form (u.last ast-elts)
res-ast (eval_ast (t.make-list most-forms) env)]
;; tco
(set ast last-body-form))
;;
(= "if" head-name)
(let [cond-res (EVAL (. ast-elts 2) env)]
(if (or (t.nil?* cond-res)
(t.false?* cond-res))
(let [else-ast (. ast-elts 4)]
(if (not else-ast)
;; tco
(set result t.mal-nil)
(set ast else-ast)))
;; tco
(set ast (. ast-elts 3))))
;;
(= "fn*" head-name)
(let [params (t.get-value (. ast-elts 2))
body (. ast-elts 3)]
;; tco
(set result
(t.make-fn (fn [args]
(EVAL body
(e.make-env env params args)))
body params env)))
;;
(let [eval-list (t.get-value (eval_ast ast env))
f (. eval-list 1)
args (u.slice eval-list 2 -1)]
(let [body (t.get-ast f)] ;; tco
(if body
(do
(set ast body)
(set env (e.make-env (t.get-env f)
(t.get-params f) args)))
(set result
((t.get-value f) args)))))))))
result))
(fn PRINT
[ast]
(printer.pr_str ast true))
(fn rep
[code-str]
(PRINT (EVAL (READ code-str) repl_env)))
(rep "(def! not (fn* (a) (if a false true)))")
(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))))

View File

@ -46,9 +46,12 @@
:content elts})
(fn make-fn
[a-fn]
[a-fn ast params env]
{:tag :fn
:content a-fn})
:content a-fn
:ast ast
:params params
:env env})
(local mal-true (make-boolean true))
@ -66,6 +69,20 @@
;;
(fn get-ast
[ast]
(. ast :ast))
(fn get-params
[ast]
(. ast :params))
(fn get-env
[ast]
(. ast :env))
;;
(fn nil?*
[ast]
(= :nil (. ast :tag)))
@ -102,6 +119,10 @@
[ast]
(= :hash-map (. ast :tag)))
(fn fn?*
[ast]
(= :fn (. ast :tag)))
;;
(fn empty?*
@ -194,6 +215,9 @@
:mal-false mal-false
;;
:get-value get-value
:get-ast get-ast
:get-params get-params
:get-env get-env
;;
:nil?* nil?*
:boolean?* boolean?*
@ -204,6 +228,7 @@
:list?* list?*
:vector?* vector?*
:hash-map?* hash-map?*
:fn?* fn?*
;;
:empty?* empty?*
:true?* true?*