mirror of
https://github.com/skyline-emu/skyline.git
synced 2025-01-25 19:01:17 +01:00
Refactor Condition Variable Waiting/Signalling
The way we handled waking/timeouts of condition variables was fairly inaccurate to HOS as we moved locking of the mutex to the waker thread which could change the order of operations and would cause what were functionally spurious wakeups for all awoken threads. This commit fixes it by doing all locks on the waker thread and only awakening the waiter thread once the condition variable was signalled and the mutex was unlocked. In addition, this fixes races between a timeout and a signal that could lead to double-insertion as a result of a refactor of how timeouts work in the new system.
This commit is contained in:
parent
1eb4eec103
commit
f487d81769
@ -218,6 +218,16 @@ namespace skyline::kernel::type {
|
||||
Result KProcess::ConditionalVariableWait(u32 *key, u32 *mutex, KHandle tag, i64 timeout) {
|
||||
TRACE_EVENT_FMT("kernel", "ConditionalVariableWait 0x{:X} (0x{:X})", key, mutex);
|
||||
|
||||
{
|
||||
// Update all waiter information
|
||||
std::unique_lock lock{state.thread->waiterMutex};
|
||||
state.thread->waitKey = mutex;
|
||||
state.thread->waitTag = tag;
|
||||
state.thread->waitConditionVariable = key;
|
||||
state.thread->waitSignalled = false;
|
||||
state.thread->waitResult = {};
|
||||
}
|
||||
|
||||
{
|
||||
std::scoped_lock lock{syncWaiterMutex};
|
||||
auto queue{syncWaiters.equal_range(key)};
|
||||
@ -230,15 +240,72 @@ namespace skyline::kernel::type {
|
||||
}
|
||||
|
||||
if (timeout > 0 && !state.scheduler->TimedWaitSchedule(std::chrono::nanoseconds(timeout))) {
|
||||
bool inQueue{true};
|
||||
{
|
||||
std::unique_lock lock(syncWaiterMutex);
|
||||
// Attempt to remove ourselves from the queue so we cannot be signalled
|
||||
std::unique_lock syncLock{syncWaiterMutex};
|
||||
auto queue{syncWaiters.equal_range(key)};
|
||||
auto iterator{std::find(queue.first, queue.second, SyncWaiters::value_type{key, state.thread})};
|
||||
if (iterator != queue.second)
|
||||
if (syncWaiters.erase(iterator) == queue.second)
|
||||
__atomic_store_n(key, false, __ATOMIC_SEQ_CST);
|
||||
|
||||
syncWaiters.erase(iterator);
|
||||
else
|
||||
inQueue = false;
|
||||
}
|
||||
|
||||
bool shouldWait{false};
|
||||
if (!inQueue) {
|
||||
// If we weren't in the queue then we need to check if we were signalled already
|
||||
while (true) {
|
||||
std::unique_lock lock{state.thread->waiterMutex};
|
||||
|
||||
if (state.thread->waitSignalled) {
|
||||
if (state.thread->waitKey) {
|
||||
auto waitThread{state.thread->waitThread};
|
||||
std::unique_lock waitLock{waitThread->waiterMutex, std::try_to_lock};
|
||||
if (!waitLock) {
|
||||
// If we can't lock the waitThread's waiterMutex then we need to wait without holding the current thread's waiterMutex to avoid a deadlock
|
||||
lock.unlock();
|
||||
waitLock.lock();
|
||||
continue;
|
||||
}
|
||||
|
||||
auto &waiters{waitThread->waiters};
|
||||
auto it{std::find(waiters.begin(), waiters.end(), state.thread)};
|
||||
if (it != waiters.end()) {
|
||||
// If we were signalled but are waiting on locking the associated mutex then we need to cancel our wait
|
||||
waiters.erase(it);
|
||||
state.thread->UpdatePriorityInheritance();
|
||||
|
||||
state.thread->waitKey = nullptr;
|
||||
state.thread->waitTag = 0;
|
||||
state.thread->waitThread = nullptr;
|
||||
} else {
|
||||
// If we were signalled and are no longer waiting on the associated mutex then we're already scheduled
|
||||
shouldWait = true;
|
||||
}
|
||||
} else {
|
||||
// If the waitKey is null then we were signalled and are no longer waiting on the associated mutex
|
||||
shouldWait = true;
|
||||
}
|
||||
} else {
|
||||
// If we were in the process of being signalled but prior to the mutex being locked then we can just cancel our wait
|
||||
state.thread->waitConditionVariable = nullptr;
|
||||
state.thread->waitSignalled = true;
|
||||
}
|
||||
break;
|
||||
}
|
||||
} else {
|
||||
// If we were in the queue then we can just cancel our wait
|
||||
state.thread->waitConditionVariable = nullptr;
|
||||
state.thread->waitSignalled = true;
|
||||
}
|
||||
|
||||
if (shouldWait) {
|
||||
// Wait if we've been signalled in the meantime as it would be problematic to double insert a thread into the scheduler
|
||||
state.scheduler->WaitSchedule();
|
||||
return state.thread->waitResult;
|
||||
}
|
||||
|
||||
state.scheduler->InsertThread(state.thread);
|
||||
state.scheduler->WaitSchedule();
|
||||
|
||||
@ -247,38 +314,78 @@ namespace skyline::kernel::type {
|
||||
state.scheduler->WaitSchedule();
|
||||
}
|
||||
|
||||
KHandle value{};
|
||||
if (!__atomic_compare_exchange_n(mutex, &value, tag, false, __ATOMIC_SEQ_CST, __ATOMIC_SEQ_CST))
|
||||
while (MutexLock(state.thread, mutex, value & ~HandleWaitersBit, tag, true) != Result{})
|
||||
if ((value = __atomic_or_fetch(mutex, HandleWaitersBit, __ATOMIC_SEQ_CST)) == HandleWaitersBit)
|
||||
if (__atomic_compare_exchange_n(mutex, &value, tag, false, __ATOMIC_SEQ_CST, __ATOMIC_SEQ_CST))
|
||||
break;
|
||||
|
||||
return {};
|
||||
return state.thread->waitResult;
|
||||
}
|
||||
|
||||
void KProcess::ConditionalVariableSignal(u32 *key, i32 amount) {
|
||||
TRACE_EVENT_FMT("kernel", "ConditionalVariableSignal 0x{:X}", key);
|
||||
|
||||
i32 waiterCount{amount};
|
||||
while (amount <= 0 || waiterCount) {
|
||||
std::shared_ptr<type::KThread> thread;
|
||||
do {
|
||||
if (thread) {
|
||||
state.scheduler->InsertThread(thread);
|
||||
thread = {};
|
||||
}
|
||||
|
||||
void *conditionVariable{};
|
||||
{
|
||||
// Try to find a thread to signal
|
||||
std::scoped_lock lock{syncWaiterMutex};
|
||||
auto queue{syncWaiters.equal_range(key)};
|
||||
|
||||
if (queue.first != queue.second && (amount <= 0 || waiterCount)) {
|
||||
if (queue.first != queue.second) {
|
||||
// If we found a thread then we need to remove it from the queue
|
||||
thread = queue.first->second;
|
||||
conditionVariable = thread->waitConditionVariable;
|
||||
#ifndef NDEBUG
|
||||
if (conditionVariable != key)
|
||||
Logger::Warn("Condition variable mismatch: 0x{:X} != 0x{:X}", conditionVariable, key);
|
||||
#endif
|
||||
|
||||
syncWaiters.erase(queue.first);
|
||||
waiterCount--;
|
||||
} else if (queue.first == queue.second) {
|
||||
__atomic_store_n(key, false, __ATOMIC_SEQ_CST); // We need to update the boolean flag denoting that there are no more threads waiting on this conditional variable
|
||||
// If we didn't find a thread then we need to clear the boolean flag denoting that there are no more threads waiting on this conditional variable
|
||||
__atomic_store_n(key, false, __ATOMIC_SEQ_CST);
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
std::scoped_lock lock{thread->waiterMutex};
|
||||
if (thread->waitConditionVariable == conditionVariable) {
|
||||
// If the thread is still waiting on the same condition variable then we can signal it (It could no longer be waiting due to a timeout)
|
||||
u32 *mutex{thread->waitKey};
|
||||
KHandle tag{thread->waitTag};
|
||||
|
||||
while (true) {
|
||||
// We need to lock the mutex before the thread can be scheduled
|
||||
KHandle value{};
|
||||
if (__atomic_compare_exchange_n(mutex, &value, tag, false, __ATOMIC_SEQ_CST, __ATOMIC_SEQ_CST)) {
|
||||
// A quick CAS to lock the mutex for the thread, we can just schedule the thread if we succeed
|
||||
state.scheduler->InsertThread(thread);
|
||||
break;
|
||||
}
|
||||
|
||||
if ((value & HandleWaitersBit) == 0)
|
||||
// Set the waiters bit in the mutex if it wasn't already set
|
||||
if (!__atomic_compare_exchange_n(mutex, &value, value | HandleWaitersBit, false, __ATOMIC_SEQ_CST, __ATOMIC_SEQ_CST))
|
||||
continue; // If we failed to set the waiters bit due to an outdated value then try again
|
||||
|
||||
// If we couldn't CAS the lock then we need to let the mutex holder schedule the thread instead of us during an unlock
|
||||
auto result{MutexLock(thread, mutex, value & ~HandleWaitersBit, tag, true)};
|
||||
if (result == result::InvalidCurrentMemory) {
|
||||
continue;
|
||||
} else if (result == result::InvalidHandle) {
|
||||
thread->waitResult = result::InvalidState;
|
||||
state.scheduler->InsertThread(thread);
|
||||
} else if (result != Result{}) {
|
||||
throw exception("Failed to lock mutex: 0x{:X}", result);
|
||||
}
|
||||
break;
|
||||
}
|
||||
|
||||
// Update the thread's wait state to avoid incorrect timeout cancellation behavior
|
||||
thread->waitConditionVariable = nullptr;
|
||||
thread->waitSignalled = true;
|
||||
thread->waitResult = {};
|
||||
}
|
||||
}
|
||||
} while (thread);
|
||||
}
|
||||
|
||||
Result KProcess::WaitForAddress(u32 *address, u32 value, i64 timeout, ArbitrationType type) {
|
||||
|
@ -62,11 +62,14 @@ namespace skyline {
|
||||
bool pendingYield{}; //!< If the thread has been yielded and hasn't been acted upon it yet
|
||||
bool forceYield{}; //!< If the thread has been forcefully yielded by another thread
|
||||
|
||||
std::mutex waiterMutex; //!< Synchronizes operations on mutation of the waiter members
|
||||
std::recursive_mutex waiterMutex; //!< Synchronizes operations on mutation of the waiter members
|
||||
u32 *waitKey; //!< The key of the mutex which this thread is waiting on
|
||||
KHandle waitTag; //!< The handle of the thread which requested the mutex lock
|
||||
std::shared_ptr<KThread> waitThread; //!< The thread which this thread is waiting on
|
||||
std::list<std::shared_ptr<type::KThread>> waiters; //!< A queue of threads waiting on this thread sorted by priority
|
||||
void *waitConditionVariable; //!< The condition variable which this thread is waiting on
|
||||
bool waitSignalled{}; //!< If the conditional variable has been signalled already
|
||||
Result waitResult; //!< The result of the wait operation
|
||||
|
||||
bool isCancellable{false}; //!< If the thread is currently in a position where it's cancellable
|
||||
bool cancelSync{false}; //!< Whether to cancel the SvcWaitSynchronization call this thread currently is in/the next one it joins
|
||||
|
Loading…
x
Reference in New Issue
Block a user