diff options
Diffstat (limited to 'day16')
-rw-r--r-- | day16/Cargo.toml | 9 | ||||
-rw-r--r-- | day16/resources/input.txt | 1 | ||||
-rw-r--r-- | day16/src/main.rs | 239 |
3 files changed, 249 insertions, 0 deletions
diff --git a/day16/Cargo.toml b/day16/Cargo.toml new file mode 100644 index 0000000..da61ec5 --- /dev/null +++ b/day16/Cargo.toml | |||
@@ -0,0 +1,9 @@ | |||
1 | [package] | ||
2 | name = "day16" | ||
3 | version = "0.1.0" | ||
4 | authors = ["Federico Igne <git@federicoigne.com>"] | ||
5 | edition = "2018" | ||
6 | |||
7 | # See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html | ||
8 | |||
9 | [dependencies] | ||
diff --git a/day16/resources/input.txt b/day16/resources/input.txt new file mode 100644 index 0000000..bcdd198 --- /dev/null +++ b/day16/resources/input.txt | |||
@@ -0,0 +1 @@ | |||
4054460802532B12FEE8B180213B19FA5AA77601C010E4EC2571A9EDFE356C7008E7B141898C1F4E50DA7438C011D005E4F6E727B738FC40180CB3ED802323A8C3FED8C4E8844297D88C578C26008E004373BCA6B1C1C99945423798025800D0CFF7DC199C9094E35980253FB50A00D4C401B87104A0C8002171CE31C41201062C01393AE2F5BCF7B6E969F3C553F2F0A10091F2D719C00CD0401A8FB1C6340803308A0947B30056803361006615C468E4200E47E8411D26697FC3F91740094E164DFA0453F46899015002A6E39F3B9802B800D04A24CC763EDBB4AFF923A96ED4BDC01F87329FA491E08180253A4DE0084C5B7F5B978CC410012F9CFA84C93900A5135BD739835F00540010F8BF1D22A0803706E0A47B3009A587E7D5E4D3A59B4C00E9567300AE791E0DCA3C4A32CDBDC4830056639D57C00D4C401C8791162380021108E26C6D991D10082549218CDC671479A97233D43993D70056663FAC630CB44D2E380592FB93C4F40CA7D1A60FE64348039CE0069E5F565697D59424B92AF246AC065DB01812805AD901552004FDB801E200738016403CC000DD2E0053801E600700091A801ED20065E60071801A800AEB00151316450014388010B86105E13980350423F447200436164688A4001E0488AC90FCDF31074929452E7612B151803A200EC398670E8401B82D04E31880390463446520040A44AA71C25653B6F2FE80124C9FF18EDFCA109275A140289CDF7B3AEEB0C954F4B5FC7CD2623E859726FB6E57DA499EA77B6B68E0401D996D9C4292A881803926FB26232A133598A118023400FA4ADADD5A97CEEC0D37696FC0E6009D002A937B459BDA3CC7FFD65200F2E531581AD80230326E11F52DFAEAAA11DCC01091D8BE0039B296AB9CE5B576130053001529BE38CDF1D22C100509298B9950020B309B3098C002F419100226DC \ No newline at end of file | |||
diff --git a/day16/src/main.rs b/day16/src/main.rs new file mode 100644 index 0000000..9f8352c --- /dev/null +++ b/day16/src/main.rs | |||
@@ -0,0 +1,239 @@ | |||
1 | use std::fs; | ||
2 | use std::path::Path; | ||
3 | |||
4 | #[derive(Debug)] | ||
5 | struct BITS { | ||
6 | version: u8, | ||
7 | body: Box<BITSPacket> | ||
8 | } | ||
9 | |||
10 | #[derive(Debug)] | ||
11 | enum BITSPacket { | ||
12 | Literal(u64), | ||
13 | Operator(u8, Vec<Box<BITS>>) | ||
14 | } | ||
15 | |||
16 | /* AOC21 Day 16: https://adventofcode.com/2021/day/16 */ | ||
17 | fn main() { | ||
18 | let input = Path::new("resources").join("input.txt"); | ||
19 | let content = fs::read_to_string(input).expect("Unable to read input file"); | ||
20 | let bits = to_bits(&content); | ||
21 | let (packet,_) = parse_packet(&bits); | ||
22 | println!("Ex1: the result is {}", sum_versions(&packet)); | ||
23 | println!("Ex2: the result is {}", evaluate(&packet)); | ||
24 | } | ||
25 | |||
26 | fn to_bits(input: &str) -> Vec<u8> { | ||
27 | input.chars().fold(vec![], |mut bits,c| { | ||
28 | let c = c.to_digit(16).expect("Malformed input"); | ||
29 | for i in 0..4 { bits.push(((c>>(3-i))&1) as u8) } | ||
30 | bits | ||
31 | }) | ||
32 | } | ||
33 | |||
34 | fn decimal(bits: &[u8]) -> u64 { | ||
35 | // NOTE: using `acc<<1` will potentially overflow *silently*. Use `2*acc` instead. | ||
36 | bits.iter().fold(0, |acc,&b| 2*acc + b as u64) | ||
37 | } | ||
38 | |||
39 | |||
40 | fn parse_packets_by_length(bits: &[u8], len: usize) -> Vec<Box<BITS>> { | ||
41 | let mut vec = vec![]; | ||
42 | let mut cur = 0; | ||
43 | while cur < len { | ||
44 | let (pkg,size) = parse_packet(&bits[cur..]); | ||
45 | vec.push(Box::new(pkg)); | ||
46 | cur += size; | ||
47 | } | ||
48 | vec | ||
49 | } | ||
50 | |||
51 | fn parse_packets_by_number(bits: &[u8], num: usize) -> (Vec<Box<BITS>>, usize) { | ||
52 | let mut vec = vec![]; | ||
53 | let mut cur = 0; | ||
54 | for _ in 0..num { | ||
55 | let (pkg,size) = parse_packet(&bits[cur..]); | ||
56 | vec.push(Box::new(pkg)); | ||
57 | cur += size; | ||
58 | } | ||
59 | (vec,cur) | ||
60 | } | ||
61 | |||
62 | fn parse_packet(bits: &[u8]) -> (BITS, usize) { | ||
63 | let version = decimal(&bits[..3]) as u8; | ||
64 | let typeid = decimal(&bits[3..6]) as u8; | ||
65 | match typeid { | ||
66 | 4 => { | ||
67 | let blocks = (0..).find(|b| bits[6+5*b] == 0).unwrap(); | ||
68 | let lit = (0..=blocks).fold(0, |acc,blk| 16*acc + decimal(&bits[6+5*blk+1..6+5*(blk+1)])); | ||
69 | let body = Box::new(BITSPacket::Literal(lit)); | ||
70 | (BITS { version, body }, 6+5*(blocks+1)) | ||
71 | }, | ||
72 | _ if bits[6] == 0 => { | ||
73 | let len = decimal(&bits[7..22]) as usize; | ||
74 | let subpkgs = parse_packets_by_length(&bits[22..], len); | ||
75 | let body = Box::new(BITSPacket::Operator(typeid, subpkgs)); | ||
76 | (BITS { version, body }, 7+15+len) | ||
77 | }, | ||
78 | _ => { | ||
79 | let num = decimal(&bits[7..18]) as usize; | ||
80 | let (subpkgs,len) = parse_packets_by_number(&bits[18..], num); | ||
81 | let body = Box::new(BITSPacket::Operator(typeid, subpkgs)); | ||
82 | (BITS { version, body }, 7+11+len) | ||
83 | } | ||
84 | } | ||
85 | } | ||
86 | |||
87 | fn sum_versions(packet: &BITS) -> u32 { | ||
88 | packet.version as u32 + match packet.body.as_ref() { | ||
89 | BITSPacket::Literal(_) => 0, | ||
90 | BITSPacket::Operator(_,packets) => packets.iter().map(|p| sum_versions(p)).sum() | ||
91 | } | ||
92 | } | ||
93 | |||
94 | fn evaluate(packet: &BITS) -> u64 { | ||
95 | match packet.body.as_ref() { | ||
96 | BITSPacket::Literal(l) => *l, | ||
97 | BITSPacket::Operator(t,ps) => { | ||
98 | let evals: Vec<u64> = ps.iter().map(|p| evaluate(p)).collect(); | ||
99 | match t { | ||
100 | 0 => evals.iter().sum(), | ||
101 | 1 => evals.iter().product(), | ||
102 | 2 => *evals.iter().min().unwrap(), | ||
103 | 3 => *evals.iter().max().unwrap(), | ||
104 | 5 => (evals[0] > evals[1]) as u64, | ||
105 | 6 => (evals[0] < evals[1]) as u64, | ||
106 | 7 => (evals[0] == evals[1]) as u64, | ||
107 | _ => unreachable!() | ||
108 | } | ||
109 | } | ||
110 | } | ||
111 | } | ||
112 | |||
113 | #[cfg(test)] | ||
114 | mod tests { | ||
115 | use super::*; | ||
116 | |||
117 | #[test] | ||
118 | fn literal() { | ||
119 | let input = "D2FE28"; | ||
120 | let bits = to_bits(input); | ||
121 | let (packet,size) = parse_packet(&bits); | ||
122 | assert_eq!(21, size); | ||
123 | assert_eq!(6, packet.version); | ||
124 | matches!(*packet.body, BITSPacket::Literal(2021)); | ||
125 | } | ||
126 | |||
127 | #[test] | ||
128 | fn bit_literal() { | ||
129 | let input = "3232D42BF9400"; | ||
130 | let bits = to_bits(input); | ||
131 | let (packet,_) = parse_packet(&bits); | ||
132 | assert_eq!(5000000000, evaluate(&packet)); | ||
133 | } | ||
134 | |||
135 | #[test] | ||
136 | fn operator_t0() { | ||
137 | let input = "38006F45291200"; | ||
138 | let bits = to_bits(input); | ||
139 | let (packet,size) = parse_packet(&bits); | ||
140 | assert_eq!(49, size); | ||
141 | assert_eq!(1, packet.version); | ||
142 | matches!(*packet.body, BITSPacket::Operator(6,_)); | ||
143 | if let BITSPacket::Operator(_,packets) = *packet.body { | ||
144 | assert_eq!(2, packets.len()); | ||
145 | let packet = packets[0].as_ref(); | ||
146 | assert_eq!(6, packet.version); | ||
147 | matches!(*packet.body, BITSPacket::Literal(10)); | ||
148 | let packet = packets[1].as_ref(); | ||
149 | assert_eq!(2, packet.version); | ||
150 | matches!(*packet.body, BITSPacket::Literal(20)); | ||
151 | } | ||
152 | } | ||
153 | |||
154 | #[test] | ||
155 | fn operator_t1() { | ||
156 | let input = "EE00D40C823060"; | ||
157 | let bits = to_bits(input); | ||
158 | let (packet,size) = parse_packet(&bits); | ||
159 | assert_eq!(51, size); | ||
160 | assert_eq!(7, packet.version); | ||
161 | matches!(*packet.body, BITSPacket::Operator(3,_)); | ||
162 | if let BITSPacket::Operator(_,packets) = *packet.body { | ||
163 | assert_eq!(3, packets.len()); | ||
164 | let packet = packets[0].as_ref(); | ||
165 | assert_eq!(2, packet.version); | ||
166 | matches!(*packet.body, BITSPacket::Literal(1)); | ||
167 | let packet = packets[1].as_ref(); | ||
168 | assert_eq!(4, packet.version); | ||
169 | matches!(*packet.body, BITSPacket::Literal(2)); | ||
170 | let packet = packets[2].as_ref(); | ||
171 | assert_eq!(1, packet.version); | ||
172 | matches!(*packet.body, BITSPacket::Literal(3)); | ||
173 | } | ||
174 | } | ||
175 | |||
176 | #[test] | ||
177 | fn sum() { | ||
178 | let input = "C200B40A82"; | ||
179 | let bits = to_bits(input); | ||
180 | let (packet,_) = parse_packet(&bits); | ||
181 | assert_eq!(3, evaluate(&packet)); | ||
182 | } | ||
183 | |||
184 | #[test] | ||
185 | fn product() { | ||
186 | let input = "04005AC33890"; | ||
187 | let bits = to_bits(input); | ||
188 | let (packet,_) = parse_packet(&bits); | ||
189 | assert_eq!(54, evaluate(&packet)); | ||
190 | } | ||
191 | |||
192 | #[test] | ||
193 | fn min() { | ||
194 | let input = "880086C3E88112"; | ||
195 | let bits = to_bits(input); | ||
196 | let (packet,_) = parse_packet(&bits); | ||
197 | assert_eq!(7, evaluate(&packet)); | ||
198 | } | ||
199 | |||
200 | #[test] | ||
201 | fn max() { | ||
202 | let input = "CE00C43D881120"; | ||
203 | let bits = to_bits(input); | ||
204 | let (packet,_) = parse_packet(&bits); | ||
205 | assert_eq!(9, evaluate(&packet)); | ||
206 | } | ||
207 | |||
208 | #[test] | ||
209 | fn less_then() { | ||
210 | let input = "D8005AC2A8F0"; | ||
211 | let bits = to_bits(input); | ||
212 | let (packet,_) = parse_packet(&bits); | ||
213 | assert_eq!(1, evaluate(&packet)); | ||
214 | } | ||
215 | |||
216 | #[test] | ||
217 | fn greater_then() { | ||
218 | let input = "F600BC2D8F"; | ||
219 | let bits = to_bits(input); | ||
220 | let (packet,_) = parse_packet(&bits); | ||
221 | assert_eq!(0, evaluate(&packet)); | ||
222 | } | ||
223 | |||
224 | #[test] | ||
225 | fn equals1() { | ||
226 | let input = "9C005AC2F8F0"; | ||
227 | let bits = to_bits(input); | ||
228 | let (packet,_) = parse_packet(&bits); | ||
229 | assert_eq!(0, evaluate(&packet)); | ||
230 | } | ||
231 | |||
232 | #[test] | ||
233 | fn equals2() { | ||
234 | let input = "9C0141080250320F1802104A08"; | ||
235 | let bits = to_bits(input); | ||
236 | let (packet,_) = parse_packet(&bits); | ||
237 | assert_eq!(1, evaluate(&packet)); | ||
238 | } | ||
239 | } | ||