VirtualBox

source: vbox/trunk/src/VBox/Additions/x11/VBoxClient/main.cpp@ 29813

Last change on this file since 29813 was 28800, checked in by vboxsync, 15 years ago

Automated rebranding to Oracle copyright/license strings via filemuncher

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id
File size: 8.7 KB
Line 
1/** @file
2 *
3 * VirtualBox Guest Service:
4 * Linux guest.
5 */
6
7/*
8 * Copyright (C) 2006-2010 Oracle Corporation
9 *
10 * This file is part of VirtualBox Open Source Edition (OSE), as
11 * available from http://www.virtualbox.org. This file is free software;
12 * you can redistribute it and/or modify it under the terms of the GNU
13 * General Public License (GPL) as published by the Free Software
14 * Foundation, in version 2 as it comes in the "COPYING" file of the
15 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
16 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
17 */
18
19#include <sys/types.h>
20#include <stdlib.h> /* For exit */
21#include <stdio.h>
22#include <string.h>
23#include <unistd.h>
24#include <errno.h>
25#include <signal.h>
26
27#include <X11/Xlib.h>
28
29#include <iprt/env.h>
30#include <iprt/initterm.h>
31#include <iprt/path.h>
32#include <iprt/param.h>
33#include <iprt/stream.h>
34#include <iprt/string.h>
35#include <VBox/VBoxGuestLib.h>
36#include <VBox/log.h>
37
38#include "VBoxClient.h"
39
40#define TRACE RTPrintf("%s: %d\n", __PRETTY_FUNCTION__, __LINE__); LogRel(("%s: %d\n", __PRETTY_FUNCTION__, __LINE__))
41
42static int (*gpfnOldIOErrorHandler)(Display *) = NULL;
43
44/** Object representing the service we are running. This has to be global
45 * so that the cleanup routine can access it. */
46VBoxClient::Service *g_pService;
47/** The name of our pidfile. It is global for the benefit of the cleanup
48 * routine. */
49static char g_szPidFile[RTPATH_MAX];
50/** The file handle of our pidfile. It is global for the benefit of the
51 * cleanup routine. */
52static RTFILE g_hPidFile;
53
54/** Clean up if we get a signal or something. This is extern so that we
55 * can call it from other compilation units. */
56void VBoxClient::CleanUp()
57{
58 if (g_pService)
59 {
60 g_pService->cleanup();
61 delete g_pService;
62 }
63 if (g_szPidFile[0] && g_hPidFile)
64 VbglR3ClosePidFile(g_szPidFile, g_hPidFile);
65 VbglR3Term();
66 exit(0);
67}
68
69/**
70 * A standard signal handler which cleans up and exits.
71 */
72void vboxClientSignalHandler(int cSignal)
73{
74 LogRel(("VBoxClient: terminated with signal %d\n", cSignal));
75 /** Disable seamless mode */
76 RTPrintf(("VBoxClient: terminating...\n"));
77 VBoxClient::CleanUp();
78}
79
80/**
81 * Xlib error handler for certain errors that we can't avoid.
82 */
83int vboxClientXLibErrorHandler(Display *pDisplay, XErrorEvent *pError)
84{
85 char errorText[1024];
86
87 XGetErrorText(pDisplay, pError->error_code, errorText, sizeof(errorText));
88 LogRelFlow(("VBoxClient: an X Window protocol error occurred: %s (error code %d). Request code: %d, minor code: %d, serial number: %d\n", errorText, pError->error_code, pError->request_code, pError->minor_code, pError->serial));
89 return 0; /* We should never reach this. */
90}
91
92/**
93 * Xlib error handler for fatal errors. This often means that the programme is still running
94 * when X exits.
95 */
96static int vboxClientXLibIOErrorHandler(Display *pDisplay)
97{
98 LogRel(("VBoxClient: a fatal guest X Window error occurred. This may just mean that the Window system was shut down while the client was still running.\n"));
99 VBoxClient::CleanUp();
100 return 0; /* We should never reach this. */
101}
102
103/**
104 * Reset all standard termination signals to call our signal handler, which
105 * cleans up and exits.
106 */
107void vboxClientSetSignalHandlers(void)
108{
109 struct sigaction sigAction;
110
111 LogRelFlowFunc(("\n"));
112 sigAction.sa_handler = vboxClientSignalHandler;
113 sigemptyset(&sigAction.sa_mask);
114 sigAction.sa_flags = 0;
115 sigaction(SIGHUP, &sigAction, NULL);
116 sigaction(SIGINT, &sigAction, NULL);
117 sigaction(SIGQUIT, &sigAction, NULL);
118 sigaction(SIGABRT, &sigAction, NULL);
119 sigaction(SIGPIPE, &sigAction, NULL);
120 sigaction(SIGALRM, &sigAction, NULL);
121 sigaction(SIGTERM, &sigAction, NULL);
122 sigaction(SIGUSR1, &sigAction, NULL);
123 sigaction(SIGUSR2, &sigAction, NULL);
124 LogRelFlowFunc(("returning\n"));
125}
126
127/**
128 * Print out a usage message and exit with success.
129 */
130void vboxClientUsage(const char *pcszFileName)
131{
132 RTPrintf("Usage: %s --clipboard|--display|--checkhostversion|--seamless [-d|--nodaemon]\n", pcszFileName);
133 RTPrintf("Start the VirtualBox X Window System guest services.\n\n");
134 RTPrintf("Options:\n");
135 RTPrintf(" --clipboard start the shared clipboard service\n");
136 RTPrintf(" --display start the display management service\n");
137# ifdef VBOX_WITH_GUEST_PROPS
138 RTPrintf(" --checkhostversion start the host version notifier service\n");
139# endif
140 RTPrintf(" --seamless start the seamless windows service\n");
141 RTPrintf(" -d, --nodaemon continue running as a system service\n");
142 RTPrintf("\n");
143 exit(0);
144}
145
146/**
147 * The main loop for the VBoxClient daemon.
148 */
149int main(int argc, char *argv[])
150{
151 int rcClipboard, rc = VINF_SUCCESS;
152 const char *pszFileName = RTPathFilename(argv[0]);
153 bool fDaemonise = true;
154 /* Have any fatal errors occurred yet? */
155 bool fSuccess = true;
156 /* Do we know which service we wish to run? */
157 bool fHaveService = false;
158
159 if (NULL == pszFileName)
160 pszFileName = "VBoxClient";
161
162 /* Initialise our runtime before all else. */
163 RTR3Init();
164
165 /* Parse our option(s) */
166 /** @todo Use RTGetOpt() if the arguments become more complex. */
167 for (int i = 1; i < argc; ++i)
168 {
169 if (!strcmp(argv[i], "-d") || !strcmp(argv[i], "--nodaemon"))
170 fDaemonise = false;
171 else if (!strcmp(argv[i], "--clipboard"))
172 {
173 if (g_pService == NULL)
174 g_pService = VBoxClient::GetClipboardService();
175 else
176 fSuccess = false;
177 }
178 else if (!strcmp(argv[i], "--display"))
179 {
180 if (g_pService == NULL)
181 g_pService = VBoxClient::GetDisplayService();
182 else
183 fSuccess = false;
184 }
185 else if (!strcmp(argv[i], "--seamless"))
186 {
187 if (g_pService == NULL)
188 g_pService = VBoxClient::GetSeamlessService();
189 else
190 fSuccess = false;
191 }
192 else if (!strcmp(argv[i], "--checkhostversion"))
193 {
194 if (g_pService == NULL)
195 g_pService = VBoxClient::GetHostVersionService();
196 else
197 fSuccess = false;
198 }
199 else if (!strcmp(argv[i], "-h") || !strcmp(argv[i], "--help"))
200 {
201 vboxClientUsage(pszFileName);
202 exit(0);
203 }
204 else
205 {
206 RTPrintf("%s: unrecognized option `%s'\n", pszFileName, argv[i]);
207 RTPrintf("Try `%s --help' for more information\n", pszFileName);
208 exit(1);
209 }
210 }
211 if (!fSuccess || !g_pService)
212 {
213 vboxClientUsage(pszFileName);
214 exit(1);
215 }
216 if (fDaemonise)
217 {
218 rc = VbglR3Daemonize(false /* fNoChDir */, false /* fNoClose */);
219 if (RT_FAILURE(rc))
220 {
221 RTPrintf("VBoxClient: failed to daemonize. Exiting.\n");
222 LogRel(("VBoxClient: failed to daemonize. Exiting.\n"));
223# ifdef DEBUG
224 RTPrintf("Error %Rrc\n", rc);
225# endif
226 return 1;
227 }
228 }
229 /** @todo explain why we aren't using RTPathUserHome here so it doesn't get
230 * changed accidentally during some cleanup effort. */
231 rc = RTEnvGetEx(RTENV_DEFAULT, "HOME", g_szPidFile, sizeof(g_szPidFile), NULL);
232 if (RT_FAILURE(rc))
233 {
234 RTPrintf("VBoxClient: failed to get home directory, rc=%Rrc. Exiting.\n", rc);
235 LogRel(("VBoxClient: failed to get home directory, rc=%Rrc. Exiting.\n", rc));
236 return 1;
237 }
238 rc = RTPathAppend(g_szPidFile, sizeof(g_szPidFile), g_pService->getPidFilePath());
239 if (RT_FAILURE(rc))
240 {
241 RTPrintf("VBoxClient: RTPathAppend failed with rc=%Rrc. Exiting.\n", rc);
242 LogRel(("VBoxClient: RTPathAppend failed with rc=%Rrc. Exiting.\n", rc));
243 return 1;
244 }
245
246 /* Initialise the guest library. */
247 if (RT_FAILURE(VbglR3InitUser()))
248 {
249 RTPrintf("Failed to connect to the VirtualBox kernel service\n");
250 LogRel(("Failed to connect to the VirtualBox kernel service\n"));
251 return 1;
252 }
253 if (g_szPidFile[0] && RT_FAILURE(VbglR3PidFile(g_szPidFile, &g_hPidFile)))
254 {
255 RTPrintf("Failed to create a pidfile. Exiting.\n");
256 LogRel(("Failed to create a pidfile. Exiting.\n"));
257 VbglR3Term();
258 return 1;
259 }
260 /* Set signal handlers to clean up on exit. */
261 vboxClientSetSignalHandlers();
262 /* Set an X11 error handler, so that we don't die when we get unavoidable errors. */
263 XSetErrorHandler(vboxClientXLibErrorHandler);
264 /* Set an X11 I/O error handler, so that we can shutdown properly on fatal errors. */
265 XSetIOErrorHandler(vboxClientXLibIOErrorHandler);
266 g_pService->run(fDaemonise);
267 VBoxClient::CleanUp();
268 return 1; /* We should never get here. */
269}
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