Skip to content

Commit

Permalink
Hy: step5
Browse files Browse the repository at this point in the history
  • Loading branch information
kanaka committed Sep 21, 2017
1 parent 379c799 commit 45f4d83
Show file tree
Hide file tree
Showing 2 changed files with 124 additions and 0 deletions.
109 changes: 109 additions & 0 deletions hy/step5_tco.hy
Original file line number Diff line number Diff line change
@@ -0,0 +1,109 @@
#!/usr/bin/env hy

(import [hy.models [HyDict :as Map HySymbol :as Sym]])
(import sys traceback)
(import [reader [read-str Blank]])
(import [printer [pr-str]])
(import [env [env-new env-get env-set]])
(import core)

;; read
(defn READ [str]
(read-str str))

;; eval
(defn eval-ast [ast env]
;;(print "eval-ast:" ast (type ast))
(if
(symbol? ast) (env-get env ast)
(instance? Map ast) (Map (map (fn [x] (EVAL x env)) ast))
(instance? tuple ast) (tuple (map (fn [x] (EVAL x env)) ast))
(instance? list ast) (list (map (fn [x] (EVAL x env)) ast))
True ast))

(defn EVAL [ast env]
;;(print "EVAL:" ast (type ast) (instance? tuple ast))
(setv res None)
(while True
(if (not (instance? tuple ast))
(setv res (eval-ast ast env))

;; apply list
(do
(setv [a0 a1 a2] [(nth ast 0) (nth ast 1) (nth ast 2)])
(if
(none? a0)
(setv res ast)

(= (Sym "def!") a0)
(setv res (env-set env a1 (EVAL a2 env)))

(= (Sym "let*") a0)
(do
(setv env (env-new env))
(for [[b e] (partition a1 2)]
(env-set env b (EVAL e env)))
(setv ast a2)
(continue)) ;; TCO

(= (Sym "do") a0)
(do (eval-ast (list (butlast (rest ast))) env)
(setv ast (last ast))
(continue)) ;; TCO

(= (Sym "if") a0)
(do
(setv cond (EVAL a1 env))
(if (or (none? cond) (and (instance? bool cond)
(= cond False)))
(if (> (len ast) 2)
(do (setv ast (nth ast 3)) (continue)) ;; TCO
(setv res None))
(do (setv ast a2) (continue)))) ;; TCO

(= (Sym "fn*") a0)
(setv func (fn [&rest args]
(EVAL a2 (env-new env a1 (or args []))))
func.ast a2
func.env env
func.params a1
res func)

;; apply
(do
(setv el (eval-ast ast env)
f (first el)
args (list (rest el)))
(if (hasattr f "ast")
(do (setv ast f.ast
env (env-new f.env f.params args))
(continue)) ;; TCO
(setv res (apply f args)))))))
(break))
res)

;; print
(defn PRINT [exp]
(pr-str exp True))

;; repl
(def repl-env (env-new))
(defn REP [str]
(PRINT (EVAL (READ str) repl-env)))

;; core.hy: defined using Hy
(for [k core.ns]
(env-set repl-env (Sym k) (get core.ns k)))

;; core.mal: defined using the language itself
(REP "(def! not (fn* [a] (if a false true)))")

(while True
(try
(do (setv line (raw_input "user> "))
(if (= "" line) (continue))
(print (REP line)))
(except [EOFError] (break))
(except [Blank])
(except []
(print (.join "" (apply traceback.format_exception (.exc_info sys)))))))
15 changes: 15 additions & 0 deletions hy/tests/step5_tco.mal
Original file line number Diff line number Diff line change
@@ -0,0 +1,15 @@
;; Test recursive non-tail call function

(def! sum-to (fn* (n) (if (= n 0) 0 (+ n (sum-to (- n 1))))))

(sum-to 10)
;=>55

;;; no try* yet, so test completion of side-effects
(def! res1 nil)
;=>nil
;;; For implementations without their own TCO this should fail and
;;; leave res1 unchanged
(def! res1 (sum-to 10000))
res1
;=>nil

0 comments on commit 45f4d83

Please sign in to comment.