VirtualBox

source: vbox/trunk/src/VBox/Runtime/r3/os2/thread-os2.cpp@ 46231

Last change on this file since 46231 was 44528, checked in by vboxsync, 12 years ago

header (C) fixes

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id
File size: 7.6 KB
Line 
1/* $Id: thread-os2.cpp 44528 2013-02-04 14:27:54Z vboxsync $ */
2/** @file
3 * IPRT - Threads, OS/2.
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#define INCL_BASE
33#include <os2.h>
34#undef RT_MAX
35
36#include <errno.h>
37#include <process.h>
38#include <stdlib.h>
39#include <signal.h>
40#include <InnoTekLIBC/FastInfoBlocks.h>
41#include <InnoTekLIBC/thread.h>
42
43#include <iprt/thread.h>
44#include <iprt/log.h>
45#include <iprt/assert.h>
46#include <iprt/alloc.h>
47#include <iprt/asm-amd64-x86.h>
48#include <iprt/cpuset.h>
49#include <iprt/string.h>
50#include <iprt/err.h>
51#include "internal/thread.h"
52
53
54/*******************************************************************************
55* Global Variables *
56*******************************************************************************/
57/** Pointer to thread local memory which points to the current thread. */
58static PRTTHREADINT *g_ppCurThread;
59
60
61/*******************************************************************************
62* Internal Functions *
63*******************************************************************************/
64static void rtThreadNativeMain(void *pvArgs);
65
66
67DECLHIDDEN(int) rtThreadNativeInit(void)
68{
69 /*
70 * Allocate thread local memory.
71 */
72 PULONG pul;
73 int rc = DosAllocThreadLocalMemory(1, &pul);
74 if (rc)
75 return VERR_NO_TLS_FOR_SELF;
76 g_ppCurThread = (PRTTHREADINT *)(void *)pul;
77 return VINF_SUCCESS;
78}
79
80
81DECLHIDDEN(int) rtThreadNativeAdopt(PRTTHREADINT pThread)
82{
83 /*
84 * Block SIGALRM - required for timer-posix.cpp.
85 * This is done to limit harm done by OSes which doesn't do special SIGALRM scheduling.
86 * It will not help much if someone creates threads directly using pthread_create. :/
87 */
88 sigset_t SigSet;
89 sigemptyset(&SigSet);
90 sigaddset(&SigSet, SIGALRM);
91 sigprocmask(SIG_BLOCK, &SigSet, NULL);
92
93 *g_ppCurThread = pThread;
94 return VINF_SUCCESS;
95}
96
97
98DECLHIDDEN(void) rtThreadNativeDestroy(PRTTHREADINT pThread)
99{
100 if (pThread == *g_ppCurThread)
101 *g_ppCurThread = NULL;
102}
103
104
105/**
106 * Wrapper which unpacks the params and calls thread function.
107 */
108static void rtThreadNativeMain(void *pvArgs)
109{
110 /*
111 * Block SIGALRM - required for timer-posix.cpp.
112 * This is done to limit harm done by OSes which doesn't do special SIGALRM scheduling.
113 * It will not help much if someone creates threads directly using pthread_create. :/
114 */
115 sigset_t SigSet;
116 sigemptyset(&SigSet);
117 sigaddset(&SigSet, SIGALRM);
118 sigprocmask(SIG_BLOCK, &SigSet, NULL);
119
120 /*
121 * Call common main.
122 */
123 PRTTHREADINT pThread = (PRTTHREADINT)pvArgs;
124 *g_ppCurThread = pThread;
125
126#ifdef fibGetTidPid
127 rtThreadMain(pThread, fibGetTidPid(), &pThread->szName[0]);
128#else
129 rtThreadMain(pThread, _gettid(), &pThread->szName[0]);
130#endif
131
132 *g_ppCurThread = NULL;
133 _endthread();
134}
135
136
137DECLHIDDEN(int) rtThreadNativeCreate(PRTTHREADINT pThread, PRTNATIVETHREAD pNativeThread)
138{
139 /*
140 * Default stack size.
141 */
142 if (!pThread->cbStack)
143 pThread->cbStack = 512*1024;
144
145 /*
146 * Create the thread.
147 */
148 int iThreadId = _beginthread(rtThreadNativeMain, NULL, pThread->cbStack, pThread);
149 if (iThreadId > 0)
150 {
151#ifdef fibGetTidPid
152 *pNativeThread = iThreadId | (fibGetPid() << 16);
153#else
154 *pNativeThread = iThreadId;
155#endif
156 return VINF_SUCCESS;
157 }
158 return RTErrConvertFromErrno(errno);
159}
160
161
162RTDECL(RTTHREAD) RTThreadSelf(void)
163{
164 PRTTHREADINT pThread = *g_ppCurThread;
165 if (pThread)
166 return (RTTHREAD)pThread;
167 /** @todo import alien threads? */
168 return NULL;
169}
170
171
172RTDECL(RTNATIVETHREAD) RTThreadNativeSelf(void)
173{
174#ifdef fibGetTidPid
175 return fibGetTidPid();
176#else
177 return _gettid();
178#endif
179}
180
181
182RTDECL(int) RTThreadSleep(RTMSINTERVAL cMillies)
183{
184 LogFlow(("RTThreadSleep: cMillies=%d\n", cMillies));
185 DosSleep(cMillies);
186 LogFlow(("RTThreadSleep: returning (cMillies=%d)\n", cMillies));
187 return VINF_SUCCESS;
188}
189
190
191RTDECL(int) RTThreadSleepNoLog(RTMSINTERVAL cMillies)
192{
193 DosSleep(cMillies);
194 return VINF_SUCCESS;
195}
196
197
198RTDECL(bool) RTThreadYield(void)
199{
200 uint64_t u64TS = ASMReadTSC();
201 DosSleep(0);
202 u64TS = ASMReadTSC() - u64TS;
203 bool fRc = u64TS > 1750;
204 LogFlow(("RTThreadYield: returning %d (%llu ticks)\n", fRc, u64TS));
205 return fRc;
206}
207
208
209RTR3DECL(int) RTThreadGetAffinity(PRTCPUSET pCpuSet)
210{
211 union
212 {
213 uint64_t u64;
214 MPAFFINITY mpaff;
215 } u;
216
217 APIRET rc = DosQueryThreadAffinity(AFNTY_THREAD, &u.mpaff);
218 if (!rc)
219 {
220 RTCpuSetFromU64(pCpuSet, u.u64);
221 return VINF_SUCCESS;
222 }
223 return RTErrConvertFromOS2(rc);
224}
225
226
227RTR3DECL(int) RTThreadSetAffinity(PCRTCPUSET pCpuSet)
228{
229 union
230 {
231 uint64_t u64;
232 MPAFFINITY mpaff;
233 } u;
234 u.u64 = pCpuSet ? RTCpuSetToU64(pCpuSet) : UINT64_MAX;
235 int rc = DosSetThreadAffinity(&u.mpaff);
236 if (!rc)
237 return VINF_SUCCESS;
238 return RTErrConvertFromOS2(rc);
239}
240
241
242RTR3DECL(RTTLS) RTTlsAlloc(void)
243{
244 AssertCompile(NIL_RTTLS == -1);
245 return __libc_TLSAlloc();
246}
247
248
249RTR3DECL(int) RTTlsAllocEx(PRTTLS piTls, PFNRTTLSDTOR pfnDestructor)
250{
251 int rc;
252 int iTls = __libc_TLSAlloc();
253 if (iTls != -1)
254 {
255 if ( !pfnDestructor
256 || __libc_TLSDestructor(iTls, (void (*)(void *, int, unsigned))pfnDestructor, 0) != -1)
257 {
258 *piTls = iTls;
259 return VINF_SUCCESS;
260 }
261
262 rc = RTErrConvertFromErrno(errno);
263 __libc_TLSFree(iTls);
264 }
265 else
266 rc = RTErrConvertFromErrno(errno);
267
268 *piTls = NIL_RTTLS;
269 return rc;
270}
271
272
273RTR3DECL(int) RTTlsFree(RTTLS iTls)
274{
275 if (iTls == NIL_RTTLS)
276 return VINF_SUCCESS;
277 if (__libc_TLSFree(iTls) != -1)
278 return VINF_SUCCESS;
279 return RTErrConvertFromErrno(errno);
280}
281
282
283RTR3DECL(void *) RTTlsGet(RTTLS iTls)
284{
285 return __libc_TLSGet(iTls);
286}
287
288
289RTR3DECL(int) RTTlsGetEx(RTTLS iTls, void **ppvValue)
290{
291 int rc = VINF_SUCCESS;
292 void *pv = __libc_TLSGet(iTls);
293 if (RT_UNLIKELY(!pv))
294 {
295 errno = 0;
296 pv = __libc_TLSGet(iTls);
297 if (!pv && errno)
298 rc = RTErrConvertFromErrno(errno);
299 }
300
301 *ppvValue = pv;
302 return rc;
303}
304
305
306RTR3DECL(int) RTTlsSet(RTTLS iTls, void *pvValue)
307{
308 if (__libc_TLSSet(iTls, pvValue) != -1)
309 return VINF_SUCCESS;
310 return RTErrConvertFromErrno(errno);
311}
312
313
314RTR3DECL(int) RTThreadGetExecutionTimeMilli(uint64_t *pKernelTime, uint64_t *pUserTime)
315{
316 return VERR_NOT_IMPLEMENTED;
317}
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