code-reviews

2020-09-29T21:00:06.003500Z

(def instruction-label first)
(defn ->label+instruction [raw-instructions]
  (reduce (fn [res part]
            (if (symbol? part)
              (conj res [part nil])
              (let [last-label (instruction-label (last res))]
                (conj res [last-label part]))))
          []
          raw-instructions))

(comment
  (->label+instruction '((assign)
                         label-one
                         (test)
                         (branch)
                         label-two
                         (goto))))

=> [[nil (assign)] [label-one nil] [label-one (test)] [label-one (branch)] [label-two nil] [label-two (goto)]]
Hey team, wrote up a quick function, which given a [sym|list], transforms it like so: [[curr-sym list]] If there’s a better way to write this function, would appreciate it! (It’s used in a machine language simulator: https://github.com/nezaj/clj-sicp/blob/master/src/machine_simulator.clj#L21-L27)

seancorfield 2020-09-29T21:04:57.004500Z

Instead of using last, I would probably track the current symbol in the accumulator part.

seancorfield 2020-09-29T21:05:28.005Z

Using peek instead of last would be much faster -- last will walk the whole of res every time.

😮 1
seancorfield 2020-09-29T21:07:14.006800Z

(second (reduce (fn [[sym res] part] 
                  (if (symbol? part) 
                    [part (conj res [part nil])] 
                    [sym (conj res [sym part])])) 
                [nil []] 
                raw-instructions))

seancorfield 2020-09-29T21:08:28.007100Z

^ @stopachka

2020-09-29T21:08:48.007600Z

Great point, thanks Sean! cc @joeaverbukh

seancorfield 2020-09-29T21:08:50.007700Z

(untested but I think that'll work)

2020-09-29T21:09:34.007900Z

works great!

2020-09-29T21:14:55.010700Z

(defn label-idx [transformed-instructions label]
  (->> transformed-instructions
       (map-indexed (fn [i x] [i (instruction-label x)]))
       (filter (comp (partial = label) last))
       ffirst))

(comment
  (label-idx '[[nil (assign)] [label-one nil] [label-two nil]]
             'label-two))
; => 2
The idea of label-idx is to find the first idx where the instruction-label is = the given label Seems like a lot of work to do above — thoughts on better one much appreciated!

2020-09-29T21:25:27.011300Z

(def indexed (partial map-indexed vector))

(defn positions [filter-fn coll]
  (->> coll
       indexed
       (filter (fn [[_ v]] (filter-fn v)))
       (map first)))

(defn label-idx [label instructions]
  (first (positions (comp (partial = label) instruction-label)
                    instructions)))

(comment
  (label-idx
    'label-two
    '[[nil (assign)] [label-one nil] [label-two nil]]))
^ wrote above. I see there used to be a clojure.contrib.seq-utils, but it is now deprecated. lmk if other way to do this : }

seancorfield 2020-09-29T21:42:24.012500Z

@stopachka That's a case where SICP is somewhat hampered by only having lists as data structures. In Clojure, you'd probably use a different data structure, or a pair of data structures, to efficiently track where labels referred.

❤️ 1
seancorfield 2020-09-29T21:43:40.014500Z

The whole of clojure.contrib went away between Clojure 1.2 and 1.3. Some parts of it became new Contrib libraries, the rest of it just fell by the wayside. Here's some of that fallout: https://clojure.org/dev/contrib_history

👍 1
seancorfield 2020-09-29T21:48:29.015400Z

Regarding searching for the first index of a given label, I'd probably use reduce and return (reduced idx) when I hit a match.

❤️ 1
seancorfield 2020-09-29T21:52:42.019100Z

(defn label-idx [label instructions]
  (reduce (fn [idx [sym instr]]
            (cond (= label sym) (reduced idx) 
                  (= ::sentinel sym) (reduced -1)
                  :else
                  (inc idx)))
          0
          (conj instructions [::sentinel nil])))

seancorfield 2020-09-29T21:53:09.019700Z

Or you could use a local gensym to create a more unique sentinel item...

2020-09-29T22:09:30.020400Z

Thanks Sean! Makes sense — will play with keeping an auxiliary data struct for labels — something like label->idx! Makes sense

2020-09-29T22:19:47.021500Z

If you get time, thoughts on:

(defn extract-labels [raw-instructions]
  (rest (reduce
          (fn [[idx label->idx instructions] part]
            (if (symbol? part)
              [idx
               (assoc label->idx part (inc idx))
               instructions]
              [(inc idx)
               label->idx
               (conj instructions part)]))
          [-1 {} []]
          raw-instructions)))

(comment
  (let [[label->idx instructions] (extract-labels '((assign foo (const 1))
                                                    label-one
                                                    (test (op =) (const 1) (reg foo))
                                                    (branch (label label-two))
                                                    label-two
                                                    (assign bar (const 2))))]
    (println label->idx)
    (println (map-indexed vector instructions))))
extract-labels now returns two things:
a. a mapping of label->idx 
b. a cleaned up list of instructions
Full code: https://github.com/nezaj/clj-sicp/blob/master/src/machine_simulator.clj#L11-L65

seancorfield 2020-09-29T23:22:44.025200Z

Yup, that's probably a good way to deal with it. I suspect there's a more optimized way that could rely on the shared nature of persistent data structures... Probably a single pass that produces an actual list of instructions (not a vector) such that you could have your map from labels to actual code so you could naturally iterate through the instructions steps and have the label lookup return the sequence of instructions to continue executing. You'd probably need to construct that via lazy-seq to avoid blowing the stack since you'd essentially process it recursively.

❤️ 1
2020-09-29T23:46:12.025900Z

Ah, very interesting! Thanks Sean