VirtualBox

source: vbox/trunk/src/VBox/Additions/x11/xclient/main.cpp@ 10301

Last change on this file since 10301 was 9791, checked in by vboxsync, 17 years ago

Additions/x11: remove some of the libstdc++ dependencies in VBoxClient (std::cout -> printf)

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id
File size: 10.1 KB
Line 
1/** @file
2 *
3 * VirtualBox Guest Service:
4 * Linux guest.
5 */
6
7/*
8 * Copyright (C) 2006-2007 Sun Microsystems, Inc.
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 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa
19 * Clara, CA 95054 USA or visit http://www.sun.com if you need
20 * additional information or have any questions.
21 */
22
23#include <VBox/VBoxGuest.h>
24#include <VBox/log.h>
25#include <iprt/initterm.h>
26#include <iprt/path.h>
27
28#include <sys/types.h>
29#include <stdlib.h> /* For exit */
30#include <stdio.h>
31#include <unistd.h>
32#include <errno.h>
33#include <signal.h>
34
35#include <X11/Xlib.h>
36#include <X11/Intrinsic.h>
37
38#include "clipboard.h"
39
40#ifdef DYNAMIC_RESIZE
41# include "displaychange.h"
42# ifdef SEAMLESS_GUEST
43# include "seamless.h"
44# endif
45#endif
46
47#define TRACE printf("%s: %d\n", __PRETTY_FUNCTION__, __LINE__); Log(("%s: %d\n", __PRETTY_FUNCTION__, __LINE__))
48
49static int (*gpfnOldIOErrorHandler)(Display *) = NULL;
50
51/* Make these global so that the destructors are called if we make an "emergency exit",
52 i.e. a (handled) signal or an X11 error. */
53#ifdef DYNAMIC_RESIZE
54VBoxGuestDisplayChangeMonitor gDisplayChange;
55# ifdef SEAMLESS_GUEST
56 /** Our instance of the seamless class. This only makes sense if dynamic resizing
57 is enabled. */
58 VBoxGuestSeamless gSeamless;
59# endif /* SEAMLESS_GUEST defined */
60#endif /* DYNAMIC_RESIZE */
61#ifdef VBOX_X11_CLIPBOARD
62 VBoxGuestClipboard gClipboard;
63#endif
64
65/**
66 * Drop the programmes privileges to the caller's.
67 * @returns IPRT status code
68 * @todo move this into the R3 guest library
69 */
70int vboxClientDropPrivileges(void)
71{
72 int rc = VINF_SUCCESS;
73 int rcSystem, rcErrno;
74
75 LogFlowFunc(("\n"));
76#ifdef _POSIX_SAVED_IDS
77 rcSystem = setuid(getuid());
78#else
79 rcSystem = setreuid(-1, getuid());
80#endif
81 if (rcSystem < 0)
82 {
83 rcErrno = errno;
84 rc = RTErrConvertFromErrno(rcErrno);
85 LogRel(("VBoxClient: failed to drop privileges, error %Rrc.\n", rc));
86 }
87 LogFlowFunc(("returning %Rrc\n", rc));
88 return rc;
89}
90
91/**
92 * Xlib error handler for certain errors that we can't avoid.
93 */
94int vboxClientXLibErrorHandler(Display *pDisplay, XErrorEvent *pError)
95{
96 char errorText[1024];
97
98 if (pError->error_code == BadAtom)
99 {
100 /* This can be triggered in debug builds if a guest application passes a bad atom
101 in its list of supported clipboard formats. As such it is harmless. */
102 Log(("VBoxClient: ignoring BadAtom error and returning\n"));
103 return 0;
104 }
105 if (pError->error_code == BadWindow)
106 {
107 /* This can be triggered if a guest application destroys a window before we notice. */
108 Log(("VBoxClient: ignoring BadWindow error and returning\n"));
109 return 0;
110 }
111 XGetErrorText(pDisplay, pError->error_code, errorText, sizeof(errorText));
112 LogRel(("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));
113 /** Disable seamless mode */
114 VbglR3SeamlessSetCap(false);
115 VbglR3Term();
116 exit(1);
117}
118
119/**
120 * Xlib error handler for fatal errors. This often means that the programme is still running
121 * when X exits.
122 */
123int vboxClientXLibIOErrorHandler(Display *pDisplay)
124{
125 Log(("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"));
126 /** Disable seamless mode */
127 VbglR3SeamlessSetCap(false);
128 VbglR3Term();
129 return gpfnOldIOErrorHandler(pDisplay);
130}
131
132/**
133 * A standard signal handler which cleans up and exits. Our global static objects will
134 * be cleaned up properly as we exit using "exit".
135 */
136void vboxClientSignalHandler(int cSignal)
137{
138 Log(("VBoxClient: terminated with signal %d\n", cSignal));
139 /** Disable seamless mode */
140 VbglR3SeamlessSetCap(false);
141 printf(("VBoxClient: terminating...\n"));
142 /* don't call VbglR3Term() here otherwise the /dev/vboxadd filehandle is closed */
143 /* Our pause() call will now return and exit. */
144}
145
146/**
147 * Reset all standard termination signals to call our signal handler, which cleans up
148 * and exits.
149 */
150void vboxClientSetSignalHandlers(void)
151{
152 struct sigaction sigAction;
153
154 LogFlowFunc(("\n"));
155 sigAction.sa_handler = vboxClientSignalHandler;
156 sigemptyset(&sigAction.sa_mask);
157 sigAction.sa_flags = 0;
158 sigaction(SIGHUP, &sigAction, NULL);
159 sigaction(SIGINT, &sigAction, NULL);
160 sigaction(SIGQUIT, &sigAction, NULL);
161 sigaction(SIGABRT, &sigAction, NULL);
162 sigaction(SIGPIPE, &sigAction, NULL);
163 sigaction(SIGALRM, &sigAction, NULL);
164 sigaction(SIGTERM, &sigAction, NULL);
165 sigaction(SIGUSR1, &sigAction, NULL);
166 sigaction(SIGUSR2, &sigAction, NULL);
167 LogFlowFunc(("returning\n"));
168}
169
170/**
171 * Print out a usage message and exit with success.
172 */
173void vboxClientUsage(const char *pcszFileName)
174{
175 printf("Usage: %s [-d|--nodaemon]\n", pcszFileName);
176 printf("Start the VirtualBox X Window System guest services.\n\n");
177 printf("Options:\n");
178 printf(" -d, --nodaemon do not lower privileges and continue running as a system\n");
179 printf(" service\n");
180 exit(0);
181}
182
183/**
184 * The main loop for the VBoxClient daemon.
185 */
186int main(int argc, char *argv[])
187{
188 int rcClipboard, rc = VINF_SUCCESS;
189 const char *pszFileName = RTPathFilename(argv[0]);
190 bool fDaemonise = true;
191
192 if (NULL == pszFileName)
193 pszFileName = "VBoxClient";
194
195 /* Parse our option(s) */
196 /** @todo Use RTGetOpt() if the arguments become more complex. */
197 for (int i = 1; i < argc; ++i)
198 {
199 if (!strcmp(argv[i], "-d") || !strcmp(argv[i], "--nodaemon"))
200 fDaemonise = false;
201 else if (!strcmp(argv[i], "-h") || !strcmp(argv[i], "--help"))
202 {
203 vboxClientUsage(pszFileName);
204 exit(0);
205 }
206 else
207 {
208 printf("%s: unrecognized option `%s'\n", pszFileName, argv[i]);
209 printf("Try `%s --help' for more information\n", pszFileName);
210 exit(1);
211 }
212 }
213 if (fDaemonise)
214 {
215 rc = VbglR3Daemonize(false /* fNoChDir */, false /* fNoClose */);
216 if (RT_FAILURE(rc))
217 {
218 printf("VBoxClient: failed to daemonize. exiting.");
219 return 1;
220 }
221 }
222 /* Initialise our runtime before all else. */
223 RTR3Init(false);
224 if (RT_FAILURE(VbglR3Init()))
225 {
226 printf("Failed to connect to the VirtualBox kernel service");
227 return 1;
228 }
229 if (fDaemonise && RT_FAILURE(vboxClientDropPrivileges()))
230 return 1;
231 LogRel(("VBoxClient: starting...\n"));
232 /* Initialise threading in X11 and in Xt. */
233 if (!XInitThreads() || !XtToolkitThreadInitialize())
234 {
235 LogRel(("VBoxClient: error initialising threads in X11, exiting."));
236 return 1;
237 }
238 /* Set an X11 error handler, so that we don't die when we get unavoidable errors. */
239 XSetErrorHandler(vboxClientXLibErrorHandler);
240 /* Set an X11 I/O error handler, so that we can shutdown properly on fatal errors. */
241 gpfnOldIOErrorHandler = XSetIOErrorHandler(vboxClientXLibIOErrorHandler);
242 vboxClientSetSignalHandlers();
243 try
244 {
245#ifdef VBOX_X11_CLIPBOARD
246 /* Connect to the host clipboard. */
247 LogRel(("VBoxClient: starting clipboard Guest Additions...\n"));
248 rcClipboard = gClipboard.init();
249 if (RT_FAILURE(rcClipboard))
250 {
251 LogRel(("VBoxClient: vboxClipboardConnect failed with rc = %Rrc\n", rc));
252 }
253#endif /* VBOX_X11_CLIPBOARD defined */
254#ifdef DYNAMIC_RESIZE
255 LogRel(("VBoxClient: starting dynamic guest resizing...\n"));
256 rc = gDisplayChange.init();
257 if (RT_FAILURE(rc))
258 {
259 LogRel(("VBoxClient: failed to start dynamic guest resizing, rc = %Rrc\n", rc));
260 }
261# ifdef SEAMLESS_GUEST
262 if (RT_SUCCESS(rc))
263 {
264 LogRel(("VBoxClient: starting seamless Guest Additions...\n"));
265 rc = gSeamless.init();
266 if (RT_FAILURE(rc))
267 {
268 LogRel(("VBoxClient: failed to start seamless Additions, rc = %Rrc\n", rc));
269 }
270 }
271# endif /* SEAMLESS_GUEST defined */
272#endif /* DYNAMIC_RESIZE defined */
273 }
274 catch (std::exception e)
275 {
276 LogRel(("VBoxClient: failed to initialise Guest Additions - caught exception: %s\n", e.what()));
277 rc = VERR_UNRESOLVED_ERROR;
278 }
279 catch (...)
280 {
281 LogRel(("VBoxClient: failed to initialise Guest Additions - caught unknown exception.\n"));
282 rc = VERR_UNRESOLVED_ERROR;
283 }
284 LogRel(("VBoxClient: sleeping...\n"));
285 pause();
286 LogRel(("VBoxClient: exiting...\n"));
287 try
288 {
289 /* r=frank: Why all these 2s delays? What are we waiting for? */
290#ifdef DYNAMIC_RESIZE
291# ifdef SEAMLESS_GUEST
292 LogRel(("VBoxClient: shutting down seamless Guest Additions...\n"));
293 gSeamless.uninit(2000);
294# endif /* SEAMLESS_GUEST defined */
295 LogRel(("VBoxClient: shutting down dynamic guest resizing...\n"));
296 gDisplayChange.uninit(2000);
297#endif /* DYNAMIC_RESIZE defined */
298#ifdef VBOX_X11_CLIPBOARD
299 /* Connect to the host clipboard. */
300 LogRel(("VBoxClient: shutting down clipboard Guest Additions...\n"));
301 gClipboard.uninit(2000);
302#endif /* VBOX_X11_CLIPBOARD defined */
303 }
304 catch (std::exception e)
305 {
306 LogRel(("VBoxClient: failed to shut down Guest Additions - caught exception: %s\n", e.what()));
307 rc = VERR_UNRESOLVED_ERROR;
308 }
309 catch (...)
310 {
311 LogRel(("VBoxClient: failed to shut down Guest Additions - caught unknown exception.\n"));
312 rc = VERR_UNRESOLVED_ERROR;
313 }
314 VbglR3Term();
315 return RT_SUCCESS(rc) ? 0 : 1;
316}
Note: See TracBrowser for help on using the repository browser.

© 2025 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette