(CLJS) 'and' does not short circuit within go block in clojurescript

Description

I have the following piece of ClojureScript code within a go block:

(cond
(and (vector? x) (= (first x) :some-key)) ...)

This generates the following piece of JavaScript code for the cond case:

if (40 === f) {
return
d = b[9],
f = cljs.core.vector_QMARK_.call(null, d),
d = cljs.core.first.call(null, d),
d = cljs.core.EQ.call(null, d, new cljs.core.Keyword(null, "some-key", "some-key", -978969032)),
cljs.core.truth_(f && d) ? b[1] = 42 : b[1] = 43,
new cljs.core.Keyword(null, "recur", "recur", -437573268);
}

This looks to me like both and arguments would actually get evaluated. As a result my code crashes whenever it hits this cond case and 'x' is not seqable.

Environment

core.async 0.1.338.0-5c5012-alpha

Activity

Show:

David Nolen May 12, 2021 at 7:46 PM

The issue is in ClojureScript

Daniel Sutton July 18, 2020 at 10:57 PM

Possible fix in cljs to not optimize the and/or. ticket: https://clojure.atlassian.net/browse/CLJS-3260

Daniel Sutton June 27, 2020 at 5:48 PM
Edited

another workaround is to split the check from the usage branch like so:

 

 

;; throws: (let [x :nope] (go (and (vector? x) (< (count x) 2)))) ;; break up the vector? check with the usage of the fact that it is a vector ;; note if :deoptimization is first it will not work (let [x :nope] (go (and (vector? x) :deoptimization (< (count x) 2)))) ;; use tag (let [x :nope] (go (and ^{:tag :not-boolean} (vector? x) (< (count x) 2))))

Daniel Sutton June 27, 2020 at 3:53 PM

its hard to read the output, but i think

 

(cljs.pprint/pprint (macroexpand '(go (and ^{:tag :dont-optimize} (vector? cards) (< 0 (count cards))))))

 

provides a workaround. The and macro checks to see if it can put all the logic into js with

(core/defn- simple-test-expr? [env ast] (core/and (#{:var :js-var :local :invoke :const :host-field :host-call :js :quote} (:op ast)) ('#{boolean seq} (cljs.analyzer/infer-tag env ast))))

 

Is a good approach to this problem figuring out a good way to prevent this optimization? If so, what would that look like?

Seem to be three options

  1. add information to env to prevent this optimization. Would require a change to cljs

  2. Walk the form to be expanded and remove boolean tags and seq tags such that the macroexpander thinks it can’t be done in javascripts notion of truthiness. Might be heavy-handed? Is the only downside just emitting a bit slower code?

  3. Prevent the boolean tag from being attached to the and forms? I’m not entirely sure

Won't Fix

Details

Assignee

Reporter

Priority

Created September 14, 2014 at 1:08 PM
Updated May 12, 2021 at 7:49 PM
Resolved May 12, 2021 at 7:46 PM

Flag notifications