mirror of
https://github.com/wheremyfoodat/Panda3DS.git
synced 2025-04-07 06:35:40 +12:00
Starting on sync object rewrite #3
This commit is contained in:
parent
517d1abee8
commit
c6c45408fe
4 changed files with 43 additions and 4 deletions
|
@ -72,10 +72,11 @@ struct Process {
|
||||||
};
|
};
|
||||||
|
|
||||||
struct Event {
|
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;
|
ResetType resetType = ResetType::OneShot;
|
||||||
bool fired = false;
|
bool fired = false;
|
||||||
|
|
||||||
Event(ResetType resetType) : resetType(resetType) {}
|
Event(ResetType resetType) : resetType(resetType), waitlist(0) {}
|
||||||
};
|
};
|
||||||
|
|
||||||
struct Port {
|
struct Port {
|
||||||
|
@ -138,6 +139,9 @@ struct Thread {
|
||||||
u32 cpsr;
|
u32 cpsr;
|
||||||
u32 fpscr;
|
u32 fpscr;
|
||||||
u32 tlsBase; // Base pointer for thread-local storage
|
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) {
|
static const char* kernelObjectTypeToString(KernelObjectType t) {
|
||||||
|
@ -161,17 +165,19 @@ static const char* kernelObjectTypeToString(KernelObjectType t) {
|
||||||
}
|
}
|
||||||
|
|
||||||
struct Mutex {
|
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
|
Handle ownerThread = 0; // Index of the thread that holds the mutex if it's locked
|
||||||
bool locked;
|
bool locked;
|
||||||
|
|
||||||
Mutex(bool lock = false) : locked(lock) {}
|
Mutex(bool lock = false) : locked(lock), waitlist(0) {}
|
||||||
};
|
};
|
||||||
|
|
||||||
struct Semaphore {
|
struct Semaphore {
|
||||||
|
u64 waitlist; // Refer to the getWaitlist function below for documentation
|
||||||
u32 initialCount;
|
u32 initialCount;
|
||||||
u32 maximumCount;
|
u32 maximumCount;
|
||||||
|
|
||||||
Semaphore(u32 initialCount, u32 maximumCount) : initialCount(initialCount), maximumCount(maximumCount) {}
|
Semaphore(u32 initialCount, u32 maximumCount) : initialCount(initialCount), maximumCount(maximumCount), waitlist(0) {}
|
||||||
};
|
};
|
||||||
|
|
||||||
struct MemoryBlock {
|
struct MemoryBlock {
|
||||||
|
@ -205,4 +211,23 @@ struct KernelObject {
|
||||||
const char* getTypeName() {
|
const char* getTypeName() {
|
||||||
return kernelObjectTypeToString(type);
|
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<Event>()->waitlist;
|
||||||
|
case KernelObjectType::Mutex: return getData<Mutex>()->waitlist;
|
||||||
|
case KernelObjectType::Semaphore: return getData<Mutex>()->waitlist;
|
||||||
|
case KernelObjectType::Thread: return getData<Thread>()->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());
|
||||||
|
}
|
||||||
|
}
|
||||||
};
|
};
|
|
@ -111,6 +111,10 @@ void Kernel::waitSynchronization1() {
|
||||||
t.sleepTick = cpu.getTicks();
|
t.sleepTick = cpu.getTicks();
|
||||||
t.waitingNanoseconds = ns;
|
t.waitingNanoseconds = ns;
|
||||||
t.waitList[0] = handle;
|
t.waitList[0] = handle;
|
||||||
|
|
||||||
|
// Add the current thread to the object's wait list
|
||||||
|
object->getWaitlist() |= (1ull << currentThreadIndex);
|
||||||
|
|
||||||
switchToNextThread();
|
switchToNextThread();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -148,8 +152,12 @@ void Kernel::waitSynchronizationN() {
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!isWaitable(object)) [[unlikely]] {
|
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;
|
regs[0] = SVCResult::Success;
|
||||||
|
|
|
@ -111,6 +111,7 @@ void Kernel::reset() {
|
||||||
for (auto& t : threads) {
|
for (auto& t : threads) {
|
||||||
t.status = ThreadStatus::Dead;
|
t.status = ThreadStatus::Dead;
|
||||||
t.waitList.clear();
|
t.waitList.clear();
|
||||||
|
t.threadsWaitingForTermination = 0; // No threads are waiting for this thread to terminate cause it's dead
|
||||||
}
|
}
|
||||||
|
|
||||||
for (auto& object : objects) {
|
for (auto& object : objects) {
|
||||||
|
|
|
@ -147,6 +147,7 @@ Handle Kernel::makeThread(u32 entrypoint, u32 initialSP, u32 priority, s32 id, u
|
||||||
t.status = status;
|
t.status = status;
|
||||||
t.handle = ret;
|
t.handle = ret;
|
||||||
t.waitingAddress = 0;
|
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.cpsr = CPSR::UserMode | (isThumb ? CPSR::Thumb : 0);
|
||||||
t.fpscr = FPSCR::ThreadDefault;
|
t.fpscr = FPSCR::ThreadDefault;
|
||||||
|
@ -310,6 +311,10 @@ void Kernel::exitThread() {
|
||||||
t.status = ThreadStatus::Dead;
|
t.status = ThreadStatus::Dead;
|
||||||
aliveThreadCount--;
|
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();
|
switchToNextThread();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Add table
Reference in a new issue