VirtualBox

source: vbox/trunk/src/VBox/Runtime/r3/win/thread-win.cpp@ 68033

Last change on this file since 68033 was 66120, checked in by vboxsync, 8 years ago

IPRT/RTThreadWait: Quick and dirty workaround for Ctrl-C deadlock with VirtualBox.exe.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id Revision
File size: 12.3 KB
Line 
1/* $Id: thread-win.cpp 66120 2017-03-15 19:59:48Z vboxsync $ */
2/** @file
3 * IPRT - Threads, Windows.
4 */
5
6/*
7 * Copyright (C) 2006-2016 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 *
17 * The contents of this file may alternatively be used under the terms
18 * of the Common Development and Distribution License Version 1.0
19 * (CDDL) only, as it comes in the "COPYING.CDDL" file of the
20 * VirtualBox OSE distribution, in which case the provisions of the
21 * CDDL are applicable instead of those of the GPL.
22 *
23 * You may elect to license modified versions of this file under the
24 * terms and conditions of either the GPL or the CDDL or both.
25 */
26
27
28/*********************************************************************************************************************************
29* Header Files *
30*********************************************************************************************************************************/
31#define LOG_GROUP RTLOGGROUP_THREAD
32#include <iprt/nt/nt-and-windows.h>
33
34#include <errno.h>
35#include <process.h>
36
37#include <iprt/thread.h>
38#include "internal/iprt.h"
39
40#include <iprt/asm-amd64-x86.h>
41#include <iprt/assert.h>
42#include <iprt/cpuset.h>
43#include <iprt/err.h>
44#include <iprt/log.h>
45#include <iprt/mem.h>
46#include "internal/thread.h"
47
48
49/*********************************************************************************************************************************
50* Defined Constants And Macros *
51*********************************************************************************************************************************/
52/** The TLS index allocated for storing the RTTHREADINT pointer. */
53static DWORD g_dwSelfTLS = TLS_OUT_OF_INDEXES;
54
55
56/*********************************************************************************************************************************
57* Internal Functions *
58*********************************************************************************************************************************/
59static unsigned __stdcall rtThreadNativeMain(void *pvArgs);
60static void rtThreadWinTellDebuggerThreadName(uint32_t idThread, const char *pszName);
61
62
63DECLHIDDEN(int) rtThreadNativeInit(void)
64{
65 g_dwSelfTLS = TlsAlloc();
66 if (g_dwSelfTLS == TLS_OUT_OF_INDEXES)
67 return VERR_NO_TLS_FOR_SELF;
68 return VINF_SUCCESS;
69}
70
71
72DECLHIDDEN(void) rtThreadNativeReInitObtrusive(void)
73{
74 /* nothing to do here. */
75}
76
77
78DECLHIDDEN(void) rtThreadNativeDetach(void)
79{
80 /*
81 * Deal with alien threads.
82 */
83 PRTTHREADINT pThread = (PRTTHREADINT)TlsGetValue(g_dwSelfTLS);
84 if ( pThread
85 && (pThread->fIntFlags & RTTHREADINT_FLAGS_ALIEN))
86 {
87 rtThreadTerminate(pThread, 0);
88 TlsSetValue(g_dwSelfTLS, NULL);
89 }
90}
91
92
93DECLHIDDEN(void) rtThreadNativeDestroy(PRTTHREADINT pThread)
94{
95 if (pThread == (PRTTHREADINT)TlsGetValue(g_dwSelfTLS))
96 TlsSetValue(g_dwSelfTLS, NULL);
97
98 if ((HANDLE)pThread->hThread != INVALID_HANDLE_VALUE)
99 {
100 CloseHandle((HANDLE)pThread->hThread);
101 pThread->hThread = (uintptr_t)INVALID_HANDLE_VALUE;
102 }
103}
104
105
106DECLHIDDEN(int) rtThreadNativeAdopt(PRTTHREADINT pThread)
107{
108 if (!TlsSetValue(g_dwSelfTLS, pThread))
109 return VERR_FAILED_TO_SET_SELF_TLS;
110 if (IsDebuggerPresent())
111 rtThreadWinTellDebuggerThreadName(GetCurrentThreadId(), pThread->szName);
112 return VINF_SUCCESS;
113}
114
115
116DECLHIDDEN(void) rtThreadNativeInformDebugger(PRTTHREADINT pThread)
117{
118 rtThreadWinTellDebuggerThreadName((uint32_t)(uintptr_t)pThread->Core.Key, pThread->szName);
119}
120
121
122/**
123 * Communicates the thread name to the debugger, if we're begin debugged that
124 * is.
125 *
126 * See http://msdn.microsoft.com/en-us/library/xcb2z8hs.aspx for debugger
127 * interface details.
128 *
129 * @param idThread The thread ID. UINT32_MAX for current thread.
130 * @param pszName The name.
131 */
132static void rtThreadWinTellDebuggerThreadName(uint32_t idThread, const char *pszName)
133{
134 struct
135 {
136 uint32_t uType;
137 const char *pszName;
138 uint32_t idThread;
139 uint32_t fFlags;
140 } Pkg = { 0x1000, pszName, idThread, 0 };
141 __try
142 {
143 RaiseException(0x406d1388, 0, sizeof(Pkg)/sizeof(ULONG_PTR), (ULONG_PTR *)&Pkg);
144 }
145 __except(EXCEPTION_CONTINUE_EXECUTION)
146 {
147
148 }
149}
150
151
152/**
153 * Bitch about dangling COM and OLE references, dispose of them
154 * afterwards so we don't end up deadlocked somewhere below
155 * OLE32!DllMain.
156 */
157static void rtThreadNativeUninitComAndOle(void)
158{
159#if 1 /* experimental code */
160 /*
161 * Read the counters.
162 */
163 struct MySOleTlsData
164 {
165 void *apvReserved0[2]; /**< x86=0x00 W7/64=0x00 */
166 DWORD adwReserved0[3]; /**< x86=0x08 W7/64=0x10 */
167 void *apvReserved1[1]; /**< x86=0x14 W7/64=0x20 */
168 DWORD cComInits; /**< x86=0x18 W7/64=0x28 */
169 DWORD cOleInits; /**< x86=0x1c W7/64=0x2c */
170 DWORD dwReserved1; /**< x86=0x20 W7/64=0x30 */
171 void *apvReserved2[4]; /**< x86=0x24 W7/64=0x38 */
172 DWORD adwReserved2[1]; /**< x86=0x34 W7/64=0x58 */
173 void *pvCurrentCtx; /**< x86=0x38 W7/64=0x60 */
174 IUnknown *pCallState; /**< x86=0x3c W7/64=0x68 */
175 } *pOleTlsData = NULL; /* outside the try/except for debugging */
176 DWORD cComInits = 0;
177 DWORD cOleInits = 0;
178 __try
179 {
180 void *pvTeb = NtCurrentTeb();
181# ifdef RT_ARCH_AMD64
182 pOleTlsData = *(struct MySOleTlsData **)((uintptr_t)pvTeb + 0x1758); /*TEB.ReservedForOle*/
183# elif RT_ARCH_X86
184 pOleTlsData = *(struct MySOleTlsData **)((uintptr_t)pvTeb + 0x0f80); /*TEB.ReservedForOle*/
185# else
186# error "Port me!"
187# endif
188 if (pOleTlsData)
189 {
190 cComInits = pOleTlsData->cComInits;
191 cOleInits = pOleTlsData->cOleInits;
192 }
193 }
194 __except(EXCEPTION_EXECUTE_HANDLER)
195 {
196 AssertFailedReturnVoid();
197 }
198
199 /*
200 * Assert sanity. If any of these breaks, the structure layout above is
201 * probably not correct any longer.
202 */
203 AssertMsgReturnVoid(cComInits < 1000, ("%u (%#x)\n", cComInits, cComInits));
204 AssertMsgReturnVoid(cOleInits < 1000, ("%u (%#x)\n", cOleInits, cOleInits));
205 AssertMsgReturnVoid(cComInits >= cOleInits, ("cComInits=%#x cOleInits=%#x\n", cComInits, cOleInits));
206
207 /*
208 * Do the uninitializing.
209 */
210 if (cComInits)
211 {
212 AssertMsgFailed(("cComInits=%u (%#x) cOleInits=%u (%#x) - dangling COM/OLE inits!\n",
213 cComInits, cComInits, cOleInits, cOleInits));
214
215 HMODULE hOle32 = GetModuleHandle("ole32.dll");
216 AssertReturnVoid(hOle32 != NULL);
217
218 typedef void (WINAPI *PFNOLEUNINITIALIZE)(void);
219 PFNOLEUNINITIALIZE pfnOleUninitialize = (PFNOLEUNINITIALIZE)GetProcAddress(hOle32, "OleUninitialize");
220 AssertReturnVoid(pfnOleUninitialize);
221
222 typedef void (WINAPI *PFNCOUNINITIALIZE)(void);
223 PFNCOUNINITIALIZE pfnCoUninitialize = (PFNCOUNINITIALIZE)GetProcAddress(hOle32, "CoUninitialize");
224 AssertReturnVoid(pfnCoUninitialize);
225
226 while (cOleInits-- > 0)
227 {
228 pfnOleUninitialize();
229 cComInits--;
230 }
231
232 while (cComInits-- > 0)
233 pfnCoUninitialize();
234 }
235#endif
236}
237
238
239/**
240 * Wrapper which unpacks the param stuff and calls thread function.
241 */
242static unsigned __stdcall rtThreadNativeMain(void *pvArgs)
243{
244 DWORD dwThreadId = GetCurrentThreadId();
245 PRTTHREADINT pThread = (PRTTHREADINT)pvArgs;
246
247 if (!TlsSetValue(g_dwSelfTLS, pThread))
248 AssertReleaseMsgFailed(("failed to set self TLS. lasterr=%d thread '%s'\n", GetLastError(), pThread->szName));
249 if (IsDebuggerPresent())
250 rtThreadWinTellDebuggerThreadName(dwThreadId, &pThread->szName[0]);
251
252 int rc = rtThreadMain(pThread, dwThreadId, &pThread->szName[0]);
253
254 TlsSetValue(g_dwSelfTLS, NULL);
255 rtThreadNativeUninitComAndOle();
256 _endthreadex(rc);
257 return rc;
258}
259
260
261DECLHIDDEN(int) rtThreadNativeCreate(PRTTHREADINT pThread, PRTNATIVETHREAD pNativeThread)
262{
263 AssertReturn(pThread->cbStack < ~(unsigned)0, VERR_INVALID_PARAMETER);
264
265 /*
266 * Create the thread.
267 */
268 pThread->hThread = (uintptr_t)INVALID_HANDLE_VALUE;
269 unsigned uThreadId = 0;
270 uintptr_t hThread = _beginthreadex(NULL, (unsigned)pThread->cbStack, rtThreadNativeMain, pThread, 0, &uThreadId);
271 if (hThread != 0 && hThread != ~0U)
272 {
273 pThread->hThread = hThread;
274 *pNativeThread = uThreadId;
275 return VINF_SUCCESS;
276 }
277 return RTErrConvertFromErrno(errno);
278}
279
280
281DECLHIDDEN(bool) rtThreadNativeIsAliveKludge(PRTTHREADINT pThread)
282{
283 PPEB_COMMON pPeb = NtCurrentPeb();
284 if (!pPeb || !pPeb->Ldr || !pPeb->Ldr->ShutdownInProgress)
285 return true;
286 DWORD rcWait = WaitForSingleObject((HANDLE)pThread->hThread, 0);
287 return rcWait != WAIT_OBJECT_0;
288}
289
290
291RTDECL(RTTHREAD) RTThreadSelf(void)
292{
293 PRTTHREADINT pThread = (PRTTHREADINT)TlsGetValue(g_dwSelfTLS);
294 /** @todo import alien threads ? */
295 return pThread;
296}
297
298
299#if 0 /* noone is using this ... */
300/**
301 * Returns the processor number the current thread was running on during this call
302 *
303 * @returns processor nr
304 */
305static int rtThreadGetCurrentProcessorNumber(void)
306{
307 static bool fInitialized = false;
308 static DWORD (WINAPI *pfnGetCurrentProcessorNumber)(void) = NULL;
309 if (!fInitialized)
310 {
311 HMODULE hmodKernel32 = GetModuleHandle("kernel32.dll");
312 if (hmodKernel32)
313 pfnGetCurrentProcessorNumber = (DWORD (WINAPI*)(void))GetProcAddress(hmodKernel32, "GetCurrentProcessorNumber");
314 fInitialized = true;
315 }
316 if (pfnGetCurrentProcessorNumber)
317 return pfnGetCurrentProcessorNumber();
318 return -1;
319}
320#endif
321
322
323RTR3DECL(int) RTThreadSetAffinity(PCRTCPUSET pCpuSet)
324{
325 DWORD_PTR fNewMask = pCpuSet ? RTCpuSetToU64(pCpuSet) : ~(DWORD_PTR)0;
326 DWORD_PTR dwRet = SetThreadAffinityMask(GetCurrentThread(), fNewMask);
327 if (dwRet)
328 return VINF_SUCCESS;
329
330 int iLastError = GetLastError();
331 AssertMsgFailed(("SetThreadAffinityMask failed, LastError=%d\n", iLastError));
332 return RTErrConvertFromWin32(iLastError);
333}
334
335
336RTR3DECL(int) RTThreadGetAffinity(PRTCPUSET pCpuSet)
337{
338 /*
339 * Haven't found no query api, but the set api returns the old mask, so let's use that.
340 */
341 DWORD_PTR dwIgnored;
342 DWORD_PTR dwProcAff = 0;
343 if (GetProcessAffinityMask(GetCurrentProcess(), &dwProcAff, &dwIgnored))
344 {
345 HANDLE hThread = GetCurrentThread();
346 DWORD_PTR dwRet = SetThreadAffinityMask(hThread, dwProcAff);
347 if (dwRet)
348 {
349 DWORD_PTR dwSet = SetThreadAffinityMask(hThread, dwRet);
350 Assert(dwSet == dwProcAff); NOREF(dwRet);
351
352 RTCpuSetFromU64(pCpuSet, (uint64_t)dwSet);
353 return VINF_SUCCESS;
354 }
355 }
356
357 int iLastError = GetLastError();
358 AssertMsgFailed(("SetThreadAffinityMask or GetProcessAffinityMask failed, LastError=%d\n", iLastError));
359 return RTErrConvertFromWin32(iLastError);
360}
361
362
363RTR3DECL(int) RTThreadGetExecutionTimeMilli(uint64_t *pKernelTime, uint64_t *pUserTime)
364{
365 uint64_t u64CreationTime, u64ExitTime, u64KernelTime, u64UserTime;
366
367 if (GetThreadTimes(GetCurrentThread(), (LPFILETIME)&u64CreationTime, (LPFILETIME)&u64ExitTime, (LPFILETIME)&u64KernelTime, (LPFILETIME)&u64UserTime))
368 {
369 *pKernelTime = u64KernelTime / 10000; /* GetThreadTimes returns time in 100 ns units */
370 *pUserTime = u64UserTime / 10000; /* GetThreadTimes returns time in 100 ns units */
371 return VINF_SUCCESS;
372 }
373
374 int iLastError = GetLastError();
375 AssertMsgFailed(("GetThreadTimes failed, LastError=%d\n", iLastError));
376 return RTErrConvertFromWin32(iLastError);
377}
378
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette