2020-01-18 09:38:21 +01:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2018-2020, Andreas Kling <kling@serenityos.org>
|
|
|
|
* All rights reserved.
|
|
|
|
*
|
|
|
|
* Redistribution and use in source and binary forms, with or without
|
|
|
|
* modification, are permitted provided that the following conditions are met:
|
|
|
|
*
|
|
|
|
* 1. Redistributions of source code must retain the above copyright notice, this
|
|
|
|
* list of conditions and the following disclaimer.
|
|
|
|
*
|
|
|
|
* 2. Redistributions in binary form must reproduce the above copyright notice,
|
|
|
|
* this list of conditions and the following disclaimer in the documentation
|
|
|
|
* and/or other materials provided with the distribution.
|
|
|
|
*
|
|
|
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
|
|
|
|
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
|
|
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
|
|
|
* DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
|
|
|
|
* FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
|
|
|
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
|
|
|
|
* SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
|
|
|
|
* CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
|
|
|
|
* OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
|
|
|
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
*/
|
|
|
|
|
2019-12-30 18:46:17 +01:00
|
|
|
#include <AK/QuickSort.h>
|
2020-08-01 14:37:40 -06:00
|
|
|
#include <AK/ScopeGuard.h>
|
2019-02-06 15:05:47 +01:00
|
|
|
#include <AK/TemporaryChange.h>
|
2020-08-03 09:43:19 -06:00
|
|
|
#include <AK/Time.h>
|
2019-06-07 19:29:34 +02:00
|
|
|
#include <Kernel/Process.h>
|
2019-12-11 20:36:56 +01:00
|
|
|
#include <Kernel/Profiling.h>
|
2019-06-07 19:29:34 +02:00
|
|
|
#include <Kernel/RTC.h>
|
|
|
|
#include <Kernel/Scheduler.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>
|
2019-12-27 11:58:28 +11:00
|
|
|
#include <Kernel/TimerQueue.h>
|
2018-11-07 22:15:02 +01:00
|
|
|
|
2019-12-27 00:46:38 +01:00
|
|
|
//#define LOG_EVERY_CONTEXT_SWITCH
|
|
|
|
//#define SCHEDULER_DEBUG
|
|
|
|
//#define SCHEDULER_RUNNABLE_DEBUG
|
|
|
|
|
2020-02-16 01:27:42 +01:00
|
|
|
namespace Kernel {
|
|
|
|
|
2020-08-01 14:37:40 -06:00
|
|
|
class SchedulerPerProcessorData {
|
|
|
|
AK_MAKE_NONCOPYABLE(SchedulerPerProcessorData);
|
|
|
|
AK_MAKE_NONMOVABLE(SchedulerPerProcessorData);
|
|
|
|
|
|
|
|
public:
|
|
|
|
SchedulerPerProcessorData() = default;
|
|
|
|
|
2020-09-27 08:53:35 -06:00
|
|
|
WeakPtr<Thread> m_pending_beneficiary;
|
2020-09-15 13:53:15 -06:00
|
|
|
const char* m_pending_donate_reason { nullptr };
|
2020-08-01 14:37:40 -06:00
|
|
|
bool m_in_scheduler { true };
|
|
|
|
};
|
|
|
|
|
2019-08-07 20:43:54 +02:00
|
|
|
SchedulerData* g_scheduler_data;
|
2020-06-28 15:34:31 -06:00
|
|
|
RecursiveSpinLock g_scheduler_lock;
|
2019-07-19 17:21:13 +02:00
|
|
|
|
|
|
|
void Scheduler::init_thread(Thread& thread)
|
|
|
|
{
|
2020-06-28 22:36:12 -06:00
|
|
|
ASSERT(g_scheduler_data);
|
2019-07-19 17:21:13 +02:00
|
|
|
g_scheduler_data->m_nonrunnable_threads.append(thread);
|
|
|
|
}
|
|
|
|
|
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)
|
2020-06-28 15:34:31 -06:00
|
|
|
if (&thread == Processor::current().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
|
|
|
|
2019-03-23 22:03:17 +01:00
|
|
|
Thread* g_finalizer;
|
2019-12-01 19:17:17 +01:00
|
|
|
WaitQueue* g_finalizer_wait_queue;
|
2020-07-30 21:46:06 +02:00
|
|
|
Atomic<bool> g_finalizer_has_work { false };
|
2018-11-07 22:15:02 +01:00
|
|
|
static Process* s_colonel_process;
|
|
|
|
|
2020-06-27 13:42:28 -06:00
|
|
|
void Scheduler::start()
|
2018-11-07 22:15:02 +01:00
|
|
|
{
|
|
|
|
ASSERT_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();
|
2020-08-01 14:37:40 -06:00
|
|
|
processor.set_scheduler_data(*new SchedulerPerProcessorData());
|
2020-06-28 22:36:12 -06:00
|
|
|
ASSERT(processor.is_initialized());
|
2020-06-28 15:34:31 -06:00
|
|
|
auto& idle_thread = *processor.idle_thread();
|
2020-06-28 22:36:12 -06:00
|
|
|
ASSERT(processor.current_thread() == &idle_thread);
|
|
|
|
ASSERT(processor.idle_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);
|
2020-06-28 15:34:31 -06:00
|
|
|
idle_thread.set_state(Thread::Running);
|
2020-06-28 22:36:12 -06:00
|
|
|
ASSERT(idle_thread.affinity() == (1u << processor.id()));
|
2020-06-28 15:34:31 -06:00
|
|
|
processor.initialize_context_switching(idle_thread);
|
2020-06-27 13:42:28 -06:00
|
|
|
ASSERT_NOT_REACHED();
|
|
|
|
}
|
2018-11-07 22:15:02 +01:00
|
|
|
|
2020-06-27 13:42:28 -06:00
|
|
|
bool Scheduler::pick_next()
|
|
|
|
{
|
|
|
|
ASSERT_INTERRUPTS_DISABLED();
|
2018-11-07 22:15:02 +01:00
|
|
|
|
2020-06-28 15:34:31 -06:00
|
|
|
auto current_thread = Thread::current();
|
2019-03-13 13:13:23 +01:00
|
|
|
|
2020-08-01 14:37:40 -06:00
|
|
|
// Set the m_in_scheduler flag before acquiring the spinlock. This
|
|
|
|
// prevents a recursive call into Scheduler::invoke_async upon
|
|
|
|
// leaving the scheduler lock.
|
|
|
|
ScopedCritical critical;
|
2020-09-15 13:53:15 -06:00
|
|
|
auto& scheduler_data = Processor::current().get_scheduler_data();
|
|
|
|
scheduler_data.m_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!
|
|
|
|
auto& scheduler_data = Processor::current().get_scheduler_data();
|
|
|
|
ASSERT(scheduler_data.m_in_scheduler);
|
|
|
|
scheduler_data.m_in_scheduler = false;
|
|
|
|
});
|
|
|
|
|
2020-06-28 15:34:31 -06:00
|
|
|
ScopedSpinLock lock(g_scheduler_lock);
|
|
|
|
|
2020-08-10 14:05:24 -06:00
|
|
|
if (current_thread->should_die() && current_thread->state() == Thread::Running) {
|
|
|
|
// Rather than immediately killing threads, yanking the kernel stack
|
|
|
|
// away from them (which can lead to e.g. reference leaks), we always
|
|
|
|
// allow Thread::wait_on to return. This allows the kernel stack to
|
|
|
|
// clean up and eventually we'll get here shortly before transitioning
|
|
|
|
// back to user mode (from Processor::exit_trap). At this point we
|
|
|
|
// no longer want to schedule this thread. We can't wait until
|
|
|
|
// Scheduler::enter_current because we don't want to allow it to
|
|
|
|
// transition back to user mode.
|
|
|
|
#ifdef SCHEDULER_DEBUG
|
|
|
|
dbg() << "Scheduler[" << Processor::current().id() << "]: Thread " << *current_thread << " is dying";
|
|
|
|
#endif
|
|
|
|
current_thread->set_state(Thread::Dying);
|
|
|
|
}
|
|
|
|
|
2019-07-17 14:17:49 +02:00
|
|
|
#ifdef SCHEDULER_RUNNABLE_DEBUG
|
2020-10-25 22:02:14 -06:00
|
|
|
dbg() << "Scheduler[" << Processor::current().id() << "]: Non-runnables:";
|
|
|
|
Scheduler::for_each_nonrunnable([&](Thread& thread) -> IterationDecision {
|
2020-12-07 21:29:41 -07:00
|
|
|
if (thread.state() == Thread::Dying)
|
2020-12-25 17:05:05 +01:00
|
|
|
dbg() << " " << String::format("%-12s", thread.state_string()) << " " << thread << " @ " << String::formatted("{:04x}:{:08x}", thread.tss().cs, thread.tss().eip) << " Finalizable: " << thread.is_finalizable();
|
2020-08-02 16:59:01 -06:00
|
|
|
else
|
2020-12-25 17:05:05 +01:00
|
|
|
dbg() << " " << String::format("%-12s", thread.state_string()) << " " << thread << " @ " << String::formatted("{:04x}:{:08x}", thread.tss().cs, thread.tss().eip);
|
2019-07-19 12:18:40 +02:00
|
|
|
return IterationDecision::Continue;
|
|
|
|
});
|
2019-05-20 03:44:45 +02:00
|
|
|
|
2020-10-25 22:02:14 -06:00
|
|
|
dbg() << "Scheduler[" << Processor::current().id() << "]: Runnables:";
|
2019-12-30 18:46:17 +01:00
|
|
|
Scheduler::for_each_runnable([](Thread& thread) -> IterationDecision {
|
2020-12-25 17:05:05 +01:00
|
|
|
dbg() << " " << String::format("%3u", thread.effective_priority()) << "/" << String::format("%2u", thread.priority()) << " " << String::format("%-12s", thread.state_string()) << " " << thread << " @ " << String::formatted("{:04x}:{:08x}", thread.tss().cs, thread.tss().eip);
|
2019-12-30 18:46:17 +01:00
|
|
|
return IterationDecision::Continue;
|
|
|
|
});
|
2018-11-07 22:15:02 +01:00
|
|
|
#endif
|
|
|
|
|
2020-09-15 13:53:15 -06:00
|
|
|
Thread* thread_to_schedule = nullptr;
|
|
|
|
|
AK: Make RefPtr, NonnullRefPtr, WeakPtr thread safe
This makes most operations thread safe, especially so that they
can safely be used in the Kernel. This includes obtaining a strong
reference from a weak reference, which now requires an explicit
call to WeakPtr::strong_ref(). Another major change is that
Weakable::make_weak_ref() may require the explicit target type.
Previously we used reinterpret_cast in WeakPtr, assuming that it
can be properly converted. But WeakPtr does not necessarily have
the knowledge to be able to do this. Instead, we now ask the class
itself to deliver a WeakPtr to the type that we want.
Also, WeakLink is no longer specific to a target type. The reason
for this is that we want to be able to safely convert e.g. WeakPtr<T>
to WeakPtr<U>, and before this we just reinterpret_cast the internal
WeakLink<T> to WeakLink<U>, which is a bold assumption that it would
actually produce the correct code. Instead, WeakLink now operates
on just a raw pointer and we only make those constructors/operators
available if we can verify that it can be safely cast.
In order to guarantee thread safety, we now use the least significant
bit in the pointer for locking purposes. This also means that only
properly aligned pointers can be used.
2020-09-29 16:26:13 -06:00
|
|
|
auto pending_beneficiary = scheduler_data.m_pending_beneficiary.strong_ref();
|
2019-12-30 18:46:17 +01:00
|
|
|
Vector<Thread*, 128> sorted_runnables;
|
2020-09-15 13:53:15 -06:00
|
|
|
for_each_runnable([&](auto& thread) {
|
2020-10-25 22:02:14 -06:00
|
|
|
if ((thread.affinity() & (1u << Processor::current().id())) == 0)
|
|
|
|
return IterationDecision::Continue;
|
|
|
|
if (thread.state() == Thread::Running && &thread != current_thread)
|
|
|
|
return IterationDecision::Continue;
|
|
|
|
sorted_runnables.append(&thread);
|
AK: Make RefPtr, NonnullRefPtr, WeakPtr thread safe
This makes most operations thread safe, especially so that they
can safely be used in the Kernel. This includes obtaining a strong
reference from a weak reference, which now requires an explicit
call to WeakPtr::strong_ref(). Another major change is that
Weakable::make_weak_ref() may require the explicit target type.
Previously we used reinterpret_cast in WeakPtr, assuming that it
can be properly converted. But WeakPtr does not necessarily have
the knowledge to be able to do this. Instead, we now ask the class
itself to deliver a WeakPtr to the type that we want.
Also, WeakLink is no longer specific to a target type. The reason
for this is that we want to be able to safely convert e.g. WeakPtr<T>
to WeakPtr<U>, and before this we just reinterpret_cast the internal
WeakLink<T> to WeakLink<U>, which is a bold assumption that it would
actually produce the correct code. Instead, WeakLink now operates
on just a raw pointer and we only make those constructors/operators
available if we can verify that it can be safely cast.
In order to guarantee thread safety, we now use the least significant
bit in the pointer for locking purposes. This also means that only
properly aligned pointers can be used.
2020-09-29 16:26:13 -06:00
|
|
|
if (&thread == pending_beneficiary) {
|
2020-09-15 13:53:15 -06:00
|
|
|
thread_to_schedule = &thread;
|
|
|
|
return IterationDecision::Break;
|
|
|
|
}
|
2019-12-30 18:46:17 +01:00
|
|
|
return IterationDecision::Continue;
|
|
|
|
});
|
2019-05-18 20:24:55 +02:00
|
|
|
|
2020-09-15 13:53:15 -06:00
|
|
|
if (thread_to_schedule) {
|
|
|
|
// The thread we're supposed to donate to still exists
|
|
|
|
const char* reason = scheduler_data.m_pending_donate_reason;
|
|
|
|
scheduler_data.m_pending_beneficiary = nullptr;
|
|
|
|
scheduler_data.m_pending_donate_reason = nullptr;
|
|
|
|
|
|
|
|
// 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();
|
|
|
|
|
|
|
|
#ifdef SCHEDULER_DEBUG
|
|
|
|
dbg() << "Processing pending donate to " << *thread_to_schedule << " reason=" << reason;
|
|
|
|
#endif
|
|
|
|
return donate_to_and_switch(thread_to_schedule, reason);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Either we're not donating or the beneficiary disappeared.
|
|
|
|
// Either way clear any pending information
|
|
|
|
scheduler_data.m_pending_beneficiary = nullptr;
|
|
|
|
scheduler_data.m_pending_donate_reason = nullptr;
|
|
|
|
|
|
|
|
quick_sort(sorted_runnables, [](auto& a, auto& b) { return a->effective_priority() >= b->effective_priority(); });
|
2018-11-07 22:15:02 +01:00
|
|
|
|
2019-12-30 18:46:17 +01:00
|
|
|
for (auto* thread : sorted_runnables) {
|
Kernel: Allow process with multiple threads to call exec and exit
This allows a process wich has more than 1 thread to call exec, even
from a thread. This kills all the other threads, but it won't wait for
them to finish, just makes sure that they are not in a running/runable
state.
In the case where a thread does exec, the new program PID will be the
thread TID, to keep the PID == TID in the new process.
This introduces a new function inside the Process class,
kill_threads_except_self which is called on exit() too (exit with
multiple threads wasn't properly working either).
Inside the Lock class, there is the need for a new function,
clear_waiters, which removes all the waiters from the
Process::big_lock. This is needed since after a exit/exec, there should
be no other threads waiting for this lock, the threads should be simply
killed. Only queued threads should wait for this lock at this point,
since blocked threads are handled in set_should_die.
2020-02-18 14:28:28 +02:00
|
|
|
if (thread->process().exec_tid() && thread->process().exec_tid() != thread->tid())
|
|
|
|
continue;
|
|
|
|
|
2019-12-30 18:46:17 +01:00
|
|
|
ASSERT(thread->state() == Thread::Runnable || thread->state() == Thread::Running);
|
|
|
|
|
|
|
|
if (!thread_to_schedule) {
|
|
|
|
thread->m_extra_priority = 0;
|
|
|
|
thread_to_schedule = thread;
|
|
|
|
} else {
|
|
|
|
thread->m_extra_priority++;
|
2018-11-07 22:15:02 +01:00
|
|
|
}
|
|
|
|
}
|
2019-12-27 00:46:38 +01:00
|
|
|
|
2019-12-30 18:46:17 +01:00
|
|
|
if (!thread_to_schedule)
|
2020-06-28 15:34:31 -06:00
|
|
|
thread_to_schedule = Processor::current().idle_thread();
|
2019-12-30 18:46:17 +01:00
|
|
|
|
|
|
|
#ifdef SCHEDULER_DEBUG
|
2020-06-28 22:36:12 -06:00
|
|
|
dbg() << "Scheduler[" << Processor::current().id() << "]: Switch to " << *thread_to_schedule << " @ " << String::format("%04x:%08x", thread_to_schedule->tss().cs, thread_to_schedule->tss().eip);
|
2019-12-30 18:46:17 +01:00
|
|
|
#endif
|
|
|
|
|
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();
|
|
|
|
|
2020-12-03 22:12:50 -07:00
|
|
|
thread_to_schedule->set_ticks_left(time_slice_for(*thread_to_schedule));
|
2020-07-05 14:32:07 -06:00
|
|
|
return context_switch(thread_to_schedule);
|
2018-11-07 22:15:02 +01:00
|
|
|
}
|
|
|
|
|
2020-06-27 13:42:28 -06:00
|
|
|
bool Scheduler::yield()
|
|
|
|
{
|
2020-07-03 05:19:50 -06:00
|
|
|
InterruptDisabler disabler;
|
2020-06-28 15:34:31 -06:00
|
|
|
auto& proc = Processor::current();
|
2020-09-15 13:53:15 -06:00
|
|
|
auto& scheduler_data = proc.get_scheduler_data();
|
|
|
|
|
|
|
|
// Clear any pending beneficiary
|
|
|
|
scheduler_data.m_pending_beneficiary = nullptr;
|
|
|
|
scheduler_data.m_pending_donate_reason = nullptr;
|
2020-08-01 14:37:40 -06:00
|
|
|
|
2020-06-28 15:34:31 -06:00
|
|
|
auto current_thread = Thread::current();
|
|
|
|
#ifdef SCHEDULER_DEBUG
|
2020-07-03 05:19:50 -06:00
|
|
|
dbg() << "Scheduler[" << proc.id() << "]: yielding thread " << *current_thread << " in_irq: " << proc.in_irq();
|
2020-06-27 13:42:28 -06:00
|
|
|
#endif
|
2020-06-28 15:34:31 -06:00
|
|
|
ASSERT(current_thread != nullptr);
|
2020-07-03 05:19:50 -06:00
|
|
|
if (proc.in_irq() || proc.in_critical()) {
|
|
|
|
// 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
|
2020-06-28 15:34:31 -06:00
|
|
|
proc.invoke_scheduler_async();
|
2020-07-03 05:19:50 -06:00
|
|
|
return false;
|
2020-08-01 14:37:40 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!Scheduler::pick_next())
|
2020-06-27 13:42:28 -06:00
|
|
|
return false;
|
2020-06-28 15:34:31 -06:00
|
|
|
#ifdef SCHEDULER_DEBUG
|
2020-08-01 14:37:40 -06:00
|
|
|
dbg() << "Scheduler[" << Processor::current().id() << "]: yield returns to thread " << *current_thread << " in_irq: " << Processor::current().in_irq();
|
2020-06-27 13:42:28 -06:00
|
|
|
#endif
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2020-12-20 16:09:48 -07:00
|
|
|
bool Scheduler::donate_to_and_switch(Thread* beneficiary, [[maybe_unused]] const char* reason)
|
2020-09-15 13:53:15 -06:00
|
|
|
{
|
|
|
|
ASSERT(g_scheduler_lock.own_lock());
|
|
|
|
|
|
|
|
auto& proc = Processor::current();
|
|
|
|
ASSERT(proc.in_critical() == 1);
|
|
|
|
|
|
|
|
unsigned ticks_left = Thread::current()->ticks_left();
|
|
|
|
if (!beneficiary || beneficiary->state() != Thread::Runnable || ticks_left <= 1)
|
|
|
|
return Scheduler::yield();
|
|
|
|
|
|
|
|
unsigned ticks_to_donate = min(ticks_left - 1, time_slice_for(*beneficiary));
|
|
|
|
#ifdef SCHEDULER_DEBUG
|
|
|
|
dbg() << "Scheduler[" << proc.id() << "]: Donating " << ticks_to_donate << " ticks to " << *beneficiary << ", reason=" << reason;
|
|
|
|
#endif
|
|
|
|
beneficiary->set_ticks_left(ticks_to_donate);
|
|
|
|
|
|
|
|
return Scheduler::context_switch(beneficiary);
|
|
|
|
}
|
|
|
|
|
2020-09-27 08:53:35 -06:00
|
|
|
bool Scheduler::donate_to(RefPtr<Thread>& beneficiary, const char* reason)
|
2019-02-07 11:12:23 +01:00
|
|
|
{
|
2020-08-01 20:04:56 -06:00
|
|
|
ASSERT(beneficiary);
|
2020-08-08 17:32:34 +02:00
|
|
|
|
2020-09-15 13:53:15 -06:00
|
|
|
if (beneficiary == Thread::current())
|
|
|
|
return Scheduler::yield();
|
|
|
|
|
2020-08-01 14:37:40 -06:00
|
|
|
// Set the m_in_scheduler flag before acquiring the spinlock. This
|
|
|
|
// prevents a recursive call into Scheduler::invoke_async upon
|
|
|
|
// leaving the scheduler lock.
|
|
|
|
ScopedCritical critical;
|
2020-07-03 05:19:50 -06:00
|
|
|
auto& proc = Processor::current();
|
2020-09-15 13:53:15 -06:00
|
|
|
auto& scheduler_data = proc.get_scheduler_data();
|
|
|
|
scheduler_data.m_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!
|
|
|
|
auto& scheduler_data = Processor::current().get_scheduler_data();
|
|
|
|
ASSERT(scheduler_data.m_in_scheduler);
|
|
|
|
scheduler_data.m_in_scheduler = false;
|
|
|
|
});
|
|
|
|
|
2020-07-03 05:19:50 -06:00
|
|
|
ASSERT(!proc.in_irq());
|
2019-04-17 12:41:51 +02:00
|
|
|
|
2020-09-15 13:53:15 -06:00
|
|
|
if (proc.in_critical() > 1) {
|
2020-11-29 17:09:14 -07:00
|
|
|
scheduler_data.m_pending_beneficiary = beneficiary; // Save the beneficiary
|
2020-09-15 13:53:15 -06:00
|
|
|
scheduler_data.m_pending_donate_reason = reason;
|
2020-07-03 05:19:50 -06:00
|
|
|
proc.invoke_scheduler_async();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2020-09-15 13:53:15 -06:00
|
|
|
ScopedSpinLock lock(g_scheduler_lock);
|
2019-02-07 11:12:23 +01:00
|
|
|
|
2020-09-15 13:53:15 -06:00
|
|
|
// "Leave" the critical section before switching context. Since we
|
|
|
|
// still hold the scheduler lock, we're not actually leaving it.
|
|
|
|
// Processor::switch_context expects Processor::in_critical() to be 1
|
|
|
|
critical.leave();
|
|
|
|
donate_to_and_switch(beneficiary, reason);
|
2019-04-17 12:41:51 +02:00
|
|
|
return false;
|
2019-02-07 11:12:23 +01:00
|
|
|
}
|
|
|
|
|
2020-07-05 14:32:07 -06:00
|
|
|
bool Scheduler::context_switch(Thread* thread)
|
2018-11-07 22:15:02 +01:00
|
|
|
{
|
2020-07-05 14:32:07 -06:00
|
|
|
thread->did_schedule();
|
2018-11-07 22:15:02 +01:00
|
|
|
|
2020-07-05 14:32:07 -06:00
|
|
|
auto from_thread = Thread::current();
|
|
|
|
if (from_thread == thread)
|
2018-11-07 22:15:02 +01:00
|
|
|
return false;
|
|
|
|
|
2020-07-05 14:32:07 -06:00
|
|
|
if (from_thread) {
|
2018-11-07 22:15:02 +01:00
|
|
|
// If the last process hasn't blocked (still marked as running),
|
|
|
|
// mark it as runnable for the next round.
|
2020-07-05 14:32:07 -06:00
|
|
|
if (from_thread->state() == Thread::Running)
|
|
|
|
from_thread->set_state(Thread::Runnable);
|
2018-11-07 22:24:20 +01:00
|
|
|
|
|
|
|
#ifdef LOG_EVERY_CONTEXT_SWITCH
|
2020-12-25 17:05:05 +01:00
|
|
|
dbgln("Scheduler[{}]: {} -> {} [prio={}] {:04x}:{:08x}", Processor::current().id(), from_thread->tid().value(), thread->tid().value(), thread->priority(), thread->tss().cs, thread->tss().eip);
|
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
|
|
|
}
|
2020-07-05 14:32:07 -06:00
|
|
|
thread->set_state(Thread::Running);
|
|
|
|
|
|
|
|
// 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.
|
|
|
|
thread->set_active(true);
|
|
|
|
|
|
|
|
proc.switch_context(from_thread, thread);
|
|
|
|
|
|
|
|
// NOTE: from_thread at this point reflects the thread we were
|
|
|
|
// switched from, and thread reflects Thread::current()
|
2020-12-07 21:29:41 -07:00
|
|
|
enter_current(*from_thread, false);
|
2020-07-05 14:32:07 -06:00
|
|
|
ASSERT(thread == Thread::current());
|
2019-09-07 15:50:44 +02:00
|
|
|
|
2020-12-23 14:18:13 +01:00
|
|
|
#if ARCH(I386)
|
|
|
|
auto iopl = get_iopl_from_eflags(Thread::current()->get_register_dump_from_stack().eflags);
|
|
|
|
if (thread->process().is_user_process() && iopl != 0) {
|
|
|
|
dbgln("PANIC: Switched to thread {} with non-zero IOPL={}", Thread::current()->tid().value(), iopl);
|
|
|
|
Processor::halt();
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2018-11-07 22:15:02 +01:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2020-12-07 21:29:41 -07:00
|
|
|
void Scheduler::enter_current(Thread& prev_thread, bool is_first)
|
2020-07-05 14:32:07 -06:00
|
|
|
{
|
2020-12-07 21:29:41 -07:00
|
|
|
ASSERT(g_scheduler_lock.own_lock());
|
2020-07-05 14:32:07 -06:00
|
|
|
prev_thread.set_active(false);
|
|
|
|
if (prev_thread.state() == Thread::Dying) {
|
|
|
|
// 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-12-07 21:29:41 -07:00
|
|
|
} else if (!is_first) {
|
|
|
|
// Check if we have any signals we should deliver (even if we don't
|
|
|
|
// end up switching to another thread).
|
|
|
|
auto current_thread = Thread::current();
|
|
|
|
if (!current_thread->is_in_block()) {
|
|
|
|
ScopedSpinLock lock(current_thread->get_lock());
|
|
|
|
if (current_thread->state() == Thread::Running && current_thread->pending_signals_for_state()) {
|
|
|
|
current_thread->dispatch_one_pending_signal();
|
|
|
|
}
|
|
|
|
}
|
2020-07-05 14:32:07 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
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);
|
|
|
|
auto& scheduler_data = Processor::current().get_scheduler_data();
|
|
|
|
ASSERT(scheduler_data.m_in_scheduler);
|
|
|
|
scheduler_data.m_in_scheduler = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
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
|
|
|
|
ASSERT(g_scheduler_lock.own_lock());
|
|
|
|
auto& scheduler_data = Processor::current().get_scheduler_data();
|
|
|
|
ASSERT(!scheduler_data.m_in_scheduler);
|
|
|
|
scheduler_data.m_in_scheduler = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
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
|
|
|
|
ASSERT(!g_scheduler_lock.own_lock());
|
|
|
|
g_scheduler_lock.lock();
|
|
|
|
auto& scheduler_data = Processor::current().get_scheduler_data();
|
|
|
|
ASSERT(!scheduler_data.m_in_scheduler);
|
|
|
|
scheduler_data.m_in_scheduler = true;
|
|
|
|
}
|
|
|
|
|
2019-02-04 10:28:12 +01:00
|
|
|
Process* Scheduler::colonel()
|
|
|
|
{
|
2020-07-06 07:27:22 -06:00
|
|
|
ASSERT(s_colonel_process);
|
2019-02-04 10:28:12 +01:00
|
|
|
return s_colonel_process;
|
|
|
|
}
|
|
|
|
|
2020-07-06 07:27:22 -06:00
|
|
|
void Scheduler::initialize()
|
2018-11-07 22:15:02 +01:00
|
|
|
{
|
2020-06-27 13:42:28 -06:00
|
|
|
ASSERT(&Processor::current() != nullptr); // sanity check
|
|
|
|
|
2020-09-27 08:53:35 -06:00
|
|
|
RefPtr<Thread> idle_thread;
|
2020-07-06 07:27:22 -06:00
|
|
|
g_scheduler_data = new SchedulerData;
|
|
|
|
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);
|
2020-11-16 20:51:34 -07:00
|
|
|
s_colonel_process = &Process::create_kernel_process(idle_thread, "colonel", idle_loop, nullptr, 1).leak_ref();
|
2020-07-06 07:27:22 -06:00
|
|
|
ASSERT(s_colonel_process);
|
|
|
|
ASSERT(idle_thread);
|
|
|
|
idle_thread->set_priority(THREAD_PRIORITY_MIN);
|
2020-09-11 21:11:07 -06:00
|
|
|
idle_thread->set_name(StringView("idle thread #0"));
|
2020-06-28 15:34:31 -06:00
|
|
|
|
2020-07-06 07:27:22 -06:00
|
|
|
set_idle_thread(idle_thread);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Scheduler::set_idle_thread(Thread* idle_thread)
|
|
|
|
{
|
2020-06-28 15:34:31 -06:00
|
|
|
Processor::current().set_idle_thread(*idle_thread);
|
2020-06-28 22:36:12 -06:00
|
|
|
Processor::current().set_current_thread(*idle_thread);
|
2020-07-06 07:27:22 -06:00
|
|
|
}
|
2020-06-28 16:05:52 -06:00
|
|
|
|
2020-07-06 07:27:22 -06:00
|
|
|
Thread* Scheduler::create_ap_idle_thread(u32 cpu)
|
|
|
|
{
|
|
|
|
ASSERT(cpu != 0);
|
|
|
|
// This function is called on the bsp, but creates an idle thread for another AP
|
|
|
|
ASSERT(Processor::current().id() == 0);
|
|
|
|
|
|
|
|
ASSERT(s_colonel_process);
|
2020-11-16 20:51:34 -07:00
|
|
|
Thread* idle_thread = s_colonel_process->create_kernel_thread(idle_loop, nullptr, THREAD_PRIORITY_MIN, String::format("idle thread #%u", cpu), 1 << cpu, false);
|
2020-07-06 07:27:22 -06:00
|
|
|
ASSERT(idle_thread);
|
|
|
|
return idle_thread;
|
2018-11-07 22:15:02 +01:00
|
|
|
}
|
2018-11-08 00:24:59 +01:00
|
|
|
|
2020-03-09 16:24:29 +02:00
|
|
|
void Scheduler::timer_tick(const RegisterState& regs)
|
2018-11-08 00:24:59 +01:00
|
|
|
{
|
2020-06-27 13:42:28 -06:00
|
|
|
ASSERT_INTERRUPTS_DISABLED();
|
|
|
|
ASSERT(Processor::current().in_irq());
|
2020-07-30 23:38:15 +02:00
|
|
|
|
2020-06-28 15:34:31 -06:00
|
|
|
auto current_thread = Processor::current().current_thread();
|
|
|
|
if (!current_thread)
|
2018-11-08 00:24:59 +01:00
|
|
|
return;
|
|
|
|
|
2020-10-25 09:13:47 -06:00
|
|
|
bool is_bsp = Processor::current().id() == 0;
|
|
|
|
if (!is_bsp)
|
|
|
|
return; // TODO: This prevents scheduling on other CPUs!
|
2020-06-28 15:34:31 -06:00
|
|
|
if (current_thread->process().is_profiling()) {
|
2020-01-12 02:02:29 +01:00
|
|
|
SmapDisabler disabler;
|
2020-06-28 15:34:31 -06:00
|
|
|
auto backtrace = current_thread->raw_backtrace(regs.ebp, regs.eip);
|
2019-12-11 20:36:56 +01:00
|
|
|
auto& sample = Profiling::next_sample_slot();
|
2020-08-09 01:08:24 +02:00
|
|
|
sample.pid = current_thread->process().pid();
|
|
|
|
sample.tid = current_thread->tid();
|
2020-11-15 11:58:19 -07:00
|
|
|
sample.timestamp = TimeManagement::the().uptime_ms();
|
2020-03-01 12:35:09 +01:00
|
|
|
for (size_t i = 0; i < min(backtrace.size(), Profiling::max_stack_frame_count); ++i) {
|
2019-12-11 20:36:56 +01:00
|
|
|
sample.frames[i] = backtrace[i];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-12-03 22:12:50 -07:00
|
|
|
if (current_thread->tick((regs.cs & 3) == 0))
|
2018-11-08 00:24:59 +01:00
|
|
|
return;
|
|
|
|
|
2020-06-27 13:42:28 -06:00
|
|
|
ASSERT_INTERRUPTS_DISABLED();
|
|
|
|
ASSERT(Processor::current().in_irq());
|
|
|
|
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
|
|
|
{
|
2020-06-27 13:42:28 -06:00
|
|
|
ASSERT_INTERRUPTS_DISABLED();
|
2020-08-01 14:37:40 -06:00
|
|
|
auto& proc = Processor::current();
|
|
|
|
ASSERT(!proc.in_irq());
|
|
|
|
|
|
|
|
// Since this function is called when leaving critical sections (such
|
|
|
|
// as a SpinLock), we need to check if we're not already doing this
|
|
|
|
// to prevent recursion
|
|
|
|
if (!proc.get_scheduler_data().m_in_scheduler)
|
|
|
|
pick_next();
|
2019-09-14 19:44:22 +02:00
|
|
|
}
|
|
|
|
|
2020-11-30 19:04:36 -07:00
|
|
|
void Scheduler::yield_from_critical()
|
|
|
|
{
|
|
|
|
auto& proc = Processor::current();
|
|
|
|
ASSERT(proc.in_critical());
|
|
|
|
ASSERT(!proc.in_irq());
|
|
|
|
|
|
|
|
yield(); // Flag a context switch
|
|
|
|
|
|
|
|
u32 prev_flags;
|
|
|
|
u32 prev_crit = Processor::current().clear_critical(prev_flags, false);
|
|
|
|
|
|
|
|
// Note, we may now be on a different CPU!
|
|
|
|
Processor::current().restore_critical(prev_crit, prev_flags);
|
|
|
|
}
|
|
|
|
|
2020-07-05 14:32:07 -06:00
|
|
|
void Scheduler::notify_finalizer()
|
|
|
|
{
|
|
|
|
if (g_finalizer_has_work.exchange(true, AK::MemoryOrder::memory_order_acq_rel) == false)
|
|
|
|
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-06-28 22:36:12 -06:00
|
|
|
dbg() << "Scheduler[" << Processor::current().id() << "]: idle loop running";
|
2020-06-27 13:42:28 -06:00
|
|
|
ASSERT(are_interrupts_enabled());
|
2020-07-06 07:27:22 -06:00
|
|
|
|
2019-09-14 19:44:22 +02:00
|
|
|
for (;;) {
|
|
|
|
asm("hlt");
|
2020-07-30 21:46:06 +02:00
|
|
|
|
|
|
|
if (Processor::current().id() == 0)
|
|
|
|
yield();
|
2019-09-14 19:44:22 +02:00
|
|
|
}
|
|
|
|
}
|
2020-02-16 01:27:42 +01:00
|
|
|
|
|
|
|
}
|