VirtualBox

source: vbox/trunk/src/VBox/Main/src-server/xpcom/server_module.cpp@ 57437

Last change on this file since 57437 was 55401, checked in by vboxsync, 10 years ago

added a couple of missing Id headers

  • Property svn:eol-style set to native
  • Property svn:keywords set to Date Revision Author Id
File size: 11.0 KB
Line 
1/* $Id: server_module.cpp 55401 2015-04-23 10:03:17Z vboxsync $ */
2/** @file
3 *
4 * XPCOM server process helper module implementation functions
5 */
6
7/*
8 * Copyright (C) 2006-2014 Oracle Corporation
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
19#ifdef RT_OS_OS2
20# include <prproces.h>
21#endif
22
23#include <nsMemory.h>
24#include <nsString.h>
25#include <nsCOMPtr.h>
26#include <nsIFile.h>
27#include <nsIGenericFactory.h>
28#include <nsIServiceManagerUtils.h>
29#include <nsICategoryManager.h>
30#include <nsDirectoryServiceDefs.h>
31
32#include <ipcIService.h>
33#include <ipcIDConnectService.h>
34#include <ipcCID.h>
35#include <ipcdclient.h>
36
37#include "prio.h"
38#include "prproces.h"
39
40// official XPCOM headers don't define it yet
41#define IPC_DCONNECTSERVICE_CONTRACTID \
42 "@mozilla.org/ipc/dconnect-service;1"
43
44// generated file
45#include <VirtualBox_XPCOM.h>
46
47#include "server.h"
48#include "Logging.h"
49
50#include <VBox/err.h>
51
52#include <iprt/assert.h>
53#include <iprt/param.h>
54#include <iprt/path.h>
55#include <iprt/process.h>
56#include <iprt/env.h>
57#include <iprt/string.h>
58#include <iprt/thread.h>
59
60#if defined(RT_OS_SOLARIS)
61# include <sys/systeminfo.h>
62#endif
63
64/// @todo move this to RT headers (and use them in MachineImpl.cpp as well)
65#if defined(RT_OS_WINDOWS) || defined(RT_OS_OS2)
66#define HOSTSUFF_EXE ".exe"
67#else /* !RT_OS_WINDOWS */
68#define HOSTSUFF_EXE ""
69#endif /* !RT_OS_WINDOWS */
70
71
72/** Name of the server executable. */
73const char VBoxSVC_exe[] = RTPATH_SLASH_STR "VBoxSVC" HOSTSUFF_EXE;
74
75enum
76{
77 /** Amount of time to wait for the server to establish a connection, ms */
78 VBoxSVC_Timeout = 30000,
79 /** How often to perform a connection check, ms */
80 VBoxSVC_WaitSlice = 100
81};
82
83/**
84 * Full path to the VBoxSVC executable.
85 */
86static char VBoxSVCPath[RTPATH_MAX];
87static bool IsVBoxSVCPathSet = false;
88
89/*
90 * The following macros define the method necessary to provide a list of
91 * interfaces implemented by the VirtualBox component. Note that this must be
92 * in sync with macros used for VirtualBox in server.cpp for the same purpose.
93 */
94
95NS_DECL_CLASSINFO(VirtualBoxWrap)
96NS_IMPL_CI_INTERFACE_GETTER1(VirtualBoxWrap, IVirtualBox)
97
98static nsresult vboxsvcSpawnDaemon(void)
99{
100 PRFileDesc *readable = nsnull, *writable = nsnull;
101 PRProcessAttr *attr = nsnull;
102 nsresult rv = NS_ERROR_FAILURE;
103 PRFileDesc *devNull;
104 // The ugly casts are necessary because the PR_CreateProcessDetached has
105 // a const array of writable strings as a parameter. It won't write. */
106 char * const args[] = { (char *)VBoxSVCPath, (char *)"--auto-shutdown", 0 };
107
108 // Use a pipe to determine when the daemon process is in the position
109 // to actually process requests. The daemon will write "READY" to the pipe.
110 if (PR_CreatePipe(&readable, &writable) != PR_SUCCESS)
111 goto end;
112 PR_SetFDInheritable(writable, PR_TRUE);
113
114 attr = PR_NewProcessAttr();
115 if (!attr)
116 goto end;
117
118 if (PR_ProcessAttrSetInheritableFD(attr, writable, VBOXSVC_STARTUP_PIPE_NAME) != PR_SUCCESS)
119 goto end;
120
121 devNull = PR_Open("/dev/null", PR_RDWR, 0);
122 if (!devNull)
123 goto end;
124
125 PR_ProcessAttrSetStdioRedirect(attr, PR_StandardInput, devNull);
126 PR_ProcessAttrSetStdioRedirect(attr, PR_StandardOutput, devNull);
127 PR_ProcessAttrSetStdioRedirect(attr, PR_StandardError, devNull);
128
129 if (PR_CreateProcessDetached(VBoxSVCPath, args, nsnull, attr) != PR_SUCCESS)
130 goto end;
131
132 // Close /dev/null
133 PR_Close(devNull);
134 // Close the child end of the pipe to make it the only owner of the
135 // file descriptor, so that unexpected closing can be detected.
136 PR_Close(writable);
137 writable = nsnull;
138
139 char msg[10];
140 RT_ZERO(msg);
141 if ( PR_Read(readable, msg, sizeof(msg)-1) != 5
142 || strcmp(msg, "READY"))
143 {
144 /* If several clients start VBoxSVC simultaneously only one can
145 * succeed. So treat this as success as well. */
146 rv = NS_OK;
147 goto end;
148 }
149
150 rv = NS_OK;
151
152end:
153 if (readable)
154 PR_Close(readable);
155 if (writable)
156 PR_Close(writable);
157 if (attr)
158 PR_DestroyProcessAttr(attr);
159 return rv;
160}
161
162
163/**
164 * VirtualBox component constructor.
165 *
166 * This constructor is responsible for starting the VirtualBox server
167 * process, connecting to it, and redirecting the constructor request to the
168 * VirtualBox component defined on the server.
169 */
170static NS_IMETHODIMP
171VirtualBoxConstructor(nsISupports *aOuter, REFNSIID aIID,
172 void **aResult)
173{
174 LogFlowFuncEnter();
175
176 nsresult rc = NS_OK;
177 int vrc = VINF_SUCCESS;
178
179 do
180 {
181 *aResult = NULL;
182 if (NULL != aOuter)
183 {
184 rc = NS_ERROR_NO_AGGREGATION;
185 break;
186 }
187
188 if (!IsVBoxSVCPathSet)
189 {
190 /* Get the directory containing XPCOM components -- the VBoxSVC
191 * executable is expected in the parent directory. */
192 nsCOMPtr<nsIProperties> dirServ = do_GetService(NS_DIRECTORY_SERVICE_CONTRACTID, &rc);
193 if (NS_SUCCEEDED(rc))
194 {
195 nsCOMPtr<nsIFile> componentDir;
196 rc = dirServ->Get(NS_XPCOM_COMPONENT_DIR,
197 NS_GET_IID(nsIFile), getter_AddRefs(componentDir));
198
199 if (NS_SUCCEEDED(rc))
200 {
201 nsCAutoString path;
202 componentDir->GetNativePath(path);
203
204 LogFlowFunc(("component directory = \"%s\"\n", path.get()));
205 AssertBreakStmt(path.Length() + strlen(VBoxSVC_exe) < RTPATH_MAX,
206 rc = NS_ERROR_FAILURE);
207
208#if defined(RT_OS_SOLARIS) && defined(VBOX_WITH_HARDENING)
209 char achKernArch[128];
210 int cbKernArch = sysinfo(SI_ARCHITECTURE_K, achKernArch, sizeof(achKernArch));
211 if (cbKernArch > 0)
212 {
213 sprintf(VBoxSVCPath, "/opt/VirtualBox/%s%s", achKernArch, VBoxSVC_exe);
214 IsVBoxSVCPathSet = true;
215 }
216 else
217 rc = NS_ERROR_UNEXPECTED;
218#else
219 strcpy(VBoxSVCPath, path.get());
220 RTPathStripFilename(VBoxSVCPath);
221 strcat(VBoxSVCPath, VBoxSVC_exe);
222
223 IsVBoxSVCPathSet = true;
224#endif
225 }
226 }
227 if (NS_FAILED(rc))
228 break;
229 }
230
231 nsCOMPtr<ipcIService> ipcServ = do_GetService(IPC_SERVICE_CONTRACTID, &rc);
232 if (NS_FAILED(rc))
233 break;
234
235 /* connect to the VBoxSVC server process */
236
237 bool startedOnce = false;
238 unsigned timeLeft = VBoxSVC_Timeout;
239
240 do
241 {
242 LogFlowFunc(("Resolving server name \"%s\"...\n", VBOXSVC_IPC_NAME));
243
244 PRUint32 serverID = 0;
245 rc = ipcServ->ResolveClientName(VBOXSVC_IPC_NAME, &serverID);
246 if (NS_FAILED(rc))
247 {
248 LogFlowFunc(("Starting server \"%s\"...\n", VBoxSVCPath));
249
250 startedOnce = true;
251
252 rc = vboxsvcSpawnDaemon();
253 if (NS_FAILED(rc))
254 break;
255
256 /* wait for the server process to establish a connection */
257 do
258 {
259 RTThreadSleep(VBoxSVC_WaitSlice);
260 rc = ipcServ->ResolveClientName(VBOXSVC_IPC_NAME, &serverID);
261 if (NS_SUCCEEDED(rc))
262 break;
263 if (timeLeft <= VBoxSVC_WaitSlice)
264 {
265 timeLeft = 0;
266 break;
267 }
268 timeLeft -= VBoxSVC_WaitSlice;
269 }
270 while (1);
271
272 if (!timeLeft)
273 {
274 rc = IPC_ERROR_WOULD_BLOCK;
275 break;
276 }
277 }
278
279 LogFlowFunc(("Connecting to server (ID=%d)...\n", serverID));
280
281 nsCOMPtr<ipcIDConnectService> dconServ =
282 do_GetService(IPC_DCONNECTSERVICE_CONTRACTID, &rc);
283 if (NS_FAILED(rc))
284 break;
285
286 rc = dconServ->CreateInstance(serverID,
287 CLSID_VirtualBox,
288 aIID, aResult);
289 if (NS_SUCCEEDED(rc))
290 break;
291
292 LogFlowFunc(("Failed to connect (rc=%Rhrc (%#08x))\n", rc, rc));
293
294 /* It's possible that the server gets shut down after we
295 * successfully resolve the server name but before it
296 * receives our CreateInstance() request. So, check for the
297 * name again, and restart the cycle if it fails. */
298 if (!startedOnce)
299 {
300 nsresult rc2 =
301 ipcServ->ResolveClientName(VBOXSVC_IPC_NAME, &serverID);
302 if (NS_SUCCEEDED(rc2))
303 break;
304
305 LogFlowFunc(("Server seems to have terminated before receiving our request. Will try again.\n"));
306 }
307 else
308 break;
309 }
310 while (1);
311 }
312 while (0);
313
314 LogFlowFunc(("rc=%Rhrc (%#08x), vrc=%Rrc\n", rc, rc, vrc));
315 LogFlowFuncLeave();
316
317 return rc;
318}
319
320#if 0
321/// @todo not really necessary for the moment
322/**
323 *
324 * @param aCompMgr
325 * @param aPath
326 * @param aLoaderStr
327 * @param aType
328 * @param aInfo
329 *
330 * @return
331 */
332static NS_IMETHODIMP
333VirtualBoxRegistration(nsIComponentManager *aCompMgr,
334 nsIFile *aPath,
335 const char *aLoaderStr,
336 const char *aType,
337 const nsModuleComponentInfo *aInfo)
338{
339 nsCAutoString modulePath;
340 aPath->GetNativePath(modulePath);
341 nsCAutoString moduleTarget;
342 aPath->GetNativeTarget(moduleTarget);
343
344 LogFlowFunc(("aPath=%s, aTarget=%s, aLoaderStr=%s, aType=%s\n",
345 modulePath.get(), moduleTarget.get(), aLoaderStr, aType));
346
347 nsresult rc = NS_OK;
348
349 return rc;
350}
351#endif
352
353/**
354 * Component definition table.
355 * Lists all components defined in this module.
356 */
357static const nsModuleComponentInfo components[] =
358{
359 {
360 "VirtualBox component", // description
361 NS_VIRTUALBOX_CID, NS_VIRTUALBOX_CONTRACTID, // CID/ContractID
362 VirtualBoxConstructor, // constructor function
363 NULL, /* VirtualBoxRegistration, */ // registration function
364 NULL, // deregistration function
365 NULL, // destructor function
366 /// @todo
367 NS_CI_INTERFACE_GETTER_NAME(VirtualBoxWrap), // interfaces function
368 NULL, // language helper
369 /// @todo
370 &NS_CLASSINFO_NAME(VirtualBoxWrap) // global class info & flags
371 }
372};
373
374NS_IMPL_NSGETMODULE(VirtualBox_Server_Module, components)
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