1
Fork 0
mirror of https://github.com/RGBCube/serenity synced 2025-07-25 15:27:35 +00:00

Move timer tick handling into Scheduler.

This commit is contained in:
Andreas Kling 2018-11-08 00:24:59 +01:00
parent 27fded7002
commit ac1d12465f
3 changed files with 56 additions and 59 deletions

View file

@ -258,3 +258,50 @@ void Scheduler::initialize()
current = nullptr; current = nullptr;
load_task_register(s_redirection.selector); load_task_register(s_redirection.selector);
} }
void Scheduler::timer_tick(RegisterDump& regs)
{
if (!current)
return;
system.uptime++;
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"
"popf\n"
);
}

View file

@ -1,20 +1,24 @@
#pragma once #pragma once
#include <AK/Assertions.h> #include <AK/Assertions.h>
class Process; class Process;
struct RegisterDump;
extern Process* current; extern Process* current;
class Scheduler { class Scheduler {
public: public:
static void initialize(); static void initialize();
static void timer_tick(RegisterDump&);
static bool pick_next(); static bool pick_next();
static void pick_next_and_switch_now(); static void pick_next_and_switch_now();
static void switch_now(); static void switch_now();
static bool yield(); static bool yield();
static bool context_switch(Process&); static bool context_switch(Process&);
static void prepare_for_iret_to_new_process();
static void prepare_to_modify_tss(Process&); static void prepare_to_modify_tss(Process&);
private:
static void prepare_for_iret_to_new_process();
}; };
int sched_yield(); int sched_yield();

View file

@ -1,24 +1,16 @@
#include "i8253.h" #include "i8253.h"
#include "i386.h" #include "i386.h"
#include "IO.h" #include "IO.h"
#include "VGA.h"
#include "Process.h"
#include "system.h"
#include "PIC.h" #include "PIC.h"
#include "Scheduler.h" #include "Scheduler.h"
#define IRQ_TIMER 0 #define IRQ_TIMER 0
extern "C" void tick_ISR(); extern "C" void tick_ISR();
extern "C" void clock_handle(); extern "C" void clock_handle(RegisterDump&);
extern volatile DWORD state_dump;
asm( asm(
".globl tick_ISR \n" ".globl tick_ISR \n"
".globl state_dump \n"
"state_dump: \n"
".long 0\n"
"tick_ISR: \n" "tick_ISR: \n"
" pusha\n" " pusha\n"
" pushw %ds\n" " pushw %ds\n"
@ -34,7 +26,7 @@ asm(
" popw %es\n" " popw %es\n"
" popw %fs\n" " popw %fs\n"
" popw %gs\n" " popw %gs\n"
" mov %esp, state_dump\n" " mov %esp, %eax\n"
" call clock_handle\n" " call clock_handle\n"
" popw %gs\n" " popw %gs\n"
" popw %gs\n" " popw %gs\n"
@ -66,54 +58,10 @@ asm(
/* Miscellaneous */ /* Miscellaneous */
#define BASE_FREQUENCY 1193182 #define BASE_FREQUENCY 1193182
void clock_handle() void clock_handle(RegisterDump& regs)
{ {
IRQHandlerScope scope(IRQ_TIMER); IRQHandlerScope scope(IRQ_TIMER);
Scheduler::timer_tick(regs);
if (!current)
return;
system.uptime++;
if (current->tick())
return;
auto& regs = *reinterpret_cast<RegisterDump*>(state_dump);
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 (!Scheduler::pick_next())
return;
Scheduler::prepare_for_iret_to_new_process();
// Set the NT (nested task) flag.
asm(
"pushf\n"
"orl $0x00004000, (%esp)\n"
"popf\n"
);
} }
namespace PIT { namespace PIT {
@ -126,8 +74,6 @@ void initialize()
timer_reload = (BASE_FREQUENCY / TICKS_PER_SECOND); timer_reload = (BASE_FREQUENCY / TICKS_PER_SECOND);
/* Send LSB and MSB of timer reload value. */
kprintf("PIT(i8253): %u Hz, square wave (%x)\n", TICKS_PER_SECOND, timer_reload); kprintf("PIT(i8253): %u Hz, square wave (%x)\n", TICKS_PER_SECOND, timer_reload);
IO::out8(TIMER0_CTL, LSB(timer_reload)); IO::out8(TIMER0_CTL, LSB(timer_reload));