summaryrefslogtreecommitdiff
path: root/2019/rust/day11
diff options
context:
space:
mode:
Diffstat (limited to '2019/rust/day11')
-rwxr-xr-x2019/rust/day11/both_parts.sh3
-rw-r--r--2019/rust/day11/src/main.rs410
2 files changed, 413 insertions, 0 deletions
diff --git a/2019/rust/day11/both_parts.sh b/2019/rust/day11/both_parts.sh
new file mode 100755
index 0000000..c9ed7ef
--- /dev/null
+++ b/2019/rust/day11/both_parts.sh
@@ -0,0 +1,3 @@
+../target/release/day11 --program input
+../target/release/day11 --program input --use-emergency-panel
+
diff --git a/2019/rust/day11/src/main.rs b/2019/rust/day11/src/main.rs
new file mode 100644
index 0000000..6add91a
--- /dev/null
+++ b/2019/rust/day11/src/main.rs
@@ -0,0 +1,410 @@
+use std::env;
+use std::io::{BufRead, BufReader};
+use std::fs::File;
+use std::sync::mpsc::{channel, Receiver, Sender};
+use std::thread;
+use std::collections::{HashMap};
+
+type BusWidth = i128;
+type IOType = i128;
+
+fn read_program(filename: &str) -> Vec<BusWidth> {
+ let f = File::open(filename).expect(filename);
+ let f = BufReader::new(f);
+
+ f.lines()
+ .filter(|line| match line {
+ Ok(s) => {
+ let b = s.as_bytes();
+ if b.len() == 0 || b.len() > 0 && b[0] == b'#' { false } else { true }
+ },
+ _ => true,
+ })
+ .map(|line| match line {
+ Ok(line) => {
+ let data:Vec<BusWidth> = line.split(',').map(|e| { e.trim_start().trim_end() })
+ .filter(|v| match v.parse::<BusWidth>() { Ok(_) => true, _ => {
+ eprintln!("Could not parse: {:?}", v);
+ false
+ },
+ })
+ .map(|r| r.parse().unwrap())
+ .collect();
+
+ data
+ },
+ Err(_) => panic!("Could not parse {}.", filename),
+ }).flatten().collect()
+}
+
+#[derive(Clone, Debug)]
+struct Memory {
+ mem: Vec<BusWidth>,
+}
+
+impl Memory {
+ fn new() -> Memory {
+ Memory { mem: Vec::new(), }
+ }
+
+ fn init(&mut self, mem:Vec<BusWidth>) {
+ self.mem = mem;
+ }
+
+ fn read(&self, address: usize) -> BusWidth {
+ match self.mem.get(address) {
+ Some(value) => *value,
+ None => 0,
+ }
+ }
+
+ fn write(&mut self, address: usize, value: BusWidth) {
+ if self.len() < address {
+ self.mem.resize(address*2, 0);
+ }
+ self.mem[address] = value;
+ }
+
+ fn len(&self) -> usize {
+ self.mem.len()
+ }
+}
+
+#[derive(Debug)]
+struct Registers {
+ ip: usize,
+ base: usize,
+}
+
+#[derive(Debug)]
+enum Arithm {
+ Add,
+ Mul,
+ LessThan,
+ Equals,
+}
+
+#[derive(Debug)]
+enum IO {
+ In,
+ Out,
+}
+
+#[derive(Debug)]
+enum Jmp {
+ JmpTrue,
+ JmpFalse,
+}
+
+#[derive(Debug)]
+enum RegOp {
+ BaseMod,
+}
+
+#[derive(Debug, PartialEq)]
+enum ParameterMode {
+ Position,
+ Immediate,
+ Relative,
+}
+
+#[derive(Debug, PartialEq)]
+enum DataDirection {
+ Read,
+ Write,
+}
+
+struct Intcode {
+ r: Registers,
+ sender: Sender<IOType>,
+ receiver: Receiver<IOType>,
+}
+
+struct Robot {
+ canvas: HashMap<(isize, isize), u8>,
+ sender: Sender<IOType>,
+ receiver: Receiver<IOType>,
+}
+
+enum Direction {
+ Up,
+ Down,
+ Left,
+ Right,
+}
+
+impl Robot {
+ fn new(sender: Sender<IOType>, receiver: Receiver<IOType>) -> Robot {
+ Robot {
+ canvas: HashMap::new(),
+ sender,
+ receiver,
+ }
+ }
+
+ fn run(&mut self) -> usize {
+ let mut x = 0;
+ let mut y = 0;
+ let mut direction = Direction::Up;
+ let mut painted = 0;
+
+ loop {
+ match self.sender.send(match self.canvas.get(&(x, y)) { Some(1) => 1, _ => 0, }) {
+ Ok(_) => {}, _ => break,
+ }
+ let (brush, turn) = match self.receiver.iter().zip(self. receiver.iter()).next() {
+ Some((b, t)) => (b as u8, t), _ => break,
+ };
+
+ match self.canvas.insert((x, y), brush as u8) {
+ None => painted += 1,
+ _ => {},
+ }
+
+ self.propulse(turn, &mut x, &mut y, &mut direction);
+ }
+
+ painted
+ }
+
+ fn place_emergency_panel(&mut self) {
+ self.canvas.insert((0, 0), 1);
+ }
+
+ fn propulse(&self, turn: IOType, x: &mut isize, y: &mut isize, direction: &mut Direction) {
+ *direction = if turn == 0 {
+ match direction {
+ Direction::Up => Direction::Left,
+ Direction::Left => Direction::Down,
+ Direction::Down => Direction::Right,
+ Direction::Right => Direction::Up,
+ }
+ } else if turn == 1 {
+ match direction {
+ Direction::Up => Direction::Right,
+ Direction::Right => Direction::Down,
+ Direction::Down => Direction::Left,
+ Direction::Left => Direction::Up,
+ }
+ } else {
+ panic!("Invalid turn instruction: {}", turn);
+ };
+
+ match direction {
+ Direction::Up => *y -= 1,
+ Direction::Right => *x += 1,
+ Direction::Down => *y += 1,
+ Direction::Left => *x -= 1,
+ }
+ }
+
+ fn show_painted_panels(&self) {
+ let boundary_msg = &"Could not determine plot boundary";
+
+ let min_x = *self.canvas.keys().map(|(x, _)| x).min().expect(boundary_msg);
+ let max_x = *self.canvas.keys().map(|(x, _)| x).max().expect(boundary_msg);
+ let min_y = *self.canvas.keys().map(|(_, y)| y).min().expect(boundary_msg);
+ let max_y = *self.canvas.keys().map(|(_, y)| y).max().expect(boundary_msg);
+ for y in min_y..=max_y {
+ for x in min_x..=max_x {
+ print!("{}", match self.canvas.get(&(x, y)) {
+ Some(1) => "██",
+ _ => " ",
+ });
+ }
+ println!("");
+ }
+ println!("");
+ }
+}
+
+impl Intcode {
+ fn new(sender: Sender<IOType>, receiver: Receiver<IOType>) -> Intcode {
+ Intcode {
+ r: Registers { ip: 0, base: 0, },
+ sender,
+ receiver,
+ }
+ }
+
+ fn param_to_argument(&self, mode: &ParameterMode, mem: &Memory, offset: usize,
+ dir: &DataDirection) -> usize
+ {
+ match mode {
+ ParameterMode::Position => {
+ if mem.read(offset) < 0 {
+ panic!("Attempt to address negative memory at IP: {}");
+ }
+ mem.read(offset) as usize
+ },
+ ParameterMode::Immediate => {
+ if *dir == DataDirection::Write {
+ panic!("Immediate parameter mode at IP: {}", self.r.ip);
+ }
+ offset
+ },
+ ParameterMode::Relative => {
+ if (mem.read(offset) + self.r.base as BusWidth) < 0 {
+ panic!("Attempt to address negative memory at IP: {}");
+ }
+ (mem.read(offset) + self.r.base as BusWidth) as usize
+ },
+ }
+ }
+
+ fn arithm(&self, op: Arithm, modes: Vec<ParameterMode>, memory: &mut Memory) {
+ let parameter_directions =
+ [DataDirection::Read, DataDirection::Read, DataDirection::Write];
+ let params:Vec<usize> = (0..parameter_directions.len())
+ .zip(parameter_directions.iter())
+ .map(|(p, d)| {
+ self.param_to_argument(&modes[p], memory, self.r.ip + 1 + p, d)
+ }).collect();
+ let left = params[0];
+ let right = params[1];
+ let result = params[2];
+
+ match op {
+ Arithm::Add => memory.write(result, memory.read(left) + memory.read(right)),
+ Arithm::Mul => memory.write(result, memory.read(left) * memory.read(right)),
+ Arithm::LessThan => memory.write(result, if memory.read(left) < memory.read(right) {
+ 1 } else { 0 }),
+ Arithm::Equals => memory.write(result, if memory.read(left) == memory.read(right) {
+ 1 } else { 0 }),
+ }
+ }
+
+ fn jmp(&mut self, op: Jmp, modes: Vec<ParameterMode>, memory: &mut Memory)
+ {
+ let parameter_directions = [DataDirection::Read, DataDirection::Read];
+ let params:Vec<usize> = (0..parameter_directions.len())
+ .zip(parameter_directions.iter())
+ .map(|(p, d)| {
+ self.param_to_argument(&modes[p], memory, self.r.ip + 1 + p, d)
+ }).collect();
+ let value = params[0];
+ let address = params[1];
+
+ self.r.ip = match op {
+ Jmp::JmpTrue => (if memory.read(value) != 0 {
+ memory.read(address) as usize } else { self.r.ip + 3 }),
+ Jmp::JmpFalse => (if memory.read(value) == 0 {
+ memory.read(address) as usize } else { self.r.ip + 3 }),
+ };
+
+ }
+
+ fn regop(&mut self, op: RegOp, modes: Vec<ParameterMode>, memory: &mut Memory)
+ {
+ let parameter_directions = [DataDirection::Read];
+ let params:Vec<usize> = (0..parameter_directions.len())
+ .zip(parameter_directions.iter())
+ .map(|(p, d)| {
+ self.param_to_argument(&modes[p], memory, self.r.ip + 1 + p, d)
+ }).collect();
+ let value = memory.read(params[0]);
+
+ match op {
+ RegOp::BaseMod => {
+ if value < 0 {
+ self.r.base -= (-1 * value) as usize;
+ } else {
+ self.r.base += value as usize;
+ }
+ },
+ };
+ }
+
+ fn io_in(&mut self, _op: IO, modes: Vec<ParameterMode>, memory:&mut Memory)
+ {
+ let parameter_directions = [DataDirection::Write];
+ let params:Vec<usize> = (0..parameter_directions.len())
+ .zip(parameter_directions.iter())
+ .map(|(p, d)| {
+ self.param_to_argument(&modes[p], memory, self.r.ip + 1 + p, d)
+ }).collect();
+ let address = params[0];
+
+ let value = self.receiver.recv();
+ match value {
+ Ok(v) => memory.write(address, v),
+ _ => panic!("Could not receive input. Remote disappeared."),
+ }
+ }
+
+ fn io_out(&mut self, _op: IO, modes: Vec<ParameterMode>, memory:&mut Memory )
+ {
+ let parameter_directions = [DataDirection::Read];
+ let params:Vec<usize> = (0..parameter_directions.len())
+ .zip(parameter_directions.iter())
+ .map(|(p, d)| {
+ self.param_to_argument(&modes[p], memory, self.r.ip + 1 + p, d)
+ }).collect();
+ let address = params[0];
+
+ let o = memory.read(address);
+// println!("{}", o);
+ self.sender.send(o as IOType).expect("Communication error. Nothing connected to CPU?");
+ }
+
+ fn intcode(&mut self, mut memory: Memory)
+ {
+ while self.r.ip < memory.len() {
+ let op = memory.read(self.r.ip) % 100;
+ let modes:Vec<ParameterMode> = (2..=4)
+ .map(|e| memory.read(self.r.ip) / 10i32.pow(e) as BusWidth % 10)
+ .map(|n| match n {
+ 0 => ParameterMode::Position,
+ 1 => ParameterMode::Immediate,
+ 2 => ParameterMode::Relative,
+ p => panic!("Invalid parameter mode {} at IP: {}", p, self.r.ip),
+ }).collect();
+
+ match op {
+ 1 => { self.arithm(Arithm::Add, modes, &mut memory); self.r.ip += 4 },
+ 2 => { self.arithm(Arithm::Mul, modes, &mut memory); self.r.ip += 4 },
+ 7 => { self.arithm(Arithm::LessThan, modes, &mut memory); self.r.ip += 4 },
+ 8 => { self.arithm(Arithm::Equals, modes, &mut memory); self.r.ip += 4 },
+ 3 => { self.io_in( IO::In, modes, &mut memory); self.r.ip += 2 },
+ 4 => { self.io_out(IO::Out, modes, &mut memory); self.r.ip += 2 },
+ 5 => { self.jmp( Jmp::JmpTrue, modes, &mut memory); },
+ 6 => { self.jmp( Jmp::JmpFalse, modes, &mut memory); },
+ 9 => { self.regop(RegOp::BaseMod, modes, &mut memory); self.r.ip += 2},
+ 99 => break,
+ op => panic!("Invalid operation {} at IP: {}", op, self.r.ip),
+ }
+ }
+ }
+}
+
+fn main() {
+ let args: Vec<String> = env::args().collect();
+
+ let mut opts = getopts::Options::new();
+ opts.optflag("e", "use-emergency-panel", "Place white panel at (0,0)");
+ opts.reqopt("p", "program", "File to load program from", "FILE");
+ let matches = match opts.parse(&args[1..]) {
+ Ok(m) => { m }
+ Err(f) => { panic!(f.to_string()) }
+ };
+ let program: String = matches.opt_str("program").unwrap();
+ let mut memory = Memory::new();
+ memory.init(read_program(&program));
+ let ( to_cpu, from_robot ): (Sender<IOType>, Receiver<IOType>) = channel();
+ let ( to_robot, from_cpu ): (Sender<IOType>, Receiver<IOType>) = channel();
+ let mut cpu = Intcode::new(to_cpu, from_cpu);
+ let mut robot = Robot::new(to_robot, from_robot);
+ if matches.opt_present("use-emergency-panel") {
+ robot.place_emergency_panel();
+ }
+
+ let cpu_handler = thread::spawn(move || cpu.intcode(memory.clone()));
+ let robot_handler = thread::spawn(move || {
+ let n = robot.run();
+ robot.show_painted_panels();
+ n
+ });
+
+ cpu_handler.join().expect("Problem with cpu thread.");
+ println!("Panes painted: {:?}", robot_handler.join().unwrap());
+}