kernel/scheduler.rs
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102
// Licensed under the Apache License, Version 2.0 or the MIT License.
// SPDX-License-Identifier: Apache-2.0 OR MIT
// Copyright Tock Contributors 2022.
//! Interface for Tock kernel schedulers.
pub mod cooperative;
pub mod mlfq;
pub mod priority;
pub mod round_robin;
use crate::deferred_call::DeferredCall;
use crate::platform::chip::Chip;
use crate::process::ProcessId;
use crate::process::StoppedExecutingReason;
use core::num::NonZeroU32;
/// Trait which any scheduler must implement.
pub trait Scheduler<C: Chip> {
/// Decide which process to run next.
///
/// The scheduler must decide whether to run a process, and if so, which
/// one. If the scheduler chooses not to run a process, it can request that
/// the chip enter sleep mode.
///
/// If the scheduler selects a process to run it must provide its `ProcessId`
/// and an optional timeslice length in microseconds to provide to that
/// process. If the timeslice is `None`, the process will be run
/// cooperatively (i.e. without preemption). Otherwise the process will run
/// with a timeslice set to the specified length.
fn next(&self) -> SchedulingDecision;
/// Inform the scheduler of why the last process stopped executing, and how
/// long it executed for. Notably, `execution_time_us` will be `None`
/// if the the scheduler requested this process be run cooperatively.
fn result(&self, result: StoppedExecutingReason, execution_time_us: Option<u32>);
/// Tell the scheduler to execute kernel work such as interrupt bottom
/// halves and dynamic deferred calls. Most schedulers will use this default
/// implementation, but schedulers which at times wish to defer interrupt
/// handling will reimplement it.
///
/// Providing this interface allows schedulers to fully manage how the main
/// kernel loop executes. For example, a more advanced scheduler that
/// attempts to help processes meet their deadlines may need to defer bottom
/// half interrupt handling or to selectively service certain interrupts.
/// Or, a power aware scheduler may want to selectively choose what work to
/// complete at any time to meet power requirements.
///
/// Custom implementations of this function must be very careful, however,
/// as this function is called in the core kernel loop.
unsafe fn execute_kernel_work(&self, chip: &C) {
chip.service_pending_interrupts();
while DeferredCall::has_tasks() && !chip.has_pending_interrupts() {
DeferredCall::service_next_pending();
}
}
/// Ask the scheduler whether to take a break from executing userspace
/// processes to handle kernel tasks. Most schedulers will use this default
/// implementation, which always prioritizes kernel work, but schedulers
/// that wish to defer interrupt handling may reimplement it.
unsafe fn do_kernel_work_now(&self, chip: &C) -> bool {
chip.has_pending_interrupts() || DeferredCall::has_tasks()
}
/// Ask the scheduler whether to continue trying to execute a process.
///
/// Once a process is scheduled the kernel will try to execute it until it
/// has no more work to do or exhausts its timeslice. The kernel will call
/// this function before every loop to check with the scheduler if it wants
/// to continue trying to execute this process.
///
/// Most schedulers will use this default implementation, which causes the
/// `do_process()` loop to return if there are interrupts or deferred calls
/// that need to be serviced. However, schedulers which wish to defer
/// interrupt handling may change this, or priority schedulers which wish to
/// check if the execution of the current process has caused a higher
/// priority process to become ready (such as in the case of IPC). If this
/// returns `false`, then `do_process` will exit with a `KernelPreemption`.
///
/// `id` is the identifier of the currently active process.
unsafe fn continue_process(&self, _id: ProcessId, chip: &C) -> bool {
!(chip.has_pending_interrupts() || DeferredCall::has_tasks())
}
}
/// Enum representing the actions the scheduler can request in each call to
/// `scheduler.next()`.
#[derive(Copy, Clone)]
pub enum SchedulingDecision {
/// Tell the kernel to run the specified process with the passed timeslice.
/// If `None` is passed as a timeslice, the process will be run
/// cooperatively.
RunProcess((ProcessId, Option<NonZeroU32>)),
/// Tell the kernel to go to sleep. Notably, if the scheduler asks the
/// kernel to sleep when kernel tasks are ready, the kernel will not sleep,
/// and will instead restart the main loop and call `next()` again.
TrySleep,
}