From 4dfe83af6cdfab2dbf682cec35b46cb4b0351515 Mon Sep 17 00:00:00 2001
From: gibbed <rick@gibbed.us>
Date: Sun, 25 Nov 2018 05:08:16 -0600
Subject: [PATCH] [Kernel] Fix threading issues caused by direct use of
 exports.

---
 src/xenia/kernel/xam/xam_net.cc               |  6 +-
 src/xenia/kernel/xboxkrnl/xboxkrnl_rtl.cc     |  6 +-
 .../kernel/xboxkrnl/xboxkrnl_threading.cc     | 82 +++++++++++--------
 .../kernel/xboxkrnl/xboxkrnl_threading.h      | 21 +++--
 4 files changed, 64 insertions(+), 51 deletions(-)

diff --git a/src/xenia/kernel/xam/xam_net.cc b/src/xenia/kernel/xam/xam_net.cc
index 68fc8a853..f64637133 100644
--- a/src/xenia/kernel/xam/xam_net.cc
+++ b/src/xenia/kernel/xam/xam_net.cc
@@ -350,7 +350,7 @@ dword_result_t NetDll_WSAWaitForMultipleEvents(dword_t num_events,
 
   X_STATUS result = 0;
   do {
-    result = xboxkrnl::NtWaitForMultipleObjectsEx(
+    result = xboxkrnl::xeNtWaitForMultipleObjectsEx(
         num_events, events, wait_all, 1, alertable,
         timeout != -1 ? &timeout_wait : nullptr);
   } while (result == X_STATUS_ALERTED);
@@ -384,7 +384,7 @@ dword_result_t NetDll_WSACloseEvent(dword_t event_handle) {
 DECLARE_XAM_EXPORT1(NetDll_WSACloseEvent, kNetworking, kImplemented);
 
 dword_result_t NetDll_WSAResetEvent(dword_t event_handle) {
-  X_STATUS result = xboxkrnl::NtClearEvent(event_handle);
+  X_STATUS result = xboxkrnl::xeNtClearEvent(event_handle);
   if (XFAILED(result)) {
     uint32_t error = xboxkrnl::RtlNtStatusToDosError(result);
     XThread::SetLastError(error);
@@ -395,7 +395,7 @@ dword_result_t NetDll_WSAResetEvent(dword_t event_handle) {
 DECLARE_XAM_EXPORT1(NetDll_WSAResetEvent, kNetworking, kImplemented);
 
 dword_result_t NetDll_WSASetEvent(dword_t event_handle) {
-  X_STATUS result = xboxkrnl::NtSetEvent(event_handle, nullptr);
+  X_STATUS result = xboxkrnl::xeNtSetEvent(event_handle, nullptr);
   if (XFAILED(result)) {
     uint32_t error = xboxkrnl::RtlNtStatusToDosError(result);
     XThread::SetLastError(error);
diff --git a/src/xenia/kernel/xboxkrnl/xboxkrnl_rtl.cc b/src/xenia/kernel/xboxkrnl/xboxkrnl_rtl.cc
index af08243b2..94c36fbc7 100644
--- a/src/xenia/kernel/xboxkrnl/xboxkrnl_rtl.cc
+++ b/src/xenia/kernel/xboxkrnl/xboxkrnl_rtl.cc
@@ -416,8 +416,8 @@ void RtlEnterCriticalSection(pointer_t<X_RTL_CRITICAL_SECTION> cs) {
 
   if (xe::atomic_inc(&cs->lock_count) != 0) {
     // Create a full waiter.
-    KeWaitForSingleObject(reinterpret_cast<void*>(cs.host_address()), 8, 0, 0,
-                          nullptr);
+    xeKeWaitForSingleObject(reinterpret_cast<void*>(cs.host_address()), 8, 0, 0,
+                            nullptr);
   }
 
   assert_true(cs->owning_thread == 0);
@@ -465,7 +465,7 @@ void RtlLeaveCriticalSection(pointer_t<X_RTL_CRITICAL_SECTION> cs) {
   cs->owning_thread = 0;
   if (xe::atomic_dec(&cs->lock_count) != -1) {
     // There were waiters - wake one of them.
-    KeSetEvent(reinterpret_cast<X_KEVENT*>(cs.host_address()), 1, 0);
+    xeKeSetEvent(reinterpret_cast<X_KEVENT*>(cs.host_address()), 1, 0);
   }
 }
 DECLARE_XBOXKRNL_EXPORT2(RtlLeaveCriticalSection, kNone, kImplemented,
diff --git a/src/xenia/kernel/xboxkrnl/xboxkrnl_threading.cc b/src/xenia/kernel/xboxkrnl/xboxkrnl_threading.cc
index 3a26b2ff1..09d7e5439 100644
--- a/src/xenia/kernel/xboxkrnl/xboxkrnl_threading.cc
+++ b/src/xenia/kernel/xboxkrnl/xboxkrnl_threading.cc
@@ -373,7 +373,7 @@ void KeInitializeEvent(pointer_t<X_KEVENT> event_ptr, dword_t event_type,
 }
 DECLARE_XBOXKRNL_EXPORT1(KeInitializeEvent, kThreading, kImplemented);
 
-uint32_t keSetEvent(X_KEVENT* event_ptr, uint32_t increment, uint32_t wait) {
+uint32_t xeKeSetEvent(X_KEVENT* event_ptr, uint32_t increment, uint32_t wait) {
   auto ev = XObject::GetNativeObject<XEvent>(kernel_state(), event_ptr);
   if (!ev) {
     assert_always();
@@ -385,7 +385,7 @@ uint32_t keSetEvent(X_KEVENT* event_ptr, uint32_t increment, uint32_t wait) {
 
 dword_result_t KeSetEvent(pointer_t<X_KEVENT> event_ptr, dword_t increment,
                           dword_t wait) {
-  return keSetEvent(event_ptr, increment, wait);
+  return xeKeSetEvent(event_ptr, increment, wait);
 }
 DECLARE_XBOXKRNL_EXPORT2(KeSetEvent, kThreading, kImplemented, kHighFrequency);
 
@@ -446,7 +446,7 @@ dword_result_t NtCreateEvent(lpdword_t handle_ptr,
 }
 DECLARE_XBOXKRNL_EXPORT1(NtCreateEvent, kThreading, kImplemented);
 
-dword_result_t NtSetEvent(dword_t handle, lpdword_t previous_state_ptr) {
+uint32_t xeNtSetEvent(uint32_t handle, xe::be<uint32_t>* previous_state_ptr) {
   X_STATUS result = X_STATUS_SUCCESS;
 
   auto ev = kernel_state()->object_table()->LookupObject<XEvent>(handle);
@@ -461,6 +461,10 @@ dword_result_t NtSetEvent(dword_t handle, lpdword_t previous_state_ptr) {
 
   return result;
 }
+
+dword_result_t NtSetEvent(dword_t handle, lpdword_t previous_state_ptr) {
+  return xeNtSetEvent(handle, previous_state_ptr);
+}
 DECLARE_XBOXKRNL_EXPORT2(NtSetEvent, kThreading, kImplemented, kHighFrequency);
 
 dword_result_t NtPulseEvent(dword_t handle, lpdword_t previous_state_ptr) {
@@ -481,7 +485,7 @@ dword_result_t NtPulseEvent(dword_t handle, lpdword_t previous_state_ptr) {
 DECLARE_XBOXKRNL_EXPORT2(NtPulseEvent, kThreading, kImplemented,
                          kHighFrequency);
 
-dword_result_t NtClearEvent(dword_t handle) {
+uint32_t xeNtClearEvent(uint32_t handle) {
   X_STATUS result = X_STATUS_SUCCESS;
 
   auto ev = kernel_state()->object_table()->LookupObject<XEvent>(handle);
@@ -493,6 +497,8 @@ dword_result_t NtClearEvent(dword_t handle) {
 
   return result;
 }
+
+dword_result_t NtClearEvent(dword_t handle) { return xeNtClearEvent(handle); }
 DECLARE_XBOXKRNL_EXPORT2(NtClearEvent, kThreading, kImplemented,
                          kHighFrequency);
 
@@ -512,8 +518,8 @@ void KeInitializeSemaphore(pointer_t<X_KSEMAPHORE> semaphore_ptr, dword_t count,
 }
 DECLARE_XBOXKRNL_EXPORT1(KeInitializeSemaphore, kThreading, kImplemented);
 
-uint32_t keReleaseSemaphore(X_KSEMAPHORE* semaphore_ptr, uint32_t increment,
-                            uint32_t adjustment, uint32_t wait) {
+uint32_t xeKeReleaseSemaphore(X_KSEMAPHORE* semaphore_ptr, uint32_t increment,
+                              uint32_t adjustment, uint32_t wait) {
   auto sem =
       XObject::GetNativeObject<XSemaphore>(kernel_state(), semaphore_ptr);
   if (!sem) {
@@ -530,7 +536,7 @@ uint32_t keReleaseSemaphore(X_KSEMAPHORE* semaphore_ptr, uint32_t increment,
 dword_result_t KeReleaseSemaphore(pointer_t<X_KSEMAPHORE> semaphore_ptr,
                                   dword_t increment, dword_t adjustment,
                                   dword_t wait) {
-  return keReleaseSemaphore(semaphore_ptr, increment, adjustment, wait);
+  return xeKeReleaseSemaphore(semaphore_ptr, increment, adjustment, wait);
 }
 DECLARE_XBOXKRNL_EXPORT1(KeReleaseSemaphore, kThreading, kImplemented);
 
@@ -728,9 +734,9 @@ dword_result_t NtCancelTimer(dword_t timer_handle,
 }
 DECLARE_XBOXKRNL_EXPORT1(NtCancelTimer, kThreading, kImplemented);
 
-uint32_t keWaitForSingleObject(void* object_ptr, uint32_t wait_reason,
-                               uint32_t processor_mode, uint32_t alertable,
-                               uint64_t* timeout) {
+uint32_t xeKeWaitForSingleObject(void* object_ptr, uint32_t wait_reason,
+                                 uint32_t processor_mode, uint32_t alertable,
+                                 uint64_t* timeout) {
   auto object = XObject::GetNativeObject<XObject>(kernel_state(), object_ptr);
 
   if (!object) {
@@ -749,8 +755,8 @@ dword_result_t KeWaitForSingleObject(lpvoid_t object_ptr, dword_t wait_reason,
                                      dword_t processor_mode, dword_t alertable,
                                      lpqword_t timeout_ptr) {
   uint64_t timeout = timeout_ptr ? static_cast<uint64_t>(*timeout_ptr) : 0u;
-  return keWaitForSingleObject(object_ptr, wait_reason, processor_mode,
-                               alertable, &timeout);
+  return xeKeWaitForSingleObject(object_ptr, wait_reason, processor_mode,
+                                 alertable, &timeout);
 }
 DECLARE_XBOXKRNL_EXPORT3(KeWaitForSingleObject, kThreading, kImplemented,
                          kBlocking, kHighFrequency);
@@ -808,10 +814,10 @@ dword_result_t KeWaitForMultipleObjects(dword_t count, lpdword_t objects_ptr,
 DECLARE_XBOXKRNL_EXPORT3(KeWaitForMultipleObjects, kThreading, kImplemented,
                          kBlocking, kHighFrequency);
 
-dword_result_t NtWaitForMultipleObjectsEx(dword_t count, lpdword_t handles,
-                                          dword_t wait_type, dword_t wait_mode,
-                                          dword_t alertable,
-                                          lpqword_t timeout_ptr) {
+uint32_t xeNtWaitForMultipleObjectsEx(uint32_t count, xe::be<uint32_t>* handles,
+                                      uint32_t wait_type, uint32_t wait_mode,
+                                      uint32_t alertable,
+                                      uint64_t* timeout_ptr) {
   assert_true(wait_type <= 1);
   X_STATUS result = X_STATUS_SUCCESS;
 
@@ -826,13 +832,21 @@ dword_result_t NtWaitForMultipleObjectsEx(dword_t count, lpdword_t handles,
     objects.push_back(std::move(object));
   }
 
-  uint64_t timeout = timeout_ptr ? static_cast<uint64_t>(*timeout_ptr) : 0u;
-  result = XObject::WaitMultiple(
-      count, reinterpret_cast<XObject**>(objects.data()), wait_type, 6,
-      wait_mode, alertable, timeout_ptr ? &timeout : nullptr);
+  result =
+      XObject::WaitMultiple(count, reinterpret_cast<XObject**>(objects.data()),
+                            wait_type, 6, wait_mode, alertable, timeout_ptr);
 
   return result;
 }
+
+dword_result_t NtWaitForMultipleObjectsEx(dword_t count, lpdword_t handles,
+                                          dword_t wait_type, dword_t wait_mode,
+                                          dword_t alertable,
+                                          lpqword_t timeout_ptr) {
+  uint64_t timeout = timeout_ptr ? static_cast<uint64_t>(*timeout_ptr) : 0u;
+  return xeNtWaitForMultipleObjectsEx(count, handles, wait_type, wait_mode,
+                                      alertable, &timeout);
+}
 DECLARE_XBOXKRNL_EXPORT3(NtWaitForMultipleObjectsEx, kThreading, kImplemented,
                          kBlocking, kHighFrequency);
 
@@ -860,7 +874,7 @@ dword_result_t NtSignalAndWaitForSingleObjectEx(dword_t signal_handle,
 DECLARE_XBOXKRNL_EXPORT3(NtSignalAndWaitForSingleObjectEx, kThreading,
                          kImplemented, kBlocking, kHighFrequency);
 
-uint32_t keKfAcquireSpinLock(uint32_t* lock) {
+uint32_t xeKeKfAcquireSpinLock(uint32_t* lock) {
   // XELOGD(
   //     "KfAcquireSpinLock(%.8X)",
   //     lock_ptr);
@@ -881,12 +895,12 @@ uint32_t keKfAcquireSpinLock(uint32_t* lock) {
 
 dword_result_t KfAcquireSpinLock(lpdword_t lock_ptr) {
   auto lock = reinterpret_cast<uint32_t*>(lock_ptr.host_address());
-  return keKfAcquireSpinLock(lock);
+  return xeKeKfAcquireSpinLock(lock);
 }
 DECLARE_XBOXKRNL_EXPORT3(KfAcquireSpinLock, kThreading, kImplemented, kBlocking,
                          kHighFrequency);
 
-void keKfReleaseSpinLock(uint32_t* lock, dword_t old_irql) {
+void xeKeKfReleaseSpinLock(uint32_t* lock, dword_t old_irql) {
   // Restore IRQL.
   XThread* thread = XThread::GetCurrentThread();
   thread->LowerIrql(old_irql);
@@ -897,7 +911,7 @@ void keKfReleaseSpinLock(uint32_t* lock, dword_t old_irql) {
 
 void KfReleaseSpinLock(lpdword_t lock_ptr, dword_t old_irql) {
   auto lock = reinterpret_cast<uint32_t*>(lock_ptr.host_address());
-  keKfReleaseSpinLock(lock, old_irql);
+  xeKeKfReleaseSpinLock(lock, old_irql);
 }
 DECLARE_XBOXKRNL_EXPORT2(KfReleaseSpinLock, kThreading, kImplemented,
                          kHighFrequency);
@@ -1130,26 +1144,26 @@ void ExInitializeReadWriteLock(pointer_t<X_ERWLOCK> lock_ptr) {
 DECLARE_XBOXKRNL_EXPORT1(ExInitializeReadWriteLock, kThreading, kImplemented);
 
 void ExAcquireReadWriteLockExclusive(pointer_t<X_ERWLOCK> lock_ptr) {
-  auto old_irql = keKfAcquireSpinLock(&lock_ptr->spin_lock);
+  auto old_irql = xeKeKfAcquireSpinLock(&lock_ptr->spin_lock);
   lock_ptr->lock_count++;
-  keKfReleaseSpinLock(&lock_ptr->spin_lock, old_irql);
+  xeKeKfReleaseSpinLock(&lock_ptr->spin_lock, old_irql);
 
   if (!lock_ptr->lock_count) {
     return;
   }
 
   lock_ptr->writers_waiting_count++;
-  keWaitForSingleObject(&lock_ptr->writer_event, 0, 0, 0, nullptr);
+  xeKeWaitForSingleObject(&lock_ptr->writer_event, 0, 0, 0, nullptr);
 }
 DECLARE_XBOXKRNL_EXPORT4(ExAcquireReadWriteLockExclusive, kThreading,
                          kImplemented, kBlocking, kHighFrequency, kSketchy);
 
 void ExReleaseReadWriteLock(pointer_t<X_ERWLOCK> lock_ptr) {
-  auto old_irql = keKfAcquireSpinLock(&lock_ptr->spin_lock);
+  auto old_irql = xeKeKfAcquireSpinLock(&lock_ptr->spin_lock);
   lock_ptr->lock_count--;
 
   if (lock_ptr->lock_count < 0) {
-    keKfReleaseSpinLock(&lock_ptr->spin_lock, old_irql);
+    xeKeKfReleaseSpinLock(&lock_ptr->spin_lock, old_irql);
     return;
   }
 
@@ -1158,17 +1172,17 @@ void ExReleaseReadWriteLock(pointer_t<X_ERWLOCK> lock_ptr) {
     if (readers_waiting_count) {
       lock_ptr->readers_waiting_count = 0;
       lock_ptr->readers_entry_count = readers_waiting_count;
-      keKfReleaseSpinLock(&lock_ptr->spin_lock, old_irql);
-      keReleaseSemaphore(&lock_ptr->reader_semaphore, 1, readers_waiting_count,
-                         0);
+      xeKeKfReleaseSpinLock(&lock_ptr->spin_lock, old_irql);
+      xeKeReleaseSemaphore(&lock_ptr->reader_semaphore, 1,
+                           readers_waiting_count, 0);
       return;
     }
   }
 
   auto count = lock_ptr->readers_entry_count--;
-  keKfReleaseSpinLock(&lock_ptr->spin_lock, old_irql);
+  xeKeKfReleaseSpinLock(&lock_ptr->spin_lock, old_irql);
   if (!count) {
-    keSetEvent(&lock_ptr->writer_event, 1, 0);
+    xeKeSetEvent(&lock_ptr->writer_event, 1, 0);
   }
 }
 DECLARE_XBOXKRNL_EXPORT2(ExReleaseReadWriteLock, kThreading, kImplemented,
diff --git a/src/xenia/kernel/xboxkrnl/xboxkrnl_threading.h b/src/xenia/kernel/xboxkrnl/xboxkrnl_threading.h
index f69f767e2..d405d4ce2 100644
--- a/src/xenia/kernel/xboxkrnl/xboxkrnl_threading.h
+++ b/src/xenia/kernel/xboxkrnl/xboxkrnl_threading.h
@@ -19,19 +19,18 @@ struct X_KEVENT;
 
 namespace xboxkrnl {
 
-dword_result_t NtSetEvent(dword_t handle, lpdword_t previous_state_ptr);
-dword_result_t NtClearEvent(dword_t handle);
+uint32_t xeNtSetEvent(uint32_t handle, xe::be<uint32_t>* previous_state_ptr);
+uint32_t xeNtClearEvent(uint32_t handle);
 
-dword_result_t NtWaitForMultipleObjectsEx(dword_t count, lpdword_t handles,
-                                          dword_t wait_type, dword_t wait_mode,
-                                          dword_t alertable,
-                                          lpqword_t timeout_ptr);
+uint32_t xeNtWaitForMultipleObjectsEx(uint32_t count, xe::be<uint32_t>* handles,
+                                      uint32_t wait_type, uint32_t wait_mode,
+                                      uint32_t alertable,
+                                      uint64_t* timeout_ptr);
 
-dword_result_t KeWaitForSingleObject(lpvoid_t object_ptr, dword_t wait_reason,
-                                     dword_t processor_mode, dword_t alertable,
-                                     lpqword_t timeout_ptr);
-dword_result_t KeSetEvent(pointer_t<X_KEVENT> event_ptr, dword_t increment,
-                          dword_t wait);
+uint32_t xeKeWaitForSingleObject(void* object_ptr, uint32_t wait_reason,
+                                 uint32_t processor_mode, uint32_t alertable,
+                                 uint64_t* timeout_ptr);
+uint32_t xeKeSetEvent(X_KEVENT* event_ptr, uint32_t increment, uint32_t wait);
 
 }  // namespace xboxkrnl
 }  // namespace kernel