Schedule tasks to APs in aster-nix

This commit is contained in:
Zhang Junyang
2024-08-22 19:23:33 +08:00
committed by Tate, Hongliang Tian
parent 5feb8f5de8
commit 47be0a909b
11 changed files with 548 additions and 481 deletions

View File

@ -35,6 +35,7 @@ use core::sync::atomic::Ordering;
use ostd::{
arch::qemu::{exit_qemu, QemuExitCode},
boot,
cpu::PinCurrentCpu,
};
use process::Process;
@ -83,7 +84,13 @@ pub fn main() {
component::init_all(component::parse_metadata!()).unwrap();
init();
ostd::IN_BOOTSTRAP_CONTEXT.store(false, Ordering::Relaxed);
// Spawn all AP idle threads.
ostd::boot::smp::register_ap_entry(ap_init);
// Spawn the first kernel thread on BSP.
Thread::spawn_kernel_thread(ThreadOptions::new(init_thread));
// Spawning functions in the bootstrap context will not return.
unreachable!()
}
@ -100,6 +107,25 @@ pub fn init() {
process::init();
}
fn ap_init() -> ! {
fn ap_idle_thread() {
let preempt_guard = ostd::task::disable_preempt();
let cpu_id = preempt_guard.current_cpu();
drop(preempt_guard);
log::info!("Kernel idle thread for CPU #{} started.", cpu_id);
loop {
Thread::yield_now();
}
}
let preempt_guard = ostd::task::disable_preempt();
let cpu_id = preempt_guard.current_cpu();
drop(preempt_guard);
Thread::spawn_kernel_thread(ThreadOptions::new(ap_idle_thread).cpu_affinity(cpu_id.into()));
// Spawning functions in the bootstrap context will not return.
unreachable!()
}
fn init_thread() {
println!(
"[kernel] Spawn init thread, tid = {}",

View File

@ -1,8 +1,9 @@
// SPDX-License-Identifier: MPL-2.0
use ostd::{
cpu::{num_cpus, PinCurrentCpu},
cpu::{num_cpus, CpuSet, PinCurrentCpu},
task::{
disable_preempt,
scheduler::{inject_scheduler, EnqueueFlags, LocalRunQueue, Scheduler, UpdateFlags},
AtomicCpuId, Priority, Task,
},
@ -36,10 +37,31 @@ impl<T: PreemptSchedInfo> PreemptScheduler<T> {
Self { rq }
}
/// Selects a cpu for task to run on.
fn select_cpu(&self, _runnable: &Arc<T>) -> u32 {
// FIXME: adopt more reasonable policy once we fully enable SMP.
0
/// Selects a CPU for task to run on for the first time.
fn select_cpu(&self, runnable: &Arc<T>) -> u32 {
// If the CPU of a runnable task has been set before, keep scheduling
// the task to that one.
// TODO: Consider migrating tasks between CPUs for load balancing.
if let Some(cpu_id) = runnable.cpu().get() {
return cpu_id;
}
let preempt_guard = disable_preempt();
let mut selected = preempt_guard.current_cpu();
let mut minimum_load = usize::MAX;
for candidate in runnable.cpu_affinity().iter() {
let rq = self.rq[candidate].lock();
// A wild guess measuring the load of a runqueue. We assume that
// real-time tasks are 4-times as important as normal tasks.
let load = rq.real_time_entities.len() * 4 + rq.normal_entities.len();
if load < minimum_load {
selected = candidate as u32;
minimum_load = load;
}
}
selected
}
}
@ -214,11 +236,15 @@ impl PreemptSchedInfo for Task {
const REAL_TIME_TASK_PRIORITY: Self::PRIORITY = Priority::new(100);
fn priority(&self) -> Self::PRIORITY {
self.priority()
self.schedule_info().priority
}
fn cpu(&self) -> &AtomicCpuId {
self.cpu()
&self.schedule_info().cpu
}
fn cpu_affinity(&self) -> &CpuSet {
&self.schedule_info().cpu_affinity
}
}
@ -231,6 +257,8 @@ trait PreemptSchedInfo {
fn cpu(&self) -> &AtomicCpuId;
fn cpu_affinity(&self) -> &CpuSet;
fn is_real_time(&self) -> bool {
self.priority() < Self::REAL_TIME_TASK_PRIORITY
}