//const ADD_XY: u8 = 0x8; // //struct CPU { // current_operation: u16, // registers: [u8; 2], //} // //impl CPU { // fn run(&mut self) { // let encoded_op = self.current_operation; // let op = ((encoded_op & 0xf000) >> 12) as u8; // let x = ((encoded_op & 0x0f00) >> 8) as u8; // let y = ((encoded_op & 0x00f0) >> 4) as u8; // // match op { // ADD_XY => { // self.add_xy(x, y); // }, // _ => unimplemented!(), // } // } // // fn add_xy(&mut self, x: u8, y: u8) { // self.registers[x as usize] += self.registers[y as usize]; // } //} // // //fn main() { // let mut cpu = CPU { // current_operation: 0x8014, // registers: [0; 2], // }; // // cpu.registers[0] = 5; // cpu.registers[1] = 10; // cpu.run(); // // assert_eq!(cpu.registers[0], 15); // println!("5 + 10 = {}", cpu.registers[0]); //} //const ARITHMETIC_AND_LOGIC: u8 = 0x8; //const HALT: u8 = 0x0; //const ADD_XY: u8 = 0x4; // //struct CPU { // // current_operation: u16 // registers: [u8; 16], // position_in_memory: usize, // memory: [u8; 4096], //} // //impl CPU { // fn run(&mut self) { // loop { // let op_byte1 = self.memory[self.position_in_memory] as u16; // let op_byte2 = self.memory[self.position_in_memory + 1] as u16; // let raw_op = op_byte1 << 8 | op_byte2; // // let op_major = ((raw_op & 0xf000) >> 12) as u8; // let x = ((raw_op & 0x0f00) >> 8) as u8; // let y = ((raw_op & 0x00f0) >> 4) as u8; // let op_minor = (raw_op & 0x000f) as u8; // // self.position_in_memory += 2; // // match (op_major, op_minor) { // (HALT, HALT) => { return; }, // (ARITHMETIC_AND_LOGIC, ADD_XY) => self.add_xy(x, y), // _ => unimplemented!("opcode: {:04x}", raw_op), // } // } // } // // fn add_xy(&mut self, x: u8, y: u8) { // self.registers[x as usize] += self.registers[y as usize]; // } //} // //fn main() { // let mut cpu = CPU { // registers: [0; 16], // memory: [0; 4096], // position_in_memory: 0, // }; // // cpu.registers[0] = 5; // cpu.registers[1] = 10; // cpu.registers[2] = 10; // cpu.registers[3] = 10; // // cpu.memory[0] = 0x80; // cpu.memory[1] = 0x14; // cpu.memory[2] = 0x80; // cpu.memory[3] = 0x24; // cpu.memory[4] = 0x80; // cpu.memory[5] = 0x34; // // cpu.run(); // // assert_eq!(cpu.registers[0], 35); // println!("5 + 10 + 10 + 10 = {}", cpu.registers[0]); // //} struct CPU { registers: [u8; 16], position_in_memory: usize, memory: [u8; 4096], stack: [u16; 16], stack_pointer: usize, } impl CPU { fn run(&mut self) { loop { let op_byte1 = self.memory[self.position_in_memory] as u16; let op_byte2 = self.memory[self.position_in_memory + 1] as u16; let opcode = op_byte1 << 8 | op_byte2; let x = ((opcode & 0x0f00) >> 8) as u8; let y = ((opcode & 0x00f0) >> 4) as u8; let op_minor = (opcode & 0x000f) as u8; let addr = opcode & 0x0fff; self.position_in_memory += 2; match opcode { 0x0000 => { return; }, 0x00ee => { self.ret(); }, 0x2000...0x2fff => { self.call(addr); }, 0x8000...0x8fff => { match op_minor { 4 => { self.add_xy(x, y); } _ => { unimplemented!("opcode: {:04x}", opcode); }, } }, _ => unimplemented!("opcode {:04x}", opcode), } } } fn call(&mut self, addr: u16) { let sp = self.stack_pointer; let stack = &mut self.stack; if sp > stack.len() { panic!("stack overflow!"); } stack[sp] = self.position_in_memory as u16; self.stack_pointer += 1; self.position_in_memory = addr as usize; } fn ret(&mut self) { if self.stack_pointer == 0 { panic!("stack underflow!"); } self.stack_pointer -= 1; self.position_in_memory = self.stack[self.stack_pointer] as usize; } fn add_xy(&mut self, x: u8, y: u8) { self.registers[x as usize] += self.registers[y as usize]; } } fn main() { let mut cpu = CPU { registers: [0; 16], memory: [0; 4096], position_in_memory: 0, stack: [0; 16], stack_pointer: 0, }; cpu.registers[0] = 5; cpu.registers[1] = 10; cpu.memory[0x000] = 0x21; cpu.memory[0x001] = 0x00; cpu.memory[0x002] = 0x21; cpu.memory[0x003] = 0x00; cpu.memory[0x100] = 0x80; cpu.memory[0x101] = 0x14; cpu.memory[0x102] = 0x80; cpu.memory[0x103] = 0x14; cpu.memory[0x104] = 0x00; cpu.memory[0x105] = 0xee; cpu.run(); assert_eq!(cpu.registers[0], 45); dbg!(cpu.registers[0]); }