use core::{
cell::RefCell,
mem,
sync::atomic::{AtomicBool, Ordering},
};
use alloc::{
boxed::Box,
collections::{BTreeMap, BinaryHeap},
vec::Vec,
};
use tracing::{info, trace};
use crate::{
cpu::{self, idt::InterruptAllSavedState, interrupts},
devices::clock::{self, ClockTime},
memory_management::virtual_memory_mapper,
process::{syscalls, FxSave},
sync::spin::mutex::Mutex,
};
use super::{Process, ProcessContext};
static SCHEDULER: Mutex<Scheduler> = Mutex::new(Scheduler::new());
static SHUTDOWN: AtomicBool = AtomicBool::new(false);
const MIN_PRIORITY_VALUE: u64 = 100;
#[derive(Debug, Clone, Copy, PartialEq, Eq)]
pub enum ProcessState {
Running,
Scheduled,
WaitingForPid(u64),
WaitingForTime(ClockTime),
}
struct SchedulerProcess {
process: RefCell<Box<Process>>,
state: ProcessState,
priority_counter: u64,
}
impl PartialEq for SchedulerProcess {
fn eq(&self, other: &Self) -> bool {
self.priority_counter == other.priority_counter
}
}
impl PartialOrd for SchedulerProcess {
fn partial_cmp(&self, other: &Self) -> Option<core::cmp::Ordering> {
Some(self.cmp(other))
}
}
impl Eq for SchedulerProcess {}
impl Ord for SchedulerProcess {
fn cmp(&self, other: &Self) -> core::cmp::Ordering {
self.priority_counter.cmp(&other.priority_counter)
}
}
struct Scheduler {
interrupt_initialized: bool,
scheduled_processes: BinaryHeap<SchedulerProcess>,
running_waiting_procs: BTreeMap<u64, SchedulerProcess>,
exited_processes: Vec<Process>,
max_priority: u64,
}
impl Scheduler {
const fn new() -> Self {
Self {
interrupt_initialized: false,
scheduled_processes: BinaryHeap::new(),
running_waiting_procs: BTreeMap::new(),
exited_processes: Vec::new(),
max_priority: u64::MAX,
}
}
pub fn push_process(&mut self, process: Process) {
self.reschedule_process(SchedulerProcess {
process: RefCell::new(Box::new(process)),
state: ProcessState::Scheduled,
priority_counter: self.max_priority,
})
}
fn init_interrupt(&mut self) {
if self.interrupt_initialized {
return;
}
self.interrupt_initialized = true;
interrupts::create_scheduler_interrupt(scheduler_interrupt_handler);
interrupts::create_syscall_interrupt(syscall_interrupt_handler);
}
fn reschedule_process(&mut self, mut process: SchedulerProcess) {
if SHUTDOWN.load(Ordering::Acquire) {
let mut inner_proc = process.process.into_inner();
info!(
"Process {} is not rescheduled as the scheduler is shutting down",
inner_proc.id
);
inner_proc.exit(0xFF);
self.exited_processes.push(*inner_proc);
return;
}
process.priority_counter = self.max_priority;
process.state = ProcessState::Scheduled;
self.scheduled_processes.push(process);
}
fn reset_scheduled_processes_counters(&mut self) {
let max_priority = u64::MAX;
self.scheduled_processes = self
.scheduled_processes
.drain()
.map(|mut p| {
p.priority_counter = max_priority;
p
})
.collect::<BinaryHeap<_>>();
}
fn try_wake_waiting_processes(&mut self) {
let time_now = clock::clocks().time_since_startup();
let extracted = self
.running_waiting_procs
.extract_if(|_, process| {
let mut remove = false;
let mut inner_proc = process.process.borrow_mut();
match process.state {
ProcessState::WaitingForPid(_) | ProcessState::Running => {
self.exited_processes.retain_mut(|exited_proc| {
let found_parent = exited_proc.parent_id == inner_proc.id;
if found_parent {
inner_proc.add_child_exit(exited_proc.id, exited_proc.exit_code);
}
if let ProcessState::WaitingForPid(pid) = process.state {
if pid == exited_proc.id {
remove = true;
assert_eq!(
inner_proc.context.cs & 0x3,
3,
"must be from user only"
);
inner_proc.context.rax = exited_proc.exit_code as u64;
}
}
!found_parent
});
}
ProcessState::WaitingForTime(t) => {
if t <= time_now {
remove = true;
}
}
_ => unreachable!("We can't have Scheduled state here"),
}
remove
})
.collect::<Vec<_>>();
for (_, process) in extracted {
self.reschedule_process(process);
}
for exited_proc in self.exited_processes.drain(..) {
for process in self.scheduled_processes.iter() {
let mut inner_proc = process.process.borrow_mut();
if inner_proc.id == exited_proc.parent_id {
inner_proc.add_child_exit(exited_proc.id, exited_proc.exit_code);
}
}
}
self.exited_processes.clear();
}
fn exit_idle_processes(&mut self) {
for process in self.scheduled_processes.drain() {
let mut inner_proc = process.process.into_inner();
info!("Force stopping process {}", inner_proc.id);
inner_proc.exit(0);
}
self.running_waiting_procs
.retain(|_, process| match process.state {
ProcessState::Running => true,
ProcessState::Scheduled
| ProcessState::WaitingForPid(_)
| ProcessState::WaitingForTime(_) => {
let mut inner_proc = process.process.borrow_mut();
info!("Force stopping process {}", inner_proc.id);
inner_proc.exit(0);
false
}
});
}
}
pub fn push_process(process: Process) {
SCHEDULER.lock().push_process(process);
}
pub fn stop_scheduler() {
SHUTDOWN.store(true, Ordering::Relaxed);
}
pub fn schedule() {
SCHEDULER.lock().init_interrupt();
loop {
let current_cpu = cpu::cpu();
assert!(current_cpu.context.is_none());
let mut scheduler = SCHEDULER.lock();
let shutdown = SHUTDOWN.load(Ordering::Acquire);
if shutdown {
scheduler.exit_idle_processes();
}
current_cpu.push_cli();
scheduler.try_wake_waiting_processes();
if scheduler
.scheduled_processes
.peek()
.map(|p| p.priority_counter < MIN_PRIORITY_VALUE)
.unwrap_or(false)
{
scheduler.reset_scheduled_processes_counters();
}
let top = scheduler.scheduled_processes.pop();
if let Some(mut top) = top {
assert_eq!(top.state, ProcessState::Scheduled);
top.state = ProcessState::Running;
let pid;
if !shutdown {
{
let mut inner_proc = top.process.borrow_mut();
pid = inner_proc.id;
let decrement = 6 - inner_proc.priority as u64;
top.priority_counter -= decrement;
scheduler.max_priority = top.priority_counter;
unsafe { inner_proc.switch_to_this_vm() };
current_cpu.process_id = inner_proc.id;
current_cpu.context = Some(inner_proc.context);
current_cpu.scheduling = true;
}
scheduler.running_waiting_procs.insert(pid, top);
}
current_cpu.pop_cli();
}
if shutdown
&& scheduler.scheduled_processes.is_empty()
&& scheduler.running_waiting_procs.is_empty()
&& current_cpu.context.is_none()
{
break;
}
drop(scheduler);
if current_cpu.context.is_some() {
unsafe { core::arch::asm!("int 0xff", out("rax") _) }
unsafe { virtual_memory_mapper::switch_to_kernel() };
} else {
unsafe { cpu::halt() };
}
}
}
fn with_current_process_and_state<F, U>(f: F) -> U
where
F: FnOnce(&mut SchedulerProcess) -> U,
{
let current_cpu = cpu::cpu();
let mut scheduler = SCHEDULER.lock();
let process = scheduler
.running_waiting_procs
.get_mut(¤t_cpu.process_id)
.expect("current process not found");
assert_eq!(process.state, ProcessState::Running);
f(process)
}
unsafe fn take_current_process() -> SchedulerProcess {
let current_cpu = cpu::cpu();
let process = SCHEDULER
.lock()
.running_waiting_procs
.remove(¤t_cpu.process_id)
.expect("current process not found");
assert_eq!(process.state, ProcessState::Running);
process
}
pub fn with_current_process<F, U>(f: F) -> U
where
F: FnOnce(&mut Process) -> U,
{
with_current_process_and_state(|p| f(&mut p.process.borrow_mut()))
}
pub fn with_process<F, U>(pid: u64, f: F) -> U
where
F: FnOnce(&mut Process) -> U,
{
let scheduler = SCHEDULER.lock();
let process = scheduler
.running_waiting_procs
.get(&pid)
.unwrap_or_else(|| {
scheduler
.scheduled_processes
.iter()
.find(|p| p.process.borrow().id == pid)
.expect("process not found")
});
let r = f(&mut process.process.borrow_mut());
r
}
pub fn exit_current_process(exit_code: i32, all_state: &mut InterruptAllSavedState) {
let current_cpu = cpu::cpu();
assert!(current_cpu.context.is_some());
current_cpu.push_cli();
let process = unsafe { take_current_process() };
let mut inner_proc = process.process.into_inner();
trace!("Process {} exited with code {}", inner_proc.id, exit_code);
swap_context(current_cpu.context.as_mut().unwrap(), all_state);
inner_proc.context = current_cpu.context.take().unwrap();
inner_proc.exit(exit_code);
SCHEDULER.lock().exited_processes.push(*inner_proc);
current_cpu.pop_cli();
}
pub fn sleep_current_process(time: ClockTime, all_state: &mut InterruptAllSavedState) {
let current_cpu = cpu::cpu();
assert!(current_cpu.context.is_some());
let deadline = clock::clocks().time_since_startup() + time;
with_current_process_and_state(|p| {
current_cpu.push_cli();
let mut inner_proc = p.process.borrow_mut();
p.state = ProcessState::WaitingForTime(deadline);
trace!(
"Process {} is waiting for time {:?}",
inner_proc.id,
deadline
);
swap_context(current_cpu.context.as_mut().unwrap(), all_state);
inner_proc.context = current_cpu.context.take().unwrap();
});
current_cpu.pop_cli();
}
pub fn yield_current_if_any(all_state: &mut InterruptAllSavedState) {
let current_cpu = cpu::cpu();
if current_cpu.context.is_none() || current_cpu.scheduling {
return;
}
current_cpu.push_cli();
let process = unsafe { take_current_process() };
swap_context(current_cpu.context.as_mut().unwrap(), all_state);
process.process.borrow_mut().context = current_cpu.context.take().unwrap();
SCHEDULER.lock().reschedule_process(process);
current_cpu.pop_cli();
}
pub fn is_process_running(pid: u64) -> bool {
let scheduler = SCHEDULER.lock();
scheduler
.running_waiting_procs
.keys()
.cloned()
.chain(
scheduler
.scheduled_processes
.iter()
.map(|p| p.process.borrow().id),
)
.any(|id| id == pid)
}
pub fn wait_for_pid(all_state: &mut InterruptAllSavedState, pid: u64) -> bool {
let current_cpu = cpu::cpu();
assert!(current_cpu.context.is_some());
let process_found = is_process_running(pid);
if !process_found {
return false;
}
with_current_process_and_state(|p| {
current_cpu.push_cli();
let mut inner_proc = p.process.borrow_mut();
p.state = ProcessState::WaitingForPid(pid);
trace!("Process {} is waiting for process {}", inner_proc.id, pid);
swap_context(current_cpu.context.as_mut().unwrap(), all_state);
inner_proc.context = current_cpu.context.take().unwrap();
});
current_cpu.pop_cli();
true
}
pub fn swap_context(context: &mut ProcessContext, all_state: &mut InterruptAllSavedState) {
let mut fxsave = FxSave::default();
unsafe { core::arch::x86_64::_fxsave64(&mut fxsave as *mut FxSave as _) };
unsafe { core::arch::x86_64::_fxrstor64(context.fxsave.0.as_ptr() as _) };
context.fxsave = fxsave;
mem::swap(&mut all_state.frame.rflags, &mut context.rflags);
mem::swap(&mut all_state.frame.rip, &mut context.rip);
all_state.frame.cs = mem::replace(&mut context.cs, all_state.frame.cs as _) as _;
mem::swap(&mut all_state.frame.rsp, &mut context.rsp);
all_state.frame.ss = mem::replace(&mut context.ss, all_state.frame.ss as _) as _;
mem::swap(&mut all_state.rest.ds, &mut context.ds);
mem::swap(&mut all_state.rest.es, &mut context.es);
mem::swap(&mut all_state.rest.fs, &mut context.fs);
mem::swap(&mut all_state.rest.gs, &mut context.gs);
mem::swap(&mut all_state.rest.dr0, &mut context.dr0);
mem::swap(&mut all_state.rest.dr1, &mut context.dr1);
mem::swap(&mut all_state.rest.dr2, &mut context.dr2);
mem::swap(&mut all_state.rest.dr3, &mut context.dr3);
mem::swap(&mut all_state.rest.dr6, &mut context.dr6);
mem::swap(&mut all_state.rest.dr7, &mut context.dr7);
mem::swap(&mut all_state.rest.rax, &mut context.rax);
mem::swap(&mut all_state.rest.rbx, &mut context.rbx);
mem::swap(&mut all_state.rest.rcx, &mut context.rcx);
mem::swap(&mut all_state.rest.rdx, &mut context.rdx);
mem::swap(&mut all_state.rest.rsi, &mut context.rsi);
mem::swap(&mut all_state.rest.rdi, &mut context.rdi);
mem::swap(&mut all_state.rest.rbp, &mut context.rbp);
mem::swap(&mut all_state.rest.r8, &mut context.r8);
mem::swap(&mut all_state.rest.r9, &mut context.r9);
mem::swap(&mut all_state.rest.r10, &mut context.r10);
mem::swap(&mut all_state.rest.r11, &mut context.r11);
mem::swap(&mut all_state.rest.r12, &mut context.r12);
mem::swap(&mut all_state.rest.r13, &mut context.r13);
mem::swap(&mut all_state.rest.r14, &mut context.r14);
mem::swap(&mut all_state.rest.r15, &mut context.r15);
}
extern "cdecl" fn scheduler_interrupt_handler(all_state: &mut InterruptAllSavedState) {
assert_eq!(all_state.frame.cs & 0x3, 0, "must be from kernel only");
let current_cpu = cpu::cpu();
assert!(current_cpu.context.is_some());
assert!(current_cpu.scheduling);
assert!(current_cpu.interrupts_disabled());
current_cpu.scheduling = false;
swap_context(current_cpu.context.as_mut().unwrap(), all_state);
}
extern "cdecl" fn syscall_interrupt_handler(all_state: &mut InterruptAllSavedState) {
assert_eq!(all_state.frame.cs & 0x3, 3, "must be from user only");
let current_cpu = cpu::cpu();
assert!(current_cpu.context.is_some());
syscalls::handle_syscall(all_state);
}