burritos/src/kernel/thread.rs

117 lines
3.3 KiB
Rust
Raw Normal View History

2023-04-11 17:47:36 +02:00
use std::{rc::Rc, cell::RefCell};
2023-04-12 15:32:46 +02:00
use super::process::Process;
2023-03-08 15:48:33 +01:00
use crate::{simulator::machine::{NUM_INT_REGS, NUM_FP_REGS, STACK_REG}};
2023-02-28 14:43:40 +01:00
2023-03-06 16:31:35 +01:00
const STACK_FENCEPOST: u32 = 0xdeadbeef;
2023-02-28 14:43:40 +01:00
/// Polymorphic macro to get thread without passing a name by default
#[macro_export]
macro_rules! get_new_thread {
() => { Thread::new(DEFAULT_THREAD_NAME) };
($a:literal) => {
Thread::new(&$a.to_string())
};
}
#[derive(PartialEq, Debug)]
pub struct ThreadContext {
2023-02-28 14:43:40 +01:00
pub int_registers: [i64; NUM_INT_REGS],
pub float_registers: [f32; NUM_FP_REGS],
pub pc: u64,
2023-02-28 14:43:40 +01:00
}
#[derive(PartialEq, Debug)]
2023-02-28 14:43:40 +01:00
pub struct Thread {
name: String,
2023-04-11 17:47:36 +02:00
pub process: Option<Rc<RefCell<Process>>>,
pub thread_context: ThreadContext,
pub stack_pointer: i32,
2023-02-28 14:43:40 +01:00
}
impl Thread {
2023-03-08 15:48:33 +01:00
/// Thread constructor
pub fn new(name: &str) -> Self {
2023-02-28 14:43:40 +01:00
Self {
name: String::from(name),
2023-02-28 14:43:40 +01:00
process: None,
2023-03-06 16:31:35 +01:00
// simulation_context: UContextT::new(),
2023-02-28 14:43:40 +01:00
thread_context: ThreadContext {
int_registers: [0; NUM_INT_REGS],
float_registers: [0f32; NUM_FP_REGS],
2023-02-28 14:43:40 +01:00
pc: 0
},
stack_pointer: 0,
}
}
pub fn init_thread_context(&mut self, initial_pc_reg: u64, initial_sp: u64, arg: i64) {
2023-03-01 15:45:49 +01:00
self.thread_context.pc = initial_pc_reg;
self.thread_context.int_registers[10] = arg;
self.thread_context.int_registers[STACK_REG] = initial_sp as i64;
2023-03-01 15:45:49 +01:00
}
2023-02-28 14:43:40 +01:00
/// Check if a thread has overflowed its stack
///
/// This assertion doesn't catch all stack overflow conditions and your program may still crash because of an overflow.
///
2023-02-28 14:43:40 +01:00
pub fn check_overflow(&self) {
// if self.simulator_context.stackBottom != STACK_FENCEPOST {
// panic!("thread {} has overflowed", self.get_name())
// }
2023-02-28 14:43:40 +01:00
}
pub fn get_name(&self) -> String {
self.name.clone()
}
2023-04-11 17:47:36 +02:00
/// Return reference to an optional Process
/// can be None if Thread hasn't been initialize
pub fn get_process_owner(&self) -> &Option<Rc<RefCell<Process>>> {
&self.process
}
2023-03-01 15:45:49 +01:00
}
#[cfg(test)]
mod test {
2023-04-12 15:32:46 +02:00
use super::{Thread, ThreadContext, NUM_INT_REGS, NUM_FP_REGS};
2023-03-09 13:33:00 +01:00
const DEFAULT_THREAD_NAME: &str = "test_thread";
2023-03-09 14:03:35 +01:00
/// This macro allows for getting a Thread for which we've ensured proper initial state
/// in case a commit further down the line changes the initial state of threads generated
/// from Thread::new
macro_rules! expected_initial_state {
2023-03-09 13:33:00 +01:00
() => { expected_initial_state!(DEFAULT_THREAD_NAME) };
($a:expr) => { {
let mut x = Thread::new($a);
x.name = $a.to_string();
x.process = Option::None;
x.thread_context = ThreadContext {
int_registers: [0; NUM_INT_REGS],
float_registers: [0f32; NUM_FP_REGS],
pc: 0
};
x.stack_pointer = 0;
x }
};
}
#[test]
fn test_macro() {
let t = get_new_thread!("hello");
assert_eq!(t.get_name(), "hello");
let t = get_new_thread!(1);
assert_eq!(t.get_name(), "1");
}
#[test]
fn check_init() {
let t = get_new_thread!();
let expected_state = expected_initial_state!();
assert_eq!(t, expected_state)
}
2023-02-28 14:43:40 +01:00
}