diff options
Diffstat (limited to '')
-rw-r--r-- | src/lux/analyser.clj | 403 | ||||
-rw-r--r-- | src/lux/compiler.clj | 529 | ||||
-rw-r--r-- | src/lux/type.clj | 6 | ||||
-rw-r--r-- | src/lux/util.clj | 36 |
4 files changed, 599 insertions, 375 deletions
diff --git a/src/lux/analyser.clj b/src/lux/analyser.clj index cde2dd9bf..f2e589646 100644 --- a/src/lux/analyser.clj +++ b/src/lux/analyser.clj @@ -3,7 +3,6 @@ [clojure.core.match :refer [match]] (lux [util :as &util :refer [exec return* return fail fail* repeat-m try-all-m map-m mapcat-m reduce-m - within normalize-ident]] [parser :as &parser] [type :as &type] @@ -13,53 +12,49 @@ ;; [Util] (def ^:private +dont-care-type+ [::&type/Any]) -(defn ^:private fresh-env [name] - {:name name - :inner-closures 0 - :locals &util/+init-env+ - :closure &util/+init-env+}) - -(defn ^:private annotate [name access macro? type] +(defn ^:private annotate [module name access type] (fn [state] - (let [full-name (str (::&util/current-module state) &util/+name-separator+ name) - bound [::Expression [::global (::&util/current-module state) name] type]] + (let [full-name (str module &util/+name-separator+ name) + bound [::Expression [::global module name] type]] [::&util/ok [(-> state - (assoc-in [::&util/modules (::&util/current-module state) name] {:args-n [:None] - :access access - :macro? macro? - :type type - :defined? false}) + (assoc-in [::&util/modules module name] {:args-n [:None] + :access access + :type type + :defined? false}) (update-in [::&util/global-env] merge {full-name bound, name bound})) nil]]))) +(defn ^:private declare-macro [module name] + (fn [state] + [::&util/ok [(assoc-in state [::&util/modules module :macros name] true) + nil]])) + (defn ^:private expr-type [syntax+] (match syntax+ [::Expression _ type] (return type) _ - (fail "Can't retrieve the type of a statement."))) + (fail "Can't retrieve the type of a non-expression."))) -(defn ^:private define [name] - (fn [state] - (if-let [{:keys [type]} (get-in state [::&util/modules (::&util/current-module state) name])] - [::&util/ok [(-> state - (assoc-in [::&util/modules (::&util/current-module state) name :defined?] true) - (update-in [::&util/global-env] merge {full-name bound, name bound})) - nil]] - (fail* (str "[Analyser Error] Can't define an unannotated element: " name))))) +(defn ^:private define [module name] + (exec [? annotated? + _ (assert! ? (str "[Analyser Error] Can't define an unannotated element: " name))] + (fn [state] + [::&util/ok [(assoc-in state [::&util/modules module name :defined?] true) + nil]]))) -(defn ^:private defined? [name] +(defn ^:private defined? [module name] (fn [state] - [::&util/ok [state (get-in state [::&util/modules (::&util/current-module state) name :defined?])]])) + [::&util/ok [state (get-in state [::&util/modules module name :defined?])]])) -(defn ^:private annotated? [name] +(defn ^:private annotated? [module name] (fn [state] - [::&util/ok [state (boolean (get-in state [::&util/modules (::&util/current-module state) name]))]])) + [::&util/ok [state (boolean (get-in state [::&util/modules module name]))]])) -(defn ^:private is-macro? [module name] +(defn ^:private macro? [module name] (fn [state] - [::&util/ok [state (boolean (get-in state [::&util/modules module name :macro?]))]])) + [::&util/ok [state (boolean (get-in state [::&util/modules module :macros name]))]])) (def ^:private next-local-idx (fn [state] @@ -114,12 +109,16 @@ (fn [state] [::&util/ok [state (-> state ::&util/local-envs first :closure :mappings)]])) -(defn ^:private analyse-n [elems] - (let [num-inputs (count elems)] - (exec [output (mapcat-m analyse-ast elems) - _ (&util/assert! (= num-inputs (count output)) - (str "[Analyser Error] Can't expand to other than " num-inputs " elements."))] - (return output)))) +(defn ^:private analyse-1 [elem] + (exec [output (analyse-ast elem) + _ (&util/assert! (= 1 (count output)) "[Analyser Error] Can't expand to other than 1 element.")] + (return (first output)))) + +(defn ^:private analyse-2 [el1 el2] + (exec [output (mapcat-m analyse-ast (list el1 el2)) + _ (&util/assert! (= 2 (count output)) + "[Analyser Error] Can't expand to other than 2 elements.")] + (return [(first output) (second output)]))) (defn ^:private with-lambda [self self-type arg arg-type body] (fn [state] @@ -128,7 +127,6 @@ (with-let self :self self-type (with-let arg :local arg-type (exec [=return body - =next next-local-idx =captured captured-vars] (return [$scope =next =captured =return]))))))] (body* (update-in state [::&util/local-envs] #(cons (update-in (first %) [:inner-closures] inc) @@ -184,13 +182,13 @@ )) (defn ^:private analyse-call [analyse-ast ?fn ?args] - (exec [[=fn] (analyse-n (list ?fn)) + (exec [=fn (analyse-1 ?fn) loader &util/loader] (match =fn [::Expression =fn-form =fn-type] (match =fn-form [::global ?module ?name] - (exec [macro? (is-macro? ?module ?name)] + (exec [macro? (macro? ?module ?name)] (if macro? (let [macro-class (str ?module "$" (normalize-ident ?name)) output (-> (.loadClass loader macro-class) @@ -344,7 +342,7 @@ (let [branches* (reduce fold-branch base-struct data)] (match branches* [::BoolTests _] branches* - [::IntTests _] branches* + [::IntTests _] branches* [::RealTests _] branches* [::CharTests _] branches* [::TextTests _] branches* @@ -438,84 +436,85 @@ max-registers (reduce max 0 (map count vars))] [max-registers branch-mappings (generate-branches branches**)]))) +(defn ^:private locals-getter [?member] + (match ?member + [::&parser/Ident ?name] + (list [?name +dont-care-type+]) + + [::&parser/Tuple ?submembers] + (mapcat locals-getter ?submembers) + + [::&parser/Form ([[::&parser/Tag ?subtag] & ?submembers] :seq)] + (mapcat locals-getter ?submembers) + + _ + (list) + )) + (defn ^:private analyse-case-branches [branches] (map-m (fn [[?pattern ?body]] (match ?pattern [::&parser/Bool ?token] - (exec [[=body] (analyse-n (list ?body))] + (exec [=body (analyse-1 ?body)] (return [::case-branch ?pattern =body])) [::&parser/Int ?token] - (exec [[=body] (analyse-n (list ?body))] + (exec [=body (analyse-1 ?body)] (return [::case-branch ?pattern =body])) [::&parser/Real ?token] - (exec [[=body] (analyse-n (list ?body))] + (exec [=body (analyse-1 ?body)] (return [::case-branch ?pattern =body])) [::&parser/Char ?token] - (exec [[=body] (analyse-n (list ?body))] + (exec [=body (analyse-1 ?body)] (return [::case-branch ?pattern =body])) [::&parser/Text ?token] - (exec [[=body] (analyse-n (list ?body))] + (exec [=body (analyse-1 ?body)] (return [::case-branch ?pattern =body])) [::&parser/Ident ?name] - (exec [[=body] (with-let ?name :local +dont-care-type+ - (analyse-n (list ?body)))] + (exec [=body (with-let ?name :local +dont-care-type+ + (analyse-1 ?body))] (return [::case-branch ?pattern =body])) [::&parser/Tag ?tag] - (exec [[=body] (analyse-n (list ?body))] + (exec [=body (analyse-1 ?body)] (return [::case-branch ?pattern =body])) - + [::&parser/Tuple ?members] - (exec [[=body] (with-lets (mapcat locals-getter ?members) - (analyse-n (list ?body)))] + (exec [=body (with-lets (mapcat locals-getter ?members) + (analyse-1 ?body))] (return [::case-branch ?pattern =body])) [::&parser/Form ([[::&parser/Tag ?tag] & ?members] :seq)] - (exec [[=body] (with-lets (mapcat locals-getter ?members) - (analyse-n (list ?body)))] + (exec [=body (with-lets (mapcat locals-getter ?members) + (analyse-1 ?body))] (return [::case-branch ?pattern =body])) )) branches)) -(let [locals-getter (fn locals-getter [?member] - (match ?member - [::&parser/Ident ?name] - (list [?name +dont-care-type+]) - - [::&parser/Tuple ?submembers] - (mapcat locals-getter ?submembers) - - [::&parser/Form ([[::&parser/Tag ?subtag] & ?submembers] :seq)] - (mapcat locals-getter ?submembers) - - _ - (list) - ))] - (defn ^:private analyse-case [analyse-ast ?variant ?branches] - (exec [[=variant] (analyse-n (list ?variant)) - _ (assert! (and (> (count ?branches) 0) (even? (count ?branches))) - "Imbalanced branches in \"case'\" expression.") - $base next-local-idx - [registers mappings tree] (exec [=branches (analyse-case-branches (partition 2 ?branches))] - (return (->decision-tree $base =branches)))] - (return (list [::Expression [::case $base =variant registers mappings tree] +dont-care-type+]))))) +(defn ^:private analyse-case [analyse-ast ?variant ?branches] + (exec [=variant (analyse-1 ?variant) + _ (assert! (and (> (count ?branches) 0) (even? (count ?branches))) + "Imbalanced branches in \"case'\" expression.") + $base next-local-idx + [num-registers mappings tree] (exec [=branches (analyse-case-branches (partition 2 ?branches))] + (return (->decision-tree $base =branches)))] + (return (list [::Expression [::case $base =variant num-registers mappings tree] +dont-care-type+])))) (defn ^:private analyse-let [analyse-ast ?label ?value ?body] - (exec [[=value] (analyse-n (list ?value)) + (exec [=value (analyse-1 ?value) =value-type (expr-type =value) idx next-local-idx - [=body] (with-let ?label :local =value-type - (analyse-n (list ?body))) + =body (with-let ?label :local =value-type + (analyse-1 ?body)) =body-type (expr-type =body)] (return (list [::Expression [::let idx =value =body] =body-type])))) -(defn ^:private raise-tree-bindings [raise-expr ?tree] - (let [tree-partial-f (partial raise-tree-bindings raise-expr)] +(defn ^:private raise-tree-bindings [raise-expr arg ?tree] + (let [tree-partial-f (partial raise-tree-bindings raise-expr arg)] (case (:type ?tree) (::tuple ::variant) (-> ?tree @@ -525,23 +524,24 @@ (update-in [:default] (fn [[tag local $branch :as total]] (if total - (match (raise-expr [::Expression local [::&type/Nothing]]) + (match (raise-expr arg [::Expression local [::&type/Nothing]]) [::Expression local* [::&type/Nothing]] [tag local* $branch]))))) ::defaults (update-in ?tree [:stores] #(into {} (for [[?store ?branches] %] - (match (raise-expr [::Expression ?store [::&type/Nothing]]) + (match (raise-expr arg [::Expression ?store [::&type/Nothing]]) [::Expression =store [::&type/Nothing]] [=store ?branches])))) ;; else (assert false (pr-str ?tree)) ))) -(defn ^:private raise-expr [syntax] +(defn ^:private raise-expr [arg syntax] ;; (prn 'raise-bindings body) - (let [tree-partial-f (partial raise-tree-bindings raise-expr)] + (let [partial-f (partial raise-expr arg) + tree-partial-f (partial raise-tree-bindings raise-expr arg)] (match syntax [::Expression ?form ?type] (match ?form @@ -561,10 +561,10 @@ syntax [::tuple ?members] - [::Expression [::tuple (map raise-expr ?members)] ?type] + [::Expression [::tuple (map partial-f ?members)] ?type] [::variant ?tag ?members] - [::Expression [::variant ?tag (map raise-expr ?members)] ?type] + [::Expression [::variant ?tag (map partial-f ?members)] ?type] [::local ?idx] [::Expression [::local (inc ?idx)] ?type] @@ -573,76 +573,125 @@ ?source [::self ?curried] - [::Expression [::self (map raise-expr ?curried)] ?type] + [::Expression [::self (cons arg (map partial-f ?curried))] ?type] [::global _ _] syntax - [::jvm-iadd ?x ?y] - [::Expression [::jvm-iadd (raise-expr ?x) (raise-expr ?y)] ?type] - - [::jvm-isub ?x ?y] - [::Expression [::jvm-isub (raise-expr ?x) (raise-expr ?y)] ?type] - - [::jvm-imul ?x ?y] - [::Expression [::jvm-imul (raise-expr ?x) (raise-expr ?y)] ?type] - - [::jvm-idiv ?x ?y] - [::Expression [::jvm-idiv (raise-expr ?x) (raise-expr ?y)] ?type] - - [::jvm-irem ?x ?y] - [::Expression [::jvm-irem (raise-expr ?x) (raise-expr ?y)] ?type] - [::let ?idx ?value ?body] - [::Expression [::let (inc ?idx) (raise-expr ?value) - (raise-expr ?body)] + [::Expression [::let (inc ?idx) (partial-f ?value) + (partial-f ?body)] ?type] [::case ?base ?variant ?registers ?mappings ?tree] - (let [=variant (raise-expr ?variant) + (let [=variant (partial-f ?variant) =mappings (into {} (for [[idx syntax] ?mappings] - [idx (raise-expr syntax)])) + [idx (partial-f syntax)])) =tree (tree-partial-f ?tree)] [::Expression [::case (inc ?base) =variant ?registers =mappings =tree] ?type]) [::lambda ?scope ?captured ?args ?value] [::Expression [::lambda (pop ?scope) (into {} (for [[?name ?sub-syntax] ?captured] - [?name (raise-expr ?sub-syntax)])) + [?name (partial-f ?sub-syntax)])) ?args ?value] ?type] + [::call ?func ?args] + [::Expression [::call (partial-f ?func) (map partial-f ?args)] ?type] + + [::do ?asts] + [::Expression [::do (map partial-f ?asts)] ?type] + [::jvm-getstatic _ _] syntax [::jvm-invokevirtual ?class ?method ?arg-classes ?obj ?args] [::Expression [::jvm-invokevirtual ?class ?method ?arg-classes - (raise-expr ?obj) - (map raise-expr ?args)] + (partial-f ?obj) + (map partial-f ?args)] ?type] - [::do ?asts] - [::Expression [::do (map raise-expr ?asts)] ?type] + ;; Integer arithmetic + [::jvm-iadd ?x ?y] + [::Expression [::jvm-iadd (partial-f ?x) (partial-f ?y)] ?type] - [::call ?func ?args] - [::Expression [::call (raise-expr ?func) (map raise-expr ?args)] ?type] + [::jvm-isub ?x ?y] + [::Expression [::jvm-isub (partial-f ?x) (partial-f ?y)] ?type] + + [::jvm-imul ?x ?y] + [::Expression [::jvm-imul (partial-f ?x) (partial-f ?y)] ?type] + + [::jvm-idiv ?x ?y] + [::Expression [::jvm-idiv (partial-f ?x) (partial-f ?y)] ?type] + + [::jvm-irem ?x ?y] + [::Expression [::jvm-irem (partial-f ?x) (partial-f ?y)] ?type] + + ;; Long arithmetic + [::jvm-ladd ?x ?y] + [::Expression [::jvm-ladd (partial-f ?x) (partial-f ?y)] ?type] + + [::jvm-lsub ?x ?y] + [::Expression [::jvm-lsub (partial-f ?x) (partial-f ?y)] ?type] + + [::jvm-lmul ?x ?y] + [::Expression [::jvm-lmul (partial-f ?x) (partial-f ?y)] ?type] + + [::jvm-ldiv ?x ?y] + [::Expression [::jvm-ldiv (partial-f ?x) (partial-f ?y)] ?type] + + [::jvm-lrem ?x ?y] + [::Expression [::jvm-lrem (partial-f ?x) (partial-f ?y)] ?type] + + ;; Float arithmetic + [::jvm-fadd ?x ?y] + [::Expression [::jvm-fadd (partial-f ?x) (partial-f ?y)] ?type] + + [::jvm-fsub ?x ?y] + [::Expression [::jvm-fsub (partial-f ?x) (partial-f ?y)] ?type] + + [::jvm-fmul ?x ?y] + [::Expression [::jvm-fmul (partial-f ?x) (partial-f ?y)] ?type] + + [::jvm-fdiv ?x ?y] + [::Expression [::jvm-fdiv (partial-f ?x) (partial-f ?y)] ?type] + + [::jvm-frem ?x ?y] + [::Expression [::jvm-frem (partial-f ?x) (partial-f ?y)] ?type] + + ;; Double arithmetic + [::jvm-dadd ?x ?y] + [::Expression [::jvm-dadd (partial-f ?x) (partial-f ?y)] ?type] + + [::jvm-dsub ?x ?y] + [::Expression [::jvm-dsub (partial-f ?x) (partial-f ?y)] ?type] + + [::jvm-dmul ?x ?y] + [::Expression [::jvm-dmul (partial-f ?x) (partial-f ?y)] ?type] + + [::jvm-ddiv ?x ?y] + [::Expression [::jvm-ddiv (partial-f ?x) (partial-f ?y)] ?type] + + [::jvm-drem ?x ?y] + [::Expression [::jvm-drem (partial-f ?x) (partial-f ?y)] ?type] _ (assert false syntax) )))) (defn ^:private analyse-lambda [analyse-ast ?self ?arg ?body] - (exec [[_ =arg =return :as =function] (within ::&util/types &type/fresh-function) - [=scope =next-local =captured [=body]] (with-lambda ?self =function - ?arg =arg - (analyse-n (list ?body))) + (exec [[_ =arg =return :as =function] &type/fresh-function + [=scope =captured =body] (with-lambda ?self =function + ?arg =arg + (analyse-1 ?body)) =body-type (expr-type =body) - =function (within ::&util/types (exec [_ (&type/solve =return =body-type)] - (&type/clean =function))) + =function (exec [_ (&type/solve =return =body-type)] + (&type/clean =function)) :let [=lambda (match =body [::Expression [::lambda ?sub-scope ?sub-captured ?sub-args ?sub-body] =body-type] - [::Expression [::lambda =scope =captured (cons ?arg ?sub-args) (raise-expr ?sub-body)] =body-type] + [::Expression [::lambda =scope =captured (cons ?arg ?sub-args) (raise-expr ?arg ?sub-body)] =body-type] _ [::Expression [::lambda =scope =captured (list ?arg) =body] =body-type])]] @@ -652,10 +701,10 @@ ;; (prn 'analyse-def ?name ?value) (exec [def?? (defined? ?name)] (if def?? - (fail (str "Can't redefine function/constant: " ?name)) + (fail (str "Can't redefine " ?name)) (exec [ann?? (annotated? ?name) $module &util/get-module-name - [=value] (analyse-n (list ?value)) + =value (analyse-1 ?value) =value (match =value [::Expression =value-form =value-type] (return (match =value-form @@ -674,7 +723,7 @@ _ (define ?name)] (return (list [::Statement [::def ?name =value]])))))) -(defn ^:private analyse-annotate [?ident] +(defn ^:private analyse-declare-macro [?ident] (exec [_ (annotate ?ident ::public true [::&type/Any])] (return (list)))) @@ -682,16 +731,39 @@ (assert false) (return (list))) -(do-template [<name> <ident> <output-tag>] +(do-template [<name> <ident> <output-tag> <wrapper-class>] (defn <name> [analyse-ast ?x ?y] - (exec [[=x =y] (analyse-n (list ?x ?y))] - (return (list [::Expression [<output-tag> =x =y] [::&type/Data "java.lang.Integer"]])))) - - ^:private analyse-jvm-iadd "jvm;iadd" ::jvm-iadd - ^:private analyse-jvm-isub "jvm;isub" ::jvm-isub - ^:private analyse-jvm-imul "jvm;imul" ::jvm-imul - ^:private analyse-jvm-idiv "jvm;idiv" ::jvm-idiv - ^:private analyse-jvm-irem "jvm;irem" ::jvm-irem + (exec [:let [=type [::&type/Data <wrapper-class>]] + [=x =y] (analyse-2 ?x ?y) + =x-type (expr-type =x) + =y-type (expr-type =y) + _ (&type/solve =type =x-type) + _ (&type/solve =type =y-type)] + (return (list [::Expression [<output-tag> =x =y] =type])))) + + ^:private analyse-jvm-iadd "jvm;iadd" ::jvm-iadd "java.lang.Integer" + ^:private analyse-jvm-isub "jvm;isub" ::jvm-isub "java.lang.Integer" + ^:private analyse-jvm-imul "jvm;imul" ::jvm-imul "java.lang.Integer" + ^:private analyse-jvm-idiv "jvm;idiv" ::jvm-idiv "java.lang.Integer" + ^:private analyse-jvm-irem "jvm;irem" ::jvm-irem "java.lang.Integer" + + ^:private analyse-jvm-ladd "jvm;ladd" ::jvm-ladd "java.lang.Long" + ^:private analyse-jvm-lsub "jvm;lsub" ::jvm-lsub "java.lang.Long" + ^:private analyse-jvm-lmul "jvm;lmul" ::jvm-lmul "java.lang.Long" + ^:private analyse-jvm-ldiv "jvm;ldiv" ::jvm-ldiv "java.lang.Long" + ^:private analyse-jvm-lrem "jvm;lrem" ::jvm-lrem "java.lang.Long" + + ^:private analyse-jvm-iadd "jvm;fadd" ::jvm-fadd "java.lang.Float" + ^:private analyse-jvm-isub "jvm;fsub" ::jvm-fsub "java.lang.Float" + ^:private analyse-jvm-imul "jvm;fmul" ::jvm-fmul "java.lang.Float" + ^:private analyse-jvm-idiv "jvm;fdiv" ::jvm-fdiv "java.lang.Float" + ^:private analyse-jvm-irem "jvm;frem" ::jvm-frem "java.lang.Float" + + ^:private analyse-jvm-iadd "jvm;dadd" ::jvm-dadd "java.lang.Double" + ^:private analyse-jvm-isub "jvm;dsub" ::jvm-dsub "java.lang.Double" + ^:private analyse-jvm-imul "jvm;dmul" ::jvm-dmul "java.lang.Double" + ^:private analyse-jvm-idiv "jvm;ddiv" ::jvm-ddiv "java.lang.Double" + ^:private analyse-jvm-irem "jvm;drem" ::jvm-drem "java.lang.Double" ) (defn ^:private analyse-jvm-getstatic [analyse-ast ?class ?field] @@ -702,7 +774,7 @@ (defn ^:private analyse-jvm-getfield [analyse-ast ?class ?field ?object] (exec [=class (full-class-name ?class) =type (lookup-static-field =class ?field) - [=object] (analyse-n (list ?object))] + =object (analyse-1 ?object)] (return (list [::Expression [::jvm-getfield =class ?field =object] =type])))) (defn ^:private analyse-jvm-invokestatic [analyse-ast ?class ?method ?classes ?args] @@ -716,7 +788,7 @@ (exec [=class (full-class-name ?class) =classes (map-m extract-jvm-param ?classes) =return (lookup-virtual-method =class ?method =classes) - [=object] (analyse-n (list ?object)) + =object (analyse-1 ?object) =args (mapcat-m analyse-ast ?args)] (return (list [::Expression [::jvm-invokevirtual =class ?method =classes =object =args] =return])))) @@ -731,12 +803,12 @@ (return (list [::Expression [::jvm-new-array =class ?length] [::&type/Array [::&type/Data =class]]])))) (defn ^:private analyse-jvm-aastore [analyse-ast ?array ?idx ?elem] - (exec [[=array =elem] (analyse-n (list ?array ?elem)) + (exec [[=array =elem] (analyse-2 ?array ?elem) =array-type (expr-type =array)] (return (list [::Expression [::jvm-aastore =array ?idx =elem] =array-type])))) (defn ^:private analyse-jvm-aaload [analyse-ast ?array ?idx] - (exec [[=array] (analyse-n (list ?array)) + (exec [=array (analyse-1 ?array) =array-type (expr-type =array)] (return (list [::Expression [::jvm-aaload =array ?idx] =array-type])))) @@ -779,10 +851,10 @@ (return (list [::Expression [::bool ?value] [::&type/Data "java.lang.Boolean"]])) [::&parser/int ?value] - (return (list [::Expression [::int ?value] [::&type/Data "java.lang.Integer"]])) + (return (list [::Expression [::int ?value] [::&type/Data "java.lang.Long"]])) [::&parser/real ?value] - (return (list [::Expression [::real ?value] [::&type/Data "java.lang.Float"]])) + (return (list [::Expression [::real ?value] [::&type/Data "java.lang.Double"]])) [::&parser/char ?value] (return (list [::Expression [::char ?value] [::&type/Data "java.lang.Character"]])) @@ -811,8 +883,8 @@ [::&parser/form ([[::&parser/ident "def'"] [::&parser/ident ?name] ?value] :seq)] (analyse-def analyse-ast ?name ?value) - [::&parser/form ([[::&parser/ident "annotate"] [::&parser/ident ?ident] [::&parser/ident "Macro"]] :seq)] - (analyse-annotate ?ident) + [::&parser/form ([[::&parser/ident "declare-macro"] [::&parser/ident ?ident]] :seq)] + (analyse-declare-macro ?ident) [::&parser/form ([[::&parser/ident "require"] [::&parser/text ?path]] :seq)] (analyse-require analyse-ast ?path) @@ -820,7 +892,8 @@ ;; Host special forms [::&parser/form ([[::&parser/ident "do"] & ?exprs] :seq)] (analyse-do analyse-ast ?exprs) - + + ;; Integer arithmetic [::&parser/form ([[::&parser/ident "jvm;iadd"] ?x ?y] :seq)] (analyse-jvm-iadd analyse-ast ?x ?y) @@ -836,6 +909,54 @@ [::&parser/form ([[::&parser/ident "jvm;irem"] ?x ?y] :seq)] (analyse-jvm-irem analyse-ast ?x ?y) + ;; Long arithmetic + [::&parser/form ([[::&parser/ident "jvm;ladd"] ?x ?y] :seq)] + (analyse-jvm-ladd analyse-ast ?x ?y) + + [::&parser/form ([[::&parser/ident "jvm;lsub"] ?x ?y] :seq)] + (analyse-jvm-lsub analyse-ast ?x ?y) + + [::&parser/form ([[::&parser/ident "jvm;lmul"] ?x ?y] :seq)] + (analyse-jvm-lmul analyse-ast ?x ?y) + + [::&parser/form ([[::&parser/ident "jvm;ldiv"] ?x ?y] :seq)] + (analyse-jvm-ldiv analyse-ast ?x ?y) + + [::&parser/form ([[::&parser/ident "jvm;lrem"] ?x ?y] :seq)] + (analyse-jvm-lrem analyse-ast ?x ?y) + + ;; Float arithmetic + [::&parser/form ([[::&parser/ident "jvm;fadd"] ?x ?y] :seq)] + (analyse-jvm-fadd analyse-ast ?x ?y) + + [::&parser/form ([[::&parser/ident "jvm;fsub"] ?x ?y] :seq)] + (analyse-jvm-fsub analyse-ast ?x ?y) + + [::&parser/form ([[::&parser/ident "jvm;fmul"] ?x ?y] :seq)] + (analyse-jvm-fmul analyse-ast ?x ?y) + + [::&parser/form ([[::&parser/ident "jvm;fdiv"] ?x ?y] :seq)] + (analyse-jvm-fdiv analyse-ast ?x ?y) + + [::&parser/form ([[::&parser/ident "jvm;frem"] ?x ?y] :seq)] + (analyse-jvm-frem analyse-ast ?x ?y) + + ;; Double arithmetic + [::&parser/form ([[::&parser/ident "jvm;dadd"] ?x ?y] :seq)] + (analyse-jvm-dadd analyse-ast ?x ?y) + + [::&parser/form ([[::&parser/ident "jvm;dsub"] ?x ?y] :seq)] + (analyse-jvm-dsub analyse-ast ?x ?y) + + [::&parser/form ([[::&parser/ident "jvm;dmul"] ?x ?y] :seq)] + (analyse-jvm-dmul analyse-ast ?x ?y) + + [::&parser/form ([[::&parser/ident "jvm;ddiv"] ?x ?y] :seq)] + (analyse-jvm-ddiv analyse-ast ?x ?y) + + [::&parser/form ([[::&parser/ident "jvm;drem"] ?x ?y] :seq)] + (analyse-jvm-drem analyse-ast ?x ?y) + [::&parser/form ([[::&parser/ident "jvm;getstatic"] [::&parser/ident ?class] [::&parser/ident ?field]] :seq)] (analyse-jvm-getstatic analyse-ast ?class ?field) diff --git a/src/lux/compiler.clj b/src/lux/compiler.clj index a62f66c35..daf2f1e09 100644 --- a/src/lux/compiler.clj +++ b/src/lux/compiler.clj @@ -18,9 +18,10 @@ ClassWriter MethodVisitor))) -(def +prefix+ "lux") - ;; [Utils/General] +(defn ^:private storage-id [scope] + (->> scope reverse (map normalize-ident) (interpose "$") (reduce str ""))) + (defn ^:private write-file [file data] (with-open [stream (java.io.BufferedOutputStream. (java.io.FileOutputStream. file))] (.write stream data))) @@ -31,31 +32,19 @@ (defn ^:private load-class! [loader name] (.loadClass loader name)) -(defn save-class! [name bytecode] +(defn ^:private save-class! [name bytecode] (exec [loader &util/loader :let [_ (write-class name bytecode) _ (load-class! loader (string/replace name #"/" "."))]] (return nil))) -(def ^:private +variant-class+ (str +prefix+ ".Variant")) -(def ^:private +tuple-class+ (str +prefix+ ".Tuple")) - -(defn ^:private unwrap-ident [ident] - (match ident - [::&parser/ident ?label] - ?label)) - -(defn ^:private with-writer [writer body] - (fn [state] - ;; (prn 'with-writer/_0 body) - (let [result (body (assoc state ::&util/writer writer))] - ;; (prn 'with-writer/_1 result) - (match result - [::&util/ok [?state ?value]] - [::&util/ok [(assoc ?state ::&util/writer (::&util/writer state)) ?value]] - - _ - result)))) +(def ^:private +prefix+ "lux.") +(def ^:private +variant-class+ (str +prefix+ "Variant")) +(def ^:private +tuple-class+ (str +prefix+ "Tuple")) +(def ^:private +function-class+ (str +prefix+ "Function")) +(def ^:private +local-prefix+ "l") +(def ^:private +partial-prefix+ "p") +(def ^:private +closure-prefix+ "c") (def ^:private ->package ->class) @@ -92,7 +81,7 @@ (->type-signature +variant-class+) [::&type/Lambda _ _] - (->type-signature (str +prefix+ "/Function")))) + (->type-signature +function-class+))) ;; [Utils/Compilers] (let [+class+ (->class "java.lang.Boolean") @@ -113,8 +102,8 @@ (.visitMethodInsn Opcodes/INVOKESPECIAL <class> "<init>" <sig>))]] (return nil)))) - ^:private compile-int "java.lang.Integer" "(I)V" - ^:private compile-real "java.lang.Float" "(F)V" + ^:private compile-int "java.lang.Long" "(J)V" + ^:private compile-real "java.lang.Double" "(D)V" ^:private compile-char "java.lang.Character" "(C)V" ) @@ -126,7 +115,7 @@ (defn ^:private compile-tuple [compile *type* ?elems] (exec [*writer* &util/get-writer :let [num-elems (count ?elems) - tuple-class (str +prefix+ "/Tuple" num-elems) + tuple-class (->class (str +tuple-class+ num-elems)) _ (doto *writer* (.visitTypeInsn Opcodes/NEW tuple-class) (.visitInsn Opcodes/DUP) @@ -134,14 +123,32 @@ _ (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;")]] + :let [_ (.visitFieldInsn *writer* Opcodes/PUTFIELD tuple-class (str +partial-prefix+ idx) "Ljava/lang/Object;")]] (return ret))) (range num-elems))] (return nil))) +(defn ^:private compile-variant [compile *type* ?tag ?members] + (exec [*writer* &util/get-writer + :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 +partial-prefix+ ?tfield) "Ljava/lang/Object;")]] + (return ret))) + (map vector (range (count ?members)) ?members))] + (return nil))) + (defn ^:private compile-local [compile *type* ?idx] (exec [*writer* &util/get-writer - :let [_ (.visitVarInsn *writer* Opcodes/ALOAD (int (inc ?idx)))]] + :let [_ (.visitVarInsn *writer* Opcodes/ALOAD (int ?idx))]] (return nil))) (defn ^:private compile-captured [compile *type* ?scope ?captured-id ?source] @@ -150,13 +157,13 @@ (.visitVarInsn Opcodes/ALOAD 0) (.visitFieldInsn Opcodes/GETFIELD (normalize-ident ?scope) - (str "__" ?captured-id) + (str +closure-prefix+ ?captured-id) "Ljava/lang/Object;"))]] (return nil))) (defn ^:private compile-global [compile *type* ?owner-class ?name] (exec [*writer* &util/get-writer - :let [_ (.visitFieldInsn *writer* Opcodes/GETSTATIC (->class (str ?owner-class "$" (normalize-ident ?name))) "_datum" "Ljava/lang/Object;")]] + :let [_ (.visitFieldInsn *writer* Opcodes/GETSTATIC (->class (storage-id (list ?name ?owner-class))) "_datum" "Ljava/lang/Object;")]] (return nil))) (def +apply-signature+ "(Ljava/lang/Object;)Ljava/lang/Object;") @@ -166,7 +173,7 @@ _ (compile ?fn) _ (map-m (fn [arg] (exec [ret (compile arg) - :let [_ (.visitMethodInsn *writer* Opcodes/INVOKEINTERFACE (str +prefix+ "/Function") "apply" +apply-signature+)]] + :let [_ (.visitMethodInsn *writer* Opcodes/INVOKEINTERFACE (->class +function-class+) "apply" +apply-signature+)]] (return ret))) ?args)] (return nil))) @@ -177,7 +184,7 @@ :let [_ (match (:form ?fn) [::&analyser/global ?owner-class ?fn-name] (let [arg-sig (->type-signature "java.lang.Object") - call-class (str (->class ?owner-class) "$" (normalize-ident ?fn-name)) + call-class (storage-id (list ?fn-name ?owner-class)) provides-num (count ?args)] (if (>= provides-num ?needs-num) (let [impl-sig (str "(" (reduce str "" (repeat ?needs-num arg-sig)) ")" arg-sig)] @@ -186,7 +193,7 @@ (->> (doseq [arg (take ?needs-num ?args)]))) (.visitMethodInsn Opcodes/INVOKESTATIC call-class "impl" impl-sig) (-> (doto (do (compile arg)) - (.visitMethodInsn Opcodes/INVOKEINTERFACE (str +prefix+ "/Function") "apply" +apply-signature+)) + (.visitMethodInsn Opcodes/INVOKEINTERFACE (->class +function-class+) "apply" +apply-signature+)) (->> (doseq [arg (drop ?needs-num ?args)]))))) (let [counter-sig "I" init-signature (str "(" (apply str counter-sig (repeat (dec ?needs-num) arg-sig)) ")" "V")] @@ -196,8 +203,7 @@ (.visitLdcInsn (int provides-num)) (-> (do (compile arg)) (->> (doseq [arg ?args]))) - (-> (.visitInsn Opcodes/ACONST_NULL) - (->> (dotimes [_ (dec (- ?needs-num provides-num))]))) + (add-nulls (dec (- ?needs-num provides-num))) (.visitMethodInsn Opcodes/INVOKESPECIAL call-class "<init>" init-signature))) )) )]] @@ -329,29 +335,59 @@ _ (compile (last ?exprs))] (return nil))) -(let [oclass (->class "java.lang.Object") - equals-sig (str "(" (->type-signature "java.lang.Object") ")Z")] - (defn ^:private compile-compare-primitive [writer mappings default-label ?pairs wrapper-class signature] - (let [wrapper-class (->class wrapper-class)] - (doseq [[?token $body] ?pairs - :let [$else (new Label)]] - (doto writer - ;; object - (.visitInsn Opcodes/DUP) ;; object, object - (-> (doto (.visitTypeInsn Opcodes/NEW wrapper-class) - (.visitInsn Opcodes/DUP) - (.visitLdcInsn ?token) - (.visitMethodInsn Opcodes/INVOKESPECIAL wrapper-class "<init>" signature)) - (->> (if (nil? wrapper-class) - (.visitLdcInsn writer ?token)))) - (.visitMethodInsn Opcodes/INVOKEVIRTUAL oclass "equals" equals-sig) ;; object, B - (.visitJumpInsn Opcodes/IFEQ $else) ;; object - (.visitInsn Opcodes/POP) - (.visitJumpInsn Opcodes/GOTO (get mappings $body)) - (.visitLabel $else))) +(do-template [<name> <wrapper-class> <value-method> <method-sig>] + (defn <name> [writer mappings default-label ?pairs] + (doseq [[?token $body] ?pairs + :let [$else (new Label)]] + (doto writer + (.visitMethodInsn Opcodes/INVOKEVIRTUAL (->class <wrapper-class>) <value-method> <method-sig>) + (.visitLdcInsn ?token) + (.visitJumpInsn Opcodes/IF_ICMPNE $else) + (.visitInsn Opcodes/POP) + (.visitJumpInsn Opcodes/GOTO (get mappings $body)) + (.visitLabel $else))) + (doto writer + (.visitInsn Opcodes/POP) + (.visitJumpInsn Opcodes/GOTO default-label))) + + ^:private compile-compare-bools "java.lang.Boolean" "booleanValue" "()Z" + ^:private compile-compare-chars "java.lang.Character" "charValue" "()C" + ) + +(do-template [<name> <wrapper-class> <value-method> <method-sig> <cmp-op>] + (defn <name> [writer mappings default-label ?pairs] + (doseq [[?token $body] ?pairs + :let [$else (new Label)]] (doto writer + (.visitMethodInsn Opcodes/INVOKEVIRTUAL (->class <wrapper-class>) <value-method> <method-sig>) + (.visitLdcInsn ?token) + (.visitInsn <cmp-op>) + (.visitJumpInsn Opcodes/IFNE $else) (.visitInsn Opcodes/POP) - (.visitJumpInsn Opcodes/GOTO default-label))))) + (.visitJumpInsn Opcodes/GOTO (get mappings $body)) + (.visitLabel $else))) + (doto writer + (.visitInsn Opcodes/POP) + (.visitJumpInsn Opcodes/GOTO default-label))) + + ^:private compile-compare-ints "java.lang.Long" "longValue" "()J" Opcodes/LCMP + ^:private compile-compare-reals "java.lang.Double" "doubleValue" "()D" Opcodes/DCMPL + ) + +(defn ^:private compile-compare-texts [writer mappings default-label ?pairs] + (doseq [[?token $body] ?pairs + :let [$else (new Label)]] + (doto writer + (.visitInsn Opcodes/DUP) + (.visitLdcInsn ?token) + (.visitMethodInsn Opcodes/INVOKEVIRTUAL (->class "java.lang.Object") "equals" (str "(" (->type-signature "java.lang.Object") ")Z")) + (.visitJumpInsn Opcodes/IFEQ $else) + (.visitInsn Opcodes/POP) + (.visitJumpInsn Opcodes/GOTO (get mappings $body)) + (.visitLabel $else))) + (doto writer + (.visitInsn Opcodes/POP) + (.visitJumpInsn Opcodes/GOTO default-label))) (let [+tag-sig+ (->type-signature "java.lang.String") variant-class* (->class +variant-class+) @@ -362,23 +398,23 @@ (defn ^:private compile-decision-tree [writer mappings default-label decision-tree] (match decision-tree [::test-bool ?pairs] - (compile-compare-primitive writer mappings default-label ?pairs "java.lang.Boolean" "(Z)V") + (compile-compare-bools writer mappings default-label ?pairs) - [::test-int ?pairs] - (compile-compare-primitive writer mappings default-label ?pairs "java.lang.Integer" "(I)V") + [::test-int ?pairs] + (compile-compare-ints writer mappings default-label ?pairs) [::test-real ?pairs] - (compile-compare-primitive writer mappings default-label ?pairs "java.lang.Float" "(F)V") + (compile-compare-reals writer mappings default-label ?pairs) [::test-char ?pairs] - (compile-compare-primitive writer mappings default-label ?pairs "java.lang.Character" "(C)V") + (compile-compare-chars writer mappings default-label ?pairs) [::test-text ?pairs] - (compile-compare-primitive writer mappings default-label ?pairs nil nil) + (compile-compare-texts writer mappings default-label ?pairs) [::store [::&analyser/local ?idx] $body] (doto writer - (.visitVarInsn Opcodes/ASTORE (inc ?idx)) + (.visitVarInsn Opcodes/ASTORE ?idx) (.visitJumpInsn Opcodes/GOTO (get mappings $body))) [::test-tuple ?branches ?cases] @@ -396,7 +432,7 @@ :let [sub-next-elem (new Label)]] (doto writer (.visitInsn Opcodes/DUP) ;; tuple, tuple - (.visitFieldInsn Opcodes/GETFIELD tuple-class** (str "_" (inc ?subidx)) +variant-field-sig+) ;; tuple, object + (.visitFieldInsn Opcodes/GETFIELD tuple-class** (str +partial-prefix+ ?subidx) +variant-field-sig+) ;; tuple, object (compile-decision-tree (assoc mappings $body sub-next-elem) next-subcase ?subpart) ;; tuple (.visitLabel sub-next-elem))) (doto writer @@ -429,7 +465,7 @@ :let [sub-next-elem (new Label)]] (doto writer (.visitInsn Opcodes/DUP) ;; variant, variant - (.visitFieldInsn Opcodes/GETFIELD variant-class** (str "_" (inc ?subidx)) +variant-field-sig+) ;; variant, object + (.visitFieldInsn Opcodes/GETFIELD variant-class** (str +partial-prefix+ ?subidx) +variant-field-sig+) ;; variant, object (compile-decision-tree (assoc mappings $body sub-next-elem) next-subcase ?subpart) ;; variant (.visitLabel sub-next-elem))) (doto writer @@ -528,7 +564,7 @@ mappings* (into {} (map first entries)) _ (dotimes [offset ?max-registers] (let [idx (+ ?base-idx offset)] - (.visitLocalVariable *writer* (str "v" idx) (->java-sig [::&type/Any]) nil start-label end-label idx)))] + (.visitLocalVariable *writer* (str +local-prefix+ idx) (->java-sig [::&type/Any]) nil start-label end-label idx)))] _ (compile ?variant) :let [_ (doto *writer* (.visitInsn Opcodes/DUP) @@ -546,7 +582,7 @@ (first (:defaults ?decision-tree)))] (doto *writer* (.visitInsn Opcodes/DUP) - (.visitVarInsn Opcodes/ASTORE (inc ?idx)) + (.visitVarInsn Opcodes/ASTORE ?idx) (.visitJumpInsn Opcodes/GOTO (get mappings* ?body))) (doto *writer* (.visitInsn Opcodes/POP) @@ -570,39 +606,13 @@ :let [start-label (new Label) end-label (new Label) _ (doto *writer* - (.visitLocalVariable (str "v" ?idx) (->java-sig (:type ?value)) nil start-label end-label ?idx) + (.visitLocalVariable (str +local-prefix+ ?idx) (->java-sig (:type ?value)) nil start-label end-label ?idx) (.visitLabel start-label) - (.visitVarInsn Opcodes/ASTORE (inc ?idx)))] + (.visitVarInsn Opcodes/ASTORE ?idx))] _ (compile ?body) :let [_ (.visitLabel *writer* end-label)]] (return nil))) -(defn ^:private compile-field [compile ?name body] - (exec [*writer* &util/get-writer - class-name &analyser/module-name - :let [outer-class (->class class-name) - datum-sig (->type-signature "java.lang.Object") - current-class (str outer-class "$" (normalize-ident ?name)) - _ (.visitInnerClass *writer* current-class outer-class nil (+ Opcodes/ACC_STATIC Opcodes/ACC_SYNTHETIC)) - =class (doto (new ClassWriter ClassWriter/COMPUTE_MAXS) - (.visit Opcodes/V1_5 (+ Opcodes/ACC_PUBLIC Opcodes/ACC_FINAL Opcodes/ACC_SUPER) - current-class nil "java/lang/Object" (into-array [(str +prefix+ "/Function")])) - (-> (.visitField (+ Opcodes/ACC_PUBLIC Opcodes/ACC_FINAL Opcodes/ACC_STATIC) "_datum" datum-sig nil nil) - (doto (.visitEnd))))] - _ (with-writer (.visitMethod =class Opcodes/ACC_PUBLIC "<clinit>" "()V" nil nil) - (exec [*writer* &util/get-writer - :let [_ (.visitCode *writer*)] - _ (compile body) - :let [_ (doto *writer* - (.visitFieldInsn Opcodes/PUTSTATIC current-class "_datum" datum-sig) - (.visitInsn Opcodes/RETURN) - (.visitMaxs 0 0) - (.visitEnd))]] - (return nil))) - :let [_ (.visitEnd *writer*)] - _ (save-class! current-class (.toByteArray =class))] - (return nil))) - (let [clo-field-sig (->type-signature "java.lang.Object") lambda-return-sig (->type-signature "java.lang.Object") <init>-return "V" @@ -627,19 +637,19 @@ (.visitVarInsn Opcodes/ALOAD 0) (.visitMethodInsn Opcodes/INVOKESPECIAL "java/lang/Object" "<init>" "()V") (-> (doto (.visitVarInsn Opcodes/ALOAD 0) - (.visitVarInsn Opcodes/ALOAD (inc ?captured-id)) + (.visitVarInsn Opcodes/ALOAD ?captured-id) (.visitFieldInsn Opcodes/PUTFIELD class-name captured-name clo-field-sig)) - (->> (let [captured-name (str "__" ?captured-id)]) + (->> (let [captured-name (str +closure-prefix+ ?captured-id)]) (match (:form ?captured) [::&analyser/captured ?closure-id ?captured-id ?source]) (doseq [[?name ?captured] closed-over]))) (-> (doto (.visitVarInsn Opcodes/ALOAD 0) - (.visitVarInsn Opcodes/ILOAD (inc num-mappings)) + (.visitInsn Opcodes/ICONST_0) (.visitFieldInsn Opcodes/PUTFIELD class-name "_counter" counter-sig) (-> (doto (.visitVarInsn Opcodes/ALOAD 0) (.visitVarInsn Opcodes/ALOAD (+ clo_idx offset)) (.visitFieldInsn Opcodes/PUTFIELD class-name field-name clo-field-sig)) - (->> (let [field-name (str "_" clo_idx)] + (->> (let [field-name (str +partial-prefix+ clo_idx)] (doto (.visitField class (+ Opcodes/ACC_PRIVATE Opcodes/ACC_FINAL) field-name clo-field-sig nil nil) (.visitEnd))) (dotimes [clo_idx (dec num-args)]) @@ -656,8 +666,8 @@ (.visitVarInsn Opcodes/ALOAD 0) (.visitFieldInsn Opcodes/GETFIELD class-name (str <prefix> idx) clo-field-sig)))) - ^:private add-closed-over-vars "__" - ^:private add-partial-vars "_" + ^:private add-closure-vars +closure-prefix+ + ^:private add-partial-vars +partial-prefix+ ) (defn ^:private add-nulls [writer amount] @@ -678,8 +688,8 @@ (-> (doto (.visitLabel branch-label) (.visitTypeInsn Opcodes/NEW class-name) (.visitInsn Opcodes/DUP) - (add-closed-over-vars class-name closed-over) - (.visitLdcInsn (-> current-captured inc int)) + (add-closure-vars class-name closed-over) + (.visitLdcInsn (int current-captured)) (add-partial-vars class-name (take current-captured args)) (.visitVarInsn Opcodes/ALOAD 1) (add-nulls (- (dec num-captured) current-captured)) @@ -697,8 +707,8 @@ (.visitEnd)))) (defn ^:private add-lambda-impl [class compile impl-signature impl-body] - (with-writer (doto (.visitMethod class Opcodes/ACC_PUBLIC "impl" impl-signature nil nil) - (.visitCode)) + (&util/with-writer (doto (.visitMethod class Opcodes/ACC_PUBLIC "impl" impl-signature nil nil) + (.visitCode)) (exec [;; :let [_ (prn 'add-lambda-impl/_0)] *writer* &util/get-writer ;; :let [_ (prn 'add-lambda-impl/_1 *writer*)] @@ -714,9 +724,7 @@ (defn ^:private instance-closure [compile lambda-class closed-over args init-signature] (exec [*writer* &util/get-writer - :let [;; _ (prn 'instance-closure/*writer* *writer*) - num-args (count args) - _ (doto *writer* + :let [_ (doto *writer* (.visitTypeInsn Opcodes/NEW lambda-class) (.visitInsn Opcodes/DUP))] _ (->> closed-over @@ -726,10 +734,10 @@ (match (:form ?captured) [::&analyser/captured ?closure-id ?captured-id ?source] (compile ?source))))) - :let [_ (do (when (> num-args 1) + :let [num-args (count args) + _ (do (when (> num-args 1) (.visitInsn *writer* Opcodes/ICONST_0) - (dotimes [_ (dec num-args)] - (.visitInsn *writer* Opcodes/ACONST_NULL))) + (add-nulls *writer* (dec num-args))) (.visitMethodInsn *writer* Opcodes/INVOKESPECIAL lambda-class "<init>" init-signature))]] (return nil))) @@ -739,26 +747,25 @@ (.visitCode) (.visitTypeInsn Opcodes/NEW class-name) (.visitInsn Opcodes/DUP) - (-> (doto (.visitLdcInsn (int 0)) - (-> (.visitInsn Opcodes/ACONST_NULL) - (->> (dotimes [_ (dec num-args)])))) + (-> (doto (.visitInsn *writer* Opcodes/ICONST_0) + (add-nulls (dec num-args))) (->> (when (> num-args 1)))) (.visitMethodInsn Opcodes/INVOKESPECIAL class-name "<init>" <init>-sig) (.visitFieldInsn Opcodes/PUTSTATIC class-name "_datum" +datum-sig+) (.visitInsn Opcodes/RETURN) (.visitMaxs 0 0) (.visitEnd)))) - + (defn ^:private compile-lambda [compile *type* ?scope ?closure ?args ?body with-datum? instance?] - (exec [:let [lambda-class (reduce str "" (interpose "$" (map normalize-ident ?scope))) + (exec [:let [lambda-class (storage-id ?scope) impl-signature (lambda-impl-signature ?args) <init>-sig (lambda-<init>-signature ?closure ?args) =class (doto (new ClassWriter ClassWriter/COMPUTE_MAXS) (.visit Opcodes/V1_5 (+ Opcodes/ACC_PUBLIC Opcodes/ACC_FINAL Opcodes/ACC_SUPER) - lambda-class nil "java/lang/Object" (into-array [(str +prefix+ "/Function")])) + lambda-class nil "java/lang/Object" (into-array [(->class +function-class+)])) (-> (doto (.visitField (+ Opcodes/ACC_PRIVATE Opcodes/ACC_FINAL) captured-name clo-field-sig nil nil) (.visitEnd)) - (->> (let [captured-name (str "__" ?captured-id)]) + (->> (let [captured-name (str +closure-prefix+ ?captured-id)]) (match (:form ?captured) [::&analyser/captured ?closure-id ?captured-id ?source]) (doseq [[?name ?captured] ?closure]))) @@ -779,13 +786,44 @@ (return nil)))) ) -(defn ^:private compile-def [compile *type* ?name ?value] - (exec [_ (match (:form ?value) - [::&analyser/lambda ?scope ?captured ?args ?body] - (compile-lambda compile *type* ?scope ?closure ?args ?body true false) +(defn ^:private compile-field [compile *type* ?name body] + (exec [*writer* &util/get-writer + class-name &analyser/module-name + :let [outer-class (->class class-name) + datum-sig (->type-signature "java.lang.Object") + current-class (storage-id (list ?name outer-class)) + _ (.visitInnerClass *writer* current-class outer-class nil (+ Opcodes/ACC_STATIC Opcodes/ACC_SYNTHETIC)) + =class (doto (new ClassWriter ClassWriter/COMPUTE_MAXS) + (.visit Opcodes/V1_5 (+ Opcodes/ACC_PUBLIC Opcodes/ACC_FINAL Opcodes/ACC_SUPER) + current-class nil "java/lang/Object" (into-array [(->class +function-class+)])) + (-> (.visitField (+ Opcodes/ACC_PUBLIC Opcodes/ACC_FINAL Opcodes/ACC_STATIC) "_datum" datum-sig nil nil) + (doto (.visitEnd))))] + _ (&util/with-writer (.visitMethod =class Opcodes/ACC_PUBLIC "<clinit>" "()V" nil nil) + (exec [*writer* &util/get-writer + :let [_ (.visitCode *writer*)] + _ (compile body) + :let [_ (doto *writer* + (.visitFieldInsn Opcodes/PUTSTATIC current-class "_datum" datum-sig) + (.visitInsn Opcodes/RETURN) + (.visitMaxs 0 0) + (.visitEnd))]] + (return nil))) + :let [_ (.visitEnd *writer*)] + _ (save-class! current-class (.toByteArray =class))] + (return nil))) + +(defn ^:private compile-def [compile *type* name value] + (exec [_ (match value + [::&analyser/Expression ?form _] + (match ?form + [::&analyser/lambda ?scope ?captured ?args ?body] + (compile-lambda compile *type* ?scope ?closure ?args ?body true false) + _ + (compile-field compile *type* name value)) + _ - (compile-field compile ?name ?value))] + (fail "Can only define expressions."))] (return nil))) (defn ^:private compile-jvm-class [compile *type* ?package ?name ?super-class ?fields ?methods] @@ -829,163 +867,221 @@ _ (save-class! full-name (.toByteArray =interface))] (return nil))) -(defn ^:private compile-variant [compile *type* ?tag ?members] - (exec [*writer* &util/get-writer - :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))) +(do-template [<name> <opcode> <wrapper-class> <value-method> <value-method-sig> <wrapper-method> <wrapper-method-sig>] + (defn <name> [compile *type* ?x ?y] + (exec [:let [+wrapper-class+ (->class <wrapper-class>)] + *writer* &util/get-writer + _ (compile ?x) + :let [_ (doto *writer* + (.visitTypeInsn Opcodes/CHECKCAST +wrapper-class+) + (.visitMethodInsn Opcodes/INVOKEVIRTUAL +wrapper-class+ <value-method> <value-method-sig>))] + _ (compile ?y) + :let [_ (doto *writer* + (.visitTypeInsn Opcodes/CHECKCAST +wrapper-class+) + (.visitMethodInsn Opcodes/INVOKEVIRTUAL +wrapper-class+ <value-method> <value-method-sig>)) + _ (doto *writer* + (.visitInsn <opcode>) + (.visitMethodInsn Opcodes/INVOKESTATIC +wrapper-class+ <wrapper-method> (str <wrapper-method-sig> (->type-signature <wrapper-class>))))]] + (return nil))) -(let [+int-class+ (->class "java.lang.Integer")] - (do-template [<name> <opcode>] - (defn <name> [compile *type* ?x ?y] - (exec [*writer* &util/get-writer - _ (compile ?x) - :let [_ (doto *writer* - (.visitTypeInsn Opcodes/CHECKCAST +int-class+) - (.visitMethodInsn Opcodes/INVOKEVIRTUAL +int-class+ "intValue" "()I"))] - _ (compile ?y) - :let [_ (doto *writer* - (.visitTypeInsn Opcodes/CHECKCAST +int-class+) - (.visitMethodInsn Opcodes/INVOKEVIRTUAL +int-class+ "intValue" "()I")) - _ (doto *writer* - (.visitInsn <opcode>) - (.visitMethodInsn Opcodes/INVOKESTATIC +int-class+ "valueOf" (str "(I)" (->type-signature "java.lang.Integer"))))]] - (return nil))) - - ^:private compile-jvm-iadd Opcodes/IADD - ^:private compile-jvm-isub Opcodes/ISUB - ^:private compile-jvm-imul Opcodes/IMUL - ^:private compile-jvm-idiv Opcodes/IDIV - ^:private compile-jvm-irem Opcodes/IREM - )) + ^:private compile-jvm-iadd Opcodes/IADD "java.lang.Integer" "intValue" "()I" "valueOf" "(I)" + ^:private compile-jvm-isub Opcodes/ISUB "java.lang.Integer" "intValue" "()I" "valueOf" "(I)" + ^:private compile-jvm-imul Opcodes/IMUL "java.lang.Integer" "intValue" "()I" "valueOf" "(I)" + ^:private compile-jvm-idiv Opcodes/IDIV "java.lang.Integer" "intValue" "()I" "valueOf" "(I)" + ^:private compile-jvm-irem Opcodes/IREM "java.lang.Integer" "intValue" "()I" "valueOf" "(I)" + + ^:private compile-jvm-ladd Opcodes/LADD "java.lang.Long" "longValue" "()J" "valueOf" "(J)" + ^:private compile-jvm-lsub Opcodes/LSUB "java.lang.Long" "longValue" "()J" "valueOf" "(J)" + ^:private compile-jvm-lmul Opcodes/LMUL "java.lang.Long" "longValue" "()J" "valueOf" "(J)" + ^:private compile-jvm-ldiv Opcodes/LDIV "java.lang.Long" "longValue" "()J" "valueOf" "(J)" + ^:private compile-jvm-lrem Opcodes/LREM "java.lang.Long" "longValue" "()J" "valueOf" "(J)" + + ^:private compile-jvm-fadd Opcodes/FADD "java.lang.Float" "floatValue" "()F" "valueOf" "(F)" + ^:private compile-jvm-fsub Opcodes/FSUB "java.lang.Float" "floatValue" "()F" "valueOf" "(F)" + ^:private compile-jvm-fmul Opcodes/FMUL "java.lang.Float" "floatValue" "()F" "valueOf" "(F)" + ^:private compile-jvm-fdiv Opcodes/FDIV "java.lang.Float" "floatValue" "()F" "valueOf" "(F)" + ^:private compile-jvm-frem Opcodes/FREM "java.lang.Float" "floatValue" "()F" "valueOf" "(F)" + + ^:private compile-jvm-dadd Opcodes/DADD "java.lang.Double" "doubleValue" "()D" "valueOf" "(D)" + ^:private compile-jvm-dsub Opcodes/DSUB "java.lang.Double" "doubleValue" "()D" "valueOf" "(D)" + ^:private compile-jvm-dmul Opcodes/DMUL "java.lang.Double" "doubleValue" "()D" "valueOf" "(D)" + ^:private compile-jvm-ddiv Opcodes/DDIV "java.lang.Double" "doubleValue" "()D" "valueOf" "(D)" + ^:private compile-jvm-drem Opcodes/DREM "java.lang.Double" "doubleValue" "()D" "valueOf" "(D)" + ) -(defn compile-self-call [compile ?assumed-args] +(defn ^:private compile-self-call [compile ?assumed-args] (exec [*writer* &util/get-writer :let [_ (.visitVarInsn *writer* Opcodes/ALOAD 0)] _ (map-m (fn [arg] (exec [ret (compile arg) - :let [_ (.visitMethodInsn *writer* Opcodes/INVOKEINTERFACE (str +prefix+ "/Function") "apply" +apply-signature+)]] + :let [_ (.visitMethodInsn *writer* Opcodes/INVOKEINTERFACE (->class +function-class+) "apply" +apply-signature+)]] (return ret))) ?assumed-args)] (return nil))) -(defn ^:private compile [syntax] +(defn ^:private compile-expression [syntax] (match (:form syntax) [::&analyser/bool ?value] - (compile-bool compile (:type syntax) ?value) + (compile-bool compile-expression (:type syntax) ?value) [::&analyser/int ?value] - (compile-int compile (:type syntax) ?value) + (compile-int compile-expression (:type syntax) ?value) [::&analyser/real ?value] - (compile-real compile (:type syntax) ?value) + (compile-real compile-expression (:type syntax) ?value) [::&analyser/char ?value] - (compile-char compile (:type syntax) ?value) + (compile-char compile-expression (:type syntax) ?value) [::&analyser/text ?value] - (compile-text compile (:type syntax) ?value) + (compile-text compile-expression (:type syntax) ?value) [::&analyser/tuple ?elems] - (compile-tuple compile (:type syntax) ?elems) + (compile-tuple compile-expression (:type syntax) ?elems) [::&analyser/local ?idx] - (compile-local compile (:type syntax) ?idx) + (compile-local compile-expression (:type syntax) ?idx) [::&analyser/captured ?scope ?captured-id ?source] - (compile-captured compile (:type syntax) ?scope ?captured-id ?source) + (compile-captured compile-expression (:type syntax) ?scope ?captured-id ?source) [::&analyser/global ?owner-class ?name] - (compile-global compile (:type syntax) ?owner-class ?name) + (compile-global compile-expression (:type syntax) ?owner-class ?name) [::&analyser/call ?fn ?args] - (compile-call compile (:type syntax) ?fn ?args) + (compile-call compile-expression (:type syntax) ?fn ?args) [::&analyser/static-call ?needs-num ?fn ?args] - (compile-static-call compile (:type syntax) ?needs-num ?fn ?args) + (compile-static-call compile-expression (:type syntax) ?needs-num ?fn ?args) [::&analyser/variant ?tag ?members] - (compile-variant compile (:type syntax) ?tag ?members) + (compile-variant compile-expression (:type syntax) ?tag ?members) [::&analyser/let ?idx ?value ?body] - (compile-let compile (:type syntax) ?idx ?value ?body) + (compile-let compile-expression (:type syntax) ?idx ?value ?body) [::&analyser/case ?base-idx ?variant ?max-registers ?branch-mappings ?decision-tree] - (compile-case compile (:type syntax) ?base-idx ?variant ?max-registers ?branch-mappings ?decision-tree) + (compile-case compile-expression (:type syntax) ?base-idx ?variant ?max-registers ?branch-mappings ?decision-tree) [::&analyser/lambda ?scope ?frame ?args ?body] - (compile-lambda compile (:type syntax) ?scope ?frame ?args ?body false true) + (compile-lambda compile-expression (:type syntax) ?scope ?frame ?args ?body false true) - [::&analyser/def ?form ?body] - (compile-def compile (:type syntax) ?form ?body) - + ;; Integer arithmetic [::&analyser/jvm-iadd ?x ?y] - (compile-jvm-iadd compile (:type syntax) ?x ?y) + (compile-jvm-iadd compile-expression (:type syntax) ?x ?y) [::&analyser/jvm-isub ?x ?y] - (compile-jvm-isub compile (:type syntax) ?x ?y) + (compile-jvm-isub compile-expression (:type syntax) ?x ?y) [::&analyser/jvm-imul ?x ?y] - (compile-jvm-imul compile (:type syntax) ?x ?y) + (compile-jvm-imul compile-expression (:type syntax) ?x ?y) [::&analyser/jvm-idiv ?x ?y] - (compile-jvm-idiv compile (:type syntax) ?x ?y) + (compile-jvm-idiv compile-expression (:type syntax) ?x ?y) [::&analyser/jvm-irem ?x ?y] - (compile-jvm-irem compile (:type syntax) ?x ?y) + (compile-jvm-irem compile-expression (:type syntax) ?x ?y) + + ;; Long arithmetic + [::&analyser/jvm-ladd ?x ?y] + (compile-jvm-ladd compile-expression (:type syntax) ?x ?y) + + [::&analyser/jvm-lsub ?x ?y] + (compile-jvm-lsub compile-expression (:type syntax) ?x ?y) + + [::&analyser/jvm-lmul ?x ?y] + (compile-jvm-lmul compile-expression (:type syntax) ?x ?y) + + [::&analyser/jvm-ldiv ?x ?y] + (compile-jvm-ldiv compile-expression (:type syntax) ?x ?y) + + [::&analyser/jvm-lrem ?x ?y] + (compile-jvm-lrem compile-expression (:type syntax) ?x ?y) + ;; Float arithmetic + [::&analyser/jvm-fadd ?x ?y] + (compile-jvm-fadd compile-expression (:type syntax) ?x ?y) + + [::&analyser/jvm-fsub ?x ?y] + (compile-jvm-fsub compile-expression (:type syntax) ?x ?y) + + [::&analyser/jvm-fmul ?x ?y] + (compile-jvm-fmul compile-expression (:type syntax) ?x ?y) + + [::&analyser/jvm-fdiv ?x ?y] + (compile-jvm-fdiv compile-expression (:type syntax) ?x ?y) + + [::&analyser/jvm-frem ?x ?y] + (compile-jvm-frem compile-expression (:type syntax) ?x ?y) + + ;; Double arithmetic + [::&analyser/jvm-dadd ?x ?y] + (compile-jvm-dadd compile-expression (:type syntax) ?x ?y) + + [::&analyser/jvm-dsub ?x ?y] + (compile-jvm-dsub compile-expression (:type syntax) ?x ?y) + + [::&analyser/jvm-dmul ?x ?y] + (compile-jvm-dmul compile-expression (:type syntax) ?x ?y) + + [::&analyser/jvm-ddiv ?x ?y] + (compile-jvm-ddiv compile-expression (:type syntax) ?x ?y) + + [::&analyser/jvm-drem ?x ?y] + (compile-jvm-drem compile-expression (:type syntax) ?x ?y) + [::&analyser/do ?exprs] - (compile-do compile (:type syntax) ?exprs) + (compile-do compile-expression (:type syntax) ?exprs) [::&analyser/jvm-new ?class ?classes ?args] - (compile-jvm-new compile (:type syntax) ?class ?classes ?args) + (compile-jvm-new compile-expression (:type syntax) ?class ?classes ?args) [::&analyser/jvm-getstatic ?class ?field] - (compile-jvm-getstatic compile (:type syntax) ?class ?field) + (compile-jvm-getstatic compile-expression (:type syntax) ?class ?field) [::&analyser/jvm-getfield ?class ?field ?object] - (compile-jvm-getfield compile (:type syntax) ?class ?field ?object) + (compile-jvm-getfield compile-expression (:type syntax) ?class ?field ?object) [::&analyser/jvm-invokestatic ?class ?method ?classes ?args] - (compile-jvm-invokestatic compile (:type syntax) ?class ?method ?classes ?args) + (compile-jvm-invokestatic compile-expression (:type syntax) ?class ?method ?classes ?args) [::&analyser/jvm-invokevirtual ?class ?method ?classes ?object ?args] - (compile-jvm-invokevirtual compile (:type syntax) ?class ?method ?classes ?object ?args) + (compile-jvm-invokevirtual compile-expression (:type syntax) ?class ?method ?classes ?object ?args) [::&analyser/jvm-new-array ?class ?length] - (compile-jvm-new-array compile (:type syntax) ?class ?length) + (compile-jvm-new-array compile-expression (:type syntax) ?class ?length) [::&analyser/jvm-aastore ?array ?idx ?elem] - (compile-jvm-aastore compile (:type syntax) ?array ?idx ?elem) + (compile-jvm-aastore compile-expression (:type syntax) ?array ?idx ?elem) [::&analyser/jvm-aaload ?array ?idx] - (compile-jvm-aaload compile (:type syntax) ?array ?idx) + (compile-jvm-aaload compile-expression (:type syntax) ?array ?idx) + [::&analyser/self ?assumed-args] + (compile-self-call compile-expression ?assumed-args) + + _ + (fail "[Compiler Error] Can't compile expressions as top-level forms.") + )) + +(defn ^:private compile-statement [syntax] + (match (:form syntax) + [::&analyser/def ?form ?body] + (compile-def compile-expression (:type syntax) ?form ?body) + [::&analyser/jvm-interface [?package ?name] ?members] - (compile-jvm-interface compile (:type syntax) ?package ?name ?members) + (compile-jvm-interface compile-expression (:type syntax) ?package ?name ?members) [::&analyser/jvm-class [?package ?name] ?super-class ?members] - (compile-jvm-class compile (:type syntax) ?package ?name ?super-class ?members) + (compile-jvm-class compile-expression (:type syntax) ?package ?name ?super-class ?members) - [::&analyser/self ?assumed-args] - (compile-self-call compile ?assumed-args) + _ + (fail "[Compiler Error] Can't compile expressions as top-level forms.") )) ;; [Interface] (let [compiler-step (exec [analysis+ &analyser/analyse] - (map-m compile analysis+))] + (map-m compile-statement analysis+))] (defn compile-module [name] (exec [loader &util/loader] (fn [state] @@ -994,28 +1090,25 @@ (let [=class (doto (new ClassWriter ClassWriter/COMPUTE_MAXS) (.visit Opcodes/V1_5 (+ Opcodes/ACC_PUBLIC Opcodes/ACC_SUPER) (->class name) nil "java/lang/Object" nil))] - (match ((repeat-m compiler-step) (assoc state - ::&util/source (slurp (str "source/" name ".lux")) - ::&util/current-module name - ::&util/writer =class)) - [::&util/ok [?state ?forms]] - (if (empty? (::&util/source ?state)) - (do (.visitEnd =class) - ((save-class! name (.toByteArray =class)) ?state)) - (assert false (str "[Compiler Error] Can't compile: " (::&util/source ?state)))) + (match (&util/run-state (exhaust-m compiler-step) (assoc state + ::&util/source (slurp (str "source/" name ".lux")) + ::&util/current-module name + ::&util/writer =class)) + [::&util/ok [?state _]] + (do (.visitEnd =class) + (&util/run-state (save-class! name (.toByteArray =class)) ?state)) [::&util/failure ?message] (fail* ?message)))))))) (defn compile-all [modules] (.mkdir (java.io.File. "output")) - (let [state (&util/init-state)] - (match ((map-m compile-module modules) state) - [::&util/ok [?state ?forms]] - (println (str "Compilation complete! " (pr-str modules))) + (match (&util/run-state (map-m compile-module modules) (&util/init-state)) + [::&util/ok [?state _]] + (println (str "Compilation complete! " (pr-str modules))) - [::&util/failure ?message] - (assert false ?message)))) + [::&util/failure ?message] + (assert false ?message))) (comment (compile-all ["lux"]) diff --git a/src/lux/type.clj b/src/lux/type.clj index a7bc8b522..9d9cf77aa 100644 --- a/src/lux/type.clj +++ b/src/lux/type.clj @@ -11,14 +11,14 @@ (defn ^:private resolve [id] (fn [state] - (if-let [top+bottom (get-in state [::mappings id])] + (if-let [top+bottom (get-in state [::&util/types ::mappings id])] [::&util/ok [state top+bottom]] [::&util/failure (str "Unknown type-var: " id)]))) (defn ^:private update [id top bottom] (fn [state] - (if-let [top+bottom (get-in state [::mappings id])] - [::&util/ok [(assoc-in state [::mappings id] [top bottom]) nil]] + (if-let [top+bottom (get-in state [::&util/types ::mappings id])] + [::&util/ok [(assoc-in state [::&util/types ::mappings id] [top bottom]) nil]] [::&util/failure (str "Unknown type-var: " id)]))) ;; [Interface] diff --git a/src/lux/util.clj b/src/lux/util.clj index 9bd8ed42c..207a07203 100644 --- a/src/lux/util.clj +++ b/src/lux/util.clj @@ -150,15 +150,6 @@ (exec [head (first m-values)] (sequence-m (rest monads))))) -(defn within [slot monad] - (fn [state] - (let [=return (monad (get state slot))] - (match =return - [::ok [?state ?value]] - [::ok [(assoc state slot ?state) ?value]] - _ - =return)))) - (defn ^:private normalize-char [char] (case char \* "_ASTER_" @@ -201,10 +192,15 @@ {:counter 0 :mappings {}}) +(defn scope [name] + {:name name + :inner-lambdas 0 + :locals +init-env+ + :closure +init-env+}) + (defn init-state [] {::source nil ::current-module nil - ::scope (list) ::modules {} ::global-env {} ::local-envs (list) @@ -212,12 +208,26 @@ ::writer nil ::loader (class-loader!)}) -(do-template [<name>] +(do-template [<name> <tag>] (def <name> (fn [state] - [::ok [state (::current-module state)]])) + (if-let [datum (<tag> state)] + [::ok [state datum]] + [::failure (str "Data does not exist: " <tag>)]))) get-module-name ::current-module - get-scope ::scope get-writer ::writer ) + +(defn with-writer [writer body] + (fn [state] + (let [output (body (assoc state ::writer writer))] + (match output + [::ok [?state ?value]] + [::ok [(assoc ?state ::writer (::writer state)) ?value]] + + _ + output)))) + +(defn run-state [monad state] + (monad state)) |