1use std::collections::HashSet;
2use utils::array::ArrayVec;
3use utils::prelude::*;
4
5#[derive(Clone, Debug)]
12pub struct Day19 {
13 rules: Vec<(Option<Atom>, ArrayVec<Atom, 8>)>,
14 molecule: Vec<Atom>,
15}
16
17parser::parsable_enum! {
18 #[derive(Copy, Clone, Debug, Default, PartialEq, Eq, Hash)]
19 #[repr(u8)]
20 enum Atom {
21 #[default]
22 "Al" => Al,
23 "Ar" => Ar,
24 "B" => B,
25 "Ca" => Ca,
26 "C" => C,
27 "F" => F,
28 "H" => H,
29 "Mg" => Mg,
30 "N" => N,
31 "O" => O,
32 "P" => P,
33 "Rn" => Rn,
34 "Si" => Si,
35 "Th" => Th,
36 "Ti" => Ti,
37 "Y" => Y,
38 }
39}
40
41const _: () = {
42 assert!(Atom::ALL.len() <= 16);
43};
44
45impl Day19 {
46 pub fn new(input: &str, _: InputType) -> Result<Self, InputError> {
47 let Some((rules_str, molecule)) = input.rsplit_once("\n\n") else {
48 return Err(InputError::new(
49 input,
50 0,
51 "expected rules then a blank line then the molecule",
52 ));
53 };
54
55 let rules = Atom::PARSER
56 .map(Some)
57 .or(b'e'.map(|_| None))
58 .with_suffix(" => ")
59 .then(Atom::PARSER.repeat_arrayvec(parser::noop(), 1))
60 .parse_lines(rules_str)?;
61
62 if rules.len() > 64 {
63 return Err(InputError::new(input, rules_str.len(), "too many rules"));
64 }
65
66 Ok(Self {
67 rules,
68 molecule: Atom::PARSER.parse_all(molecule)?,
69 })
70 }
71
72 #[must_use]
73 pub fn part1(&self) -> usize {
74 let mut set = HashSet::new();
75 for (from, to) in &self.rules {
76 let Some(from) = *from else { continue };
77 let new_length = self.molecule.len() + to.len() - 1;
78 for i in 0..self.molecule.len() {
79 if self.molecule[i] == from {
80 let mut molecule = Vec::with_capacity(new_length);
81 molecule.extend_from_slice(&self.molecule[..i]);
82 molecule.extend_from_slice(to);
83 molecule.extend_from_slice(&self.molecule[i + 1..]);
84
85 set.insert(molecule.into_iter().map(|x| x as u8).collect::<Vec<_>>());
91 }
92 }
93 }
94 set.len()
95 }
96
97 #[must_use]
98 pub fn part2(&self) -> u32 {
99 #[derive(Copy, Clone, Debug)]
100 struct State {
101 rule: usize,
102 dot: usize,
103 origin: usize,
104 }
105
106 let mut chart = vec![Vec::new(); self.molecule.len() + 1];
111
112 let mut current_bitset = vec![[0u64; 9]; self.molecule.len() + 1];
116 let mut next_bitset = vec![[0u64; 9]; self.molecule.len() + 1];
117
118 let mut rules_by_lhs = vec![Vec::new(); 16];
121 for (i, (lhs, _)) in self.rules.iter().enumerate() {
122 if let Some(lhs) = *lhs {
123 rules_by_lhs[lhs as usize].push(i);
124 } else {
125 let state = State {
126 rule: i,
127 dot: 0,
128 origin: 0,
129 };
130 current_bitset[state.origin][state.dot] |= 1 << state.rule;
131 chart[0].push((state, 1));
132 }
133 }
134
135 let mut predictions_done = 0u16;
137 let mut completions_done = vec![0u16; self.molecule.len() + 1];
139
140 for pos in 0..chart.len() {
141 let mut set_idx = 0;
142 while let Some(&(state, steps)) = chart[pos].get(set_idx) {
143 let (lhs, rhs) = &self.rules[state.rule];
144
145 if state.dot < rhs.len() {
146 if predictions_done & (1 << rhs[state.dot] as usize) == 0 {
148 predictions_done |= 1 << rhs[state.dot] as usize;
149
150 for &i in &rules_by_lhs[rhs[state.dot] as usize] {
151 let new = State {
152 rule: i,
153 dot: 0,
154 origin: pos,
155 };
156 if current_bitset[new.origin][new.dot] & (1 << new.rule) == 0 {
157 current_bitset[new.origin][new.dot] |= 1 << new.rule;
158 chart[pos].push((new, 1));
159 }
160 }
161 }
162
163 if self.molecule.get(pos) == Some(&rhs[state.dot]) {
165 let new = State {
166 rule: state.rule,
167 dot: state.dot + 1,
168 origin: state.origin,
169 };
170 if next_bitset[new.origin][new.dot] & (1 << new.rule) == 0 {
171 next_bitset[new.origin][new.dot] |= 1 << new.rule;
172 chart[pos + 1].push((new, steps));
173 }
174 }
175 } else if let Some(lhs) = *lhs {
176 if completions_done[state.origin] & (1 << lhs as usize) == 0 {
178 completions_done[state.origin] |= 1 << lhs as usize;
179
180 let [current_chart, origin_chart] = chart
181 .get_disjoint_mut([pos, state.origin])
182 .expect("origin must be less than pos");
183
184 for (prev_state, prev_steps) in origin_chart.iter() {
185 let (_, prev_rhs) = &self.rules[prev_state.rule];
186 if prev_state.dot < prev_rhs.len() && prev_rhs[prev_state.dot] == lhs {
187 let new = State {
188 rule: prev_state.rule,
189 dot: prev_state.dot + 1,
190 origin: prev_state.origin,
191 };
192 if current_bitset[new.origin][new.dot] & (1 << new.rule) == 0 {
193 current_bitset[new.origin][new.dot] |= 1 << new.rule;
194 current_chart.push((new, steps + prev_steps));
195 }
196 }
197 }
198 }
199 } else if pos == self.molecule.len() {
200 return steps;
202 }
203
204 set_idx += 1;
205 }
206
207 (current_bitset, next_bitset) = (next_bitset, current_bitset);
208 next_bitset[..=pos].fill([0; 9]);
209
210 predictions_done = 0u16;
212 completions_done[..pos].fill(0);
213 }
214
215 panic!("no solution found");
216 }
217}
218
219examples!(Day19 -> (usize, u32) [
220 {input: "H => HO\nH => OH\nO => HH\n\nHOH", part1: 4},
221 {input: "e => H\ne => O\nH => HO\nH => OH\nO => HH\n\nHOH", part2: 3},
222 {input: "e => H\ne => O\nH => HO\nH => OH\nO => HH\n\nHOHOHO", part2: 6},
223]);