use alloc::{
string::String,
sync::{Arc, Weak},
};
use ax_kernel_guard::NoPreemptIrqSave;
pub(crate) use crate::run_queue::{current_run_queue, select_run_queue};
#[cfg_attr(doc, doc(cfg(all(feature = "multitask", feature = "task-ext"))))]
#[cfg(feature = "task-ext")]
pub use crate::task::{AxTaskExt, TaskExt};
#[cfg_attr(doc, doc(cfg(all(feature = "multitask", feature = "irq"))))]
#[cfg(feature = "irq")]
pub use crate::timers::register_timer_callback;
#[cfg_attr(doc, doc(cfg(feature = "multitask")))]
pub use crate::{
task::{CurrentTask, TaskId, TaskInner, TaskState},
wait_queue::WaitQueue,
};
pub type AxTaskRef = Arc<AxTask>;
pub type WeakAxTaskRef = Weak<AxTask>;
pub type AxCpuMask = ax_cpumask::CpuMask<{ ax_config::plat::MAX_CPU_NUM }>;
cfg_if::cfg_if! {
if #[cfg(feature = "sched-rr")] {
const MAX_TIME_SLICE: usize = 5;
pub(crate) type AxTask = ax_sched::RRTask<TaskInner, MAX_TIME_SLICE>;
pub(crate) type Scheduler = ax_sched::RRScheduler<TaskInner, MAX_TIME_SLICE>;
} else if #[cfg(feature = "sched-cfs")] {
pub(crate) type AxTask = ax_sched::CFSTask<TaskInner>;
pub(crate) type Scheduler = ax_sched::CFScheduler<TaskInner>;
} else {
pub(crate) type AxTask = ax_sched::FifoTask<TaskInner>;
pub(crate) type Scheduler = ax_sched::FifoScheduler<TaskInner>;
}
}
#[cfg(feature = "preempt")]
struct KernelGuardIfImpl;
#[cfg(feature = "preempt")]
#[ax_crate_interface::impl_interface]
impl ax_kernel_guard::KernelGuardIf for KernelGuardIfImpl {
fn disable_preempt() {
if let Some(curr) = current_may_uninit() {
curr.disable_preempt();
}
}
fn enable_preempt() {
if let Some(curr) = current_may_uninit() {
curr.enable_preempt(true);
}
}
}
pub fn current_may_uninit() -> Option<CurrentTask> {
CurrentTask::try_get()
}
pub fn current() -> CurrentTask {
CurrentTask::get()
}
pub fn init_scheduler() {
info!("Initialize scheduling...");
crate::run_queue::init();
info!(" use {} scheduler.", Scheduler::scheduler_name());
}
pub(crate) fn cpu_mask_full() -> AxCpuMask {
use spin::Lazy;
static CPU_MASK_FULL: Lazy<AxCpuMask> = Lazy::new(|| {
let cpu_num = ax_hal::cpu_num();
let mut cpumask = AxCpuMask::new();
for cpu_id in 0..cpu_num {
cpumask.set(cpu_id, true);
}
cpumask
});
*CPU_MASK_FULL
}
pub fn init_scheduler_secondary() {
crate::run_queue::init_secondary();
}
#[cfg(feature = "irq")]
#[cfg_attr(doc, doc(cfg(feature = "irq")))]
pub fn on_timer_tick() {
use ax_kernel_guard::NoOp;
crate::timers::check_events();
current_run_queue::<NoOp>().scheduler_timer_tick();
}
pub fn spawn_task(task: TaskInner) -> AxTaskRef {
let task_ref = task.into_arc();
select_run_queue::<NoPreemptIrqSave>(&task_ref).add_task(task_ref.clone());
task_ref
}
pub fn spawn_raw<F>(f: F, name: String, stack_size: usize) -> AxTaskRef
where
F: FnOnce() + Send + 'static,
{
spawn_task(TaskInner::new(f, name, stack_size))
}
pub fn spawn_with_name<F>(f: F, name: String) -> AxTaskRef
where
F: FnOnce() + Send + 'static,
{
spawn_raw(f, name, ax_config::TASK_STACK_SIZE)
}
pub fn spawn<F>(f: F) -> AxTaskRef
where
F: FnOnce() + Send + 'static,
{
spawn_with_name(f, String::new())
}
pub fn set_priority(prio: isize) -> bool {
current_run_queue::<NoPreemptIrqSave>().set_current_priority(prio)
}
pub fn set_current_affinity(cpumask: AxCpuMask) -> bool {
might_sleep();
if cpumask.is_empty() {
false
} else {
let curr = current().clone();
curr.set_cpumask(cpumask);
#[cfg(feature = "smp")]
if !cpumask.get(ax_hal::percpu::this_cpu_id()) {
const MIGRATION_TASK_STACK_SIZE: usize = 4096;
let migration_task = TaskInner::new(
move || crate::run_queue::migrate_entry(curr),
"migration-task".into(),
MIGRATION_TASK_STACK_SIZE,
)
.into_arc();
current_run_queue::<NoPreemptIrqSave>().migrate_current(migration_task);
assert!(
cpumask.get(ax_hal::percpu::this_cpu_id()),
"Migration failed"
);
}
true
}
}
#[track_caller]
pub fn yield_now() {
might_sleep();
yield_now_unchecked();
}
#[doc(hidden)]
pub(crate) fn yield_now_unchecked() {
current_run_queue::<NoPreemptIrqSave>().yield_current()
}
pub fn sleep(dur: core::time::Duration) {
sleep_until(ax_hal::time::wall_time() + dur);
}
pub fn sleep_until(deadline: ax_hal::time::TimeValue) {
#[cfg(feature = "irq")]
might_sleep();
#[cfg(feature = "irq")]
current_run_queue::<NoPreemptIrqSave>().sleep_until(deadline);
#[cfg(not(feature = "irq"))]
ax_hal::time::busy_wait_until(deadline);
}
pub fn exit(exit_code: i32) -> ! {
might_sleep();
current_run_queue::<NoPreemptIrqSave>().exit_current(exit_code)
}
fn current_preempt_count() -> usize {
#[cfg(feature = "preempt")]
{
current_may_uninit().map_or(0, |curr| curr.preempt_count())
}
#[cfg(not(feature = "preempt"))]
{
0
}
}
pub(crate) fn in_atomic_context() -> bool {
#[cfg(feature = "irq")]
if !ax_hal::asm::irqs_enabled() {
return true;
}
#[cfg(feature = "preempt")]
if current_preempt_count() != 0 {
return true;
}
false
}
#[track_caller]
pub(crate) fn might_sleep() {
if in_atomic_context() {
panic!(
"sleeping or rescheduling is not allowed in atomic context: irq_enabled={}, \
preempt_count={}",
ax_hal::asm::irqs_enabled(),
current_preempt_count()
);
}
}
pub fn run_idle() -> ! {
loop {
yield_now_unchecked();
trace!("idle task: waiting for IRQs...");
#[cfg(feature = "irq")]
ax_hal::asm::wait_for_irqs();
}
}