Source code of Windows XP (NT5)
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

382 lines
9.1 KiB

  1. /*++
  2. Copyright (c) 2000 Microsoft Corporation
  3. Module Name:
  4. EmulateDirectDrawSync.cpp
  5. Abstract:
  6. DirectDraw uses per-thread exclusive mode arbitration on NT. On Win9x this
  7. is done per process. What this means is that if an app releases exclusive
  8. mode from a different thread than that which acquired it, it will be in a
  9. permanently bad state.
  10. This shim ensures that the mutex is obtained and released on the same
  11. thread. During DLL_PROCESS_ATTACH, a new thread is started: this thread
  12. manages the acquisition and release of the mutex.
  13. Note we can't get the mutex by catching CreateMutex because it's called
  14. from the dllmain of ddraw.dll: so it wouldn't work on win2k.
  15. Notes:
  16. This is a general purpose shim.
  17. History:
  18. 09/11/2000 prashkud Created
  19. 10/28/2000 linstev Rewrote to work on win2k
  20. 02/23/2001 linstev Modified to handle cases where DirectDraw was used
  21. inside DllMains
  22. --*/
  23. #include "precomp.h"
  24. IMPLEMENT_SHIM_BEGIN(EmulateDirectDrawSync)
  25. #include "ShimHookMacro.h"
  26. APIHOOK_ENUM_BEGIN
  27. APIHOOK_ENUM_ENTRY(WaitForSingleObject)
  28. APIHOOK_ENUM_ENTRY(ReleaseMutex)
  29. APIHOOK_ENUM_ENTRY(CloseHandle)
  30. APIHOOK_ENUM_END
  31. // Enum used to tell our thread what to do
  32. enum {sNone, sWaitForSingleObject, sReleaseMutex};
  33. // Events we use to signal our thread to do work and wait until its done
  34. HANDLE g_hWaitEvent;
  35. HANDLE g_hDoneEvent;
  36. HANDLE g_hThread = NULL;
  37. //
  38. // Parameters that are passed between the caller thread and our thread
  39. // Access is synchronized with a critical section
  40. //
  41. CRITICAL_SECTION g_csSync;
  42. DWORD g_dwWait;
  43. DWORD g_dwWaitRetValue;
  44. DWORD g_dwTime;
  45. BOOL g_bRetValue;
  46. // Store the DirectDraw mutex handle
  47. HANDLE g_hDDMutex = 0;
  48. // Thread tracking data so we can identify degenerate cases
  49. DWORD g_dwMutexOwnerThreadId = 0;
  50. // Find the DirectDraw mutex
  51. DWORD g_dwFindMutexThread = 0;
  52. /*++
  53. Unfortunately we don't get in early enough on Win2k to get the mutex from the
  54. ddraw call to CreateMutex, so we have to make use of a special hack that knows
  55. about the ddraw internals.
  56. Ddraw has an export called GetOLEThunkData. The name is chosen to prevent
  57. people from calling it. It is designed to be used by the test harness. One of
  58. the things it can do, is release the exclusive mode mutex. This is the hack
  59. we're exploiting so we can determine the mutex handle.
  60. --*/
  61. BOOL
  62. FindMutex()
  63. {
  64. typedef VOID (WINAPI *_pfn_GetOLEThunkData)(ULONG_PTR dwOrdinal);
  65. HMODULE hMod;
  66. _pfn_GetOLEThunkData pfnGetOLEThunkData;
  67. hMod = GetModuleHandleA("ddraw");
  68. if (!hMod) {
  69. DPFN( eDbgLevelError, "[FindMutex] DirectDraw not loaded");
  70. return FALSE;
  71. }
  72. pfnGetOLEThunkData = (_pfn_GetOLEThunkData) GetProcAddress(hMod, "GetOLEThunkData");
  73. if (!pfnGetOLEThunkData) {
  74. DPFN( eDbgLevelError, "[FindMutex] Failed to get GetOLEThunkData API");
  75. return FALSE;
  76. }
  77. //
  78. // Now we plan to go and find the mutex by getting Ddraw to call
  79. // ReleaseMutex.
  80. //
  81. EnterCriticalSection(&g_csSync);
  82. //
  83. // Set the mutex to the current thread so it can be picked up in the
  84. // ReleaseMutex hook
  85. //
  86. g_dwFindMutexThread = GetCurrentThreadId();
  87. //
  88. // Call to the hard-coded (in ddraw) ReleaseMutex hack which releases the
  89. // mutex
  90. //
  91. pfnGetOLEThunkData(6);
  92. g_dwFindMutexThread = 0;
  93. LeaveCriticalSection(&g_csSync);
  94. return (g_hDDMutex != 0);
  95. }
  96. /*++
  97. Hook WaitForSingleObject to determine when DirectDraw is testing or acquiring
  98. the mutex. If we haven't got the mutex yet, we attempt to find it using our
  99. hack.
  100. --*/
  101. DWORD
  102. APIHOOK(WaitForSingleObject)(
  103. HANDLE hHandle,
  104. DWORD dwMilliSeconds
  105. )
  106. {
  107. if (g_hThread) {
  108. //
  109. // Hack to find the DirectDraw mutex
  110. //
  111. if (!g_hDDMutex) {
  112. FindMutex();
  113. }
  114. if (g_hDDMutex && (hHandle == g_hDDMutex)) {
  115. //
  116. // Use our thread to acquire the mutex. We synchronize since we're
  117. // accessing globals to communicate with our thread.
  118. //
  119. DWORD dwRet;
  120. EnterCriticalSection(&g_csSync);
  121. // Set globals to communicate with our thread
  122. g_dwTime = dwMilliSeconds;
  123. g_dwWait = sWaitForSingleObject;
  124. ResetEvent(g_hDoneEvent);
  125. // Signal our thread to obtain the mutex
  126. SetEvent(g_hWaitEvent);
  127. // Wait until the state of the mutex has been determined
  128. WaitForSingleObject(g_hDoneEvent, INFINITE);
  129. // Code to detect the degenerate
  130. if (g_dwWaitRetValue == WAIT_OBJECT_0) {
  131. g_dwMutexOwnerThreadId = GetCurrentThreadId();
  132. }
  133. dwRet = g_dwWaitRetValue;
  134. LeaveCriticalSection(&g_csSync);
  135. return dwRet;
  136. }
  137. }
  138. return ORIGINAL_API(WaitForSingleObject)(hHandle, dwMilliSeconds);
  139. }
  140. /*++
  141. Hook ReleaseMutex and release the mutex on our thread.
  142. --*/
  143. BOOL
  144. APIHOOK(ReleaseMutex)(
  145. HANDLE hMutex
  146. )
  147. {
  148. if (g_hThread && (g_dwFindMutexThread == GetCurrentThreadId())) {
  149. //
  150. // We're using our hack to find the DirectDraw mutex
  151. //
  152. DPFN( eDbgLevelInfo, "DDraw exclusive mode mutex found");
  153. g_hDDMutex = hMutex;
  154. // Don't release it, since we never acquired it
  155. return TRUE;
  156. }
  157. //
  158. // First try to release it on the current thread. This will only succeed if
  159. // it was obtained on this thread.
  160. //
  161. BOOL bRet = ORIGINAL_API(ReleaseMutex)(hMutex);
  162. if (!bRet && g_hThread && g_hDDMutex && (hMutex == g_hDDMutex)) {
  163. //
  164. // Use our thread to release the mutex. We synchronize since we're
  165. // accessing globals to communicate with our thread.
  166. //
  167. EnterCriticalSection(&g_csSync);
  168. // Set globals to communicate with our thread
  169. g_dwWait = sReleaseMutex;
  170. ResetEvent(g_hDoneEvent);
  171. // Wait until our thread returns
  172. SetEvent(g_hWaitEvent);
  173. // Signal our thread to release the mutex
  174. WaitForSingleObject(g_hDoneEvent, INFINITE);
  175. // Detect degenerate case
  176. if (GetCurrentThreadId() != g_dwMutexOwnerThreadId) {
  177. LOGN( eDbgLevelError, "[ReleaseMutex] DirectDraw synchronization error - correcting");
  178. }
  179. if (g_bRetValue) {
  180. g_dwMutexOwnerThreadId = 0;
  181. }
  182. bRet = g_bRetValue;
  183. LeaveCriticalSection(&g_csSync);
  184. }
  185. return bRet;
  186. }
  187. /*++
  188. Clear our handle in case the app frees ddraw and reloads it.
  189. --*/
  190. BOOL
  191. APIHOOK(CloseHandle)(HANDLE hObject)
  192. {
  193. if (g_hThread && (hObject == g_hDDMutex))
  194. {
  195. DPFN( eDbgLevelInfo, "DDraw exclusive mode mutex closed");
  196. g_hDDMutex = 0;
  197. }
  198. return ORIGINAL_API(CloseHandle)(hObject);
  199. }
  200. /*++
  201. Thread used to do all the mutex operations so we can guarantee that the thread
  202. that acquired the mutex is the same one that releases it.
  203. --*/
  204. VOID
  205. WINAPI
  206. ThreadSyncMutex(
  207. LPVOID lpParameter
  208. )
  209. {
  210. while (1) {
  211. // Wait until we need to acquire or release the mutex
  212. WaitForSingleObject(g_hWaitEvent, INFINITE);
  213. if (g_dwWait == sWaitForSingleObject) {
  214. // WaitForSingleobject() has been called on the Mutex object
  215. g_dwWaitRetValue = ORIGINAL_API(WaitForSingleObject)(
  216. g_hDDMutex, g_dwTime);
  217. }
  218. else if (g_dwWait == sReleaseMutex) {
  219. // ReleaseMutex has been called
  220. g_bRetValue = ORIGINAL_API(ReleaseMutex)(g_hDDMutex);
  221. }
  222. g_dwWait = sNone;
  223. ResetEvent(g_hWaitEvent);
  224. SetEvent(g_hDoneEvent);
  225. }
  226. }
  227. /*++
  228. Register hooked functions
  229. --*/
  230. BOOL
  231. NOTIFY_FUNCTION(
  232. DWORD fdwReason
  233. )
  234. {
  235. if (fdwReason == SHIM_STATIC_DLLS_INITIALIZED) {
  236. //
  237. // We need the critical section all the time
  238. //
  239. InitializeCriticalSection(&g_csSync);
  240. //
  241. // Create Events that will be used for the thread synchronization, i.e
  242. // to synchronize this thread and the one we will be creating ahead. We
  243. // don't clean these up by design. We have to do this stuff here, rather
  244. // than in the process attach, since OpenGL apps and others do DirectX
  245. // stuff in their dllmains.
  246. //
  247. g_hWaitEvent = CreateEvent(NULL, TRUE, FALSE, NULL);
  248. if (!g_hWaitEvent) {
  249. DPFN( eDbgLevelInfo, "Failed to create Event 1");
  250. return FALSE;
  251. }
  252. g_hDoneEvent = CreateEvent(NULL, TRUE, FALSE, NULL);
  253. if (!g_hDoneEvent) {
  254. DPFN( eDbgLevelInfo, "Failed to create Event 2");
  255. return FALSE;
  256. }
  257. // Create our thread
  258. g_hThread = CreateThread(NULL, 0,
  259. (LPTHREAD_START_ROUTINE) ThreadSyncMutex, NULL, 0,
  260. NULL);
  261. if (!g_hThread) {
  262. DPFN( eDbgLevelInfo, "Failed to create Thread");
  263. return FALSE;
  264. }
  265. }
  266. return TRUE;
  267. }
  268. HOOK_BEGIN
  269. CALL_NOTIFY_FUNCTION
  270. APIHOOK_ENTRY(KERNEL32.DLL, WaitForSingleObject)
  271. APIHOOK_ENTRY(KERNEL32.DLL, ReleaseMutex)
  272. APIHOOK_ENTRY(KERNEL32.DLL, CloseHandle)
  273. HOOK_END
  274. IMPLEMENT_SHIM_END