aboutsummaryrefslogtreecommitdiffstats
path: root/year2021
diff options
context:
space:
mode:
authorClyne Sullivan <clyne@bitgloo.com>2022-11-30 19:55:31 -0500
committerClyne Sullivan <clyne@bitgloo.com>2022-11-30 19:55:31 -0500
commit8d43e37df99f280377bed90284d6ac2428334804 (patch)
tree3a5042c9af29da52b4bac38fd78b3ccde77a1dbc /year2021
parent66ed0b9d27850dc653abc8baa75884f3de311bfa (diff)
move 2021 days to folder; update README
Diffstat (limited to 'year2021')
-rw-r--r--year2021/day1/part1.clj15
-rw-r--r--year2021/day1/part2.clj15
-rw-r--r--year2021/day10/part1.clj22
-rw-r--r--year2021/day10/part2.clj24
-rw-r--r--year2021/day10/partboth.clj34
-rw-r--r--year2021/day11/part1.clj41
-rw-r--r--year2021/day11/part2.clj42
-rw-r--r--year2021/day12/part1.clj27
-rw-r--r--year2021/day12/part2.clj50
-rw-r--r--year2021/day13/part1.clj30
-rw-r--r--year2021/day13/part2.clj33
-rw-r--r--year2021/day13/partboth.clj49
-rw-r--r--year2021/day14/part1.clj27
-rw-r--r--year2021/day14/part2.clj38
-rw-r--r--year2021/day15/part1.clj30
-rw-r--r--year2021/day15/part2.clj41
-rw-r--r--year2021/day16/part1.clj67
-rw-r--r--year2021/day16/part2.cpp131
-rw-r--r--year2021/day17/part3.clj59
-rw-r--r--year2021/day17/partboth.clj54
-rw-r--r--year2021/day18/partboth.clj95
-rw-r--r--year2021/day19/partboth.clj98
-rw-r--r--year2021/day2/part1.clj26
-rw-r--r--year2021/day2/part2.clj27
-rw-r--r--year2021/day20/partboth.clj43
-rw-r--r--year2021/day21/part1.clj22
-rw-r--r--year2021/day21/part2.clj37
-rw-r--r--year2021/day22/part1.clj25
-rw-r--r--year2021/day22/part2.clj60
-rw-r--r--year2021/day23/part1-done-by-hand0
-rw-r--r--year2021/day23/part2.clj90
-rw-r--r--year2021/day24/checker.clj47
-rw-r--r--year2021/day24/partboth.clj41
-rw-r--r--year2021/day24/solved-by-hand-with-help.md4
-rw-r--r--year2021/day25/part1.clj40
-rw-r--r--year2021/day3/part1.clj25
-rw-r--r--year2021/day3/part2.clj36
-rw-r--r--year2021/day4/part1.clj70
-rw-r--r--year2021/day4/part2.clj70
-rw-r--r--year2021/day5/part1.clj75
-rw-r--r--year2021/day5/part2.clj83
-rw-r--r--year2021/day6/consteval.cpp32
-rw-r--r--year2021/day6/part1.clj20
-rw-r--r--year2021/day6/part2.clj25
-rw-r--r--year2021/day7/maxima.mac10
-rw-r--r--year2021/day7/part1.clj18
-rw-r--r--year2021/day7/part2.clj20
-rw-r--r--year2021/day8/part1.clj15
-rw-r--r--year2021/day8/part2.clj66
-rw-r--r--year2021/day9/part1.clj21
-rw-r--r--year2021/day9/part2.clj45
51 files changed, 2115 insertions, 0 deletions
diff --git a/year2021/day1/part1.clj b/year2021/day1/part1.clj
new file mode 100644
index 0000000..1c2c756
--- /dev/null
+++ b/year2021/day1/part1.clj
@@ -0,0 +1,15 @@
+; Day 1, part 1
+; Read a list of numbers from stdin, separated by newlines.
+; Count occurances of the current number being greater than
+; the previous.
+;
+
+(as-> (slurp "./in") $
+ (clojure.string/split-lines $)
+ (map read-string $)
+ (reduce
+ #(cond-> (assoc %1 0 %2) (> %2 (first %1)) (update 1 inc))
+ [(first $) 0]
+ (rest $))
+ (println (second $)))
+
diff --git a/year2021/day1/part2.clj b/year2021/day1/part2.clj
new file mode 100644
index 0000000..e496c7d
--- /dev/null
+++ b/year2021/day1/part2.clj
@@ -0,0 +1,15 @@
+; Day 1, part 2
+; Read a list of numbers from stdin, separated by newlines.
+; Count occurances of the current sum being greater than
+; the previous, where a sum is that of the current number,
+; the previous number, and the next number.
+;
+
+(let [input (->> (slurp "./in")
+ clojure.string/split-lines
+ (mapv read-string))]
+ (println
+ (count
+ (filter #(< (get input %) (get input (+ % 3)))
+ (range 0 (- (count input) 3))))))
+
diff --git a/year2021/day10/part1.clj b/year2021/day10/part1.clj
new file mode 100644
index 0000000..60ed1e6
--- /dev/null
+++ b/year2021/day10/part1.clj
@@ -0,0 +1,22 @@
+(def to-closing {\{ \} \( \) \[ \] \< \>})
+(def to-score {\) 3 \] 57 \} 1197 \> 25137})
+
+(defn check-line [input]
+ (loop [in input open '()]
+ (when-let [c (first in)]
+ (if (some->> (#{\} \) \] \>} c) (not= (first open)))
+ c
+ (recur
+ (rest in)
+ (if-let [op (to-closing c)]
+ (conj open op)
+ (rest open)
+ ))))))
+
+(->> (slurp "./in")
+ (clojure.string/split-lines)
+ (map (comp to-score check-line vec))
+ (filter some?)
+ (apply +)
+ (println))
+
diff --git a/year2021/day10/part2.clj b/year2021/day10/part2.clj
new file mode 100644
index 0000000..4dfaf3b
--- /dev/null
+++ b/year2021/day10/part2.clj
@@ -0,0 +1,24 @@
+(def to-closing {\{ \} \( \) \[ \] \< \>})
+(def to-score {\) 1 \] 2 \} 3 \> 4})
+
+(defn check-line [input]
+ (loop [in input open '()]
+ (if-let [c (first in)]
+ (when (or (nil? (#{\} \) \] \>} c)) (= (first open) c))
+ (recur
+ (rest in)
+ (if-let [op (to-closing c)]
+ (conj open op)
+ (rest open))))
+ open
+ )))
+
+(->> (slurp "./in")
+ (clojure.string/split-lines)
+ (map (comp check-line vec))
+ (filter some?)
+ (map (partial reduce #(+ (* 5 %1) (to-score %2)) 0))
+ (sort)
+ (#(nth % (quot (count %) 2)))
+ (println))
+
diff --git a/year2021/day10/partboth.clj b/year2021/day10/partboth.clj
new file mode 100644
index 0000000..78080d9
--- /dev/null
+++ b/year2021/day10/partboth.clj
@@ -0,0 +1,34 @@
+(def to-closing {\{ \} \( \) \[ \] \< \>})
+(def to-score {\) 3 \] 57 \} 1197 \> 25137})
+(def to-score2 {\) 1 \] 2 \} 3 \> 4})
+
+(defn check-line [input]
+ (loop [in input open '()]
+ (if-let [c (first in)]
+ (if (some->> (#{\} \) \] \>} c) (not= (first open)))
+ c
+ (recur
+ (rest in)
+ (if-let [op (to-closing c)]
+ (conj open op)
+ (rest open)
+ )))
+ open
+ )))
+
+(->> (slurp "./in")
+ (clojure.string/split-lines)
+ (map (comp check-line vec))
+ ; check-line returns first invalid character, or list of characters
+ ; necessary to close the line. Work through these through `reduce`
+ ; and build the answers for both parts:
+ (reduce
+ (fn [tots nxt]
+ (if (seq? nxt)
+ (update tots 1 #(conj % (reduce (fn [a b] (+ (* 5 a) (to-score2 b))) 0 nxt)))
+ (update tots 0 #(+ % (to-score nxt)))))
+ [0 '()])
+ ; Get part 2 answer from the list of scores:
+ (#(update % 1 (fn [ns] (nth ns (quot (count ns) 2)))))
+ (println))
+
diff --git a/year2021/day11/part1.clj b/year2021/day11/part1.clj
new file mode 100644
index 0000000..5113fd4
--- /dev/null
+++ b/year2021/day11/part1.clj
@@ -0,0 +1,41 @@
+(defn get-adj [in y x]
+ (filter
+ #(some? (get-in in %))
+ [[(dec y) (dec x)] [(dec y) x] [(dec y) (inc x)]
+ [y (dec x)] [y (inc x)]
+ [(inc y) (dec x)] [(inc y) x] [(inc y) (inc x)]]
+ ))
+
+(defn apply-incs [in] (mapv (partial mapv inc) in))
+
+(defn next-step [indata]
+ (loop [in (apply-incs (indata :grid)) bl (indata :blinks) y 0 x 0]
+ (cond
+ (> (get-in in [y x]) 9)
+ (do
+ (recur
+ (-> (reduce
+ (fn [i n] (update-in i n #(cond-> % (pos? %) inc)))
+ in
+ (get-adj in y x))
+ (update-in [y x] #(do % 0)))
+ (inc bl)
+ 0 0))
+ (< x (dec (count (first in))))
+ (recur in bl y (inc x))
+ (< y (dec (count in)))
+ (recur in bl (inc y) 0)
+ :else
+ {:grid in :blinks bl}
+ )
+ )
+ )
+
+(->> (slurp "./in")
+ (clojure.string/split-lines)
+ (map (partial map #(- (int %) 48)))
+ (assoc {} :blinks 0 :grid)
+ (iterate next-step)
+ (#(nth % 100))
+ (#(println (% :blinks))))
+
diff --git a/year2021/day11/part2.clj b/year2021/day11/part2.clj
new file mode 100644
index 0000000..4c4663c
--- /dev/null
+++ b/year2021/day11/part2.clj
@@ -0,0 +1,42 @@
+(defn get-adj [in y x]
+ (filter
+ #(some? (get-in in %))
+ [[(dec y) (dec x)] [(dec y) x] [(dec y) (inc x)]
+ [y (dec x)] [y (inc x)]
+ [(inc y) (dec x)] [(inc y) x] [(inc y) (inc x)]]
+ ))
+
+(defn apply-incs [in] (mapv (partial mapv inc) in))
+
+(defn next-step [indata]
+ (loop [in (apply-incs (indata :grid)) bl (indata :blinks) y 0 x 0]
+ (cond
+ (> (get-in in [y x]) 9)
+ (do
+ (recur
+ (-> (reduce
+ (fn [i n] (update-in i n #(cond-> % (pos? %) inc)))
+ in
+ (get-adj in y x))
+ (update-in [y x] #(do % 0)))
+ (inc bl)
+ 0 0))
+ (< x (dec (count (first in))))
+ (recur in bl y (inc x))
+ (< y (dec (count in)))
+ (recur in bl (inc y) 0)
+ :else
+ {:grid in :blinks bl}
+ )
+ )
+ )
+
+(->> (slurp "./in")
+ (clojure.string/split-lines)
+ (map (partial map #(- (int %) 48)))
+ (assoc {} :blinks 0 :grid)
+ (iterate next-step)
+ (take-while #(pos? (apply + (flatten (% :grid)))))
+ (count)
+ (println))
+
diff --git a/year2021/day12/part1.clj b/year2021/day12/part1.clj
new file mode 100644
index 0000000..cdd4a43
--- /dev/null
+++ b/year2021/day12/part1.clj
@@ -0,0 +1,27 @@
+(require '[clojure.string :as str])
+
+(def caves (->> (slurp "./in")
+ (str/split-lines)
+ (map #(str/split % #"-"))
+ (map (partial map str))
+ (#(concat % (map reverse %)))
+ ))
+
+(defn get-caves-forward [klst]
+ (map #(cons (second %) klst)
+ (filter
+ #(or (< (int (first (second %))) 96)
+ (not-any? (partial = (second %)) klst))
+ (filter #(= (first %) (first klst)) caves)
+ )))
+
+(loop [lst (get-caves-forward ["start"]) ms '()]
+ (let [nxt (->> lst
+ (map get-caves-forward)
+ (apply concat))
+ mtchs (concat ms (filter #(= (first %) "end") nxt))
+ nxtlst (filter #(not= (first %) "end") nxt)]
+ (if (empty? nxtlst)
+ (println (count mtchs))
+ (recur nxtlst mtchs))))
+
diff --git a/year2021/day12/part2.clj b/year2021/day12/part2.clj
new file mode 100644
index 0000000..06b2bcd
--- /dev/null
+++ b/year2021/day12/part2.clj
@@ -0,0 +1,50 @@
+(require '[clojure.string :as str])
+(require '[clojure.core.reducers :as r])
+
+; Build list of cave connections:
+(def caves (->> (slurp "./in")
+ (str/split-lines)
+ (map #(str/split % #"-"))
+ (map (partial map str))
+ (#(concat % (map reverse %)))
+ ))
+
+; Try to help execution speed by memoizing cave searches:
+(def search-caves
+ (memoize (fn [id]
+ (filter
+ #(and (= (first %) id)
+ (not (str/starts-with? (second %) "start")))
+ caves
+ ))))
+
+; Given current path 'path', return a list of valid paths that branch
+; from 'path'. Note, paths are stored in reverse order.
+(defn get-caves-forward [path]
+ (r/map #(cons (second %) path)
+ (r/filter
+ (fn [cv]
+ (or
+ ; Always allow uppercase paths
+ (< (int (first (second cv))) 96)
+ ; Or, allow lowercase paths that have never been visited
+ (not-any? #(= % (second cv)) path)
+ ; Or, allow one duplicate if there are none yet
+ (apply distinct? (filter #(= (str/lower-case %) %) path))))
+ ; Only work with paths that we connect to
+ (search-caves (first path))
+ )))
+
+(loop [paths (into [] (get-caves-forward ["start"])) complete-paths 0]
+ (let [results (->> paths
+ (r/mapcat get-caves-forward)
+ (r/foldcat)
+ (r/reduce
+ (fn [r b] (if (not= (first b) "end")
+ (update r :next-paths #(conj % b))
+ (update r :complete-counts inc)))
+ {:next-paths [] :complete-counts complete-paths}))]
+ (println (results :complete-counts))
+ (when-not (empty? (results :next-paths))
+ (recur (results :next-paths) (results :complete-counts)))))
+
diff --git a/year2021/day13/part1.clj b/year2021/day13/part1.clj
new file mode 100644
index 0000000..c3f08d4
--- /dev/null
+++ b/year2021/day13/part1.clj
@@ -0,0 +1,30 @@
+(require '[clojure.string :as str])
+
+(def input (->> (slurp "./in")
+ (str/split-lines)
+ (split-with not-empty)
+ ((juxt
+ (fn [lst]
+ (reduce
+ #(conj %1 (mapv read-string (str/split %2 #",")))
+ #{} (first lst)))
+ (fn [lst]
+ (->> (second lst)
+ (drop 1)
+ (map #(-> %
+ (str/split #"=")
+ (update 0 (comp {\x 0 \y 1} last))
+ (update 1 read-string)))
+ (first)
+ ))))))
+
+(defn fold-point [idx chg pt]
+ (cond-> pt (> (get pt idx) chg) (update idx #(- % (* 2 (- % chg))))))
+
+(let [instruction (second input)]
+ (->> (first input)
+ (map (partial fold-point (first instruction) (second instruction)))
+ (distinct)
+ (count)
+ (println)))
+
diff --git a/year2021/day13/part2.clj b/year2021/day13/part2.clj
new file mode 100644
index 0000000..6db6555
--- /dev/null
+++ b/year2021/day13/part2.clj
@@ -0,0 +1,33 @@
+(require '[clojure.string :as str])
+
+(def input (->> (slurp "./in")
+ (str/split-lines)
+ (split-with not-empty)
+ ((juxt
+ (fn [lst]
+ (reduce
+ #(conj %1 (mapv read-string (str/split %2 #",")))
+ #{} (first lst)))
+ (fn [lst]
+ (->> (rest (second lst))
+ (map #(-> %
+ (str/split #"=")
+ (update 0 (comp {\x 0 \y 1} last))
+ (update 1 read-string)))
+ ))))))
+
+(defn print-grid [pts]
+ (doseq [p pts] (println (str/join [(first p) "," (second p)]))))
+
+(defn fold-point [idx chg pt]
+ (cond-> pt (> (get pt idx) chg) (update idx #(- % (* 2 (- % chg))))))
+
+(print-grid
+ (apply
+ (partial
+ reduce
+ #(let [ins (first %2)]
+ (set (map (partial fold-point (first %2) (second %2)) %1))))
+ input
+ ))
+
diff --git a/year2021/day13/partboth.clj b/year2021/day13/partboth.clj
new file mode 100644
index 0000000..c24980d
--- /dev/null
+++ b/year2021/day13/partboth.clj
@@ -0,0 +1,49 @@
+(require '[clojure.string :as str])
+
+(def input (->> (slurp "./in")
+ (str/split-lines)
+ (split-with not-empty)
+ ((juxt
+ (fn [lst]
+ (reduce
+ #(conj %1 (mapv read-string (str/split %2 #",")))
+ #{} (first lst)))
+ (fn [lst]
+ (->> (second lst)
+ (drop 1)
+ (map #(-> %
+ (str/split #"=")
+ (update 0 (comp {\x 0 \y 1} last))
+ (update 1 read-string)))
+ ))))))
+
+(defn fold-point [idx chg pt]
+ (cond-> pt (> (get pt idx) chg) (update idx #(- % (* 2 (- % chg))))))
+
+(defn print-grid [pts]
+ (let [xmax (inc (apply max (map first pts)))
+ ymax (inc (apply max (map second pts)))]
+ (doseq [y (range 0 ymax)]
+ (println
+ (str/join
+ (for [x (range 0 xmax)] (if (contains? pts [x y]) \@ \ ))
+ )))))
+
+; Part 1
+(let [instruction (first (second input))]
+ (->> (first input)
+ (map (partial fold-point (first instruction) (second instruction)))
+ (distinct)
+ (count)
+ (println)))
+
+; Part 2
+(print-grid
+ (apply
+ (partial
+ reduce
+ #(let [ins (first %2)]
+ (set (map (partial fold-point (first %2) (second %2)) %1))))
+ input
+ ))
+
diff --git a/year2021/day14/part1.clj b/year2021/day14/part1.clj
new file mode 100644
index 0000000..de03bda
--- /dev/null
+++ b/year2021/day14/part1.clj
@@ -0,0 +1,27 @@
+(require '[clojure.string :as str])
+
+(def input (->> (slurp "./in")
+ str/split-lines
+ ((juxt
+ first
+ (fn [lines]
+ (->> lines
+ (drop 2)
+ (map #(str/split % #" -> "))
+ (flatten)
+ (apply (partial assoc {}))
+ ))))))
+
+(defn grow-polymer [polymer insertion-rules]
+ (str/join
+ (cons
+ (first polymer)
+ (mapcat (juxt insertion-rules second)
+ (for [i (range 0 (dec (count polymer)))]
+ (subs polymer i (+ i 2)))))))
+
+(def growth-seq (iterate #(grow-polymer % (second input)) (first input)))
+
+(let [freqs (vals (frequencies (nth growth-seq 10)))]
+ (println (- (apply max freqs) (apply min freqs))))
+
diff --git a/year2021/day14/part2.clj b/year2021/day14/part2.clj
new file mode 100644
index 0000000..f5de5fe
--- /dev/null
+++ b/year2021/day14/part2.clj
@@ -0,0 +1,38 @@
+(require '[clojure.string :as str])
+
+(def input (->> (slurp "./in")
+ str/split-lines
+ ((juxt
+ #(let [init-polymer (first %)]
+ (for [i (range 0 (dec (count init-polymer)))]
+ (subs init-polymer i (+ i 2))))
+ (fn [lines]
+ (->> lines
+ (drop 2)
+ (map #(str/split % #" -> "))
+ (map (fn [[[a b] c]]
+ {(str/join [a b]) (map str/join [[a c] [c b]])}))
+ (reduce into)
+ ))))))
+
+(def blank-map (zipmap (keys (second input)) (repeat 0)))
+
+(defn grow-polymer [polymer insertion-rules]
+ (reduce
+ #(let [[p1 p2] (insertion-rules (key %2)) v (val %2)]
+ (-> %1 (update p1 + v) (update p2 + v)))
+ blank-map
+ (filter (comp pos? val) polymer)))
+
+(def growth-seq
+ (iterate #(grow-polymer % (second input))
+ (reduce #(update %1 %2 inc) blank-map (first input))))
+
+(let [final-polymer (nth growth-seq 40)
+ letter-counts (reduce
+ (fn [r [k v]] (-> r (update (first k) + v) (update (second k) + v)))
+ (zipmap (str/join (keys final-polymer)) (repeat 0))
+ final-polymer)
+ unique-counts (filter pos? (map #(Math/ceil (/ (val %) 2)) letter-counts))]
+ (println (- (apply max unique-counts) (apply min unique-counts))))
+
diff --git a/year2021/day15/part1.clj b/year2021/day15/part1.clj
new file mode 100644
index 0000000..7838e34
--- /dev/null
+++ b/year2021/day15/part1.clj
@@ -0,0 +1,30 @@
+(def input (->> (slurp "./in")
+ (clojure.string/split-lines)
+ (mapv (partial mapv (comp read-string str)))
+ (#(for [y (range 0 (count %)) x (range 0 (count (first %)))]
+ {[y x] (get-in % [y x])}))
+ (into {})))
+
+(def dim (apply max (map first (keys input))))
+
+(defn find-neighbors [u]
+ (filter #(contains? input %) [(update u 0 inc)
+ (update u 0 dec)
+ (update u 1 inc)
+ (update u 1 dec)]))
+
+(loop [dist (zipmap (keys input) (repeat ##Inf))
+ Q {[0 0] 0}]
+ (if (empty? Q)
+ (println (dist [dim dim]))
+ (let [[u distu] (first Q)
+ NN (reduce
+ #(let [dv (+ distu (input %2))]
+ (cond-> %1
+ (> ((first %1) %2) dv)
+ (-> (update 0 assoc %2 dv)
+ (update 1 assoc %2 dv))))
+ [dist {}]
+ (find-neighbors u))]
+ (recur (first NN) (sort-by val < (into (rest Q) (second NN)))))))
+
diff --git a/year2021/day15/part2.clj b/year2021/day15/part2.clj
new file mode 100644
index 0000000..a422ed0
--- /dev/null
+++ b/year2021/day15/part2.clj
@@ -0,0 +1,41 @@
+(def input (->> (slurp "./in")
+ (clojure.string/split-lines)
+ (mapv (partial mapv (comp read-string str)))
+ (#(for [y (range 0 (count %)) x (range 0 (count (first %)))]
+ {[y x] (get-in % [y x])}))
+ (into {})
+ ((fn [lst]
+ (reduce
+ #(into %1
+ (for [j (range 0 5) i (range 0 5)]
+ {[(+ (first (key %2)) (* 100 j)) (+ (second (key %2)) (* 100 i))]
+ (let [s (+ i j (val %2))] (if (> s 9) (- s 9) s))}))
+ {}
+ lst
+ )))
+ (into {})
+ ))
+
+(def dim (apply max (map first (keys input))))
+
+(defn find-neighbors [u]
+ (filter #(contains? input %) [(update u 0 inc)
+ (update u 0 dec)
+ (update u 1 inc)
+ (update u 1 dec)]))
+
+(loop [dist (zipmap (keys input) (repeat ##Inf))
+ Q {[0 0] 0}]
+ (if (empty? Q)
+ (println (dist [dim dim]))
+ (let [[u distu] (first Q)
+ NN (reduce
+ #(let [dv (+ distu (input %2))]
+ (cond-> %1
+ (> ((first %1) %2) dv)
+ (-> (update 0 assoc %2 dv)
+ (update 1 assoc %2 dv))))
+ [dist {}]
+ (find-neighbors u))]
+ (recur (first NN) (sort-by val < (into (rest Q) (second NN)))))))
+
diff --git a/year2021/day16/part1.clj b/year2021/day16/part1.clj
new file mode 100644
index 0000000..2358e59
--- /dev/null
+++ b/year2021/day16/part1.clj
@@ -0,0 +1,67 @@
+(require '[clojure.string :as str])
+(require '[clojure.set :as set])
+
+(def hex-lookup
+ {\0 "0000" \1 "0001" \2 "0010" \3 "0011"
+ \4 "0100" \5 "0101" \6 "0110" \7 "0111"
+ \8 "1000" \9 "1001" \A "1010" \B "1011"
+ \C "1100" \D "1101" \E "1110" \F "1111"})
+
+(defn binary-to-num [binstr]
+ (reduce
+ #(cond-> %1 true (* 2) (= \1 %2) inc)
+ 0
+ binstr))
+
+(defn take-packet-version [packet]
+ (-> (split-at 3 packet)
+ (update 0 binary-to-num)))
+
+(defn take-literal-value [packet]
+ (let [[tid data] (split-at 3 packet)]
+ (when (= '(\1 \0 \0) tid)
+ (let [sd (into [] data)
+ spl (split-with #(= \1 (first %)) (partition 5 sd))
+ nums (map rest (concat (first spl) (take 1 (second spl))))]
+ [(binary-to-num (reduce concat nums))
+ (drop (* 5 (inc (count (first spl)))) sd)]
+ ))))
+
+(defn take-operator [packet]
+ (let [id (first packet) ps (str/join (rest packet))]
+ (if (= \0 id)
+ (let [bits (binary-to-num (subs ps 0 15))]
+ [(subs ps 15) bits :bits])
+ (let [pkts (binary-to-num (subs ps 0 11))]
+ [(subs ps 11) pkts :pkts]))))
+
+(defn process-packet [[input vcnt]]
+ (let [[version data] (take-packet-version input)]
+ (if (every? #(= \0 %) data)
+ ['() vcnt]
+ (do
+ (if-let [value (take-literal-value data)]
+ [(second value) (+ vcnt version)]
+ (do
+ (let [info (take-operator (drop 3 data))]
+ (if (= :bits (get info 2))
+ (loop [inpt [(subs (first info) 0 (get info 1)) (+ vcnt version)]]
+ (if (empty? (first inpt))
+ [(subs (first info) (get info 1)) (second inpt)]
+ (recur (process-packet inpt))))
+ (loop [n (get info 1) inpt [(get info 0) (+ vcnt version)]]
+ (if (pos? n)
+ (recur (dec n) (process-packet inpt))
+ inpt))))))))))
+
+(loop [packet (->> (slurp "./in")
+ (map hex-lookup)
+ (take-while some?)
+ (reduce concat)
+ (str/join)
+ (#(vector % 0))
+ (process-packet))]
+ (if (not (empty? (first packet)))
+ (recur (process-packet packet))
+ (println (second packet))))
+
diff --git a/year2021/day16/part2.cpp b/year2021/day16/part2.cpp
new file mode 100644
index 0000000..5a02df9
--- /dev/null
+++ b/year2021/day16/part2.cpp
@@ -0,0 +1,131 @@
+#include <algorithm>
+#include <cstdint>
+#include <iostream>
+#include <list>
+#include <map>
+#include <string>
+#include <string_view>
+#include <tuple>
+
+static const std::map<char, const char *> hexToBin = {
+ {'0', "0000"}, {'1', "0001"}, {'2', "0010"}, {'3', "0011"},
+ {'4', "0100"}, {'5', "0101"}, {'6', "0110"}, {'7', "0111"},
+ {'8', "1000"}, {'9', "1001"}, {'A', "1010"}, {'B', "1011"},
+ {'C', "1100"}, {'D', "1101"}, {'E', "1110"}, {'F', "1111"}
+};
+
+static std::pair<uint64_t, std::string_view> solve(std::string_view packet);
+
+int main(int argc, const char *argv[])
+{
+ if (argc != 2)
+ return -1;
+
+ std::string packetBin;
+ std::string packetHex (argv[1]);
+ for (char c : packetHex)
+ packetBin += hexToBin.at(c);
+
+ std::cout << solve(packetBin).first << std::endl;
+
+ return 0;
+}
+
+uint64_t binStrToInt(std::string_view bin)
+{
+ uint64_t n = 0;
+
+ for (char c : bin) {
+ n = n * 2ull;
+ if (c == '1')
+ ++n;
+ }
+
+ return n;
+}
+
+bool isEmptyPacket(std::string_view packet)
+{
+ return packet.empty() ||
+ std::all_of(packet.cbegin(), packet.cend(),
+ [](char c) { return c == '0'; });
+}
+
+std::pair<uint64_t, std::string_view> solve(std::string_view packet)
+{
+ // Remove version
+ packet = packet.substr(3);
+ // Pull type ID
+ const auto typeId = packet.substr(0, 3);
+ packet = packet.substr(3);
+
+ if (typeId == "100") {
+ std::string numberStr;
+ while (1) {
+ const auto chunk = packet.substr(0, 5);
+ packet = packet.substr(5);
+ numberStr += chunk.substr(1);
+ if (chunk.front() == '0')
+ break;
+ }
+
+ return {binStrToInt(numberStr), isEmptyPacket(packet) ? "" : packet};
+ } else {
+ const bool isLengthBits = packet.front() == '0';
+ packet = packet.substr(1);
+ const auto lengthStr = packet.substr(0, isLengthBits ? 15 : 11);
+ const auto length = binStrToInt(lengthStr);
+ packet = packet.substr(isLengthBits ? 15 : 11);
+
+ std::list<uint64_t> args;
+ if (isLengthBits) {
+ auto rem = packet.substr(0, length);
+ while (1) {
+ const auto ret = solve(rem);
+ args.emplace_front(ret.first);
+ if (ret.second.empty() || isEmptyPacket(ret.second))
+ break;
+ rem = ret.second;
+ }
+ packet = packet.substr(length);
+ } else {
+ for (uint64_t i = 0; i < length; ++i) {
+ const auto ret = solve(packet);
+ args.emplace_front(ret.first);
+ packet = ret.second;
+ if (isEmptyPacket(packet))
+ break;
+ }
+ }
+
+ uint64_t result;
+ if (typeId == "000") {
+ result = 0;
+ for (auto a : args)
+ result += a;
+ } else if (typeId == "001") {
+ result = 1;
+ for (auto a : args)
+ result *= a;
+ } else if (typeId == "010") {
+ result = *std::min_element(args.cbegin(), args.cend());
+ } else if (typeId == "011") {
+ result = *std::max_element(args.cbegin(), args.cend());
+ } else if (typeId == "101") {
+ const auto a1 = args.back();
+ args.pop_back();
+ result = args.back() < a1;
+ } else if (typeId == "110") {
+ const auto a1 = args.back();
+ args.pop_back();
+ result = args.back() > a1;
+ } else if (typeId == "111") {
+ const auto a1 = args.back();
+ args.pop_back();
+ result = args.back() == a1;
+ }
+
+ return {result, isEmptyPacket(packet) ? "" : packet};
+ }
+}
+
diff --git a/year2021/day17/part3.clj b/year2021/day17/part3.clj
new file mode 100644
index 0000000..ae51260
--- /dev/null
+++ b/year2021/day17/part3.clj
@@ -0,0 +1,59 @@
+;
+;
+; Put into a lein project so that you can pass Java more RAM (e.g. -Xmx15G)
+; https://www.reddit.com/r/adventofcode/comments/riqtwx/2021_day_17_day_17_part_3/
+;
+;
+
+(require '[clojure.core.reducers :as r])
+
+(def target-area [[20000 -5000] [30000 -10000]])
+(def initial-pos [0 0])
+
+(defn beyond-target? [[[tbx tby] [tex tey]] [x y]]
+ (or (> x tex) (< y tey)))
+
+(defn within-target? [[[tbx tby] [tex tey]] [x y]]
+ (and (>= x tbx) (<= x tex) (<= y tby) (>= y tey)))
+
+(defn apply-velocity [[[px py] [vx vy]]]
+ [[(+ px vx) (+ py vy)] [(cond-> vx (pos? vx) dec) (dec vy)]])
+
+(defn take-last-while [pred coll]
+ (loop [v (first coll) r (rest coll)]
+ (if (not (pred (first r)))
+ v
+ (recur (first r) (rest r)))))
+
+(defn build-path [target vel]
+ (->> (iterate apply-velocity [initial-pos vel])
+ (take-last-while (comp not (partial beyond-target? target) first))
+ (first)))
+
+; Used to determine best x velocity for highest path
+(def tnum-seq (iterate #(do [(apply + %) (inc (second %))]) [0 1]))
+
+(let [[tb te] target-area
+ lowest-x (second (last (take-while #(< (first %) (first tb)) tnum-seq)))
+ highest-y (dec (Math/abs (second te)))]
+ (->> (for [y (range (second te) (inc highest-y))
+ x (range lowest-x (inc (first te)))] [x y])
+ (#(do (println "Generated xy pairs...") %))
+ (#(do (println "Total: " (* (- highest-y (second te)) (- (inc highest-y) lowest-x))) %))
+ (partition 1000000)
+ (#(do (println "Prepared partitions...") %))
+ (reduce
+ (fn [sum nlst]
+ (println sum)
+ (+ sum
+ (r/fold +
+ (r/monoid
+ (fn [tot xy]
+ (cond-> tot
+ (within-target? target-area (build-path target-area xy))
+ inc))
+ (constantly 0))
+ (into [] nlst))))
+ 0)
+ (println)))
+
diff --git a/year2021/day17/partboth.clj b/year2021/day17/partboth.clj
new file mode 100644
index 0000000..c585bf9
--- /dev/null
+++ b/year2021/day17/partboth.clj
@@ -0,0 +1,54 @@
+(require '[clojure.core.reducers :as r])
+
+;(def target-area [[169 -68] [206 -108]])
+(def target-area [[20000 -5000] [30000 -10000]])
+;(def target-area [[2000 -500] [3000 -1000]])
+(def initial-pos [0 0])
+
+(defn beyond-target? [[[tbx tby] [tex tey]] [x y]]
+ (or (> x tex) (< y tey)))
+
+(defn within-target? [[[tbx tby] [tex tey]] [x y]]
+ (and (>= x tbx) (<= x tex) (<= y tby) (>= y tey)))
+
+(defn apply-velocity [[[px py] [vx vy]]]
+ [[(+ px vx) (+ py vy)] [(cond-> vx (pos? vx) dec) (dec vy)]])
+
+(defn take-last-while [pred coll]
+ (loop [v (first coll) r (rest coll)]
+ (if (not (pred (first r)))
+ v
+ (recur (first r) (rest r)))))
+
+(defn build-path [target vel]
+ (->> (iterate apply-velocity [initial-pos vel])
+ (take-last-while (comp not (partial beyond-target? target) first))
+ (first)))
+
+; Used to determine best x velocity for highest path
+(def tnum-seq (iterate #(do [(apply + %) (inc (second %))]) [0 1]))
+
+(let [[tb te] target-area
+ lowest-x (second (last (take-while #(< (first %) (first tb)) tnum-seq)))
+ highest-y (dec (Math/abs (second te)))]
+ (->> (for [y (range (second te) (inc highest-y))
+ x (range lowest-x (inc (first te)))] [x y])
+ (#(do (println "Generated xy pairs...") %))
+ (#(do (println "Total: " (* (- highest-y (second te)) (- (inc highest-y) lowest-x))) %))
+ (partition 50000)
+ (#(do (println "Prepared partitions...") %))
+ (reduce
+ (fn [sum nlst]
+ (println sum)
+ (+ sum
+ (r/fold +
+ (r/monoid
+ (fn [tot xy]
+ (cond-> tot
+ (within-target? target-area (build-path target-area xy))
+ inc))
+ (constantly 0))
+ (into [] nlst))))
+ 0)
+ (println)))
+
diff --git a/year2021/day18/partboth.clj b/year2021/day18/partboth.clj
new file mode 100644
index 0000000..83fc42c
--- /dev/null
+++ b/year2021/day18/partboth.clj
@@ -0,0 +1,95 @@
+(require '[clojure.core.reducers :as r])
+(require '[clojure.set :as set])
+(require '[clojure.zip :as z])
+
+(defn zip-depth [loc]
+ (loop [l loc d 0] (if (nil? l) d (recur (z/up l) (inc d)))))
+
+(defn zip-prev-num [loc]
+ (loop [l loc]
+ (if-let [prev (z/prev l)]
+ (if (not (z/branch? prev))
+ prev
+ (recur prev)))))
+
+(defn zip-next-num [loc]
+ (loop [l loc]
+ (if-let [nxt (z/next l)]
+ (if (not (z/branch? nxt))
+ (when (some? (z/node nxt)) nxt)
+ (recur nxt)))))
+
+(defn snailfish-explode [tree]
+ (loop [node (z/vector-zip tree)]
+ (if (z/end? node)
+ node
+ (if (and (z/branch? node)
+ (= 2 (count (z/children node)))
+ (and (not (z/branch? (z/down node))) (not (z/branch? (z/next (z/down node)))))
+ (> (zip-depth node) 5))
+ (let [children (z/children node)
+ nnode (z/replace node 0)]
+ (if-let [prev (zip-prev-num nnode)]
+ (z/root
+ (let [new-prev (z/edit prev + (first children))]
+ (if-let [nxt (nth (iterate zip-next-num new-prev) 2)]
+ (z/edit nxt + (second children))
+ new-prev)))
+ (when-let [nxt (zip-next-num nnode)]
+ (z/root (z/edit nxt + (second children))))))
+ (recur (z/next node))))))
+
+(defn snailfish-split [tree]
+ (loop [node (z/vector-zip tree)]
+ (if (or (z/end? node) (nil? (z/node node)))
+ node
+ (if (and (not (z/branch? node)) (> (z/node node) 9))
+ (z/root
+ (z/replace node
+ (z/make-node node (z/node node)
+ (map long [(Math/floor (/ (z/node node) 2))
+ (Math/ceil (/ (z/node node) 2))]))))
+ (recur (z/next node))))))
+
+(defn snailfish-magnitude [ntree]
+ (if (= 2 (count (z/children ntree)))
+ (let [cl (z/down ntree) cr (z/right cl)]
+ (+ (* 3 (if (z/branch? cl) (snailfish-magnitude cl) (z/node cl)))
+ (* 2 (if (z/branch? cr) (snailfish-magnitude cr) (z/node cr)))))
+ (loop [node (z/down ntree) sum 0]
+ (if (nil? node)
+ sum
+ (recur (z/right node)
+ (+ sum (if (z/branch? node) (snailfish-magnitude node) (z/node node))))))))
+
+(defn snailfish-solve [add-list]
+ (snailfish-magnitude
+ (loop [result (z/vector-zip (vec (take 2 add-list)))
+ next-input (drop 2 add-list)]
+ (let [explode (snailfish-explode result)]
+ (if (nil? (second explode))
+ (recur explode next-input)
+ (let [splt (snailfish-split result)]
+ (if (nil? (second splt))
+ (recur splt next-input)
+ (if (empty? next-input)
+ result
+ (recur (z/vector-zip
+ [(vec (z/children result)) (first next-input)])
+ (rest next-input))))))))))
+
+; Build input list
+(def input (->> (slurp "./in")
+ clojure.string/split-lines
+ (map read-string)))
+
+; Part 1
+(println (snailfish-solve input))
+
+; Part 2
+(->> (for [in input in2 input :when (not= in in2)] [in in2])
+ (vec)
+ (r/map snailfish-solve)
+ (r/fold (r/monoid max (constantly 0)))
+ (println))
+
diff --git a/year2021/day19/partboth.clj b/year2021/day19/partboth.clj
new file mode 100644
index 0000000..48f1a9c
--- /dev/null
+++ b/year2021/day19/partboth.clj
@@ -0,0 +1,98 @@
+(require '[clojure.core.reducers :as r])
+(require 'clojure.string)
+
+(def orientation-vectors
+ [[[1 0 0] [0 1 0] [0 0 1]]
+ [[1 0 0] [0 0 -1] [0 1 0]]
+ [[1 0 0] [0 -1 0] [0 0 -1]]
+ [[1 0 0] [0 0 1] [0 -1 0]]
+ [[-1 0 0] [0 -1 0] [0 0 1]]
+ [[-1 0 0] [0 0 -1] [0 -1 0]]
+ [[-1 0 0] [0 1 0] [0 0 -1]]
+ [[-1 0 0] [0 0 1] [0 1 0]]
+ [[0 0 -1] [1 0 0] [0 -1 0]]
+ [[0 1 0] [1 0 0] [0 0 -1]]
+ [[0 0 1] [1 0 0] [0 1 0]]
+ [[0 -1 0] [1 0 0] [0 0 1]]
+ [[0 0 1] [-1 0 0] [0 -1 0]]
+ [[0 1 0] [-1 0 0] [0 0 1]]
+ [[0 0 -1] [-1 0 0] [0 1 0]]
+ [[0 -1 0] [-1 0 0] [0 0 -1]]
+ [[0 -1 0] [0 0 -1] [1 0 0]]
+ [[0 0 1] [0 -1 0] [1 0 0]]
+ [[0 1 0] [0 0 1] [1 0 0]]
+ [[0 0 -1] [0 1 0] [1 0 0]]
+ [[0 0 1] [0 1 0] [-1 0 0]]
+ [[0 -1 0] [0 0 1] [-1 0 0]]
+ [[0 0 -1] [0 -1 0] [-1 0 0]]
+ [[0 1 0] [0 0 -1] [-1 0 0]]])
+
+(defn build-scanner-reports
+ "Splits slurped, split-line'd input by scanner."
+ [input-lines]
+ (loop [scanners [] rem-input input-lines]
+ (if (empty? rem-input)
+ scanners
+ (let [[this nxt] (split-with (comp not empty?) rem-input)]
+ (recur (->> (rest this)
+ (map #(read-string (clojure.string/join ["[" % "]"])))
+ (conj scanners))
+ (rest nxt))))))
+
+(defn point-add [[a b c] [d e f]] [(+ a d) (+ b e) (+ c f)])
+
+(defn orient [[x y z] [[a b c] [d e f] [g h i]]]
+ [(+ (* x a) (* y d) (* z g)) (+ (* x b) (* y e) (* z h)) (+ (* x c) (* y f) (* z i))])
+
+(defn scanner-orientation [report or-index]
+ (map #(orient (map - %) (get orientation-vectors or-index)) report))
+
+(defn scanner-orientations
+ "Builds list of scanner reports adjusted for all possible orientations."
+ [report]
+ (for [or-vec orientation-vectors]
+ (map #(orient (map - %) or-vec) report)))
+
+(defn scanner-build-potential-links
+ "Builds a list for each s2 orientation that contains lists of each s1 point
+ added to the corresponding s2 point."
+ [s1 s2]
+ (for [s2-or (scanner-orientations s2)]
+ (for [i (range 0 (count s1))]
+ (for [j (range 0 (count s2-or))]
+ (point-add (nth s1 i) (nth s2-or j))))))
+
+(defn scanner-find-connection
+ "Attempt to determine if s1 and s2 are linked through common beacons."
+ [s1 s2]
+ (loop [potential-links (as-> (scanner-build-potential-links s1 s2) $
+ (for [i (range 0 (count $))] [i (nth $ i)])
+ (into [] $))]
+ (when-let [[orientation link] (first potential-links)]
+ (if-let [match (first (drop-while #(< (val %) 12) (frequencies (reduce concat link))))]
+ [orientation (key match)]
+ (recur (rest potential-links))))))
+
+(defn scanner-merge
+ "Merges the report of linked scanner s2 into scanner s1."
+ [s1 s2 s2-or s2-coord]
+ (vec (into (set s1) (map #(point-add s2-coord (map - %)) (scanner-orientation s2 s2-or)))))
+
+(let [[beacon-count scanner-coords]
+ (loop [new-reps (->> (slurp "./in") clojure.string/split-lines build-scanner-reports)
+ sc-coords '([0 0 0])
+ next-reps (rest new-reps)]
+ (if-let [next-rep (first next-reps)]
+ (if-let [[found-or found-coord] (scanner-find-connection (first new-reps) next-rep)]
+ (let [new-base (scanner-merge (first new-reps) next-rep found-or found-coord)
+ newest-reps (->> (assoc new-reps 0 new-base) (filterv (partial not= next-rep)))]
+ (println "found" found-coord)
+ (recur newest-reps (conj sc-coords found-coord) (rest newest-reps)))
+ (recur new-reps sc-coords (rest next-reps)))
+ [(count (first new-reps)) sc-coords]))]
+ (println "Part 1:" beacon-count)
+ (println "Part 2:"
+ (apply max
+ (for [p1 scanner-coords p2 scanner-coords :when (not= p1 p2)]
+ (apply + (map #(Math/abs %) (point-add (map - p1) p2)))))))
+
diff --git a/year2021/day2/part1.clj b/year2021/day2/part1.clj
new file mode 100644
index 0000000..e4e9d49
--- /dev/null
+++ b/year2021/day2/part1.clj
@@ -0,0 +1,26 @@
+; Day 2, part 1
+; Read a list of instructions from stdin:
+; "down X" increases depth number by X,
+; "up X" decreases depth by X,
+; "forward X" increases xpos by X.
+; Print (xpos * depth) after end of data.
+;
+
+(require '[clojure.string :as str])
+
+(println
+ (reduce *
+ (vals
+ (reduce
+ #(case (first %2)
+ "forward" (update %1 :xpos + (second %2))
+ "up" (update %1 :depth - (second %2))
+ "down" (update %1 :depth + (second %2))
+ )
+ {:xpos 0 :depth 0}
+ (->> (slurp "./in")
+ str/split-lines
+ (map #(str/split % #" "))
+ (map #(update % 1 read-string))
+ )))))
+
diff --git a/year2021/day2/part2.clj b/year2021/day2/part2.clj
new file mode 100644
index 0000000..9487a49
--- /dev/null
+++ b/year2021/day2/part2.clj
@@ -0,0 +1,27 @@
+; Day 2, part 2
+; Read a list of instructions from stdin:
+; "down X" increases aim number by X,
+; "up X" decreases aim by X,
+; "forward X" increases xpos by X and depth by (aim * X).
+; Print (xpos * depth) after end of data.
+;
+
+(require '[clojure.string :as str])
+
+(println
+ (apply *
+ (map
+ (reduce
+ #(case (first %2)
+ "forward" (-> %1 (update :xpos + (second %2))
+ (update :depth + (* (%1 :aim) (second %2))))
+ "up" (update %1 :aim - (second %2))
+ "down" (update %1 :aim + (second %2)))
+ {:xpos 0 :depth 0 :aim 0}
+ (->> (slurp "./in")
+ str/split-lines
+ (map #(str/split % #" "))
+ (map #(update % 1 read-string))))
+ [:xpos :depth]
+ )))
+
diff --git a/year2021/day20/partboth.clj b/year2021/day20/partboth.clj
new file mode 100644
index 0000000..33a4d4f
--- /dev/null
+++ b/year2021/day20/partboth.clj
@@ -0,0 +1,43 @@
+(require '[clojure.string :as str])
+
+(defn pad-input-map [in-map pad-val]
+ (let [ly (count in-map) lx (count (first in-map))
+ sy (+ 4 ly) sx (+ 4 lx)
+ buffer (byte-array (repeat (* sy sx) pad-val))]
+ (doseq [y (range 0 ly) x (range 0 lx)]
+ (aset-byte buffer (+ (* sx (+ 2 y)) 2 x) (get-in in-map [y x])))
+ (mapv vec (partition sx buffer))))
+
+(defn get-number [in-map y x]
+ (map (partial get-in in-map)
+ [[(dec y) (dec x)] [(dec y) x] [(dec y) (inc x)]
+ [y (dec x)] [y x] [y (inc x)]
+ [(inc y) (dec x)] [(inc y) x] [(inc y) (inc x)]]))
+
+(defn get-new-pixel [enhance-algo in-map y x]
+ (->> (get-number in-map y x)
+ (reduce #(cond-> (* 2 %1) (= 1 %2) inc) 0)
+ (get enhance-algo)))
+
+(defn build-output-number-map [enhance-algo in-map]
+ (let [pad-val (if (pos? (first enhance-algo)) (first (first in-map)) 0)
+ padded-in-map (pad-input-map in-map pad-val)]
+ (->> (for [y (range 1 (dec (count padded-in-map)))]
+ (future
+ (mapv (partial get-new-pixel enhance-algo padded-in-map y)
+ (range 1 (dec (count (first padded-in-map)))))))
+ (mapv deref))))
+
+(defn count-lit-pixels [in-map] ((frequencies (flatten in-map)) 1))
+
+(let [[enhance-algo in-map]
+ (->> (slurp "./in")
+ str/split-lines
+ ((juxt (comp (partial mapv {\. 0 \# 1}) first)
+ (comp (partial mapv #(mapv {\. 0 \# 1} %)) (partial drop 2)))))
+ image-output (iterate (partial build-output-number-map enhance-algo) in-map)]
+ (println "Part 1:" (count-lit-pixels (nth image-output 2)))
+ (println "Part 2:" (count-lit-pixels (nth image-output 50))))
+
+(shutdown-agents)
+
diff --git a/year2021/day21/part1.clj b/year2021/day21/part1.clj
new file mode 100644
index 0000000..174f5a1
--- /dev/null
+++ b/year2021/day21/part1.clj
@@ -0,0 +1,22 @@
+(require 'clojure.string)
+
+(loop [player-positions (->> (slurp "./in")
+ clojure.string/split-lines
+ (map (comp read-string second #(clojure.string/split % #": ")))
+ (mapv #(drop (dec %) (cycle (range 1 11)))))
+ player-scores (into [] (repeat (count player-positions) 0))
+ player-turn (cycle (range 0 (count player-scores)))
+ deterministic-dice (cycle (range 1 101))
+ dice-roll-count 0]
+ (if-not (every? #(< % 1000) player-scores)
+ (println (* dice-roll-count (apply min player-scores)))
+ (let [roll (reduce + (take 3 deterministic-dice))
+ turn (first player-turn)
+ new-position (drop roll (get player-positions turn))]
+ (recur
+ (assoc player-positions turn new-position)
+ (update player-scores turn + (first new-position))
+ (next player-turn)
+ (drop 3 deterministic-dice)
+ (+ dice-roll-count 3)))))
+
diff --git a/year2021/day21/part2.clj b/year2021/day21/part2.clj
new file mode 100644
index 0000000..0042aaa
--- /dev/null
+++ b/year2021/day21/part2.clj
@@ -0,0 +1,37 @@
+(defn dice-probability-spread [sides rolls]
+ (let [sr (range 1 (inc sides))]
+ (->> sr
+ (iterate #(flatten (for [i sr] (map (partial + i) %))))
+ (drop (dec rolls))
+ ((comp frequencies first)))))
+
+(def rolls (dice-probability-spread 3 3))
+
+(defn advance-pos [pos roll] (let [n (+ pos roll)] (if (> n 10) (- n 10) n)))
+
+(defn add-roll [state roll p1-turn]
+ (let [pos (if p1-turn :pos1 :pos2)
+ score (if p1-turn :score1 :score2)
+ new-pos (advance-pos (state pos) roll)]
+ (-> state
+ (assoc pos new-pos)
+ (update score + new-pos))))
+
+(defonce wins (atom [0 0]))
+
+(loop [turn 0 states {{:pos1 4 :score1 0 :pos2 8 :score2 0} 1}]
+ (if (empty? states)
+ (println (apply max @wins))
+ (recur
+ (if (= 0 turn) 1 0)
+ (reduce
+ #(let [kvp (first %2)]
+ (if (< ((key kvp) (if (= 0 turn) :score1 :score2)) 21)
+ (if (contains? %1 (key kvp))
+ (update %1 (key kvp) +' (val kvp))
+ (conj %1 kvp))
+ (do (swap! wins update turn +' (val kvp)) %1)))
+ {}
+ (for [s states r rolls]
+ {(add-roll (first s) (key r) (= 0 turn)) (*' (second s) (val r))})))))
+
diff --git a/year2021/day22/part1.clj b/year2021/day22/part1.clj
new file mode 100644
index 0000000..bffc1a9
--- /dev/null
+++ b/year2021/day22/part1.clj
@@ -0,0 +1,25 @@
+(require '[clojure.string :as str])
+
+(def input (->> (slurp "./in")
+ (str/split-lines)
+ (map
+ (fn [line]
+ [(if (str/starts-with? line "on") true false)
+ (->> (str/split line #"[^-\d]+")
+ (rest)
+ (map #(Integer/parseInt %))
+ (partition 2))]))))
+
+(println
+ (frequencies
+ (vals
+ (reduce
+ (fn [cmap c]
+ (into cmap
+ (let [[xx yy zz] (second c)]
+ (for [x (range (max -50 (first xx)) (inc (min 50 (second xx))))
+ y (range (max -50 (first yy)) (inc (min 50 (second yy))))
+ z (range (max -50 (first zz)) (inc (min 50 (second zz))))]
+ [[x y z] (first c)]))))
+ {} input))))
+
diff --git a/year2021/day22/part2.clj b/year2021/day22/part2.clj
new file mode 100644
index 0000000..c5d2803
--- /dev/null
+++ b/year2021/day22/part2.clj
@@ -0,0 +1,60 @@
+(require '[clojure.core.reducers :as r])
+(require '[clojure.string :as str])
+
+(def input (->> (slurp "./in")
+ (str/split-lines)
+ (map
+ (fn [line]
+ [(if (str/starts-with? line "on") true false)
+ (->> (str/split line #"[^-\d]+")
+ (rest)
+ (map #(Integer/parseInt %))
+ (partition 2)
+ (mapv vec))]))
+ (reverse)))
+
+(defn build-coord-list [coords idx]
+ (->> coords
+ (map #(update (get (second %) idx) 1 inc))
+ (flatten)
+ (sort)
+ (vec)))
+
+(defn filter-coord-list [coords idx v]
+ (filter
+ #(let [[p0 p1] (get (second %) idx)]
+ (and (<= p0 v) (<= v p1)))
+ coords))
+
+(def xs (build-coord-list input 0))
+(def ys (build-coord-list input 1))
+(def zs (build-coord-list input 2))
+
+(defonce on-count (atom 0))
+
+(loop [x xs]
+ (when-not (empty? (rest x))
+ (println "x=" (first x))
+ (let [inputx (filter-coord-list input 0 (first x))]
+ (swap! on-count +
+ (r/fold
+ 32
+ +
+ (fn [onc2 iy]
+ (let [inputy (filter-coord-list inputx 1 (get ys iy))]
+ (+ onc2
+ (reduce
+ (fn [onc iz]
+ (let [inputz (first (filter-coord-list inputy 2 (get zs iz)))]
+ (cond-> onc
+ (first inputz)
+ (+ (* (- (second x) (first x))
+ (- (get ys (inc iy)) (get ys iy))
+ (- (get zs (inc iz)) (get zs iz)))))))
+ 0
+ (range 0 (dec (count zs)))))))
+ (into [] (range 0 (dec (count ys))))))
+ (recur (rest x)))))
+
+(println @on-count)
+
diff --git a/year2021/day23/part1-done-by-hand b/year2021/day23/part1-done-by-hand
new file mode 100644
index 0000000..e69de29
--- /dev/null
+++ b/year2021/day23/part1-done-by-hand
diff --git a/year2021/day23/part2.clj b/year2021/day23/part2.clj
new file mode 100644
index 0000000..adbefc1
--- /dev/null
+++ b/year2021/day23/part2.clj
@@ -0,0 +1,90 @@
+(require '[clojure.core.reducers :as r])
+
+(def init-field
+ [nil
+ nil
+ (seq [:b :d :d :d])
+ nil
+ (seq [:b :c :b :a])
+ nil
+ (seq [:c :b :a :a])
+ nil
+ (seq [:d :a :c :c])
+ nil
+ nil])
+
+(defn do-slot [[field energy] idx]
+ (let [slot (get field idx)]
+ (cond
+ ; Moving value out of a room
+ (and (seq? slot)
+ (not (empty? slot))
+ (not (every? #(= % ({2 :a 4 :b 6 :c 8 :d} idx)) slot)))
+ (let [open-slots
+ (filterv
+ #(contains? #{0 1 3 5 7 9 10} %)
+ (concat
+ (for [i (reverse (range 0 idx))
+ :while (or (nil? (get field i)) (seq? (get field i)))] i)
+ (for [i (range idx (count field))
+ :while (or (nil? (get field i)) (seq? (get field i)))] i)))]
+ (when-not (empty? open-slots)
+ (map
+ (fn [os]
+ [(-> field
+ (assoc os (first slot))
+ (update idx rest))
+ (+ energy (* ({:a 1 :b 10 :c 100 :d 1000} (first slot))
+ (+ (inc (- 4 (count slot))) (Math/abs (- os idx)))))])
+ open-slots)))
+ ; Moving value into a room
+ (and (not (seq? slot)) (some? slot))
+ (let [our-room ({:a 2 :b 4 :c 6 :d 8} slot)]
+ (if (every? #(or (nil? (get field %)) (seq? (get field %)))
+ (range (inc (min our-room idx)) (max our-room idx)))
+ (let [room (get field our-room)]
+ (when (or (empty? room) (every? #(= slot %) room))
+ [(-> field
+ (assoc idx nil)
+ (update our-room conj slot))
+ (+ energy (* ({:a 1 :b 10 :c 100 :d 1000} slot)
+ (+ (Math/abs (- idx our-room)) (- 4 (count room)))))])))))))
+
+(defn winner? [[field q s]]
+ (= field
+ [nil
+ nil
+ (seq [:a :a :a :a])
+ nil
+ (seq [:b :b :b :b])
+ nil
+ (seq [:c :c :c :c])
+ nil
+ (seq [:d :d :d :d])
+ nil
+ nil]))
+
+(defn do-turns [fields]
+ (into #{}
+ (r/fold
+ r/cat
+ #(if-let [t (apply do-slot %2)]
+ (if (seq? t)
+ (reduce r/append! %1 t)
+ (r/append! %1 t))
+ %1)
+ (into [] (for [f fields i (range 0 11)] [f i])))))
+
+(defn play-games [turns tc]
+ (println "Games:" (count turns) "Turn:" tc)
+ (let [new-turns (do-turns turns)
+ winners (filter winner? new-turns)]
+ (if (seq winners)
+ (map second winners)
+ (recur new-turns (inc tc)))))
+
+(->> (play-games #{[init-field 0]} 0)
+ sort
+ first
+ ((partial println "Lowest energy:")))
+
diff --git a/year2021/day24/checker.clj b/year2021/day24/checker.clj
new file mode 100644
index 0000000..4874f62
--- /dev/null
+++ b/year2021/day24/checker.clj
@@ -0,0 +1,47 @@
+(require '[clojure.string :as str])
+
+(defn alu-parse-argument [arg]
+ (if (contains? #{\w \x \y \z} (first arg))
+ (str/join ["@" arg])
+ (read-string arg)))
+
+(defn alu-parse-instruction [[ins arg1 arg2]]
+ (let [arg2-parsed (if (some? arg2) (alu-parse-argument arg2) 0)]
+ (case ins
+ "inp"
+ (str/join ["(reset! " arg1 " (- (int (get input (swap! index inc))) 48))\n"])
+ "add"
+ (str/join ["(swap! " arg1 " +' " arg2-parsed ")\n"])
+ "mul"
+ (str/join ["(swap! " arg1 " *' " arg2-parsed ")\n"])
+ "div"
+ (str/join ["(swap! " arg1 " quot " arg2-parsed ")\n"])
+ "mod"
+ (str/join ["(swap! " arg1 " rem " arg2-parsed ")\n"])
+ "eql"
+ (str/join ["(swap! " arg1 " #(if (= % " arg2-parsed ") 1 0))\n"]))))
+
+(def alu-program-header
+ "(defn alu-program [input]
+ (let [index (atom -1) w (atom 0) x (atom 0) y (atom 0) z (atom 0)]\n")
+
+(defn alu-compile-program [instructions]
+ (let [parsed-ins (mapv alu-parse-instruction instructions)]
+ (eval (read-string
+ (str/join [alu-program-header
+ (str/join parsed-ins)
+ "@z))"])))))
+
+(def program (->> (slurp "./in")
+ str/split-lines
+ (map #(str/split % #"\s+"))
+ (alu-compile-program)))
+
+; to split by `inp w`s:
+; (partition 18)
+; (reverse)
+; (mapv alu-compile-program)))
+
+(println "Part 1:" (program "16181111641521"))
+(println "Part 2:" (program "59692994994998"))
+
diff --git a/year2021/day24/partboth.clj b/year2021/day24/partboth.clj
new file mode 100644
index 0000000..9548ec3
--- /dev/null
+++ b/year2021/day24/partboth.clj
@@ -0,0 +1,41 @@
+(require '[clojure.string :as str])
+
+; The given program operates on a base-26 stack, either pushing or
+; (conditionally) popping values. This code extracts those operations
+; from the input, then works through them to determine the highest
+; and lowest values that produce an output of zero.
+; See the .md file in this directory for more info.
+
+(def program (->> (slurp "./in")
+ str/split-lines
+ (map #(str/split % #"\s+"))
+ (partition 18)
+ (map (fn [step]
+ (if (= "1" (last (nth step 4)))
+ [:push (read-string (last (nth step 15)))]
+ [:pop (read-string (last (nth step 5)))])
+ ))))
+
+(defn program-index [prog] (- 14 (count prog)))
+
+(loop [numbers (vec (repeat 2 (vec (repeat 14 0))))
+ stack '()
+ prog program]
+ (if-let [step (first prog)]
+ (if (= :push (first step))
+ (recur numbers
+ (conj stack [(program-index prog) (second step)])
+ (rest prog))
+ (let [i2 (program-index prog)
+ [i1 v] (first stack)
+ diff (+ v (second step))]
+ (recur
+ [(if (neg? diff)
+ (-> (first numbers) (assoc i1 9) (assoc i2 (+ 9 diff)))
+ (-> (first numbers) (assoc i2 9) (assoc i1 (- 9 diff))))
+ (if (neg? diff)
+ (-> (second numbers) (assoc i2 1) (assoc i1 (- 1 diff)))
+ (-> (second numbers) (assoc i1 1) (assoc i2 (+ 1 diff))))]
+ (rest stack) (rest prog))))
+ (println (map str/join numbers))))
+
diff --git a/year2021/day24/solved-by-hand-with-help.md b/year2021/day24/solved-by-hand-with-help.md
new file mode 100644
index 0000000..1ddf03e
--- /dev/null
+++ b/year2021/day24/solved-by-hand-with-help.md
@@ -0,0 +1,4 @@
+Went through this wonderful guide:
+
+[https://github.com/mrphlip/aoc/blob/master/2021/24.md](https://github.com/mrphlip/aoc/blob/master/2021/24.md)
+
diff --git a/year2021/day25/part1.clj b/year2021/day25/part1.clj
new file mode 100644
index 0000000..df2947c
--- /dev/null
+++ b/year2021/day25/part1.clj
@@ -0,0 +1,40 @@
+(require '[clojure.string :as str])
+
+(defn cucumber-step-east [cuc-map]
+ (for [cuc-row cuc-map]
+ (let [shifted-cuc (str/replace cuc-row #">\." ".>")]
+ (if (and (= \> (last cuc-row)) (= \. (first cuc-row)))
+ (-> shifted-cuc
+ (str/replace #">$" ".")
+ (str/replace #"^\." ">"))
+ shifted-cuc))))
+
+(defn cucumber-step-south [cuc-map]
+ (let [extra-cuc-map
+ (conj (vec (cons (last cuc-map) cuc-map)) (first cuc-map))
+ new-extra-cuc-map
+ (for [y (reverse (range 1 (count extra-cuc-map)))]
+ (str/join
+ (for [x (range 0 (count (first cuc-map)))]
+ (cond
+ (and (= \. (get-in extra-cuc-map [y x])) (= \v (get-in extra-cuc-map [(dec y) x])))
+ \v
+ (and (= \v (get-in extra-cuc-map [y x])) (= \. (get-in extra-cuc-map [(inc y) x])))
+ \.
+ :else
+ (get-in extra-cuc-map [y x])))))]
+ (into [] (reverse (rest new-extra-cuc-map)))))
+
+(defn cucumber-seq [cuc-map-init]
+ (iterate (comp cucumber-step-south cucumber-step-east) cuc-map-init))
+
+(def input (->> (slurp "./in")
+ (str/split-lines)
+ (vec)))
+
+(loop [cuc-hist '() cuc-list (cucumber-seq input)]
+ (let [next-cuc (first cuc-list)]
+ (if (or (nil? next-cuc) (= (first cuc-hist) next-cuc))
+ (println "Cucs are stuck! Steps:" (count cuc-hist))
+ (recur (conj cuc-hist next-cuc) (rest cuc-list)))))
+
diff --git a/year2021/day3/part1.clj b/year2021/day3/part1.clj
new file mode 100644
index 0000000..3fe888e
--- /dev/null
+++ b/year2021/day3/part1.clj
@@ -0,0 +1,25 @@
+(require '[clojure.string :as str])
+
+(->> "./in"
+ (slurp)
+ (str/split-lines)
+ (map (fn [l] (map #(if (= % \1) 1 0) l)))
+ (apply (partial map +))
+ (map #(if (< % 500) \1 \0))
+ (str/join)
+ (#(Integer/parseInt % 2))
+ (#(* % (bit-xor % (dec (int (Math/pow 2 12))))))
+ (println)
+ )
+
+; (->> input data file name
+; read in entire contents
+; split contents into array of lines
+; for each line, transform characters '1'/'0' to numbers
+; build sum array using the lines
+; convert back to array of characters
+; join characters into single string
+; convert binary string to a number (gamma)
+; multiply gamma by its bit-inverse (bit length hard-coded)
+; print results
+
diff --git a/year2021/day3/part2.clj b/year2021/day3/part2.clj
new file mode 100644
index 0000000..00e656f
--- /dev/null
+++ b/year2021/day3/part2.clj
@@ -0,0 +1,36 @@
+(require '[clojure.string :as str])
+
+(def bitcount 12)
+(def input
+ (->> "./in"
+ (slurp)
+ (str/split-lines)
+ (map #(Integer/parseInt % 2))
+ )
+ )
+
+(defn countbit [lst bit]
+ (->> lst
+ (map #(if (bit-test % bit) 1 0))
+ (apply +)
+ )
+ )
+
+(defn filterbit [lst bit v]
+ (if (= 1 (count lst))
+ lst
+ (filter #(= (bit-test % bit) v) lst)
+ )
+ )
+
+(loop [bit (dec bitcount) lst0 input lst1 input]
+ (if (and (= 1 (count lst0)) (= 1 (count lst1)))
+ (println (map * lst0 lst1))
+ (recur
+ (dec bit)
+ (filterbit lst0 bit (>= (countbit lst0 bit) (/ (count lst0) 2)))
+ (filterbit lst1 bit (< (countbit lst1 bit) (/ (count lst1) 2)))
+ )
+ )
+ )
+
diff --git a/year2021/day4/part1.clj b/year2021/day4/part1.clj
new file mode 100644
index 0000000..b9fcceb
--- /dev/null
+++ b/year2021/day4/part1.clj
@@ -0,0 +1,70 @@
+(require '[clojure.string :as str])
+
+(def calls
+ (->> (read-line)
+ (#(str/split % #","))
+ (map #(Integer/parseInt %))
+ )
+ )
+
+(defn read-board []
+ (when (some? (read-line))
+ (mapv
+ (fn [line] (->> line
+ (str/join)
+ (str/trim)
+ (#(str/split % #"\s+"))
+ (map #(Integer/parseInt %))
+ )
+ )
+ (repeatedly 5 read-line)
+ )
+ )
+ )
+
+(defn bingo? [board]
+ (some?
+ (some
+ #(every? nil? %)
+ (concat board (apply mapv vector board))
+ )
+ )
+ )
+
+(defn bingo-mark [board n]
+ (mapv (partial replace {n nil}) board)
+ )
+
+(defn get-bingo [init-board]
+ (loop [board init-board nums calls turns 1]
+ (let [new-board (bingo-mark board (first nums))]
+ (if (bingo? new-board)
+ [new-board (first nums) turns]
+ (recur new-board (rest nums) (inc turns))
+ )
+ )
+ )
+ )
+
+(loop [best [999 0] board (read-board)]
+ (if (nil? board)
+ (println best)
+ (let [bingo (get-bingo board)]
+ (recur
+ (if (< (get bingo 2) (first best))
+ [(get bingo 2)
+ (->> (first bingo)
+ (flatten)
+ (filter some?)
+ (apply +)
+ (* (second bingo))
+ )
+ ]
+ best
+ )
+ (read-board)
+ )
+ )
+ )
+ )
+
diff --git a/year2021/day4/part2.clj b/year2021/day4/part2.clj
new file mode 100644
index 0000000..516e83c
--- /dev/null
+++ b/year2021/day4/part2.clj
@@ -0,0 +1,70 @@
+(require '[clojure.string :as str])
+
+(def calls
+ (->> (read-line)
+ (#(str/split % #","))
+ (map #(Integer/parseInt %))
+ )
+ )
+
+(defn read-board []
+ (when (some? (read-line))
+ (mapv
+ (fn [line] (->> line
+ (str/join)
+ (str/trim)
+ (#(str/split % #"\s+"))
+ (map #(Integer/parseInt %))
+ )
+ )
+ (repeatedly 5 read-line)
+ )
+ )
+ )
+
+(defn bingo? [board]
+ (some?
+ (some
+ #(every? nil? %)
+ (concat board (apply mapv vector board))
+ )
+ )
+ )
+
+(defn bingo-mark [board n]
+ (mapv (partial replace {n nil}) board)
+ )
+
+(defn get-bingo [init-board]
+ (loop [board init-board nums calls turns 1]
+ (let [new-board (bingo-mark board (first nums))]
+ (if (bingo? new-board)
+ [new-board (first nums) turns]
+ (recur new-board (rest nums) (inc turns))
+ )
+ )
+ )
+ )
+
+(loop [best [0 0] board (read-board)]
+ (if (nil? board)
+ (println best)
+ (let [bingo (get-bingo board)]
+ (recur
+ (if (> (get bingo 2) (first best))
+ [(get bingo 2)
+ (->> (first bingo)
+ (flatten)
+ (filter some?)
+ (apply +)
+ (* (second bingo))
+ )
+ ]
+ best
+ )
+ (read-board)
+ )
+ )
+ )
+ )
+
diff --git a/year2021/day5/part1.clj b/year2021/day5/part1.clj
new file mode 100644
index 0000000..95d04ba
--- /dev/null
+++ b/year2021/day5/part1.clj
@@ -0,0 +1,75 @@
+(require '[clojure.string :as str])
+
+(defn read-coords []
+ (let [line (read-line)]
+ (when (not (empty? line))
+ (mapv
+ #(Integer/parseInt %)
+ (str/split
+ line
+ #"[^\d]+"
+ )
+ )
+ )
+ )
+ )
+
+(defn read-all-coords []
+ (loop [cds [] c (read-coords)]
+ (if (empty? c)
+ cds
+ (recur
+ (conj cds c)
+ (read-coords)
+ )
+ )
+ )
+ )
+
+(defn mark-coord [cmap x y]
+ (update cmap y #(update % x inc))
+ )
+
+(defn mark-coords [cmap x1 y1 x2 y2]
+ (cond
+ (= y1 y2)
+ (reduce
+ #(mark-coord %1 %2 y1)
+ cmap
+ (range (min x1 x2) (inc (max x1 x2)))
+ )
+ (= x1 x2)
+ (reduce
+ #(mark-coord %1 x1 %2)
+ cmap
+ (range (min y1 y2) (inc (max y1 y2)))
+ )
+ :else
+ cmap
+ )
+ )
+
+(defn empty-map []
+ (vec
+ (repeat 1000
+ (vec (repeat 1000 0))
+ )
+ )
+ )
+
+(def finished-map
+ (reduce
+ #(apply (partial mark-coords %1) %2)
+ (empty-map)
+ (read-all-coords)
+ )
+ )
+
+(->> finished-map
+ (flatten)
+ (map dec)
+ (filter pos?)
+ (count)
+ (println)
+ )
+
diff --git a/year2021/day5/part2.clj b/year2021/day5/part2.clj
new file mode 100644
index 0000000..d324618
--- /dev/null
+++ b/year2021/day5/part2.clj
@@ -0,0 +1,83 @@
+(require '[clojure.string :as str])
+
+(defn read-coords []
+ (let [line (read-line)]
+ (when (not (empty? line))
+ (mapv
+ #(Integer/parseInt %)
+ (str/split
+ line
+ #"[^\d]+"
+ )
+ )
+ )
+ )
+ )
+
+(defn read-all-coords []
+ (loop [cds [] c (read-coords)]
+ (if (empty? c)
+ cds
+ (recur
+ (conj cds c)
+ (read-coords)
+ )
+ )
+ )
+ )
+
+(defn mark-coord [cmap x y]
+ (update
+ cmap
+ [x y]
+ #(if (nil? %) 0 (inc %))
+ )
+ )
+
+(defn mark-coords [cmap x1 y1 x2 y2]
+ (cond
+ (= y1 y2)
+ (reduce
+ #(mark-coord %1 %2 y1)
+ cmap
+ (range (min x1 x2) (inc (max x1 x2)))
+ )
+ (= x1 x2)
+ (reduce
+ #(mark-coord %1 x1 %2)
+ cmap
+ (range (min y1 y2) (inc (max y1 y2)))
+ )
+ :else
+ (let [ic (if (< x1 x2) [x1 y1] [x2 y2])
+ ec (if (> x1 x2) [x1 y1] [x2 y2])
+ dy (if (> (second ec) (second ic)) 1 -1)
+ ]
+ (loop [cm cmap c ic]
+ (if (> (first c) (first ec))
+ cm
+ (recur
+ (apply (partial mark-coord cm) c)
+ [(inc (first c)) (+ dy (second c))]
+ )
+ )
+ )
+ )
+ )
+ )
+
+(def finished-map
+ (reduce
+ #(apply (partial mark-coords %1) %2)
+ {}
+ (read-all-coords)
+ )
+ )
+
+(->> finished-map
+ (vals)
+ (filter pos?)
+ (count)
+ (println)
+ )
+
diff --git a/year2021/day6/consteval.cpp b/year2021/day6/consteval.cpp
new file mode 100644
index 0000000..f46da89
--- /dev/null
+++ b/year2021/day6/consteval.cpp
@@ -0,0 +1,32 @@
+#include <algorithm>
+#include <cstdint>
+#include <numeric>
+#include <iostream>
+
+consteval auto countFish(unsigned int day)
+{
+ unsigned char input[] = {
+ //3,4,3,1,2
+#include "in"
+ };
+
+ uint64_t counts[9];
+ std::fill(counts, counts + 9, 0);
+ for (int i = 0; i < sizeof(input); ++i)
+ ++counts[input[i]];
+
+ for (int i = 0; i < day; ++i) {
+ std::rotate(counts, counts + 1, counts + 9);
+ counts[6] += counts[8];
+ }
+
+ return std::accumulate(counts, counts + 9, 0ull);
+}
+
+int main()
+{
+ std::cout << "80: " << countFish(80) << std::endl;
+ std::cout << "256: " << countFish(256) << std::endl;
+ return 0;
+}
+
diff --git a/year2021/day6/part1.clj b/year2021/day6/part1.clj
new file mode 100644
index 0000000..70293f7
--- /dev/null
+++ b/year2021/day6/part1.clj
@@ -0,0 +1,20 @@
+(require '[clojure.string :as str])
+
+(def fish-init
+ (->> (read-line)
+ (#(str/split % #","))
+ (map #(Integer/parseInt %))
+ (vec)
+ )
+ )
+
+(defn cycle-day [fish]
+ (flatten
+ (for [f (map dec fish)]
+ (if (neg? f) [6 8] f)
+ )
+ )
+ )
+
+(println (count (nth (iterate cycle-day fish-init) 80)))
+
diff --git a/year2021/day6/part2.clj b/year2021/day6/part2.clj
new file mode 100644
index 0000000..ad6fe91
--- /dev/null
+++ b/year2021/day6/part2.clj
@@ -0,0 +1,25 @@
+(require '[clojure.string :as str])
+
+(->> (read-line)
+ (#(str/split % #","))
+ (map read-string)
+ (reduce #(update %1 %2 inc) (vec (repeat 9 0)))
+ (iterate
+ #(let [nf (conj (vec (rest %)) (first %))]
+ (update nf 6 (partial + (get nf 8)))
+ )
+ )
+ (#(nth % 256))
+ (apply +)
+ (println)
+ )
+
+; ->> read input from stdin
+; split input string by commas
+; convert string array into number array
+; reduce to frequency counts
+; create iterator that returns next day's counts
+; get 256th iteration
+; sum all frequency counts
+; print results
+
diff --git a/year2021/day7/maxima.mac b/year2021/day7/maxima.mac
new file mode 100644
index 0000000..a1c2d18
--- /dev/null
+++ b/year2021/day7/maxima.mac
@@ -0,0 +1,10 @@
+load("descriptive")$
+load("in.mac")$ /* Should define `input` list of numbers. */
+
+calcfuel(pos, distmod) :=
+ lreduce("+", map(lambda([n], distmod(abs(pos - n))), input))$
+
+calcfuel(median(input), lambda([x],x));
+calcfuel(floor(mean(input)), lambda([x], (x*(x+1)/2)));
+calcfuel(floor(mean(input)) + 1, lambda([x], (x*(x+1)/2)));
+
diff --git a/year2021/day7/part1.clj b/year2021/day7/part1.clj
new file mode 100644
index 0000000..f08e5b4
--- /dev/null
+++ b/year2021/day7/part1.clj
@@ -0,0 +1,18 @@
+(defn median [lst]
+ (as-> (count lst) $
+ (quot $ 2)
+ (subvec (vec (sort lst)) (dec $) (inc $))
+ (if (even? (count lst)) (apply + $) (second $))
+ (quot $ 2)
+ )
+ )
+
+(as-> (slurp "./in") $ ; "16,1,2,0,4,2,7,1,2,14"
+ (clojure.string/split $ #",")
+ (mapv read-string $)
+ (map (partial - (median $)) $)
+ (map #(Math/abs %) $)
+ (apply + $)
+ (println $)
+ )
+
diff --git a/year2021/day7/part2.clj b/year2021/day7/part2.clj
new file mode 100644
index 0000000..b96b55d
--- /dev/null
+++ b/year2021/day7/part2.clj
@@ -0,0 +1,20 @@
+(defn calc-fuel [lst pos]
+ (reduce #(as-> %2 $
+ (- pos $)
+ (Math/abs $)
+ (/ (* $ (inc $)) 2)
+ (+ %1 $)
+ )
+ 0 lst
+ )
+ )
+
+(let [input (as-> (slurp "./in") $ ;"16,1,2,0,4,2,7,1,2,14"
+ (clojure.string/split $ #",")
+ (map read-string $)
+ )
+ mean (quot (apply + input) (count input))]
+ (println (min (calc-fuel input mean)
+ (calc-fuel input (inc mean))))
+ )
+
diff --git a/year2021/day8/part1.clj b/year2021/day8/part1.clj
new file mode 100644
index 0000000..fe06fdf
--- /dev/null
+++ b/year2021/day8/part1.clj
@@ -0,0 +1,15 @@
+(require '[clojure.string :as str])
+
+(->> (str/split-lines (slurp "./in"))
+ (map
+ (fn [line]
+ (as-> line $
+ (str/split $ #" ")
+ (subvec $ 11 15)
+ (filter #(.contains [2 3 4 7] (count %)) $)
+ (count $)
+ )))
+ (apply +)
+ (println)
+ )
+
diff --git a/year2021/day8/part2.clj b/year2021/day8/part2.clj
new file mode 100644
index 0000000..1ee3c08
--- /dev/null
+++ b/year2021/day8/part2.clj
@@ -0,0 +1,66 @@
+(require '[clojure.string :as str])
+(require '[clojure.set :as set])
+
+(defn find-match
+ "
+ Searches for digit that `has segs-left` segments remaining
+ after the segments of `dig-to-cmp` are removed from the
+ digits in `cnts` with `grp-to-cmp` segments.
+ "
+ [segs-left dig-to-cmp cnts grp-to-cmp]
+ (as-> cnts $
+ (filterv #(= (first %) grp-to-cmp) $)
+ (filterv
+ #(-> (second %)
+ (str/replace (re-pattern (str/join ["[" dig-to-cmp "]"])) "")
+ (count)
+ (= segs-left)
+ )
+ $
+ )
+ (get-in $ [0 1])
+ )
+ )
+
+(defn determine-digits [line]
+ (let [counts (mapv #(do [(count %) %]) (take 10 line))
+ mcounts (into {} counts)]
+ (as-> {} $
+ (assoc $ 1 (mcounts 2)
+ 4 (mcounts 4)
+ 7 (mcounts 3)
+ 8 (mcounts 7)
+ )
+ (assoc $ 3 (find-match 2 ($ 7) counts 5)
+ 6 (find-match 4 ($ 7) counts 6)
+ 2 (find-match 3 ($ 4) counts 5)
+ 9 (find-match 2 ($ 4) counts 6)
+ )
+ (assoc $ 5 (find-match 2 ($ 2) counts 5))
+ (assoc $ 0 (find-match 2 ($ 5) counts 6))
+ )
+ )
+ )
+
+(println
+ (reduce
+ (fn [sum input]
+ (let [line (as-> input $
+ (str/split $ #" ")
+ (mapv (comp str/join sort) $)
+ )
+ number (subvec line 11 15)
+ decoder (set/map-invert (determine-digits line))]
+ (->> number
+ (map (comp str decoder))
+ (str/join)
+ (#(Integer/parseInt %))
+ (+ sum)
+ )
+ )
+ )
+ 0
+ (str/split-lines (slurp "./in"))
+ )
+ )
+
diff --git a/year2021/day9/part1.clj b/year2021/day9/part1.clj
new file mode 100644
index 0000000..7bd8e23
--- /dev/null
+++ b/year2021/day9/part1.clj
@@ -0,0 +1,21 @@
+(def input-map
+ (->> (slurp "./in")
+ (clojure.string/split-lines)
+ (mapv vec)
+ (mapv (partial mapv #(- (int %) 48)))
+ ))
+
+(defn get-adj [y x]
+ (map (partial get-in input-map)
+ [[(dec y) x] [(inc y) x] [y (dec x)] [y (inc x)]]))
+
+(->> (for [y (range 0 (count input-map))
+ x (range 0 (count (first input-map)))
+ :let [height (get-in input-map [y x])]
+ :when (every? #(or (nil? %) (< height %))
+ (get-adj y x))]
+ (inc height))
+ (apply +)
+ (println)
+ )
+
diff --git a/year2021/day9/part2.clj b/year2021/day9/part2.clj
new file mode 100644
index 0000000..3030c4e
--- /dev/null
+++ b/year2021/day9/part2.clj
@@ -0,0 +1,45 @@
+(require '[clojure.set])
+
+(defn get-adj
+ "Gets vector of coords surrounding (x, y)."
+ [y x] [[(dec y) x] [(inc y) x] [y (dec x)] [y (inc x)]])
+
+(defn basin-continues?
+ "Determines if `b` is a continuation of the basin that includes `a`."
+ [a b] (and (some? b) (not= b 9) (> b a)))
+
+(defn basin-bottom?
+ "Determines if point `p` in `hmap` is the bottom of a basin surrounded by
+ points `adj`."
+ [hmap p adj]
+ (every?
+ #(let [q (get-in hmap %)] (or (nil? q) (< (get-in hmap p) q)))
+ adj
+ ))
+
+(defn find-basin
+ "If point `yx` in `hmap` is in a basin (determined using `adj`), return a
+ list of points in the basin that are above `yx`."
+ [hmap yx adj]
+ (let [res (filter #(basin-continues? (get-in hmap yx)
+ (get-in hmap %))
+ adj)]
+ (apply (partial clojure.set/union res)
+ (map #(set (find-basin hmap % (apply get-adj %))) res)
+ )))
+
+(->> (slurp "./in")
+ (clojure.string/split-lines)
+ (mapv vec)
+ (mapv (partial mapv #(- (int %) 48)))
+ (#(for [y (range 0 (count %))
+ x (range 0 (count (first %)))
+ :let [adj (get-adj y x)]
+ :when (basin-bottom? % [y x] adj)]
+ (inc (count (find-basin % [y x] adj)))))
+ (sort >)
+ (take 3)
+ (apply *)
+ (println)
+ )
+