From 119ce4405f437e9a6a52861e52a9c30b3f70b40f Mon Sep 17 00:00:00 2001 From: Erin <erin@erindesu.cz> Date: Sun, 28 May 2023 16:49:01 +0200 Subject: [PATCH] Changed register handling --- hbvm/Cargo.toml | 2 - hbvm/src/engine/mod.rs | 24 ++++++----- hbvm/src/engine/regs.rs | 93 ++++++++++++++++++++++++++++------------- hbvm/src/main.rs | 11 ++--- hbvm/src/memory.rs | 2 +- 5 files changed, 84 insertions(+), 48 deletions(-) diff --git a/hbvm/Cargo.toml b/hbvm/Cargo.toml index 92f7c089..bac4d36e 100644 --- a/hbvm/Cargo.toml +++ b/hbvm/Cargo.toml @@ -3,8 +3,6 @@ name = "hbvm" version = "0.1.0" edition = "2021" -# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html - [dependencies] log = "*" hashbrown = "0.13.2" diff --git a/hbvm/src/engine/mod.rs b/hbvm/src/engine/mod.rs index 595929da..4370689d 100644 --- a/hbvm/src/engine/mod.rs +++ b/hbvm/src/engine/mod.rs @@ -1,3 +1,5 @@ +use log::info; + pub mod call_stack; pub mod config; pub mod enviroment_calls; @@ -7,12 +9,13 @@ pub mod tests; use { self::call_stack::CallStack, - crate::{memory, HaltStatus, RuntimeErrors}, + crate::{engine::enviroment_calls::EnviromentCall, memory, HaltStatus, RuntimeErrors}, alloc::vec::Vec, config::EngineConfig, log::trace, regs::Registers, }; + // pub const PAGE_SIZE: usize = 8192; pub struct RealPage { @@ -23,8 +26,8 @@ pub struct RealPage { pub struct VMPage { pub data: [u8; 8192], } -impl VMPage { - pub fn new() -> Self { +impl Default for VMPage { + fn default() -> Self { Self { data: [0; 4096 * 2], } @@ -52,10 +55,10 @@ pub fn empty_enviroment_call(engine: &mut Engine) -> Result<&mut Engine, u64> { } pub struct Engine { - pub index: usize, - pub program: Vec<u8>, + pub pc: usize, + pub program: Vec<u8>, pub registers: Registers, - pub config: EngineConfig, + pub config: EngineConfig, /// BUG: This DOES NOT account for overflowing pub last_timer_count: u32, @@ -64,12 +67,11 @@ pub struct Engine { pub enviroment_call_table: [Option<EnviromentCall>; 256], pub call_stack: CallStack, } -use crate::engine::enviroment_calls::EnviromentCall; + impl Engine { pub fn set_timer_callback(&mut self, func: fn() -> u32) { self.timer_callback = Some(func); } - pub fn set_register(&mut self, register: u8, value: u64) {} } impl Engine { @@ -81,9 +83,9 @@ impl Engine { trace!("{:?}", mem.read_addr8(0)); let ecall_table: [Option<EnviromentCall>; 256] = [None; 256]; Self { - index: 0, + pc: 0, program, - registers: Registers::new(), + registers: Registers::default(), config: EngineConfig::default(), last_timer_count: 0, timer_callback: None, @@ -95,6 +97,6 @@ impl Engine { pub fn dump(&self) {} pub fn run(&mut self) -> Result<HaltStatus, RuntimeErrors> { - Ok(HaltStatus::Halted) + Ok(HaltStatus::Running) } } diff --git a/hbvm/src/engine/regs.rs b/hbvm/src/engine/regs.rs index 11d51c0a..5600f98a 100644 --- a/hbvm/src/engine/regs.rs +++ b/hbvm/src/engine/regs.rs @@ -1,32 +1,67 @@ -#[rustfmt::skip] -#[derive(Debug, Clone, Copy)] -pub struct Registers { - pub a0: u64, pub b0: u64, pub c0: u64, pub d0: u64, pub e0: u64, pub f0: u64, - pub a1: u64, pub b1: u64, pub c1: u64, pub d1: u64, pub e1: u64, pub f1: u64, - pub a2: u64, pub b2: u64, pub c2: u64, pub d2: u64, pub e2: u64, pub f2: u64, - pub a3: u64, pub b3: u64, pub c3: u64, pub d3: u64, pub e3: u64, pub f3: u64, - pub a4: u64, pub b4: u64, pub c4: u64, pub d4: u64, pub e4: u64, pub f4: u64, - pub a5: u64, pub b5: u64, pub c5: u64, pub d5: u64, pub e5: u64, pub f5: u64, - pub a6: u64, pub b6: u64, pub c6: u64, pub d6: u64, pub e6: u64, pub f6: u64, - pub a7: u64, pub b7: u64, pub c7: u64, pub d7: u64, pub e7: u64, pub f7: u64, - pub a8: u64, pub b8: u64, pub c8: u64, pub d8: u64, pub e8: u64, pub f8: u64, - pub a9: u64, pub b9: u64, pub c9: u64, pub d9: u64, pub e9: u64, pub f9: u64, -} +use core::{ + fmt::Debug, + ops::{Index, IndexMut}, +}; -impl Registers { - #[rustfmt::skip] - pub fn new() -> Self{ - Self { - a0: 0, b0: 0, c0: 0, d0: 0, e0: 0, f0: 0, - a1: 0, b1: 0, c1: 0, d1: 0, e1: 0, f1: 0, - a2: 0, b2: 0, c2: 0, d2: 0, e2: 0, f2: 0, - a3: 0, b3: 0, c3: 0, d3: 0, e3: 0, f3: 0, - a4: 0, b4: 0, c4: 0, d4: 0, e4: 0, f4: 0, - a5: 0, b5: 0, c5: 0, d5: 0, e5: 0, f5: 0, - a6: 0, b6: 0, c6: 0, d6: 0, e6: 0, f6: 0, - a7: 0, b7: 0, c7: 0, d7: 0, e7: 0, f7: 0, - a8: 0, b8: 0, c8: 0, d8: 0, e8: 0, f8: 0, - a9: 0, b9: 0, c9: 0, d9: 0, e9: 0, f9: 0, - } +#[derive(Debug, Clone, Copy)] +pub struct Registers([Value; 60]); + +impl Index<u8> for Registers { + type Output = Value; + + #[inline] + fn index(&self, index: u8) -> &Self::Output { + &self.0[index as usize] + } +} + +impl IndexMut<u8> for Registers { + #[inline] + fn index_mut(&mut self, index: u8) -> &mut Self::Output { + &mut self.0[index as usize] + } +} + +impl Default for Registers { + fn default() -> Self { + Self([Value { u: 0 }; 60]) + } +} + +/// # Safety +/// The macro invoker shall make sure that byte reinterpret-cast +/// won't cause undefined behaviour. +macro_rules! value_def { + ($($fname:ident : $fty:ident, $getter:ident);* $(;)?) => { + #[derive(Clone, Copy)] + pub union Value { + $($fname: $fty),* + } + + impl Value {$( + #[inline] + pub fn $getter(&self) -> $fty { + unsafe { self.$fname } + } + )*} + + $(impl From<$fty> for Value { + #[inline] + fn from($fname: $fty) -> Self { + Self { $fname } + } + })* + } +} + +value_def! { + u: u64, unsigned; + s: i64, signed; + f: f64, float; +} + +impl Debug for Value { + fn fmt(&self, f: &mut core::fmt::Formatter<'_>) -> core::fmt::Result { + self.unsigned().fmt(f) } } diff --git a/hbvm/src/main.rs b/hbvm/src/main.rs index 61494688..efc7ef2b 100644 --- a/hbvm/src/main.rs +++ b/hbvm/src/main.rs @@ -1,21 +1,21 @@ use hbvm::{ - bytecode::ops::{Operations::*, RWSubTypes::*}, + bytecode::ops::{opcode::*}, engine::Engine, - RuntimeErrors, + RuntimeErrors, HaltStatus, }; fn main() -> Result<(), RuntimeErrors> { // TODO: Grab program from cmdline #[rustfmt::skip] let prog: Vec<u8> = vec![ - NOP as u8, - JUMP as u8, 0, 0, 0, 0, 0, 0, 0, 0, + NOP, + JUMP, 0, 0, 0, 0, 0, 0, 0, 0, ]; let mut eng = Engine::new(prog); // eng.set_timer_callback(time); eng.enviroment_call_table[10] = Some(print_fn); - eng.run()?; + while eng.run()? != HaltStatus::Halted {} eng.dump(); println!("{:#?}", eng.registers); @@ -25,6 +25,7 @@ fn main() -> Result<(), RuntimeErrors> { pub fn time() -> u32 { 9 } + pub fn print_fn(engine: &mut Engine) -> Result<&mut Engine, u64> { println!("hello"); Ok(engine) diff --git a/hbvm/src/memory.rs b/hbvm/src/memory.rs index 5d7425d9..1b1ff8af 100644 --- a/hbvm/src/memory.rs +++ b/hbvm/src/memory.rs @@ -52,7 +52,7 @@ impl Memory { page.data()[offset as usize] = value; } None => { - let mut pg = VMPage::new(); + let mut pg = VMPage::default(); pg.data[offset as usize] = value; self.inner.insert(page, Page::VMPage(pg)); trace!("Mapped page {}", page);