VirtualBox

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

Last change on this file since 45733 was 45375, checked in by vboxsync, 12 years ago

RTPathParse -> RTPathParseSimple.

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