VirtualBox

source: vbox/trunk/src/VBox/Runtime/common/misc/thread.cpp@ 54358

Last change on this file since 54358 was 54358, checked in by vboxsync, 10 years ago

IPRT/R0Drv: Fix (windows & solaris) / kludge (the rest) for thread termination racing IPRT termination race in ring-0.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id Revision
File size: 47.9 KB
Line 
1/* $Id: thread.cpp 54358 2015-02-22 23:29:25Z vboxsync $ */
2/** @file
3 * IPRT - Threads, common routines.
4 */
5
6/*
7 * Copyright (C) 2006-2012 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/thread.h>
33#include "internal/iprt.h"
34
35#include <iprt/log.h>
36#include <iprt/avl.h>
37#include <iprt/alloc.h>
38#include <iprt/assert.h>
39#include <iprt/lockvalidator.h>
40#include <iprt/semaphore.h>
41#ifdef IN_RING0
42# include <iprt/spinlock.h>
43#endif
44#include <iprt/asm.h>
45#include <iprt/err.h>
46#include <iprt/string.h>
47#include "internal/magics.h"
48#include "internal/thread.h"
49#include "internal/sched.h"
50#include "internal/process.h"
51#ifdef RT_WITH_ICONV_CACHE
52# include "internal/string.h"
53#endif
54
55
56/*******************************************************************************
57* Defined Constants And Macros *
58*******************************************************************************/
59#ifdef IN_RING0
60# define RT_THREAD_LOCK_RW() RTSpinlockAcquire(g_ThreadSpinlock)
61# define RT_THREAD_UNLOCK_RW() RTSpinlockRelease(g_ThreadSpinlock)
62# define RT_THREAD_LOCK_RD() RTSpinlockAcquire(g_ThreadSpinlock)
63# define RT_THREAD_UNLOCK_RD() RTSpinlockRelease(g_ThreadSpinlock)
64#else
65# define RT_THREAD_LOCK_RW() rtThreadLockRW()
66# define RT_THREAD_UNLOCK_RW() rtThreadUnLockRW()
67# define RT_THREAD_LOCK_RD() rtThreadLockRD()
68# define RT_THREAD_UNLOCK_RD() rtThreadUnLockRD()
69#endif
70
71
72/*******************************************************************************
73* Global Variables *
74*******************************************************************************/
75/** The AVL thread containing the threads. */
76static PAVLPVNODECORE g_ThreadTree;
77/** The number of threads in the tree (for ring-0 termination kludge). */
78static uint32_t volatile g_cThreadInTree;
79#ifdef IN_RING3
80/** The RW lock protecting the tree. */
81static RTSEMRW g_ThreadRWSem = NIL_RTSEMRW;
82#else
83/** The spinlocks protecting the tree. */
84static RTSPINLOCK g_ThreadSpinlock = NIL_RTSPINLOCK;
85#endif
86/** Indicates whether we've been initialized or not. */
87static bool g_frtThreadInitialized;
88
89
90/*******************************************************************************
91* Internal Functions *
92*******************************************************************************/
93static void rtThreadDestroy(PRTTHREADINT pThread);
94#ifdef IN_RING3
95static int rtThreadAdopt(RTTHREADTYPE enmType, unsigned fFlags, uint32_t fIntFlags, const char *pszName);
96#endif
97static void rtThreadRemoveLocked(PRTTHREADINT pThread);
98static PRTTHREADINT rtThreadAlloc(RTTHREADTYPE enmType, unsigned fFlags, uint32_t fIntFlags, const char *pszName);
99
100
101/** @page pg_rt_thread IPRT Thread Internals
102 *
103 * IPRT provides interface to whatever native threading that the host provides,
104 * preferably using a CRT level interface to better integrate with other libraries.
105 *
106 * Internally IPRT keeps track of threads by means of the RTTHREADINT structure.
107 * All the RTTHREADINT structures are kept in a AVL tree which is protected by a
108 * read/write lock for efficient access. A thread is inserted into the tree in
109 * three places in the code. The main thread is 'adopted' by IPRT on rtR3Init()
110 * by rtThreadAdopt(). When creating a new thread there the child and the parent
111 * race inserting the thread, this is rtThreadMain() and RTThreadCreate.
112 *
113 * RTTHREADINT objects are using reference counting as a mean of sticking around
114 * till no-one needs them any longer. Waitable threads is created with one extra
115 * reference so they won't go away until they are waited on. This introduces a
116 * major problem if we use the host thread identifier as key in the AVL tree - the
117 * host may reuse the thread identifier before the thread was waited on. So, on
118 * most platforms we are using the RTTHREADINT pointer as key and not the
119 * thread id. RTThreadSelf() then have to be implemented using a pointer stored
120 * in thread local storage (TLS).
121 *
122 * In Ring-0 we only try keep track of kernel threads created by RTThreadCreate
123 * at the moment. There we really only need the 'join' feature, but doing things
124 * the same way allow us to name threads and similar stuff.
125 */
126
127
128/**
129 * Initializes the thread database.
130 *
131 * @returns iprt status code.
132 */
133DECLHIDDEN(int) rtThreadInit(void)
134{
135#ifdef IN_RING3
136 int rc = VINF_ALREADY_INITIALIZED;
137 if (g_ThreadRWSem == NIL_RTSEMRW)
138 {
139 /*
140 * We assume the caller is the 1st thread, which we'll call 'main'.
141 * But first, we'll create the semaphore.
142 */
143 rc = RTSemRWCreateEx(&g_ThreadRWSem, RTSEMRW_FLAGS_NO_LOCK_VAL, NIL_RTLOCKVALCLASS, RTLOCKVAL_SUB_CLASS_NONE, NULL);
144 if (RT_SUCCESS(rc))
145 {
146 rc = rtThreadNativeInit();
147 if (RT_SUCCESS(rc))
148 rc = rtThreadAdopt(RTTHREADTYPE_DEFAULT, 0, RTTHREADINT_FLAGS_MAIN, "main");
149 if (RT_SUCCESS(rc))
150 rc = rtSchedNativeCalcDefaultPriority(RTTHREADTYPE_DEFAULT);
151 if (RT_SUCCESS(rc))
152 {
153 g_frtThreadInitialized = true;
154 return VINF_SUCCESS;
155 }
156
157 /* failed, clear out */
158 RTSemRWDestroy(g_ThreadRWSem);
159 g_ThreadRWSem = NIL_RTSEMRW;
160 }
161 }
162
163#elif defined(IN_RING0)
164 int rc;
165 /*
166 * Create the spinlock and to native init.
167 */
168 Assert(g_ThreadSpinlock == NIL_RTSPINLOCK);
169 rc = RTSpinlockCreate(&g_ThreadSpinlock, RTSPINLOCK_FLAGS_INTERRUPT_SAFE, "RTThread");
170 if (RT_SUCCESS(rc))
171 {
172 rc = rtThreadNativeInit();
173 if (RT_SUCCESS(rc))
174 {
175 g_frtThreadInitialized = true;
176 return VINF_SUCCESS;
177 }
178
179 /* failed, clear out */
180 RTSpinlockDestroy(g_ThreadSpinlock);
181 g_ThreadSpinlock = NIL_RTSPINLOCK;
182 }
183#else
184# error "!IN_RING0 && !IN_RING3"
185#endif
186 return rc;
187}
188
189
190#ifdef IN_RING3
191/**
192 * Called when IPRT was first initialized in unobtrusive mode and later changed
193 * to obtrustive.
194 *
195 * This is only applicable in ring-3.
196 */
197DECLHIDDEN(void) rtThreadReInitObtrusive(void)
198{
199 rtThreadNativeReInitObtrusive();
200}
201#endif
202
203
204/**
205 * Terminates the thread database.
206 */
207DECLHIDDEN(void) rtThreadTerm(void)
208{
209#ifdef IN_RING3
210 /* we don't cleanup here yet */
211
212#elif defined(IN_RING0)
213 /* just destroy the spinlock and assume the thread is fine... */
214 RTSpinlockDestroy(g_ThreadSpinlock);
215 g_ThreadSpinlock = NIL_RTSPINLOCK;
216 if (g_ThreadTree != NULL)
217 RTAssertMsg2Weak("WARNING: g_ThreadTree=%p\n", g_ThreadTree);
218#endif
219}
220
221
222#ifdef IN_RING3
223
224DECLINLINE(void) rtThreadLockRW(void)
225{
226 if (g_ThreadRWSem == NIL_RTSEMRW)
227 rtThreadInit();
228 int rc = RTSemRWRequestWrite(g_ThreadRWSem, RT_INDEFINITE_WAIT);
229 AssertReleaseRC(rc);
230}
231
232
233DECLINLINE(void) rtThreadLockRD(void)
234{
235 if (g_ThreadRWSem == NIL_RTSEMRW)
236 rtThreadInit();
237 int rc = RTSemRWRequestRead(g_ThreadRWSem, RT_INDEFINITE_WAIT);
238 AssertReleaseRC(rc);
239}
240
241
242DECLINLINE(void) rtThreadUnLockRW(void)
243{
244 int rc = RTSemRWReleaseWrite(g_ThreadRWSem);
245 AssertReleaseRC(rc);
246}
247
248
249DECLINLINE(void) rtThreadUnLockRD(void)
250{
251 int rc = RTSemRWReleaseRead(g_ThreadRWSem);
252 AssertReleaseRC(rc);
253}
254
255
256/**
257 * Adopts the calling thread.
258 * No locks are taken or released by this function.
259 */
260static int rtThreadAdopt(RTTHREADTYPE enmType, unsigned fFlags, uint32_t fIntFlags, const char *pszName)
261{
262 int rc;
263 PRTTHREADINT pThread;
264 Assert(!(fFlags & RTTHREADFLAGS_WAITABLE));
265 fFlags &= ~RTTHREADFLAGS_WAITABLE;
266
267 /*
268 * Allocate and insert the thread.
269 * (It is vital that rtThreadNativeAdopt updates the TLS before
270 * we try inserting the thread because of locking.)
271 */
272 rc = VERR_NO_MEMORY;
273 pThread = rtThreadAlloc(enmType, fFlags, RTTHREADINT_FLAGS_ALIEN | fIntFlags, pszName);
274 if (pThread)
275 {
276 RTNATIVETHREAD NativeThread = RTThreadNativeSelf();
277 rc = rtThreadNativeAdopt(pThread);
278 if (RT_SUCCESS(rc))
279 {
280 rtThreadInsert(pThread, NativeThread);
281 rtThreadSetState(pThread, RTTHREADSTATE_RUNNING);
282 rtThreadRelease(pThread);
283 }
284 }
285 return rc;
286}
287
288/**
289 * Adopts a non-IPRT thread.
290 *
291 * @returns IPRT status code.
292 * @param enmType The thread type.
293 * @param fFlags The thread flags. RTTHREADFLAGS_WAITABLE is not currently allowed.
294 * @param pszName The thread name. Optional.
295 * @param pThread Where to store the thread handle. Optional.
296 */
297RTDECL(int) RTThreadAdopt(RTTHREADTYPE enmType, unsigned fFlags, const char *pszName, PRTTHREAD pThread)
298{
299 int rc;
300 RTTHREAD Thread;
301
302 AssertReturn(!(fFlags & RTTHREADFLAGS_WAITABLE), VERR_INVALID_PARAMETER);
303 AssertReturn(!pszName || VALID_PTR(pszName), VERR_INVALID_POINTER);
304 AssertReturn(!pThread || VALID_PTR(pThread), VERR_INVALID_POINTER);
305
306 rc = VINF_SUCCESS;
307 Thread = RTThreadSelf();
308 if (Thread == NIL_RTTHREAD)
309 {
310 /* generate a name if none was given. */
311 char szName[RTTHREAD_NAME_LEN];
312 if (!pszName || !*pszName)
313 {
314 static uint32_t s_i32AlienId = 0;
315 uint32_t i32Id = ASMAtomicIncU32(&s_i32AlienId);
316 RTStrPrintf(szName, sizeof(szName), "ALIEN-%RX32", i32Id);
317 pszName = szName;
318 }
319
320 /* try adopt it */
321 rc = rtThreadAdopt(enmType, fFlags, 0, pszName);
322 Thread = RTThreadSelf();
323 Log(("RTThreadAdopt: %RTthrd %RTnthrd '%s' enmType=%d fFlags=%#x rc=%Rrc\n",
324 Thread, RTThreadNativeSelf(), pszName, enmType, fFlags, rc));
325 }
326 else
327 Log(("RTThreadAdopt: %RTthrd %RTnthrd '%s' enmType=%d fFlags=%#x - already adopted!\n",
328 Thread, RTThreadNativeSelf(), pszName, enmType, fFlags));
329
330 if (pThread)
331 *pThread = Thread;
332 return rc;
333}
334RT_EXPORT_SYMBOL(RTThreadAdopt);
335
336
337/**
338 * Get the thread handle of the current thread, automatically adopting alien
339 * threads.
340 *
341 * @returns Thread handle.
342 */
343RTDECL(RTTHREAD) RTThreadSelfAutoAdopt(void)
344{
345 RTTHREAD hSelf = RTThreadSelf();
346 if (RT_UNLIKELY(hSelf == NIL_RTTHREAD))
347 RTThreadAdopt(RTTHREADTYPE_DEFAULT, 0, NULL, &hSelf);
348 return hSelf;
349}
350RT_EXPORT_SYMBOL(RTThreadSelfAutoAdopt);
351
352#endif /* IN_RING3 */
353
354/**
355 * Allocates a per thread data structure and initializes the basic fields.
356 *
357 * @returns Pointer to per thread data structure.
358 * This is reference once.
359 * @returns NULL on failure.
360 * @param enmType The thread type.
361 * @param fFlags The thread flags.
362 * @param fIntFlags The internal thread flags.
363 * @param pszName Pointer to the thread name.
364 */
365PRTTHREADINT rtThreadAlloc(RTTHREADTYPE enmType, unsigned fFlags, uint32_t fIntFlags, const char *pszName)
366{
367 PRTTHREADINT pThread = (PRTTHREADINT)RTMemAllocZ(sizeof(RTTHREADINT));
368 if (pThread)
369 {
370 size_t cchName;
371 int rc;
372
373 pThread->Core.Key = (void*)NIL_RTTHREAD;
374 pThread->u32Magic = RTTHREADINT_MAGIC;
375 cchName = strlen(pszName);
376 if (cchName >= RTTHREAD_NAME_LEN)
377 cchName = RTTHREAD_NAME_LEN - 1;
378 memcpy(pThread->szName, pszName, cchName);
379 pThread->szName[cchName] = '\0';
380 pThread->cRefs = 2 + !!(fFlags & RTTHREADFLAGS_WAITABLE); /* And extra reference if waitable. */
381 pThread->rc = VERR_PROCESS_RUNNING; /** @todo get a better error code! */
382 pThread->enmType = enmType;
383 pThread->fFlags = fFlags;
384 pThread->fIntFlags = fIntFlags;
385 pThread->enmState = RTTHREADSTATE_INITIALIZING;
386 pThread->fReallySleeping = false;
387#ifdef IN_RING3
388 rtLockValidatorInitPerThread(&pThread->LockValidator);
389#endif
390#ifdef RT_WITH_ICONV_CACHE
391 rtStrIconvCacheInit(pThread);
392#endif
393 rc = RTSemEventMultiCreate(&pThread->EventUser);
394 if (RT_SUCCESS(rc))
395 {
396 rc = RTSemEventMultiCreate(&pThread->EventTerminated);
397 if (RT_SUCCESS(rc))
398 return pThread;
399 RTSemEventMultiDestroy(pThread->EventUser);
400 }
401 RTMemFree(pThread);
402 }
403 return NULL;
404}
405
406
407/**
408 * Insert the per thread data structure into the tree.
409 *
410 * This can be called from both the thread it self and the parent,
411 * thus it must handle insertion failures in a nice manner.
412 *
413 * @param pThread Pointer to thread structure allocated by rtThreadAlloc().
414 * @param NativeThread The native thread id.
415 */
416DECLHIDDEN(void) rtThreadInsert(PRTTHREADINT pThread, RTNATIVETHREAD NativeThread)
417{
418 Assert(pThread);
419 Assert(pThread->u32Magic == RTTHREADINT_MAGIC);
420
421 {
422 RT_THREAD_LOCK_RW();
423
424 /*
425 * Do not insert a terminated thread.
426 *
427 * This may happen if the thread finishes before the RTThreadCreate call
428 * gets this far. Since the OS may quickly reuse the native thread ID
429 * it should not be reinserted at this point.
430 */
431 if (rtThreadGetState(pThread) != RTTHREADSTATE_TERMINATED)
432 {
433 /*
434 * Before inserting we must check if there is a thread with this id
435 * in the tree already. We're racing parent and child on insert here
436 * so that the handle is valid in both ends when they return / start.
437 *
438 * If it's not ourself we find, it's a dead alien thread and we will
439 * unlink it from the tree. Alien threads will be released at this point.
440 */
441 PRTTHREADINT pThreadOther = (PRTTHREADINT)RTAvlPVGet(&g_ThreadTree, (void *)NativeThread);
442 if (pThreadOther != pThread)
443 {
444 bool fRc;
445 /* remove dead alien if any */
446 if (pThreadOther)
447 {
448 AssertMsg(pThreadOther->fIntFlags & RTTHREADINT_FLAGS_ALIEN, ("%p:%s; %p:%s\n", pThread, pThread->szName, pThreadOther, pThreadOther->szName));
449 ASMAtomicBitClear(&pThread->fIntFlags, RTTHREADINT_FLAG_IN_TREE_BIT);
450 rtThreadRemoveLocked(pThreadOther);
451 if (pThreadOther->fIntFlags & RTTHREADINT_FLAGS_ALIEN)
452 rtThreadRelease(pThreadOther);
453 }
454
455 /* insert the thread */
456 ASMAtomicWritePtr(&pThread->Core.Key, (void *)NativeThread);
457 fRc = RTAvlPVInsert(&g_ThreadTree, &pThread->Core);
458 ASMAtomicOrU32(&pThread->fIntFlags, RTTHREADINT_FLAG_IN_TREE);
459 if (fRc)
460 ASMAtomicIncU32(&g_cThreadInTree);
461
462 AssertReleaseMsg(fRc, ("Lock problem? %p (%RTnthrd) %s\n", pThread, NativeThread, pThread->szName));
463 NOREF(fRc);
464 }
465 }
466
467 RT_THREAD_UNLOCK_RW();
468 }
469}
470
471
472/**
473 * Removes the thread from the AVL tree, call owns the tree lock
474 * and has cleared the RTTHREADINT_FLAG_IN_TREE bit.
475 *
476 * @param pThread The thread to remove.
477 */
478static void rtThreadRemoveLocked(PRTTHREADINT pThread)
479{
480 PRTTHREADINT pThread2 = (PRTTHREADINT)RTAvlPVRemove(&g_ThreadTree, pThread->Core.Key);
481#if !defined(RT_OS_OS2) /** @todo this asserts for threads created by NSPR */
482 AssertMsg(pThread2 == pThread, ("%p(%s) != %p (%p/%s)\n", pThread2, pThread2 ? pThread2->szName : "<null>",
483 pThread, pThread->Core.Key, pThread->szName));
484#endif
485 if (pThread2)
486 ASMAtomicDecU32(&g_cThreadInTree);
487}
488
489
490/**
491 * Removes the thread from the AVL tree.
492 *
493 * @param pThread The thread to remove.
494 */
495static void rtThreadRemove(PRTTHREADINT pThread)
496{
497 RT_THREAD_LOCK_RW();
498 if (ASMAtomicBitTestAndClear(&pThread->fIntFlags, RTTHREADINT_FLAG_IN_TREE_BIT))
499 rtThreadRemoveLocked(pThread);
500 RT_THREAD_UNLOCK_RW();
501}
502
503
504/**
505 * Checks if a thread is alive or not.
506 *
507 * @returns true if the thread is alive (or we don't really know).
508 * @returns false if the thread has surely terminate.
509 */
510DECLINLINE(bool) rtThreadIsAlive(PRTTHREADINT pThread)
511{
512 return !(pThread->fIntFlags & RTTHREADINT_FLAGS_TERMINATED);
513}
514
515
516/**
517 * Gets a thread by it's native ID.
518 *
519 * @returns pointer to the thread structure.
520 * @returns NULL if not a thread IPRT knows.
521 * @param NativeThread The native thread id.
522 */
523DECLHIDDEN(PRTTHREADINT) rtThreadGetByNative(RTNATIVETHREAD NativeThread)
524{
525 PRTTHREADINT pThread;
526 /*
527 * Simple tree lookup.
528 */
529 RT_THREAD_LOCK_RD();
530 pThread = (PRTTHREADINT)RTAvlPVGet(&g_ThreadTree, (void *)NativeThread);
531 RT_THREAD_UNLOCK_RD();
532 return pThread;
533}
534
535
536/**
537 * Gets the per thread data structure for a thread handle.
538 *
539 * @returns Pointer to the per thread data structure for Thread.
540 * The caller must release the thread using rtThreadRelease().
541 * @returns NULL if Thread was not found.
542 * @param Thread Thread id which structure is to be returned.
543 */
544DECLHIDDEN(PRTTHREADINT) rtThreadGet(RTTHREAD Thread)
545{
546 if ( Thread != NIL_RTTHREAD
547 && VALID_PTR(Thread))
548 {
549 PRTTHREADINT pThread = (PRTTHREADINT)Thread;
550 if ( pThread->u32Magic == RTTHREADINT_MAGIC
551 && pThread->cRefs > 0)
552 {
553 ASMAtomicIncU32(&pThread->cRefs);
554 return pThread;
555 }
556 }
557
558 AssertMsgFailed(("Thread=%RTthrd\n", Thread));
559 return NULL;
560}
561
562/**
563 * Release a per thread data structure.
564 *
565 * @returns New reference count.
566 * @param pThread The thread structure to release.
567 */
568DECLHIDDEN(uint32_t) rtThreadRelease(PRTTHREADINT pThread)
569{
570 uint32_t cRefs;
571
572 Assert(pThread);
573 if (pThread->cRefs >= 1)
574 {
575 cRefs = ASMAtomicDecU32(&pThread->cRefs);
576 if (!cRefs)
577 rtThreadDestroy(pThread);
578 }
579 else
580 {
581 cRefs = 0;
582 AssertFailed();
583 }
584 return cRefs;
585}
586
587
588/**
589 * Destroys the per thread data.
590 *
591 * @param pThread The thread to destroy.
592 */
593static void rtThreadDestroy(PRTTHREADINT pThread)
594{
595 RTSEMEVENTMULTI hEvt1, hEvt2;
596 /*
597 * Remove it from the tree and mark it as dead.
598 *
599 * Threads that has seen rtThreadTerminate and should already have been
600 * removed from the tree. There is probably no thread that should
601 * require removing here. However, be careful making sure that cRefs
602 * isn't 0 if we do or we'll blow up because the strict locking code
603 * will be calling us back.
604 */
605 if (ASMBitTest(&pThread->fIntFlags, RTTHREADINT_FLAG_IN_TREE_BIT))
606 {
607 ASMAtomicIncU32(&pThread->cRefs);
608 rtThreadRemove(pThread);
609 ASMAtomicDecU32(&pThread->cRefs);
610 }
611
612 /*
613 * Invalidate the thread structure.
614 */
615#ifdef IN_RING3
616 rtLockValidatorSerializeDestructEnter();
617
618 rtLockValidatorDeletePerThread(&pThread->LockValidator);
619#endif
620#ifdef RT_WITH_ICONV_CACHE
621 rtStrIconvCacheDestroy(pThread);
622#endif
623 ASMAtomicXchgU32(&pThread->u32Magic, RTTHREADINT_MAGIC_DEAD);
624 ASMAtomicWritePtr(&pThread->Core.Key, (void *)NIL_RTTHREAD);
625 pThread->enmType = RTTHREADTYPE_INVALID;
626 hEvt1 = pThread->EventUser;
627 pThread->EventUser = NIL_RTSEMEVENTMULTI;
628 hEvt2 = pThread->EventTerminated;
629 pThread->EventTerminated = NIL_RTSEMEVENTMULTI;
630
631#ifdef IN_RING3
632 rtLockValidatorSerializeDestructLeave();
633#endif
634
635 /*
636 * Destroy semaphore resources and free the bugger.
637 */
638 RTSemEventMultiDestroy(hEvt1);
639 if (hEvt2 != NIL_RTSEMEVENTMULTI)
640 RTSemEventMultiDestroy(hEvt2);
641
642 rtThreadNativeDestroy(pThread);
643 RTMemFree(pThread);
644}
645
646
647/**
648 * Terminates the thread.
649 * Called by the thread wrapper function when the thread terminates.
650 *
651 * @param pThread The thread structure.
652 * @param rc The thread result code.
653 */
654DECLHIDDEN(void) rtThreadTerminate(PRTTHREADINT pThread, int rc)
655{
656 Assert(pThread->cRefs >= 1);
657
658#ifdef IPRT_WITH_GENERIC_TLS
659 /*
660 * Destroy TLS entries.
661 */
662 rtThreadTlsDestruction(pThread);
663#endif /* IPRT_WITH_GENERIC_TLS */
664
665 /*
666 * Set the rc, mark it terminated and signal anyone waiting.
667 */
668 pThread->rc = rc;
669 rtThreadSetState(pThread, RTTHREADSTATE_TERMINATED);
670 ASMAtomicOrU32(&pThread->fIntFlags, RTTHREADINT_FLAGS_TERMINATED);
671 if (pThread->EventTerminated != NIL_RTSEMEVENTMULTI)
672 RTSemEventMultiSignal(pThread->EventTerminated);
673
674 /*
675 * Remove the thread from the tree so that there will be no
676 * key clashes in the AVL tree and release our reference to ourself.
677 */
678 rtThreadRemove(pThread);
679 rtThreadRelease(pThread);
680}
681
682
683/**
684 * The common thread main function.
685 * This is called by rtThreadNativeMain().
686 *
687 * @returns The status code of the thread.
688 * pThread is dereference by the thread before returning!
689 * @param pThread The thread structure.
690 * @param NativeThread The native thread id.
691 * @param pszThreadName The name of the thread (purely a dummy for backtrace).
692 */
693DECLCALLBACK(DECLHIDDEN(int)) rtThreadMain(PRTTHREADINT pThread, RTNATIVETHREAD NativeThread, const char *pszThreadName)
694{
695 int rc;
696 NOREF(pszThreadName);
697 rtThreadInsert(pThread, NativeThread);
698 Log(("rtThreadMain: Starting: pThread=%p NativeThread=%RTnthrd Name=%s pfnThread=%p pvUser=%p\n",
699 pThread, NativeThread, pThread->szName, pThread->pfnThread, pThread->pvUser));
700
701 /*
702 * Change the priority.
703 */
704 rc = rtThreadNativeSetPriority(pThread, pThread->enmType);
705#ifdef IN_RING3
706 AssertMsgRC(rc, ("Failed to set priority of thread %p (%RTnthrd / %s) to enmType=%d enmPriority=%d rc=%Rrc\n",
707 pThread, NativeThread, pThread->szName, pThread->enmType, g_enmProcessPriority, rc));
708#else
709 AssertMsgRC(rc, ("Failed to set priority of thread %p (%RTnthrd / %s) to enmType=%d rc=%Rrc\n",
710 pThread, NativeThread, pThread->szName, pThread->enmType, rc));
711#endif
712
713 /*
714 * Call thread function and terminate when it returns.
715 */
716 rtThreadSetState(pThread, RTTHREADSTATE_RUNNING);
717 rc = pThread->pfnThread(pThread, pThread->pvUser);
718
719 /*
720 * Paranoia checks for leftover resources.
721 */
722#ifdef RTSEMRW_STRICT
723 int32_t cWrite = ASMAtomicReadS32(&pThread->cWriteLocks);
724 Assert(!cWrite);
725 int32_t cRead = ASMAtomicReadS32(&pThread->cReadLocks);
726 Assert(!cRead);
727#endif
728
729 Log(("rtThreadMain: Terminating: rc=%d pThread=%p NativeThread=%RTnthrd Name=%s pfnThread=%p pvUser=%p\n",
730 rc, pThread, NativeThread, pThread->szName, pThread->pfnThread, pThread->pvUser));
731 rtThreadTerminate(pThread, rc);
732 return rc;
733}
734
735
736/**
737 * Create a new thread.
738 *
739 * @returns iprt status code.
740 * @param pThread Where to store the thread handle to the new thread. (optional)
741 * @param pfnThread The thread function.
742 * @param pvUser User argument.
743 * @param cbStack The size of the stack for the new thread.
744 * Use 0 for the default stack size.
745 * @param enmType The thread type. Used for deciding scheduling attributes
746 * of the thread.
747 * @param fFlags Flags of the RTTHREADFLAGS type (ORed together).
748 * @param pszName Thread name.
749 */
750RTDECL(int) RTThreadCreate(PRTTHREAD pThread, PFNRTTHREAD pfnThread, void *pvUser, size_t cbStack,
751 RTTHREADTYPE enmType, unsigned fFlags, const char *pszName)
752{
753 int rc;
754 PRTTHREADINT pThreadInt;
755
756 LogFlow(("RTThreadCreate: pThread=%p pfnThread=%p pvUser=%p cbStack=%#x enmType=%d fFlags=%#x pszName=%p:{%s}\n",
757 pThread, pfnThread, pvUser, cbStack, enmType, fFlags, pszName, pszName));
758
759 /*
760 * Validate input.
761 */
762 if (!VALID_PTR(pThread) && pThread)
763 {
764 Assert(VALID_PTR(pThread));
765 return VERR_INVALID_PARAMETER;
766 }
767 if (!VALID_PTR(pfnThread))
768 {
769 Assert(VALID_PTR(pfnThread));
770 return VERR_INVALID_PARAMETER;
771 }
772 if (!pszName || !*pszName || strlen(pszName) >= RTTHREAD_NAME_LEN)
773 {
774 AssertMsgFailed(("pszName=%s (max len is %d because of logging)\n", pszName, RTTHREAD_NAME_LEN - 1));
775 return VERR_INVALID_PARAMETER;
776 }
777 if (fFlags & ~RTTHREADFLAGS_MASK)
778 {
779 AssertMsgFailed(("fFlags=%#x\n", fFlags));
780 return VERR_INVALID_PARAMETER;
781 }
782
783 /*
784 * Allocate thread argument.
785 */
786 pThreadInt = rtThreadAlloc(enmType, fFlags, 0, pszName);
787 if (pThreadInt)
788 {
789 RTNATIVETHREAD NativeThread;
790
791 pThreadInt->pfnThread = pfnThread;
792 pThreadInt->pvUser = pvUser;
793 pThreadInt->cbStack = cbStack;
794
795 rc = rtThreadNativeCreate(pThreadInt, &NativeThread);
796 if (RT_SUCCESS(rc))
797 {
798 rtThreadInsert(pThreadInt, NativeThread);
799 rtThreadRelease(pThreadInt);
800 Log(("RTThreadCreate: Created thread %p (%p) %s\n", pThreadInt, NativeThread, pszName));
801 if (pThread)
802 *pThread = pThreadInt;
803 return VINF_SUCCESS;
804 }
805
806 pThreadInt->cRefs = 1;
807 rtThreadRelease(pThreadInt);
808 }
809 else
810 rc = VERR_NO_TMP_MEMORY;
811 LogFlow(("RTThreadCreate: Failed to create thread, rc=%Rrc\n", rc));
812 AssertReleaseRC(rc);
813 return rc;
814}
815RT_EXPORT_SYMBOL(RTThreadCreate);
816
817
818/**
819 * Create a new thread.
820 *
821 * Same as RTThreadCreate except the name is given in the RTStrPrintfV form.
822 *
823 * @returns iprt status code.
824 * @param pThread See RTThreadCreate.
825 * @param pfnThread See RTThreadCreate.
826 * @param pvUser See RTThreadCreate.
827 * @param cbStack See RTThreadCreate.
828 * @param enmType See RTThreadCreate.
829 * @param fFlags See RTThreadCreate.
830 * @param pszNameFmt Thread name format.
831 * @param va Format arguments.
832 */
833RTDECL(int) RTThreadCreateV(PRTTHREAD pThread, PFNRTTHREAD pfnThread, void *pvUser, size_t cbStack,
834 RTTHREADTYPE enmType, uint32_t fFlags, const char *pszNameFmt, va_list va)
835{
836 char szName[RTTHREAD_NAME_LEN * 2];
837 RTStrPrintfV(szName, sizeof(szName), pszNameFmt, va);
838 return RTThreadCreate(pThread, pfnThread, pvUser, cbStack, enmType, fFlags, szName);
839}
840RT_EXPORT_SYMBOL(RTThreadCreateV);
841
842
843/**
844 * Create a new thread.
845 *
846 * Same as RTThreadCreate except the name is given in the RTStrPrintf form.
847 *
848 * @returns iprt status code.
849 * @param pThread See RTThreadCreate.
850 * @param pfnThread See RTThreadCreate.
851 * @param pvUser See RTThreadCreate.
852 * @param cbStack See RTThreadCreate.
853 * @param enmType See RTThreadCreate.
854 * @param fFlags See RTThreadCreate.
855 * @param pszNameFmt Thread name format.
856 * @param ... Format arguments.
857 */
858RTDECL(int) RTThreadCreateF(PRTTHREAD pThread, PFNRTTHREAD pfnThread, void *pvUser, size_t cbStack,
859 RTTHREADTYPE enmType, uint32_t fFlags, const char *pszNameFmt, ...)
860{
861 va_list va;
862 int rc;
863 va_start(va, pszNameFmt);
864 rc = RTThreadCreateV(pThread, pfnThread, pvUser, cbStack, enmType, fFlags, pszNameFmt, va);
865 va_end(va);
866 return rc;
867}
868RT_EXPORT_SYMBOL(RTThreadCreateF);
869
870
871/**
872 * Gets the native thread id of a IPRT thread.
873 *
874 * @returns The native thread id.
875 * @param Thread The IPRT thread.
876 */
877RTDECL(RTNATIVETHREAD) RTThreadGetNative(RTTHREAD Thread)
878{
879 PRTTHREADINT pThread = rtThreadGet(Thread);
880 if (pThread)
881 {
882 RTNATIVETHREAD NativeThread = (RTNATIVETHREAD)pThread->Core.Key;
883 rtThreadRelease(pThread);
884 return NativeThread;
885 }
886 return NIL_RTNATIVETHREAD;
887}
888RT_EXPORT_SYMBOL(RTThreadGetNative);
889
890
891/**
892 * Gets the IPRT thread of a native thread.
893 *
894 * @returns The IPRT thread handle
895 * @returns NIL_RTTHREAD if not a thread known to IPRT.
896 * @param NativeThread The native thread handle/id.
897 */
898RTDECL(RTTHREAD) RTThreadFromNative(RTNATIVETHREAD NativeThread)
899{
900 PRTTHREADINT pThread = rtThreadGetByNative(NativeThread);
901 if (pThread)
902 return pThread;
903 return NIL_RTTHREAD;
904}
905RT_EXPORT_SYMBOL(RTThreadFromNative);
906
907
908/**
909 * Gets the name of the current thread thread.
910 *
911 * @returns Pointer to readonly name string.
912 * @returns NULL on failure.
913 */
914RTDECL(const char *) RTThreadSelfName(void)
915{
916 RTTHREAD Thread = RTThreadSelf();
917 if (Thread != NIL_RTTHREAD)
918 {
919 PRTTHREADINT pThread = rtThreadGet(Thread);
920 if (pThread)
921 {
922 const char *szName = pThread->szName;
923 rtThreadRelease(pThread);
924 return szName;
925 }
926 }
927 return NULL;
928}
929RT_EXPORT_SYMBOL(RTThreadSelfName);
930
931
932/**
933 * Gets the name of a thread.
934 *
935 * @returns Pointer to readonly name string.
936 * @returns NULL on failure.
937 * @param Thread Thread handle of the thread to query the name of.
938 */
939RTDECL(const char *) RTThreadGetName(RTTHREAD Thread)
940{
941 PRTTHREADINT pThread;
942 if (Thread == NIL_RTTHREAD)
943 return NULL;
944 pThread = rtThreadGet(Thread);
945 if (pThread)
946 {
947 const char *szName = pThread->szName;
948 rtThreadRelease(pThread);
949 return szName;
950 }
951 return NULL;
952}
953RT_EXPORT_SYMBOL(RTThreadGetName);
954
955
956/**
957 * Sets the name of a thread.
958 *
959 * @returns iprt status code.
960 * @param Thread Thread handle of the thread to query the name of.
961 * @param pszName The thread name.
962 */
963RTDECL(int) RTThreadSetName(RTTHREAD Thread, const char *pszName)
964{
965 /*
966 * Validate input.
967 */
968 PRTTHREADINT pThread;
969 size_t cchName = strlen(pszName);
970 if (cchName >= RTTHREAD_NAME_LEN)
971 {
972 AssertMsgFailed(("pszName=%s is too long, max is %d\n", pszName, RTTHREAD_NAME_LEN - 1));
973 return VERR_INVALID_PARAMETER;
974 }
975 pThread = rtThreadGet(Thread);
976 if (!pThread)
977 return VERR_INVALID_HANDLE;
978
979 /*
980 * Update the name.
981 */
982 pThread->szName[cchName] = '\0'; /* paranoia */
983 memcpy(pThread->szName, pszName, cchName);
984 rtThreadRelease(pThread);
985 return VINF_SUCCESS;
986}
987RT_EXPORT_SYMBOL(RTThreadSetName);
988
989
990/**
991 * Checks if the specified thread is the main thread.
992 *
993 * @returns true if it is, false if it isn't.
994 *
995 * @param hThread The thread handle.
996 *
997 * @remarks This function may not return the correct value when rtR3Init was
998 * called on a thread of the than the main one. This could for
999 * instance happen when the DLL/DYLIB/SO containing IPRT is dynamically
1000 * loaded at run time by a different thread.
1001 */
1002RTDECL(bool) RTThreadIsMain(RTTHREAD hThread)
1003{
1004 PRTTHREADINT pThread = rtThreadGet(hThread);
1005 if (pThread)
1006 {
1007 bool fRc = !!(pThread->fIntFlags & RTTHREADINT_FLAGS_MAIN);
1008 rtThreadRelease(pThread);
1009 return fRc;
1010 }
1011 return false;
1012}
1013RT_EXPORT_SYMBOL(RTThreadIsMain);
1014
1015
1016RTDECL(bool) RTThreadIsSelfAlive(void)
1017{
1018 if (g_frtThreadInitialized)
1019 {
1020 RTTHREAD hSelf = RTThreadSelf();
1021 if (hSelf != NIL_RTTHREAD)
1022 {
1023 /*
1024 * Inspect the thread state. ASSUMES thread state order.
1025 */
1026 RTTHREADSTATE enmState = rtThreadGetState(hSelf);
1027 if ( enmState >= RTTHREADSTATE_RUNNING
1028 && enmState <= RTTHREADSTATE_END)
1029 return true;
1030 }
1031 }
1032 return false;
1033}
1034RT_EXPORT_SYMBOL(RTThreadIsSelfAlive);
1035
1036
1037RTDECL(bool) RTThreadIsSelfKnown(void)
1038{
1039 if (g_frtThreadInitialized)
1040 {
1041 RTTHREAD hSelf = RTThreadSelf();
1042 if (hSelf != NIL_RTTHREAD)
1043 return true;
1044 }
1045 return false;
1046}
1047RT_EXPORT_SYMBOL(RTThreadIsSelfKnown);
1048
1049
1050RTDECL(bool) RTThreadIsInitialized(void)
1051{
1052 return g_frtThreadInitialized;
1053}
1054RT_EXPORT_SYMBOL(RTThreadIsInitialized);
1055
1056
1057/**
1058 * Signal the user event.
1059 *
1060 * @returns iprt status code.
1061 */
1062RTDECL(int) RTThreadUserSignal(RTTHREAD Thread)
1063{
1064 int rc;
1065 PRTTHREADINT pThread = rtThreadGet(Thread);
1066 if (pThread)
1067 {
1068 rc = RTSemEventMultiSignal(pThread->EventUser);
1069 rtThreadRelease(pThread);
1070 }
1071 else
1072 rc = VERR_INVALID_HANDLE;
1073 return rc;
1074}
1075RT_EXPORT_SYMBOL(RTThreadUserSignal);
1076
1077
1078/**
1079 * Wait for the user event, resume on interruption.
1080 *
1081 * @returns iprt status code.
1082 * @param Thread The thread to wait for.
1083 * @param cMillies The number of milliseconds to wait. Use RT_INDEFINITE_WAIT for
1084 * an indefinite wait.
1085 */
1086RTDECL(int) RTThreadUserWait(RTTHREAD Thread, RTMSINTERVAL cMillies)
1087{
1088 int rc;
1089 PRTTHREADINT pThread = rtThreadGet(Thread);
1090 if (pThread)
1091 {
1092 rc = RTSemEventMultiWait(pThread->EventUser, cMillies);
1093 rtThreadRelease(pThread);
1094 }
1095 else
1096 rc = VERR_INVALID_HANDLE;
1097 return rc;
1098}
1099RT_EXPORT_SYMBOL(RTThreadUserWait);
1100
1101
1102/**
1103 * Wait for the user event, return on interruption.
1104 *
1105 * @returns iprt status code.
1106 * @param Thread The thread to wait for.
1107 * @param cMillies The number of milliseconds to wait. Use RT_INDEFINITE_WAIT for
1108 * an indefinite wait.
1109 */
1110RTDECL(int) RTThreadUserWaitNoResume(RTTHREAD Thread, RTMSINTERVAL cMillies)
1111{
1112 int rc;
1113 PRTTHREADINT pThread = rtThreadGet(Thread);
1114 if (pThread)
1115 {
1116 rc = RTSemEventMultiWaitNoResume(pThread->EventUser, cMillies);
1117 rtThreadRelease(pThread);
1118 }
1119 else
1120 rc = VERR_INVALID_HANDLE;
1121 return rc;
1122}
1123RT_EXPORT_SYMBOL(RTThreadUserWaitNoResume);
1124
1125
1126/**
1127 * Reset the user event.
1128 *
1129 * @returns iprt status code.
1130 * @param Thread The thread to reset.
1131 */
1132RTDECL(int) RTThreadUserReset(RTTHREAD Thread)
1133{
1134 int rc;
1135 PRTTHREADINT pThread = rtThreadGet(Thread);
1136 if (pThread)
1137 {
1138 rc = RTSemEventMultiReset(pThread->EventUser);
1139 rtThreadRelease(pThread);
1140 }
1141 else
1142 rc = VERR_INVALID_HANDLE;
1143 return rc;
1144}
1145RT_EXPORT_SYMBOL(RTThreadUserReset);
1146
1147
1148/**
1149 * Wait for the thread to terminate.
1150 *
1151 * @returns iprt status code.
1152 * @param Thread The thread to wait for.
1153 * @param cMillies The number of milliseconds to wait. Use RT_INDEFINITE_WAIT for
1154 * an indefinite wait.
1155 * @param prc Where to store the return code of the thread. Optional.
1156 * @param fAutoResume Whether or not to resume the wait on VERR_INTERRUPTED.
1157 */
1158static int rtThreadWait(RTTHREAD Thread, RTMSINTERVAL cMillies, int *prc, bool fAutoResume)
1159{
1160 int rc = VERR_INVALID_HANDLE;
1161 if (Thread != NIL_RTTHREAD)
1162 {
1163 PRTTHREADINT pThread = rtThreadGet(Thread);
1164 if (pThread)
1165 {
1166 if (pThread->fFlags & RTTHREADFLAGS_WAITABLE)
1167 {
1168 if (fAutoResume)
1169 rc = RTSemEventMultiWait(pThread->EventTerminated, cMillies);
1170 else
1171 rc = RTSemEventMultiWaitNoResume(pThread->EventTerminated, cMillies);
1172 if (RT_SUCCESS(rc))
1173 {
1174 if (prc)
1175 *prc = pThread->rc;
1176
1177 /*
1178 * If the thread is marked as waitable, we'll do one additional
1179 * release in order to free up the thread structure (see how we
1180 * init cRef in rtThreadAlloc()).
1181 */
1182 if (ASMAtomicBitTestAndClear(&pThread->fFlags, RTTHREADFLAGS_WAITABLE_BIT))
1183 {
1184 rtThreadRelease(pThread);
1185#ifdef IN_RING0
1186 /*
1187 * IPRT termination kludge. Call native code to make sure
1188 * the last thread is really out of IPRT to prevent it from
1189 * crashing after we destroyed the spinlock in rtThreadTerm.
1190 */
1191 if ( ASMAtomicReadU32(&g_cThreadInTree) == 1
1192 && ASMAtomicReadU32(&pThread->cRefs) > 1)
1193 rtThreadNativeWaitKludge(pThread);
1194#endif
1195 }
1196 }
1197 }
1198 else
1199 {
1200 rc = VERR_THREAD_NOT_WAITABLE;
1201 AssertRC(rc);
1202 }
1203 rtThreadRelease(pThread);
1204 }
1205 }
1206 return rc;
1207}
1208
1209
1210/**
1211 * Wait for the thread to terminate, resume on interruption.
1212 *
1213 * @returns iprt status code.
1214 * Will not return VERR_INTERRUPTED.
1215 * @param Thread The thread to wait for.
1216 * @param cMillies The number of milliseconds to wait. Use RT_INDEFINITE_WAIT for
1217 * an indefinite wait.
1218 * @param prc Where to store the return code of the thread. Optional.
1219 */
1220RTDECL(int) RTThreadWait(RTTHREAD Thread, RTMSINTERVAL cMillies, int *prc)
1221{
1222 int rc = rtThreadWait(Thread, cMillies, prc, true);
1223 Assert(rc != VERR_INTERRUPTED);
1224 return rc;
1225}
1226RT_EXPORT_SYMBOL(RTThreadWait);
1227
1228
1229/**
1230 * Wait for the thread to terminate, return on interruption.
1231 *
1232 * @returns iprt status code.
1233 * @param Thread The thread to wait for.
1234 * @param cMillies The number of milliseconds to wait. Use RT_INDEFINITE_WAIT for
1235 * an indefinite wait.
1236 * @param prc Where to store the return code of the thread. Optional.
1237 */
1238RTDECL(int) RTThreadWaitNoResume(RTTHREAD Thread, RTMSINTERVAL cMillies, int *prc)
1239{
1240 return rtThreadWait(Thread, cMillies, prc, false);
1241}
1242RT_EXPORT_SYMBOL(RTThreadWaitNoResume);
1243
1244
1245/**
1246 * Changes the type of the specified thread.
1247 *
1248 * @returns iprt status code.
1249 * @param Thread The thread which type should be changed.
1250 * @param enmType The new thread type.
1251 */
1252RTDECL(int) RTThreadSetType(RTTHREAD Thread, RTTHREADTYPE enmType)
1253{
1254 /*
1255 * Validate input.
1256 */
1257 int rc;
1258 if ( enmType > RTTHREADTYPE_INVALID
1259 && enmType < RTTHREADTYPE_END)
1260 {
1261 PRTTHREADINT pThread = rtThreadGet(Thread);
1262 if (pThread)
1263 {
1264 if (rtThreadIsAlive(pThread))
1265 {
1266 /*
1267 * Do the job.
1268 */
1269 RT_THREAD_LOCK_RW();
1270 rc = rtThreadNativeSetPriority(pThread, enmType);
1271 if (RT_SUCCESS(rc))
1272 ASMAtomicXchgSize(&pThread->enmType, enmType);
1273 RT_THREAD_UNLOCK_RW();
1274 if (RT_FAILURE(rc))
1275 Log(("RTThreadSetType: failed on thread %p (%s), rc=%Rrc!!!\n", Thread, pThread->szName, rc));
1276 }
1277 else
1278 rc = VERR_THREAD_IS_DEAD;
1279 rtThreadRelease(pThread);
1280 }
1281 else
1282 rc = VERR_INVALID_HANDLE;
1283 }
1284 else
1285 {
1286 AssertMsgFailed(("enmType=%d\n", enmType));
1287 rc = VERR_INVALID_PARAMETER;
1288 }
1289 return rc;
1290}
1291RT_EXPORT_SYMBOL(RTThreadSetType);
1292
1293
1294/**
1295 * Gets the type of the specified thread.
1296 *
1297 * @returns The thread type.
1298 * @returns RTTHREADTYPE_INVALID if the thread handle is invalid.
1299 * @param Thread The thread in question.
1300 */
1301RTDECL(RTTHREADTYPE) RTThreadGetType(RTTHREAD Thread)
1302{
1303 RTTHREADTYPE enmType = RTTHREADTYPE_INVALID;
1304 PRTTHREADINT pThread = rtThreadGet(Thread);
1305 if (pThread)
1306 {
1307 enmType = pThread->enmType;
1308 rtThreadRelease(pThread);
1309 }
1310 return enmType;
1311}
1312RT_EXPORT_SYMBOL(RTThreadGetType);
1313
1314#ifdef IN_RING3
1315
1316/**
1317 * Recalculates scheduling attributes for the default process
1318 * priority using the specified priority type for the calling thread.
1319 *
1320 * The scheduling attributes are targeted at threads and they are protected
1321 * by the thread read-write semaphore, that's why RTProc is forwarding the
1322 * operation to RTThread.
1323 *
1324 * @returns iprt status code.
1325 * @remarks Will only work for strict builds.
1326 */
1327int rtThreadDoCalcDefaultPriority(RTTHREADTYPE enmType)
1328{
1329 RT_THREAD_LOCK_RW();
1330 int rc = rtSchedNativeCalcDefaultPriority(enmType);
1331 RT_THREAD_UNLOCK_RW();
1332 return rc;
1333}
1334
1335
1336/**
1337 * Thread enumerator - sets the priority of one thread.
1338 *
1339 * @returns 0 to continue.
1340 * @returns !0 to stop. In our case a VERR_ code.
1341 * @param pNode The thread node.
1342 * @param pvUser The new priority.
1343 */
1344static DECLCALLBACK(int) rtThreadSetPriorityOne(PAVLPVNODECORE pNode, void *pvUser)
1345{
1346 PRTTHREADINT pThread = (PRTTHREADINT)pNode;
1347 if (!rtThreadIsAlive(pThread))
1348 return VINF_SUCCESS;
1349 int rc = rtThreadNativeSetPriority(pThread, pThread->enmType);
1350 if (RT_SUCCESS(rc)) /* hide any warnings */
1351 return VINF_SUCCESS;
1352 NOREF(pvUser);
1353 return rc;
1354}
1355
1356
1357/**
1358 * Attempts to alter the priority of the current process.
1359 *
1360 * The scheduling attributes are targeted at threads and they are protected
1361 * by the thread read-write semaphore, that's why RTProc is forwarding the
1362 * operation to RTThread. This operation also involves updating all thread
1363 * which is much faster done from RTThread.
1364 *
1365 * @returns iprt status code.
1366 * @param enmPriority The new priority.
1367 */
1368DECLHIDDEN(int) rtThreadDoSetProcPriority(RTPROCPRIORITY enmPriority)
1369{
1370 LogFlow(("rtThreadDoSetProcPriority: enmPriority=%d\n", enmPriority));
1371
1372 /*
1373 * First validate that we're allowed by the OS to use all the
1374 * scheduling attributes defined by the specified process priority.
1375 */
1376 RT_THREAD_LOCK_RW();
1377 int rc = rtProcNativeSetPriority(enmPriority);
1378 if (RT_SUCCESS(rc))
1379 {
1380 /*
1381 * Update the priority of existing thread.
1382 */
1383 rc = RTAvlPVDoWithAll(&g_ThreadTree, true, rtThreadSetPriorityOne, NULL);
1384 if (RT_SUCCESS(rc))
1385 ASMAtomicXchgSize(&g_enmProcessPriority, enmPriority);
1386 else
1387 {
1388 /*
1389 * Failed, restore the priority.
1390 */
1391 rtProcNativeSetPriority(g_enmProcessPriority);
1392 RTAvlPVDoWithAll(&g_ThreadTree, true, rtThreadSetPriorityOne, NULL);
1393 }
1394 }
1395 RT_THREAD_UNLOCK_RW();
1396 LogFlow(("rtThreadDoSetProcPriority: returns %Rrc\n", rc));
1397 return rc;
1398}
1399
1400
1401/**
1402 * Change the thread state to blocking.
1403 *
1404 * @param hThread The current thread.
1405 * @param enmState The sleep state.
1406 * @param fReallySleeping Really going to sleep now.
1407 */
1408RTDECL(void) RTThreadBlocking(RTTHREAD hThread, RTTHREADSTATE enmState, bool fReallySleeping)
1409{
1410 Assert(RTTHREAD_IS_SLEEPING(enmState));
1411 PRTTHREADINT pThread = hThread;
1412 if (pThread != NIL_RTTHREAD)
1413 {
1414 Assert(pThread == RTThreadSelf());
1415 if (rtThreadGetState(pThread) == RTTHREADSTATE_RUNNING)
1416 rtThreadSetState(pThread, enmState);
1417 ASMAtomicWriteBool(&pThread->fReallySleeping, fReallySleeping);
1418 }
1419}
1420RT_EXPORT_SYMBOL(RTThreadBlocking);
1421
1422
1423/**
1424 * Unblocks a thread.
1425 *
1426 * This function is paired with rtThreadBlocking.
1427 *
1428 * @param hThread The current thread.
1429 * @param enmCurState The current state, used to check for nested blocking.
1430 * The new state will be running.
1431 */
1432RTDECL(void) RTThreadUnblocked(RTTHREAD hThread, RTTHREADSTATE enmCurState)
1433{
1434 PRTTHREADINT pThread = hThread;
1435 if (pThread != NIL_RTTHREAD)
1436 {
1437 Assert(pThread == RTThreadSelf());
1438 ASMAtomicWriteBool(&pThread->fReallySleeping, false);
1439
1440 RTTHREADSTATE enmActualState = rtThreadGetState(pThread);
1441 if (enmActualState == enmCurState)
1442 {
1443 rtThreadSetState(pThread, RTTHREADSTATE_RUNNING);
1444 if ( pThread->LockValidator.pRec
1445 && pThread->LockValidator.enmRecState == enmCurState)
1446 ASMAtomicWriteNullPtr(&pThread->LockValidator.pRec);
1447 }
1448 /* This is a bit ugly... :-/ */
1449 else if ( ( enmActualState == RTTHREADSTATE_TERMINATED
1450 || enmActualState == RTTHREADSTATE_INITIALIZING)
1451 && pThread->LockValidator.pRec)
1452 ASMAtomicWriteNullPtr(&pThread->LockValidator.pRec);
1453 Assert( pThread->LockValidator.pRec == NULL
1454 || RTTHREAD_IS_SLEEPING(enmActualState));
1455 }
1456}
1457RT_EXPORT_SYMBOL(RTThreadUnblocked);
1458
1459
1460/**
1461 * Get the current thread state.
1462 *
1463 * @returns The thread state.
1464 * @param hThread The thread.
1465 */
1466RTDECL(RTTHREADSTATE) RTThreadGetState(RTTHREAD hThread)
1467{
1468 RTTHREADSTATE enmState = RTTHREADSTATE_INVALID;
1469 PRTTHREADINT pThread = rtThreadGet(hThread);
1470 if (pThread)
1471 {
1472 enmState = rtThreadGetState(pThread);
1473 rtThreadRelease(pThread);
1474 }
1475 return enmState;
1476}
1477RT_EXPORT_SYMBOL(RTThreadGetState);
1478
1479
1480RTDECL(RTTHREADSTATE) RTThreadGetReallySleeping(RTTHREAD hThread)
1481{
1482 RTTHREADSTATE enmState = RTTHREADSTATE_INVALID;
1483 PRTTHREADINT pThread = rtThreadGet(hThread);
1484 if (pThread)
1485 {
1486 enmState = rtThreadGetState(pThread);
1487 if (!ASMAtomicUoReadBool(&pThread->fReallySleeping))
1488 enmState = RTTHREADSTATE_RUNNING;
1489 rtThreadRelease(pThread);
1490 }
1491 return enmState;
1492}
1493RT_EXPORT_SYMBOL(RTThreadGetReallySleeping);
1494
1495
1496/**
1497 * Translate a thread state into a string.
1498 *
1499 * @returns Pointer to a read-only string containing the state name.
1500 * @param enmState The state.
1501 */
1502RTDECL(const char *) RTThreadStateName(RTTHREADSTATE enmState)
1503{
1504 switch (enmState)
1505 {
1506 case RTTHREADSTATE_INVALID: return "INVALID";
1507 case RTTHREADSTATE_INITIALIZING: return "INITIALIZING";
1508 case RTTHREADSTATE_TERMINATED: return "TERMINATED";
1509 case RTTHREADSTATE_RUNNING: return "RUNNING";
1510 case RTTHREADSTATE_CRITSECT: return "CRITSECT";
1511 case RTTHREADSTATE_EVENT: return "EVENT";
1512 case RTTHREADSTATE_EVENT_MULTI: return "EVENT_MULTI";
1513 case RTTHREADSTATE_FAST_MUTEX: return "FAST_MUTEX";
1514 case RTTHREADSTATE_MUTEX: return "MUTEX";
1515 case RTTHREADSTATE_RW_READ: return "RW_READ";
1516 case RTTHREADSTATE_RW_WRITE: return "RW_WRITE";
1517 case RTTHREADSTATE_SLEEP: return "SLEEP";
1518 case RTTHREADSTATE_SPIN_MUTEX: return "SPIN_MUTEX";
1519 default: return "UnknownThreadState";
1520 }
1521}
1522RT_EXPORT_SYMBOL(RTThreadStateName);
1523
1524#endif /* IN_RING3 */
1525#ifdef IPRT_WITH_GENERIC_TLS
1526
1527/**
1528 * Thread enumerator - clears a TLS entry.
1529 *
1530 * @returns 0.
1531 * @param pNode The thread node.
1532 * @param pvUser The TLS index.
1533 */
1534static DECLCALLBACK(int) rtThreadClearTlsEntryCallback(PAVLPVNODECORE pNode, void *pvUser)
1535{
1536 PRTTHREADINT pThread = (PRTTHREADINT)pNode;
1537 RTTLS iTls = (RTTLS)(uintptr_t)pvUser;
1538 ASMAtomicWriteNullPtr(&pThread->apvTlsEntries[iTls]);
1539 return 0;
1540}
1541
1542
1543/**
1544 * Helper for the generic TLS implementation that clears a given TLS
1545 * entry on all threads.
1546 *
1547 * @param iTls The TLS entry. (valid)
1548 */
1549DECLHIDDEN(void) rtThreadClearTlsEntry(RTTLS iTls)
1550{
1551 RT_THREAD_LOCK_RD();
1552 RTAvlPVDoWithAll(&g_ThreadTree, true /* fFromLeft*/, rtThreadClearTlsEntryCallback, (void *)(uintptr_t)iTls);
1553 RT_THREAD_UNLOCK_RD();
1554}
1555
1556#endif /* IPRT_WITH_GENERIC_TLS */
1557
1558
1559#if defined(RT_OS_WINDOWS) && defined(IN_RING3)
1560
1561/**
1562 * Thread enumeration callback for RTThreadNameThreads
1563 */
1564static DECLCALLBACK(int) rtThreadNameThreadCallback(PAVLPVNODECORE pNode, void *pvUser)
1565{
1566 PRTTHREADINT pThread = (PRTTHREADINT)pNode;
1567 rtThreadNativeInformDebugger(pThread);
1568 return 0;
1569}
1570
1571/**
1572 * A function that can be called from the windows debugger to get the names of
1573 * all threads when attaching to a process.
1574 *
1575 * Usage: .call VBoxRT!RTThreadNameThreads()
1576 *
1577 * @returns 0
1578 * @remarks Do not call from source code as it skips locks.
1579 */
1580extern "C" RTDECL(int) RTThreadNameThreads(void);
1581RTDECL(int) RTThreadNameThreads(void)
1582{
1583 return RTAvlPVDoWithAll(&g_ThreadTree, true /* fFromLeft*/, rtThreadNameThreadCallback, NULL);
1584}
1585
1586#endif
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