2020-01-18 09:38:21 +01:00
|
|
|
/*
|
2021-01-11 09:52:18 +01:00
|
|
|
* Copyright (c) 2018-2021, Andreas Kling <kling@serenityos.org>
|
2020-01-18 09:38:21 +01:00
|
|
|
*
|
2021-04-22 01:24:48 -07:00
|
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
2020-01-18 09:38:21 +01:00
|
|
|
*/
|
|
|
|
|
2021-12-19 15:46:55 -06:00
|
|
|
#include <AK/BuiltinWrappers.h>
|
2020-08-01 14:37:40 -06:00
|
|
|
#include <AK/ScopeGuard.h>
|
2021-08-08 12:31:42 +02:00
|
|
|
#include <AK/Singleton.h>
|
2020-08-03 09:43:19 -06:00
|
|
|
#include <AK/Time.h>
|
2021-06-21 17:34:09 +02:00
|
|
|
#include <Kernel/Arch/x86/InterruptDisabler.h>
|
2021-12-01 17:15:46 +00:00
|
|
|
#include <Kernel/Arch/x86/TrapFrame.h>
|
2021-01-25 16:07:10 +01:00
|
|
|
#include <Kernel/Debug.h>
|
2021-02-14 09:30:31 +01:00
|
|
|
#include <Kernel/Panic.h>
|
2021-05-14 07:48:53 +02:00
|
|
|
#include <Kernel/PerformanceManager.h>
|
2019-06-07 19:29:34 +02:00
|
|
|
#include <Kernel/Process.h>
|
|
|
|
#include <Kernel/RTC.h>
|
|
|
|
#include <Kernel/Scheduler.h>
|
2021-06-22 17:40:16 +02:00
|
|
|
#include <Kernel/Sections.h>
|
Kernel: Introduce the new Time management subsystem
This new subsystem includes better abstractions of how time will be
handled in the OS. We take advantage of the existing RTC timer to aid
in keeping time synchronized. This is standing in contrast to how we
handled time-keeping in the kernel, where the PIT was responsible for
that function in addition to update the scheduler about ticks.
With that new advantage, we can easily change the ticking dynamically
and still keep the time synchronized.
In the process context, we no longer use a fixed declaration of
TICKS_PER_SECOND, but we call the TimeManagement singleton class to
provide us the right value. This allows us to use dynamic ticking in
the future, a feature known as tickless kernel.
The scheduler no longer does by himself the calculation of real time
(Unix time), and just calls the TimeManagment singleton class to provide
the value.
Also, we can use 2 new boot arguments:
- the "time" boot argument accpets either the value "modern", or
"legacy". If "modern" is specified, the time management subsystem will
try to setup HPET. Otherwise, for "legacy" value, the time subsystem
will revert to use the PIT & RTC, leaving HPET disabled.
If this boot argument is not specified, the default pattern is to try
to setup HPET.
- the "hpet" boot argumet accepts either the value "periodic" or
"nonperiodic". If "periodic" is specified, the HPET will scan for
periodic timers, and will assert if none are found. If only one is
found, that timer will be assigned for the time-keeping task. If more
than one is found, both time-keeping task & scheduler-ticking task
will be assigned to periodic timers.
If this boot argument is not specified, the default pattern is to try
to scan for HPET periodic timers. This boot argument has no effect if
HPET is disabled.
In hardware context, PIT & RealTimeClock classes are merely inheriting
from the HardwareTimer class, and they allow to use the old i8254 (PIT)
and RTC devices, managing them via IO ports. By default, the RTC will be
programmed to a frequency of 1024Hz. The PIT will be programmed to a
frequency close to 1000Hz.
About HPET, depending if we need to scan for periodic timers or not,
we try to set a frequency close to 1000Hz for the time-keeping timer
and scheduler-ticking timer. Also, if possible, we try to enable the
Legacy replacement feature of the HPET. This feature if exists,
instructs the chipset to disconnect both i8254 (PIT) and RTC.
This behavior is observable on QEMU, and was verified against the source
code:
https://github.com/qemu/qemu/commit/ce967e2f33861b0e17753f97fa4527b5943c94b6
The HPETComparator class is inheriting from HardwareTimer class, and is
responsible for an individual HPET comparator, which is essentially a
timer. Therefore, it needs to call the singleton HPET class to perform
HPET-related operations.
The new abstraction of Hardware timers brings an opportunity of more new
features in the foreseeable future. For example, we can change the
callback function of each hardware timer, thus it makes it possible to
swap missions between hardware timers, or to allow to use a hardware
timer for other temporary missions (e.g. calibrating the LAPIC timer,
measuring the CPU frequency, etc).
2020-03-09 17:03:27 +02:00
|
|
|
#include <Kernel/Time/TimeManagement.h>
|
2022-01-15 21:19:41 +02:00
|
|
|
#include <Kernel/kstdio.h>
|
2018-11-07 22:15:02 +01:00
|
|
|
|
2021-01-26 22:45:30 -07:00
|
|
|
// Remove this once SMP is stable and can be enabled by default
|
|
|
|
#define SCHEDULE_ON_ALL_PROCESSORS 0
|
|
|
|
|
2020-02-16 01:27:42 +01:00
|
|
|
namespace Kernel {
|
|
|
|
|
2021-08-22 01:37:17 +02:00
|
|
|
RecursiveSpinlock g_scheduler_lock;
|
2019-07-19 17:21:13 +02:00
|
|
|
|
2019-12-30 18:46:17 +01:00
|
|
|
static u32 time_slice_for(const Thread& thread)
|
2019-02-07 12:21:17 +01:00
|
|
|
{
|
2020-12-03 22:12:50 -07:00
|
|
|
// One time slice unit == 4ms (assuming 250 ticks/second)
|
2021-01-28 20:07:41 -07:00
|
|
|
if (thread.is_idle_thread())
|
2019-04-20 15:58:45 +02:00
|
|
|
return 1;
|
2020-12-03 22:12:50 -07:00
|
|
|
return 2;
|
2019-02-07 12:21:17 +01:00
|
|
|
}
|
2018-11-07 22:15:02 +01:00
|
|
|
|
2021-02-14 17:39:41 +01:00
|
|
|
READONLY_AFTER_INIT Thread* g_finalizer;
|
|
|
|
READONLY_AFTER_INIT WaitQueue* g_finalizer_wait_queue;
|
2020-07-30 21:46:06 +02:00
|
|
|
Atomic<bool> g_finalizer_has_work { false };
|
2021-02-14 17:39:41 +01:00
|
|
|
READONLY_AFTER_INIT static Process* s_colonel_process;
|
2018-11-07 22:15:02 +01:00
|
|
|
|
2021-01-22 16:56:08 -07:00
|
|
|
struct ThreadReadyQueue {
|
2021-09-09 16:30:59 +04:30
|
|
|
IntrusiveList<&Thread::m_ready_queue_node> thread_list;
|
2021-01-22 16:56:08 -07:00
|
|
|
};
|
2021-08-08 12:31:42 +02:00
|
|
|
|
|
|
|
struct ThreadReadyQueues {
|
|
|
|
u32 mask {};
|
|
|
|
static constexpr size_t count = sizeof(mask) * 8;
|
|
|
|
Array<ThreadReadyQueue, count> queues;
|
|
|
|
};
|
|
|
|
|
2021-08-22 01:37:17 +02:00
|
|
|
static Singleton<SpinlockProtected<ThreadReadyQueues>> g_ready_queues;
|
2021-07-14 21:46:32 -06:00
|
|
|
|
2021-08-22 01:37:17 +02:00
|
|
|
static SpinlockProtected<TotalTimeScheduled> g_total_time_scheduled;
|
2021-07-14 21:46:32 -06:00
|
|
|
|
|
|
|
// The Scheduler::current_time function provides a current time for scheduling purposes,
|
|
|
|
// which may not necessarily relate to wall time
|
|
|
|
u64 (*Scheduler::current_time)();
|
|
|
|
|
2021-07-15 14:54:19 -06:00
|
|
|
static void dump_thread_list(bool = false);
|
2021-01-22 16:56:08 -07:00
|
|
|
|
|
|
|
static inline u32 thread_priority_to_priority_index(u32 thread_priority)
|
|
|
|
{
|
|
|
|
// Converts the priority in the range of THREAD_PRIORITY_MIN...THREAD_PRIORITY_MAX
|
|
|
|
// to a index into g_ready_queues where 0 is the highest priority bucket
|
2021-02-23 20:42:32 +01:00
|
|
|
VERIFY(thread_priority >= THREAD_PRIORITY_MIN && thread_priority <= THREAD_PRIORITY_MAX);
|
2021-01-22 16:56:08 -07:00
|
|
|
constexpr u32 thread_priority_count = THREAD_PRIORITY_MAX - THREAD_PRIORITY_MIN + 1;
|
|
|
|
static_assert(thread_priority_count > 0);
|
2021-08-08 12:31:42 +02:00
|
|
|
auto priority_bucket = ((thread_priority_count - (thread_priority - THREAD_PRIORITY_MIN)) / thread_priority_count) * (ThreadReadyQueues::count - 1);
|
|
|
|
VERIFY(priority_bucket < ThreadReadyQueues::count);
|
2021-01-22 16:56:08 -07:00
|
|
|
return priority_bucket;
|
|
|
|
}
|
|
|
|
|
|
|
|
Thread& Scheduler::pull_next_runnable_thread()
|
|
|
|
{
|
2021-08-22 12:37:50 +02:00
|
|
|
auto affinity_mask = 1u << Processor::current_id();
|
2021-01-22 16:56:08 -07:00
|
|
|
|
2021-08-08 12:31:42 +02:00
|
|
|
return g_ready_queues->with([&](auto& ready_queues) -> Thread& {
|
|
|
|
auto priority_mask = ready_queues.mask;
|
|
|
|
while (priority_mask != 0) {
|
2021-12-19 15:46:55 -06:00
|
|
|
auto priority = bit_scan_forward(priority_mask);
|
2021-08-08 12:31:42 +02:00
|
|
|
VERIFY(priority > 0);
|
|
|
|
auto& ready_queue = ready_queues.queues[--priority];
|
|
|
|
for (auto& thread : ready_queue.thread_list) {
|
|
|
|
VERIFY(thread.m_runnable_priority == (int)priority);
|
|
|
|
if (thread.is_active())
|
|
|
|
continue;
|
|
|
|
if (!(thread.affinity() & affinity_mask))
|
|
|
|
continue;
|
|
|
|
thread.m_runnable_priority = -1;
|
|
|
|
ready_queue.thread_list.remove(thread);
|
|
|
|
if (ready_queue.thread_list.is_empty())
|
|
|
|
ready_queues.mask &= ~(1u << priority);
|
|
|
|
// Mark it as active because we are using this thread. This is similar
|
|
|
|
// to comparing it with Processor::current_thread, but when there are
|
|
|
|
// multiple processors there's no easy way to check whether the thread
|
|
|
|
// is actually still needed. This prevents accidental finalization when
|
|
|
|
// a thread is no longer in Running state, but running on another core.
|
|
|
|
|
|
|
|
// We need to mark it active here so that this thread won't be
|
|
|
|
// scheduled on another core if it were to be queued before actually
|
|
|
|
// switching to it.
|
|
|
|
// FIXME: Figure out a better way maybe?
|
|
|
|
thread.set_active(true);
|
|
|
|
return thread;
|
|
|
|
}
|
|
|
|
priority_mask &= ~(1u << priority);
|
2021-01-22 16:56:08 -07:00
|
|
|
}
|
2021-08-08 12:31:42 +02:00
|
|
|
return *Processor::idle_thread();
|
|
|
|
});
|
2021-01-22 16:56:08 -07:00
|
|
|
}
|
|
|
|
|
2021-07-11 20:12:42 -06:00
|
|
|
Thread* Scheduler::peek_next_runnable_thread()
|
|
|
|
{
|
2021-08-22 12:37:50 +02:00
|
|
|
auto affinity_mask = 1u << Processor::current_id();
|
2021-07-11 20:12:42 -06:00
|
|
|
|
2021-08-08 12:31:42 +02:00
|
|
|
return g_ready_queues->with([&](auto& ready_queues) -> Thread* {
|
|
|
|
auto priority_mask = ready_queues.mask;
|
|
|
|
while (priority_mask != 0) {
|
2021-12-19 15:46:55 -06:00
|
|
|
auto priority = bit_scan_forward(priority_mask);
|
2021-08-08 12:31:42 +02:00
|
|
|
VERIFY(priority > 0);
|
|
|
|
auto& ready_queue = ready_queues.queues[--priority];
|
|
|
|
for (auto& thread : ready_queue.thread_list) {
|
|
|
|
VERIFY(thread.m_runnable_priority == (int)priority);
|
|
|
|
if (thread.is_active())
|
|
|
|
continue;
|
|
|
|
if (!(thread.affinity() & affinity_mask))
|
|
|
|
continue;
|
|
|
|
return &thread;
|
|
|
|
}
|
|
|
|
priority_mask &= ~(1u << priority);
|
2021-07-11 20:12:42 -06:00
|
|
|
}
|
|
|
|
|
2021-08-08 12:31:42 +02:00
|
|
|
// Unlike in pull_next_runnable_thread() we don't want to fall back to
|
|
|
|
// the idle thread. We just want to see if we have any other thread ready
|
|
|
|
// to be scheduled.
|
|
|
|
return nullptr;
|
|
|
|
});
|
2021-07-11 20:12:42 -06:00
|
|
|
}
|
|
|
|
|
2021-01-22 16:56:08 -07:00
|
|
|
bool Scheduler::dequeue_runnable_thread(Thread& thread, bool check_affinity)
|
|
|
|
{
|
2021-01-28 20:07:41 -07:00
|
|
|
if (thread.is_idle_thread())
|
2021-01-22 16:56:08 -07:00
|
|
|
return true;
|
|
|
|
|
2021-08-08 12:31:42 +02:00
|
|
|
return g_ready_queues->with([&](auto& ready_queues) {
|
|
|
|
auto priority = thread.m_runnable_priority;
|
|
|
|
if (priority < 0) {
|
|
|
|
VERIFY(!thread.m_ready_queue_node.is_in_list());
|
|
|
|
return false;
|
|
|
|
}
|
2021-01-22 16:56:08 -07:00
|
|
|
|
2021-08-22 12:37:50 +02:00
|
|
|
if (check_affinity && !(thread.affinity() & (1 << Processor::current_id())))
|
2021-08-08 12:31:42 +02:00
|
|
|
return false;
|
|
|
|
|
|
|
|
VERIFY(ready_queues.mask & (1u << priority));
|
|
|
|
auto& ready_queue = ready_queues.queues[priority];
|
|
|
|
thread.m_runnable_priority = -1;
|
|
|
|
ready_queue.thread_list.remove(thread);
|
|
|
|
if (ready_queue.thread_list.is_empty())
|
|
|
|
ready_queues.mask &= ~(1u << priority);
|
|
|
|
return true;
|
|
|
|
});
|
2021-01-22 16:56:08 -07:00
|
|
|
}
|
|
|
|
|
2021-08-08 14:19:55 +02:00
|
|
|
void Scheduler::enqueue_runnable_thread(Thread& thread)
|
2021-01-22 16:56:08 -07:00
|
|
|
{
|
2021-08-29 20:10:24 +02:00
|
|
|
VERIFY(g_scheduler_lock.is_locked_by_current_processor());
|
2021-01-28 20:07:41 -07:00
|
|
|
if (thread.is_idle_thread())
|
2021-01-22 16:56:08 -07:00
|
|
|
return;
|
2021-01-28 08:25:05 +01:00
|
|
|
auto priority = thread_priority_to_priority_index(thread.priority());
|
2021-01-22 16:56:08 -07:00
|
|
|
|
2021-08-08 12:31:42 +02:00
|
|
|
g_ready_queues->with([&](auto& ready_queues) {
|
|
|
|
VERIFY(thread.m_runnable_priority < 0);
|
|
|
|
thread.m_runnable_priority = (int)priority;
|
|
|
|
VERIFY(!thread.m_ready_queue_node.is_in_list());
|
|
|
|
auto& ready_queue = ready_queues.queues[priority];
|
|
|
|
bool was_empty = ready_queue.thread_list.is_empty();
|
|
|
|
ready_queue.thread_list.append(thread);
|
|
|
|
if (was_empty)
|
|
|
|
ready_queues.mask |= (1u << priority);
|
|
|
|
});
|
2021-01-22 16:56:08 -07:00
|
|
|
}
|
|
|
|
|
2021-02-19 18:41:50 +01:00
|
|
|
UNMAP_AFTER_INIT void Scheduler::start()
|
2018-11-07 22:15:02 +01:00
|
|
|
{
|
2021-02-23 20:42:32 +01:00
|
|
|
VERIFY_INTERRUPTS_DISABLED();
|
2020-07-30 21:46:06 +02:00
|
|
|
|
2020-07-05 14:32:07 -06:00
|
|
|
// We need to acquire our scheduler lock, which will be released
|
|
|
|
// by the idle thread once control transferred there
|
|
|
|
g_scheduler_lock.lock();
|
|
|
|
|
2020-06-28 15:34:31 -06:00
|
|
|
auto& processor = Processor::current();
|
2021-02-23 20:42:32 +01:00
|
|
|
VERIFY(processor.is_initialized());
|
2021-01-28 20:07:41 -07:00
|
|
|
auto& idle_thread = *Processor::idle_thread();
|
2021-02-23 20:42:32 +01:00
|
|
|
VERIFY(processor.current_thread() == &idle_thread);
|
2020-06-28 15:34:31 -06:00
|
|
|
idle_thread.set_ticks_left(time_slice_for(idle_thread));
|
|
|
|
idle_thread.did_schedule();
|
|
|
|
idle_thread.set_initialized(true);
|
2020-07-03 05:19:50 -06:00
|
|
|
processor.init_context(idle_thread, false);
|
2022-01-30 11:38:50 +01:00
|
|
|
idle_thread.set_state(Thread::State::Running);
|
2021-08-22 12:37:50 +02:00
|
|
|
VERIFY(idle_thread.affinity() == (1u << processor.id()));
|
2020-06-28 15:34:31 -06:00
|
|
|
processor.initialize_context_switching(idle_thread);
|
2021-02-23 20:42:32 +01:00
|
|
|
VERIFY_NOT_REACHED();
|
2020-06-27 13:42:28 -06:00
|
|
|
}
|
2018-11-07 22:15:02 +01:00
|
|
|
|
2022-01-30 14:29:37 +01:00
|
|
|
void Scheduler::pick_next()
|
2020-06-27 13:42:28 -06:00
|
|
|
{
|
2021-02-23 20:42:32 +01:00
|
|
|
VERIFY_INTERRUPTS_DISABLED();
|
2018-11-07 22:15:02 +01:00
|
|
|
|
2021-08-08 11:09:35 +02:00
|
|
|
// Set the in_scheduler flag before acquiring the spinlock. This
|
2020-08-01 14:37:40 -06:00
|
|
|
// prevents a recursive call into Scheduler::invoke_async upon
|
|
|
|
// leaving the scheduler lock.
|
|
|
|
ScopedCritical critical;
|
2021-08-29 12:43:39 +02:00
|
|
|
Processor::set_current_in_scheduler(true);
|
2020-08-01 14:37:40 -06:00
|
|
|
ScopeGuard guard(
|
|
|
|
[]() {
|
|
|
|
// We may be on a different processor after we got switched
|
|
|
|
// back to this thread!
|
2021-08-29 12:43:39 +02:00
|
|
|
VERIFY(Processor::current_in_scheduler());
|
|
|
|
Processor::set_current_in_scheduler(false);
|
2020-08-01 14:37:40 -06:00
|
|
|
});
|
|
|
|
|
2021-08-22 01:49:22 +02:00
|
|
|
SpinlockLocker lock(g_scheduler_lock);
|
2020-06-28 15:34:31 -06:00
|
|
|
|
2021-01-23 23:59:27 +01:00
|
|
|
if constexpr (SCHEDULER_RUNNABLE_DEBUG) {
|
2021-04-18 19:57:17 +02:00
|
|
|
dump_thread_list();
|
2021-01-12 22:30:52 +01:00
|
|
|
}
|
2018-11-07 22:15:02 +01:00
|
|
|
|
2021-01-22 16:56:08 -07:00
|
|
|
auto& thread_to_schedule = pull_next_runnable_thread();
|
2021-01-23 23:59:27 +01:00
|
|
|
if constexpr (SCHEDULER_DEBUG) {
|
2021-07-25 19:51:44 +02:00
|
|
|
dbgln("Scheduler[{}]: Switch to {} @ {:#04x}:{:p}",
|
2021-08-22 12:37:50 +02:00
|
|
|
Processor::current_id(),
|
2021-01-22 16:56:08 -07:00
|
|
|
thread_to_schedule,
|
2021-07-18 16:50:08 -07:00
|
|
|
thread_to_schedule.regs().cs, thread_to_schedule.regs().ip());
|
2021-01-12 22:30:52 +01:00
|
|
|
}
|
2019-12-30 18:46:17 +01:00
|
|
|
|
2020-08-01 14:37:40 -06:00
|
|
|
// We need to leave our first critical section before switching context,
|
|
|
|
// but since we're still holding the scheduler lock we're still in a critical section
|
|
|
|
critical.leave();
|
|
|
|
|
2021-01-22 16:56:08 -07:00
|
|
|
thread_to_schedule.set_ticks_left(time_slice_for(thread_to_schedule));
|
2022-01-30 14:29:37 +01:00
|
|
|
context_switch(&thread_to_schedule);
|
2018-11-07 22:15:02 +01:00
|
|
|
}
|
|
|
|
|
2022-01-30 14:29:37 +01:00
|
|
|
void Scheduler::yield()
|
2020-06-27 13:42:28 -06:00
|
|
|
{
|
2020-07-03 05:19:50 -06:00
|
|
|
InterruptDisabler disabler;
|
2020-08-01 14:37:40 -06:00
|
|
|
|
2021-12-29 01:01:27 +01:00
|
|
|
auto const* current_thread = Thread::current();
|
2021-08-22 12:37:50 +02:00
|
|
|
dbgln_if(SCHEDULER_DEBUG, "Scheduler[{}]: yielding thread {} in_irq={}", Processor::current_id(), *current_thread, Processor::current_in_irq());
|
2021-02-23 20:42:32 +01:00
|
|
|
VERIFY(current_thread != nullptr);
|
2021-08-22 12:21:31 +02:00
|
|
|
if (Processor::current_in_irq() || Processor::in_critical()) {
|
2020-07-03 05:19:50 -06:00
|
|
|
// If we're handling an IRQ we can't switch context, or we're in
|
|
|
|
// a critical section where we don't want to switch contexts, then
|
|
|
|
// delay until exiting the trap or critical section
|
2021-08-22 12:21:31 +02:00
|
|
|
Processor::current().invoke_scheduler_async();
|
2022-01-30 14:29:37 +01:00
|
|
|
return;
|
2020-08-01 14:37:40 -06:00
|
|
|
}
|
|
|
|
|
2022-01-30 14:29:37 +01:00
|
|
|
Scheduler::pick_next();
|
2020-06-27 13:42:28 -06:00
|
|
|
}
|
|
|
|
|
2022-01-30 14:29:37 +01:00
|
|
|
void Scheduler::context_switch(Thread* thread)
|
2018-11-07 22:15:02 +01:00
|
|
|
{
|
2021-08-29 20:10:24 +02:00
|
|
|
if (Memory::s_mm_lock.is_locked_by_current_processor()) {
|
2021-08-06 13:49:36 +02:00
|
|
|
PANIC("In context switch while holding Memory::s_mm_lock");
|
2021-05-25 01:02:19 -07:00
|
|
|
}
|
|
|
|
|
2020-07-05 14:32:07 -06:00
|
|
|
thread->did_schedule();
|
2018-11-07 22:15:02 +01:00
|
|
|
|
2021-12-29 01:01:27 +01:00
|
|
|
auto* from_thread = Thread::current();
|
2022-01-30 14:04:46 +01:00
|
|
|
VERIFY(from_thread);
|
|
|
|
|
2020-07-05 14:32:07 -06:00
|
|
|
if (from_thread == thread)
|
2022-01-30 14:29:37 +01:00
|
|
|
return;
|
2018-11-07 22:15:02 +01:00
|
|
|
|
2022-01-30 14:04:46 +01:00
|
|
|
// If the last process hasn't blocked (still marked as running),
|
|
|
|
// mark it as runnable for the next round.
|
|
|
|
if (from_thread->state() == Thread::State::Running)
|
|
|
|
from_thread->set_state(Thread::State::Runnable);
|
2018-11-07 22:24:20 +01:00
|
|
|
|
|
|
|
#ifdef LOG_EVERY_CONTEXT_SWITCH
|
2022-01-30 14:04:46 +01:00
|
|
|
const auto msg = "Scheduler[{}]: {} -> {} [prio={}] {:#04x}:{:p}";
|
2021-07-18 16:50:08 -07:00
|
|
|
|
2022-01-30 14:04:46 +01:00
|
|
|
dbgln(msg,
|
|
|
|
Processor::current_id(), from_thread->tid().value(),
|
|
|
|
thread->tid().value(), thread->priority(), thread->regs().cs, thread->regs().ip());
|
2018-11-07 22:24:20 +01:00
|
|
|
#endif
|
2018-11-07 22:15:02 +01:00
|
|
|
|
2020-07-03 05:19:50 -06:00
|
|
|
auto& proc = Processor::current();
|
2020-07-05 14:32:07 -06:00
|
|
|
if (!thread->is_initialized()) {
|
|
|
|
proc.init_context(*thread, false);
|
|
|
|
thread->set_initialized(true);
|
2019-09-07 15:50:44 +02:00
|
|
|
}
|
2022-01-30 11:38:50 +01:00
|
|
|
thread->set_state(Thread::State::Running);
|
2020-07-05 14:32:07 -06:00
|
|
|
|
2021-05-14 07:48:53 +02:00
|
|
|
PerformanceManager::add_context_switch_perf_event(*from_thread, *thread);
|
|
|
|
|
2020-07-05 14:32:07 -06:00
|
|
|
proc.switch_context(from_thread, thread);
|
|
|
|
|
|
|
|
// NOTE: from_thread at this point reflects the thread we were
|
|
|
|
// switched from, and thread reflects Thread::current()
|
2022-01-29 13:57:39 +01:00
|
|
|
enter_current(*from_thread);
|
2021-02-23 20:42:32 +01:00
|
|
|
VERIFY(thread == Thread::current());
|
2018-11-07 22:15:02 +01:00
|
|
|
}
|
|
|
|
|
2022-01-29 13:57:39 +01:00
|
|
|
void Scheduler::enter_current(Thread& prev_thread)
|
2020-07-05 14:32:07 -06:00
|
|
|
{
|
2021-08-29 20:10:24 +02:00
|
|
|
VERIFY(g_scheduler_lock.is_locked_by_current_processor());
|
2021-07-14 21:46:32 -06:00
|
|
|
|
|
|
|
// We already recorded the scheduled time when entering the trap, so this merely accounts for the kernel time since then
|
|
|
|
auto scheduler_time = Scheduler::current_time();
|
|
|
|
prev_thread.update_time_scheduled(scheduler_time, true, true);
|
|
|
|
auto* current_thread = Thread::current();
|
|
|
|
current_thread->update_time_scheduled(scheduler_time, true, false);
|
|
|
|
|
2020-07-05 14:32:07 -06:00
|
|
|
prev_thread.set_active(false);
|
2022-01-30 11:38:50 +01:00
|
|
|
if (prev_thread.state() == Thread::State::Dying) {
|
2020-07-05 14:32:07 -06:00
|
|
|
// If the thread we switched from is marked as dying, then notify
|
|
|
|
// the finalizer. Note that as soon as we leave the scheduler lock
|
|
|
|
// the finalizer may free from_thread!
|
|
|
|
notify_finalizer();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-08-01 14:37:40 -06:00
|
|
|
void Scheduler::leave_on_first_switch(u32 flags)
|
|
|
|
{
|
2020-10-02 22:14:37 +01:00
|
|
|
// This is called when a thread is switched into for the first time.
|
2020-08-01 14:37:40 -06:00
|
|
|
// At this point, enter_current has already be called, but because
|
|
|
|
// Scheduler::context_switch is not in the call stack we need to
|
|
|
|
// clean up and release locks manually here
|
|
|
|
g_scheduler_lock.unlock(flags);
|
2021-08-29 12:43:39 +02:00
|
|
|
|
|
|
|
VERIFY(Processor::current_in_scheduler());
|
|
|
|
Processor::set_current_in_scheduler(false);
|
2020-08-01 14:37:40 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
void Scheduler::prepare_after_exec()
|
|
|
|
{
|
|
|
|
// This is called after exec() when doing a context "switch" into
|
|
|
|
// the new process. This is called from Processor::assume_context
|
2021-08-29 20:10:24 +02:00
|
|
|
VERIFY(g_scheduler_lock.is_locked_by_current_processor());
|
2021-08-29 12:43:39 +02:00
|
|
|
|
|
|
|
VERIFY(!Processor::current_in_scheduler());
|
|
|
|
Processor::set_current_in_scheduler(true);
|
2020-08-01 14:37:40 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
void Scheduler::prepare_for_idle_loop()
|
|
|
|
{
|
|
|
|
// This is called when the CPU finished setting up the idle loop
|
|
|
|
// and is about to run it. We need to acquire he scheduler lock
|
2021-08-29 20:10:24 +02:00
|
|
|
VERIFY(!g_scheduler_lock.is_locked_by_current_processor());
|
2020-08-01 14:37:40 -06:00
|
|
|
g_scheduler_lock.lock();
|
2021-08-29 12:43:39 +02:00
|
|
|
|
|
|
|
VERIFY(!Processor::current_in_scheduler());
|
|
|
|
Processor::set_current_in_scheduler(true);
|
2020-08-01 14:37:40 -06:00
|
|
|
}
|
|
|
|
|
2019-02-04 10:28:12 +01:00
|
|
|
Process* Scheduler::colonel()
|
|
|
|
{
|
2021-02-23 20:42:32 +01:00
|
|
|
VERIFY(s_colonel_process);
|
2019-02-04 10:28:12 +01:00
|
|
|
return s_colonel_process;
|
|
|
|
}
|
|
|
|
|
2021-07-14 21:46:32 -06:00
|
|
|
static u64 current_time_tsc()
|
|
|
|
{
|
|
|
|
return read_tsc();
|
|
|
|
}
|
|
|
|
|
|
|
|
static u64 current_time_monotonic()
|
|
|
|
{
|
|
|
|
// We always need a precise timestamp here, we cannot rely on a coarse timestamp
|
|
|
|
return (u64)TimeManagement::the().monotonic_time(TimePrecision::Precise).to_nanoseconds();
|
|
|
|
}
|
|
|
|
|
2021-02-19 18:41:50 +01:00
|
|
|
UNMAP_AFTER_INIT void Scheduler::initialize()
|
2018-11-07 22:15:02 +01:00
|
|
|
{
|
2021-06-30 07:15:21 +02:00
|
|
|
VERIFY(Processor::is_initialized()); // sanity check
|
2020-06-27 13:42:28 -06:00
|
|
|
|
2021-07-14 21:46:32 -06:00
|
|
|
// Figure out a good scheduling time source
|
|
|
|
if (Processor::current().has_feature(CPUFeature::TSC)) {
|
|
|
|
// TODO: only use if TSC is running at a constant frequency?
|
|
|
|
current_time = current_time_tsc;
|
|
|
|
} else {
|
|
|
|
// TODO: Using HPET is rather slow, can we use any other time source that may be faster?
|
|
|
|
current_time = current_time_monotonic;
|
|
|
|
}
|
|
|
|
|
2020-09-27 08:53:35 -06:00
|
|
|
RefPtr<Thread> idle_thread;
|
2020-07-06 07:27:22 -06:00
|
|
|
g_finalizer_wait_queue = new WaitQueue;
|
2020-06-28 16:05:52 -06:00
|
|
|
|
2020-07-06 07:27:22 -06:00
|
|
|
g_finalizer_has_work.store(false, AK::MemoryOrder::memory_order_release);
|
2021-09-07 12:53:28 +02:00
|
|
|
s_colonel_process = Process::create_kernel_process(idle_thread, KString::must_create("colonel"), idle_loop, nullptr, 1, Process::RegisterProcess::No).leak_ref();
|
2021-02-23 20:42:32 +01:00
|
|
|
VERIFY(s_colonel_process);
|
|
|
|
VERIFY(idle_thread);
|
2020-07-06 07:27:22 -06:00
|
|
|
idle_thread->set_priority(THREAD_PRIORITY_MIN);
|
2021-09-06 12:44:27 +02:00
|
|
|
idle_thread->set_name(KString::must_create("idle thread #0"));
|
2020-06-28 15:34:31 -06:00
|
|
|
|
2020-07-06 07:27:22 -06:00
|
|
|
set_idle_thread(idle_thread);
|
|
|
|
}
|
|
|
|
|
2021-02-19 21:29:46 +01:00
|
|
|
UNMAP_AFTER_INIT void Scheduler::set_idle_thread(Thread* idle_thread)
|
2020-07-06 07:27:22 -06:00
|
|
|
{
|
2021-01-28 20:07:41 -07:00
|
|
|
idle_thread->set_idle_thread();
|
2020-06-28 15:34:31 -06:00
|
|
|
Processor::current().set_idle_thread(*idle_thread);
|
2021-07-25 19:48:55 +02:00
|
|
|
Processor::set_current_thread(*idle_thread);
|
2020-07-06 07:27:22 -06:00
|
|
|
}
|
2020-06-28 16:05:52 -06:00
|
|
|
|
2021-02-19 21:29:46 +01:00
|
|
|
UNMAP_AFTER_INIT Thread* Scheduler::create_ap_idle_thread(u32 cpu)
|
2020-07-06 07:27:22 -06:00
|
|
|
{
|
2021-02-23 20:42:32 +01:00
|
|
|
VERIFY(cpu != 0);
|
2020-07-06 07:27:22 -06:00
|
|
|
// This function is called on the bsp, but creates an idle thread for another AP
|
2021-05-05 16:48:26 +00:00
|
|
|
VERIFY(Processor::is_bootstrap_processor());
|
2020-07-06 07:27:22 -06:00
|
|
|
|
2021-02-23 20:42:32 +01:00
|
|
|
VERIFY(s_colonel_process);
|
2021-12-28 09:38:41 +01:00
|
|
|
Thread* idle_thread = s_colonel_process->create_kernel_thread(idle_loop, nullptr, THREAD_PRIORITY_MIN, MUST(KString::formatted("idle thread #{}", cpu)), 1 << cpu, false);
|
2021-02-23 20:42:32 +01:00
|
|
|
VERIFY(idle_thread);
|
2020-07-06 07:27:22 -06:00
|
|
|
return idle_thread;
|
2018-11-07 22:15:02 +01:00
|
|
|
}
|
2018-11-08 00:24:59 +01:00
|
|
|
|
2021-07-14 21:46:32 -06:00
|
|
|
void Scheduler::add_time_scheduled(u64 time_to_add, bool is_kernel)
|
|
|
|
{
|
2021-08-08 12:34:30 +02:00
|
|
|
g_total_time_scheduled.with([&](auto& total_time_scheduled) {
|
|
|
|
total_time_scheduled.total += time_to_add;
|
|
|
|
if (is_kernel)
|
|
|
|
total_time_scheduled.total_kernel += time_to_add;
|
|
|
|
});
|
2021-07-14 21:46:32 -06:00
|
|
|
}
|
|
|
|
|
2020-03-09 16:24:29 +02:00
|
|
|
void Scheduler::timer_tick(const RegisterState& regs)
|
2018-11-08 00:24:59 +01:00
|
|
|
{
|
2021-02-23 20:42:32 +01:00
|
|
|
VERIFY_INTERRUPTS_DISABLED();
|
2021-08-22 12:21:31 +02:00
|
|
|
VERIFY(Processor::current_in_irq());
|
2020-07-30 23:38:15 +02:00
|
|
|
|
2021-12-29 01:01:27 +01:00
|
|
|
auto* current_thread = Processor::current_thread();
|
2020-06-28 15:34:31 -06:00
|
|
|
if (!current_thread)
|
2018-11-08 00:24:59 +01:00
|
|
|
return;
|
|
|
|
|
2021-01-25 13:19:34 -07:00
|
|
|
// Sanity checks
|
2021-02-23 20:42:32 +01:00
|
|
|
VERIFY(current_thread->current_trap());
|
|
|
|
VERIFY(current_thread->current_trap()->regs == ®s);
|
2021-01-25 13:19:34 -07:00
|
|
|
|
2021-01-26 22:45:30 -07:00
|
|
|
#if !SCHEDULE_ON_ALL_PROCESSORS
|
2021-05-05 16:48:26 +00:00
|
|
|
if (!Processor::is_bootstrap_processor())
|
2020-10-25 09:13:47 -06:00
|
|
|
return; // TODO: This prevents scheduling on other CPUs!
|
2021-01-26 22:45:30 -07:00
|
|
|
#endif
|
2021-03-02 17:19:35 +01:00
|
|
|
|
2021-07-14 21:46:32 -06:00
|
|
|
if (current_thread->process().is_kernel_process()) {
|
|
|
|
// Because the previous mode when entering/exiting kernel threads never changes
|
|
|
|
// we never update the time scheduled. So we need to update it manually on the
|
|
|
|
// timer interrupt
|
|
|
|
current_thread->update_time_scheduled(current_time(), true, false);
|
|
|
|
}
|
2021-07-14 12:05:59 -06:00
|
|
|
|
2021-07-14 21:46:32 -06:00
|
|
|
if (current_thread->previous_mode() == Thread::PreviousMode::UserMode && current_thread->should_die() && !current_thread->is_blocked()) {
|
2021-08-22 01:49:22 +02:00
|
|
|
SpinlockLocker scheduler_lock(g_scheduler_lock);
|
2021-08-22 12:37:50 +02:00
|
|
|
dbgln_if(SCHEDULER_DEBUG, "Scheduler[{}]: Terminating user mode thread {}", Processor::current_id(), *current_thread);
|
2022-01-30 11:38:50 +01:00
|
|
|
current_thread->set_state(Thread::State::Dying);
|
2021-07-14 21:46:32 -06:00
|
|
|
Processor::current().invoke_scheduler_async();
|
|
|
|
return;
|
2021-07-13 11:44:30 -06:00
|
|
|
}
|
2018-11-08 00:24:59 +01:00
|
|
|
|
2021-07-14 21:46:32 -06:00
|
|
|
if (current_thread->tick())
|
|
|
|
return;
|
|
|
|
|
2021-07-11 20:12:42 -06:00
|
|
|
if (!current_thread->is_idle_thread() && !peek_next_runnable_thread()) {
|
|
|
|
// If no other thread is ready to be scheduled we don't need to
|
|
|
|
// switch to the idle thread. Just give the current thread another
|
|
|
|
// time slice and let it run!
|
|
|
|
current_thread->set_ticks_left(time_slice_for(*current_thread));
|
|
|
|
current_thread->did_schedule();
|
2021-08-22 12:37:50 +02:00
|
|
|
dbgln_if(SCHEDULER_DEBUG, "Scheduler[{}]: No other threads ready, give {} another timeslice", Processor::current_id(), *current_thread);
|
2021-07-11 20:12:42 -06:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2021-02-23 20:42:32 +01:00
|
|
|
VERIFY_INTERRUPTS_DISABLED();
|
2021-08-22 12:21:31 +02:00
|
|
|
VERIFY(Processor::current_in_irq());
|
2020-06-27 13:42:28 -06:00
|
|
|
Processor::current().invoke_scheduler_async();
|
2018-11-08 00:24:59 +01:00
|
|
|
}
|
2019-09-14 19:44:22 +02:00
|
|
|
|
2020-06-27 13:42:28 -06:00
|
|
|
void Scheduler::invoke_async()
|
2019-09-14 19:44:22 +02:00
|
|
|
{
|
2021-02-23 20:42:32 +01:00
|
|
|
VERIFY_INTERRUPTS_DISABLED();
|
2021-08-22 12:21:31 +02:00
|
|
|
VERIFY(!Processor::current_in_irq());
|
2020-08-01 14:37:40 -06:00
|
|
|
|
|
|
|
// Since this function is called when leaving critical sections (such
|
2021-08-22 01:37:17 +02:00
|
|
|
// as a Spinlock), we need to check if we're not already doing this
|
2020-08-01 14:37:40 -06:00
|
|
|
// to prevent recursion
|
2021-08-29 12:43:39 +02:00
|
|
|
if (!Processor::current_in_scheduler())
|
2020-08-01 14:37:40 -06:00
|
|
|
pick_next();
|
2019-09-14 19:44:22 +02:00
|
|
|
}
|
|
|
|
|
2020-07-05 14:32:07 -06:00
|
|
|
void Scheduler::notify_finalizer()
|
|
|
|
{
|
2021-12-29 01:03:09 +01:00
|
|
|
if (!g_finalizer_has_work.exchange(true, AK::MemoryOrder::memory_order_acq_rel))
|
2020-07-05 14:32:07 -06:00
|
|
|
g_finalizer_wait_queue->wake_all();
|
|
|
|
}
|
|
|
|
|
2020-11-16 20:51:34 -07:00
|
|
|
void Scheduler::idle_loop(void*)
|
2019-09-14 19:44:22 +02:00
|
|
|
{
|
2020-10-28 16:06:16 -06:00
|
|
|
auto& proc = Processor::current();
|
2021-08-22 12:37:50 +02:00
|
|
|
dbgln("Scheduler[{}]: idle loop running", proc.id());
|
2021-02-23 20:42:32 +01:00
|
|
|
VERIFY(are_interrupts_enabled());
|
2020-07-06 07:27:22 -06:00
|
|
|
|
2019-09-14 19:44:22 +02:00
|
|
|
for (;;) {
|
2020-10-28 16:06:16 -06:00
|
|
|
proc.idle_begin();
|
2019-09-14 19:44:22 +02:00
|
|
|
asm("hlt");
|
2020-07-30 21:46:06 +02:00
|
|
|
|
2020-10-28 16:06:16 -06:00
|
|
|
proc.idle_end();
|
2021-02-23 20:42:32 +01:00
|
|
|
VERIFY_INTERRUPTS_ENABLED();
|
2021-01-26 22:45:30 -07:00
|
|
|
#if SCHEDULE_ON_ALL_PROCESSORS
|
|
|
|
yield();
|
|
|
|
#else
|
2021-08-22 12:37:50 +02:00
|
|
|
if (Processor::current_id() == 0)
|
2020-07-30 21:46:06 +02:00
|
|
|
yield();
|
2021-01-26 22:45:30 -07:00
|
|
|
#endif
|
2019-09-14 19:44:22 +02:00
|
|
|
}
|
|
|
|
}
|
2020-02-16 01:27:42 +01:00
|
|
|
|
2021-07-15 14:54:19 -06:00
|
|
|
void Scheduler::dump_scheduler_state(bool with_stack_traces)
|
2021-04-18 19:57:17 +02:00
|
|
|
{
|
2021-07-15 14:54:19 -06:00
|
|
|
dump_thread_list(with_stack_traces);
|
2021-04-18 19:57:17 +02:00
|
|
|
}
|
|
|
|
|
2021-06-12 23:32:56 +10:00
|
|
|
bool Scheduler::is_initialized()
|
|
|
|
{
|
2021-06-28 17:06:26 +02:00
|
|
|
// The scheduler is initialized iff the idle thread exists
|
2021-06-12 23:32:56 +10:00
|
|
|
return Processor::idle_thread() != nullptr;
|
|
|
|
}
|
|
|
|
|
2021-07-14 21:46:32 -06:00
|
|
|
TotalTimeScheduled Scheduler::get_total_time_scheduled()
|
2021-07-14 12:05:59 -06:00
|
|
|
{
|
2021-08-08 12:34:30 +02:00
|
|
|
return g_total_time_scheduled.with([&](auto& total_time_scheduled) { return total_time_scheduled; });
|
2021-07-14 12:05:59 -06:00
|
|
|
}
|
|
|
|
|
2021-07-15 14:54:19 -06:00
|
|
|
void dump_thread_list(bool with_stack_traces)
|
2021-04-18 19:57:17 +02:00
|
|
|
{
|
2021-08-22 12:37:50 +02:00
|
|
|
dbgln("Scheduler thread list for processor {}:", Processor::current_id());
|
2021-04-18 19:57:17 +02:00
|
|
|
|
|
|
|
auto get_cs = [](Thread& thread) -> u16 {
|
|
|
|
if (!thread.current_trap())
|
2021-06-26 19:57:16 +02:00
|
|
|
return thread.regs().cs;
|
2021-04-18 19:57:17 +02:00
|
|
|
return thread.get_register_dump_from_stack().cs;
|
|
|
|
};
|
|
|
|
|
|
|
|
auto get_eip = [](Thread& thread) -> u32 {
|
2021-06-28 17:06:36 +02:00
|
|
|
if (!thread.current_trap())
|
2021-07-18 16:50:08 -07:00
|
|
|
return thread.regs().ip();
|
|
|
|
return thread.get_register_dump_from_stack().ip();
|
2021-04-18 19:57:17 +02:00
|
|
|
};
|
|
|
|
|
2021-05-16 02:36:52 -07:00
|
|
|
Thread::for_each([&](Thread& thread) {
|
2021-04-18 19:57:17 +02:00
|
|
|
switch (thread.state()) {
|
2022-01-30 11:38:50 +01:00
|
|
|
case Thread::State::Dying:
|
2021-05-13 19:36:31 +03:00
|
|
|
dmesgln(" {:14} {:30} @ {:04x}:{:08x} Finalizable: {}, (nsched: {})",
|
2021-04-18 19:57:17 +02:00
|
|
|
thread.state_string(),
|
|
|
|
thread,
|
|
|
|
get_cs(thread),
|
|
|
|
get_eip(thread),
|
|
|
|
thread.is_finalizable(),
|
|
|
|
thread.times_scheduled());
|
|
|
|
break;
|
|
|
|
default:
|
2021-05-13 19:36:31 +03:00
|
|
|
dmesgln(" {:14} Pr:{:2} {:30} @ {:04x}:{:08x} (nsched: {})",
|
2021-04-18 19:57:17 +02:00
|
|
|
thread.state_string(),
|
|
|
|
thread.priority(),
|
|
|
|
thread,
|
|
|
|
get_cs(thread),
|
|
|
|
get_eip(thread),
|
|
|
|
thread.times_scheduled());
|
|
|
|
break;
|
|
|
|
}
|
2022-01-15 21:19:41 +02:00
|
|
|
if (with_stack_traces) {
|
|
|
|
auto trace_or_error = thread.backtrace();
|
|
|
|
if (!trace_or_error.is_error()) {
|
|
|
|
auto trace = trace_or_error.release_value();
|
|
|
|
dbgln("Backtrace:");
|
|
|
|
kernelputstr(trace->characters(), trace->length());
|
|
|
|
}
|
|
|
|
}
|
2021-08-15 12:38:02 +02:00
|
|
|
return IterationDecision::Continue;
|
2021-04-18 19:57:17 +02:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2020-02-16 01:27:42 +01:00
|
|
|
}
|