use std::collections::HashSet; use std::fmt::Write; use aoc::err; const INPUT: &str = include_str!("../input/day08.txt"); pub fn run() -> aoc::Result { let mut res = String::with_capacity(128); writeln!(res, "part 1: {}", part1(INPUT)?)?; Ok(res) } fn part1(input: &str) -> aoc::Result { let instructions = input .lines() .map(|line| line.parse()) .collect::>>()?; let mut interpreter = Interpreter::new(instructions); let mut set = HashSet::new(); loop { if !set.insert(interpreter.idx) { break; } interpreter.step(); } Ok(interpreter.accumulator) } struct Interpreter { idx: usize, accumulator: i64, memory: Vec, } impl Interpreter { fn new(instructions: Vec) -> Self { Self { idx: 0, accumulator: 0, memory: instructions, } } fn step(&mut self) { match self.memory[self.idx] { Instruction::Acc(arg) => { self.accumulator += arg; self.idx += 1; } Instruction::Jmp(offset) => self.idx = self.idx.wrapping_add(offset as usize), Instruction::Nop => self.idx += 1, } } } enum Instruction { Acc(i64), Jmp(i64), Nop, } impl std::str::FromStr for Instruction { type Err = aoc::Error; fn from_str(s: &str) -> aoc::Result { let space = s.find(' ').ok_or_else(|| err!("couldn't split on space"))?; let inst = &s[..space]; let arg = s[(space + 1)..] .parse() .map_err(|e| err!("couldn't parse argument for instruction: {}", e))?; Ok(match inst { "acc" => Self::Acc(arg), "jmp" => Self::Jmp(arg), "nop" => Self::Nop, _ => return Err(err!("unrecognized instruction `{}`", inst)), }) } } #[cfg(test)] mod tests { use super::*; const PROVIDED: &str = include_str!("../input/day08_provided.txt"); #[test] fn part1_provided() { assert_eq!(part1(PROVIDED).unwrap(), 5); } #[test] fn part1_real() { assert_eq!(part1(INPUT).unwrap(), 1675); } }