VirtualBox

source: vbox/trunk/src/VBox/Main/src-server/win/svcmain.cpp@ 40040

Last change on this file since 40040 was 38636, checked in by vboxsync, 13 years ago

*,IPRT: Redid the ring-3 init to always convert the arguments to UTF-8.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 12.9 KB
Line 
1/** @file
2 *
3 * SVCMAIN - COM out-of-proc server main entry
4 */
5
6/*
7 * Copyright (C) 2006-2011 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
18#include <Windows.h>
19#include <stdio.h>
20#include <stdlib.h>
21
22#include "VBox/com/defs.h"
23
24#include "VBox/com/com.h"
25
26#include "VBox/com/VirtualBox.h"
27
28#include "VirtualBoxImpl.h"
29#include "Logging.h"
30
31#include "svchlp.h"
32
33#include <VBox/err.h>
34#include <iprt/buildconfig.h>
35#include <iprt/initterm.h>
36#include <iprt/string.h>
37#include <iprt/uni.h>
38#include <iprt/path.h>
39#include <iprt/getopt.h>
40
41#include <atlbase.h>
42#include <atlcom.h>
43
44#define _ATL_FREE_THREADED
45
46class CExeModule : public CComModule
47{
48public:
49 LONG Unlock();
50 DWORD dwThreadID;
51 HANDLE hEventShutdown;
52 void MonitorShutdown();
53 bool StartMonitor();
54 bool bActivity;
55};
56
57const DWORD dwTimeOut = 5000; /* time for EXE to be idle before shutting down */
58const DWORD dwPause = 1000; /* time to wait for threads to finish up */
59
60/* Passed to CreateThread to monitor the shutdown event */
61static DWORD WINAPI MonitorProc(void* pv)
62{
63 CExeModule* p = (CExeModule*)pv;
64 p->MonitorShutdown();
65 return 0;
66}
67
68LONG CExeModule::Unlock()
69{
70 LONG l = CComModule::Unlock();
71 if (l == 0)
72 {
73 bActivity = true;
74 SetEvent(hEventShutdown); /* tell monitor that we transitioned to zero */
75 }
76 return l;
77}
78
79/* Monitors the shutdown event */
80void CExeModule::MonitorShutdown()
81{
82 while (1)
83 {
84 WaitForSingleObject(hEventShutdown, INFINITE);
85 DWORD dwWait=0;
86 do
87 {
88 bActivity = false;
89 dwWait = WaitForSingleObject(hEventShutdown, dwTimeOut);
90 } while (dwWait == WAIT_OBJECT_0);
91 /* timed out */
92 if (!bActivity && m_nLockCnt == 0) /* if no activity let's really bail */
93 {
94#if _WIN32_WINNT >= 0x0400 & defined(_ATL_FREE_THREADED)
95 CoSuspendClassObjects();
96 if (!bActivity && m_nLockCnt == 0)
97#endif
98 break;
99 }
100 }
101 CloseHandle(hEventShutdown);
102 PostThreadMessage(dwThreadID, WM_QUIT, 0, 0);
103}
104
105bool CExeModule::StartMonitor()
106{
107 hEventShutdown = CreateEvent(NULL, false, false, NULL);
108 if (hEventShutdown == NULL)
109 return false;
110 DWORD dwThreadID;
111 HANDLE h = CreateThread(NULL, 0, MonitorProc, this, 0, &dwThreadID);
112 return (h != NULL);
113}
114
115CExeModule _Module;
116
117BEGIN_OBJECT_MAP(ObjectMap)
118 OBJECT_ENTRY(CLSID_VirtualBox, VirtualBox)
119END_OBJECT_MAP()
120
121
122LPCTSTR FindOneOf(LPCTSTR p1, LPCTSTR p2)
123{
124 while (p1 != NULL && *p1 != NULL)
125 {
126 LPCTSTR p = p2;
127 while (p != NULL && *p != NULL)
128 {
129 if (*p1 == *p)
130 return CharNext(p1);
131 p = CharNext(p);
132 }
133 p1 = CharNext(p1);
134 }
135 return NULL;
136}
137
138static int WordCmpI(LPCTSTR psz1, LPCTSTR psz2) throw()
139{
140 TCHAR c1 = (TCHAR)CharUpper((LPTSTR)*psz1);
141 TCHAR c2 = (TCHAR)CharUpper((LPTSTR)*psz2);
142 while (c1 != NULL && c1 == c2 && c1 != ' ' && c1 != '\t')
143 {
144 psz1 = CharNext(psz1);
145 psz2 = CharNext(psz2);
146 c1 = (TCHAR)CharUpper((LPTSTR)*psz1);
147 c2 = (TCHAR)CharUpper((LPTSTR)*psz2);
148 }
149 if ((c1 == NULL || c1 == ' ' || c1 == '\t') && (c2 == NULL || c2 == ' ' || c2 == '\t'))
150 return 0;
151
152 return (c1 < c2) ? -1 : 1;
153}
154
155/////////////////////////////////////////////////////////////////////////////
156//
157int WINAPI WinMain(HINSTANCE hInstance, HINSTANCE /*hPrevInstance*/, LPSTR /*lpCmdLine*/, int /*nShowCmd*/)
158{
159 LPCTSTR lpCmdLine = GetCommandLine(); /* this line necessary for _ATL_MIN_CRT */
160
161 /*
162 * Need to parse the command line before initializing the VBox runtime.
163 */
164 TCHAR szTokens[] = _T("-/");
165 LPCTSTR lpszToken = FindOneOf(lpCmdLine, szTokens);
166 while (lpszToken != NULL)
167 {
168 if (WordCmpI(lpszToken, _T("Embedding")) == 0)
169 {
170 /* %HOMEDRIVE%%HOMEPATH% */
171 wchar_t wszHome[RTPATH_MAX];
172 DWORD cEnv = GetEnvironmentVariable(L"HOMEDRIVE", &wszHome[0], RTPATH_MAX);
173 if (cEnv && cEnv < RTPATH_MAX)
174 {
175 DWORD cwc = cEnv; /* doesn't include NUL */
176 cEnv = GetEnvironmentVariable(L"HOMEPATH", &wszHome[cEnv], RTPATH_MAX - cwc);
177 if (cEnv && cEnv < RTPATH_MAX - cwc)
178 {
179 /* If this fails there is nothing we can do. Ignore. */
180 SetCurrentDirectory(wszHome);
181 }
182 }
183 }
184
185 lpszToken = FindOneOf(lpszToken, szTokens);
186 }
187
188 /*
189 * Initialize the VBox runtime without loading
190 * the support driver.
191 */
192 int argc = __argc;
193 char **argv = __argv;
194 RTR3InitExe(argc, &argv, 0);
195
196 /* Note that all options are given lowercase/camel case/uppercase to
197 * approximate case insensitive matching, which RTGetOpt doesn't offer. */
198 static const RTGETOPTDEF s_aOptions[] =
199 {
200 { "--embedding", 'e', RTGETOPT_REQ_NOTHING | RTGETOPT_FLAG_ICASE },
201 { "-embedding", 'e', RTGETOPT_REQ_NOTHING | RTGETOPT_FLAG_ICASE },
202 { "/embedding", 'e', RTGETOPT_REQ_NOTHING | RTGETOPT_FLAG_ICASE },
203 { "--unregserver", 'u', RTGETOPT_REQ_NOTHING | RTGETOPT_FLAG_ICASE },
204 { "-unregserver", 'u', RTGETOPT_REQ_NOTHING | RTGETOPT_FLAG_ICASE },
205 { "/unregserver", 'u', RTGETOPT_REQ_NOTHING | RTGETOPT_FLAG_ICASE },
206 { "--regserver", 'r', RTGETOPT_REQ_NOTHING | RTGETOPT_FLAG_ICASE },
207 { "-regserver", 'r', RTGETOPT_REQ_NOTHING | RTGETOPT_FLAG_ICASE },
208 { "/regserver", 'r', RTGETOPT_REQ_NOTHING | RTGETOPT_FLAG_ICASE },
209 { "--reregserver", 'f', RTGETOPT_REQ_NOTHING | RTGETOPT_FLAG_ICASE },
210 { "-reregserver", 'f', RTGETOPT_REQ_NOTHING | RTGETOPT_FLAG_ICASE },
211 { "/reregserver", 'f', RTGETOPT_REQ_NOTHING | RTGETOPT_FLAG_ICASE },
212 { "--helper", 'H', RTGETOPT_REQ_STRING | RTGETOPT_FLAG_ICASE },
213 { "-helper", 'H', RTGETOPT_REQ_STRING | RTGETOPT_FLAG_ICASE },
214 { "/helper", 'H', RTGETOPT_REQ_STRING | RTGETOPT_FLAG_ICASE },
215 { "--logfile", 'F', RTGETOPT_REQ_STRING | RTGETOPT_FLAG_ICASE },
216 { "-logfile", 'F', RTGETOPT_REQ_STRING | RTGETOPT_FLAG_ICASE },
217 { "/logfile", 'F', RTGETOPT_REQ_STRING | RTGETOPT_FLAG_ICASE },
218 { "--logrotate", 'R', RTGETOPT_REQ_UINT32 | RTGETOPT_FLAG_ICASE },
219 { "-logrotate", 'R', RTGETOPT_REQ_UINT32 | RTGETOPT_FLAG_ICASE },
220 { "/logrotate", 'R', RTGETOPT_REQ_UINT32 | RTGETOPT_FLAG_ICASE },
221 { "--logsize", 'S', RTGETOPT_REQ_UINT64 | RTGETOPT_FLAG_ICASE },
222 { "-logsize", 'S', RTGETOPT_REQ_UINT64 | RTGETOPT_FLAG_ICASE },
223 { "/logsize", 'S', RTGETOPT_REQ_UINT64 | RTGETOPT_FLAG_ICASE },
224 { "--loginterval", 'I', RTGETOPT_REQ_UINT32 | RTGETOPT_FLAG_ICASE },
225 { "-loginterval", 'I', RTGETOPT_REQ_UINT32 | RTGETOPT_FLAG_ICASE },
226 { "/loginterval", 'I', RTGETOPT_REQ_UINT32 | RTGETOPT_FLAG_ICASE },
227 };
228
229 bool fRun = true;
230 bool fRegister = false;
231 bool fUnregister = false;
232 const char *pszPipeName = NULL;
233 const char *pszLogFile = NULL;
234 uint32_t cHistory = 10; // enable log rotation, 10 files
235 uint32_t uHistoryFileTime = RT_SEC_1DAY; // max 1 day per file
236 uint64_t uHistoryFileSize = 100 * _1M; // max 100MB per file
237
238 RTGETOPTSTATE GetOptState;
239 int vrc = RTGetOptInit(&GetOptState, argc, argv, &s_aOptions[0], RT_ELEMENTS(s_aOptions), 1, 0 /*fFlags*/);
240 AssertRC(vrc);
241
242 RTGETOPTUNION ValueUnion;
243 while ((vrc = RTGetOpt(&GetOptState, &ValueUnion)))
244 {
245 switch (vrc)
246 {
247 case 'e':
248 /* already handled above */
249 break;
250
251 case 'u':
252 fUnregister = true;
253 fRun = false;
254 break;
255
256 case 'r':
257 fRegister = true;
258 fRun = false;
259 break;
260
261 case 'f':
262 fUnregister = true;
263 fRegister = true;
264 fRun = false;
265 break;
266
267 case 'H':
268 pszPipeName = ValueUnion.psz;
269 if (!pszPipeName)
270 pszPipeName = "";
271 fRun = false;
272 break;
273
274 case 'F':
275 pszLogFile = ValueUnion.psz;
276 break;
277
278 case 'R':
279 cHistory = ValueUnion.u32;
280 break;
281
282 case 'S':
283 uHistoryFileSize = ValueUnion.u64;
284 break;
285
286 case 'I':
287 uHistoryFileTime = ValueUnion.u32;
288 break;
289
290 case 'h':
291 {
292 TCHAR txt[]= L"Options:\n\n"
293 L"/RegServer:\tregister COM out-of-proc server\n"
294 L"/UnregServer:\tunregister COM out-of-proc server\n"
295 L"/ReregServer:\tunregister and register COM server\n"
296 L"no options:\trun the server";
297 TCHAR title[]=_T("Usage");
298 fRun = false;
299 MessageBox(NULL, txt, title, MB_OK);
300 return 0;
301 }
302
303 case 'V':
304 {
305 char *psz = NULL;
306 RTStrAPrintf(&psz, "%sr%s\n", RTBldCfgVersion(), RTBldCfgRevisionStr());
307 PRTUTF16 txt = NULL;
308 RTStrToUtf16(psz, &txt);
309 TCHAR title[]=_T("Version");
310 fRun = false;
311 MessageBox(NULL, txt, title, MB_OK);
312 RTStrFree(psz);
313 RTUtf16Free(txt);
314 return 0;
315 }
316
317 default:
318 /** @todo this assumes that stderr is visible, which is not
319 * true for standard Windows applications. */
320 /* continue on command line errors... */
321 RTGetOptPrintError(vrc, &ValueUnion);
322 }
323 }
324
325 /* Only create the log file when running VBoxSVC normally, but not when
326 * registering/unregistering or calling the helper functionality. */
327 if (fRun)
328 {
329 if (!pszLogFile)
330 {
331 char szLogFile[RTPATH_MAX];
332 vrc = com::GetVBoxUserHomeDirectory(szLogFile, sizeof(szLogFile));
333 if (RT_SUCCESS(vrc))
334 vrc = RTPathAppend(szLogFile, sizeof(szLogFile), "VBoxSVC.log");
335 if (RT_SUCCESS(vrc))
336 pszLogFile = RTStrDup(szLogFile);
337 }
338 VBoxSVCLogRelCreate(pszLogFile, cHistory, uHistoryFileTime, uHistoryFileSize);
339 }
340
341 int nRet = 0;
342 HRESULT hRes = com::Initialize();
343
344 _ASSERTE(SUCCEEDED(hRes));
345 _Module.Init(ObjectMap, hInstance, &LIBID_VirtualBox);
346 _Module.dwThreadID = GetCurrentThreadId();
347
348 if (!fRun)
349 {
350 if (fUnregister)
351 {
352 _Module.UpdateRegistryFromResource(IDR_VIRTUALBOX, FALSE);
353 nRet = _Module.UnregisterServer(TRUE);
354 }
355 if (fRegister)
356 {
357 _Module.UpdateRegistryFromResource(IDR_VIRTUALBOX, TRUE);
358 nRet = _Module.RegisterServer(TRUE);
359 }
360 if (pszPipeName)
361 {
362 Log(("SVCMAIN: Processing Helper request (cmdline=\"%s\")...\n", pszPipeName));
363
364 if (!*pszPipeName)
365 vrc = VERR_INVALID_PARAMETER;
366
367 if (RT_SUCCESS(vrc))
368 {
369 /* do the helper job */
370 SVCHlpServer server;
371 vrc = server.open(pszPipeName);
372 if (RT_SUCCESS(vrc))
373 vrc = server.run();
374 }
375 if (RT_FAILURE(vrc))
376 {
377 Log(("SVCMAIN: Failed to process Helper request (%Rrc).", vrc));
378 nRet = 1;
379 }
380 }
381 }
382 else
383 {
384 _Module.StartMonitor();
385#if _WIN32_WINNT >= 0x0400 & defined(_ATL_FREE_THREADED)
386 hRes = _Module.RegisterClassObjects(CLSCTX_LOCAL_SERVER, REGCLS_MULTIPLEUSE | REGCLS_SUSPENDED);
387 _ASSERTE(SUCCEEDED(hRes));
388 hRes = CoResumeClassObjects();
389#else
390 hRes = _Module.RegisterClassObjects(CLSCTX_LOCAL_SERVER, REGCLS_MULTIPLEUSE);
391#endif
392 _ASSERTE(SUCCEEDED(hRes));
393
394 MSG msg;
395 while (GetMessage(&msg, 0, 0, 0))
396 DispatchMessage(&msg);
397
398 _Module.RevokeClassObjects();
399 Sleep(dwPause); //wait for any threads to finish
400 }
401
402 _Module.Term();
403
404 com::Shutdown();
405
406 Log(("SVCMAIN: Returning, COM server process ends.\n"));
407 return nRet;
408}
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