VirtualBox

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

Last change on this file since 20606 was 19924, checked in by vboxsync, 15 years ago

IPRT,SUP: Renamed RTPathProgram to RTPathExecDir to make it clear what it returns. Renamed hardened version of it as well.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id
File size: 9.7 KB
Line 
1/* $Id: init.cpp 19924 2009-05-22 21:52:47Z vboxsync $ */
2/** @file
3 * IPRT - Init Ring-3.
4 */
5
6/*
7 * Copyright (C) 2006-2009 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_DEFAULT
36#ifdef RT_OS_WINDOWS
37# include <process.h>
38#else
39# include <unistd.h>
40# ifndef RT_OS_OS2
41# include <pthread.h>
42# endif
43#endif
44#ifdef RT_OS_OS2
45# include <InnoTekLIBC/fork.h>
46#endif
47#include <locale.h>
48
49#include <iprt/initterm.h>
50#include <iprt/asm.h>
51#include <iprt/assert.h>
52#include <iprt/err.h>
53#include <iprt/log.h>
54#include <iprt/path.h>
55#include <iprt/time.h>
56#include <iprt/string.h>
57#include <iprt/param.h>
58#if !defined(IN_GUEST) && !defined(RT_NO_GIP)
59# include <iprt/file.h>
60# include <VBox/sup.h>
61# include <stdlib.h>
62#endif
63
64#include "internal/path.h"
65#include "internal/process.h"
66#include "internal/thread.h"
67#include "internal/thread.h"
68#include "internal/time.h"
69
70
71/*******************************************************************************
72* Global Variables *
73*******************************************************************************/
74/** The number of calls to RTR3Init. */
75static int32_t volatile g_cUsers = 0;
76/** Whether we're currently initializing the IPRT. */
77static bool volatile g_fInitializing = false;
78
79/** The process path.
80 * This is used by RTPathExecDir and RTProcGetExecutableName and set by rtProcInitName. */
81char g_szrtProcExePath[RTPATH_MAX];
82/** The length of g_szrtProcExePath. */
83size_t g_cchrtProcExePath;
84/** The length of directory path component of g_szrtProcExePath. */
85size_t g_cchrtProcDir;
86/** The offset of the process name into g_szrtProcExePath. */
87size_t g_offrtProcName;
88
89/**
90 * Program start nanosecond TS.
91 */
92uint64_t g_u64ProgramStartNanoTS;
93
94/**
95 * Program start microsecond TS.
96 */
97uint64_t g_u64ProgramStartMicroTS;
98
99/**
100 * Program start millisecond TS.
101 */
102uint64_t g_u64ProgramStartMilliTS;
103
104/**
105 * The process identifier of the running process.
106 */
107RTPROCESS g_ProcessSelf = NIL_RTPROCESS;
108
109/**
110 * The current process priority.
111 */
112RTPROCPRIORITY g_enmProcessPriority = RTPROCPRIORITY_DEFAULT;
113
114
115#ifndef RT_OS_WINDOWS
116/**
117 * Fork callback, child context.
118 */
119static void rtR3ForkChildCallback(void)
120{
121 g_ProcessSelf = getpid();
122}
123#endif /* RT_OS_WINDOWS */
124
125#ifdef RT_OS_OS2
126/** Low-level fork callback for OS/2. */
127int rtR3ForkOs2Child(__LIBC_PFORKHANDLE pForkHandle, __LIBC_FORKOP enmOperation)
128{
129 if (enmOperation == __LIBC_FORK_STAGE_COMPLETION_CHILD)
130 rtR3ForkChildCallback();
131 return 0;
132}
133
134_FORK_CHILD1(0, rtR3ForkOs2Child);
135#endif /* RT_OS_OS2 */
136
137
138
139/**
140 * Internal worker which initializes or re-initializes the
141 * program path, name and directory globals.
142 *
143 * @returns IPRT status code.
144 * @param pszProgramPath The program path, NULL if not specified.
145 */
146static int rtR3InitProgramPath(const char *pszProgramPath)
147{
148 /*
149 * We're reserving 32 bytes here for file names as what not.
150 */
151 if (!pszProgramPath)
152 {
153 int rc = rtProcInitExePath(g_szrtProcExePath, sizeof(g_szrtProcExePath) - 32);
154 if (RT_FAILURE(rc))
155 return rc;
156 }
157 else
158 {
159 size_t cch = strlen(pszProgramPath);
160 Assert(cch > 1);
161 AssertMsgReturn(cch < sizeof(g_szrtProcExePath) - 32, ("%zu\n", cch), VERR_BUFFER_OVERFLOW);
162 memcpy(g_szrtProcExePath, pszProgramPath, cch + 1);
163 }
164
165 /*
166 * Parse the name.
167 */
168 ssize_t offName;
169 g_cchrtProcExePath = RTPathParse(g_szrtProcExePath, &g_cchrtProcDir, &offName, NULL);
170 g_offrtProcName = offName;
171 return VINF_SUCCESS;
172}
173
174/**
175 * rtR3Init worker.
176 */
177static int rtR3InitBody(bool fInitSUPLib, const char *pszProgramPath)
178{
179 /*
180 * The Process ID.
181 */
182#ifdef _MSC_VER
183 g_ProcessSelf = _getpid(); /* crappy ansi compiler */
184#else
185 g_ProcessSelf = getpid();
186#endif
187
188#if !defined(IN_GUEST) && !defined(RT_NO_GIP)
189# ifdef VBOX
190 /*
191 * This MUST be done as the very first thing, before any file is opened.
192 * The log is opened on demand, but the first log entries may be caused
193 * by rtThreadInit() below.
194 */
195 const char *pszDisableHostCache = getenv("VBOX_DISABLE_HOST_DISK_CACHE");
196 if ( pszDisableHostCache != NULL
197 && strlen(pszDisableHostCache) > 0
198 && strcmp(pszDisableHostCache, "0") != 0)
199 {
200 RTFileSetForceFlags(RTFILE_O_WRITE, RTFILE_O_WRITE_THROUGH, 0);
201 RTFileSetForceFlags(RTFILE_O_READWRITE, RTFILE_O_WRITE_THROUGH, 0);
202 }
203# endif /* VBOX */
204#endif /* !IN_GUEST && !RT_NO_GIP */
205
206 /*
207 * Thread Thread database and adopt the caller thread as 'main'.
208 * This must be done before everything else or else we'll call into threading
209 * without having initialized TLS entries and suchlike.
210 */
211 int rc = rtThreadInit();
212 AssertMsgRCReturn(rc, ("Failed to initialize threads, rc=%Rrc!\n", rc), rc);
213
214#if !defined(IN_GUEST) && !defined(RT_NO_GIP)
215 if (fInitSUPLib)
216 {
217 /*
218 * Init GIP first.
219 * (The more time for updates before real use, the better.)
220 */
221 rc = SUPR3Init(NULL);
222 AssertMsgRCReturn(rc, ("Failed to initializeble the support library, rc=%Rrc!\n", rc), rc);
223 }
224#endif
225
226 /*
227 * The executable path, name and directory.
228 */
229 rc = rtR3InitProgramPath(pszProgramPath);
230 AssertLogRelMsgRCReturn(rc, ("Failed to get executable directory path, rc=%Rrc!\n", rc), rc);
231
232#if !defined(IN_GUEST) && !defined(RT_NO_GIP)
233 /*
234 * The threading is initialized we can safely sleep a bit if GIP
235 * needs some time to update itself updating.
236 */
237 if (fInitSUPLib && g_pSUPGlobalInfoPage)
238 {
239 RTThreadSleep(20);
240 RTTimeNanoTS();
241 }
242#endif
243
244 /*
245 * Init the program start TSes.
246 * Do that here to be sure that the GIP time was properly updated the 1st time.
247 */
248 g_u64ProgramStartNanoTS = RTTimeNanoTS();
249 g_u64ProgramStartMicroTS = g_u64ProgramStartNanoTS / 1000;
250 g_u64ProgramStartMilliTS = g_u64ProgramStartNanoTS / 1000000;
251
252 /*
253 * The remainder cannot easily be undone, so it has to go last.
254 */
255
256 /* Init C runtime locale. */
257 setlocale(LC_CTYPE, "");
258
259 /* Fork callbacks. */
260#if !defined(RT_OS_WINDOWS) && !defined(RT_OS_OS2)
261 rc = pthread_atfork(NULL, NULL, rtR3ForkChildCallback);
262 AssertMsg(rc == 0, ("%d\n", rc));
263#endif
264
265 return VINF_SUCCESS;
266}
267
268
269/**
270 * Internal initialization worker.
271 *
272 * @returns IPRT status code.
273 * @param fInitSUPLib Whether to call SUPR3Init.
274 * @param pszProgramPath The program path, NULL if not specified.
275 */
276static int rtR3Init(bool fInitSUPLib, const char *pszProgramPath)
277{
278 /* no entry log flow, because prefixes and thread may freak out. */
279
280 /*
281 * Do reference counting, only initialize the first time around.
282 *
283 * We are ASSUMING that nobody will be able to race RTR3Init calls when the
284 * first one, the real init, is running (second assertion).
285 */
286 int32_t cUsers = ASMAtomicIncS32(&g_cUsers);
287 if (cUsers != 1)
288 {
289 AssertMsg(cUsers > 1, ("%d\n", cUsers));
290 Assert(!g_fInitializing);
291#if !defined(IN_GUEST) && !defined(RT_NO_GIP)
292 if (fInitSUPLib)
293 SUPR3Init(NULL);
294#endif
295 if (!pszProgramPath)
296 return VINF_SUCCESS;
297 return rtR3InitProgramPath(pszProgramPath);
298 }
299 ASMAtomicWriteBool(&g_fInitializing, true);
300
301 /*
302 * Do the initialization.
303 */
304 int rc = rtR3InitBody(fInitSUPLib, pszProgramPath);
305 if (RT_FAILURE(rc))
306 {
307 /* failure */
308 ASMAtomicWriteBool(&g_fInitializing, false);
309 ASMAtomicDecS32(&g_cUsers);
310 return rc;
311 }
312
313 /* success */
314 LogFlow(("RTR3Init: returns VINF_SUCCESS\n"));
315 ASMAtomicWriteBool(&g_fInitializing, false);
316 return VINF_SUCCESS;
317}
318
319
320RTR3DECL(int) RTR3Init(void)
321{
322 return rtR3Init(false /* fInitSUPLib */, NULL);
323}
324
325
326RTR3DECL(int) RTR3InitEx(uint32_t iVersion, const char *pszProgramPath, bool fInitSUPLib)
327{
328 AssertReturn(iVersion == 0, VERR_NOT_SUPPORTED);
329 return rtR3Init(fInitSUPLib, pszProgramPath);
330}
331
332
333RTR3DECL(int) RTR3InitWithProgramPath(const char *pszProgramPath)
334{
335 return rtR3Init(false /* fInitSUPLib */, pszProgramPath);
336}
337
338
339RTR3DECL(int) RTR3InitAndSUPLib(void)
340{
341 return rtR3Init(true /* fInitSUPLib */, NULL /* pszProgramPath */);
342}
343
344
345RTR3DECL(int) RTR3InitAndSUPLibWithProgramPath(const char *pszProgramPath)
346{
347 return rtR3Init(true /* fInitSUPLib */, pszProgramPath);
348}
349
350
351#if 0 /** @todo implement RTR3Term. */
352RTR3DECL(void) RTR3Term(void)
353{
354}
355#endif
356
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