2019-02-06 15:05:47 +01:00
# include <AK/TemporaryChange.h>
2019-07-09 14:23:12 +02:00
# include <Kernel/Arch/i386/PIT.h>
2019-05-15 21:40:41 +02:00
# include <Kernel/Devices/PCSpeaker.h>
2019-06-07 11:43:58 +02:00
# include <Kernel/FileSystem/FileDescription.h>
2019-06-07 19:29:34 +02:00
# include <Kernel/Process.h>
# include <Kernel/RTC.h>
# include <Kernel/Scheduler.h>
2018-11-07 22:15:02 +01:00
2018-11-07 22:24:20 +01:00
//#define LOG_EVERY_CONTEXT_SWITCH
2018-11-07 22:15:02 +01:00
//#define SCHEDULER_DEBUG
2019-07-17 14:17:49 +02:00
//#define SCHEDULER_RUNNABLE_DEBUG
2018-11-07 22:15:02 +01:00
2019-07-03 21:17:35 +02:00
static u32 time_slice_for ( Process : : Priority priority )
2019-02-07 12:21:17 +01:00
{
// One time slice unit == 1ms
switch ( priority ) {
case Process : : HighPriority :
return 50 ;
2019-02-12 12:11:22 +01:00
case Process : : NormalPriority :
return 15 ;
case Process : : LowPriority :
return 5 ;
2019-04-20 15:58:45 +02:00
case Process : : IdlePriority :
return 1 ;
2019-02-07 12:21:17 +01:00
}
2019-02-12 12:11:22 +01:00
ASSERT_NOT_REACHED ( ) ;
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 * current ;
Thread * g_last_fpu_thread ;
Thread * g_finalizer ;
2018-11-07 22:15:02 +01:00
static Process * s_colonel_process ;
2019-07-03 21:17:35 +02:00
u64 g_uptime ;
static u64 s_beep_timeout ;
2018-11-07 22:15:02 +01:00
2018-11-07 23:13:38 +01:00
struct TaskRedirectionData {
2019-07-03 21:17:35 +02:00
u16 selector ;
2018-11-07 23:13:38 +01:00
TSS32 tss ;
} ;
static TaskRedirectionData s_redirection ;
2019-02-06 15:05:47 +01:00
static bool s_active ;
bool Scheduler : : is_active ( )
{
return s_active ;
}
2018-11-07 23:13:38 +01:00
2019-05-15 21:40:41 +02:00
void Scheduler : : beep ( )
{
PCSpeaker : : tone_on ( 440 ) ;
s_beep_timeout = g_uptime + 100 ;
}
2019-07-18 16:22:26 +02:00
Thread : : ThreadBlockerFileDescription : : ThreadBlockerFileDescription ( const RefPtr < FileDescription > & description )
: m_blocked_description ( description )
{ }
RefPtr < FileDescription > Thread : : ThreadBlockerFileDescription : : blocked_description ( ) const
{
return m_blocked_description ;
}
Thread : : ThreadBlockerAccept : : ThreadBlockerAccept ( const RefPtr < FileDescription > & description )
: ThreadBlockerFileDescription ( description )
{
}
2019-07-18 17:39:49 +02:00
bool Thread : : ThreadBlockerAccept : : should_unblock ( Thread & , time_t , long )
2019-07-18 16:22:26 +02:00
{
auto & description = * blocked_description ( ) ;
auto & socket = * description . socket ( ) ;
return socket . can_accept ( ) ;
}
Thread : : ThreadBlockerReceive : : ThreadBlockerReceive ( const RefPtr < FileDescription > & description )
: ThreadBlockerFileDescription ( description )
{
}
2019-07-18 17:39:49 +02:00
bool Thread : : ThreadBlockerReceive : : should_unblock ( Thread & , time_t now_sec , long now_usec )
2019-07-18 16:22:26 +02:00
{
auto & description = * blocked_description ( ) ;
auto & socket = * description . socket ( ) ;
// FIXME: Block until the amount of data wanted is available.
bool timed_out = now_sec > socket . receive_deadline ( ) . tv_sec | | ( now_sec = = socket . receive_deadline ( ) . tv_sec & & now_usec > = socket . receive_deadline ( ) . tv_usec ) ;
if ( timed_out | | description . can_read ( ) )
return true ;
return false ;
}
Thread : : ThreadBlockerConnect : : ThreadBlockerConnect ( const RefPtr < FileDescription > & description )
: ThreadBlockerFileDescription ( description )
{
}
2019-07-18 17:39:49 +02:00
bool Thread : : ThreadBlockerConnect : : should_unblock ( Thread & , time_t , long )
2019-07-18 16:22:26 +02:00
{
auto & description = * blocked_description ( ) ;
auto & socket = * description . socket ( ) ;
return socket . is_connected ( ) ;
}
Thread : : ThreadBlockerWrite : : ThreadBlockerWrite ( const RefPtr < FileDescription > & description )
: ThreadBlockerFileDescription ( description )
{
}
2019-07-18 17:39:49 +02:00
bool Thread : : ThreadBlockerWrite : : should_unblock ( Thread & , time_t , long )
2019-07-18 16:22:26 +02:00
{
return blocked_description ( ) - > can_write ( ) ;
}
Thread : : ThreadBlockerRead : : ThreadBlockerRead ( const RefPtr < FileDescription > & description )
: ThreadBlockerFileDescription ( description )
{
}
2019-07-18 17:39:49 +02:00
bool Thread : : ThreadBlockerRead : : should_unblock ( Thread & , time_t , long )
2019-07-18 16:22:26 +02:00
{
// FIXME: Block until the amount of data wanted is available.
return blocked_description ( ) - > can_read ( ) ;
}
Thread : : ThreadBlockerCondition : : ThreadBlockerCondition ( Function < bool ( ) > & condition )
: m_block_until_condition ( move ( condition ) )
{
ASSERT ( m_block_until_condition ) ;
}
2019-07-18 17:39:49 +02:00
bool Thread : : ThreadBlockerCondition : : should_unblock ( Thread & , time_t , long )
2019-07-18 16:22:26 +02:00
{
return m_block_until_condition ( ) ;
}
2019-07-18 17:26:11 +02:00
Thread : : ThreadBlockerSleep : : ThreadBlockerSleep ( u64 wakeup_time )
: m_wakeup_time ( wakeup_time )
{
}
2019-07-18 17:39:49 +02:00
bool Thread : : ThreadBlockerSleep : : should_unblock ( Thread & , time_t , long )
2019-07-18 17:26:11 +02:00
{
return m_wakeup_time < = g_uptime ;
}
2019-07-18 17:39:49 +02:00
Thread : : ThreadBlockerSelect : : ThreadBlockerSelect ( const timeval & tv , bool select_has_timeout , const Vector < int > & read_fds , const Vector < int > & write_fds , const Vector < int > & except_fds )
: m_select_timeout ( tv )
, m_select_has_timeout ( select_has_timeout )
, m_select_read_fds ( read_fds )
, m_select_write_fds ( write_fds )
, m_select_exceptional_fds ( except_fds )
{
}
bool Thread : : ThreadBlockerSelect : : should_unblock ( Thread & thread , time_t now_sec , long now_usec )
{
if ( m_select_has_timeout ) {
if ( now_sec > m_select_timeout . tv_sec | | ( now_sec = = m_select_timeout . tv_sec & & now_usec > = m_select_timeout . tv_usec ) )
return true ;
}
auto & process = thread . process ( ) ;
for ( int fd : m_select_read_fds ) {
if ( process . m_fds [ fd ] . description - > can_read ( ) )
return true ;
}
for ( int fd : m_select_write_fds ) {
if ( process . m_fds [ fd ] . description - > can_write ( ) )
return true ;
}
return false ;
}
2019-07-18 14:10:28 +02:00
// Called by the scheduler on threads that are blocked for some reason.
// Make a decision as to whether to unblock them or not.
void Thread : : consider_unblock ( time_t now_sec , long now_usec )
{
auto & process = this - > process ( ) ;
switch ( state ( ) ) {
case Thread : : __Begin_Blocked_States__ :
case Thread : : __End_Blocked_States__ :
ASSERT_NOT_REACHED ( ) ;
[[fallthrough]] ;
case Thread : : Invalid :
case Thread : : Runnable :
case Thread : : Running :
case Thread : : Dead :
case Thread : : Stopped :
case Thread : : BlockedLurking :
case Thread : : BlockedSignal :
/* don't know, don't care */
return ;
case Thread : : BlockedWait :
process . for_each_child ( [ & ] ( Process & child ) {
if ( waitee_pid ( ) ! = - 1 & & waitee_pid ( ) ! = child . pid ( ) )
return IterationDecision : : Continue ;
bool child_exited = child . is_dead ( ) ;
bool child_stopped = child . main_thread ( ) . state ( ) = = Thread : : State : : Stopped ;
bool wait_finished = ( ( m_wait_options & WEXITED ) & & child_exited )
| | ( ( m_wait_options & WSTOPPED ) & & child_stopped ) ;
if ( ! wait_finished )
return IterationDecision : : Continue ;
m_waitee_pid = child . pid ( ) ;
unblock ( ) ;
return IterationDecision : : Break ;
} ) ;
return ;
case Thread : : BlockedCondition :
2019-07-18 16:22:26 +02:00
ASSERT ( m_blocker ) ;
2019-07-18 17:39:49 +02:00
if ( m_blocker - > should_unblock ( * this , now_sec , now_usec ) ) {
2019-07-18 14:10:28 +02:00
unblock ( ) ;
2019-07-18 16:22:26 +02:00
m_blocker = nullptr ;
2019-07-18 14:10:28 +02:00
}
return ;
case Thread : : Skip1SchedulerPass :
set_state ( Thread : : Skip0SchedulerPasses ) ;
return ;
case Thread : : Skip0SchedulerPasses :
set_state ( Thread : : Runnable ) ;
return ;
case Thread : : Dying :
ASSERT ( g_finalizer ) ;
if ( g_finalizer - > state ( ) = = Thread : : BlockedLurking )
g_finalizer - > unblock ( ) ;
return ;
}
}
2018-11-07 22:15:02 +01:00
bool Scheduler : : pick_next ( )
{
ASSERT_INTERRUPTS_DISABLED ( ) ;
2019-02-06 15:05:47 +01:00
ASSERT ( ! s_active ) ;
TemporaryChange < bool > change ( s_active , true ) ;
ASSERT ( s_active ) ;
2018-11-07 22:15:02 +01:00
if ( ! current ) {
// XXX: The first ever context_switch() goes to the idle process.
// This to setup a reliable place we can return to.
2019-03-23 22:03:17 +01:00
return context_switch ( s_colonel_process - > main_thread ( ) ) ;
2018-11-07 22:15:02 +01:00
}
2019-03-25 02:06:57 +01:00
struct timeval now ;
kgettimeofday ( now ) ;
2019-07-18 14:10:28 +02:00
2019-03-25 02:06:57 +01:00
auto now_sec = now . tv_sec ;
auto now_usec = now . tv_usec ;
2019-03-13 13:13:23 +01:00
2019-03-23 22:03:17 +01:00
// Check and unblock threads whose wait conditions have been met.
2019-06-07 11:43:58 +02:00
Thread : : for_each_nonrunnable ( [ & ] ( Thread & thread ) {
2019-07-18 14:10:28 +02:00
thread . consider_unblock ( now_sec , now_usec ) ;
2019-03-23 22:03:17 +01:00
return IterationDecision : : Continue ;
} ) ;
2019-06-07 11:43:58 +02:00
Process : : for_each ( [ & ] ( Process & process ) {
2019-03-23 22:03:17 +01:00
if ( process . is_dead ( ) ) {
if ( current ! = & process . main_thread ( ) & & ( ! process . ppid ( ) | | ! Process : : from_pid ( process . ppid ( ) ) ) ) {
2018-11-28 22:01:24 +01:00
auto name = process . name ( ) ;
auto pid = process . pid ( ) ;
auto exit_status = Process : : reap ( process ) ;
2019-01-01 03:56:39 +01:00
dbgprintf ( " reaped unparented process %s(%u), exit status: %u \n " , name . characters ( ) , pid , exit_status ) ;
2018-11-28 22:01:24 +01:00
}
2019-06-07 11:30:07 +02:00
return IterationDecision : : Continue ;
}
if ( process . m_alarm_deadline & & g_uptime > process . m_alarm_deadline ) {
process . m_alarm_deadline = 0 ;
process . send_signal ( SIGALRM , nullptr ) ;
2018-11-07 23:59:49 +01:00
}
2019-06-07 11:30:07 +02:00
return IterationDecision : : Continue ;
2018-11-07 22:15:02 +01:00
} ) ;
// Dispatch any pending signals.
// FIXME: Do we really need this to be a separate pass over the process list?
2019-06-07 11:43:58 +02:00
Thread : : for_each_living ( [ ] ( Thread & thread ) {
2019-03-23 22:03:17 +01:00
if ( ! thread . has_unmasked_pending_signals ( ) )
2018-11-07 22:15:02 +01:00
return true ;
2019-03-05 12:50:55 +01:00
// FIXME: It would be nice if the Scheduler didn't have to worry about who is "current"
// For now, avoid dispatching signals to "current" and do it in a scheduling pass
// while some other process is interrupted. Otherwise a mess will be made.
2019-03-23 22:03:17 +01:00
if ( & thread = = current )
2019-03-05 12:50:55 +01:00
return true ;
2018-11-07 22:15:02 +01:00
// We know how to interrupt blocked processes, but if they are just executing
// at some random point in the kernel, let them continue. They'll be in userspace
// sooner or later and we can deliver the signal then.
// FIXME: Maybe we could check when returning from a syscall if there's a pending
// signal and dispatch it then and there? Would that be doable without the
// syscall effectively being "interrupted" despite having completed?
2019-03-23 22:03:17 +01:00
if ( thread . in_kernel ( ) & & ! thread . is_blocked ( ) & & ! thread . is_stopped ( ) )
2018-11-07 22:15:02 +01:00
return true ;
2018-11-28 23:30:06 +01:00
// NOTE: dispatch_one_pending_signal() may unblock the process.
2019-03-23 22:03:17 +01:00
bool was_blocked = thread . is_blocked ( ) ;
if ( thread . dispatch_one_pending_signal ( ) = = ShouldUnblockThread : : No )
2018-11-16 21:14:25 +01:00
return true ;
2018-11-28 23:30:06 +01:00
if ( was_blocked ) {
2019-03-23 22:03:17 +01:00
dbgprintf ( " Unblock %s(%u) due to signal \n " , thread . process ( ) . name ( ) . characters ( ) , thread . pid ( ) ) ;
thread . m_was_interrupted_while_blocked = true ;
thread . unblock ( ) ;
2018-11-07 22:15:02 +01:00
}
return true ;
} ) ;
2019-07-17 14:17:49 +02:00
# ifdef SCHEDULER_RUNNABLE_DEBUG
2019-05-20 03:44:45 +02:00
dbgprintf ( " Non-runnables: \n " ) ;
for ( auto * thread = g_nonrunnable_threads - > head ( ) ; thread ; thread = thread - > next ( ) ) {
auto * process = & thread - > process ( ) ;
2019-06-22 15:52:45 +02:00
dbgprintf ( " [K%x] %-12s %s(%u:%u) @ %w:%x \n " , process , to_string ( thread - > state ( ) ) , process - > name ( ) . characters ( ) , process - > pid ( ) , thread - > tid ( ) , thread - > tss ( ) . cs , thread - > tss ( ) . eip ) ;
2019-05-20 03:44:45 +02:00
}
dbgprintf ( " Runnables: \n " ) ;
2019-05-18 18:31:36 +02:00
for ( auto * thread = g_runnable_threads - > head ( ) ; thread ; thread = thread - > next ( ) ) {
2019-03-23 22:03:17 +01:00
auto * process = & thread - > process ( ) ;
2019-06-22 15:52:45 +02:00
dbgprintf ( " [K%x] %-12s %s(%u:%u) @ %w:%x \n " , process , to_string ( thread - > state ( ) ) , process - > name ( ) . characters ( ) , process - > pid ( ) , thread - > tid ( ) , thread - > tss ( ) . cs , thread - > tss ( ) . eip ) ;
2018-11-07 22:15:02 +01:00
}
# endif
2019-05-18 20:24:55 +02:00
if ( g_runnable_threads - > is_empty ( ) )
return context_switch ( s_colonel_process - > main_thread ( ) ) ;
2019-05-18 18:31:36 +02:00
auto * previous_head = g_runnable_threads - > head ( ) ;
2018-11-07 22:15:02 +01:00
for ( ; ; ) {
// Move head to tail.
2019-05-18 18:31:36 +02:00
g_runnable_threads - > append ( g_runnable_threads - > remove_head ( ) ) ;
auto * thread = g_runnable_threads - > head ( ) ;
2018-11-07 22:15:02 +01:00
2019-03-23 22:03:17 +01:00
if ( ! thread - > process ( ) . is_being_inspected ( ) & & ( thread - > state ( ) = = Thread : : Runnable | | thread - > state ( ) = = Thread : : Running ) ) {
2018-11-07 22:15:02 +01:00
# ifdef SCHEDULER_DEBUG
2019-07-17 14:17:49 +02:00
dbgprintf ( " switch to %s(%u:%u) @ %w:%x \n " , thread - > process ( ) . name ( ) . characters ( ) , thread - > process ( ) . pid ( ) , thread - > tid ( ) , thread - > tss ( ) . cs , thread - > tss ( ) . eip ) ;
2018-11-07 22:15:02 +01:00
# endif
2019-03-23 22:03:17 +01:00
return context_switch ( * thread ) ;
2018-11-07 22:15:02 +01:00
}
2019-03-23 22:03:17 +01:00
if ( thread = = previous_head ) {
2018-11-07 23:13:38 +01:00
// Back at process_head, nothing wants to run. Send in the colonel!
2019-03-23 22:03:17 +01:00
return context_switch ( s_colonel_process - > main_thread ( ) ) ;
2018-11-07 22:15:02 +01:00
}
}
}
2019-03-23 22:03:17 +01:00
bool Scheduler : : donate_to ( Thread * beneficiary , const char * reason )
2019-02-07 11:12:23 +01:00
{
2019-04-17 12:41:51 +02:00
InterruptDisabler disabler ;
if ( ! Thread : : is_thread ( beneficiary ) )
return false ;
2019-02-07 11:12:23 +01:00
( void ) reason ;
unsigned ticks_left = current - > ticks_left ( ) ;
2019-04-17 12:41:51 +02:00
if ( ! beneficiary | | beneficiary - > state ( ) ! = Thread : : Runnable | | ticks_left < = 1 )
2019-02-07 11:12:23 +01:00
return yield ( ) ;
2019-03-23 22:03:17 +01:00
unsigned ticks_to_donate = min ( ticks_left - 1 , time_slice_for ( beneficiary - > process ( ) . priority ( ) ) ) ;
2019-02-07 11:12:23 +01:00
# ifdef SCHEDULER_DEBUG
2019-03-23 22:03:17 +01:00
dbgprintf ( " %s(%u:%u) donating %u ticks to %s(%u:%u), reason=%s \n " , current - > process ( ) . name ( ) . characters ( ) , current - > pid ( ) , current - > tid ( ) , ticks_to_donate , beneficiary - > process ( ) . name ( ) . characters ( ) , beneficiary - > pid ( ) , beneficiary - > tid ( ) , reason ) ;
2019-02-07 11:12:23 +01:00
# endif
context_switch ( * beneficiary ) ;
beneficiary - > set_ticks_left ( ticks_to_donate ) ;
switch_now ( ) ;
2019-04-17 12:41:51 +02:00
return false ;
2019-02-07 11:12:23 +01:00
}
2018-11-07 22:15:02 +01:00
bool Scheduler : : yield ( )
{
2019-01-23 05:05:45 +01:00
InterruptDisabler disabler ;
2019-02-06 15:05:47 +01:00
ASSERT ( current ) ;
2019-06-07 11:43:58 +02:00
// dbgprintf("%s(%u:%u) yield()\n", current->process().name().characters(), current->pid(), current->tid());
2018-11-07 22:15:02 +01:00
2019-02-06 15:05:47 +01:00
if ( ! pick_next ( ) )
2019-04-17 12:41:51 +02:00
return false ;
2018-11-07 22:15:02 +01:00
2019-06-07 11:43:58 +02:00
// dbgprintf("yield() jumping to new process: sel=%x, %s(%u:%u)\n", current->far_ptr().selector, current->process().name().characters(), current->pid(), current->tid());
2018-11-07 22:15:02 +01:00
switch_now ( ) ;
2019-04-17 12:41:51 +02:00
return true ;
2018-11-07 22:15:02 +01:00
}
void Scheduler : : pick_next_and_switch_now ( )
{
bool someone_wants_to_run = pick_next ( ) ;
ASSERT ( someone_wants_to_run ) ;
switch_now ( ) ;
}
void Scheduler : : switch_now ( )
{
2018-12-03 00:39:25 +01:00
Descriptor & descriptor = get_gdt_entry ( current - > selector ( ) ) ;
2018-11-07 22:15:02 +01:00
descriptor . type = 9 ;
2018-12-03 00:39:25 +01:00
flush_gdt ( ) ;
2018-11-07 22:15:02 +01:00
asm ( " sti \n "
2019-06-07 11:43:58 +02:00
" ljmp *(%%eax) \n " : : " a " ( & current - > far_ptr ( ) ) ) ;
2018-11-07 22:15:02 +01:00
}
2019-03-23 22:03:17 +01:00
bool Scheduler : : context_switch ( Thread & thread )
2018-11-07 22:15:02 +01:00
{
2019-03-23 22:03:17 +01:00
thread . set_ticks_left ( time_slice_for ( thread . process ( ) . priority ( ) ) ) ;
thread . did_schedule ( ) ;
2018-11-07 22:15:02 +01:00
2019-03-23 22:03:17 +01:00
if ( current = = & thread )
2018-11-07 22:15:02 +01:00
return false ;
if ( current ) {
// If the last process hasn't blocked (still marked as running),
// mark it as runnable for the next round.
2019-03-23 22:03:17 +01:00
if ( current - > state ( ) = = Thread : : Running )
current - > set_state ( Thread : : Runnable ) ;
2018-11-07 22:24:20 +01:00
# ifdef LOG_EVERY_CONTEXT_SWITCH
2019-03-23 22:03:17 +01:00
dbgprintf ( " Scheduler: %s(%u:%u) -> %s(%u:%u) %w:%x \n " ,
2019-06-07 11:43:58 +02:00
current - > process ( ) . name ( ) . characters ( ) , current - > process ( ) . pid ( ) , current - > tid ( ) ,
thread . process ( ) . name ( ) . characters ( ) , thread . process ( ) . pid ( ) , thread . tid ( ) ,
thread . tss ( ) . cs , thread . tss ( ) . eip ) ;
2018-11-07 22:24:20 +01:00
# endif
2018-11-07 22:15:02 +01:00
}
2019-03-23 22:03:17 +01:00
current = & thread ;
thread . set_state ( Thread : : Running ) ;
2018-11-07 22:15:02 +01:00
2019-03-23 22:03:17 +01:00
if ( ! thread . selector ( ) ) {
thread . set_selector ( gdt_alloc_entry ( ) ) ;
auto & descriptor = get_gdt_entry ( thread . selector ( ) ) ;
descriptor . set_base ( & thread . tss ( ) ) ;
2019-01-31 17:31:23 +01:00
descriptor . set_limit ( 0xffff ) ;
2018-11-07 22:15:02 +01:00
descriptor . dpl = 0 ;
descriptor . segment_present = 1 ;
descriptor . granularity = 1 ;
descriptor . zero = 0 ;
descriptor . operation_size = 1 ;
descriptor . descriptor_type = 0 ;
}
2019-03-23 22:03:17 +01:00
auto & descriptor = get_gdt_entry ( thread . selector ( ) ) ;
2018-11-07 22:15:02 +01:00
descriptor . type = 11 ; // Busy TSS
2018-12-03 00:39:25 +01:00
flush_gdt ( ) ;
2018-11-07 22:15:02 +01:00
return true ;
}
2018-11-07 23:13:38 +01:00
static void initialize_redirection ( )
2018-11-07 22:15:02 +01:00
{
2018-12-03 00:39:25 +01:00
auto & descriptor = get_gdt_entry ( s_redirection . selector ) ;
2019-01-31 17:31:23 +01:00
descriptor . set_base ( & s_redirection . tss ) ;
descriptor . set_limit ( 0xffff ) ;
2018-11-07 23:13:38 +01:00
descriptor . dpl = 0 ;
descriptor . segment_present = 1 ;
descriptor . granularity = 1 ;
descriptor . zero = 0 ;
descriptor . operation_size = 1 ;
descriptor . descriptor_type = 0 ;
descriptor . type = 9 ;
2018-12-03 00:39:25 +01:00
flush_gdt ( ) ;
2018-11-07 22:15:02 +01:00
}
void Scheduler : : prepare_for_iret_to_new_process ( )
{
2018-12-03 00:39:25 +01:00
auto & descriptor = get_gdt_entry ( s_redirection . selector ) ;
2018-11-07 23:13:38 +01:00
descriptor . type = 9 ;
s_redirection . tss . backlink = current - > selector ( ) ;
load_task_register ( s_redirection . selector ) ;
2018-11-07 22:15:02 +01:00
}
2019-03-23 22:03:17 +01:00
void Scheduler : : prepare_to_modify_tss ( Thread & thread )
2018-11-07 22:15:02 +01:00
{
2018-11-07 23:59:49 +01:00
// This ensures that a currently running process modifying its own TSS
// in order to yield() and end up somewhere else doesn't just end up
// right after the yield().
2019-03-23 22:03:17 +01:00
if ( current = = & thread )
2018-11-07 23:59:49 +01:00
load_task_register ( s_redirection . selector ) ;
2018-11-07 22:15:02 +01:00
}
2019-02-04 10:28:12 +01:00
Process * Scheduler : : colonel ( )
{
return s_colonel_process ;
}
2018-11-07 22:15:02 +01:00
void Scheduler : : initialize ( )
{
2018-11-07 23:13:38 +01:00
s_redirection . selector = gdt_alloc_entry ( ) ;
initialize_redirection ( ) ;
2018-12-24 23:10:48 +01:00
s_colonel_process = Process : : create_kernel_process ( " colonel " , nullptr ) ;
2019-03-23 22:22:01 +01:00
// Make sure the colonel uses a smallish time slice.
2019-04-20 15:58:45 +02:00
s_colonel_process - > set_priority ( Process : : IdlePriority ) ;
2018-11-07 23:13:38 +01:00
load_task_register ( s_redirection . selector ) ;
2018-11-07 22:15:02 +01:00
}
2018-11-08 00:24:59 +01:00
void Scheduler : : timer_tick ( RegisterDump & regs )
{
if ( ! current )
return ;
2019-04-14 01:29:14 +02:00
+ + g_uptime ;
2018-11-08 00:24:59 +01:00
2019-05-15 21:40:41 +02:00
if ( s_beep_timeout & & g_uptime > s_beep_timeout ) {
PCSpeaker : : tone_off ( ) ;
s_beep_timeout = 0 ;
}
2018-11-08 00:24:59 +01:00
if ( current - > tick ( ) )
return ;
current - > tss ( ) . gs = regs . gs ;
current - > tss ( ) . fs = regs . fs ;
current - > tss ( ) . es = regs . es ;
current - > tss ( ) . ds = regs . ds ;
current - > tss ( ) . edi = regs . edi ;
current - > tss ( ) . esi = regs . esi ;
current - > tss ( ) . ebp = regs . ebp ;
current - > tss ( ) . ebx = regs . ebx ;
current - > tss ( ) . edx = regs . edx ;
current - > tss ( ) . ecx = regs . ecx ;
current - > tss ( ) . eax = regs . eax ;
current - > tss ( ) . eip = regs . eip ;
current - > tss ( ) . cs = regs . cs ;
current - > tss ( ) . eflags = regs . eflags ;
// Compute process stack pointer.
// Add 12 for CS, EIP, EFLAGS (interrupt mechanic)
current - > tss ( ) . esp = regs . esp + 12 ;
current - > tss ( ) . ss = regs . ss ;
if ( ( current - > tss ( ) . cs & 3 ) ! = 0 ) {
current - > tss ( ) . ss = regs . ss_if_crossRing ;
current - > tss ( ) . esp = regs . esp_if_crossRing ;
}
if ( ! pick_next ( ) )
return ;
prepare_for_iret_to_new_process ( ) ;
// Set the NT (nested task) flag.
asm (
" pushf \n "
" orl $0x00004000, (%esp) \n "
2019-06-07 11:43:58 +02:00
" popf \n " ) ;
2018-11-08 00:24:59 +01:00
}