Skip to content

Commit 576fb6e

Browse files
author
dnolen
committedApr 6, 2015
CLJS-1188: multi-arity fns hinder cross-module code motion
This patch makes all top level function definitions completely static, we never issue an invoke to produce a top-level function value. This is accomplished by duplicating and further enhancing the fn emission logic in cljs.compiler at the macro level. The enhancements are entirely around eliminating invokes and any property aliasing. While useful in expression contexts, at the top level both of these approaches in cljs.compiler defeat cross module code motion. - test-simple should clean builds - cljs.analyzer * remove :method info, never used * read fn information from :top-fn meta if available - cljs.closure * enhance module build reporting - cljs.core * move clojure.core/defn macro + helpers directly into macro ns * handle top level multi-arity & variadic fns - cjls.compiler-tests * include some examples
1 parent 9bf486b commit 576fb6e

File tree

5 files changed

+298
-54
lines changed

5 files changed

+298
-54
lines changed
 

‎script/test-simple

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,7 @@
11
#!/bin/sh
22

33
# stop blowing compiled stuff
4-
#rm -rf builds/out-simp
4+
rm -rf builds/out-simp
55
mkdir -p builds/out-simp
66

77
possible=4

‎src/clj/cljs/analyzer.clj

+17-24
Original file line numberDiff line numberDiff line change
@@ -864,22 +864,19 @@
864864
:impls #{}})
865865
(when fn-var?
866866
(let [params (map #(vec (map :name (:params %))) (:methods init-expr))]
867-
{:fn-var true
868-
;; protocol implementation context
869-
:protocol-impl (:protocol-impl init-expr)
870-
;; inline protocol implementation context
871-
:protocol-inline (:protocol-inline init-expr)
872-
:variadic (:variadic init-expr)
873-
:max-fixed-arity (:max-fixed-arity init-expr)
874-
:method-params params
875-
:arglists (:arglists sym-meta)
876-
:arglists-meta (doall (map meta (:arglists sym-meta)))
877-
:methods (map (fn [method]
878-
(let [tag (infer-tag env (assoc method :op :method))]
879-
(cond-> (select-keys method
880-
[:max-fixed-arity :variadic])
881-
tag (assoc :tag tag))))
882-
(:methods init-expr))}) )
867+
(merge
868+
{:fn-var true
869+
;; protocol implementation context
870+
:protocol-impl (:protocol-impl init-expr)
871+
;; inline protocol implementation context
872+
:protocol-inline (:protocol-inline init-expr)}
873+
(if-let [top-fn-meta (:top-fn sym-meta)]
874+
top-fn-meta
875+
{:variadic (:variadic init-expr)
876+
:max-fixed-arity (:max-fixed-arity init-expr)
877+
:method-params params
878+
:arglists (:arglists sym-meta)
879+
:arglists-meta (doall (map meta (:arglists sym-meta)))}))) )
883880
(when (and fn-var? tag)
884881
{:ret-tag tag})))
885882
(merge
@@ -978,8 +975,7 @@
978975
:fn-var true
979976
:variadic variadic
980977
:max-fixed-arity max-fixed-arity
981-
:method-params (map :params methods)
982-
:methods methods)
978+
:method-params (map :params methods))
983979
locals)
984980
methods (if name
985981
;; a second pass with knowledge of our function-ness/arity
@@ -1027,8 +1023,7 @@
10271023
:shadow (locals n)
10281024
:variadic (:variadic fexpr)
10291025
:max-fixed-arity (:max-fixed-arity fexpr)
1030-
:method-params (map :params (:methods fexpr))
1031-
:methods (:methods fexpr)}
1026+
:method-params (map :params (:methods fexpr))}
10321027
ret-tag (assoc :ret-tag ret-tag))]
10331028
[(assoc-in env [:locals n] be)
10341029
(conj bes be)]))
@@ -1043,8 +1038,7 @@
10431038
:init fexpr
10441039
:variadic (:variadic fexpr)
10451040
:max-fixed-arity (:max-fixed-arity fexpr)
1046-
:method-params (map :params (:methods fexpr))
1047-
:methods (:methods fexpr))]
1041+
:method-params (map :params (:methods fexpr)))]
10481042
[(assoc-in env [:locals name] be')
10491043
(conj bes be')]))
10501044
[meth-env []] bes)
@@ -1102,8 +1096,7 @@
11021096
{:fn-var true
11031097
:variadic (:variadic init-expr)
11041098
:max-fixed-arity (:max-fixed-arity init-expr)
1105-
:method-params (map :params (:methods init-expr))
1106-
:methods (:methods init-expr)})
1099+
:method-params (map :params (:methods init-expr))})
11071100
be)]
11081101
(recur (conj bes be)
11091102
(assoc-in env [:locals name] be)

‎src/clj/cljs/closure.clj

+2-2
Original file line numberDiff line numberDiff line change
@@ -701,7 +701,7 @@ should contain the source for the given namespace name."
701701
(fn [[sources ret] [name {:keys [entries output-to depends-on] :as module-desc}]]
702702
(assert (or (= name :cljs-base) (not (empty? entries)))
703703
(str "Module " name " does not define any :entries"))
704-
(when (and (:verbose opts) (not= name :cljs-base))
704+
(when (:verbose opts)
705705
(util/debug-prn "Building module" name))
706706
(let [js-module (JSModule. (clojure.core/name name))
707707
[sources' module-sources]
@@ -750,7 +750,7 @@ should contain the source for the given namespace name."
750750
cljs-base-closure-module (get-in (into {} modules) [:cljs-base :closure-module])
751751
foreign-deps (atom [])]
752752
(when (:verbose opts)
753-
(util/debug-prn "Building module" :cljs-base))
753+
(util/debug-prn "Adding remaining namespaces to" :cljs-base))
754754
;; add anything left to :cljs-base module
755755
(doseq [source sources']
756756
(when (:verbose opts)

‎src/clj/cljs/core.clj

+265-27
Original file line numberDiff line numberDiff line change
@@ -38,6 +38,7 @@
3838
(:require clojure.walk
3939
clojure.set
4040
cljs.compiler
41+
[cljs.util :as util]
4142
[cljs.env :as env])
4243
(:import [java.io File]))
4344

@@ -74,7 +75,7 @@
7475

7576
(import-macros clojure.core
7677
[-> ->> .. assert comment cond
77-
declare defn defn-
78+
declare defn-
7879
doto
7980
extend-protocol fn for
8081
if-let if-not letfn
@@ -83,6 +84,54 @@
8384
cond-> cond->> as-> some-> some->>
8485
if-some when-some])
8586

87+
(defn- ^{:dynamic true} assert-valid-fdecl
88+
"A good fdecl looks like (([a] ...) ([a b] ...)) near the end of defn."
89+
[fdecl]
90+
(when (empty? fdecl) (throw (IllegalArgumentException.
91+
"Parameter declaration missing")))
92+
(core/let [argdecls (map
93+
#(if (seq? %)
94+
(first %)
95+
(throw (IllegalArgumentException.
96+
(if (seq? (first fdecl))
97+
(core/str "Invalid signature \""
98+
%
99+
"\" should be a list")
100+
(core/str "Parameter declaration \""
101+
%
102+
"\" should be a vector")))))
103+
fdecl)
104+
bad-args (seq (remove #(vector? %) argdecls))]
105+
(when bad-args
106+
(throw (IllegalArgumentException.
107+
(core/str "Parameter declaration \"" (first bad-args)
108+
"\" should be a vector"))))))
109+
110+
(def
111+
^{:private true}
112+
sigs
113+
(fn [fdecl]
114+
(assert-valid-fdecl fdecl)
115+
(core/let [asig
116+
(fn [fdecl]
117+
(core/let [arglist (first fdecl)
118+
;elide implicit macro args
119+
arglist (if (clojure.lang.Util/equals '&form (first arglist))
120+
(clojure.lang.RT/subvec arglist 2 (clojure.lang.RT/count arglist))
121+
arglist)
122+
body (next fdecl)]
123+
(if (map? (first body))
124+
(if (next body)
125+
(with-meta arglist (conj (if (meta arglist) (meta arglist) {}) (first body)))
126+
arglist)
127+
arglist)))]
128+
(if (seq? (first fdecl))
129+
(core/loop [ret [] fdecls fdecl]
130+
(if fdecls
131+
(recur (conj ret (asig (first fdecls))) (next fdecls))
132+
(seq ret)))
133+
(core/list (asig fdecl))))))
134+
86135
(defmacro defonce [x init]
87136
`(when-not (exists? ~x)
88137
(def ~x ~init)))
@@ -95,7 +144,7 @@
95144
(when more
96145
(list* `assert-args fnname more)))))
97146

98-
(defn destructure [bindings]
147+
(core/defn destructure [bindings]
99148
(core/let [bents (partition 2 bindings)
100149
pb (fn pb [bvec b v]
101150
(core/let [pvec
@@ -227,10 +276,10 @@
227276
(apply core/str))]
228277
(list* 'js* (core/str "[" strs "].join('')") xs)))
229278

230-
(defn bool-expr [e]
279+
(defn- bool-expr [e]
231280
(vary-meta e assoc :tag 'boolean))
232281

233-
(defn simple-test-expr? [env ast]
282+
(defn- simple-test-expr? [env ast]
234283
(core/and
235284
(#{:var :invoke :constant :dot :js} (:op ast))
236285
('#{boolean seq} (cljs.analyzer/infer-tag env ast))))
@@ -607,7 +656,7 @@
607656

608657
;;; end of reducers macros
609658

610-
(defn protocol-prefix [psym]
659+
(defn- protocol-prefix [psym]
611660
(core/str (-> (core/str psym) (.replace \. \$) (.replace \/ \$)) "$"))
612661

613662
(def #^:private base-type
@@ -708,10 +757,10 @@
708757
`(let [~name (js-this)]
709758
~@body))
710759

711-
(defn to-property [sym]
760+
(defn- to-property [sym]
712761
(symbol (core/str "-" sym)))
713762

714-
(defn warn-and-update-protocol [p type env]
763+
(defn- warn-and-update-protocol [p type env]
715764
(when-not (= 'Object p)
716765
(if-let [var (cljs.analyzer/resolve-existing-var (dissoc env :locals) p)]
717766
(do
@@ -729,21 +778,21 @@
729778
(when (:undeclared cljs.analyzer/*cljs-warnings*)
730779
(cljs.analyzer/warning :undeclared-protocol-symbol env {:protocol p})))))
731780

732-
(defn resolve-var [env sym]
781+
(defn- resolve-var [env sym]
733782
(let [ret (-> (dissoc env :locals)
734783
(cljs.analyzer/resolve-var sym)
735784
:name)]
736785
(assert ret (core/str "Can't resolve: " sym))
737786
ret))
738787

739-
(defn ->impl-map [impls]
788+
(defn- ->impl-map [impls]
740789
(loop [ret {} s impls]
741790
(if (seq s)
742791
(recur (assoc ret (first s) (take-while seq? (next s)))
743792
(drop-while seq? (next s)))
744793
ret)))
745794

746-
(defn base-assign-impls [env resolve tsym type [p sigs]]
795+
(defn- base-assign-impls [env resolve tsym type [p sigs]]
747796
(warn-and-update-protocol p tsym env)
748797
(let [psym (resolve p)
749798
pfn-prefix (subs (core/str psym) 0
@@ -762,29 +811,29 @@
762811
(core/defmethod extend-prefix :default
763812
[tsym sym] `(.. ~tsym -prototype ~(to-property sym)))
764813

765-
(defn adapt-obj-params [type [[this & args :as sig] & body]]
814+
(defn- adapt-obj-params [type [[this & args :as sig] & body]]
766815
(core/list (vec args)
767816
(list* 'this-as (vary-meta this assoc :tag type) body)))
768817

769-
(defn adapt-ifn-params [type [[this & args :as sig] & body]]
818+
(defn- adapt-ifn-params [type [[this & args :as sig] & body]]
770819
(let [self-sym (with-meta 'self__ {:tag type})]
771820
`(~(vec (cons self-sym args))
772821
(this-as ~self-sym
773822
(let [~this ~self-sym]
774823
~@body)))))
775824

776825
;; for IFn invoke implementations, we need to drop first arg
777-
(defn adapt-ifn-invoke-params [type [[this & args :as sig] & body]]
826+
(defn- adapt-ifn-invoke-params [type [[this & args :as sig] & body]]
778827
`(~(vec args)
779828
(this-as ~(vary-meta this assoc :tag type)
780829
~@body)))
781830

782-
(defn adapt-proto-params [type [[this & args :as sig] & body]]
831+
(defn- adapt-proto-params [type [[this & args :as sig] & body]]
783832
`(~(vec (cons (vary-meta this assoc :tag type) args))
784833
(this-as ~this
785834
~@body)))
786835

787-
(defn add-obj-methods [type type-sym sigs]
836+
(defn- add-obj-methods [type type-sym sigs]
788837
(map (fn [[f & meths :as form]]
789838
(let [[f meths] (if (vector? (first meths))
790839
[f [(rest form)]]
@@ -793,15 +842,15 @@
793842
~(with-meta `(fn ~@(map #(adapt-obj-params type %) meths)) (meta form)))))
794843
sigs))
795844

796-
(defn ifn-invoke-methods [type type-sym [f & meths :as form]]
845+
(defn- ifn-invoke-methods [type type-sym [f & meths :as form]]
797846
(map
798847
(fn [meth]
799848
(let [arity (count (first meth))]
800849
`(set! ~(extend-prefix type-sym (symbol (core/str "cljs$core$IFn$_invoke$arity$" arity)))
801850
~(with-meta `(fn ~meth) (meta form)))))
802851
(map #(adapt-ifn-invoke-params type %) meths)))
803852

804-
(defn add-ifn-methods [type type-sym [f & meths :as form]]
853+
(defn- add-ifn-methods [type type-sym [f & meths :as form]]
805854
(let [meths (map #(adapt-ifn-params type %) meths)
806855
this-sym (with-meta 'self__ {:tag type})
807856
argsym (gensym "args")]
@@ -816,7 +865,7 @@
816865
(meta form)))]
817866
(ifn-invoke-methods type type-sym form))))
818867

819-
(defn add-proto-methods* [pprefix type type-sym [f & meths :as form]]
868+
(defn- add-proto-methods* [pprefix type type-sym [f & meths :as form]]
820869
(let [pf (core/str pprefix f)]
821870
(if (vector? (first meths))
822871
;; single method case
@@ -828,7 +877,7 @@
828877
~(with-meta `(fn ~(adapt-proto-params type meth)) (meta form))))
829878
meths))))
830879

831-
(defn proto-assign-impls [env resolve type-sym type [p sigs]]
880+
(defn- proto-assign-impls [env resolve type-sym type [p sigs]]
832881
(warn-and-update-protocol p type env)
833882
(let [psym (resolve p)
834883
pprefix (protocol-prefix psym)
@@ -845,7 +894,7 @@
845894
(add-proto-methods* pprefix type type-sym sig)))
846895
sigs)))))
847896

848-
(defn validate-impl-sigs [env p method]
897+
(defn- validate-impl-sigs [env p method]
849898
(when-not (= p 'Object)
850899
(let [var (ana/resolve-var (dissoc env :locals) p)
851900
minfo (-> var :protocol-info :methods)
@@ -865,7 +914,7 @@
865914
(ana/warning :protocol-invalid-method env {:protocol p :fname fname :invalid-arity c}))
866915
(recur (next sigs) (conj seen c))))))))
867916

868-
(defn validate-impls [env impls]
917+
(defn- validate-impls [env impls]
869918
(loop [protos #{} impls impls]
870919
(when (seq impls)
871920
(let [proto (first impls)
@@ -930,12 +979,12 @@
930979
parts
931980
(range fast-path-protocol-partitions-count))]))))
932981

933-
(defn annotate-specs [annots v [f sigs]]
982+
(defn- annotate-specs [annots v [f sigs]]
934983
(conj v
935984
(vary-meta (cons f (map #(cons (second %) (nnext %)) sigs))
936985
merge annots)))
937986

938-
(defn dt->et
987+
(core/defn dt->et
939988
([type specs fields]
940989
(dt->et type specs fields false))
941990
([type specs fields inline]
@@ -952,7 +1001,7 @@
9521001
(recur ret specs))
9531002
ret)))))
9541003

955-
(defn collect-protocols [impls env]
1004+
(defn- collect-protocols [impls env]
9561005
(->> impls
9571006
(filter core/symbol?)
9581007
(map #(:name (cljs.analyzer/resolve-var (dissoc env :locals) %)))
@@ -1738,7 +1787,7 @@
17381787
`(.fromArray cljs.core/PersistentHashSet (array ~@xs) true)
17391788
assoc :tag 'cljs.core/PersistentHashSet))))
17401789

1741-
(defn js-obj* [kvs]
1790+
(defn- js-obj* [kvs]
17421791
(let [kvs-str (->> (repeat "~{}:~{}")
17431792
(take (count kvs))
17441793
(interpose ",")
@@ -1810,7 +1859,7 @@
18101859
~@body
18111860
(recur (inc ~i)))))))
18121861

1813-
(defn ^:private check-valid-options
1862+
(defn- check-valid-options
18141863
"Throws an exception if the given option map contains keys not listed
18151864
as valid, else returns nil."
18161865
[options & valid-keys]
@@ -1897,7 +1946,7 @@
18971946

18981947
(def cs (into [] (map (comp gensym core/str core/char) (range 97 118))))
18991948

1900-
(defn gen-apply-to-helper
1949+
(defn- gen-apply-to-helper
19011950
([] (gen-apply-to-helper 1))
19021951
([n]
19031952
(let [prop (symbol (core/str "-cljs$core$IFn$_invoke$arity$" n))
@@ -2004,3 +2053,192 @@
20042053
(core/if-not (core/identical? form form')
20052054
(recur form' (ana/macroexpand-1 env form'))
20062055
`(quote ~form')))))
2056+
2057+
(defn- multi-arity-fn? [fdecl]
2058+
(core/< 1 (count fdecl)))
2059+
2060+
(defn- variadic-fn? [fdecl]
2061+
(core/and (= 1 (count fdecl))
2062+
(some '#{&} (ffirst fdecl))))
2063+
2064+
(defn- variadic-fn*
2065+
([sym method]
2066+
(variadic-fn* sym method true))
2067+
([sym [arglist & body :as method] solo]
2068+
(let [sig (remove '#{&} arglist)
2069+
restarg (gensym "seq")]
2070+
(letfn [(get-delegate []
2071+
'cljs$core$IFn$_invoke$arity$variadic)
2072+
(get-delegate-prop []
2073+
(symbol (core/str "-" (get-delegate))))
2074+
(param-bind [param]
2075+
`[~param (^::ana/no-resolve first ~restarg)
2076+
~restarg (^::ana/no-resolve next ~restarg)])
2077+
(apply-to []
2078+
(if (core/< 1 (count sig))
2079+
(let [params (repeatedly (core/dec (count sig)) gensym)]
2080+
`(fn
2081+
([~restarg]
2082+
(let [~@(mapcat param-bind params)]
2083+
(. ~sym (~(get-delegate) ~@params ~restarg))))))
2084+
`(fn
2085+
([~restarg]
2086+
(. ~sym (~(get-delegate) (seq ~restarg)))))))]
2087+
`(do
2088+
(set! (. ~sym ~(get-delegate-prop))
2089+
(fn (~(vec sig) ~@body)))
2090+
~@(when solo
2091+
`[(set! (. ~sym ~'-cljs$lang$maxFixedArity)
2092+
~(core/dec (count sig)))])
2093+
(set! (. ~sym ~'-cljs$lang$applyTo)
2094+
~(apply-to)))))))
2095+
2096+
(defn- variadic-fn [name meta [[arglist & body :as method] :as fdecl]]
2097+
(letfn [(dest-args [c]
2098+
(map (fn [n] `(aget (js-arguments) ~n))
2099+
(range c)))]
2100+
(core/let [rname (symbol (core/str ana/*cljs-ns*) (core/str name))
2101+
sig (remove '#{&} arglist)
2102+
c-1 (core/dec (count sig))
2103+
meta (assoc meta
2104+
:top-fn
2105+
{:variadic true
2106+
:max-fixed-arity c-1
2107+
:method-params [sig]
2108+
:arglists [arglist]
2109+
:arglists-meta (doall (map meta [arglist]))})]
2110+
`(do
2111+
(def ~(with-meta name meta)
2112+
(fn []
2113+
(let [argseq# (when (< ~c-1 (alength (js-arguments)))
2114+
(new ^::ana/no-resolve cljs.core/IndexedSeq
2115+
(.call js/Array.prototype.slice
2116+
(js-arguments) ~c-1) 0))]
2117+
(. ~rname
2118+
(~'cljs$core$IFn$_invoke$arity$variadic ~@(dest-args c-1) argseq#)))))
2119+
~(variadic-fn* rname method)))))
2120+
2121+
(comment
2122+
(require '[clojure.pprint :as pp])
2123+
(pp/pprint (variadic-fn 'foo {} '(([& xs]))))
2124+
(pp/pprint (variadic-fn 'foo {} '(([a & xs] xs))))
2125+
(pp/pprint (variadic-fn 'foo {} '(([a b & xs] xs))))
2126+
(pp/pprint (variadic-fn 'foo {} '(([a [b & cs] & xs] xs))))
2127+
)
2128+
2129+
(defn- multi-arity-fn [name meta fdecl]
2130+
(letfn [(dest-args [c]
2131+
(map (fn [n] `(aget (js-arguments) ~n))
2132+
(range c)))
2133+
(fixed-arity [rname sig]
2134+
(let [c (count sig)]
2135+
[c `(. ~rname
2136+
(~(symbol
2137+
(core/str "cljs$core$IFn$_invoke$arity$" c))
2138+
~@(dest-args c)))]))
2139+
(fn-method [[sig & body :as method]]
2140+
(if (some '#{&} sig)
2141+
(variadic-fn* name method false)
2142+
`(set!
2143+
(. ~name
2144+
~(symbol (core/str "-cljs$core$IFn$_invoke$arity$"
2145+
(count sig))))
2146+
(fn ~method))))]
2147+
(core/let [rname (symbol (core/str ana/*cljs-ns*) (core/str name))
2148+
arglists (map first fdecl)
2149+
variadic (boolean (some #(some '#{&} %) arglists))
2150+
sigs (remove #(some '#{&} %) arglists)
2151+
maxfa (apply core/max (map count sigs))
2152+
meta (assoc meta
2153+
:top-fn
2154+
{:variadic variadic
2155+
:max-fixed-arity maxfa
2156+
:method-params sigs
2157+
:arglists arglists
2158+
:arglists-meta (doall (map meta arglists))})]
2159+
`(do
2160+
(def ~(with-meta name meta)
2161+
(fn []
2162+
(case (alength (js-arguments))
2163+
~@(mapcat #(fixed-arity rname %) sigs)
2164+
~(if variadic
2165+
`(let [argseq# (new ^::ana/no-resolve cljs.core/IndexedSeq
2166+
(.call js/Array.prototype.slice
2167+
(js-arguments) ~maxfa) 0)]
2168+
(. ~rname
2169+
(~'cljs$core$IFn$_invoke$arity$variadic
2170+
~@(dest-args maxfa)
2171+
argseq#)))
2172+
`(throw (js/Error.
2173+
(str "Invalid arity: "
2174+
(alength (js-arguments)))))))))
2175+
~@(map fn-method fdecl)
2176+
;; optimization properties
2177+
(set! (. ~name ~'-cljs$lang$maxFixedArity) ~maxfa)))))
2178+
2179+
(comment
2180+
(require '[clojure.pprint :as pp])
2181+
(pp/pprint (multi-arity-fn 'foo {} '(([a]) ([a b]))))
2182+
(pp/pprint (multi-arity-fn 'foo {} '(([a]) ([a & xs]))))
2183+
(pp/pprint (multi-arity-fn 'foo {} '(([a]) ([a [b & cs] & xs]))))
2184+
)
2185+
2186+
(def
2187+
^{:doc "Same as (def name (fn [params* ] exprs*)) or (def
2188+
name (fn ([params* ] exprs*)+)) with any doc-string or attrs added
2189+
to the var metadata. prepost-map defines a map with optional keys
2190+
:pre and :post that contain collections of pre or post conditions."
2191+
:arglists '([name doc-string? attr-map? [params*] prepost-map? body]
2192+
[name doc-string? attr-map? ([params*] prepost-map? body)+ attr-map?])}
2193+
defn (fn defn [&form &env name & fdecl]
2194+
;; Note: Cannot delegate this check to def because of the call to (with-meta name ..)
2195+
(if (core/instance? clojure.lang.Symbol name)
2196+
nil
2197+
(throw (IllegalArgumentException. "First argument to defn must be a symbol")))
2198+
(core/let [m (if (core/string? (first fdecl))
2199+
{:doc (first fdecl)}
2200+
{})
2201+
fdecl (if (core/string? (first fdecl))
2202+
(next fdecl)
2203+
fdecl)
2204+
m (if (map? (first fdecl))
2205+
(conj m (first fdecl))
2206+
m)
2207+
fdecl (if (map? (first fdecl))
2208+
(next fdecl)
2209+
fdecl)
2210+
fdecl (if (vector? (first fdecl))
2211+
(core/list fdecl)
2212+
fdecl)
2213+
m (if (map? (last fdecl))
2214+
(conj m (last fdecl))
2215+
m)
2216+
fdecl (if (map? (last fdecl))
2217+
(butlast fdecl)
2218+
fdecl)
2219+
m (conj {:arglists (core/list 'quote (sigs fdecl))} m)
2220+
m (core/let [inline (:inline m)
2221+
ifn (first inline)
2222+
iname (second inline)]
2223+
;; same as: (if (and (= 'fn ifn) (not (symbol? iname))) ...)
2224+
(if (if (clojure.lang.Util/equiv 'fn ifn)
2225+
(if (core/instance? clojure.lang.Symbol iname) false true))
2226+
;; inserts the same fn name to the inline fn if it does not have one
2227+
(assoc m :inline (cons ifn (cons (clojure.lang.Symbol/intern (.concat (.getName ^clojure.lang.Symbol name) "__inliner"))
2228+
(next inline))))
2229+
m))
2230+
m (conj (if (meta name) (meta name) {}) m)]
2231+
(cond
2232+
(multi-arity-fn? fdecl)
2233+
(multi-arity-fn name m fdecl)
2234+
2235+
(variadic-fn? fdecl)
2236+
(variadic-fn name m fdecl)
2237+
2238+
:else
2239+
(core/list 'def (with-meta name m)
2240+
;;todo - restore propagation of fn name
2241+
;;must figure out how to convey primitive hints to self calls first
2242+
(cons `fn fdecl))))))
2243+
2244+
(. (var defn) (setMacro))

‎test/clj/cljs/compiler_tests.clj

+13
Original file line numberDiff line numberDiff line change
@@ -52,3 +52,16 @@
5252
(comp/emit
5353
(ana/analyze (assoc aenv :context :expr) 'js/-Infinity)))
5454
"-Infinity"))
55+
56+
(comment
57+
(env/with-compiler-env cenv
58+
(comp/emit
59+
(ana/analyze aenv
60+
'(defn foo ([a]) ([a b])))))
61+
62+
(env/with-compiler-env cenv
63+
(comp/munge
64+
(comp/lazy-load?
65+
(ana/analyze aenv
66+
'(defn foo ([a]) ([a b]))))))
67+
)

0 commit comments

Comments
 (0)
Please sign in to comment.