VirtualBox

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

Last change on this file since 57444 was 57416, checked in by vboxsync, 9 years ago

More DECLCALLBACK fixes; retired RTMemAutoPtr.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 37.8 KB
Line 
1/* $Id: VBoxService.cpp 57416 2015-08-18 11:19:28Z vboxsync $ */
2/** @file
3 * VBoxService - Guest Additions Service Skeleton.
4 */
5
6/*
7 * Copyright (C) 2007-2015 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 <package-generated.h>
39#include "product-generated.h"
40
41#include <iprt/asm.h>
42#include <iprt/buildconfig.h>
43#include <iprt/initterm.h>
44#include <iprt/file.h>
45#ifdef DEBUG
46# include <iprt/memtracker.h>
47#endif
48#include <iprt/message.h>
49#include <iprt/path.h>
50#include <iprt/process.h>
51#include <iprt/semaphore.h>
52#include <iprt/string.h>
53#include <iprt/stream.h>
54#include <iprt/system.h>
55#include <iprt/thread.h>
56
57#include <VBox/log.h>
58
59#include "VBoxServiceInternal.h"
60#ifdef VBOX_WITH_GUEST_CONTROL
61# include "VBoxServiceControl.h"
62#endif
63
64
65/*********************************************************************************************************************************
66* Global Variables *
67*********************************************************************************************************************************/
68/** The program name (derived from argv[0]). */
69char *g_pszProgName = (char *)"";
70/** The current verbosity level. */
71int g_cVerbosity = 0;
72char g_szLogFile[RTPATH_MAX + 128] = "";
73/** Logging parameters. */
74/** @todo Make this configurable later. */
75static PRTLOGGER g_pLoggerRelease = NULL;
76static uint32_t g_cHistory = 10; /* Enable log rotation, 10 files. */
77static uint32_t g_uHistoryFileTime = RT_SEC_1DAY; /* Max 1 day per file. */
78static uint64_t g_uHistoryFileSize = 100 * _1M; /* Max 100MB per file. */
79/** Critical section for (debug) logging. */
80#ifdef DEBUG
81 RTCRITSECT g_csLog;
82#endif
83/** The default service interval (the -i | --interval) option). */
84uint32_t g_DefaultInterval = 0;
85#ifdef RT_OS_WINDOWS
86/** Signal shutdown to the Windows service thread. */
87static bool volatile g_fWindowsServiceShutdown;
88/** Event the Windows service thread waits for shutdown. */
89static RTSEMEVENT g_hEvtWindowsService;
90#endif
91
92/**
93 * The details of the services that has been compiled in.
94 */
95static struct
96{
97 /** Pointer to the service descriptor. */
98 PCVBOXSERVICE pDesc;
99 /** The worker thread. NIL_RTTHREAD if it's the main thread. */
100 RTTHREAD Thread;
101 /** Whether Pre-init was called. */
102 bool fPreInited;
103 /** Shutdown indicator. */
104 bool volatile fShutdown;
105 /** Indicator set by the service thread exiting. */
106 bool volatile fStopped;
107 /** Whether the service was started or not. */
108 bool fStarted;
109 /** Whether the service is enabled or not. */
110 bool fEnabled;
111} g_aServices[] =
112{
113#ifdef VBOXSERVICE_CONTROL
114 { &g_Control, NIL_RTTHREAD, false, false, false, false, true },
115#endif
116#ifdef VBOXSERVICE_TIMESYNC
117 { &g_TimeSync, NIL_RTTHREAD, false, false, false, false, true },
118#endif
119#ifdef VBOXSERVICE_CLIPBOARD
120 { &g_Clipboard, NIL_RTTHREAD, false, false, false, false, true },
121#endif
122#ifdef VBOXSERVICE_VMINFO
123 { &g_VMInfo, NIL_RTTHREAD, false, false, false, false, true },
124#endif
125#ifdef VBOXSERVICE_CPUHOTPLUG
126 { &g_CpuHotPlug, NIL_RTTHREAD, false, false, false, false, true },
127#endif
128#ifdef VBOXSERVICE_MANAGEMENT
129# ifdef VBOX_WITH_MEMBALLOON
130 { &g_MemBalloon, NIL_RTTHREAD, false, false, false, false, true },
131# endif
132 { &g_VMStatistics, NIL_RTTHREAD, false, false, false, false, true },
133#endif
134#if defined(VBOXSERVICE_PAGE_SHARING)
135 { &g_PageSharing, NIL_RTTHREAD, false, false, false, false, true },
136#endif
137#ifdef VBOX_WITH_SHARED_FOLDERS
138 { &g_AutoMount, NIL_RTTHREAD, false, false, false, false, true },
139#endif
140};
141
142/* Default call-backs for services which do not need special behaviour. */
143
144/** @copydoc VBOXSERVICE::pfnPreInit */
145DECLCALLBACK(int) VBoxServiceDefaultPreInit(void)
146{
147 return VINF_SUCCESS;
148}
149
150/** @copydoc VBOXSERVICE::pfnOption */
151DECLCALLBACK(int) VBoxServiceDefaultOption(const char **ppszShort, int argc,
152 char **argv, int *pi)
153{
154 NOREF(ppszShort);
155 NOREF(argc);
156 NOREF(argv);
157 NOREF(pi);
158
159 return -1;
160}
161
162/** @copydoc VBOXSERVICE::pfnInit */
163DECLCALLBACK(int) VBoxServiceDefaultInit(void)
164{
165 return VINF_SUCCESS;
166}
167
168/** @copydoc VBOXSERVICE::pfnTerm */
169DECLCALLBACK(void) VBoxServiceDefaultTerm(void)
170{
171 return;
172}
173
174/**
175 * Release logger callback.
176 *
177 * @return IPRT status code.
178 * @param pLoggerRelease
179 * @param enmPhase
180 * @param pfnLog
181 */
182static DECLCALLBACK(void) VBoxServiceLogHeaderFooter(PRTLOGGER pLoggerRelease, RTLOGPHASE enmPhase, PFNRTLOGPHASEMSG pfnLog)
183{
184 /* Some introductory information. */
185 static RTTIMESPEC s_TimeSpec;
186 char szTmp[256];
187 if (enmPhase == RTLOGPHASE_BEGIN)
188 RTTimeNow(&s_TimeSpec);
189 RTTimeSpecToString(&s_TimeSpec, szTmp, sizeof(szTmp));
190
191 switch (enmPhase)
192 {
193 case RTLOGPHASE_BEGIN:
194 {
195 pfnLog(pLoggerRelease,
196 "VBoxService %s r%s (verbosity: %d) %s (%s %s) release log\n"
197 "Log opened %s\n",
198 RTBldCfgVersion(), RTBldCfgRevisionStr(), g_cVerbosity, VBOX_BUILD_TARGET,
199 __DATE__, __TIME__, szTmp);
200
201 int vrc = RTSystemQueryOSInfo(RTSYSOSINFO_PRODUCT, szTmp, sizeof(szTmp));
202 if (RT_SUCCESS(vrc) || vrc == VERR_BUFFER_OVERFLOW)
203 pfnLog(pLoggerRelease, "OS Product: %s\n", szTmp);
204 vrc = RTSystemQueryOSInfo(RTSYSOSINFO_RELEASE, szTmp, sizeof(szTmp));
205 if (RT_SUCCESS(vrc) || vrc == VERR_BUFFER_OVERFLOW)
206 pfnLog(pLoggerRelease, "OS Release: %s\n", szTmp);
207 vrc = RTSystemQueryOSInfo(RTSYSOSINFO_VERSION, szTmp, sizeof(szTmp));
208 if (RT_SUCCESS(vrc) || vrc == VERR_BUFFER_OVERFLOW)
209 pfnLog(pLoggerRelease, "OS Version: %s\n", szTmp);
210 if (RT_SUCCESS(vrc) || vrc == VERR_BUFFER_OVERFLOW)
211 pfnLog(pLoggerRelease, "OS Service Pack: %s\n", szTmp);
212
213 /* the package type is interesting for Linux distributions */
214 char szExecName[RTPATH_MAX];
215 char *pszExecName = RTProcGetExecutablePath(szExecName, sizeof(szExecName));
216 pfnLog(pLoggerRelease,
217 "Executable: %s\n"
218 "Process ID: %u\n"
219 "Package type: %s"
220#ifdef VBOX_OSE
221 " (OSE)"
222#endif
223 "\n",
224 pszExecName ? pszExecName : "unknown",
225 RTProcSelf(),
226 VBOX_PACKAGE_STRING);
227 break;
228 }
229
230 case RTLOGPHASE_PREROTATE:
231 pfnLog(pLoggerRelease, "Log rotated - Log started %s\n", szTmp);
232 break;
233
234 case RTLOGPHASE_POSTROTATE:
235 pfnLog(pLoggerRelease, "Log continuation - Log started %s\n", szTmp);
236 break;
237
238 case RTLOGPHASE_END:
239 pfnLog(pLoggerRelease, "End of log file - Log started %s\n", szTmp);
240 break;
241
242 default:
243 /* nothing */;
244 }
245}
246
247
248/**
249 * Creates the default release logger outputting to the specified file.
250 * Pass NULL for disabled logging.
251 *
252 * @return IPRT status code.
253 * @param pszLogFile Filename for log output. Optional.
254 */
255int VBoxServiceLogCreate(const char *pszLogFile)
256{
257 /* Create release logger (stdout + file). */
258 static const char * const s_apszGroups[] = VBOX_LOGGROUP_NAMES;
259 RTUINT fFlags = RTLOGFLAGS_PREFIX_THREAD | RTLOGFLAGS_PREFIX_TIME_PROG;
260#if defined(RT_OS_WINDOWS) || defined(RT_OS_OS2)
261 fFlags |= RTLOGFLAGS_USECRLF;
262#endif
263 char szError[RTPATH_MAX + 128] = "";
264 int rc = RTLogCreateEx(&g_pLoggerRelease, fFlags, "all",
265#ifdef DEBUG
266 "VBOXSERVICE_LOG",
267#else
268 "VBOXSERVICE_RELEASE_LOG",
269#endif
270 RT_ELEMENTS(s_apszGroups), s_apszGroups,
271 RTLOGDEST_STDOUT | RTLOGDEST_USER,
272 VBoxServiceLogHeaderFooter, g_cHistory, g_uHistoryFileSize, g_uHistoryFileTime,
273 szError, sizeof(szError), pszLogFile);
274 if (RT_SUCCESS(rc))
275 {
276 /* register this logger as the release logger */
277 RTLogRelSetDefaultInstance(g_pLoggerRelease);
278
279 /* Explicitly flush the log in case of VBOXSERVICE_RELEASE_LOG=buffered. */
280 RTLogFlush(g_pLoggerRelease);
281 }
282
283 return rc;
284}
285
286
287void VBoxServiceLogDestroy(void)
288{
289 RTLogDestroy(RTLogRelSetDefaultInstance(NULL));
290}
291
292
293/**
294 * Displays the program usage message.
295 *
296 * @returns 1.
297 */
298static int vboxServiceUsage(void)
299{
300 RTPrintf("Usage:\n"
301 " %-12s [-f|--foreground] [-v|--verbose] [-l|--logfile <file>]\n"
302 " [-i|--interval <seconds>]\n"
303 " [--disable-<service>] [--enable-<service>]\n"
304 " [--only-<service>] [-h|-?|--help]\n", g_pszProgName);
305#ifdef RT_OS_WINDOWS
306 RTPrintf(" [-r|--register] [-u|--unregister]\n");
307#endif
308 for (unsigned j = 0; j < RT_ELEMENTS(g_aServices); j++)
309 if (g_aServices[j].pDesc->pszUsage)
310 RTPrintf("%s\n", g_aServices[j].pDesc->pszUsage);
311 RTPrintf("\n"
312 "Options:\n"
313 " -i | --interval The default interval.\n"
314 " -f | --foreground Don't daemonize the program. For debugging.\n"
315 " -l | --logfile <file> Enables logging to a file.\n"
316 " -v | --verbose Increment the verbosity level. For debugging.\n"
317 " -V | --version Show version information.\n"
318 " -h | -? | --help Show this message and exit with status 1.\n"
319 );
320#ifdef RT_OS_WINDOWS
321 RTPrintf(" -r | --register Installs the service.\n"
322 " -u | --unregister Uninstall service.\n");
323#endif
324
325 RTPrintf("\n"
326 "Service-specific options:\n");
327 for (unsigned j = 0; j < RT_ELEMENTS(g_aServices); j++)
328 {
329 RTPrintf(" --enable-%-14s Enables the %s service. (default)\n", g_aServices[j].pDesc->pszName, g_aServices[j].pDesc->pszName);
330 RTPrintf(" --disable-%-13s Disables the %s service.\n", g_aServices[j].pDesc->pszName, g_aServices[j].pDesc->pszName);
331 RTPrintf(" --only-%-16s Only enables the %s service.\n", g_aServices[j].pDesc->pszName, g_aServices[j].pDesc->pszName);
332 if (g_aServices[j].pDesc->pszOptions)
333 RTPrintf("%s", g_aServices[j].pDesc->pszOptions);
334 }
335 RTPrintf("\n"
336 " Copyright (C) 2009-" VBOX_C_YEAR " " VBOX_VENDOR "\n");
337
338 return 1;
339}
340
341
342/**
343 * Displays an error message.
344 *
345 * @returns RTEXITCODE_FAILURE.
346 * @param pszFormat The message text.
347 * @param ... Format arguments.
348 */
349RTEXITCODE VBoxServiceError(const char *pszFormat, ...)
350{
351 va_list args;
352 va_start(args, pszFormat);
353 char *psz = NULL;
354 RTStrAPrintfV(&psz, pszFormat, args);
355 va_end(args);
356
357 AssertPtr(psz);
358 LogRel(("Error: %s", psz));
359
360 RTStrFree(psz);
361
362 return RTEXITCODE_FAILURE;
363}
364
365
366/**
367 * Displays a verbose message.
368 *
369 * @param iLevel Minimum log level required to display this message.
370 * @param pszFormat The message text.
371 * @param ... Format arguments.
372 */
373void VBoxServiceVerbose(int iLevel, const char *pszFormat, ...)
374{
375 if (iLevel <= g_cVerbosity)
376 {
377#ifdef DEBUG
378 int rc = RTCritSectEnter(&g_csLog);
379 if (RT_SUCCESS(rc))
380 {
381#endif
382 va_list args;
383 va_start(args, pszFormat);
384 char *psz = NULL;
385 RTStrAPrintfV(&psz, pszFormat, args);
386 va_end(args);
387
388 AssertPtr(psz);
389 LogRel(("%s", psz));
390
391 RTStrFree(psz);
392#ifdef DEBUG
393 RTCritSectLeave(&g_csLog);
394 }
395#endif
396 }
397}
398
399
400/**
401 * Reports the current VBoxService status to the host.
402 *
403 * This makes sure that the Failed state is sticky.
404 *
405 * @return IPRT status code.
406 * @param enmStatus Status to report to the host.
407 */
408int VBoxServiceReportStatus(VBoxGuestFacilityStatus enmStatus)
409{
410 /*
411 * VBoxGuestFacilityStatus_Failed is sticky.
412 */
413 static VBoxGuestFacilityStatus s_enmLastStatus = VBoxGuestFacilityStatus_Inactive;
414 VBoxServiceVerbose(4, "Setting VBoxService status to %u\n", enmStatus);
415 if (s_enmLastStatus != VBoxGuestFacilityStatus_Failed)
416 {
417 int rc = VbglR3ReportAdditionsStatus(VBoxGuestFacilityType_VBoxService,
418 enmStatus, 0 /* Flags */);
419 if (RT_FAILURE(rc))
420 {
421 VBoxServiceError("Could not report VBoxService status (%u), rc=%Rrc\n", enmStatus, rc);
422 return rc;
423 }
424 s_enmLastStatus = enmStatus;
425 }
426 return VINF_SUCCESS;
427}
428
429
430/**
431 * Gets a 32-bit value argument.
432 * @todo Get rid of this and VBoxServiceArgString() as soon as we have RTOpt handling.
433 *
434 * @returns 0 on success, non-zero exit code on error.
435 * @param argc The argument count.
436 * @param argv The argument vector
437 * @param psz Where in *pi to start looking for the value argument.
438 * @param pi Where to find and perhaps update the argument index.
439 * @param pu32 Where to store the 32-bit value.
440 * @param u32Min The minimum value.
441 * @param u32Max The maximum value.
442 */
443int VBoxServiceArgUInt32(int argc, char **argv, const char *psz, int *pi, uint32_t *pu32, uint32_t u32Min, uint32_t u32Max)
444{
445 if (*psz == ':' || *psz == '=')
446 psz++;
447 if (!*psz)
448 {
449 if (*pi + 1 >= argc)
450 return RTMsgErrorExit(RTEXITCODE_SYNTAX, "Missing value for the '%s' argument\n", argv[*pi]);
451 psz = argv[++*pi];
452 }
453
454 char *pszNext;
455 int rc = RTStrToUInt32Ex(psz, &pszNext, 0, pu32);
456 if (RT_FAILURE(rc) || *pszNext)
457 return RTMsgErrorExit(RTEXITCODE_SYNTAX, "Failed to convert interval '%s' to a number\n", psz);
458 if (*pu32 < u32Min || *pu32 > u32Max)
459 return RTMsgErrorExit(RTEXITCODE_SYNTAX, "The timesync interval of %RU32 seconds is out of range [%RU32..%RU32]\n",
460 *pu32, u32Min, u32Max);
461 return 0;
462}
463
464/** @todo Get rid of this and VBoxServiceArgUInt32() as soon as we have RTOpt handling. */
465int VBoxServiceArgString(int argc, char **argv, const char *psz, int *pi, char *pszBuf, size_t cbBuf)
466{
467 AssertPtrReturn(pszBuf, VERR_INVALID_POINTER);
468 AssertPtrReturn(cbBuf, VERR_INVALID_PARAMETER);
469
470 if (*psz == ':' || *psz == '=')
471 psz++;
472 if (!*psz)
473 {
474 if (*pi + 1 >= argc)
475 return RTMsgErrorExit(RTEXITCODE_SYNTAX, "Missing string for the '%s' argument\n", argv[*pi]);
476 psz = argv[++*pi];
477 }
478
479 if (!RTStrPrintf(pszBuf, cbBuf, "%s", psz))
480 return RTMsgErrorExit(RTEXITCODE_FAILURE, "String for '%s' argument too big\n", argv[*pi]);
481 return 0;
482}
483
484
485
486/**
487 * The service thread.
488 *
489 * @returns Whatever the worker function returns.
490 * @param ThreadSelf My thread handle.
491 * @param pvUser The service index.
492 */
493static DECLCALLBACK(int) vboxServiceThread(RTTHREAD ThreadSelf, void *pvUser)
494{
495 const unsigned i = (uintptr_t)pvUser;
496
497#ifndef RT_OS_WINDOWS
498 /*
499 * Block all signals for this thread. Only the main thread will handle signals.
500 */
501 sigset_t signalMask;
502 sigfillset(&signalMask);
503 pthread_sigmask(SIG_BLOCK, &signalMask, NULL);
504#endif
505
506 int rc = g_aServices[i].pDesc->pfnWorker(&g_aServices[i].fShutdown);
507 ASMAtomicXchgBool(&g_aServices[i].fShutdown, true);
508 RTThreadUserSignal(ThreadSelf);
509 return rc;
510}
511
512
513/**
514 * Lazily calls the pfnPreInit method on each service.
515 *
516 * @returns VBox status code, error message displayed.
517 */
518static RTEXITCODE vboxServiceLazyPreInit(void)
519{
520 for (unsigned j = 0; j < RT_ELEMENTS(g_aServices); j++)
521 if (!g_aServices[j].fPreInited)
522 {
523 int rc = g_aServices[j].pDesc->pfnPreInit();
524 if (RT_FAILURE(rc))
525 return VBoxServiceError("Service '%s' failed pre-init: %Rrc\n", g_aServices[j].pDesc->pszName, rc);
526 g_aServices[j].fPreInited = true;
527 }
528 return RTEXITCODE_SUCCESS;
529}
530
531
532/**
533 * Count the number of enabled services.
534 */
535static unsigned vboxServiceCountEnabledServices(void)
536{
537 unsigned cEnabled = 0;
538 for (unsigned i = 0; i < RT_ELEMENTS(g_aServices); i++)
539 cEnabled += g_aServices[i].fEnabled;
540 return cEnabled;
541}
542
543
544#ifdef RT_OS_WINDOWS
545static BOOL WINAPI VBoxServiceConsoleControlHandler(DWORD dwCtrlType)
546{
547 int rc = VINF_SUCCESS;
548 bool fEventHandled = FALSE;
549 switch (dwCtrlType)
550 {
551 /* User pressed CTRL+C or CTRL+BREAK or an external event was sent
552 * via GenerateConsoleCtrlEvent(). */
553 case CTRL_BREAK_EVENT:
554 case CTRL_CLOSE_EVENT:
555 case CTRL_C_EVENT:
556 VBoxServiceVerbose(2, "ControlHandler: Received break/close event\n");
557 rc = VBoxServiceStopServices();
558 fEventHandled = TRUE;
559 break;
560 default:
561 break;
562 /** @todo Add other events here. */
563 }
564
565 if (RT_FAILURE(rc))
566 VBoxServiceError("ControlHandler: Event %ld handled with error rc=%Rrc\n",
567 dwCtrlType, rc);
568 return fEventHandled;
569}
570#endif /* RT_OS_WINDOWS */
571
572
573/**
574 * Starts the service.
575 *
576 * @returns VBox status code, errors are fully bitched.
577 */
578int VBoxServiceStartServices(void)
579{
580 int rc;
581
582 VBoxServiceReportStatus(VBoxGuestFacilityStatus_Init);
583
584 /*
585 * Initialize the services.
586 */
587 VBoxServiceVerbose(2, "Initializing services ...\n");
588 for (unsigned j = 0; j < RT_ELEMENTS(g_aServices); j++)
589 if (g_aServices[j].fEnabled)
590 {
591 rc = g_aServices[j].pDesc->pfnInit();
592 if (RT_FAILURE(rc))
593 {
594 if (rc != VERR_SERVICE_DISABLED)
595 {
596 VBoxServiceError("Service '%s' failed to initialize: %Rrc\n",
597 g_aServices[j].pDesc->pszName, rc);
598 VBoxServiceReportStatus(VBoxGuestFacilityStatus_Failed);
599 return rc;
600 }
601 g_aServices[j].fEnabled = false;
602 VBoxServiceVerbose(0, "Service '%s' was disabled because of missing functionality\n",
603 g_aServices[j].pDesc->pszName);
604
605 }
606 }
607
608 /*
609 * Start the service(s).
610 */
611 VBoxServiceVerbose(2, "Starting services ...\n");
612 rc = VINF_SUCCESS;
613 for (unsigned j = 0; j < RT_ELEMENTS(g_aServices); j++)
614 {
615 if (!g_aServices[j].fEnabled)
616 continue;
617
618 VBoxServiceVerbose(2, "Starting service '%s' ...\n", g_aServices[j].pDesc->pszName);
619 rc = RTThreadCreate(&g_aServices[j].Thread, vboxServiceThread, (void *)(uintptr_t)j, 0,
620 RTTHREADTYPE_DEFAULT, RTTHREADFLAGS_WAITABLE, g_aServices[j].pDesc->pszName);
621 if (RT_FAILURE(rc))
622 {
623 VBoxServiceError("RTThreadCreate failed, rc=%Rrc\n", rc);
624 break;
625 }
626 g_aServices[j].fStarted = true;
627
628 /* Wait for the thread to initialize. */
629 /** @todo There is a race between waiting and checking
630 * the fShutdown flag of a thread here and processing
631 * the thread's actual worker loop. If the thread decides
632 * to exit the loop before we skipped the fShutdown check
633 * below the service will fail to start! */
634 /** @todo This presumably means either a one-shot service or that
635 * something has gone wrong. In the second case treating it as failure
636 * to start is probably right, so we need a way to signal the first
637 * rather than leaving the idle thread hanging around. A flag in the
638 * service description? */
639 RTThreadUserWait(g_aServices[j].Thread, 60 * 1000);
640 if (g_aServices[j].fShutdown)
641 {
642 VBoxServiceError("Service '%s' failed to start!\n", g_aServices[j].pDesc->pszName);
643 rc = VERR_GENERAL_FAILURE;
644 }
645 }
646
647 if (RT_SUCCESS(rc))
648 VBoxServiceVerbose(1, "All services started.\n");
649 else
650 {
651 VBoxServiceError("An error occcurred while the services!\n");
652 VBoxServiceReportStatus(VBoxGuestFacilityStatus_Failed);
653 }
654 return rc;
655}
656
657
658/**
659 * Stops and terminates the services.
660 *
661 * This should be called even when VBoxServiceStartServices fails so it can
662 * clean up anything that we succeeded in starting.
663 */
664int VBoxServiceStopServices(void)
665{
666 VBoxServiceReportStatus(VBoxGuestFacilityStatus_Terminating);
667
668 /*
669 * Signal all the services.
670 */
671 for (unsigned j = 0; j < RT_ELEMENTS(g_aServices); j++)
672 ASMAtomicWriteBool(&g_aServices[j].fShutdown, true);
673
674 /*
675 * Do the pfnStop callback on all running services.
676 */
677 for (unsigned j = 0; j < RT_ELEMENTS(g_aServices); j++)
678 if (g_aServices[j].fStarted)
679 {
680 VBoxServiceVerbose(3, "Calling stop function for service '%s' ...\n", g_aServices[j].pDesc->pszName);
681 g_aServices[j].pDesc->pfnStop();
682 }
683
684 VBoxServiceVerbose(3, "All stop functions for services called\n");
685
686 /*
687 * Wait for all the service threads to complete.
688 */
689 int rc = VINF_SUCCESS;
690 for (unsigned j = 0; j < RT_ELEMENTS(g_aServices); j++)
691 {
692 if (!g_aServices[j].fEnabled) /* Only stop services which were started before. */
693 continue;
694 if (g_aServices[j].Thread != NIL_RTTHREAD)
695 {
696 VBoxServiceVerbose(2, "Waiting for service '%s' to stop ...\n", g_aServices[j].pDesc->pszName);
697 int rc2 = VINF_SUCCESS;
698 for (int i = 0; i < 30; i++) /* Wait 30 seconds in total */
699 {
700 rc2 = RTThreadWait(g_aServices[j].Thread, 1000 /* Wait 1 second */, NULL);
701 if (RT_SUCCESS(rc2))
702 break;
703#ifdef RT_OS_WINDOWS
704 /* Notify SCM that it takes a bit longer ... */
705 VBoxServiceWinSetStopPendingStatus(i + j*32);
706#endif
707 }
708 if (RT_FAILURE(rc2))
709 {
710 VBoxServiceError("Service '%s' failed to stop. (%Rrc)\n", g_aServices[j].pDesc->pszName, rc2);
711 rc = rc2;
712 }
713 }
714 VBoxServiceVerbose(3, "Terminating service '%s' (%d) ...\n", g_aServices[j].pDesc->pszName, j);
715 g_aServices[j].pDesc->pfnTerm();
716 }
717
718#ifdef RT_OS_WINDOWS
719 /*
720 * Wake up and tell the main() thread that we're shutting down (it's
721 * sleeping in VBoxServiceMainWait).
722 */
723 ASMAtomicWriteBool(&g_fWindowsServiceShutdown, true);
724 if (g_hEvtWindowsService != NIL_RTSEMEVENT)
725 {
726 VBoxServiceVerbose(3, "Stopping the main thread...\n");
727 int rc2 = RTSemEventSignal(g_hEvtWindowsService);
728 AssertRC(rc2);
729 }
730#endif
731
732 VBoxServiceVerbose(2, "Stopping services returning: %Rrc\n", rc);
733 VBoxServiceReportStatus(RT_SUCCESS(rc) ? VBoxGuestFacilityStatus_Paused : VBoxGuestFacilityStatus_Failed);
734 return rc;
735}
736
737
738/**
739 * Block the main thread until the service shuts down.
740 */
741void VBoxServiceMainWait(void)
742{
743 int rc;
744
745 VBoxServiceReportStatus(VBoxGuestFacilityStatus_Active);
746
747#ifdef RT_OS_WINDOWS
748 /*
749 * Wait for the semaphore to be signalled.
750 */
751 VBoxServiceVerbose(1, "Waiting in main thread\n");
752 rc = RTSemEventCreate(&g_hEvtWindowsService);
753 AssertRC(rc);
754 while (!ASMAtomicReadBool(&g_fWindowsServiceShutdown))
755 {
756 rc = RTSemEventWait(g_hEvtWindowsService, RT_INDEFINITE_WAIT);
757 AssertRC(rc);
758 }
759 RTSemEventDestroy(g_hEvtWindowsService);
760 g_hEvtWindowsService = NIL_RTSEMEVENT;
761#else
762 /*
763 * Wait explicitly for a HUP, INT, QUIT, ABRT or TERM signal, blocking
764 * all important signals.
765 *
766 * The annoying EINTR/ERESTART loop is for the benefit of Solaris where
767 * sigwait returns when we receive a SIGCHLD. Kind of makes sense since
768 */
769 sigset_t signalMask;
770 sigemptyset(&signalMask);
771 sigaddset(&signalMask, SIGHUP);
772 sigaddset(&signalMask, SIGINT);
773 sigaddset(&signalMask, SIGQUIT);
774 sigaddset(&signalMask, SIGABRT);
775 sigaddset(&signalMask, SIGTERM);
776 pthread_sigmask(SIG_BLOCK, &signalMask, NULL);
777
778 int iSignal;
779 do
780 {
781 iSignal = -1;
782 rc = sigwait(&signalMask, &iSignal);
783 }
784 while ( rc == EINTR
785# ifdef ERESTART
786 || rc == ERESTART
787# endif
788 );
789
790 VBoxServiceVerbose(3, "VBoxServiceMainWait: Received signal %d (rc=%d)\n", iSignal, rc);
791#endif /* !RT_OS_WINDOWS */
792}
793
794
795int main(int argc, char **argv)
796{
797 RTEXITCODE rcExit;
798
799 /*
800 * Init globals and such.
801 */
802 int rc = RTR3InitExe(argc, &argv, 0);
803 if (RT_FAILURE(rc))
804 return RTMsgInitFailure(rc);
805 g_pszProgName = RTPathFilename(argv[0]);
806#ifdef DEBUG
807 rc = RTCritSectInit(&g_csLog);
808 AssertRC(rc);
809#endif
810
811#ifdef VBOXSERVICE_TOOLBOX
812 /*
813 * Run toolbox code before all other stuff since these things are simpler
814 * shell/file/text utility like programs that just happens to be inside
815 * VBoxService and shouldn't be subject to /dev/vboxguest, pid-files and
816 * global mutex restrictions.
817 */
818 if (VBoxServiceToolboxMain(argc, argv, &rcExit))
819 return rcExit;
820#endif
821
822 bool fUserSession = false;
823#ifdef VBOX_WITH_GUEST_CONTROL
824 /*
825 * Check if we're the specially spawned VBoxService.exe process that
826 * handles a guest control session.
827 */
828 if ( argc >= 2
829 && !RTStrICmp(argv[1], "guestsession"))
830 fUserSession = true;
831#endif
832
833 /*
834 * Connect to the kernel part before daemonizing so we can fail and
835 * complain if there is some kind of problem. We need to initialize the
836 * guest lib *before* we do the pre-init just in case one of services needs
837 * do to some initial stuff with it.
838 */
839 if (fUserSession)
840 rc = VbglR3InitUser();
841 else
842 rc = VbglR3Init();
843
844 if (RT_FAILURE(rc))
845 {
846 if (rc == VERR_ACCESS_DENIED)
847 return RTMsgErrorExit(RTEXITCODE_FAILURE, "Insufficient privileges to start %s! Please start with Administrator/root privileges!\n",
848 g_pszProgName);
849 return RTMsgErrorExit(RTEXITCODE_FAILURE, "VbglR3Init failed with rc=%Rrc\n", rc);
850 }
851
852#ifdef RT_OS_WINDOWS
853 /*
854 * Check if we're the specially spawned VBoxService.exe process that
855 * handles page fusion. This saves an extra executable.
856 */
857 if ( argc == 2
858 && !RTStrICmp(argv[1], "pagefusion"))
859 return VBoxServicePageSharingWorkerChild();
860#endif
861
862#ifdef VBOX_WITH_GUEST_CONTROL
863 /*
864 * Check if we're the specially spawned VBoxService.exe process that
865 * handles a guest control session.
866 */
867 if (fUserSession)
868 return VBoxServiceControlSessionForkInit(argc, argv);
869#endif
870
871 /*
872 * Parse the arguments.
873 *
874 * Note! This code predates RTGetOpt, thus the manual parsing.
875 */
876 bool fDaemonize = true;
877 bool fDaemonized = false;
878 for (int i = 1; i < argc; i++)
879 {
880 const char *psz = argv[i];
881 if (*psz != '-')
882 return RTMsgErrorExit(RTEXITCODE_SYNTAX, "Unknown argument '%s'\n", psz);
883 psz++;
884
885 /* translate long argument to short */
886 if (*psz == '-')
887 {
888 psz++;
889 size_t cch = strlen(psz);
890#define MATCHES(strconst) ( cch == sizeof(strconst) - 1 \
891 && !memcmp(psz, strconst, sizeof(strconst) - 1) )
892 if (MATCHES("foreground"))
893 psz = "f";
894 else if (MATCHES("verbose"))
895 psz = "v";
896 else if (MATCHES("version"))
897 psz = "V";
898 else if (MATCHES("help"))
899 psz = "h";
900 else if (MATCHES("interval"))
901 psz = "i";
902#ifdef RT_OS_WINDOWS
903 else if (MATCHES("register"))
904 psz = "r";
905 else if (MATCHES("unregister"))
906 psz = "u";
907#endif
908 else if (MATCHES("logfile"))
909 psz = "l";
910 else if (MATCHES("daemonized"))
911 {
912 fDaemonized = true;
913 continue;
914 }
915 else
916 {
917 bool fFound = false;
918
919 if (cch > sizeof("enable-") && !memcmp(psz, RT_STR_TUPLE("enable-")))
920 for (unsigned j = 0; !fFound && j < RT_ELEMENTS(g_aServices); j++)
921 if ((fFound = !RTStrICmp(psz + sizeof("enable-") - 1, g_aServices[j].pDesc->pszName)))
922 g_aServices[j].fEnabled = true;
923
924 if (cch > sizeof("disable-") && !memcmp(psz, RT_STR_TUPLE("disable-")))
925 for (unsigned j = 0; !fFound && j < RT_ELEMENTS(g_aServices); j++)
926 if ((fFound = !RTStrICmp(psz + sizeof("disable-") - 1, g_aServices[j].pDesc->pszName)))
927 g_aServices[j].fEnabled = false;
928
929 if (cch > sizeof("only-") && !memcmp(psz, RT_STR_TUPLE("only-")))
930 for (unsigned j = 0; j < RT_ELEMENTS(g_aServices); j++)
931 {
932 g_aServices[j].fEnabled = !RTStrICmp(psz + sizeof("only-") - 1, g_aServices[j].pDesc->pszName);
933 if (g_aServices[j].fEnabled)
934 fFound = true;
935 }
936
937 if (!fFound)
938 {
939 rcExit = vboxServiceLazyPreInit();
940 if (rcExit != RTEXITCODE_SUCCESS)
941 return rcExit;
942 for (unsigned j = 0; !fFound && j < RT_ELEMENTS(g_aServices); j++)
943 {
944 rc = g_aServices[j].pDesc->pfnOption(NULL, argc, argv, &i);
945 fFound = rc == VINF_SUCCESS;
946 if (fFound)
947 break;
948 if (rc != -1)
949 return rc;
950 }
951 }
952 if (!fFound)
953 return RTMsgErrorExit(RTEXITCODE_SYNTAX, "Unknown option '%s'\n", argv[i]);
954 continue;
955 }
956#undef MATCHES
957 }
958
959 /* handle the string of short options. */
960 do
961 {
962 switch (*psz)
963 {
964 case 'i':
965 rc = VBoxServiceArgUInt32(argc, argv, psz + 1, &i,
966 &g_DefaultInterval, 1, (UINT32_MAX / 1000) - 1);
967 if (rc)
968 return rc;
969 psz = NULL;
970 break;
971
972 case 'f':
973 fDaemonize = false;
974 break;
975
976 case 'v':
977 g_cVerbosity++;
978 break;
979
980 case 'V':
981 RTPrintf("%sr%s\n", RTBldCfgVersion(), RTBldCfgRevisionStr());
982 return RTEXITCODE_SUCCESS;
983
984 case 'h':
985 case '?':
986 return vboxServiceUsage();
987
988#ifdef RT_OS_WINDOWS
989 case 'r':
990 return VBoxServiceWinInstall();
991
992 case 'u':
993 return VBoxServiceWinUninstall();
994#endif
995
996 case 'l':
997 {
998 rc = VBoxServiceArgString(argc, argv, psz + 1, &i,
999 g_szLogFile, sizeof(g_szLogFile));
1000 if (rc)
1001 return rc;
1002 psz = NULL;
1003 break;
1004 }
1005
1006 default:
1007 {
1008 rcExit = vboxServiceLazyPreInit();
1009 if (rcExit != RTEXITCODE_SUCCESS)
1010 return rcExit;
1011
1012 bool fFound = false;
1013 for (unsigned j = 0; j < RT_ELEMENTS(g_aServices); j++)
1014 {
1015 rc = g_aServices[j].pDesc->pfnOption(&psz, argc, argv, &i);
1016 fFound = rc == VINF_SUCCESS;
1017 if (fFound)
1018 break;
1019 if (rc != -1)
1020 return rc;
1021 }
1022 if (!fFound)
1023 return RTMsgErrorExit(RTEXITCODE_SYNTAX, "Unknown option '%c' (%s)\n", *psz, argv[i]);
1024 break;
1025 }
1026 }
1027 } while (psz && *++psz);
1028 }
1029
1030 /* Check that at least one service is enabled. */
1031 if (vboxServiceCountEnabledServices() == 0)
1032 return RTMsgErrorExit(RTEXITCODE_SYNTAX, "At least one service must be enabled\n");
1033
1034 rc = VBoxServiceLogCreate(strlen(g_szLogFile) ? g_szLogFile : NULL);
1035 if (RT_FAILURE(rc))
1036 return RTMsgErrorExit(RTEXITCODE_FAILURE, "Failed to create release log \"%s\", rc=%Rrc\n",
1037 strlen(g_szLogFile) ? g_szLogFile : "<None>", rc);
1038
1039 /* Call pre-init if we didn't do it already. */
1040 rcExit = vboxServiceLazyPreInit();
1041 if (rcExit != RTEXITCODE_SUCCESS)
1042 return rcExit;
1043
1044#ifdef RT_OS_WINDOWS
1045 /*
1046 * Make sure only one instance of VBoxService runs at a time. Create a
1047 * global mutex for that.
1048 *
1049 * Note! The \\Global\ namespace was introduced with Win2K, thus the
1050 * version check.
1051 * Note! If the mutex exists CreateMutex will open it and set last error to
1052 * ERROR_ALREADY_EXISTS.
1053 */
1054 OSVERSIONINFOEX OSInfoEx;
1055 RT_ZERO(OSInfoEx);
1056 OSInfoEx.dwOSVersionInfoSize = sizeof(OSVERSIONINFOEX);
1057
1058 SetLastError(NO_ERROR);
1059 HANDLE hMutexAppRunning;
1060 if ( GetVersionEx((LPOSVERSIONINFO)&OSInfoEx)
1061 && OSInfoEx.dwPlatformId == VER_PLATFORM_WIN32_NT
1062 && OSInfoEx.dwMajorVersion >= 5 /* NT 5.0 a.k.a W2K */)
1063 hMutexAppRunning = CreateMutex(NULL, FALSE, "Global\\" VBOXSERVICE_NAME);
1064 else
1065 hMutexAppRunning = CreateMutex(NULL, FALSE, VBOXSERVICE_NAME);
1066 if (hMutexAppRunning == NULL)
1067 {
1068 DWORD dwErr = GetLastError();
1069 if ( dwErr == ERROR_ALREADY_EXISTS
1070 || dwErr == ERROR_ACCESS_DENIED)
1071 {
1072 VBoxServiceError("%s is already running! Terminating.\n", g_pszProgName);
1073 return RTEXITCODE_FAILURE;
1074 }
1075
1076 VBoxServiceError("CreateMutex failed with last error %u! Terminating.\n", GetLastError());
1077 return RTEXITCODE_FAILURE;
1078 }
1079
1080#else /* !RT_OS_WINDOWS */
1081 /** @todo Add PID file creation here? */
1082#endif /* !RT_OS_WINDOWS */
1083
1084 VBoxServiceVerbose(0, "%s r%s started. Verbose level = %d\n",
1085 RTBldCfgVersion(), RTBldCfgRevisionStr(), g_cVerbosity);
1086
1087 /*
1088 * Daemonize if requested.
1089 */
1090 if (fDaemonize && !fDaemonized)
1091 {
1092#ifdef RT_OS_WINDOWS
1093 VBoxServiceVerbose(2, "Starting service dispatcher ...\n");
1094 rcExit = VBoxServiceWinEnterCtrlDispatcher();
1095#else
1096 VBoxServiceVerbose(1, "Daemonizing...\n");
1097 rc = VbglR3Daemonize(false /* fNoChDir */, false /* fNoClose */,
1098 false /* fRespawn */, NULL /* pcRespawn */);
1099 if (RT_FAILURE(rc))
1100 return VBoxServiceError("Daemon failed: %Rrc\n", rc);
1101 /* in-child */
1102#endif
1103 }
1104#ifdef RT_OS_WINDOWS
1105 else
1106#endif
1107 {
1108 /*
1109 * Windows: We're running the service as a console application now. Start the
1110 * services, enter the main thread's run loop and stop them again
1111 * when it returns.
1112 *
1113 * POSIX: This is used for both daemons and console runs. Start all services
1114 * and return immediately.
1115 */
1116#ifdef RT_OS_WINDOWS
1117# ifndef RT_OS_NT4
1118 /* Install console control handler. */
1119 if (!SetConsoleCtrlHandler((PHANDLER_ROUTINE)VBoxServiceConsoleControlHandler, TRUE /* Add handler */))
1120 {
1121 VBoxServiceError("Unable to add console control handler, error=%ld\n", GetLastError());
1122 /* Just skip this error, not critical. */
1123 }
1124# endif /* !RT_OS_NT4 */
1125#endif /* RT_OS_WINDOWS */
1126 rc = VBoxServiceStartServices();
1127 rcExit = RT_SUCCESS(rc) ? RTEXITCODE_SUCCESS : RTEXITCODE_FAILURE;
1128 if (RT_SUCCESS(rc))
1129 VBoxServiceMainWait();
1130#ifdef RT_OS_WINDOWS
1131# ifndef RT_OS_NT4
1132 /* Uninstall console control handler. */
1133 if (!SetConsoleCtrlHandler((PHANDLER_ROUTINE)NULL, FALSE /* Remove handler */))
1134 {
1135 VBoxServiceError("Unable to remove console control handler, error=%ld\n", GetLastError());
1136 /* Just skip this error, not critical. */
1137 }
1138# endif /* !RT_OS_NT4 */
1139#else /* !RT_OS_WINDOWS */
1140 /* On Windows - since we're running as a console application - we already stopped all services
1141 * through the console control handler. So only do the stopping of services here on other platforms
1142 * where the break/shutdown/whatever signal was just received. */
1143 VBoxServiceStopServices();
1144#endif /* RT_OS_WINDOWS */
1145 }
1146 VBoxServiceReportStatus(VBoxGuestFacilityStatus_Terminated);
1147
1148#ifdef RT_OS_WINDOWS
1149 /*
1150 * Cleanup mutex.
1151 */
1152 CloseHandle(hMutexAppRunning);
1153#endif
1154
1155 VBoxServiceVerbose(0, "Ended.\n");
1156
1157#ifdef DEBUG
1158 RTCritSectDelete(&g_csLog);
1159 //RTMemTrackerDumpAllToStdOut();
1160#endif
1161
1162 VBoxServiceLogDestroy();
1163
1164 return rcExit;
1165}
1166
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