hence/hence/src/lib/emulator.rs

291 lines
8.4 KiB
Rust
Raw Normal View History

2022-09-02 09:33:46 +00:00
use anyhow::{bail, Result};
2022-07-17 18:24:49 +00:00
use itertools::Itertools;
use radix_fmt::radix;
use std::cmp::Ordering;
2022-08-23 14:20:38 +00:00
use std::io::{self, Write};
2022-07-17 18:24:49 +00:00
#[derive(Debug)]
pub struct Data {
program: [u8; 32 * 1024],
pub tmp: u16,
pub reg_pc: u16,
pub reg_opc: u8,
pub reg_arg: u16,
pub reg_s: u8,
pub reg_sp: u16,
pub reg_a: u16,
pub reg_b: u16,
pub reg_c: u16,
pub reg_d: u16,
memory: [u16; 16 * 1024],
2022-08-23 14:20:38 +00:00
term: console::Term,
2022-07-17 18:24:49 +00:00
}
impl Data {
pub fn new(program: [u8; 32 * 1024]) -> Self {
2022-08-26 13:02:48 +00:00
Self {
2022-07-17 18:24:49 +00:00
program,
tmp: 0,
reg_pc: 0,
reg_opc: 0,
reg_arg: 0,
reg_s: 0,
reg_sp: 0,
reg_a: 0,
reg_b: 0,
reg_c: 0,
reg_d: 0,
memory: [0; 16 * 1024],
2022-08-23 14:20:38 +00:00
term: console::Term::stdout(),
2022-08-26 13:02:48 +00:00
}
2022-07-17 18:24:49 +00:00
}
2022-09-02 09:33:46 +00:00
pub fn get_memory(&self, address: u16) -> Result<u16> {
2022-07-17 18:24:49 +00:00
if address < (32 * 1024) {
match self.program.get(address as usize) {
2022-09-02 09:33:46 +00:00
Some(val) => Ok(*val as u16),
None => Ok(0),
2022-07-17 18:24:49 +00:00
}
2022-09-09 12:17:37 +00:00
// } else if address < (40 * 1024) {
// Ok(self.stack[(address - (32 * 1024)) as usize])
} else if address < (48 * 1024) {
2022-09-09 12:52:07 +00:00
Ok(self.memory[(address - (32 * 1024)) as usize])
2022-09-09 12:17:37 +00:00
} else if address == (48 * 1024 + 2) {
2022-09-02 09:33:46 +00:00
Ok(self.term.read_char()? as u16)
2022-07-17 18:24:49 +00:00
} else {
2022-09-02 09:33:46 +00:00
Ok(0)
2022-07-17 18:24:49 +00:00
}
}
2022-09-02 09:33:46 +00:00
pub fn set_memory(&mut self, address: u16, value: u16) -> Result<()> {
2022-09-09 12:17:37 +00:00
// if ((32 * 1024)..(40 * 1024)).contains(&address) {
// self.stack[(address - (32 * 1024)) as usize] = value;
if ((32 * 1024)..(48 * 1024)).contains(&address) {
self.memory[(address - (32 * 1024)) as usize] = value;
} else if address == (48 * 1024) {
print!("{}", value);
2022-09-02 09:33:46 +00:00
io::stdout().flush()?;
2022-09-09 12:17:37 +00:00
} else if address == (48 * 1024 + 1) {
2022-07-17 18:24:49 +00:00
print!("{}", char::from(value as u8));
2022-09-02 09:33:46 +00:00
io::stdout().flush()?;
2022-07-17 18:24:49 +00:00
}
2022-09-02 09:33:46 +00:00
Ok(())
2022-07-17 18:24:49 +00:00
}
pub fn get_register(&self, register: u8) -> u16 {
match register {
0x0 => self.reg_pc,
0x1 => self.reg_opc as u16,
0x2 => self.reg_arg,
0x3 => self.reg_s as u16,
0x4 => self.reg_sp,
0x5 => self.reg_a,
0x6 => self.reg_b,
0x7 => self.reg_c,
0x8 => self.reg_d,
_ => 0,
}
}
pub fn set_register(&mut self, register: u8, value: u16) -> u16 {
match register {
0x0 => {
self.reg_pc = value;
value
}
0x1 => {
self.reg_opc = value as u8;
value & 0xff
}
0x2 => {
self.reg_arg = value;
value
}
0x3 => {
self.reg_s = value as u8;
value & 0xff
}
0x4 => {
self.reg_sp = value;
value
}
0x5 => {
self.reg_a = value;
value
}
0x6 => {
self.reg_b = value;
value
}
0x7 => {
self.reg_c = value;
value
}
0x8 => {
self.reg_d = value;
value
}
_ => 0,
}
}
2022-09-02 09:33:46 +00:00
fn alu(&mut self, operation: u8) -> Result<()> {
2022-07-17 18:24:49 +00:00
match operation {
0x00 => {
self.tmp = !self.reg_a;
}
0x01 => {
self.tmp = self.reg_a & self.reg_b;
}
0x02 => {
self.tmp = self.reg_a | self.reg_b;
}
0x03 => {
self.tmp = self.reg_a ^ self.reg_b;
}
0x04 => {
self.tmp = self.reg_a << self.reg_b;
}
0x05 => {
self.tmp = self.reg_a >> self.reg_b;
}
0x06 => {
self.tmp = self
.reg_a
.wrapping_add(self.reg_b)
.wrapping_add((self.reg_s & 0b00000001) as u16);
self.reg_s &= 0b11111110;
if self.tmp < self.reg_a {
self.tmp |= 0b00000001;
}
}
0x07 => {
self.tmp = self
.reg_a
.wrapping_sub(self.reg_b)
.wrapping_sub((self.reg_s & 0b00000001) as u16);
self.reg_s &= 0b11111110;
if self.tmp > self.reg_a {
self.tmp |= 0b00000001;
}
}
0x08 => {
self.tmp = self.reg_a.wrapping_mul(self.reg_b);
self.reg_s &= 0b11111110;
}
0x09 => {
self.tmp = self.reg_a / self.reg_b;
self.reg_s &= 0b11111110;
}
0x0a => {
self.tmp = match self.reg_a.cmp(&self.reg_b) {
2022-08-23 14:20:38 +00:00
Ordering::Equal => 0,
2022-07-17 18:24:49 +00:00
Ordering::Less => 1,
Ordering::Greater => 2,
}
}
0x0b => {
2022-09-02 09:33:46 +00:00
self.tmp = (self.reg_a == self.reg_b) as u16;
2022-07-17 18:24:49 +00:00
}
0x0c => {
2022-09-02 09:33:46 +00:00
self.tmp = (self.reg_a < self.reg_b) as u16;
2022-07-17 18:24:49 +00:00
}
0x0d => {
2022-09-02 09:33:46 +00:00
self.tmp = (self.reg_a > self.reg_b) as u16;
2022-07-17 18:24:49 +00:00
}
0x0e => {
2022-09-02 09:33:46 +00:00
self.tmp = (self.reg_a <= self.reg_b) as u16;
2022-07-17 18:24:49 +00:00
}
0x0f => {
2022-09-02 09:33:46 +00:00
self.tmp = (self.reg_a >= self.reg_b) as u16;
2022-07-17 18:24:49 +00:00
}
0x10 => {
2022-09-02 09:33:46 +00:00
self.tmp = (self.reg_a & 1 == 1) as u16;
2022-07-17 18:24:49 +00:00
}
0x11 => {
2022-09-02 09:33:46 +00:00
self.tmp = (self.reg_a & 1 != 1) as u16;
2022-07-17 18:24:49 +00:00
}
0x12 => {
self.tmp = rand::random();
}
2022-09-02 09:33:46 +00:00
_ => bail!("Invalid ALU operation: 0x{}", radix(operation, 16)),
2022-07-17 18:24:49 +00:00
}
Ok(())
}
}
2022-09-02 09:33:46 +00:00
pub fn emulate(data: &mut Data) -> Result<()> {
2022-07-17 18:24:49 +00:00
while data.reg_pc != 0xffff {
2022-09-02 09:33:46 +00:00
data.reg_opc = data.get_memory(data.reg_pc)? as u8;
2022-07-17 18:24:49 +00:00
data.reg_pc = data.reg_pc.wrapping_add(1);
if data.reg_opc >> 7 == 1 {
data.reg_opc &= 0b0111111;
data.reg_arg = 0;
} else {
2022-09-02 09:33:46 +00:00
data.reg_arg = data.get_memory(data.reg_pc)? << 8;
2022-07-17 18:24:49 +00:00
data.reg_pc = data.reg_pc.wrapping_add(1);
2022-09-02 09:33:46 +00:00
data.reg_arg |= data.get_memory(data.reg_pc)?;
2022-07-17 18:24:49 +00:00
data.reg_pc = data.reg_pc.wrapping_add(1);
}
match data.reg_opc {
0x00 => {}
0x01 => {
2022-10-26 09:17:19 +00:00
data.tmp = data.reg_arg;
2022-09-09 12:17:37 +00:00
data.memory[data.reg_sp as usize] = data.reg_arg;
2022-07-17 18:24:49 +00:00
data.reg_sp = data.reg_sp.wrapping_add(1);
}
0x02 => {
data.reg_sp = data.reg_sp.wrapping_sub(1);
2022-09-09 12:17:37 +00:00
data.memory[data.reg_sp as usize] = 0;
2022-07-17 18:24:49 +00:00
}
0x03 => {
data.tmp = data.reg_arg;
}
0x04 => {
data.tmp = data.get_register(data.reg_arg as u8);
}
0x05 => {
2022-09-09 12:17:37 +00:00
data.tmp = data.memory[data.reg_sp.wrapping_sub(1) as usize];
2022-07-17 18:24:49 +00:00
}
0x06 => {
data.set_register(data.reg_arg as u8, data.tmp);
}
0x07 => {
if data.reg_a & 1 == 1 {
2022-08-23 14:20:38 +00:00
data.set_register(data.reg_arg as u8, data.tmp);
}
}
0x08 => {
2022-09-09 12:17:37 +00:00
data.memory[data.reg_sp as usize] = data.tmp;
2022-07-17 18:24:49 +00:00
data.reg_sp = data.reg_sp.wrapping_add(1);
}
2022-08-23 14:20:38 +00:00
0x09 => {
2022-07-17 18:24:49 +00:00
println!(
"[DEBUG]: [{}]",
2022-09-09 12:17:37 +00:00
data.memory.iter().take(data.reg_sp as usize).join(", ")
2022-07-17 18:24:49 +00:00
);
2022-08-23 14:20:38 +00:00
print!("Press enter to continue execution...",);
2022-09-02 09:33:46 +00:00
io::stdout().flush()?;
data.term.read_line()?;
2022-07-17 18:24:49 +00:00
}
2022-09-06 09:54:57 +00:00
0x0a => {
2022-09-02 09:33:46 +00:00
data.alu(data.tmp as u8)?;
2022-07-17 18:24:49 +00:00
}
2022-09-06 09:54:57 +00:00
0x0b => {
2022-09-02 09:33:46 +00:00
data.tmp = data.get_memory(data.tmp)?;
2022-07-17 18:24:49 +00:00
}
2022-09-06 09:54:57 +00:00
0x0c => {
2022-09-02 09:33:46 +00:00
data.set_memory(data.tmp, data.reg_a)?;
2022-07-17 18:24:49 +00:00
}
2022-09-02 09:33:46 +00:00
_ => bail!("Invalid opcode: 0x{}", radix(data.reg_opc, 16)),
2022-07-17 18:24:49 +00:00
}
}
Ok(())
}