aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorClyne Sullivan <clyne@bitgloo.com>2024-09-27 10:21:26 -0400
committerClyne Sullivan <clyne@bitgloo.com>2024-09-27 10:21:26 -0400
commit35aa65037bebd800f9cba2b558d245552960a5d7 (patch)
treef307e44b7f30f7d9f6ab694c11c4a046c988b1ba
parent56f5c483ee516245867a312c7e1520c3f4df16c7 (diff)
fixed multithreading!
-rw-r--r--pit.cpp2
-rw-r--r--tasking.cpp59
-rw-r--r--tasking.hpp2
3 files changed, 46 insertions, 17 deletions
diff --git a/pit.cpp b/pit.cpp
index 4cafc57..f3baa4e 100644
--- a/pit.cpp
+++ b/pit.cpp
@@ -9,7 +9,7 @@ static void timer_callback(const Registers& regs)
{
ticks = ticks + 1;
- schedule(const_cast<Registers&>(regs));
+ schedule(regs);
}
void pit_initialize(std::uint32_t frequency)
diff --git a/tasking.cpp b/tasking.cpp
index 9e2cd49..5cfe091 100644
--- a/tasking.cpp
+++ b/tasking.cpp
@@ -4,31 +4,56 @@
struct Task
{
- Registers regs;
- bool valid = false;
+ enum class State {
+ Invalid,
+ Staging,
+ Running
+ };
+
+ using enum State;
+
+ std::uint32_t esp;
+ std::uint32_t ebp;
+ State state = State::Invalid;
};
static std::array<Task, 4> tasks;
static int current = -1;
-void schedule(Registers& regs)
+void schedule(const Registers& regs)
{
if (current < 0)
return;
- tasks[current].regs = regs;
+ asm volatile(R"(
+ mov %%esp, %0
+ mov %%ebp, %1
+ )" : "=m" (tasks[current].esp), "=m" (tasks[current].ebp));
do {
if (++current >= tasks.size())
current = 0;
- } while (!tasks[current].valid);
+ } while (tasks[current].state == Task::Invalid);
- regs = tasks[current].regs;
+ asm volatile(R"(
+ mov %0, %%esp
+ mov %1, %%ebp
+ )" :: "m" (tasks[current].esp), "m" (tasks[current].ebp));
+
+ if (tasks[current].state == Task::Staging) {
+ tasks[current].state = Task::Running;
+ asm volatile(R"(
+ pop %eax
+ popa
+ add $0x4, %esp
+ iret
+ )");
+ }
}
void tasking_initialize()
{
- tasks[0].valid = true;
+ tasks[0].state = Task::Running;
current = 0;
asm volatile("int $0x20");
}
@@ -37,7 +62,7 @@ bool tasking_spawn(void (*entry)(), unsigned ssize)
{
int i = -1;
for (i = 0; i < tasks.size(); ++i) {
- if (!tasks[i].valid)
+ if (tasks[i].state == Task::Invalid)
break;
}
@@ -46,15 +71,19 @@ bool tasking_spawn(void (*entry)(), unsigned ssize)
tasks[i] = Task();
- auto& r = tasks[i].regs;
auto stack = reinterpret_cast<std::uint32_t>(new std::uint8_t[ssize]);
- r.ebp = stack + ssize;
- r.esp = r.ebp;
- r.eip = reinterpret_cast<std::uint32_t>(entry);
- r.cs = 0x8;
- r.eflags = tasks[current].regs.eflags;
+ const auto stackend = stack + ssize;
+ const auto regbase = stackend - sizeof(Registers);
+ auto r = reinterpret_cast<Registers *>(regbase);
+ r->ebp = stackend;
+ r->esp = stackend;
+ r->eip = reinterpret_cast<std::uint32_t>(entry);
+ r->cs = 0x8;
+ asm volatile("pushfl; pop %%eax" : "=a"(r->eflags));
- tasks[i].valid = true;
+ tasks[i].esp = regbase;
+ tasks[i].ebp = stackend;
+ tasks[i].state = Task::Staging;
return true;
}
diff --git a/tasking.hpp b/tasking.hpp
index 5e03f25..64a738e 100644
--- a/tasking.hpp
+++ b/tasking.hpp
@@ -6,7 +6,7 @@
void tasking_initialize();
bool tasking_spawn(void (*entry)(), unsigned ssize);
-void schedule(Registers& regs);
+void schedule(const Registers& regs);
#endif // TASKING_HPP