VirtualBox

source: vbox/trunk/src/VBox/Main/glue/initterm.cpp@ 22911

Last change on this file since 22911 was 22847, checked in by vboxsync, 16 years ago

Python, COM glue: event processing API

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 17.9 KB
Line 
1/* $Id: initterm.cpp 22847 2009-09-08 20:37:54Z vboxsync $ */
2
3/** @file
4 * MS COM / XPCOM Abstraction Layer - Initialization and Termination.
5 */
6
7/*
8 * Copyright (C) 2006-2007 Sun Microsystems, Inc.
9 *
10 * This file is part of VirtualBox Open Source Edition (OSE), as
11 * available from http://www.virtualbox.org. This file is free software;
12 * you can redistribute it and/or modify it under the terms of the GNU
13 * General Public License (GPL) as published by the Free Software
14 * Foundation, in version 2 as it comes in the "COPYING" file of the
15 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
16 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
17 *
18 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa
19 * Clara, CA 95054 USA or visit http://www.sun.com if you need
20 * additional information or have any questions.
21 */
22
23#if !defined (VBOX_WITH_XPCOM)
24
25#include <objbase.h>
26
27#else /* !defined (VBOX_WITH_XPCOM) */
28
29#include <stdlib.h>
30
31/* XPCOM_GLUE is defined when the client uses the standalone glue
32 * (i.e. dynamically picks up the existing XPCOM shared library installation).
33 * This is not the case for VirtualBox XPCOM clients (they are always
34 * distrubuted with the self-built XPCOM library, and therefore have a binary
35 * dependency on it) but left here for clarity.
36 */
37#if defined (XPCOM_GLUE)
38#include <nsXPCOMGlue.h>
39#endif
40
41#include <nsIComponentRegistrar.h>
42#include <nsIServiceManager.h>
43#include <nsCOMPtr.h>
44#include <nsEventQueueUtils.h>
45#include <nsEmbedString.h>
46
47#include <nsILocalFile.h>
48#include <nsIDirectoryService.h>
49#include <nsDirectoryServiceDefs.h>
50
51#endif /* !defined (VBOX_WITH_XPCOM) */
52
53#include "VBox/com/com.h"
54#include "VBox/com/assert.h"
55#include "VBox/com/EventQueue.h"
56
57#include "../include/Logging.h"
58
59#include <iprt/param.h>
60#include <iprt/path.h>
61#include <iprt/string.h>
62#include <iprt/env.h>
63#include <iprt/asm.h>
64
65#include <VBox/err.h>
66
67namespace com
68{
69
70#if defined (VBOX_WITH_XPCOM)
71
72class DirectoryServiceProvider : public nsIDirectoryServiceProvider
73{
74public:
75
76 NS_DECL_ISUPPORTS
77
78 DirectoryServiceProvider()
79 : mCompRegLocation (NULL), mXPTIDatLocation (NULL)
80 , mComponentDirLocation (NULL), mCurrProcDirLocation (NULL)
81 {}
82
83 virtual ~DirectoryServiceProvider();
84
85 HRESULT init (const char *aCompRegLocation,
86 const char *aXPTIDatLocation,
87 const char *aComponentDirLocation,
88 const char *aCurrProcDirLocation);
89
90 NS_DECL_NSIDIRECTORYSERVICEPROVIDER
91
92private:
93
94 char *mCompRegLocation;
95 char *mXPTIDatLocation;
96 char *mComponentDirLocation;
97 char *mCurrProcDirLocation;
98};
99
100NS_IMPL_ISUPPORTS1 (DirectoryServiceProvider, nsIDirectoryServiceProvider)
101
102DirectoryServiceProvider::~DirectoryServiceProvider()
103{
104 if (mCompRegLocation)
105 {
106 RTStrFree (mCompRegLocation);
107 mCompRegLocation = NULL;
108 }
109 if (mXPTIDatLocation)
110 {
111 RTStrFree (mXPTIDatLocation);
112 mXPTIDatLocation = NULL;
113 }
114 if (mComponentDirLocation)
115 {
116 RTStrFree (mComponentDirLocation);
117 mComponentDirLocation = NULL;
118 }
119 if (mCurrProcDirLocation)
120 {
121 RTStrFree (mCurrProcDirLocation);
122 mCurrProcDirLocation = NULL;
123 }
124}
125
126/**
127 * @param aCompRegLocation Path to compreg.dat, in Utf8.
128 * @param aXPTIDatLocation Path to xpti.data, in Utf8.
129 */
130HRESULT
131DirectoryServiceProvider::init (const char *aCompRegLocation,
132 const char *aXPTIDatLocation,
133 const char *aComponentDirLocation,
134 const char *aCurrProcDirLocation)
135{
136 AssertReturn(aCompRegLocation, NS_ERROR_INVALID_ARG);
137 AssertReturn(aXPTIDatLocation, NS_ERROR_INVALID_ARG);
138
139 int vrc = RTStrUtf8ToCurrentCP (&mCompRegLocation, aCompRegLocation);
140 if (RT_SUCCESS(vrc))
141 vrc = RTStrUtf8ToCurrentCP (&mXPTIDatLocation, aXPTIDatLocation);
142 if (RT_SUCCESS(vrc) && aComponentDirLocation)
143 vrc = RTStrUtf8ToCurrentCP (&mComponentDirLocation, aComponentDirLocation);
144 if (RT_SUCCESS(vrc) && aCurrProcDirLocation)
145 vrc = RTStrUtf8ToCurrentCP (&mCurrProcDirLocation, aCurrProcDirLocation);
146
147 return RT_SUCCESS(vrc) ? NS_OK : NS_ERROR_OUT_OF_MEMORY;
148}
149
150NS_IMETHODIMP
151DirectoryServiceProvider::GetFile (const char *aProp,
152 PRBool *aPersistent,
153 nsIFile **aRetval)
154{
155 nsCOMPtr <nsILocalFile> localFile;
156 nsresult rv = NS_ERROR_FAILURE;
157
158 *aRetval = nsnull;
159 *aPersistent = PR_TRUE;
160
161 const char *fileLocation = NULL;
162
163 if (strcmp (aProp, NS_XPCOM_COMPONENT_REGISTRY_FILE) == 0)
164 fileLocation = mCompRegLocation;
165 else if (strcmp (aProp, NS_XPCOM_XPTI_REGISTRY_FILE) == 0)
166 fileLocation = mXPTIDatLocation;
167 else if (mComponentDirLocation && strcmp (aProp, NS_XPCOM_COMPONENT_DIR) == 0)
168 fileLocation = mComponentDirLocation;
169 else if (mCurrProcDirLocation && strcmp (aProp, NS_XPCOM_CURRENT_PROCESS_DIR) == 0)
170 fileLocation = mCurrProcDirLocation;
171 else
172 return NS_ERROR_FAILURE;
173
174 rv = NS_NewNativeLocalFile (nsEmbedCString (fileLocation),
175 PR_TRUE, getter_AddRefs (localFile));
176 if (NS_FAILED(rv))
177 return rv;
178
179 return localFile->QueryInterface (NS_GET_IID (nsIFile),
180 (void **) aRetval);
181}
182
183/**
184 * Global XPCOM initialization flag (we maintain it ourselves since XPCOM
185 * doesn't provide such functionality)
186 */
187static bool gIsXPCOMInitialized = false;
188
189/**
190 * Number of Initialize() calls on the main thread.
191 */
192static unsigned int gXPCOMInitCount = 0;
193
194#endif /* defined (VBOX_WITH_XPCOM) */
195
196/**
197 * Initializes the COM runtime.
198 *
199 * This method must be called on each thread of the client application that
200 * wants to access COM facilities. The initialization must be performed before
201 * calling any other COM method or attempting to instantiate COM objects.
202 *
203 * On platforms using XPCOM, this method uses the following scheme to search for
204 * XPCOM runtime:
205 *
206 * 1. If the VBOX_APP_HOME environment variable is set, the path it specifies
207 * is used to search XPCOM libraries and components. If this method fails to
208 * initialize XPCOM runtime using this path, it will immediately return a
209 * failure and will NOT check for other paths as described below.
210 *
211 * 2. If VBOX_APP_HOME is not set, this methods tries the following paths in the
212 * given order:
213 *
214 * a) Compiled-in application data directory (as returned by
215 * RTPathAppPrivateArch())
216 * b) "/usr/lib/virtualbox" (Linux only)
217 * c) "/opt/VirtualBox" (Linux only)
218 *
219 * The first path for which the initialization succeeds will be used.
220 *
221 * On MS COM platforms, the COM runtime is provided by the system and does not
222 * need to be searched for.
223 *
224 * Once the COM subsystem is no longer necessary on a given thread, Shutdown()
225 * must be called to free resources allocated for it. Note that a thread may
226 * call Initialize() several times but for each of tese calls there must be a
227 * corresponding Shutdown() call.
228 *
229 * @return S_OK on success and a COM result code in case of failure.
230 */
231HRESULT Initialize()
232{
233 HRESULT rc = E_FAIL;
234
235#if !defined (VBOX_WITH_XPCOM)
236
237 DWORD flags = COINIT_MULTITHREADED |
238 COINIT_DISABLE_OLE1DDE |
239 COINIT_SPEED_OVER_MEMORY;
240
241 rc = CoInitializeEx (NULL, flags);
242
243 /// @todo the below rough method of changing the aparment type doesn't
244 /// work on some systems for unknown reason (CoUninitialize() simply does
245 /// nothing there, or at least all 10 000 of subsequent CoInitializeEx()
246 /// continue to return RPC_E_CHANGED_MODE there). The problem on those
247 /// systems is related to the "Extend support for advanced text services
248 /// to all programs" checkbox in the advanced language settings dialog,
249 /// i.e. the problem appears when this checkbox is checked and disappears
250 /// if you clear it. For this reason, we disable the code below and
251 /// instead initialize COM in MTA as early as possible, before 3rd party
252 /// libraries we use have done so (i.e. Qt).
253#if 0
254 /* If we fail to set the necessary apartment model, it may mean that some
255 * DLL that was indirectly loaded by the process calling this function has
256 * already initialized COM on the given thread in an incompatible way
257 * which we can't leave with. Therefore, we try to fix this by using the
258 * brute force method: */
259
260 if (rc == RPC_E_CHANGED_MODE)
261 {
262 /* Before we use brute force, we need to check if we are in the
263 * neutral threaded apartment -- in this case there is no need to
264 * worry at all. */
265
266 rc = CoInitializeEx (NULL, COINIT_APARTMENTTHREADED);
267 if (rc == RPC_E_CHANGED_MODE)
268 {
269 /* This is a neutral apartment, reset the error */
270 rc = S_OK;
271
272 LogFlowFunc (("COM is already initialized in neutral threaded "
273 "apartment mode,\nwill accept it.\n"));
274 }
275 else if (rc == S_FALSE)
276 {
277 /* balance the test CoInitializeEx above */
278 CoUninitialize();
279 rc = RPC_E_CHANGED_MODE;
280
281 LogFlowFunc (("COM is already initialized in single threaded "
282 "apartment mode,\nwill reinitialize as "
283 "multi threaded.\n"));
284
285 enum { MaxTries = 10000 };
286 int tries = MaxTries;
287 while (rc == RPC_E_CHANGED_MODE && tries --)
288 {
289 CoUninitialize();
290 rc = CoInitializeEx (NULL, flags);
291 if (rc == S_OK)
292 {
293 /* We've successfully reinitialized COM; restore the
294 * initialization reference counter */
295
296 LogFlowFunc (("Will call CoInitializeEx() %d times.\n",
297 MaxTries - tries));
298
299 while (tries ++ < MaxTries)
300 {
301 rc = CoInitializeEx (NULL, flags);
302 Assert (rc == S_FALSE);
303 }
304 }
305 }
306 }
307 else
308 AssertMsgFailed (("rc=%08X\n", rc));
309 }
310#endif
311
312 /* the overall result must be either S_OK or S_FALSE (S_FALSE means
313 * "already initialized using the same apartment model") */
314 AssertMsg (rc == S_OK || rc == S_FALSE, ("rc=%08X\n", rc));
315
316#else /* !defined (VBOX_WITH_XPCOM) */
317
318 if (ASMAtomicXchgBool (&gIsXPCOMInitialized, true) == true)
319 {
320 /* XPCOM is already initialized on the main thread, no special
321 * initialization is necessary on additional threads. Just increase
322 * the init counter if it's a main thread again (to correctly support
323 * nested calls to Initialize()/Shutdown() for compatibility with
324 * Win32). */
325
326 nsCOMPtr <nsIEventQueue> eventQ;
327 rc = NS_GetMainEventQ (getter_AddRefs (eventQ));
328
329 if (NS_SUCCEEDED(rc))
330 {
331 PRBool isOnMainThread = PR_FALSE;
332 rc = eventQ->IsOnCurrentThread (&isOnMainThread);
333 if (NS_SUCCEEDED(rc) && isOnMainThread)
334 ++ gXPCOMInitCount;
335 }
336
337 AssertComRC (rc);
338 return rc;
339 }
340
341 /* this is the first initialization */
342 gXPCOMInitCount = 1;
343
344 /* prepare paths for registry files */
345 char userHomeDir [RTPATH_MAX];
346 int vrc = GetVBoxUserHomeDirectory (userHomeDir, sizeof (userHomeDir));
347 AssertRCReturn (vrc, NS_ERROR_FAILURE);
348
349 char compReg [RTPATH_MAX];
350 char xptiDat [RTPATH_MAX];
351
352 RTStrPrintf (compReg, sizeof (compReg), "%s%c%s",
353 userHomeDir, RTPATH_DELIMITER, "compreg.dat");
354 RTStrPrintf (xptiDat, sizeof (xptiDat), "%s%c%s",
355 userHomeDir, RTPATH_DELIMITER, "xpti.dat");
356
357 LogFlowFunc (("component registry : \"%s\"\n", compReg));
358 LogFlowFunc (("XPTI data file : \"%s\"\n", xptiDat));
359
360#if defined (XPCOM_GLUE)
361 XPCOMGlueStartup (nsnull);
362#endif
363
364 const char *kAppPathsToProbe[] =
365 {
366 NULL, /* 0: will use VBOX_APP_HOME */
367 NULL, /* 1: will try RTPathAppPrivateArch() */
368#ifdef RT_OS_LINUX
369 "/usr/lib/virtualbox",
370 "/opt/VirtualBox",
371#elif RT_OS_SOLARIS
372 "/opt/VirtualBox/amd64",
373 "/opt/VirtualBox/i386",
374#elif RT_OS_DARWIN
375 "/Application/VirtualBox.app/Contents/MacOS",
376#endif
377 };
378
379 /* Find out the directory where VirtualBox binaries are located */
380 for (size_t i = 0; i < RT_ELEMENTS (kAppPathsToProbe); ++ i)
381 {
382 char appHomeDir [RTPATH_MAX];
383 appHomeDir [RTPATH_MAX - 1] = '\0';
384
385 if (i == 0)
386 {
387 /* Use VBOX_APP_HOME if present */
388 if (!RTEnvExist ("VBOX_APP_HOME"))
389 continue;
390
391 strncpy (appHomeDir, RTEnvGet ("VBOX_APP_HOME"), RTPATH_MAX - 1);
392 }
393 else if (i == 1)
394 {
395 /* Use RTPathAppPrivateArch() first */
396 vrc = RTPathAppPrivateArch (appHomeDir, sizeof (appHomeDir));
397 AssertRC (vrc);
398 if (RT_FAILURE(vrc))
399 {
400 rc = NS_ERROR_FAILURE;
401 continue;
402 }
403 }
404 else
405 {
406 /* Iterate over all other paths */
407 strncpy (appHomeDir, kAppPathsToProbe [i], RTPATH_MAX - 1);
408 }
409
410 nsCOMPtr <DirectoryServiceProvider> dsProv;
411
412 char compDir [RTPATH_MAX];
413 RTStrPrintf (compDir, sizeof (compDir), "%s%c%s",
414 appHomeDir, RTPATH_DELIMITER, "components");
415 LogFlowFunc (("component directory : \"%s\"\n", compDir));
416
417 dsProv = new DirectoryServiceProvider();
418 if (dsProv)
419 rc = dsProv->init (compReg, xptiDat, compDir, appHomeDir);
420 else
421 rc = NS_ERROR_OUT_OF_MEMORY;
422 if (NS_FAILED (rc))
423 break;
424
425 /* Setup the application path for NS_InitXPCOM2. Note that we properly
426 * answer the NS_XPCOM_CURRENT_PROCESS_DIR query in our directory
427 * service provider but it seems to be activated after the directory
428 * service is used for the first time (see the source NS_InitXPCOM2). So
429 * use the same value here to be on the safe side. */
430 nsCOMPtr <nsIFile> appDir;
431 {
432 char *appDirCP = NULL;
433 vrc = RTStrUtf8ToCurrentCP (&appDirCP, appHomeDir);
434 if (RT_SUCCESS(vrc))
435 {
436 nsCOMPtr <nsILocalFile> file;
437 rc = NS_NewNativeLocalFile (nsEmbedCString (appDirCP),
438 PR_FALSE, getter_AddRefs (file));
439 if (NS_SUCCEEDED(rc))
440 appDir = do_QueryInterface (file, &rc);
441
442 RTStrFree (appDirCP);
443 }
444 else
445 rc = NS_ERROR_FAILURE;
446 }
447 if (NS_FAILED (rc))
448 break;
449
450 /* Set VBOX_XPCOM_HOME to the same app path to make XPCOM sources that
451 * still use it instead of the directory service happy */
452 {
453 char *pathCP = NULL;
454 vrc = RTStrUtf8ToCurrentCP (&pathCP, appHomeDir);
455 if (RT_SUCCESS(vrc))
456 {
457 vrc = RTEnvSet ("VBOX_XPCOM_HOME", pathCP);
458 RTStrFree (pathCP);
459 }
460 AssertRC (vrc);
461 }
462
463 /* Finally, initialize XPCOM */
464 {
465 nsCOMPtr <nsIServiceManager> serviceManager;
466 rc = NS_InitXPCOM2 (getter_AddRefs (serviceManager),
467 appDir, dsProv);
468
469 if (NS_SUCCEEDED(rc))
470 {
471 nsCOMPtr <nsIComponentRegistrar> registrar =
472 do_QueryInterface (serviceManager, &rc);
473 if (NS_SUCCEEDED(rc))
474 {
475 rc = registrar->AutoRegister (nsnull);
476 if (NS_SUCCEEDED(rc))
477 {
478 /* We succeeded, stop probing paths */
479 LogFlowFunc (("Succeeded.\n"));
480 break;
481 }
482 }
483 }
484 }
485
486 /* clean up before the new try */
487 rc = NS_ShutdownXPCOM (nsnull);
488
489 if (i == 0)
490 {
491 /* We failed with VBOX_APP_HOME, don't probe other paths */
492 break;
493 }
494 }
495
496#endif /* !defined (VBOX_WITH_XPCOM) */
497
498 AssertComRC (rc);
499
500 EventQueue::init();
501
502 return rc;
503}
504
505HRESULT Shutdown()
506{
507 HRESULT rc = S_OK;
508
509 EventQueue::deinit();
510
511#if !defined (VBOX_WITH_XPCOM)
512
513 CoUninitialize();
514
515#else /* !defined (VBOX_WITH_XPCOM) */
516
517 nsCOMPtr <nsIEventQueue> eventQ;
518 rc = NS_GetMainEventQ (getter_AddRefs (eventQ));
519
520 if (NS_SUCCEEDED(rc) || rc == NS_ERROR_NOT_AVAILABLE)
521 {
522 /* NS_ERROR_NOT_AVAILABLE seems to mean that
523 * nsIEventQueue::StopAcceptingEvents() has been called (see
524 * nsEventQueueService.cpp). We hope that this error code always means
525 * just that in this case and assume that we're on the main thread
526 * (it's a kind of unexpected behavior if a non-main thread ever calls
527 * StopAcceptingEvents() on the main event queue). */
528
529 PRBool isOnMainThread = PR_FALSE;
530 if (NS_SUCCEEDED(rc))
531 {
532 rc = eventQ->IsOnCurrentThread (&isOnMainThread);
533 eventQ = nsnull; /* early release before shutdown */
534 }
535 else
536 {
537 isOnMainThread = PR_TRUE;
538 rc = NS_OK;
539 }
540
541 if (NS_SUCCEEDED(rc) && isOnMainThread)
542 {
543 /* only the main thread needs to uninitialize XPCOM and only if
544 * init counter drops to zero */
545 if (-- gXPCOMInitCount == 0)
546 {
547 rc = NS_ShutdownXPCOM (nsnull);
548
549 /* This is a thread initialized XPCOM and set gIsXPCOMInitialized to
550 * true. Reset it back to false. */
551 bool wasInited = ASMAtomicXchgBool (&gIsXPCOMInitialized, false);
552 Assert (wasInited == true);
553 NOREF (wasInited);
554
555#if defined (XPCOM_GLUE)
556 XPCOMGlueShutdown();
557#endif
558 }
559 }
560 }
561
562#endif /* !defined (VBOX_WITH_XPCOM) */
563
564 AssertComRC (rc);
565
566 return rc;
567}
568
569} /* namespace com */
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