summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--day16/Cargo.toml9
-rw-r--r--day16/resources/input.txt1
-rw-r--r--day16/src/main.rs239
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]
2name = "day16"
3version = "0.1.0"
4authors = ["Federico Igne <git@federicoigne.com>"]
5edition = "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 @@
1use std::fs;
2use std::path::Path;
3
4#[derive(Debug)]
5struct BITS {
6 version: u8,
7 body: Box<BITSPacket>
8}
9
10#[derive(Debug)]
11enum BITSPacket {
12 Literal(u64),
13 Operator(u8, Vec<Box<BITS>>)
14}
15
16/* AOC21 Day 16: https://adventofcode.com/2021/day/16 */
17fn 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
26fn 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
34fn 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
40fn 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
51fn 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
62fn 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
87fn 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
94fn 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)]
114mod 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}