From c6c45408fe024541723180d944f089a0185a690c Mon Sep 17 00:00:00 2001 From: wheremyfoodat Date: Tue, 18 Apr 2023 03:14:25 +0300 Subject: [PATCH] Starting on sync object rewrite #3 --- include/kernel/kernel_types.hpp | 31 ++++++++++++++++++++++++++++--- src/core/kernel/events.cpp | 10 +++++++++- src/core/kernel/kernel.cpp | 1 + src/core/kernel/threads.cpp | 5 +++++ 4 files changed, 43 insertions(+), 4 deletions(-) diff --git a/include/kernel/kernel_types.hpp b/include/kernel/kernel_types.hpp index 86f9ae07..27893922 100644 --- a/include/kernel/kernel_types.hpp +++ b/include/kernel/kernel_types.hpp @@ -72,10 +72,11 @@ struct Process { }; struct Event { + u64 waitlist; // A bitfield where each bit symbolizes if the thread with thread with the corresponding index is waiting on the event ResetType resetType = ResetType::OneShot; bool fired = false; - Event(ResetType resetType) : resetType(resetType) {} + Event(ResetType resetType) : resetType(resetType), waitlist(0) {} }; struct Port { @@ -138,6 +139,9 @@ struct Thread { u32 cpsr; u32 fpscr; u32 tlsBase; // Base pointer for thread-local storage + + // A list of threads waiting for this thread to terminate. Yes, threads are sync objects too. + u64 threadsWaitingForTermination; }; static const char* kernelObjectTypeToString(KernelObjectType t) { @@ -161,17 +165,19 @@ static const char* kernelObjectTypeToString(KernelObjectType t) { } struct Mutex { + u64 waitlist; // Refer to the getWaitlist function below for documentation Handle ownerThread = 0; // Index of the thread that holds the mutex if it's locked bool locked; - Mutex(bool lock = false) : locked(lock) {} + Mutex(bool lock = false) : locked(lock), waitlist(0) {} }; struct Semaphore { + u64 waitlist; // Refer to the getWaitlist function below for documentation u32 initialCount; u32 maximumCount; - Semaphore(u32 initialCount, u32 maximumCount) : initialCount(initialCount), maximumCount(maximumCount) {} + Semaphore(u32 initialCount, u32 maximumCount) : initialCount(initialCount), maximumCount(maximumCount), waitlist(0) {} }; struct MemoryBlock { @@ -205,4 +211,23 @@ struct KernelObject { const char* getTypeName() { return kernelObjectTypeToString(type); } + + // Retrieves a reference to the waitlist for a specified object + // We return a reference because this function is only called in the kernel threading internals + // We want the kernel to be able to easily manage waitlists, by reading/parsing them or setting/clearing bits. + // As we mention in the definition of the "Event" struct, the format for wailists is very simple and made to be efficient. + // Each bit corresponds to a thread index and denotes whether the corresponding thread is waiting on this object + // For example if bit 0 of the wait list is set, then the thread with index 0 is waiting on our object + u64& getWaitlist() { + // This code is actually kinda trash but eh good enough + switch (type) { + case KernelObjectType::Event: return getData()->waitlist; + case KernelObjectType::Mutex: return getData()->waitlist; + case KernelObjectType::Semaphore: return getData()->waitlist; + case KernelObjectType::Thread: return getData()->threadsWaitingForTermination; + // This should be unreachable once we fully implement sync objects + default: [[unlikely]] + Helpers::panic("Called GetWaitList on kernel object without a waitlist (Type: %s)", getTypeName()); + } + } }; \ No newline at end of file diff --git a/src/core/kernel/events.cpp b/src/core/kernel/events.cpp index df729f1d..cab777f7 100644 --- a/src/core/kernel/events.cpp +++ b/src/core/kernel/events.cpp @@ -111,6 +111,10 @@ void Kernel::waitSynchronization1() { t.sleepTick = cpu.getTicks(); t.waitingNanoseconds = ns; t.waitList[0] = handle; + + // Add the current thread to the object's wait list + object->getWaitlist() |= (1ull << currentThreadIndex); + switchToNextThread(); } } @@ -148,8 +152,12 @@ void Kernel::waitSynchronizationN() { } if (!isWaitable(object)) [[unlikely]] { - //Helpers::panic("Tried to wait on a non waitable object. Type: %s, handle: %X\n", object->getTypeName(), handle); + Helpers::panic("Tried to wait on a non waitable object in WaitSyncN. Type: %s, handle: %X\n", + object->getTypeName(), handle); } + + // Add the current thread to the object's wait list + object->getWaitlist() |= (1ull << currentThreadIndex); } regs[0] = SVCResult::Success; diff --git a/src/core/kernel/kernel.cpp b/src/core/kernel/kernel.cpp index f22ab051..dd6a6e00 100644 --- a/src/core/kernel/kernel.cpp +++ b/src/core/kernel/kernel.cpp @@ -111,6 +111,7 @@ void Kernel::reset() { for (auto& t : threads) { t.status = ThreadStatus::Dead; t.waitList.clear(); + t.threadsWaitingForTermination = 0; // No threads are waiting for this thread to terminate cause it's dead } for (auto& object : objects) { diff --git a/src/core/kernel/threads.cpp b/src/core/kernel/threads.cpp index 3ff12af2..5dc11257 100644 --- a/src/core/kernel/threads.cpp +++ b/src/core/kernel/threads.cpp @@ -147,6 +147,7 @@ Handle Kernel::makeThread(u32 entrypoint, u32 initialSP, u32 priority, s32 id, u t.status = status; t.handle = ret; t.waitingAddress = 0; + t.threadsWaitingForTermination = 0; // Thread just spawned, no other threads waiting for it to terminate t.cpsr = CPSR::UserMode | (isThumb ? CPSR::Thumb : 0); t.fpscr = FPSCR::ThreadDefault; @@ -310,6 +311,10 @@ void Kernel::exitThread() { t.status = ThreadStatus::Dead; aliveThreadCount--; + // Check if any threads are sleeping, waiting for this thread to terminate, and wake them up + if (t.threadsWaitingForTermination != 0) + Helpers::panic("TODO: Implement threads sleeping until another thread terminates"); + switchToNextThread(); }