VirtualBox

source: vbox/trunk/src/VBox/Additions/darwin/VBoxClient/VBoxClient.cpp@ 76364

Last change on this file since 76364 was 69500, checked in by vboxsync, 7 years ago

*: scm --update-copyright-year

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 9.1 KB
Line 
1/** $Id: VBoxClient.cpp 69500 2017-10-28 15:14:05Z vboxsync $ */
2/** @file
3 * VBoxClient - User specific services, Darwin.
4 */
5
6/*
7 * Copyright (C) 2007-2017 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 <stdio.h>
23#include <signal.h>
24#include <unistd.h>
25#include <stdlib.h>
26
27#include <VBox/log.h>
28#include <VBox/VBoxGuestLib.h>
29#include <iprt/stream.h>
30#include <iprt/initterm.h>
31#include <iprt/message.h>
32#include <iprt/process.h>
33#include <iprt/string.h>
34
35#include "VBoxClientInternal.h"
36
37
38/*********************************************************************************************************************************
39* Glogal Variables *
40*********************************************************************************************************************************/
41
42static int g_cVerbosity = 0;
43static PRTLOGGER g_pLogger = NULL;
44
45static VBOXCLIENTSERVICE g_aServices[] =
46{
47 g_ClipboardService
48};
49
50
51/**
52 * Create default logger in order to print output to the specified file.
53 *
54 * @return IPRT status code.
55 */
56static int vbclInitLogger(char *szLogFileName)
57{
58 int rc;
59
60 uint32_t fFlags = RTLOGFLAGS_PREFIX_THREAD | RTLOGFLAGS_PREFIX_TIME_PROG;
61 static const char * const s_apszGroups[] = VBOX_LOGGROUP_NAMES;
62 uint32_t fDestFlags = RTLOGDEST_STDOUT;
63
64 rc = RTLogCreateEx(&g_pLogger,
65 fFlags, /* Logger instance flags, a combination of the RTLOGFLAGS_* values */
66 "all", /* The initial group settings */
67 "VBOXCLIENT_RELEASE_LOG", /* Base name for the environment variables for this instance */
68 RT_ELEMENTS(s_apszGroups), /* Number of groups in the array */
69 s_apszGroups, /* Pointer to array of groups. This must stick around for the life of the logger instance */
70 fDestFlags, /* The destination flags */
71 NULL, /* Callback function for starting logging and for ending or starting a new file for log history rotation */
72 szLogFileName ? 10 : 0, /* Number of old log files to keep when performing log history rotation */
73 szLogFileName ? 100 * _1M : 0, /* Maximum size of log file when performing history rotation */
74 szLogFileName ? RT_SEC_1DAY : 0, /* Maximum time interval per log file when performing history rotation */
75 0, /* A buffer which is filled with an error message if something fails */
76 0, /* The size of the error message buffer */
77 szLogFileName /* Log filename format string */
78 );
79
80 AssertReturn(RT_SUCCESS(rc), rc);
81
82 /* Register this logger as the release logger */
83 RTLogRelSetDefaultInstance(g_pLogger);
84
85 /* Explicitly flush the log in case of VBOXCLIENT_RELEASE_LOG=buffered. */
86 RTLogFlush(g_pLogger);
87
88 return rc;
89}
90
91
92/**
93 * Destroy logger.
94 */
95static void vbclTermLogger(char *szLogFileName)
96{
97 // Why SIGBUS here?
98 RTLogDestroy(RTLogRelSetDefaultInstance(NULL));
99
100 if (szLogFileName)
101 RTStrFree(szLogFileName);
102}
103
104/**
105 * Displays a verbose message.
106 *
107 * @param iLevel Minimum log level required to display this message.
108 * @param pszFormat The message text.
109 * @param ... Format arguments.
110 */
111void VBoxClientVerbose(int iLevel, const char *pszFormat, ...)
112{
113 if (iLevel > g_cVerbosity)
114 return;
115
116 va_list args;
117 va_start(args, pszFormat);
118 char *psz = NULL;
119 RTStrAPrintfV(&psz, pszFormat, args);
120 va_end(args);
121
122 AssertPtr(psz);
123 LogRel(("%s", psz));
124
125 RTStrFree(psz);
126}
127
128/**
129 * Wait for signals in order to safely terminate process.
130 */
131static void vbclWait(void)
132{
133 sigset_t signalMask;
134 int iSignal;
135
136 /* Register signals that we are waiting for */
137 sigemptyset(&signalMask);
138 sigaddset(&signalMask, SIGHUP);
139 sigaddset(&signalMask, SIGINT);
140 sigaddset(&signalMask, SIGQUIT);
141 sigaddset(&signalMask, SIGABRT);
142 sigaddset(&signalMask, SIGTERM);
143 pthread_sigmask(SIG_BLOCK, &signalMask, NULL);
144
145 /* Ignoring return status */
146 sigwait(&signalMask, &iSignal);
147}
148
149/**
150 * Start registered services.
151 *
152 * @return IPRT status code.
153 */
154static int vbclStartServices(void)
155{
156 int rc;
157 unsigned int iServiceId = 0;
158
159 VBoxClientVerbose(1, "Starting services...\n");
160 for (iServiceId = 0; iServiceId < RT_ELEMENTS(g_aServices); iServiceId++)
161 {
162 VBoxClientVerbose(1, "Starting service: %s\n", g_aServices[iServiceId].pszName);
163 rc = (g_aServices[iServiceId].pfnStart)();
164 if (RT_FAILURE(rc))
165 {
166 VBoxClientVerbose(1, "unable to start service: %s (%Rrc)\n", g_aServices[iServiceId].pszName, rc);
167 VBoxClientVerbose(1, "Rolling back..\n");
168
169 /* Stop running services */
170 do
171 {
172 VBoxClientVerbose(1, "Stopping service: %s\n", g_aServices[iServiceId].pszName);
173 int rcStop = (g_aServices[iServiceId].pfnStop)();
174 if (RT_FAILURE(rcStop))
175 VBoxClientVerbose(1, "unable to stop service: %s (%Rrc)\n", g_aServices[iServiceId].pszName, rcStop);
176 } while (--iServiceId != 0);
177
178 break;
179 }
180 }
181
182 if (RT_SUCCESS(rc))
183 VBoxClientVerbose(1, "Services start completed.\n");
184
185 return rc;
186}
187
188/**
189 * Stop registered services.
190 *
191 * @return IPRT status code.
192 */
193static void vbclStopServices(void)
194{
195 unsigned int iServiceId = 0;
196
197 VBoxClientVerbose(1, "Stopping services...\n");
198 for (iServiceId = 0; iServiceId < RT_ELEMENTS(g_aServices); iServiceId++)
199 {
200 VBoxClientVerbose(1, "Stopping service: %s\n", g_aServices[iServiceId].pszName);
201 int rc = (g_aServices[iServiceId].pfnStop)();
202 if (RT_FAILURE(rc))
203 VBoxClientVerbose(1, "unable to stop service: %s (%Rrc)\n", g_aServices[iServiceId].pszName, rc);
204 }
205 VBoxClientVerbose(1, "Services stop completed\n");
206}
207
208
209static void usage(char *sProgName)
210{
211 RTPrintf("usage: %s [-fvl]\n", sProgName);
212 RTPrintf(" -f\tRun in foreground (default: no)\n", sProgName);
213 RTPrintf(" -v\tIncrease verbosity level (default: no verbosity)\n", sProgName);
214 RTPrintf(" -l\tSpecify log file name (default: no log file)\n", sProgName);
215 exit(1);
216}
217
218int main(int argc, char *argv[])
219{
220 int rc;
221 int c;
222
223 bool fDemonize = true;
224 static char *szLogFileName = NULL;
225
226 /* Parse command line */
227 while((c = getopt(argc, argv, "fvl:")) != -1)
228 {
229 switch(c)
230 {
231 case 'f':
232 fDemonize = false;
233 break;
234 case 'v':
235 g_cVerbosity++;
236 break;
237 case 'l':
238 szLogFileName = RTStrDup(optarg);
239 break;
240
241 default : usage(argv[0]);
242 }
243 }
244
245 /* No more arguments allowed */
246 if ((argc - optind) != 0)
247 usage(argv[0]);
248
249 if (fDemonize)
250 {
251 rc = RTProcDaemonizeUsingFork(true /* fNoChDir */, false /* fNoClose */, NULL);
252 if (RT_FAILURE(rc))
253 {
254 RTPrintf("failed to run into background\n");
255 return 1;
256 }
257 }
258
259 rc = RTR3InitExe(argc, &argv, 0);
260 if (RT_FAILURE(rc))
261 {
262 RTPrintf("RTR3InitExe() failed: (%Rrc)\n", rc);
263 return RTMsgInitFailure(rc);
264 }
265
266 rc = VbglR3Init();
267 if (RT_SUCCESS(rc))
268 {
269 rc = vbclInitLogger(szLogFileName);
270 if (RT_SUCCESS(rc))
271 {
272 rc = vbclStartServices();
273 if (RT_SUCCESS(rc))
274 {
275 vbclWait();
276 vbclStopServices();
277 }
278 else
279 {
280 RTPrintf("failed to start services: (%Rrc)\n", rc);
281 }
282
283 vbclTermLogger(szLogFileName);
284 }
285 else
286 {
287 RTPrintf("failed to start logger: (%Rrc)\n", rc);
288 }
289
290 VbglR3Term();
291 }
292 else
293 {
294 RTPrintf("failed to initialize guest library: (%Rrc)\n", rc);
295 }
296
297 return 0;
298}
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