VirtualBox

source: vbox/trunk/src/VBox/Runtime/r3/init.cpp@ 39668

Last change on this file since 39668 was 38658, checked in by vboxsync, 13 years ago

IPRT: , and are now deprecated, the conversion to local codeset being moved to the streams (RTPrintf / RTStrmPrintf). Adding special detection of the windows console and talk UTF-16 to it in order to avoid lost-in-translation issues when its codepage differs from the active codepage of the process.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id
File size: 16.0 KB
Line 
1/* $Id: init.cpp 38658 2011-09-06 14:22:53Z vboxsync $ */
2/** @file
3 * IPRT - Init Ring-3.
4 */
5
6/*
7 * Copyright (C) 2006-2009 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_DEFAULT
32#include <iprt/types.h> /* darwin: UINT32_C and others. */
33
34#ifdef RT_OS_WINDOWS
35# include <process.h>
36#else
37# include <unistd.h>
38# ifndef RT_OS_OS2
39# include <pthread.h>
40# include <signal.h>
41# include <errno.h>
42# define IPRT_USE_SIG_CHILD_DUMMY
43# endif
44#endif
45#ifdef RT_OS_OS2
46# include <InnoTekLIBC/fork.h>
47#endif
48#include <locale.h>
49
50#include <iprt/initterm.h>
51#include <iprt/asm.h>
52#include <iprt/assert.h>
53#include <iprt/err.h>
54#include <iprt/log.h>
55#include <iprt/mem.h>
56#include <iprt/path.h>
57#include <iprt/time.h>
58#include <iprt/string.h>
59#include <iprt/param.h>
60#if !defined(IN_GUEST) && !defined(RT_NO_GIP)
61# include <iprt/file.h>
62# include <VBox/sup.h>
63#endif
64#include <stdlib.h>
65
66#include "internal/alignmentchecks.h"
67#include "internal/path.h"
68#include "internal/process.h"
69#include "internal/thread.h"
70#include "internal/thread.h"
71#include "internal/time.h"
72
73
74/*******************************************************************************
75* Global Variables *
76*******************************************************************************/
77/** The number of calls to RTR3Init*. */
78static int32_t volatile g_cUsers = 0;
79/** Whether we're currently initializing the IPRT. */
80static bool volatile g_fInitializing = false;
81
82/** The process path.
83 * This is used by RTPathExecDir and RTProcGetExecutablePath and set by rtProcInitName. */
84DECLHIDDEN(char) g_szrtProcExePath[RTPATH_MAX];
85/** The length of g_szrtProcExePath. */
86DECLHIDDEN(size_t) g_cchrtProcExePath;
87/** The length of directory path component of g_szrtProcExePath. */
88DECLHIDDEN(size_t) g_cchrtProcDir;
89/** The offset of the process name into g_szrtProcExePath. */
90DECLHIDDEN(size_t) g_offrtProcName;
91
92/** The argument count of the program. */
93static int g_crtArgs = -1;
94/** The arguments of the program (UTF-8). This is "leaked". */
95static char ** g_papszrtArgs;
96/** The original argument vector of the program. */
97static char ** g_papszrtOrgArgs;
98
99/**
100 * Program start nanosecond TS.
101 */
102DECLHIDDEN(uint64_t) g_u64ProgramStartNanoTS;
103
104/**
105 * Program start microsecond TS.
106 */
107DECLHIDDEN(uint64_t) g_u64ProgramStartMicroTS;
108
109/**
110 * Program start millisecond TS.
111 */
112DECLHIDDEN(uint64_t) g_u64ProgramStartMilliTS;
113
114/**
115 * The process identifier of the running process.
116 */
117DECLHIDDEN(RTPROCESS) g_ProcessSelf = NIL_RTPROCESS;
118
119/**
120 * The current process priority.
121 */
122DECLHIDDEN(RTPROCPRIORITY) g_enmProcessPriority = RTPROCPRIORITY_DEFAULT;
123
124/**
125 * Set if the atexit callback has been called, i.e. indicating
126 * that the process is terminating.
127 */
128DECLHIDDEN(bool volatile) g_frtAtExitCalled = false;
129
130#ifdef IPRT_WITH_ALIGNMENT_CHECKS
131/**
132 * Whether alignment checks are enabled.
133 * This is set if the environment variable IPRT_ALIGNMENT_CHECKS is 1.
134 */
135RTDATADECL(bool) g_fRTAlignmentChecks = false;
136#endif
137
138
139/**
140 * atexit callback.
141 *
142 * This makes sure any loggers are flushed and will later also work the
143 * termination callback chain.
144 */
145static void rtR3ExitCallback(void)
146{
147 ASMAtomicWriteBool(&g_frtAtExitCalled, true);
148
149 if (g_cUsers > 0)
150 {
151 PRTLOGGER pLogger = RTLogGetDefaultInstance();
152 if (pLogger)
153 RTLogFlush(pLogger);
154
155 pLogger = RTLogRelDefaultInstance();
156 if (pLogger)
157 RTLogFlush(pLogger);
158 }
159}
160
161
162#ifndef RT_OS_WINDOWS
163/**
164 * Fork callback, child context.
165 */
166static void rtR3ForkChildCallback(void)
167{
168 g_ProcessSelf = getpid();
169}
170#endif /* RT_OS_WINDOWS */
171
172#ifdef RT_OS_OS2
173/** Fork completion callback for OS/2. Only called in the child. */
174static void rtR3ForkOs2ChildCompletionCallback(void *pvArg, int rc, __LIBC_FORKCTX enmCtx)
175{
176 Assert(enmCtx == __LIBC_FORK_CTX_CHILD); NOREF(enmCtx);
177 NOREF(pvArg);
178
179 if (!rc)
180 rtR3ForkChildCallback();
181}
182
183/** Low-level fork callback for OS/2. */
184int rtR3ForkOs2Child(__LIBC_PFORKHANDLE pForkHandle, __LIBC_FORKOP enmOperation)
185{
186 if (enmOperation == __LIBC_FORK_OP_EXEC_CHILD)
187 return pForkHandle->pfnCompletionCallback(pForkHandle, rtR3ForkOs2ChildCompletionCallback, NULL, __LIBC_FORK_CTX_CHILD);
188 return 0;
189}
190
191_FORK_CHILD1(0, rtR3ForkOs2Child);
192#endif /* RT_OS_OS2 */
193
194
195
196/**
197 * Internal worker which initializes or re-initializes the
198 * program path, name and directory globals.
199 *
200 * @returns IPRT status code.
201 * @param pszProgramPath The program path, NULL if not specified.
202 */
203static int rtR3InitProgramPath(const char *pszProgramPath)
204{
205 /*
206 * We're reserving 32 bytes here for file names as what not.
207 */
208 if (!pszProgramPath)
209 {
210 int rc = rtProcInitExePath(g_szrtProcExePath, sizeof(g_szrtProcExePath) - 32);
211 if (RT_FAILURE(rc))
212 return rc;
213 }
214 else
215 {
216 size_t cch = strlen(pszProgramPath);
217 Assert(cch > 1);
218 AssertMsgReturn(cch < sizeof(g_szrtProcExePath) - 32, ("%zu\n", cch), VERR_BUFFER_OVERFLOW);
219 memcpy(g_szrtProcExePath, pszProgramPath, cch + 1);
220 }
221
222 /*
223 * Parse the name.
224 */
225 ssize_t offName;
226 g_cchrtProcExePath = RTPathParse(g_szrtProcExePath, &g_cchrtProcDir, &offName, NULL);
227 g_offrtProcName = offName;
228 return VINF_SUCCESS;
229}
230
231
232/**
233 * Internal worker which initializes or re-initializes the
234 * program path, name and directory globals.
235 *
236 * @returns IPRT status code.
237 * @param fFlags Flags, see RTR3INIT_XXX.
238 * @param cArgs Pointer to the argument count.
239 * @param ppapszArgs Pointer to the argument vector pointer. NULL
240 * allowed if @a cArgs is 0.
241 */
242static int rtR3InitArgv(uint32_t fFlags, int cArgs, char ***ppapszArgs)
243{
244 NOREF(fFlags);
245 if (cArgs)
246 {
247 AssertPtr(ppapszArgs);
248 AssertPtr(*ppapszArgs);
249 char **papszOrgArgs = *ppapszArgs;
250
251 /*
252 * Normally we should only be asked to convert arguments once. If we
253 * are though, it should be the already convered arguments.
254 */
255 if (g_crtArgs != -1)
256 {
257 AssertReturn( g_crtArgs == cArgs
258 && g_papszrtArgs == papszOrgArgs,
259 VERR_WRONG_ORDER); /* only init once! */
260 return VINF_SUCCESS;
261 }
262
263 /*
264 * Convert the arguments.
265 */
266 char **papszArgs = (char **)RTMemAllocZ((cArgs + 1) * sizeof(char *));
267 if (!papszArgs)
268 return VERR_NO_MEMORY;
269
270 for (int i = 0; i < cArgs; i++)
271 {
272 int rc = RTStrCurrentCPToUtf8(&papszArgs[i], papszOrgArgs[i]);
273 if (RT_FAILURE(rc))
274 {
275 while (i--)
276 RTStrFree(papszArgs[i]);
277 RTMemFree(papszArgs);
278 return rc;
279 }
280 }
281 papszArgs[cArgs] = NULL;
282
283 g_papszrtOrgArgs = papszOrgArgs;
284 g_papszrtArgs = papszArgs;
285 g_crtArgs = cArgs;
286
287 *ppapszArgs = papszArgs;
288 }
289
290 return VINF_SUCCESS;
291}
292
293
294#ifdef IPRT_USE_SIG_CHILD_DUMMY
295/**
296 * Dummy SIGCHILD handler.
297 *
298 * Assigned on rtR3Init only when SIGCHILD handler is set SIGIGN or SIGDEF to
299 * ensure waitpid works properly for the terminated processes.
300 */
301static void rtR3SigChildHandler(int iSignal)
302{
303 NOREF(iSignal);
304}
305#endif /* IPRT_USE_SIG_CHILD_DUMMY */
306
307
308/**
309 * rtR3Init worker.
310 */
311static int rtR3InitBody(uint32_t fFlags, int cArgs, char ***papszArgs, const char *pszProgramPath)
312{
313 /*
314 * Init C runtime locale before we do anything that may end up converting
315 * paths or we'll end up using the "C" locale for path conversion.
316 */
317 setlocale(LC_CTYPE, "");
318
319 /*
320 * The Process ID.
321 */
322#ifdef _MSC_VER
323 g_ProcessSelf = _getpid(); /* crappy ansi compiler */
324#else
325 g_ProcessSelf = getpid();
326#endif
327
328#if !defined(IN_GUEST) && !defined(RT_NO_GIP)
329# ifdef VBOX
330 /*
331 * This MUST be done as the very first thing, before any file is opened.
332 * The log is opened on demand, but the first log entries may be caused
333 * by rtThreadInit() below.
334 */
335 const char *pszDisableHostCache = getenv("VBOX_DISABLE_HOST_DISK_CACHE");
336 if ( pszDisableHostCache != NULL
337 && *pszDisableHostCache
338 && strcmp(pszDisableHostCache, "0") != 0)
339 {
340 RTFileSetForceFlags(RTFILE_O_WRITE, RTFILE_O_WRITE_THROUGH, 0);
341 RTFileSetForceFlags(RTFILE_O_READWRITE, RTFILE_O_WRITE_THROUGH, 0);
342 }
343# endif /* VBOX */
344#endif /* !IN_GUEST && !RT_NO_GIP */
345
346 /*
347 * Thread Thread database and adopt the caller thread as 'main'.
348 * This must be done before everything else or else we'll call into threading
349 * without having initialized TLS entries and suchlike.
350 */
351 int rc = rtThreadInit();
352 AssertMsgRCReturn(rc, ("Failed to initialize threads, rc=%Rrc!\n", rc), rc);
353
354#if !defined(IN_GUEST) && !defined(RT_NO_GIP)
355 if (fFlags & RTR3INIT_FLAGS_SUPLIB)
356 {
357 /*
358 * Init GIP first.
359 * (The more time for updates before real use, the better.)
360 */
361 rc = SUPR3Init(NULL);
362 AssertMsgRCReturn(rc, ("Failed to initializable the support library, rc=%Rrc!\n", rc), rc);
363 }
364#endif
365
366 /*
367 * The executable path, name and directory. Convert arguments.
368 */
369 rc = rtR3InitProgramPath(pszProgramPath);
370 AssertLogRelMsgRCReturn(rc, ("Failed to get executable directory path, rc=%Rrc!\n", rc), rc);
371
372 rc = rtR3InitArgv(fFlags, cArgs, papszArgs);
373 AssertLogRelMsgRCReturn(rc, ("Failed to convert the arguments, rc=%Rrc!\n", rc), rc);
374
375#if !defined(IN_GUEST) && !defined(RT_NO_GIP)
376 /*
377 * The threading is initialized we can safely sleep a bit if GIP
378 * needs some time to update itself updating.
379 */
380 if ((fFlags & RTR3INIT_FLAGS_SUPLIB) && g_pSUPGlobalInfoPage)
381 {
382 RTThreadSleep(20);
383 RTTimeNanoTS();
384 }
385#endif
386
387 /*
388 * Init the program start TSes.
389 * Do that here to be sure that the GIP time was properly updated the 1st time.
390 */
391 g_u64ProgramStartNanoTS = RTTimeNanoTS();
392 g_u64ProgramStartMicroTS = g_u64ProgramStartNanoTS / 1000;
393 g_u64ProgramStartMilliTS = g_u64ProgramStartNanoTS / 1000000;
394
395 /*
396 * The remainder cannot easily be undone, so it has to go last.
397 */
398
399 /* Fork and exit callbacks. */
400#if !defined(RT_OS_WINDOWS) && !defined(RT_OS_OS2)
401 rc = pthread_atfork(NULL, NULL, rtR3ForkChildCallback);
402 AssertMsg(rc == 0, ("%d\n", rc));
403#endif
404 atexit(rtR3ExitCallback);
405
406#ifdef IPRT_USE_SIG_CHILD_DUMMY
407 /*
408 * SIGCHLD must not be ignored (that's default), otherwise posix compliant waitpid
409 * implementations won't work right.
410 */
411 for (;;)
412 {
413 struct sigaction saOld;
414 rc = sigaction(SIGCHLD, 0, &saOld); AssertMsg(rc == 0, ("%d/%d\n", rc, errno));
415 if ( rc != 0
416 || (saOld.sa_flags & SA_SIGINFO)
417 || ( saOld.sa_handler != SIG_IGN
418 && saOld.sa_handler != SIG_DFL)
419 )
420 break;
421
422 /* Try install dummy handler. */
423 struct sigaction saNew = saOld;
424 saNew.sa_flags = SA_NOCLDSTOP | SA_RESTART;
425 saNew.sa_handler = rtR3SigChildHandler;
426 rc = sigemptyset(&saNew.sa_mask); AssertMsg(rc == 0, ("%d/%d\n", rc, errno));
427 struct sigaction saOld2;
428 rc = sigaction(SIGCHLD, &saNew, &saOld2); AssertMsg(rc == 0, ("%d/%d\n", rc, errno));
429 if ( rc != 0
430 || ( saOld2.sa_handler == saOld.sa_handler
431 && !(saOld2.sa_flags & SA_SIGINFO))
432 )
433 break;
434
435 /* Race during dynamic load, restore and try again... */
436 sigaction(SIGCHLD, &saOld2, NULL);
437 RTThreadYield();
438 }
439#endif /* IPRT_USE_SIG_CHILD_DUMMY */
440
441#ifdef IPRT_WITH_ALIGNMENT_CHECKS
442 /*
443 * Enable alignment checks.
444 */
445 const char *pszAlignmentChecks = getenv("IPRT_ALIGNMENT_CHECKS");
446 g_fRTAlignmentChecks = pszAlignmentChecks != NULL
447 && pszAlignmentChecks[0] == '1'
448 && pszAlignmentChecks[1] == '\0';
449 if (g_fRTAlignmentChecks)
450 IPRT_ALIGNMENT_CHECKS_ENABLE();
451#endif
452
453 return VINF_SUCCESS;
454}
455
456
457/**
458 * Internal initialization worker.
459 *
460 * @returns IPRT status code.
461 * @param fFlags Flags, see RTR3INIT_XXX.
462 * @param cArgs Pointer to the argument count.
463 * @param ppapszArgs Pointer to the argument vector pointer. NULL
464 * allowed if @a cArgs is 0.
465 * @param pszProgramPath The program path. Pass NULL if we're to figure it
466 * out ourselves.
467 */
468static int rtR3Init(uint32_t fFlags, int cArgs, char ***papszArgs, const char *pszProgramPath)
469{
470 /* no entry log flow, because prefixes and thread may freak out. */
471 Assert(!(fFlags & ~(RTR3INIT_FLAGS_DLL | RTR3INIT_FLAGS_SUPLIB)));
472 Assert(!(fFlags & RTR3INIT_FLAGS_DLL) || cArgs == 0);
473
474 /*
475 * Do reference counting, only initialize the first time around.
476 *
477 * We are ASSUMING that nobody will be able to race RTR3Init* calls when the
478 * first one, the real init, is running (second assertion).
479 */
480 int32_t cUsers = ASMAtomicIncS32(&g_cUsers);
481 if (cUsers != 1)
482 {
483 AssertMsg(cUsers > 1, ("%d\n", cUsers));
484 Assert(!g_fInitializing);
485#if !defined(IN_GUEST) && !defined(RT_NO_GIP)
486 if (fFlags & RTR3INIT_FLAGS_SUPLIB)
487 SUPR3Init(NULL);
488#endif
489 if (!pszProgramPath)
490 return VINF_SUCCESS;
491
492 int rc = rtR3InitProgramPath(pszProgramPath);
493 if (RT_SUCCESS(rc))
494 rc = rtR3InitArgv(fFlags, cArgs, papszArgs);
495 return rc;
496 }
497 ASMAtomicWriteBool(&g_fInitializing, true);
498
499 /*
500 * Do the initialization.
501 */
502 int rc = rtR3InitBody(fFlags, cArgs, papszArgs, pszProgramPath);
503 if (RT_FAILURE(rc))
504 {
505 /* failure */
506 ASMAtomicWriteBool(&g_fInitializing, false);
507 ASMAtomicDecS32(&g_cUsers);
508 return rc;
509 }
510
511 /* success */
512 LogFlow(("rtR3Init: returns VINF_SUCCESS\n"));
513 ASMAtomicWriteBool(&g_fInitializing, false);
514 return VINF_SUCCESS;
515}
516
517
518RTR3DECL(int) RTR3InitExe(int cArgs, char ***papszArgs, uint32_t fFlags)
519{
520 Assert(!(fFlags & RTR3INIT_FLAGS_DLL));
521 return rtR3Init(fFlags, cArgs, papszArgs, NULL);
522}
523
524
525RTR3DECL(int) RTR3InitExeNoArguments(uint32_t fFlags)
526{
527 Assert(!(fFlags & RTR3INIT_FLAGS_DLL));
528 return rtR3Init(fFlags, 0, NULL, NULL);
529}
530
531
532RTR3DECL(int) RTR3InitDll(uint32_t fFlags)
533{
534 Assert(!(fFlags & RTR3INIT_FLAGS_DLL));
535 return rtR3Init(fFlags | RTR3INIT_FLAGS_DLL, 0, NULL, NULL);
536}
537
538
539RTR3DECL(int) RTR3InitEx(uint32_t iVersion, uint32_t fFlags, int cArgs, char ***papszArgs, const char *pszProgramPath)
540{
541 AssertReturn(iVersion == RTR3INIT_VER_CUR, VERR_NOT_SUPPORTED);
542 return rtR3Init(fFlags, cArgs, papszArgs, pszProgramPath);
543}
544
545
546#if 0 /** @todo implement RTR3Term. */
547RTR3DECL(void) RTR3Term(void)
548{
549}
550#endif
551
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