VirtualBox

source: vbox/trunk/src/VBox/Additions/common/VBoxService/VBoxService.cpp@ 37423

Last change on this file since 37423 was 36745, checked in by vboxsync, 14 years ago

Another fix for internal toolbox invocation (argv0), logging.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 26.8 KB
Line 
1/* $Id: VBoxService.cpp 36745 2011-04-20 10:08:26Z vboxsync $ */
2/** @file
3 * VBoxService - Guest Additions Service Skeleton.
4 */
5
6/*
7 * Copyright (C) 2007-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
19
20/*******************************************************************************
21* Header Files *
22*******************************************************************************/
23/** @todo LOG_GROUP*/
24#ifndef _MSC_VER
25# include <unistd.h>
26#endif
27#include <errno.h>
28#ifndef RT_OS_WINDOWS
29# include <signal.h>
30# ifdef RT_OS_OS2
31# define pthread_sigmask sigprocmask
32# endif
33#endif
34#ifdef RT_OS_FREEBSD
35# include <pthread.h>
36#endif
37
38#include "product-generated.h"
39#include <iprt/asm.h>
40#include <iprt/buildconfig.h>
41#include <iprt/initterm.h>
42#include <iprt/message.h>
43#include <iprt/path.h>
44#include <iprt/semaphore.h>
45#include <iprt/string.h>
46#include <iprt/stream.h>
47#include <iprt/thread.h>
48
49#include <VBox/log.h>
50
51#include "VBoxServiceInternal.h"
52
53
54/*******************************************************************************
55* Global Variables *
56*******************************************************************************/
57/** The program name (derived from argv[0]). */
58char *g_pszProgName = (char *)"";
59/** The current verbosity level. */
60int g_cVerbosity = 0;
61/** The default service interval (the -i | --interval) option). */
62uint32_t g_DefaultInterval = 0;
63#ifdef RT_OS_WINDOWS
64/** Signal shutdown to the Windows service thread. */
65static bool volatile g_fWindowsServiceShutdown;
66/** Event the Windows service thread waits for shutdown. */
67static RTSEMEVENT g_hEvtWindowsService;
68#endif
69
70/**
71 * The details of the services that has been compiled in.
72 */
73static struct
74{
75 /** Pointer to the service descriptor. */
76 PCVBOXSERVICE pDesc;
77 /** The worker thread. NIL_RTTHREAD if it's the main thread. */
78 RTTHREAD Thread;
79 /** Whether Pre-init was called. */
80 bool fPreInited;
81 /** Shutdown indicator. */
82 bool volatile fShutdown;
83 /** Indicator set by the service thread exiting. */
84 bool volatile fStopped;
85 /** Whether the service was started or not. */
86 bool fStarted;
87 /** Whether the service is enabled or not. */
88 bool fEnabled;
89} g_aServices[] =
90{
91#ifdef VBOXSERVICE_CONTROL
92 { &g_Control, NIL_RTTHREAD, false, false, false, false, true },
93#endif
94#ifdef VBOXSERVICE_TIMESYNC
95 { &g_TimeSync, NIL_RTTHREAD, false, false, false, false, true },
96#endif
97#ifdef VBOXSERVICE_CLIPBOARD
98 { &g_Clipboard, NIL_RTTHREAD, false, false, false, false, true },
99#endif
100#ifdef VBOXSERVICE_VMINFO
101 { &g_VMInfo, NIL_RTTHREAD, false, false, false, false, true },
102#endif
103#ifdef VBOXSERVICE_CPUHOTPLUG
104 { &g_CpuHotPlug, NIL_RTTHREAD, false, false, false, false, true },
105#endif
106#ifdef VBOXSERVICE_MANAGEMENT
107# ifdef VBOX_WITH_MEMBALLOON
108 { &g_MemBalloon, NIL_RTTHREAD, false, false, false, false, true },
109# endif
110 { &g_VMStatistics, NIL_RTTHREAD, false, false, false, false, true },
111#endif
112#if defined(VBOX_WITH_PAGE_SHARING) && defined(RT_OS_WINDOWS)
113 { &g_PageSharing, NIL_RTTHREAD, false, false, false, false, true },
114#endif
115#ifdef VBOX_WITH_SHARED_FOLDERS
116 { &g_AutoMount, NIL_RTTHREAD, false, false, false, false, true },
117#endif
118};
119
120
121/**
122 * Displays the program usage message.
123 *
124 * @returns 1.
125 */
126static int vboxServiceUsage(void)
127{
128 RTPrintf("Usage:\n"
129 " %-12s [-f|--foreground] [-v|--verbose] [-i|--interval <seconds>]\n"
130 " [--disable-<service>] [--enable-<service>] [-h|-?|--help]\n", g_pszProgName);
131#ifdef RT_OS_WINDOWS
132 RTPrintf(" [-r|--register] [-u|--unregister]\n");
133#endif
134 for (unsigned j = 0; j < RT_ELEMENTS(g_aServices); j++)
135 if (g_aServices[j].pDesc->pszUsage)
136 RTPrintf("%s\n", g_aServices[j].pDesc->pszUsage);
137 RTPrintf("\n"
138 "Options:\n"
139 " -i | --interval The default interval.\n"
140 " -f | --foreground Don't daemonize the program. For debugging.\n"
141 " -v | --verbose Increment the verbosity level. For debugging.\n"
142 " -V | --version Show version information.\n"
143 " -h | -? | --help Show this message and exit with status 1.\n"
144 );
145#ifdef RT_OS_WINDOWS
146 RTPrintf(" -r | --register Installs the service.\n"
147 " -u | --unregister Uninstall service.\n");
148#endif
149
150 RTPrintf("\n"
151 "Service-specific options:\n");
152 for (unsigned j = 0; j < RT_ELEMENTS(g_aServices); j++)
153 {
154 RTPrintf(" --enable-%-14s Enables the %s service. (default)\n", g_aServices[j].pDesc->pszName, g_aServices[j].pDesc->pszName);
155 RTPrintf(" --disable-%-13s Disables the %s service.\n", g_aServices[j].pDesc->pszName, g_aServices[j].pDesc->pszName);
156 if (g_aServices[j].pDesc->pszOptions)
157 RTPrintf("%s", g_aServices[j].pDesc->pszOptions);
158 }
159 RTPrintf("\n"
160 " Copyright (C) 2009-" VBOX_C_YEAR " " VBOX_VENDOR "\n");
161
162 return 1;
163}
164
165
166/**
167 * Displays a syntax error message.
168 *
169 * @returns RTEXITCODE_SYNTAX.
170 * @param pszFormat The message text.
171 * @param ... Format arguments.
172 */
173RTEXITCODE VBoxServiceSyntax(const char *pszFormat, ...)
174{
175 RTStrmPrintf(g_pStdErr, "%s: syntax error: ", g_pszProgName);
176
177 va_list va;
178 va_start(va, pszFormat);
179 RTStrmPrintfV(g_pStdErr, pszFormat, va);
180 va_end(va);
181
182 return RTEXITCODE_SYNTAX;
183}
184
185
186/**
187 * Displays an error message.
188 *
189 * @returns RTEXITCODE_FAILURE.
190 * @param pszFormat The message text.
191 * @param ... Format arguments.
192 */
193RTEXITCODE VBoxServiceError(const char *pszFormat, ...)
194{
195 RTStrmPrintf(g_pStdErr, "%s: error: ", g_pszProgName);
196
197 va_list va;
198 va_start(va, pszFormat);
199 RTStrmPrintfV(g_pStdErr, pszFormat, va);
200 va_end(va);
201
202 va_start(va, pszFormat);
203 LogRel(("%s: Error: %N", g_pszProgName, pszFormat, &va));
204 va_end(va);
205
206 return RTEXITCODE_FAILURE;
207}
208
209
210/**
211 * Displays a verbose message.
212 *
213 * @returns 1
214 * @param pszFormat The message text.
215 * @param ... Format arguments.
216 */
217void VBoxServiceVerbose(int iLevel, const char *pszFormat, ...)
218{
219 if (iLevel <= g_cVerbosity)
220 {
221 RTStrmPrintf(g_pStdOut, "%s: ", g_pszProgName);
222 va_list va;
223 va_start(va, pszFormat);
224 RTStrmPrintfV(g_pStdOut, pszFormat, va);
225 va_end(va);
226 va_start(va, pszFormat);
227 LogRel(("%s: %N", g_pszProgName, pszFormat, &va));
228 va_end(va);
229 }
230}
231
232
233/**
234 * Reports the current VBoxService status to the host.
235 *
236 * This makes sure that the Failed state is sticky.
237 *
238 * @return IPRT status code.
239 * @param enmStatus Status to report to the host.
240 */
241int VBoxServiceReportStatus(VBoxGuestFacilityStatus enmStatus)
242{
243 /*
244 * VBoxGuestFacilityStatus_Failed is sticky.
245 */
246 static VBoxGuestFacilityStatus s_enmLastStatus = VBoxGuestFacilityStatus_Inactive;
247 VBoxServiceVerbose(4, "Setting VBoxService status to %u\n", enmStatus);
248 if (s_enmLastStatus != VBoxGuestFacilityStatus_Failed)
249 {
250 int rc = VbglR3ReportAdditionsStatus(VBoxGuestFacilityType_VBoxService,
251 enmStatus, 0 /* Flags */);
252 if (RT_FAILURE(rc))
253 {
254 VBoxServiceError("Could not report VBoxService status (%u), rc=%Rrc\n", enmStatus, rc);
255 return rc;
256 }
257 s_enmLastStatus = enmStatus;
258 }
259 return VINF_SUCCESS;
260}
261
262
263/**
264 * Gets a 32-bit value argument.
265 *
266 * @returns 0 on success, non-zero exit code on error.
267 * @param argc The argument count.
268 * @param argv The argument vector
269 * @param psz Where in *pi to start looking for the value argument.
270 * @param pi Where to find and perhaps update the argument index.
271 * @param pu32 Where to store the 32-bit value.
272 * @param u32Min The minimum value.
273 * @param u32Max The maximum value.
274 */
275int VBoxServiceArgUInt32(int argc, char **argv, const char *psz, int *pi, uint32_t *pu32, uint32_t u32Min, uint32_t u32Max)
276{
277 if (*psz == ':' || *psz == '=')
278 psz++;
279 if (!*psz)
280 {
281 if (*pi + 1 >= argc)
282 return VBoxServiceSyntax("Missing value for the '%s' argument\n", argv[*pi]);
283 psz = argv[++*pi];
284 }
285
286 char *pszNext;
287 int rc = RTStrToUInt32Ex(psz, &pszNext, 0, pu32);
288 if (RT_FAILURE(rc) || *pszNext)
289 return VBoxServiceSyntax("Failed to convert interval '%s' to a number.\n", psz);
290 if (*pu32 < u32Min || *pu32 > u32Max)
291 return VBoxServiceSyntax("The timesync interval of %RU32 seconds is out of range [%RU32..%RU32].\n",
292 *pu32, u32Min, u32Max);
293 return 0;
294}
295
296
297/**
298 * The service thread.
299 *
300 * @returns Whatever the worker function returns.
301 * @param ThreadSelf My thread handle.
302 * @param pvUser The service index.
303 */
304static DECLCALLBACK(int) vboxServiceThread(RTTHREAD ThreadSelf, void *pvUser)
305{
306 const unsigned i = (uintptr_t)pvUser;
307
308#ifndef RT_OS_WINDOWS
309 /*
310 * Block all signals for this thread. Only the main thread will handle signals.
311 */
312 sigset_t signalMask;
313 sigfillset(&signalMask);
314 pthread_sigmask(SIG_BLOCK, &signalMask, NULL);
315#endif
316
317 int rc = g_aServices[i].pDesc->pfnWorker(&g_aServices[i].fShutdown);
318 ASMAtomicXchgBool(&g_aServices[i].fShutdown, true);
319 RTThreadUserSignal(ThreadSelf);
320 return rc;
321}
322
323
324/**
325 * Lazily calls the pfnPreInit method on each service.
326 *
327 * @returns VBox status code, error message displayed.
328 */
329static RTEXITCODE vboxServiceLazyPreInit(void)
330{
331 for (unsigned j = 0; j < RT_ELEMENTS(g_aServices); j++)
332 if (!g_aServices[j].fPreInited)
333 {
334 int rc = g_aServices[j].pDesc->pfnPreInit();
335 if (RT_FAILURE(rc))
336 return VBoxServiceError("Service '%s' failed pre-init: %Rrc\n", g_aServices[j].pDesc->pszName, rc);
337 g_aServices[j].fPreInited = true;
338 }
339 return RTEXITCODE_SUCCESS;
340}
341
342
343/**
344 * Count the number of enabled services.
345 */
346static unsigned vboxServiceCountEnabledServices(void)
347{
348 unsigned cEnabled = 0;
349 for (unsigned i = 0; i < RT_ELEMENTS(g_aServices); i++)
350 cEnabled += g_aServices[i].fEnabled;
351 return cEnabled;
352}
353
354
355/**
356 * Starts the service.
357 *
358 * @returns VBox status code, errors are fully bitched.
359 */
360int VBoxServiceStartServices(void)
361{
362 int rc;
363
364 VBoxServiceReportStatus(VBoxGuestFacilityStatus_Init);
365
366 /*
367 * Initialize the services.
368 */
369 VBoxServiceVerbose(2, "Initializing services ...\n");
370 for (unsigned j = 0; j < RT_ELEMENTS(g_aServices); j++)
371 if (g_aServices[j].fEnabled)
372 {
373 rc = g_aServices[j].pDesc->pfnInit();
374 if (RT_FAILURE(rc))
375 {
376 if (rc != VERR_SERVICE_DISABLED)
377 {
378 VBoxServiceError("Service '%s' failed to initialize: %Rrc\n",
379 g_aServices[j].pDesc->pszName, rc);
380 VBoxServiceReportStatus(VBoxGuestFacilityStatus_Failed);
381 return rc;
382 }
383 g_aServices[j].fEnabled = false;
384 VBoxServiceVerbose(0, "Service '%s' was disabled because of missing functionality\n",
385 g_aServices[j].pDesc->pszName);
386
387 }
388 }
389
390 /*
391 * Start the service(s).
392 */
393 VBoxServiceVerbose(2, "Starting services ...\n");
394 rc = VINF_SUCCESS;
395 for (unsigned j = 0; j < RT_ELEMENTS(g_aServices); j++)
396 {
397 if (!g_aServices[j].fEnabled)
398 continue;
399
400 VBoxServiceVerbose(2, "Starting service '%s' ...\n", g_aServices[j].pDesc->pszName);
401 rc = RTThreadCreate(&g_aServices[j].Thread, vboxServiceThread, (void *)(uintptr_t)j, 0,
402 RTTHREADTYPE_DEFAULT, RTTHREADFLAGS_WAITABLE, g_aServices[j].pDesc->pszName);
403 if (RT_FAILURE(rc))
404 {
405 VBoxServiceError("RTThreadCreate failed, rc=%Rrc\n", rc);
406 break;
407 }
408 g_aServices[j].fStarted = true;
409
410 /* Wait for the thread to initialize. */
411 /** @todo There is a race between waiting and checking
412 * the fShutdown flag of a thread here and processing
413 * the thread's actual worker loop. If the thread decides
414 * to exit the loop before we skipped the fShutdown check
415 * below the service will fail to start! */
416 RTThreadUserWait(g_aServices[j].Thread, 60 * 1000);
417 if (g_aServices[j].fShutdown)
418 {
419 VBoxServiceError("Service '%s' failed to start!\n", g_aServices[j].pDesc->pszName);
420 rc = VERR_GENERAL_FAILURE;
421 }
422 }
423
424 if (RT_SUCCESS(rc))
425 VBoxServiceVerbose(1, "All services started.\n");
426 else
427 {
428 VBoxServiceError("An error occcurred while the services!\n");
429 VBoxServiceReportStatus(VBoxGuestFacilityStatus_Failed);
430 }
431 return rc;
432}
433
434
435/**
436 * Stops and terminates the services.
437 *
438 * This should be called even when VBoxServiceStartServices fails so it can
439 * clean up anything that we succeeded in starting.
440 */
441int VBoxServiceStopServices(void)
442{
443 VBoxServiceReportStatus(VBoxGuestFacilityStatus_Terminating);
444
445 /*
446 * Signal all the services.
447 */
448 for (unsigned j = 0; j < RT_ELEMENTS(g_aServices); j++)
449 ASMAtomicWriteBool(&g_aServices[j].fShutdown, true);
450
451 /*
452 * Do the pfnStop callback on all running services.
453 */
454 for (unsigned j = 0; j < RT_ELEMENTS(g_aServices); j++)
455 if (g_aServices[j].fStarted)
456 {
457 VBoxServiceVerbose(3, "Calling stop function for service '%s' ...\n", g_aServices[j].pDesc->pszName);
458 g_aServices[j].pDesc->pfnStop();
459 }
460
461 /*
462 * Wait for all the service threads to complete.
463 */
464 int rc = VINF_SUCCESS;
465 for (unsigned j = 0; j < RT_ELEMENTS(g_aServices); j++)
466 {
467 if (!g_aServices[j].fEnabled) /* Only stop services which were started before. */
468 continue;
469 if (g_aServices[j].Thread != NIL_RTTHREAD)
470 {
471 VBoxServiceVerbose(2, "Waiting for service '%s' to stop ...\n", g_aServices[j].pDesc->pszName);
472 int rc2 = VINF_SUCCESS;
473 for (int i = 0; i < 30; i++) /* Wait 30 seconds in total */
474 {
475 rc2 = RTThreadWait(g_aServices[j].Thread, 1000 /* Wait 1 second */, NULL);
476 if (RT_SUCCESS(rc2))
477 break;
478#ifdef RT_OS_WINDOWS
479 /* Notify SCM that it takes a bit longer ... */
480 VBoxServiceWinSetStopPendingStatus(i + j*32);
481#endif
482 }
483 if (RT_FAILURE(rc2))
484 {
485 VBoxServiceError("Service '%s' failed to stop. (%Rrc)\n", g_aServices[j].pDesc->pszName, rc2);
486 rc = rc2;
487 }
488 }
489 VBoxServiceVerbose(3, "Terminating service '%s' (%d) ...\n", g_aServices[j].pDesc->pszName, j);
490 g_aServices[j].pDesc->pfnTerm();
491 }
492
493#ifdef RT_OS_WINDOWS
494 /*
495 * Wake up and tell the main() thread that we're shutting down (it's
496 * sleeping in VBoxServiceMainWait).
497 */
498 ASMAtomicWriteBool(&g_fWindowsServiceShutdown, true);
499 if (g_hEvtWindowsService != NIL_RTSEMEVENT)
500 {
501 VBoxServiceVerbose(3, "Stopping the main thread...\n");
502 int rc2 = RTSemEventSignal(g_hEvtWindowsService);
503 AssertRC(rc2);
504 }
505#endif
506
507 VBoxServiceVerbose(2, "Stopping services returning: %Rrc\n", rc);
508 VBoxServiceReportStatus(RT_SUCCESS(rc) ? VBoxGuestFacilityStatus_Paused : VBoxGuestFacilityStatus_Failed);
509 return rc;
510}
511
512
513/**
514 * Block the main thread until the service shuts down.
515 */
516void VBoxServiceMainWait(void)
517{
518 int rc;
519
520 VBoxServiceReportStatus(VBoxGuestFacilityStatus_Active);
521
522#ifdef RT_OS_WINDOWS
523 /*
524 * Wait for the semaphore to be signalled.
525 */
526 VBoxServiceVerbose(1, "Waiting in main thread\n");
527 rc = RTSemEventCreate(&g_hEvtWindowsService);
528 AssertRC(rc);
529 while (!ASMAtomicReadBool(&g_fWindowsServiceShutdown))
530 {
531 rc = RTSemEventWait(g_hEvtWindowsService, RT_INDEFINITE_WAIT);
532 AssertRC(rc);
533 }
534 RTSemEventDestroy(g_hEvtWindowsService);
535 g_hEvtWindowsService = NIL_RTSEMEVENT;
536
537#else
538 /*
539 * Wait explicitly for a HUP, INT, QUIT, ABRT or TERM signal, blocking
540 * all important signals.
541 *
542 * The annoying EINTR/ERESTART loop is for the benefit of Solaris where
543 * sigwait returns when we receive a SIGCHLD. Kind of makes sense since
544 */
545 sigset_t signalMask;
546 sigemptyset(&signalMask);
547 sigaddset(&signalMask, SIGHUP);
548 sigaddset(&signalMask, SIGINT);
549 sigaddset(&signalMask, SIGQUIT);
550 sigaddset(&signalMask, SIGABRT);
551 sigaddset(&signalMask, SIGTERM);
552 pthread_sigmask(SIG_BLOCK, &signalMask, NULL);
553
554 int iSignal;
555 do
556 {
557 iSignal = -1;
558 rc = sigwait(&signalMask, &iSignal);
559 }
560 while ( rc == EINTR
561# ifdef ERESTART
562 || rc == ERESTART
563# endif
564 );
565
566 VBoxServiceVerbose(3, "VBoxServiceMainWait: Received signal %d (rc=%d)\n", iSignal, rc);
567#endif /* !RT_OS_WINDOWS */
568}
569
570
571int main(int argc, char **argv)
572{
573 RTEXITCODE rcExit;
574
575 /*
576 * Init globals and such.
577 */
578 int rc = RTR3Init();
579 if (RT_FAILURE(rc))
580 return RTMsgInitFailure(rc);
581 g_pszProgName = RTPathFilename(argv[0]);
582
583#ifdef VBOXSERVICE_TOOLBOX
584 /*
585 * Run toolbox code before all other stuff since these things are simpler
586 * shell/file/text utility like programs that just happens to be inside
587 * VBoxService and shouldn't be subject to /dev/vboxguest, pid-files and
588 * global mutex restrictions.
589 */
590 if (VBoxServiceToolboxMain(argc, argv, &rcExit))
591 return rcExit;
592#endif
593
594 /*
595 * Connect to the kernel part before daemonizing so we can fail and
596 * complain if there is some kind of problem. We need to initialize the
597 * guest lib *before* we do the pre-init just in case one of services needs
598 * do to some initial stuff with it.
599 */
600 VBoxServiceVerbose(2, "Calling VbgR3Init()\n");
601 rc = VbglR3Init();
602 if (RT_FAILURE(rc))
603 {
604 if (rc == VERR_ACCESS_DENIED)
605 return VBoxServiceError("Insufficient privileges to start %s! Please start with Administrator/root privileges!\n",
606 g_pszProgName);
607 return VBoxServiceError("VbglR3Init failed with rc=%Rrc.\n", rc);
608 }
609
610#ifdef RT_OS_WINDOWS
611 /*
612 * Check if we're the specially spawned VBoxService.exe process that
613 * handles page fusion. This saves an extra executable.
614 */
615 if ( argc == 2
616 && !strcmp(argv[1], "--pagefusionfork"))
617 return VBoxServicePageSharingInitFork();
618#endif
619
620 /*
621 * Parse the arguments.
622 *
623 * Note! This code predates RTGetOpt, thus the manual parsing.
624 */
625 bool fDaemonize = true;
626 bool fDaemonized = false;
627 for (int i = 1; i < argc; i++)
628 {
629 const char *psz = argv[i];
630 if (*psz != '-')
631 return VBoxServiceSyntax("Unknown argument '%s'\n", psz);
632 psz++;
633
634 /* translate long argument to short */
635 if (*psz == '-')
636 {
637 psz++;
638 size_t cch = strlen(psz);
639#define MATCHES(strconst) ( cch == sizeof(strconst) - 1 \
640 && !memcmp(psz, strconst, sizeof(strconst) - 1) )
641 if (MATCHES("foreground"))
642 psz = "f";
643 else if (MATCHES("verbose"))
644 psz = "v";
645 else if (MATCHES("version"))
646 psz = "V";
647 else if (MATCHES("help"))
648 psz = "h";
649 else if (MATCHES("interval"))
650 psz = "i";
651#ifdef RT_OS_WINDOWS
652 else if (MATCHES("register"))
653 psz = "r";
654 else if (MATCHES("unregister"))
655 psz = "u";
656#endif
657 else if (MATCHES("daemonized"))
658 {
659 fDaemonized = true;
660 continue;
661 }
662 else
663 {
664 bool fFound = false;
665
666 if (cch > sizeof("enable-") && !memcmp(psz, "enable-", sizeof("enable-") - 1))
667 for (unsigned j = 0; !fFound && j < RT_ELEMENTS(g_aServices); j++)
668 if ((fFound = !RTStrICmp(psz + sizeof("enable-") - 1, g_aServices[j].pDesc->pszName)))
669 g_aServices[j].fEnabled = true;
670
671 if (cch > sizeof("disable-") && !memcmp(psz, "disable-", sizeof("disable-") - 1))
672 for (unsigned j = 0; !fFound && j < RT_ELEMENTS(g_aServices); j++)
673 if ((fFound = !RTStrICmp(psz + sizeof("disable-") - 1, g_aServices[j].pDesc->pszName)))
674 g_aServices[j].fEnabled = false;
675
676 if (!fFound)
677 {
678 rcExit = vboxServiceLazyPreInit();
679 if (rcExit != RTEXITCODE_SUCCESS)
680 return rcExit;
681
682 for (unsigned j = 0; !fFound && j < RT_ELEMENTS(g_aServices); j++)
683 {
684 rc = g_aServices[j].pDesc->pfnOption(NULL, argc, argv, &i);
685 fFound = rc == 0;
686 if (fFound)
687 break;
688 if (rc != -1)
689 return rc;
690 }
691 }
692 if (!fFound)
693 return VBoxServiceSyntax("Unknown option '%s'\n", argv[i]);
694 continue;
695 }
696#undef MATCHES
697 }
698
699 /* handle the string of short options. */
700 do
701 {
702 switch (*psz)
703 {
704 case 'i':
705 rc = VBoxServiceArgUInt32(argc, argv, psz + 1, &i,
706 &g_DefaultInterval, 1, (UINT32_MAX / 1000) - 1);
707 if (rc)
708 return rc;
709 psz = NULL;
710 break;
711
712 case 'f':
713 fDaemonize = false;
714 break;
715
716 case 'v':
717 g_cVerbosity++;
718 break;
719
720 case 'V':
721 RTPrintf("%sr%s\n", RTBldCfgVersion(), RTBldCfgRevisionStr());
722 return RTEXITCODE_SUCCESS;
723
724 case 'h':
725 case '?':
726 return vboxServiceUsage();
727
728#ifdef RT_OS_WINDOWS
729 case 'r':
730 return VBoxServiceWinInstall();
731
732 case 'u':
733 return VBoxServiceWinUninstall();
734#endif
735
736 default:
737 {
738 rcExit = vboxServiceLazyPreInit();
739 if (rcExit != RTEXITCODE_SUCCESS)
740 return rcExit;
741
742 bool fFound = false;
743 for (unsigned j = 0; j < RT_ELEMENTS(g_aServices); j++)
744 {
745 rc = g_aServices[j].pDesc->pfnOption(&psz, argc, argv, &i);
746 fFound = rc == VINF_SUCCESS;
747 if (fFound)
748 break;
749 if (rc != -1)
750 return rc;
751 }
752 if (!fFound)
753 return VBoxServiceSyntax("Unknown option '%c' (%s)\n", *psz, argv[i]);
754 break;
755 }
756 }
757 } while (psz && *++psz);
758 }
759
760 /* Check that at least one service is enabled. */
761 if (vboxServiceCountEnabledServices() == 0)
762 return VBoxServiceSyntax("At least one service must be enabled.\n");
763
764 /* Call pre-init if we didn't do it already. */
765 rcExit = vboxServiceLazyPreInit();
766 if (rcExit != RTEXITCODE_SUCCESS)
767 return rcExit;
768
769#ifdef RT_OS_WINDOWS
770 /*
771 * Make sure only one instance of VBoxService runs at a time. Create a
772 * global mutex for that.
773 *
774 * Note! The \\Global\ namespace was introduced with Win2K, thus the
775 * version check.
776 * Note! If the mutex exists CreateMutex will open it and set last error to
777 * ERROR_ALREADY_EXISTS.
778 */
779 OSVERSIONINFOEX OSInfoEx;
780 RT_ZERO(OSInfoEx);
781 OSInfoEx.dwOSVersionInfoSize = sizeof(OSVERSIONINFOEX);
782
783 SetLastError(NO_ERROR);
784 HANDLE hMutexAppRunning;
785 if ( GetVersionEx((LPOSVERSIONINFO)&OSInfoEx)
786 && OSInfoEx.dwPlatformId == VER_PLATFORM_WIN32_NT
787 && OSInfoEx.dwMajorVersion >= 5 /* NT 5.0 a.k.a W2K */)
788 hMutexAppRunning = CreateMutex(NULL, FALSE, "Global\\" VBOXSERVICE_NAME);
789 else
790 hMutexAppRunning = CreateMutex(NULL, FALSE, VBOXSERVICE_NAME);
791 if (hMutexAppRunning == NULL)
792 {
793 VBoxServiceError("CreateMutex failed with last error %u! Terminating", GetLastError());
794 return RTEXITCODE_FAILURE;
795 }
796 if (GetLastError() == ERROR_ALREADY_EXISTS)
797 {
798 VBoxServiceError("%s is already running! Terminating.", g_pszProgName);
799 CloseHandle(hMutexAppRunning);
800 return RTEXITCODE_FAILURE;
801 }
802#else /* !RT_OS_WINDOWS */
803 /** @todo Add PID file creation here? */
804#endif /* !RT_OS_WINDOWS */
805
806 VBoxServiceVerbose(0, "%s r%s started. Verbose level = %d\n",
807 RTBldCfgVersion(), RTBldCfgRevisionStr(), g_cVerbosity);
808
809 /*
810 * Daemonize if requested.
811 */
812 if (fDaemonize && !fDaemonized)
813 {
814#ifdef RT_OS_WINDOWS
815 VBoxServiceVerbose(2, "Starting service dispatcher ...\n");
816 rcExit = VBoxServiceWinEnterCtrlDispatcher();
817#else
818 VBoxServiceVerbose(1, "Daemonizing...\n");
819 rc = VbglR3Daemonize(false /* fNoChDir */, false /* fNoClose */);
820 if (RT_FAILURE(rc))
821 return VBoxServiceError("Daemon failed: %Rrc\n", rc);
822 /* in-child */
823#endif
824 }
825#ifdef RT_OS_WINDOWS
826 else
827#endif
828 {
829 /*
830 * Windows: We're running the service as a console application now. Start the
831 * services, enter the main thread's run loop and stop them again
832 * when it returns.
833 *
834 * POSIX: This is used for both daemons and console runs. Start all services
835 * and return immediately.
836 */
837 rc = VBoxServiceStartServices();
838 rcExit = RT_SUCCESS(rc) ? RTEXITCODE_SUCCESS : RTEXITCODE_FAILURE;
839 if (RT_SUCCESS(rc))
840 VBoxServiceMainWait();
841 VBoxServiceStopServices();
842 }
843 VBoxServiceReportStatus(VBoxGuestFacilityStatus_Terminated);
844
845#ifdef RT_OS_WINDOWS
846 /*
847 * Cleanup mutex.
848 */
849 CloseHandle(hMutexAppRunning);
850#endif
851
852 VBoxServiceVerbose(0, "Ended.\n");
853 return rcExit;
854}
855
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