VirtualBox

source: vbox/trunk/src/VBox/Runtime/r3/posix/thread-posix.cpp@ 26957

Last change on this file since 26957 was 26363, checked in by vboxsync, 15 years ago

thread-posix.cpp: RTThreadPoke is optional, don't assert if the signal is in use. Build fix for IN_GUEST change.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id
File size: 10.6 KB
Line 
1/* $Id: thread-posix.cpp 26363 2010-02-09 13:26:17Z vboxsync $ */
2/** @file
3 * IPRT - Threads, POSIX.
4 */
5
6/*
7 * Copyright (C) 2006-2007 Sun Microsystems, Inc.
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 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa
27 * Clara, CA 95054 USA or visit http://www.sun.com if you need
28 * additional information or have any questions.
29 */
30
31
32/*******************************************************************************
33* Header Files *
34*******************************************************************************/
35#define LOG_GROUP RTLOGGROUP_THREAD
36#include <errno.h>
37#include <pthread.h>
38#include <signal.h>
39#if defined(RT_OS_LINUX)
40# include <unistd.h>
41# include <sys/syscall.h>
42#endif
43#if defined(RT_OS_SOLARIS)
44# include <sched.h>
45#endif
46
47#include <iprt/thread.h>
48#include <iprt/log.h>
49#include <iprt/assert.h>
50#include <iprt/asm.h>
51#include <iprt/err.h>
52#include <iprt/string.h>
53#include "internal/thread.h"
54
55
56/*******************************************************************************
57* Defined Constants And Macros *
58*******************************************************************************/
59#ifndef IN_GUEST
60/** The signal we're using for RTThreadPoke. */
61# define RTTHREAD_POSIX_POKE_SIG SIGUSR2
62#endif
63
64
65/*******************************************************************************
66* Global Variables *
67*******************************************************************************/
68/** The pthread key in which we store the pointer to our own PRTTHREAD structure. */
69static pthread_key_t g_SelfKey;
70#ifdef RTTHREAD_POSIX_POKE_SIG
71/** Set if we can poke a thread, clear if we cannot. */
72static bool g_fCanPokeThread;
73#endif
74
75
76/*******************************************************************************
77* Internal Functions *
78*******************************************************************************/
79static void *rtThreadNativeMain(void *pvArgs);
80static void rtThreadKeyDestruct(void *pvValue);
81static void rtThreadPosixPokeSignal(int iSignal);
82
83
84int rtThreadNativeInit(void)
85{
86 /*
87 * Allocate the TLS (key in posix terms) where we store the pointer to
88 * a threads RTTHREADINT structure.
89 */
90 int rc = pthread_key_create(&g_SelfKey, rtThreadKeyDestruct);
91 if (rc)
92 return VERR_NO_TLS_FOR_SELF;
93
94#ifdef RTTHREAD_POSIX_POKE_SIG
95 /*
96 * Try register the dummy signal handler for RTThreadPoke.
97 */
98 g_fCanPokeThread = false;
99 struct sigaction SigActOld;
100 if (!sigaction(RTTHREAD_POSIX_POKE_SIG, NULL, &SigActOld))
101 {
102 if ( SigActOld.sa_handler == SIG_DFL
103 || SigActOld.sa_handler == rtThreadPosixPokeSignal)
104 {
105 struct sigaction SigAct;
106 memset(&SigAct, '\0', sizeof(SigAct));
107 SigAct.sa_handler = rtThreadPosixPokeSignal;
108 sigfillset(&SigAct.sa_mask);
109 SigAct.sa_flags = 0;
110
111 /* ASSUMES no sigaction race... (lazy bird) */
112 if (!sigaction(RTTHREAD_POSIX_POKE_SIG, &SigAct, NULL))
113 g_fCanPokeThread = true;
114 else
115 {
116 AssertMsgFailed(("rc=%Rrc errno=%d\n", RTErrConvertFromErrno(errno), errno));
117 pthread_key_delete(g_SelfKey);
118 g_SelfKey = 0;
119 }
120 }
121 }
122 else
123 AssertMsgFailed(("rc=%Rrc errno=%d\n", RTErrConvertFromErrno(errno), errno));
124#endif /* RTTHREAD_POSIX_POKE_SIG */
125 return rc;
126}
127
128
129/**
130 * Destructor called when a thread terminates.
131 * @param pvValue The key value. PRTTHREAD in our case.
132 */
133static void rtThreadKeyDestruct(void *pvValue)
134{
135 /*
136 * Deal with alien threads.
137 */
138 PRTTHREADINT pThread = (PRTTHREADINT)pvValue;
139 if (pThread->fIntFlags & RTTHREADINT_FLAGS_ALIEN)
140 {
141 pthread_setspecific(g_SelfKey, pThread);
142 rtThreadTerminate(pThread, 0);
143 pthread_setspecific(g_SelfKey, NULL);
144 }
145}
146
147
148#ifdef RTTHREAD_POSIX_POKE_SIG
149/**
150 * Dummy signal handler for the poke signal.
151 *
152 * @param iSignal The signal number.
153 */
154static void rtThreadPosixPokeSignal(int iSignal)
155{
156 Assert(iSignal == RTTHREAD_POSIX_POKE_SIG);
157 NOREF(iSignal);
158}
159#endif
160
161
162/**
163 * Adopts a thread, this is called immediately after allocating the
164 * thread structure.
165 *
166 * @param pThread Pointer to the thread structure.
167 */
168int rtThreadNativeAdopt(PRTTHREADINT pThread)
169{
170 /*
171 * Block SIGALRM - required for timer-posix.cpp.
172 * This is done to limit harm done by OSes which doesn't do special SIGALRM scheduling.
173 * It will not help much if someone creates threads directly using pthread_create. :/
174 */
175 sigset_t SigSet;
176 sigemptyset(&SigSet);
177 sigaddset(&SigSet, SIGALRM);
178 sigprocmask(SIG_BLOCK, &SigSet, NULL);
179#ifdef RTTHREAD_POSIX_POKE_SIG
180 if (g_fCanPokeThread)
181 siginterrupt(RTTHREAD_POSIX_POKE_SIG, 1);
182#endif
183
184 int rc = pthread_setspecific(g_SelfKey, pThread);
185 if (!rc)
186 return VINF_SUCCESS;
187 return VERR_FAILED_TO_SET_SELF_TLS;
188}
189
190
191/**
192 * Wrapper which unpacks the params and calls thread function.
193 */
194static void *rtThreadNativeMain(void *pvArgs)
195{
196 PRTTHREADINT pThread = (PRTTHREADINT)pvArgs;
197
198#if defined(RT_OS_LINUX)
199 /*
200 * Set the TID.
201 */
202 pThread->tid = syscall(__NR_gettid);
203 ASMMemoryFence();
204#endif
205
206 /*
207 * Block SIGALRM - required for timer-posix.cpp.
208 * This is done to limit harm done by OSes which doesn't do special SIGALRM scheduling.
209 * It will not help much if someone creates threads directly using pthread_create. :/
210 */
211 sigset_t SigSet;
212 sigemptyset(&SigSet);
213 sigaddset(&SigSet, SIGALRM);
214 sigprocmask(SIG_BLOCK, &SigSet, NULL);
215#ifdef RTTHREAD_POSIX_POKE_SIG
216 if (g_fCanPokeThread)
217 siginterrupt(RTTHREAD_POSIX_POKE_SIG, 1);
218#endif
219
220 int rc = pthread_setspecific(g_SelfKey, pThread);
221 AssertReleaseMsg(!rc, ("failed to set self TLS. rc=%d thread '%s'\n", rc, pThread->szName));
222
223 /*
224 * Call common main.
225 */
226 pthread_t Self = pthread_self();
227 Assert((uintptr_t)Self == (RTNATIVETHREAD)Self && (uintptr_t)Self != NIL_RTNATIVETHREAD);
228 rc = rtThreadMain(pThread, (uintptr_t)Self, &pThread->szName[0]);
229
230 pthread_setspecific(g_SelfKey, NULL);
231 pthread_exit((void *)rc);
232 return (void *)rc;
233}
234
235
236int rtThreadNativeCreate(PRTTHREADINT pThread, PRTNATIVETHREAD pNativeThread)
237{
238 /*
239 * Set the default stack size.
240 */
241 if (!pThread->cbStack)
242 pThread->cbStack = 512*1024;
243
244#ifdef RT_OS_LINUX
245 pThread->tid = -1;
246#endif
247
248 /*
249 * Setup thread attributes.
250 */
251 pthread_attr_t ThreadAttr;
252 int rc = pthread_attr_init(&ThreadAttr);
253 if (!rc)
254 {
255 rc = pthread_attr_setdetachstate(&ThreadAttr, PTHREAD_CREATE_DETACHED);
256 if (!rc)
257 {
258 rc = pthread_attr_setstacksize(&ThreadAttr, pThread->cbStack);
259 if (!rc)
260 {
261 /*
262 * Create the thread.
263 */
264 pthread_t ThreadId;
265 rc = pthread_create(&ThreadId, &ThreadAttr, rtThreadNativeMain, pThread);
266 if (!rc)
267 {
268 *pNativeThread = (uintptr_t)ThreadId;
269 return VINF_SUCCESS;
270 }
271 }
272 }
273 pthread_attr_destroy(&ThreadAttr);
274 }
275 return RTErrConvertFromErrno(rc);
276}
277
278
279RTDECL(RTTHREAD) RTThreadSelf(void)
280{
281 PRTTHREADINT pThread = (PRTTHREADINT)pthread_getspecific(g_SelfKey);
282 /** @todo import alien threads? */
283 return pThread;
284}
285
286
287RTDECL(RTNATIVETHREAD) RTThreadNativeSelf(void)
288{
289 return (RTNATIVETHREAD)pthread_self();
290}
291
292
293RTDECL(int) RTThreadSleep(RTMSINTERVAL cMillies)
294{
295 LogFlow(("RTThreadSleep: cMillies=%d\n", cMillies));
296 if (!cMillies)
297 {
298 /* pthread_yield() isn't part of SuS, thus this fun. */
299#ifdef RT_OS_DARWIN
300 pthread_yield_np();
301#elif defined(RT_OS_FREEBSD) /* void pthread_yield */
302 pthread_yield();
303#elif defined(RT_OS_SOLARIS)
304 sched_yield();
305#else
306 if (!pthread_yield())
307#endif
308 {
309 LogFlow(("RTThreadSleep: returning %Rrc (cMillies=%d)\n", VINF_SUCCESS, cMillies));
310 return VINF_SUCCESS;
311 }
312 }
313 else
314 {
315 struct timespec ts;
316 struct timespec tsrem = {0,0};
317
318 ts.tv_nsec = (cMillies % 1000) * 1000000;
319 ts.tv_sec = cMillies / 1000;
320 if (!nanosleep(&ts, &tsrem))
321 {
322 LogFlow(("RTThreadSleep: returning %Rrc (cMillies=%d)\n", VINF_SUCCESS, cMillies));
323 return VINF_SUCCESS;
324 }
325 }
326
327 int rc = RTErrConvertFromErrno(errno);
328 LogFlow(("RTThreadSleep: returning %Rrc (cMillies=%d)\n", rc, cMillies));
329 return rc;
330}
331
332
333RTDECL(bool) RTThreadYield(void)
334{
335 uint64_t u64TS = ASMReadTSC();
336#ifdef RT_OS_DARWIN
337 pthread_yield_np();
338#elif defined(RT_OS_SOLARIS)
339 sched_yield();
340#else
341 pthread_yield();
342#endif
343 u64TS = ASMReadTSC() - u64TS;
344 bool fRc = u64TS > 1500;
345 LogFlow(("RTThreadYield: returning %d (%llu ticks)\n", fRc, u64TS));
346 return fRc;
347}
348
349
350RTR3DECL(uint64_t) RTThreadGetAffinity(void)
351{
352 return 1;
353}
354
355
356RTR3DECL(int) RTThreadSetAffinity(uint64_t u64Mask)
357{
358 if (u64Mask != 1)
359 return VERR_INVALID_PARAMETER;
360 return VINF_SUCCESS;
361}
362
363
364#ifdef RTTHREAD_POSIX_POKE_SIG
365RTDECL(int) RTThreadPoke(RTTHREAD hThread)
366{
367 AssertReturn(hThread != RTThreadSelf(), VERR_INVALID_PARAMETER);
368 PRTTHREADINT pThread = rtThreadGet(hThread);
369 AssertReturn(pThread, VERR_INVALID_HANDLE);
370 int rc;
371 if (g_fCanPokeThread)
372 {
373 rc = pthread_kill((pthread_t)(uintptr_t)pThread->Core.Key, RTTHREAD_POSIX_POKE_SIG);
374 rc = RTErrConvertFromErrno(rc);
375 }
376 else
377 rc = VERR_NOT_SUPPORTED;
378
379 rtThreadRelease(pThread);
380 return rc;
381}
382#endif
383
Note: See TracBrowser for help on using the repository browser.

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