2023-03-09 12:08:33 +01:00
|
|
|
use super::{process::Process, system::ObjectType, thread_manager::SIMULATORSTACKSIZE};
|
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
|
|
|
|
2023-03-01 11:10:15 +01:00
|
|
|
#[derive(PartialEq)]
|
2023-03-09 12:08:33 +01:00
|
|
|
pub struct ThreadContext {
|
2023-02-28 14:43:40 +01:00
|
|
|
pub int_registers: [i64; NUM_INT_REGS],
|
2023-03-09 12:08:33 +01:00
|
|
|
pub float_registers: [f32; NUM_FP_REGS],
|
2023-02-28 14:43:40 +01:00
|
|
|
pc: i64,
|
|
|
|
}
|
|
|
|
|
2023-03-01 11:10:15 +01:00
|
|
|
#[derive(PartialEq)]
|
2023-02-28 14:43:40 +01:00
|
|
|
pub struct Thread {
|
|
|
|
name: String,
|
2023-03-08 21:10:51 +01:00
|
|
|
pub process: Option<Process>,
|
2023-03-06 16:31:35 +01:00
|
|
|
// simulation_context: UContextT,
|
2023-03-09 12:08:33 +01:00
|
|
|
pub thread_context: ThreadContext,
|
|
|
|
pub stack_pointer: i32,
|
2023-03-08 21:10:51 +01:00
|
|
|
object_type: ObjectType
|
2023-02-28 14:43:40 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
impl Thread {
|
|
|
|
|
2023-03-08 15:48:33 +01:00
|
|
|
/// Thread constructor
|
2023-03-09 12:44:03 +01:00
|
|
|
pub fn new(name: &str) -> Self {
|
2023-02-28 14:43:40 +01:00
|
|
|
Self {
|
2023-03-09 12:44:03 +01:00
|
|
|
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],
|
2023-03-09 12:08:33 +01:00
|
|
|
float_registers: [0f32; NUM_FP_REGS],
|
2023-02-28 14:43:40 +01:00
|
|
|
pc: 0
|
|
|
|
},
|
|
|
|
stack_pointer: 0,
|
2023-03-08 15:48:33 +01:00
|
|
|
object_type: ObjectType::ThreadType,
|
2023-02-28 14:43:40 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-08 21:10:51 +01:00
|
|
|
pub fn init_thread_context(&mut self, initial_pc_reg: i64, initial_sp: i64, 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;
|
|
|
|
}
|
|
|
|
|
2023-03-08 21:10:51 +01:00
|
|
|
pub fn init_simulator_context(&self, base_stack_addr: [i8; SIMULATORSTACKSIZE]) {
|
2023-03-06 16:31:35 +01:00
|
|
|
// let res = self.simulation_context.get_context();
|
|
|
|
// if res != 0 {
|
|
|
|
// panic!("getcontext returns non-zero value {}", res);
|
|
|
|
// }
|
|
|
|
// self.simulation_context.buf.uc_stack.ss_sp = base_stack_addr;
|
|
|
|
// self.simulation_context.buf.uc_stack.ss_size = base_stack_addr.len();
|
|
|
|
// self.simulation_context.buf.uc_stack.ss_flags = 0;
|
|
|
|
// self.simulation_context.buf.uc_link = UContextT::new().buf;
|
|
|
|
// self.simulation_context.make_context(start_thread_execution, 0);
|
|
|
|
|
|
|
|
// self.simulation_context.stackBottom = base_stack_addr.to_vec();
|
|
|
|
// self.simulation_context.stackBottom[0] = STACK_FENCEPOST;
|
2023-02-28 14:43:40 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Check if a thread has overflowed its stack
|
2023-03-09 12:08:33 +01:00
|
|
|
///
|
|
|
|
/// 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) {
|
2023-03-09 12:08:33 +01:00
|
|
|
// if self.simulator_context.stackBottom != STACK_FENCEPOST {
|
|
|
|
// panic!("thread {} has overflowed", self.get_name())
|
|
|
|
// }
|
2023-02-28 14:43:40 +01:00
|
|
|
}
|
|
|
|
|
2023-03-09 12:08:33 +01:00
|
|
|
pub fn sleep(&self) {
|
|
|
|
unreachable!("Has been moved to thread manager");
|
2023-02-28 14:43:40 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn save_simulator_state(&self) {
|
|
|
|
todo!();
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn restore_simulator_state(&self) {
|
|
|
|
todo!();
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn get_name(&self) -> String {
|
|
|
|
self.name.clone()
|
|
|
|
}
|
|
|
|
|
2023-03-01 15:45:49 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
impl Drop for Thread {
|
|
|
|
|
|
|
|
fn drop(&mut self) {
|
|
|
|
self.object_type = ObjectType::InvalidType;
|
|
|
|
todo!();
|
|
|
|
}
|
|
|
|
|
2023-03-06 16:31:35 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
fn start_thread_execution() {
|
|
|
|
|
2023-03-09 12:44:03 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
mod test {
|
|
|
|
|
2023-03-09 13:07:50 +01:00
|
|
|
use super::{Thread, ThreadContext};
|
2023-03-09 12:44:03 +01:00
|
|
|
|
2023-03-09 13:07:50 +01:00
|
|
|
macro_rules! get_new_thread {
|
|
|
|
() => { Thread::new("test_thread") };
|
|
|
|
($a:literal) => {
|
|
|
|
Thread::new(&$a.to_string())
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
#[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!();
|
2023-03-09 12:44:03 +01:00
|
|
|
}
|
|
|
|
|
2023-02-28 14:43:40 +01:00
|
|
|
}
|