VirtualBox

source: vbox/trunk/src/VBox/Frontends/VBoxAutostart/VBoxAutostart-win.cpp@ 61673

Last change on this file since 61673 was 60759, checked in by vboxsync, 9 years ago

Frontends: various cleanups, mostly ATL related, remove code for no longer existing API method

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 33.7 KB
Line 
1/* $Id: VBoxAutostart-win.cpp 60759 2016-04-29 11:19:23Z vboxsync $ */
2/** @file
3 * VirtualBox Autostart Service - Windows Specific Code.
4 */
5
6/*
7 * Copyright (C) 2012-2016 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* Header Files *
21*********************************************************************************************************************************/
22#include <Windows.h>
23#include <tchar.h>
24
25#include <VBox/com/com.h>
26#include <VBox/com/string.h>
27#include <VBox/com/Guid.h>
28#include <VBox/com/array.h>
29#include <VBox/com/ErrorInfo.h>
30#include <VBox/com/errorprint.h>
31
32#include <VBox/com/NativeEventQueue.h>
33#include <VBox/com/listeners.h>
34#include <VBox/com/VirtualBox.h>
35
36#include <VBox/log.h>
37#include <VBox/version.h>
38#include <iprt/string.h>
39#include <iprt/mem.h>
40#include <iprt/initterm.h>
41#include <iprt/stream.h>
42#include <iprt/getopt.h>
43#include <iprt/semaphore.h>
44#include <iprt/thread.h>
45
46#include "VBoxAutostart.h"
47
48
49/*********************************************************************************************************************************
50* Defined Constants And Macros *
51*********************************************************************************************************************************/
52/** The service name. */
53#define AUTOSTART_SERVICE_NAME "VBoxAutostartSvc"
54/** The service display name. */
55#define AUTOSTART_SERVICE_DISPLAY_NAME "VirtualBox Autostart Service"
56
57ComPtr<IVirtualBoxClient> g_pVirtualBoxClient = NULL;
58bool g_fVerbose = false;
59ComPtr<IVirtualBox> g_pVirtualBox = NULL;
60ComPtr<ISession> g_pSession = NULL;
61
62
63/*********************************************************************************************************************************
64* Global Variables *
65*********************************************************************************************************************************/
66/** The service control handler handle. */
67static SERVICE_STATUS_HANDLE g_hSupSvcWinCtrlHandler = NULL;
68/** The service status. */
69static uint32_t volatile g_u32SupSvcWinStatus = SERVICE_STOPPED;
70/** The semaphore the main service thread is waiting on in autostartSvcWinServiceMain. */
71static RTSEMEVENTMULTI g_hSupSvcWinEvent = NIL_RTSEMEVENTMULTI;
72
73
74/*********************************************************************************************************************************
75* Internal Functions *
76*********************************************************************************************************************************/
77static SC_HANDLE autostartSvcWinOpenSCManager(const char *pszAction, DWORD dwAccess);
78
79/**
80 * Print out progress on the console.
81 *
82 * This runs the main event queue every now and then to prevent piling up
83 * unhandled things (which doesn't cause real problems, just makes things
84 * react a little slower than in the ideal case).
85 */
86DECLHIDDEN(HRESULT) showProgress(ComPtr<IProgress> progress)
87{
88 using namespace com;
89
90 BOOL fCompleted = FALSE;
91 ULONG ulCurrentPercent = 0;
92 ULONG ulLastPercent = 0;
93
94 ULONG ulLastOperationPercent = (ULONG)-1;
95
96 ULONG ulLastOperation = (ULONG)-1;
97 Bstr bstrOperationDescription;
98
99 NativeEventQueue::getMainEventQueue()->processEventQueue(0);
100
101 ULONG cOperations = 1;
102 HRESULT hrc = progress->COMGETTER(OperationCount)(&cOperations);
103 if (FAILED(hrc))
104 return hrc;
105
106 /* setup signal handling if cancelable */
107 bool fCanceledAlready = false;
108 BOOL fCancelable;
109 hrc = progress->COMGETTER(Cancelable)(&fCancelable);
110 if (FAILED(hrc))
111 fCancelable = FALSE;
112
113 hrc = progress->COMGETTER(Completed(&fCompleted));
114 while (SUCCEEDED(hrc))
115 {
116 progress->COMGETTER(Percent(&ulCurrentPercent));
117
118 if (fCompleted)
119 break;
120
121 /* process async cancelation */
122 if (!fCanceledAlready)
123 {
124 hrc = progress->Cancel();
125 if (SUCCEEDED(hrc))
126 fCanceledAlready = true;
127 }
128
129 /* make sure the loop is not too tight */
130 progress->WaitForCompletion(100);
131
132 NativeEventQueue::getMainEventQueue()->processEventQueue(0);
133 hrc = progress->COMGETTER(Completed(&fCompleted));
134 }
135
136 /* complete the line. */
137 LONG iRc = E_FAIL;
138 hrc = progress->COMGETTER(ResultCode)(&iRc);
139 if (SUCCEEDED(hrc))
140 {
141 hrc = iRc;
142 }
143
144 return hrc;
145}
146
147DECLHIDDEN(void) autostartSvcOsLogStr(const char *pszMsg, AUTOSTARTLOGTYPE enmLogType)
148{
149 HANDLE hEventLog = RegisterEventSourceA(NULL /* local computer */, "VBoxAutostartSvc");
150 AssertReturnVoid(hEventLog != NULL);
151 WORD wType = 0;
152 const char *apsz[2];
153 apsz[0] = "VBoxAutostartSvc";
154 apsz[1] = pszMsg;
155
156 switch (enmLogType)
157 {
158 case AUTOSTARTLOGTYPE_INFO:
159 wType = 0;
160 break;
161 case AUTOSTARTLOGTYPE_ERROR:
162 wType = EVENTLOG_ERROR_TYPE;
163 break;
164 case AUTOSTARTLOGTYPE_WARNING:
165 wType = EVENTLOG_WARNING_TYPE;
166 break;
167 case AUTOSTARTLOGTYPE_VERBOSE:
168 if (!g_fVerbose)
169 return;
170 wType = EVENTLOG_INFORMATION_TYPE;
171 break;
172 default:
173 AssertMsgFailed(("Invalid log type %d\n", enmLogType));
174 }
175
176 BOOL fRc = ReportEventA(hEventLog, /* hEventLog */
177 wType, /* wType */
178 0, /* wCategory */
179 0 /** @todo mc */, /* dwEventID */
180 NULL, /* lpUserSid */
181 RT_ELEMENTS(apsz), /* wNumStrings */
182 0, /* dwDataSize */
183 apsz, /* lpStrings */
184 NULL); /* lpRawData */
185 AssertMsg(fRc, ("%d\n", GetLastError()));
186 DeregisterEventSource(hEventLog);
187}
188
189/**
190 * Opens the service control manager.
191 *
192 * When this fails, an error message will be displayed.
193 *
194 * @returns Valid handle on success.
195 * NULL on failure, will display an error message.
196 *
197 * @param pszAction The action which is requesting access to SCM.
198 * @param dwAccess The desired access.
199 */
200static SC_HANDLE autostartSvcWinOpenSCManager(const char *pszAction, DWORD dwAccess)
201{
202 SC_HANDLE hSCM = OpenSCManager(NULL /* lpMachineName*/, NULL /* lpDatabaseName */, dwAccess);
203 if (hSCM == NULL)
204 {
205 DWORD err = GetLastError();
206 switch (err)
207 {
208 case ERROR_ACCESS_DENIED:
209 autostartSvcDisplayError("%s - OpenSCManager failure: access denied\n", pszAction);
210 break;
211 default:
212 autostartSvcDisplayError("%s - OpenSCManager failure: %d\n", pszAction, err);
213 break;
214 }
215 }
216 return hSCM;
217}
218
219
220/**
221 * Opens the service.
222 *
223 * Last error is preserved on failure and set to 0 on success.
224 *
225 * @returns Valid service handle on success.
226 * NULL on failure, will display an error message unless it's ignored.
227 *
228 * @param pszAction The action which is requesting access to the service.
229 * @param dwSCMAccess The service control manager access.
230 * @param dwSVCAccess The desired service access.
231 * @param cIgnoredErrors The number of ignored errors.
232 * @param ... Errors codes that should not cause a message to be displayed.
233 */
234static SC_HANDLE autostartSvcWinOpenService(const char *pszAction, DWORD dwSCMAccess, DWORD dwSVCAccess,
235 unsigned cIgnoredErrors, ...)
236{
237 SC_HANDLE hSCM = autostartSvcWinOpenSCManager(pszAction, dwSCMAccess);
238 if (!hSCM)
239 return NULL;
240
241 SC_HANDLE hSvc = OpenServiceA(hSCM, AUTOSTART_SERVICE_NAME, dwSVCAccess);
242 if (hSvc)
243 {
244 CloseServiceHandle(hSCM);
245 SetLastError(0);
246 }
247 else
248 {
249 DWORD err = GetLastError();
250 bool fIgnored = false;
251 va_list va;
252 va_start(va, cIgnoredErrors);
253 while (!fIgnored && cIgnoredErrors-- > 0)
254 fIgnored = va_arg(va, long) == err;
255 va_end(va);
256 if (!fIgnored)
257 {
258 switch (err)
259 {
260 case ERROR_ACCESS_DENIED:
261 autostartSvcDisplayError("%s - OpenService failure: access denied\n", pszAction);
262 break;
263 case ERROR_SERVICE_DOES_NOT_EXIST:
264 autostartSvcDisplayError("%s - OpenService failure: The service does not exist. Reinstall it.\n", pszAction);
265 break;
266 default:
267 autostartSvcDisplayError("%s - OpenService failure: %d\n", pszAction, err);
268 break;
269 }
270 }
271
272 CloseServiceHandle(hSCM);
273 SetLastError(err);
274 }
275 return hSvc;
276}
277
278static RTEXITCODE autostartSvcWinInterrogate(int argc, char **argv)
279{
280 RTPrintf("VBoxAutostartSvc: The \"interrogate\" action is not implemented.\n");
281 return RTEXITCODE_FAILURE;
282}
283
284
285static RTEXITCODE autostartSvcWinStop(int argc, char **argv)
286{
287 RTPrintf("VBoxAutostartSvc: The \"stop\" action is not implemented.\n");
288 return RTEXITCODE_FAILURE;
289}
290
291
292static RTEXITCODE autostartSvcWinContinue(int argc, char **argv)
293{
294 RTPrintf("VBoxAutostartSvc: The \"continue\" action is not implemented.\n");
295 return RTEXITCODE_FAILURE;
296}
297
298
299static RTEXITCODE autostartSvcWinPause(int argc, char **argv)
300{
301 RTPrintf("VBoxAutostartSvc: The \"pause\" action is not implemented.\n");
302 return RTEXITCODE_FAILURE;
303}
304
305
306static RTEXITCODE autostartSvcWinStart(int argc, char **argv)
307{
308 RTPrintf("VBoxAutostartSvc: The \"start\" action is not implemented.\n");
309 return RTEXITCODE_SUCCESS;
310}
311
312
313static RTEXITCODE autostartSvcWinQueryDescription(int argc, char **argv)
314{
315 RTPrintf("VBoxAutostartSvc: The \"qdescription\" action is not implemented.\n");
316 return RTEXITCODE_FAILURE;
317}
318
319
320static RTEXITCODE autostartSvcWinQueryConfig(int argc, char **argv)
321{
322 RTPrintf("VBoxAutostartSvc: The \"qconfig\" action is not implemented.\n");
323 return RTEXITCODE_FAILURE;
324}
325
326
327static RTEXITCODE autostartSvcWinDisable(int argc, char **argv)
328{
329 RTPrintf("VBoxAutostartSvc: The \"disable\" action is not implemented.\n");
330 return RTEXITCODE_FAILURE;
331}
332
333static RTEXITCODE autostartSvcWinEnable(int argc, char **argv)
334{
335 RTPrintf("VBoxAutostartSvc: The \"enable\" action is not implemented.\n");
336 return RTEXITCODE_FAILURE;
337}
338
339
340/**
341 * Handle the 'delete' action.
342 *
343 * @returns RTEXITCODE_SUCCESS or RTEXITCODE_FAILURE.
344 * @param argc The action argument count.
345 * @param argv The action argument vector.
346 */
347static int autostartSvcWinDelete(int argc, char **argv)
348{
349 /*
350 * Parse the arguments.
351 */
352 bool fVerbose = false;
353 static const RTGETOPTDEF s_aOptions[] =
354 {
355 { "--verbose", 'v', RTGETOPT_REQ_NOTHING }
356 };
357 int ch;
358 unsigned iArg = 0;
359 RTGETOPTUNION Value;
360 RTGETOPTSTATE GetState;
361 RTGetOptInit(&GetState, argc, argv, s_aOptions, RT_ELEMENTS(s_aOptions), 0, RTGETOPTINIT_FLAGS_NO_STD_OPTS);
362 while ((ch = RTGetOpt(&GetState, &Value)))
363 {
364 switch (ch)
365 {
366 case 'v':
367 fVerbose = true;
368 break;
369 case VINF_GETOPT_NOT_OPTION:
370 return autostartSvcDisplayTooManyArgsError("delete", argc, argv, iArg);
371 default:
372 return autostartSvcDisplayGetOptError("delete", ch, argc, argv, iArg, &Value);
373 }
374
375 iArg++;
376 }
377
378 /*
379 * Create the service.
380 */
381 int rc = RTEXITCODE_FAILURE;
382 SC_HANDLE hSvc = autostartSvcWinOpenService("delete", SERVICE_CHANGE_CONFIG, DELETE,
383 1, ERROR_SERVICE_DOES_NOT_EXIST);
384 if (hSvc)
385 {
386 if (DeleteService(hSvc))
387 {
388 RTPrintf("Successfully deleted the %s service.\n", AUTOSTART_SERVICE_NAME);
389 rc = RTEXITCODE_SUCCESS;
390 }
391 else
392 autostartSvcDisplayError("delete - DeleteService failed, err=%d.\n", GetLastError());
393 CloseServiceHandle(hSvc);
394 }
395 else if (GetLastError() == ERROR_SERVICE_DOES_NOT_EXIST)
396 {
397
398 if (fVerbose)
399 RTPrintf("The service %s was not installed, nothing to be done.", AUTOSTART_SERVICE_NAME);
400 else
401 RTPrintf("Successfully deleted the %s service.\n", AUTOSTART_SERVICE_NAME);
402 rc = RTEXITCODE_SUCCESS;
403 }
404 return rc;
405}
406
407
408/**
409 * Handle the 'create' action.
410 *
411 * @returns 0 or 1.
412 * @param argc The action argument count.
413 * @param argv The action argument vector.
414 */
415static RTEXITCODE autostartSvcWinCreate(int argc, char **argv)
416{
417 /*
418 * Parse the arguments.
419 */
420 bool fVerbose = false;
421 const char *pszUser = NULL;
422 const char *pszPwd = NULL;
423 static const RTGETOPTDEF s_aOptions[] =
424 {
425 { "--verbose", 'v', RTGETOPT_REQ_NOTHING },
426 { "--user", 'u', RTGETOPT_REQ_STRING },
427 { "--password", 'p', RTGETOPT_REQ_STRING }
428 };
429 int iArg = 0;
430 int ch;
431 RTGETOPTUNION Value;
432 RTGETOPTSTATE GetState;
433 RTGetOptInit(&GetState, argc, argv, s_aOptions, RT_ELEMENTS(s_aOptions), 0, RTGETOPTINIT_FLAGS_NO_STD_OPTS);
434 while ((ch = RTGetOpt(&GetState, &Value)))
435 {
436 switch (ch)
437 {
438 case 'v':
439 fVerbose = true;
440 break;
441 case 'u':
442 pszUser = Value.psz;
443 break;
444 case 'p':
445 pszPwd = Value.psz;
446 break;
447 default:
448 return autostartSvcDisplayGetOptError("create", ch, argc, argv, iArg, &Value);
449 }
450 iArg++;
451 }
452 if (iArg != argc)
453 return autostartSvcDisplayTooManyArgsError("create", argc, argv, iArg);
454
455 /*
456 * Create the service.
457 */
458 RTEXITCODE rc = RTEXITCODE_FAILURE;
459 SC_HANDLE hSCM = autostartSvcWinOpenSCManager("create", SC_MANAGER_CREATE_SERVICE); /*SC_MANAGER_ALL_ACCESS*/
460 if (hSCM)
461 {
462 char szExecPath[MAX_PATH];
463 if (GetModuleFileNameA(NULL /* the executable */, szExecPath, sizeof(szExecPath)))
464 {
465 if (fVerbose)
466 RTPrintf("Creating the %s service, binary \"%s\"...\n",
467 AUTOSTART_SERVICE_NAME, szExecPath); /* yea, the binary name isn't UTF-8, but wtf. */
468
469 SC_HANDLE hSvc = CreateServiceA(hSCM, /* hSCManager */
470 AUTOSTART_SERVICE_NAME, /* lpServiceName */
471 AUTOSTART_SERVICE_DISPLAY_NAME, /* lpDisplayName */
472 SERVICE_CHANGE_CONFIG | SERVICE_QUERY_STATUS | SERVICE_QUERY_CONFIG, /* dwDesiredAccess */
473 SERVICE_WIN32_OWN_PROCESS, /* dwServiceType ( | SERVICE_INTERACTIVE_PROCESS? ) */
474 SERVICE_AUTO_START, /* dwStartType */
475 SERVICE_ERROR_NORMAL, /* dwErrorControl */
476 szExecPath, /* lpBinaryPathName */
477 NULL, /* lpLoadOrderGroup */
478 NULL, /* lpdwTagId */
479 NULL, /* lpDependencies */
480 pszUser, /* lpServiceStartName (NULL => LocalSystem) */
481 pszPwd); /* lpPassword */
482 if (hSvc)
483 {
484 RTPrintf("Successfully created the %s service.\n", AUTOSTART_SERVICE_NAME);
485 /** @todo Set the service description or it'll look weird in the vista service manager.
486 * Anything else that should be configured? Start access or something? */
487 rc = RTEXITCODE_SUCCESS;
488 CloseServiceHandle(hSvc);
489 }
490 else
491 {
492 DWORD err = GetLastError();
493 switch (err)
494 {
495 case ERROR_SERVICE_EXISTS:
496 autostartSvcDisplayError("create - The service already exists.\n");
497 break;
498 default:
499 autostartSvcDisplayError("create - CreateService failed, err=%d.\n", GetLastError());
500 break;
501 }
502 }
503 CloseServiceHandle(hSvc);
504 }
505 else
506 autostartSvcDisplayError("create - Failed to obtain the executable path: %d\n", GetLastError());
507 }
508 return rc;
509}
510
511
512/**
513 * Sets the service status, just a SetServiceStatus Wrapper.
514 *
515 * @returns See SetServiceStatus.
516 * @param dwStatus The current status.
517 * @param iWaitHint The wait hint, if < 0 then supply a default.
518 * @param dwExitCode The service exit code.
519 */
520static bool autostartSvcWinSetServiceStatus(DWORD dwStatus, int iWaitHint, DWORD dwExitCode)
521{
522 SERVICE_STATUS SvcStatus;
523 SvcStatus.dwServiceType = SERVICE_WIN32_OWN_PROCESS;
524 SvcStatus.dwWin32ExitCode = dwExitCode;
525 SvcStatus.dwServiceSpecificExitCode = 0;
526 SvcStatus.dwWaitHint = iWaitHint >= 0 ? iWaitHint : 3000;
527 SvcStatus.dwCurrentState = dwStatus;
528 LogFlow(("autostartSvcWinSetServiceStatus: %d -> %d\n", g_u32SupSvcWinStatus, dwStatus));
529 g_u32SupSvcWinStatus = dwStatus;
530 switch (dwStatus)
531 {
532 case SERVICE_START_PENDING:
533 SvcStatus.dwControlsAccepted = 0;
534 break;
535 default:
536 SvcStatus.dwControlsAccepted = SERVICE_ACCEPT_STOP;
537 break;
538 }
539
540 static DWORD dwCheckPoint = 0;
541 switch (dwStatus)
542 {
543 case SERVICE_RUNNING:
544 case SERVICE_STOPPED:
545 SvcStatus.dwCheckPoint = 0;
546 default:
547 SvcStatus.dwCheckPoint = ++dwCheckPoint;
548 break;
549 }
550 return SetServiceStatus(g_hSupSvcWinCtrlHandler, &SvcStatus) != FALSE;
551}
552
553
554/**
555 * Service control handler (extended).
556 *
557 * @returns Windows status (see HandlerEx).
558 * @retval NO_ERROR if handled.
559 * @retval ERROR_CALL_NOT_IMPLEMENTED if not handled.
560 *
561 * @param dwControl The control code.
562 * @param dwEventType Event type. (specific to the control?)
563 * @param pvEventData Event data, specific to the event.
564 * @param pvContext The context pointer registered with the handler.
565 * Currently not used.
566 */
567static DWORD WINAPI autostartSvcWinServiceCtrlHandlerEx(DWORD dwControl, DWORD dwEventType, LPVOID pvEventData, LPVOID pvContext)
568{
569 LogFlow(("autostartSvcWinServiceCtrlHandlerEx: dwControl=%#x dwEventType=%#x pvEventData=%p\n",
570 dwControl, dwEventType, pvEventData));
571
572 switch (dwControl)
573 {
574 /*
575 * Interrogate the service about it's current status.
576 * MSDN says that this should just return NO_ERROR and does
577 * not need to set the status again.
578 */
579 case SERVICE_CONTROL_INTERROGATE:
580 return NO_ERROR;
581
582 /*
583 * Request to stop the service.
584 */
585 case SERVICE_CONTROL_STOP:
586 {
587 /*
588 * Check if the real services can be stopped and then tell them to stop.
589 */
590 autostartSvcWinSetServiceStatus(SERVICE_STOP_PENDING, 3000, NO_ERROR);
591 /*
592 * Notify the main thread that we're done, it will wait for the
593 * VMs to stop, and set the windows service status to SERVICE_STOPPED
594 * and return.
595 */
596 int rc = RTSemEventMultiSignal(g_hSupSvcWinEvent);
597 if (RT_FAILURE(rc))
598 autostartSvcLogError("SERVICE_CONTROL_STOP: RTSemEventMultiSignal failed, %Rrc\n", rc);
599
600 return NO_ERROR;
601 }
602
603 case SERVICE_CONTROL_PAUSE:
604 case SERVICE_CONTROL_CONTINUE:
605 case SERVICE_CONTROL_SHUTDOWN:
606 case SERVICE_CONTROL_PARAMCHANGE:
607 case SERVICE_CONTROL_NETBINDADD:
608 case SERVICE_CONTROL_NETBINDREMOVE:
609 case SERVICE_CONTROL_NETBINDENABLE:
610 case SERVICE_CONTROL_NETBINDDISABLE:
611 case SERVICE_CONTROL_DEVICEEVENT:
612 case SERVICE_CONTROL_HARDWAREPROFILECHANGE:
613 case SERVICE_CONTROL_POWEREVENT:
614 case SERVICE_CONTROL_SESSIONCHANGE:
615#ifdef SERVICE_CONTROL_PRESHUTDOWN /* vista */
616 case SERVICE_CONTROL_PRESHUTDOWN:
617#endif
618 default:
619 return ERROR_CALL_NOT_IMPLEMENTED;
620 }
621
622 NOREF(dwEventType);
623 NOREF(pvEventData);
624 NOREF(pvContext);
625 return NO_ERROR;
626}
627
628static int autostartWorker(RTTHREAD ThreadSelf, void *pvUser)
629{
630 int rc = autostartSetup();
631
632 /** @todo: Implement config options. */
633 rc = autostartStartMain(NULL);
634 if (RT_FAILURE(rc))
635 autostartSvcLogError("Starting VMs failed, rc=%Rrc", rc);
636
637 return rc;
638}
639
640/**
641 * Windows Service Main.
642 *
643 * This is invoked when the service is started and should not return until
644 * the service has been stopped.
645 *
646 * @param cArgs Argument count.
647 * @param papszArgs Argument vector.
648 */
649static VOID WINAPI autostartSvcWinServiceMain(DWORD cArgs, LPTSTR *papszArgs)
650{
651 LogFlowFuncEnter();
652
653 /*
654 * Register the control handler function for the service and report to SCM.
655 */
656 Assert(g_u32SupSvcWinStatus == SERVICE_STOPPED);
657 g_hSupSvcWinCtrlHandler = RegisterServiceCtrlHandlerExA(AUTOSTART_SERVICE_NAME, autostartSvcWinServiceCtrlHandlerEx, NULL);
658 if (g_hSupSvcWinCtrlHandler)
659 {
660 DWORD err = ERROR_GEN_FAILURE;
661 if (autostartSvcWinSetServiceStatus(SERVICE_START_PENDING, 3000, NO_ERROR))
662 {
663 if (cArgs == 1)
664 {
665 /*
666 * Create the event semaphore we'll be waiting on and
667 * then instantiate the actual services.
668 */
669 int rc = RTSemEventMultiCreate(&g_hSupSvcWinEvent);
670 if (RT_SUCCESS(rc))
671 {
672 /*
673 * Update the status and enter the work loop.
674 */
675 if (autostartSvcWinSetServiceStatus(SERVICE_RUNNING, 0, 0))
676 {
677 LogFlow(("autostartSvcWinServiceMain: calling RTSemEventMultiWait\n"));
678 RTTHREAD hWorker;
679 RTThreadCreate(&hWorker, autostartWorker, NULL, 0, RTTHREADTYPE_DEFAULT, 0, "WorkerThread");
680
681 LogFlow(("autostartSvcWinServiceMain: woke up\n"));
682 err = NO_ERROR;
683 rc = RTSemEventMultiWait(g_hSupSvcWinEvent, RT_INDEFINITE_WAIT);
684 if (RT_SUCCESS(rc))
685 {
686 LogFlow(("autostartSvcWinServiceMain: woke up\n"));
687 /** @todo: Autostop part. */
688 err = NO_ERROR;
689 }
690 else
691 autostartSvcLogError("RTSemEventWait failed, rc=%Rrc", rc);
692
693 autostartShutdown();
694 }
695 else
696 {
697 err = GetLastError();
698 autostartSvcLogError("SetServiceStatus failed, err=%d", err);
699 }
700
701 RTSemEventMultiDestroy(g_hSupSvcWinEvent);
702 g_hSupSvcWinEvent = NIL_RTSEMEVENTMULTI;
703 }
704 else
705 autostartSvcLogError("RTSemEventMultiCreate failed, rc=%Rrc", rc);
706 }
707 else
708 autostartSvcLogTooManyArgsError("main", cArgs, NULL, 0);
709 }
710 else
711 {
712 err = GetLastError();
713 autostartSvcLogError("SetServiceStatus failed, err=%d", err);
714 }
715 autostartSvcWinSetServiceStatus(SERVICE_STOPPED, 0, err);
716 }
717 else
718 autostartSvcLogError("RegisterServiceCtrlHandlerEx failed, err=%d", GetLastError());
719
720 LogFlowFuncLeave();
721}
722
723
724/**
725 * Handle the 'create' action.
726 *
727 * @returns RTEXITCODE_SUCCESS or RTEXITCODE_FAILURE.
728 * @param argc The action argument count.
729 * @param argv The action argument vector.
730 */
731static int autostartSvcWinRunIt(int argc, char **argv)
732{
733 LogFlowFuncEnter();
734
735 /*
736 * Initialize release logging.
737 */
738 /** @todo release logging of the system-wide service. */
739
740 /*
741 * Parse the arguments.
742 */
743 static const RTGETOPTDEF s_aOptions[] =
744 {
745 { "--dummy", 'd', RTGETOPT_REQ_NOTHING }
746 };
747 int iArg = 0;
748 int ch;
749 RTGETOPTUNION Value;
750 RTGETOPTSTATE GetState;
751 RTGetOptInit(&GetState, argc, argv, s_aOptions, RT_ELEMENTS(s_aOptions), 0, RTGETOPTINIT_FLAGS_NO_STD_OPTS);
752 while ((ch = RTGetOpt(&GetState, &Value)))
753 switch (ch)
754 {
755 default: return autostartSvcDisplayGetOptError("runit", ch, argc, argv, iArg, &Value);
756 }
757 if (iArg != argc)
758 return autostartSvcDisplayTooManyArgsError("runit", argc, argv, iArg);
759
760 /*
761 * Register the service with the service control manager
762 * and start dispatching requests from it (all done by the API).
763 */
764 static SERVICE_TABLE_ENTRY const s_aServiceStartTable[] =
765 {
766 { _T(AUTOSTART_SERVICE_NAME), autostartSvcWinServiceMain },
767 { NULL, NULL}
768 };
769 if (StartServiceCtrlDispatcher(&s_aServiceStartTable[0]))
770 {
771 LogFlowFuncLeave();
772 return RTEXITCODE_SUCCESS; /* told to quit, so quit. */
773 }
774
775 DWORD err = GetLastError();
776 switch (err)
777 {
778 case ERROR_FAILED_SERVICE_CONTROLLER_CONNECT:
779 autostartSvcWinServiceMain(0, NULL);//autostartSvcDisplayError("Cannot run a service from the command line. Use the 'start' action to start it the right way.\n");
780 break;
781 default:
782 autostartSvcLogError("StartServiceCtrlDispatcher failed, err=%d", err);
783 break;
784 }
785 return RTEXITCODE_FAILURE;
786}
787
788
789/**
790 * Show the version info.
791 *
792 * @returns RTEXITCODE_SUCCESS.
793 */
794static RTEXITCODE autostartSvcWinShowVersion(int argc, char **argv)
795{
796 /*
797 * Parse the arguments.
798 */
799 bool fBrief = false;
800 static const RTGETOPTDEF s_aOptions[] =
801 {
802 { "--brief", 'b', RTGETOPT_REQ_NOTHING }
803 };
804 int iArg = 0;
805 int ch;
806 RTGETOPTUNION Value;
807 RTGETOPTSTATE GetState;
808 RTGetOptInit(&GetState, argc, argv, s_aOptions, RT_ELEMENTS(s_aOptions), 0, RTGETOPTINIT_FLAGS_NO_STD_OPTS);
809 while ((ch = RTGetOpt(&GetState, &Value)))
810 switch (ch)
811 {
812 case 'b': fBrief = true; break;
813 default: return autostartSvcDisplayGetOptError("version", ch, argc, argv, iArg, &Value);
814
815 }
816 if (iArg != argc)
817 return autostartSvcDisplayTooManyArgsError("version", argc, argv, iArg);
818
819 /*
820 * Do the printing.
821 */
822 if (fBrief)
823 RTPrintf("%s\n", VBOX_VERSION_STRING);
824 else
825 RTPrintf("VirtualBox Autostart Service Version %s\n"
826 "(C) 2012 Oracle Corporation\n"
827 "All rights reserved.\n",
828 VBOX_VERSION_STRING);
829 return RTEXITCODE_SUCCESS;
830}
831
832
833/**
834 * Show the usage help screen.
835 *
836 * @returns RTEXITCODE_SUCCESS.
837 */
838static RTEXITCODE autostartSvcWinShowHelp(void)
839{
840 RTPrintf("VirtualBox Autostart Service Version %s\n"
841 "(C) 2012 Oracle Corporation\n"
842 "All rights reserved.\n"
843 "\n",
844 VBOX_VERSION_STRING);
845 RTPrintf("Usage:\n"
846 "\n"
847 "VBoxAutostartSvc\n"
848 " Runs the service.\n"
849 "VBoxAutostartSvc <version|-v|--version> [-brief]\n"
850 " Displays the version.\n"
851 "VBoxAutostartSvc <help|-?|-h|--help> [...]\n"
852 " Displays this help screen.\n"
853 "\n"
854 "VBoxAutostartSvc <install|/RegServer|/i>\n"
855 " Installs the service.\n"
856 "VBoxAutostartSvc <uninstall|delete|/UnregServer|/u>\n"
857 " Uninstalls the service.\n"
858 );
859 return RTEXITCODE_SUCCESS;
860}
861
862
863/**
864 * VBoxAutostart main(), Windows edition.
865 *
866 *
867 * @returns 0 on success.
868 *
869 * @param argc Number of arguments in argv.
870 * @param argv Argument vector.
871 */
872int main(int argc, char **argv)
873{
874 /*
875 * Initialize the IPRT first of all.
876 */
877 int rc = RTR3InitExe(argc, &argv, 0);
878 if (RT_FAILURE(rc))
879 {
880 autostartSvcLogError("RTR3InitExe failed with rc=%Rrc", rc);
881 return RTEXITCODE_FAILURE;
882 }
883
884 RTThreadSleep(10 * 1000);
885
886 /*
887 * Parse the initial arguments to determine the desired action.
888 */
889 enum
890 {
891 kAutoSvcAction_RunIt,
892
893 kAutoSvcAction_Create,
894 kAutoSvcAction_Delete,
895
896 kAutoSvcAction_Enable,
897 kAutoSvcAction_Disable,
898 kAutoSvcAction_QueryConfig,
899 kAutoSvcAction_QueryDescription,
900
901 kAutoSvcAction_Start,
902 kAutoSvcAction_Pause,
903 kAutoSvcAction_Continue,
904 kAutoSvcAction_Stop,
905 kAutoSvcAction_Interrogate,
906
907 kAutoSvcAction_End
908 } enmAction = kAutoSvcAction_RunIt;
909 int iArg = 1;
910 if (argc > 1)
911 {
912 if ( !stricmp(argv[iArg], "/RegServer")
913 || !stricmp(argv[iArg], "install")
914 || !stricmp(argv[iArg], "/i"))
915 enmAction = kAutoSvcAction_Create;
916 else if ( !stricmp(argv[iArg], "/UnregServer")
917 || !stricmp(argv[iArg], "/u")
918 || !stricmp(argv[iArg], "uninstall")
919 || !stricmp(argv[iArg], "delete"))
920 enmAction = kAutoSvcAction_Delete;
921
922 else if (!stricmp(argv[iArg], "enable"))
923 enmAction = kAutoSvcAction_Enable;
924 else if (!stricmp(argv[iArg], "disable"))
925 enmAction = kAutoSvcAction_Disable;
926 else if (!stricmp(argv[iArg], "qconfig"))
927 enmAction = kAutoSvcAction_QueryConfig;
928 else if (!stricmp(argv[iArg], "qdescription"))
929 enmAction = kAutoSvcAction_QueryDescription;
930
931 else if ( !stricmp(argv[iArg], "start")
932 || !stricmp(argv[iArg], "/t"))
933 enmAction = kAutoSvcAction_Start;
934 else if (!stricmp(argv[iArg], "pause"))
935 enmAction = kAutoSvcAction_Start;
936 else if (!stricmp(argv[iArg], "continue"))
937 enmAction = kAutoSvcAction_Continue;
938 else if (!stricmp(argv[iArg], "stop"))
939 enmAction = kAutoSvcAction_Stop;
940 else if (!stricmp(argv[iArg], "interrogate"))
941 enmAction = kAutoSvcAction_Interrogate;
942 else if ( !stricmp(argv[iArg], "help")
943 || !stricmp(argv[iArg], "?")
944 || !stricmp(argv[iArg], "/?")
945 || !stricmp(argv[iArg], "-?")
946 || !stricmp(argv[iArg], "/h")
947 || !stricmp(argv[iArg], "-h")
948 || !stricmp(argv[iArg], "/help")
949 || !stricmp(argv[iArg], "-help")
950 || !stricmp(argv[iArg], "--help"))
951 return autostartSvcWinShowHelp();
952 else if ( !stricmp(argv[iArg], "version")
953 || !stricmp(argv[iArg], "/v")
954 || !stricmp(argv[iArg], "-v")
955 || !stricmp(argv[iArg], "/version")
956 || !stricmp(argv[iArg], "-version")
957 || !stricmp(argv[iArg], "--version"))
958 return autostartSvcWinShowVersion(argc - iArg - 1, argv + iArg + 1);
959 else
960 iArg--;
961 iArg++;
962 }
963
964 /*
965 * Dispatch it.
966 */
967 switch (enmAction)
968 {
969 case kAutoSvcAction_RunIt:
970 return autostartSvcWinRunIt(argc - iArg, argv + iArg);
971
972 case kAutoSvcAction_Create:
973 return autostartSvcWinCreate(argc - iArg, argv + iArg);
974 case kAutoSvcAction_Delete:
975 return autostartSvcWinDelete(argc - iArg, argv + iArg);
976
977 case kAutoSvcAction_Enable:
978 return autostartSvcWinEnable(argc - iArg, argv + iArg);
979 case kAutoSvcAction_Disable:
980 return autostartSvcWinDisable(argc - iArg, argv + iArg);
981 case kAutoSvcAction_QueryConfig:
982 return autostartSvcWinQueryConfig(argc - iArg, argv + iArg);
983 case kAutoSvcAction_QueryDescription:
984 return autostartSvcWinQueryDescription(argc - iArg, argv + iArg);
985
986 case kAutoSvcAction_Start:
987 return autostartSvcWinStart(argc - iArg, argv + iArg);
988 case kAutoSvcAction_Pause:
989 return autostartSvcWinPause(argc - iArg, argv + iArg);
990 case kAutoSvcAction_Continue:
991 return autostartSvcWinContinue(argc - iArg, argv + iArg);
992 case kAutoSvcAction_Stop:
993 return autostartSvcWinStop(argc - iArg, argv + iArg);
994 case kAutoSvcAction_Interrogate:
995 return autostartSvcWinInterrogate(argc - iArg, argv + iArg);
996
997 default:
998 AssertMsgFailed(("enmAction=%d\n", enmAction));
999 return RTEXITCODE_FAILURE;
1000 }
1001}
1002
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