2022-09-20 17:58:04 +00:00
|
|
|
#define MODULE "sched"
|
|
|
|
|
|
|
|
#include "thread/Scheduler.h"
|
2022-09-20 18:02:08 +00:00
|
|
|
#include "assert.h"
|
2022-09-20 17:58:04 +00:00
|
|
|
#include "interrupts/Interrupts.h"
|
|
|
|
#include "log/Log.h"
|
2022-09-24 19:45:13 +00:00
|
|
|
#include "memory/MemoryManager.h"
|
2022-09-23 14:41:43 +00:00
|
|
|
#include "memory/VMM.h"
|
2022-09-21 15:56:53 +00:00
|
|
|
#include "misc/hang.h"
|
2022-09-25 14:56:00 +00:00
|
|
|
#include "panic/Panic.h"
|
2022-09-24 21:18:33 +00:00
|
|
|
#include "std/stdlib.h"
|
2022-09-20 17:58:04 +00:00
|
|
|
#include "std/string.h"
|
2022-09-21 15:56:53 +00:00
|
|
|
#include "thread/PIT.h"
|
2022-09-20 17:58:04 +00:00
|
|
|
#include "thread/Task.h"
|
|
|
|
|
2022-09-25 16:13:20 +00:00
|
|
|
#define TASK_PAGES_IN_STACK 4
|
|
|
|
|
2022-09-20 17:58:04 +00:00
|
|
|
static uint64_t task_num = 0;
|
|
|
|
|
|
|
|
static Task idle_task;
|
|
|
|
|
|
|
|
static uint64_t free_tid = 0;
|
|
|
|
|
|
|
|
static Task* sched_current_task;
|
|
|
|
static Task* base_task;
|
|
|
|
static Task* end_task;
|
|
|
|
|
2022-10-01 10:16:30 +00:00
|
|
|
extern "C" void idle_task_function();
|
2022-09-20 17:58:04 +00:00
|
|
|
|
|
|
|
static uint64_t frequency;
|
|
|
|
|
|
|
|
void Scheduler::init()
|
|
|
|
{
|
|
|
|
memset(&idle_task, 0, sizeof(Task));
|
|
|
|
idle_task.id = free_tid++;
|
2022-10-01 10:16:30 +00:00
|
|
|
idle_task.regs.rip = (uint64_t)idle_task_function;
|
2022-09-24 19:45:13 +00:00
|
|
|
idle_task.regs.rsp = (uint64_t)MemoryManager::get_page();
|
2022-09-20 17:58:04 +00:00
|
|
|
idle_task.regs.cs = 0x08;
|
2022-09-21 19:03:24 +00:00
|
|
|
idle_task.regs.ss = 0x10;
|
2022-10-01 10:16:30 +00:00
|
|
|
idle_task.regs.rflags = (1 << 21) | (1 << 9);
|
2022-09-20 17:58:04 +00:00
|
|
|
idle_task.task_sleep = 1000;
|
2022-09-21 19:06:00 +00:00
|
|
|
idle_task.state = idle_task.Idle;
|
2022-09-20 17:58:04 +00:00
|
|
|
|
2022-09-25 16:12:12 +00:00
|
|
|
base_task = new Task;
|
2022-09-20 17:58:04 +00:00
|
|
|
end_task = base_task;
|
|
|
|
sched_current_task = base_task;
|
|
|
|
sched_current_task->id = free_tid++;
|
2022-09-20 18:02:08 +00:00
|
|
|
sched_current_task->task_time = 20; // gets 20 ms of cpu time before next switch
|
2022-09-21 19:06:00 +00:00
|
|
|
sched_current_task->next_task = sched_current_task;
|
|
|
|
sched_current_task->prev_task = sched_current_task;
|
|
|
|
sched_current_task->state = sched_current_task->Running;
|
|
|
|
task_num++;
|
2022-09-20 17:58:04 +00:00
|
|
|
// the other registers will be saved next task switch
|
|
|
|
|
|
|
|
frequency = 1000 / PIT::frequency();
|
|
|
|
}
|
|
|
|
|
|
|
|
void Scheduler::add_kernel_task(void (*task)(void))
|
|
|
|
{
|
2022-09-25 16:12:12 +00:00
|
|
|
Task* new_task = new Task;
|
2022-09-24 21:18:33 +00:00
|
|
|
ASSERT(new_task);
|
2022-09-20 17:58:04 +00:00
|
|
|
new_task->id = free_tid++;
|
|
|
|
new_task->regs.rip = (uint64_t)task;
|
2022-09-25 16:13:20 +00:00
|
|
|
new_task->allocated_stack =
|
|
|
|
(uint64_t)MemoryManager::get_pages(TASK_PAGES_IN_STACK); // 16 KB is enough for everyone, right?
|
2022-09-21 19:06:00 +00:00
|
|
|
new_task->regs.rsp = new_task->allocated_stack + (4096 * 4) - sizeof(uintptr_t);
|
2022-09-20 17:58:04 +00:00
|
|
|
new_task->regs.cs = 0x08;
|
2022-09-21 19:03:24 +00:00
|
|
|
new_task->regs.ss = 0x10;
|
2022-09-25 18:35:05 +00:00
|
|
|
new_task->regs.ds = 0x10;
|
2022-09-20 17:58:04 +00:00
|
|
|
asm volatile("pushfq; movq (%%rsp), %%rax; movq %%rax, %0; popfq;" : "=m"(new_task->regs.rflags)::"%rax");
|
2022-09-20 18:02:08 +00:00
|
|
|
new_task->regs.rflags |= 0x200; // enable interrupts
|
2022-09-20 17:58:04 +00:00
|
|
|
new_task->task_sleep = 0;
|
|
|
|
new_task->task_time = 0;
|
2022-09-25 18:35:05 +00:00
|
|
|
new_task->cpu_time = 0;
|
2022-09-20 17:58:04 +00:00
|
|
|
end_task->next_task = new_task;
|
2022-09-21 19:06:00 +00:00
|
|
|
new_task->prev_task = end_task;
|
|
|
|
base_task->prev_task = new_task;
|
|
|
|
new_task->next_task = base_task;
|
2022-09-20 17:58:04 +00:00
|
|
|
end_task = new_task;
|
2022-09-21 19:06:00 +00:00
|
|
|
new_task->state = new_task->Running;
|
|
|
|
task_num++;
|
|
|
|
kinfoln("Adding kernel task: starts at %lx, tid %ld, stack at %lx, total tasks: %ld", new_task->regs.rip,
|
|
|
|
new_task->id, new_task->regs.rsp, task_num);
|
2022-09-20 17:58:04 +00:00
|
|
|
}
|
|
|
|
|
2022-09-23 14:41:43 +00:00
|
|
|
void Scheduler::add_user_task(void* task)
|
2022-09-21 19:06:00 +00:00
|
|
|
{
|
2022-09-25 16:12:12 +00:00
|
|
|
Task* new_task = new Task;
|
2022-09-24 21:18:33 +00:00
|
|
|
ASSERT(new_task);
|
2022-09-21 19:06:00 +00:00
|
|
|
new_task->id = free_tid++;
|
2022-09-23 14:41:43 +00:00
|
|
|
new_task->regs.rip = (uint64_t)task;
|
2022-09-25 16:13:20 +00:00
|
|
|
new_task->allocated_stack = (uint64_t)MemoryManager::get_pages(
|
|
|
|
TASK_PAGES_IN_STACK, MAP_READ_WRITE | MAP_USER); // 16 KB is enough for everyone, right?
|
2022-09-21 19:06:00 +00:00
|
|
|
new_task->regs.rsp = new_task->allocated_stack + (4096 * 4) - sizeof(uintptr_t);
|
|
|
|
new_task->regs.cs = 0x18 | 0x03;
|
|
|
|
new_task->regs.ss = 0x20 | 0x03;
|
2022-09-25 18:35:05 +00:00
|
|
|
new_task->regs.ds = 0x20 | 0x03;
|
2022-09-21 19:06:00 +00:00
|
|
|
new_task->regs.rflags = (1 << 21) | (1 << 9); // enable interrupts
|
|
|
|
new_task->task_sleep = 0;
|
|
|
|
new_task->task_time = 0;
|
2022-09-25 18:35:05 +00:00
|
|
|
new_task->cpu_time = 0;
|
2022-09-21 19:06:00 +00:00
|
|
|
end_task->next_task = new_task;
|
|
|
|
new_task->prev_task = end_task;
|
|
|
|
base_task->prev_task = new_task;
|
|
|
|
new_task->next_task = base_task;
|
|
|
|
end_task = new_task;
|
|
|
|
new_task->state = new_task->Running;
|
|
|
|
task_num++;
|
|
|
|
kinfoln("Adding user task: starts at %lx, tid %ld, stack at %lx, total tasks: %ld", new_task->regs.rip,
|
|
|
|
new_task->id, new_task->regs.rsp, task_num);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Scheduler::reap_task(Task* task)
|
|
|
|
{
|
|
|
|
ASSERT(!Interrupts::is_in_handler());
|
|
|
|
task_num--;
|
|
|
|
Task* exiting_task = task;
|
|
|
|
kinfoln("reaping task %ld", exiting_task->id);
|
2022-09-25 16:13:20 +00:00
|
|
|
if (exiting_task->allocated_stack)
|
|
|
|
MemoryManager::release_pages((void*)exiting_task->allocated_stack, TASK_PAGES_IN_STACK);
|
2022-09-25 16:12:12 +00:00
|
|
|
delete exiting_task;
|
2022-09-21 19:06:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void Scheduler::task_exit(Context* context)
|
|
|
|
{
|
|
|
|
ASSERT(Interrupts::is_in_handler());
|
2022-09-25 15:49:51 +00:00
|
|
|
kdbgln("exit: task %ld finished running, used %ld ms of cpu time", sched_current_task->id,
|
|
|
|
sched_current_task->cpu_time);
|
2022-09-21 19:06:00 +00:00
|
|
|
sched_current_task->state = sched_current_task->Exited;
|
|
|
|
task_yield(context);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Scheduler::task_misbehave(Context* context)
|
|
|
|
{
|
|
|
|
ASSERT(Interrupts::is_in_handler());
|
2022-09-25 15:49:51 +00:00
|
|
|
kdbgln("exit: task %ld misbehaved, used %ld ms of cpu time", sched_current_task->id, sched_current_task->cpu_time);
|
2022-09-21 19:06:00 +00:00
|
|
|
sched_current_task->state = sched_current_task->Exited;
|
|
|
|
task_yield(context);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Scheduler::reap_tasks()
|
2022-09-20 17:58:04 +00:00
|
|
|
{
|
2022-09-21 19:06:00 +00:00
|
|
|
Interrupts::disable();
|
|
|
|
ASSERT(!Interrupts::is_in_handler());
|
|
|
|
Task* reap_base = nullptr;
|
|
|
|
Task* reap_end = nullptr;
|
2022-09-20 17:58:04 +00:00
|
|
|
Task* task = base_task;
|
2022-09-21 19:06:00 +00:00
|
|
|
Task* task_reaping;
|
|
|
|
uint64_t iter_index = 0;
|
|
|
|
do {
|
|
|
|
if (task->state == task->Exited)
|
|
|
|
{
|
2022-09-25 14:56:00 +00:00
|
|
|
if (task == base_task && task == end_task) { panic("Last task exited"); }
|
2022-09-21 19:06:00 +00:00
|
|
|
else if (task == base_task) { base_task = task->next_task; }
|
|
|
|
else if (task == end_task) { end_task = task->prev_task; }
|
|
|
|
if (!reap_base)
|
|
|
|
{
|
|
|
|
reap_base = task;
|
|
|
|
reap_end = task;
|
|
|
|
task->prev_task->next_task = task->next_task;
|
|
|
|
task->next_task->prev_task = task->prev_task;
|
|
|
|
task->prev_task = nullptr;
|
|
|
|
task_reaping = task;
|
|
|
|
task = task->next_task;
|
|
|
|
task_reaping->next_task = nullptr;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
reap_end->next_task = task;
|
|
|
|
task->prev_task->next_task = task->next_task;
|
|
|
|
task->next_task->prev_task = task->prev_task;
|
|
|
|
task->prev_task = nullptr;
|
|
|
|
reap_end = task;
|
|
|
|
task_reaping = task;
|
|
|
|
task = task->next_task;
|
|
|
|
task_reaping->next_task = nullptr;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else { task = task->next_task; }
|
|
|
|
iter_index++;
|
|
|
|
} while (iter_index < task_num);
|
|
|
|
task = reap_base;
|
2022-09-20 17:58:04 +00:00
|
|
|
while (task)
|
|
|
|
{
|
2022-09-21 19:06:00 +00:00
|
|
|
Task* reaped_task = task;
|
|
|
|
task = task->next_task;
|
|
|
|
reap_task(reaped_task);
|
|
|
|
}
|
|
|
|
Interrupts::enable();
|
|
|
|
}
|
|
|
|
|
|
|
|
static void sched_decrement_sleep_times()
|
|
|
|
{
|
|
|
|
Task* task = base_task;
|
|
|
|
if (!task) return;
|
|
|
|
do {
|
2022-09-20 17:58:04 +00:00
|
|
|
if (task->task_sleep > 0)
|
|
|
|
{
|
|
|
|
task->task_sleep -= frequency;
|
|
|
|
if (task->task_sleep < 0) task->task_sleep = 0;
|
|
|
|
}
|
2022-09-21 19:06:00 +00:00
|
|
|
if (task->task_sleep == 0 && task->state == task->Sleeping) task->state = task->Running;
|
2022-09-20 17:58:04 +00:00
|
|
|
task = task->next_task;
|
2022-09-21 19:06:00 +00:00
|
|
|
} while (task != base_task);
|
2022-09-20 17:58:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void Scheduler::task_tick(Context* context)
|
|
|
|
{
|
2022-09-20 18:02:08 +00:00
|
|
|
ASSERT(Interrupts::is_in_handler());
|
2022-09-21 19:06:00 +00:00
|
|
|
Interrupts::disable();
|
2022-09-20 17:58:04 +00:00
|
|
|
sched_decrement_sleep_times();
|
2022-09-21 19:06:00 +00:00
|
|
|
if (sched_current_task->id == 0) return task_yield(context);
|
2022-09-20 17:58:04 +00:00
|
|
|
sched_current_task->task_time -= frequency;
|
2022-09-25 15:49:51 +00:00
|
|
|
sched_current_task->cpu_time += frequency;
|
2022-09-20 17:58:04 +00:00
|
|
|
if (sched_current_task->task_time < 0)
|
|
|
|
{
|
|
|
|
sched_current_task->task_time = 0;
|
|
|
|
task_yield(context);
|
|
|
|
}
|
2022-09-21 19:06:00 +00:00
|
|
|
Interrupts::enable();
|
2022-09-20 17:58:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void Scheduler::task_yield(Context* context)
|
|
|
|
{
|
2022-09-20 18:02:08 +00:00
|
|
|
ASSERT(Interrupts::is_in_handler());
|
2022-09-21 19:06:00 +00:00
|
|
|
Interrupts::disable();
|
2022-09-20 17:58:04 +00:00
|
|
|
get_context_to_task(*sched_current_task, context);
|
|
|
|
bool was_idle = false;
|
2022-09-21 19:06:00 +00:00
|
|
|
if (sched_current_task->state == sched_current_task->Idle)
|
2022-09-20 17:58:04 +00:00
|
|
|
{
|
2022-09-21 19:06:00 +00:00
|
|
|
sched_current_task = end_task;
|
2022-09-20 17:58:04 +00:00
|
|
|
was_idle = true;
|
|
|
|
}
|
|
|
|
Task* original_task = sched_current_task;
|
|
|
|
do {
|
|
|
|
sched_current_task = sched_current_task->next_task;
|
2022-09-21 19:06:00 +00:00
|
|
|
if (sched_current_task->state == sched_current_task->Running)
|
2022-09-20 17:58:04 +00:00
|
|
|
{
|
2022-09-20 18:02:08 +00:00
|
|
|
sched_current_task->task_time = 20;
|
2022-09-20 17:58:04 +00:00
|
|
|
set_context_from_task(*sched_current_task, context);
|
2022-09-21 19:06:00 +00:00
|
|
|
Interrupts::enable();
|
2022-09-20 17:58:04 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
} while (sched_current_task != original_task);
|
2022-09-21 19:06:00 +00:00
|
|
|
sched_current_task = &idle_task;
|
|
|
|
sched_current_task->task_time = frequency;
|
|
|
|
if (!was_idle) { set_context_from_task(*sched_current_task, context); }
|
|
|
|
Interrupts::enable();
|
2022-09-20 17:58:04 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Scheduler::yield()
|
|
|
|
{
|
2022-09-29 17:17:43 +00:00
|
|
|
asm volatile("int $0x42" : : "a"(1));
|
2022-09-20 17:58:04 +00:00
|
|
|
}
|
|
|
|
|
2022-09-21 19:06:00 +00:00
|
|
|
void Scheduler::exit()
|
|
|
|
{
|
2022-09-29 17:17:43 +00:00
|
|
|
asm volatile("int $0x42" : : "a"(0));
|
2022-09-21 19:06:00 +00:00
|
|
|
}
|
|
|
|
|
2022-09-20 17:58:04 +00:00
|
|
|
void Scheduler::sleep(unsigned long ms)
|
|
|
|
{
|
2022-09-29 17:17:43 +00:00
|
|
|
asm volatile("int $0x42" : : "D"(ms), "a"(2));
|
2022-09-20 17:58:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Task* Scheduler::current_task()
|
|
|
|
{
|
|
|
|
return sched_current_task;
|
|
|
|
}
|