generated from fspoettel/advent-of-code-rust
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy path17.rs
88 lines (65 loc) · 1.94 KB
/
17.rs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
advent_of_code::solution!(17);
use advent_of_code_macros::memoize;
use advent_of_code::maneatingape::parse::*;
fn parse_data(input: &str) -> (u64, Vec<u8>) {
let (left, right) = input.split_once("\n\n").unwrap();
(left.unsigned(), right.iter_unsigned().collect())
}
#[memoize]
fn run_program(a: u64) -> Vec<u8> {
let mut result = vec![];
let mut a = a;
while a > 0 {
let b = (a ^ 1) & 7;
result.push((b ^ 4 ^ (a >> b) & 7) as u8);
a >>= 3;
}
result
}
pub fn part_one(input: &str) -> Option<String> {
let (a, _) = parse_data(input);
let program_result = run_program(a);
let mut result = String::with_capacity(program_result.len() * 2);
for x in program_result {
result.push((b'0' + x) as char);
result.push(',');
}
result.pop();
Some(result)
}
pub fn part_two(input: &str) -> Option<u64> {
let (_, program) = parse_data(input);
let mut result = u64::MAX;
let mut queue = vec![];
queue.push((0, program.len() - 1));
while let Some((a, i)) = queue.pop() {
let p_i = program[i] as u64;
for part_a in 0..=7 {
let n_a = (a << 3) | (p_i ^ part_a ^ 5) << (part_a ^ 1) | part_a;
if i > 0 && run_program(n_a) == program[i..] {
queue.push((n_a, i - 1));
continue;
}
if i == 0 && run_program(n_a) == program {
result = result.min(n_a);
}
}
}
Some(result)
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn test_part_one() {
let input = advent_of_code::template::read_file("examples", DAY);
let result = part_one(&input);
assert_eq!(result, Some(String::from("5,0,4,5")));
}
#[test]
fn test_part_two() {
let input = advent_of_code::template::read_file("examples", DAY);
let result = part_two(&input);
assert_eq!(result, Some(188468));
}
}