diff options
author | Eduardo Julian | 2015-01-29 01:25:54 -0400 |
---|---|---|
committer | Eduardo Julian | 2015-01-29 01:25:54 -0400 |
commit | df59026eefe30d2a903adee14cea0cce95c92084 (patch) | |
tree | f256e4bf3f2b1cc57e6a075e2b7eb65cf064bd83 | |
parent | 5f492cd8f612906d25f6377731f71c7289fd4b8d (diff) |
[Bugs]
- Fixed the errors in the compiler due to its restructuring as monadic code.
- Fixed a bug in the analyser where 'case' forms has "Nothing" as their type.
[Refactor]
- Now there only one way to raise/fold lambda-bodies.
-rw-r--r-- | source/lux.lux | 59 | ||||
-rw-r--r-- | src/lux/analyser.clj | 199 | ||||
-rw-r--r-- | src/lux/compiler.clj | 172 |
3 files changed, 202 insertions, 228 deletions
diff --git a/source/lux.lux b/source/lux.lux index c40c26589..cab8a31d2 100644 --- a/source/lux.lux +++ b/source/lux.lux @@ -87,33 +87,38 @@ (annotate lambda Macro) (def' lambda (lambda' _ tokens - (case tokens - (#Cons self (#Cons (#Tuple (#Cons arg args')) (#Cons body #Nil))) - (#Form (#Cons (#Ident "lambda'") - (#Cons self - (#Cons arg - (case args' - #Nil - (#Cons body #Nil) - - _ - (#Cons (#Ident "lux:lambda") - (#Cons (#Tuple args') - (#Cons body #Nil)))))))) - - (#Cons (#Tuple (#Cons arg args')) (#Cons body #Nil)) - (#Form (#Cons (#Ident "lambda'") - (#Cons (#Ident "_") - (#Cons arg - (case args' - #Nil - (#Cons body #Nil) - - _ - (#Cons (#Ident "lux:lambda") - (#Cons (#Tuple args') - (#Cons body #Nil)))))))) - ))) + (lambda' _ state + (let output (case tokens + (#Cons self (#Cons (#Tuple (#Cons arg args')) (#Cons body #Nil))) + (#Form (#Cons (#Ident "lambda'") + (#Cons self + (#Cons arg + (case args' + #Nil + (#Cons body #Nil) + + _ + (#Cons (#Ident "lux:lambda") + (#Cons (#Tuple args') + (#Cons body #Nil)))))))) + + (#Cons (#Tuple (#Cons arg args')) (#Cons body #Nil)) + (#Form (#Cons (#Ident "lambda'") + (#Cons (#Ident "_") + (#Cons arg + (case args' + #Nil + (#Cons body #Nil) + + _ + (#Cons (#Ident "lux:lambda") + (#Cons (#Tuple args') + (#Cons body #Nil))))))))) + [(#Cons output #Nil) state])))) + +(def cons + (lambda [tail head] + (#Cons head tail))) #( (defmacro (lambda tokens) diff --git a/src/lux/analyser.clj b/src/lux/analyser.clj index 14bb533dc..d44c333b1 100644 --- a/src/lux/analyser.clj +++ b/src/lux/analyser.clj @@ -14,6 +14,8 @@ ;; [Util] (def +int-class+ "java.lang.Integer") +(def +dont-care-type+ [::&type/object "java.lang.Object" []]) + (defn ^:private annotated [form type] {:form form :type type}) @@ -411,7 +413,7 @@ (exec [[=test] (analyse-ast ?test) [=then] (analyse-ast ?then) [=else] (analyse-ast ?else)] - (return (list (annotated [::if =test =then =else] ::&type/nothing))))) + (return (list (annotated [::if =test =then =else] +dont-care-type+))))) (defn ^:private analyse-do [analyse-ast ?exprs] (exec [=exprs (do-all-m* (map analyse-ast ?exprs))] @@ -646,7 +648,7 @@ )) (partition 2 ?branches))] (return (->decision-tree $scope $base =branches)))] - (return (list (annotated [::case (dec $base) =variant registers mappings tree] ::&type/nothing)))))) + (return (list (annotated [::case (dec $base) =variant registers mappings tree] +dont-care-type+)))))) (defn ^:private analyse-let [analyse-ast ?label ?value ?body] (exec [[=value] (analyse-ast ?value) @@ -655,53 +657,82 @@ (analyse-ast ?body))] (return (list (annotated [::let idx ?label =value =body] (:type =body)))))) -(declare raise-bindings) -(defn ^:private raise-tree-bindings [outer-scope ?tree] - (case (:type ?tree) - ::adt* - (update-in ?tree [:patterns] - #(into {} (for [[?tag ?unapply] % - :let [=unapply (update-in ?unapply [:parts] (partial map (partial raise-tree-bindings outer-scope)))]] - [?tag =unapply]))) - - ::defaults - (update-in ?tree [:stores] - #(into {} (for [[?store ?branches] % - :let [=store (raise-bindings outer-scope {:form ?store :type ::&type/nothing})]] - [(:form =store) ?branches]))) - ;; else - (assert false (pr-str ?tree)) - )) +(defn ^:private raise-tree-bindings [raise-expr outer-scope ?tree] + (let [partial-f (partial raise-expr outer-scope) + tree-partial-f (partial raise-tree-bindings raise-expr outer-scope)] + (case (:type ?tree) + ::adt* + (update-in ?tree [:patterns] + #(into {} (for [[?tag ?unapply] %] + [?tag (update-in ?unapply [:parts] (partial map tree-partial-f))]))) + + ::defaults + (update-in ?tree [:stores] + #(into {} (for [[?store ?branches] % + :let [=store (partial-f {:form ?store :type ::&type/nothing})]] + [(:form =store) ?branches]))) + ;; else + (assert false (pr-str ?tree)) + ))) -(defn ^:private raise-bindings [outer-scope body] +(defn ^:private raise-expr [outer-scope syntax] ;; (prn 'raise-bindings body) - (match (:form body) - [::local ?scope ?idx] - {:form [::local outer-scope (inc ?idx)] - :type (:type body)} - - [::captured _ _ ?source] - ?source - - [::jvm:iadd ?x ?y] - {:form [::jvm:iadd - (raise-bindings outer-scope ?x) - (raise-bindings outer-scope ?y)] - :type (:type body)} - - [::case ?base ?variant ?registers ?mappings ?tree] - (let [=variant (raise-bindings outer-scope ?variant) - =mappings (into {} (for [[idx syntax] ?mappings] - [idx (raise-bindings outer-scope syntax)])) - =tree (raise-tree-bindings outer-scope ?tree)] - {:form [::case ?base =variant ?registers =mappings =tree] - :type (:type body)}) - - [::call ?func ?args] - {:form [::call (raise-bindings outer-scope ?func) - (map (partial raise-bindings outer-scope) ?args)] - :type (:type body)} - )) + (let [partial-f (partial raise-expr outer-scope) + tree-partial-f (partial raise-tree-bindings raise-expr outer-scope)] + (match (:form syntax) + [::literal ?value] + syntax + + [::tuple ?members] + {:form [::tuple (map partial-f ?members)] + :type (:type syntax)} + + [::variant ?tag ?members] + {:form [::variant ?tag (map partial-f ?members)] + :type (:type syntax)} + + [::local ?scope ?idx] + {:form [::local outer-scope (inc ?idx)] + :type (:type syntax)} + + [::captured _ _ ?source] + ?source + + [::self ?self-name ?curried] + {:form [::self outer-scope (mapv partial-f ?curried)] + :type (:type syntax)} + + [::jvm:iadd ?x ?y] + {:form [::jvm:iadd (partial-f ?x) (partial-f ?y)] + :type (:type syntax)} + + [::let ?idx ?name ?value ?body] + {:form [::let ?idx ?name (partial-f ?value) (partial-f ?body)] + :type (:type syntax)} + + [::case ?base ?variant ?registers ?mappings ?tree] + (let [=variant (partial-f ?variant) + =mappings (into {} (for [[idx syntax] ?mappings] + [idx (partial-f syntax)])) + =tree (tree-partial-f ?tree)] + {:form [::case ?base =variant ?registers =mappings =tree] + :type (:type syntax)}) + + [::lambda ?scope ?captured ?args ?value] + {:form [::lambda outer-scope + (into {} (for [[?name ?sub-syntax] ?captured] + [?name (partial-f ?sub-syntax)])) + ?args + ?value] + :type (:type syntax)} + + [::call ?func ?args] + {:form [::call (partial-f ?func) (map partial-f ?args)] + :type (:type syntax)} + + _ + (assert false (pr-str (:form syntax))) + ))) (defn ^:private analyse-lambda [analyse-ast ?self ?arg ?body] (exec [[_ =arg =return :as =function] (within ::types &type/fresh-function) @@ -717,8 +748,7 @@ :let [;; _ (prn '(:form =body) (:form =body)) =lambda (match (:form =body) [::lambda ?sub-scope ?sub-captured ?sub-args ?sub-body] - (let [?sub-body* (raise-bindings =scope ?sub-body)] - [::lambda =scope =captured (cons ?arg ?sub-args) ?sub-body*]) + [::lambda =scope =captured (cons ?arg ?sub-args) (raise-expr =scope ?sub-body)] _ [::lambda =scope =captured (list ?arg) =body])] @@ -726,72 +756,6 @@ ] (return (list (annotated =lambda =function))))) -(declare ->def-lambda) -(defn ^:private ->def-lambda-tree [old-scope new-scope ?tree] - (case (:type ?tree) - ::adt* - (update-in ?tree [:patterns] - #(into {} (for [[?tag ?unapply] % - :let [=unapply (update-in ?unapply [:parts] (partial map (partial ->def-lambda-tree old-scope new-scope)))]] - [?tag =unapply]))) - - ::defaults - (update-in ?tree [:stores] - #(into {} (for [[?store ?branches] % - :let [=store (->def-lambda old-scope new-scope {:form ?store :type ::&type/nothing})]] - [(:form =store) ?branches]))) - ;; else - (assert false (pr-str ?tree)) - )) - -(defn ^:private ->def-lambda [old-scope new-scope syntax] - (match (:form syntax) - [::literal _] - syntax - - [::variant ?tag ?elems] - {:form [::variant ?tag (map (partial ->def-lambda old-scope new-scope) ?elems)] - :type (:type syntax)} - - [::local ?local-scope ?idx] - {:form [::local new-scope (inc ?idx)] - :type (:type syntax)} - - [::self ?self-name ?curried] - (if (= ?self-name old-scope) - {:form [::self new-scope (mapv (partial ->def-lambda old-scope new-scope) ?curried)] - :type (:type syntax)} - syntax) - - - [::jvm:iadd ?x ?y] - {:form [::jvm:iadd (->def-lambda old-scope new-scope ?x) (->def-lambda old-scope new-scope ?y)] - :type (:type syntax)} - - [::case ?base ?variant ?registers ?mappings ?tree] - (let [=variant (->def-lambda old-scope new-scope ?variant) - =mappings (into {} (for [[idx syntax] ?mappings] - [idx (->def-lambda old-scope new-scope syntax)])) - =tree (->def-lambda-tree old-scope new-scope ?tree)] - {:form [::case ?base =variant ?registers =mappings =tree] - :type (:type syntax)}) - - [::call ?func ?args] - {:form [::call (->def-lambda old-scope new-scope ?func) - (map (partial ->def-lambda old-scope new-scope) ?args)] - :type (:type syntax)} - - [::lambda ?scope ?captured ?args ?value] - {:form [::lambda new-scope - (into {} (for [[?name ?sub-syntax] ?captured] - [?name (->def-lambda old-scope new-scope ?sub-syntax)])) - ?args - ?value] - :type (:type syntax)} - - _ - (assert false (pr-str (:form syntax))))) - (defn ^:private analyse-def [analyse-ast ?name ?value] (exec [def?? (defined? ?name)] (if def?? @@ -802,9 +766,10 @@ (analyse-ast ?value)) ;; :let [_ (prn 'DEF/PRE =value)] :let [;; _ (prn 'analyse-def/=value =value) + new-scope [$module ?name] =value (match (:form =value) - [::lambda ?scope ?env ?args ?body] - {:form [::lambda ?scope ?env ?args (->def-lambda ?scope [$module ?name] ?body)] + [::lambda ?old-scope ?env ?args ?body] + {:form [::lambda new-scope ?env ?args (raise-expr new-scope ?body)] :type (:type =value)} _ diff --git a/src/lux/compiler.clj b/src/lux/compiler.clj index e9a445510..f6daaca0f 100644 --- a/src/lux/compiler.clj +++ b/src/lux/compiler.clj @@ -161,22 +161,23 @@ (defn ^:private compile-tuple [compile *type* ?elems] (exec [*writer* get-writer - :let [_ (let [num-elems (count ?elems)] - (let [tuple-class (str (str +prefix+ "/Tuple") num-elems)] - (doto *writer* - (.visitTypeInsn Opcodes/NEW tuple-class) - (.visitInsn Opcodes/DUP) - (.visitMethodInsn Opcodes/INVOKESPECIAL tuple-class "<init>" "()V")) - (dotimes [idx num-elems] - (.visitInsn *writer* Opcodes/DUP) - (compile (nth ?elems idx)) - (.visitFieldInsn *writer* Opcodes/PUTFIELD tuple-class (str "_" (inc idx)) "Ljava/lang/Object;"))))]] + :let [num-elems (count ?elems) + tuple-class (str (str +prefix+ "/Tuple") num-elems) + _ (doto *writer* + (.visitTypeInsn Opcodes/NEW tuple-class) + (.visitInsn Opcodes/DUP) + (.visitMethodInsn Opcodes/INVOKESPECIAL tuple-class "<init>" "()V"))] + _ (map-m (fn [idx] + (exec [:let [_ (.visitInsn *writer* Opcodes/DUP)] + ret (compile (nth ?elems idx)) + :let [_ (.visitFieldInsn *writer* Opcodes/PUTFIELD tuple-class (str "_" (inc idx)) "Ljava/lang/Object;")]] + (return ret))) + (range num-elems))] (return nil))) (defn ^:private compile-local [compile *type* ?env ?idx] (exec [*writer* get-writer - :let [_ (doto *writer* - (.visitVarInsn Opcodes/ALOAD (int ?idx)))]] + :let [_ (.visitVarInsn *writer* Opcodes/ALOAD (int ?idx))]] (return nil))) (defn ^:private compile-captured [compile *type* ?scope ?captured-id ?source] @@ -191,8 +192,7 @@ (defn ^:private compile-global [compile *type* ?owner-class ?name] (exec [*writer* get-writer - :let [_ (doto *writer* - (.visitFieldInsn Opcodes/GETSTATIC (->class (str ?owner-class "$" (normalize-ident ?name))) "_datum" "Ljava/lang/Object;"))]] + :let [_ (.visitFieldInsn *writer* Opcodes/GETSTATIC (->class (str ?owner-class "$" (normalize-ident ?name))) "_datum" "Ljava/lang/Object;")]] (return nil))) (defn ^:private compile-global-fn [compile *type* ?owner-class ?name] @@ -247,8 +247,7 @@ (defn ^:private compile-jvm-getstatic [compile *type* ?owner ?field] (exec [*writer* get-writer - :let [_ (doto *writer* - (.visitFieldInsn Opcodes/GETSTATIC (->class ?owner) ?field (->java-sig *type*)))]] + :let [_ (.visitFieldInsn *writer* Opcodes/GETSTATIC (->class ?owner) ?field (->java-sig *type*))]] (return nil))) (defn prepare-arg! [*writer* class-name] @@ -310,30 +309,32 @@ (defn ^:private compile-jvm-invokevirtual [compile *type* ?class ?method ?classes ?object ?args] (exec [*writer* get-writer - :let [_ (let [method-sig (str "(" (reduce str "" (map ->type-signature ?classes)) ")" (->java-sig *type*))] - (compile ?object) - (.visitTypeInsn *writer* Opcodes/CHECKCAST (->class ?class)) - (doseq [[class-name arg] (map vector ?classes ?args)] - (do (compile arg) - (prepare-arg! *writer* class-name))) - (.visitMethodInsn *writer* Opcodes/INVOKEVIRTUAL (->class ?class) ?method method-sig) - (prepare-return! *writer* *type*) - )]] + :let [method-sig (str "(" (reduce str "" (map ->type-signature ?classes)) ")" (->java-sig *type*))] + _ (compile ?object) + :let [_ (.visitTypeInsn *writer* Opcodes/CHECKCAST (->class ?class))] + _ (map-m (fn [[class-name arg]] + (exec [ret (compile arg) + :let [_ (prepare-arg! *writer* class-name)]] + (return ret))) + (map vector ?classes ?args)) + :let [_ (do (.visitMethodInsn *writer* Opcodes/INVOKEVIRTUAL (->class ?class) ?method method-sig) + (prepare-return! *writer* *type*))]] (return nil))) (defn ^:private compile-jvm-new [compile *type* ?class ?classes ?args] (exec [*writer* get-writer - :let [_ (let [init-sig (str "(" (reduce str "" (map ->type-signature ?classes)) ")V") - class* (->class ?class)] - (doto *writer* - (.visitTypeInsn Opcodes/NEW class*) - (.visitInsn Opcodes/DUP)) - (doseq [[class-name arg] (map vector ?classes ?args)] - (do (compile arg) - (prepare-arg! *writer* class-name))) - (doto *writer* - (.visitMethodInsn Opcodes/INVOKESPECIAL class* "<init>" init-sig)) - )]] + :let [init-sig (str "(" (reduce str "" (map ->type-signature ?classes)) ")V") + class* (->class ?class) + _ (doto *writer* + (.visitTypeInsn Opcodes/NEW class*) + (.visitInsn Opcodes/DUP))] + _ (map-m (fn [[class-name arg]] + (exec [ret (compile arg) + :let [_ (prepare-arg! *writer* class-name)]] + (return ret))) + (map vector ?classes ?args)) + :let [_ (doto *writer* + (.visitMethodInsn Opcodes/INVOKESPECIAL class* "<init>" init-sig))]] (return nil))) (defn ^:private compile-jvm-new-array [compile *type* ?class ?length] @@ -345,18 +346,18 @@ (defn ^:private compile-jvm-aastore [compile *type* ?array ?idx ?elem] (exec [*writer* get-writer + _ (compile ?array) :let [_ (doto *writer* - (do (compile ?array)) (.visitInsn Opcodes/DUP) - (.visitLdcInsn (int ?idx)) - (do (compile ?elem)) - (.visitInsn Opcodes/AASTORE))]] + (.visitLdcInsn (int ?idx)))] + _ (compile ?elem) + :let [_ (.visitInsn *writer* Opcodes/AASTORE)]] (return nil))) (defn ^:private compile-jvm-aaload [compile *type* ?array ?idx] (exec [*writer* get-writer + _ (compile ?array) :let [_ (doto *writer* - (do (compile ?array)) (.visitLdcInsn (int ?idx)) (.visitInsn Opcodes/AALOAD))]] (return nil))) @@ -364,27 +365,29 @@ (let [+bool-class+ (->class "java.lang.Boolean")] (defn ^:private compile-if [compile *type* ?test ?then ?else] (exec [*writer* get-writer - :let [_ (let [else-label (new Label) - end-label (new Label)] - (compile ?test) - (doto *writer* - (.visitTypeInsn Opcodes/CHECKCAST +bool-class+) - (.visitMethodInsn Opcodes/INVOKEVIRTUAL +bool-class+ "booleanValue" "()Z") - (.visitJumpInsn Opcodes/IFEQ else-label)) - (compile ?then) - (doto *writer* - (.visitJumpInsn Opcodes/GOTO end-label) - (.visitLabel else-label)) - (compile ?else) - (.visitLabel *writer* end-label))]] + :let [else-label (new Label) + end-label (new Label)] + _ (compile ?test) + :let [_ (doto *writer* + (.visitTypeInsn Opcodes/CHECKCAST +bool-class+) + (.visitMethodInsn Opcodes/INVOKEVIRTUAL +bool-class+ "booleanValue" "()Z") + (.visitJumpInsn Opcodes/IFEQ else-label))] + _ (compile ?then) + :let [_ (doto *writer* + (.visitJumpInsn Opcodes/GOTO end-label) + (.visitLabel else-label))] + _ (compile ?else) + :let [_ (.visitLabel *writer* end-label)]] (return nil)))) (defn ^:private compile-do [compile *type* ?exprs] (exec [*writer* get-writer - :let [_ (do (doseq [expr (butlast ?exprs)] - (compile expr) - (.visitInsn *writer* Opcodes/POP)) - (compile (last ?exprs)))]] + _ (map-m (fn [expr] + (exec [ret (compile expr) + :let [_ (.visitInsn *writer* Opcodes/POP)]] + (return ret))) + (butlast ?exprs)) + _ (compile (last ?exprs))] (return nil))) (let [+tag-sig+ (->type-signature "java.lang.String") @@ -633,16 +636,16 @@ (defn ^:private compile-let [compile *type* ?idx ?label ?value ?body] (exec [*writer* get-writer - :let [_ (let [start-label (new Label) - end-label (new Label) - ?idx (int ?idx)] - (.visitLocalVariable *writer* (normalize-ident ?label) (->java-sig (:type ?value)) nil start-label end-label ?idx) - (compile ?value) - (doto *writer* - (.visitVarInsn Opcodes/ASTORE ?idx) - (.visitLabel start-label)) - (compile ?body) - (.visitLabel *writer* end-label))]] + :let [start-label (new Label) + end-label (new Label) + ?idx (int ?idx) + _ (.visitLocalVariable *writer* (normalize-ident ?label) (->java-sig (:type ?value)) nil start-label end-label ?idx)] + _ (compile ?value) + :let [_ (doto *writer* + (.visitVarInsn Opcodes/ASTORE ?idx) + (.visitLabel start-label))] + _ (compile ?body) + :let [_ (.visitLabel *writer* end-label)]] (return nil))) (defn compile-field [compile ?name body] @@ -874,7 +877,7 @@ (compile-method compile ?name ?value) _ - (compile-field compile ?name ?value))] + (compile-field compile ?name ?value))] (return nil))) (defn ^:private compile-defclass [compile *type* ?package ?name ?super-class ?members] @@ -919,19 +922,20 @@ (defn ^:private compile-variant [compile *type* ?tag ?members] (exec [*writer* get-writer - :let [_ (let [variant-class* (str (->class +variant-class+) (count ?members))] - (doto *writer* - (.visitTypeInsn Opcodes/NEW variant-class*) - (.visitInsn Opcodes/DUP) - (.visitMethodInsn Opcodes/INVOKESPECIAL variant-class* "<init>" "()V") - (.visitInsn Opcodes/DUP) - (.visitLdcInsn ?tag) - (.visitFieldInsn Opcodes/PUTFIELD variant-class* "tag" (->type-signature "java.lang.String")) - (-> (doto (.visitInsn Opcodes/DUP) - (do (compile ?member)) - (.visitFieldInsn Opcodes/PUTFIELD variant-class* (str "_" (inc ?tfield)) "Ljava/lang/Object;")) - (->> (doseq [[?tfield ?member] (mapv vector (range (count ?members)) ?members)])))) - )]] + :let [variant-class* (str (->class +variant-class+) (count ?members)) + _ (doto *writer* + (.visitTypeInsn Opcodes/NEW variant-class*) + (.visitInsn Opcodes/DUP) + (.visitMethodInsn Opcodes/INVOKESPECIAL variant-class* "<init>" "()V") + (.visitInsn Opcodes/DUP) + (.visitLdcInsn ?tag) + (.visitFieldInsn Opcodes/PUTFIELD variant-class* "tag" (->type-signature "java.lang.String")))] + _ (map-m (fn [[?tfield ?member]] + (exec [:let [_ (.visitInsn *writer* Opcodes/DUP)] + ret (compile ?member) + :let [_ (.visitFieldInsn *writer* Opcodes/PUTFIELD variant-class* (str "_" (inc ?tfield)) "Ljava/lang/Object;")]] + (return ret))) + (map vector (range (count ?members)) ?members))] (return nil))) (let [+int-class+ (->class "java.lang.Integer")] @@ -958,7 +962,7 @@ ^:private compile-jvm-irem Opcodes/IREM )) -(defn compile-self-call [?scope ?assumed-args] +(defn compile-self-call [compile ?scope ?assumed-args] (exec [*writer* get-writer :let [lambda-class (->class (reduce str "" (interpose "$" (map normalize-ident ?scope)))) _ (doto *writer* @@ -1058,7 +1062,7 @@ (compile-defclass compile (:type syntax) ?package ?name ?super-class ?members) [::&analyser/self ?scope ?assumed-args] - (compile-self-call ?scope ?assumed-args) + (compile-self-call compile ?scope ?assumed-args) )) ;; [Interface] |