VirtualBox

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

Last change on this file since 37608 was 37375, checked in by vboxsync, 14 years ago

GuestCtrl: Added APIs for guest directory enumeration, guest file existence and copy from guest support, some API renaming/cleanup (work in progress).

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 8.7 KB
Line 
1/* $Id: VBoxServiceControl.cpp 37375 2011-06-08 10:51:26Z vboxsync $ */
2/** @file
3 * VBoxServiceControl - Host-driven Guest Control.
4 */
5
6/*
7 * Copyright (C) 2011 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 <iprt/assert.h>
23#include <iprt/getopt.h>
24#include <iprt/mem.h>
25#include <iprt/semaphore.h>
26#include <iprt/thread.h>
27#include <VBox/VBoxGuestLib.h>
28#include <VBox/HostServices/GuestControlSvc.h>
29#include "VBoxServiceInternal.h"
30#include "VBoxServiceUtils.h"
31#include "VBoxServiceControlExecThread.h"
32
33using namespace guestControl;
34
35/*******************************************************************************
36* Global Variables *
37*******************************************************************************/
38/** The control interval (milliseconds). */
39uint32_t g_ControlInterval = 0;
40/** The semaphore we're blocking on. */
41static RTSEMEVENTMULTI g_hControlEvent = NIL_RTSEMEVENTMULTI;
42/** The Guest Control service client ID. */
43static uint32_t g_GuestControlSvcClientID = 0;
44/** List of spawned processes. */
45RTLISTNODE g_GuestControlExecThreads;
46/** Critical section protecting g_GuestControlExecThreads. */
47RTCRITSECT g_GuestControlExecThreadsCritSect;
48
49/** @copydoc VBOXSERVICE::pfnPreInit */
50static DECLCALLBACK(int) VBoxServiceControlPreInit(void)
51{
52 return VINF_SUCCESS;
53}
54
55
56/** @copydoc VBOXSERVICE::pfnOption */
57static DECLCALLBACK(int) VBoxServiceControlOption(const char **ppszShort, int argc, char **argv, int *pi)
58{
59 int rc = -1;
60 if (ppszShort)
61 /* no short options */;
62 else if (!strcmp(argv[*pi], "--control-interval"))
63 rc = VBoxServiceArgUInt32(argc, argv, "", pi,
64 &g_ControlInterval, 1, UINT32_MAX - 1);
65 return rc;
66}
67
68
69/** @copydoc VBOXSERVICE::pfnInit */
70static DECLCALLBACK(int) VBoxServiceControlInit(void)
71{
72 /*
73 * If not specified, find the right interval default.
74 * Then create the event sem to block on.
75 */
76 if (!g_ControlInterval)
77 g_ControlInterval = 1000;
78
79 int rc = RTSemEventMultiCreate(&g_hControlEvent);
80 AssertRCReturn(rc, rc);
81
82 rc = VbglR3GuestCtrlConnect(&g_GuestControlSvcClientID);
83 if (RT_SUCCESS(rc))
84 {
85 VBoxServiceVerbose(3, "Control: Service Client ID: %#x\n", g_GuestControlSvcClientID);
86
87 /* Init thread list. */
88 RTListInit(&g_GuestControlExecThreads);
89 rc = RTCritSectInit(&g_GuestControlExecThreadsCritSect);
90 AssertRC(rc);
91 }
92 else
93 {
94 /* If the service was not found, we disable this service without
95 causing VBoxService to fail. */
96 if (rc == VERR_HGCM_SERVICE_NOT_FOUND) /* Host service is not available. */
97 {
98 VBoxServiceVerbose(0, "Control: Guest control service is not available\n");
99 rc = VERR_SERVICE_DISABLED;
100 }
101 else
102 VBoxServiceError("Control: Failed to connect to the guest control service! Error: %Rrc\n", rc);
103 RTSemEventMultiDestroy(g_hControlEvent);
104 g_hControlEvent = NIL_RTSEMEVENTMULTI;
105 }
106 return rc;
107}
108
109
110/** @copydoc VBOXSERVICE::pfnWorker */
111DECLCALLBACK(int) VBoxServiceControlWorker(bool volatile *pfShutdown)
112{
113 /*
114 * Tell the control thread that it can continue
115 * spawning services.
116 */
117 RTThreadUserSignal(RTThreadSelf());
118 Assert(g_GuestControlSvcClientID > 0);
119
120 int rc = VINF_SUCCESS;
121
122 /*
123 * Execution loop.
124 *
125 * @todo
126 */
127 for (;;)
128 {
129 uint32_t uMsg;
130 uint32_t uNumParms;
131 VBoxServiceVerbose(3, "Control: Waiting for host msg ...\n");
132 rc = VbglR3GuestCtrlWaitForHostMsg(g_GuestControlSvcClientID, &uMsg, &uNumParms);
133 if (RT_FAILURE(rc))
134 {
135 if (rc == VERR_TOO_MUCH_DATA)
136 {
137 VBoxServiceVerbose(4, "Control: Message requires %ld parameters, but only 2 supplied -- retrying request (no error!)...\n", uNumParms);
138 rc = VINF_SUCCESS; /* Try to get "real" message in next block below. */
139 }
140 else
141 VBoxServiceVerbose(3, "Control: Getting host message failed with %Rrc\n", rc); /* VERR_GEN_IO_FAILURE seems to be normal if ran into timeout. */
142 }
143
144 if (RT_SUCCESS(rc))
145 {
146 VBoxServiceVerbose(3, "Control: Msg=%u (%u parms) retrieved\n", uMsg, uNumParms);
147 switch(uMsg)
148 {
149 case HOST_CANCEL_PENDING_WAITS:
150 VBoxServiceVerbose(3, "Control: Host asked us to quit ...\n");
151 break;
152
153 case HOST_EXEC_CMD:
154 rc = VBoxServiceControlExecHandleCmdStartProcess(g_GuestControlSvcClientID, uNumParms);
155 break;
156
157 case HOST_EXEC_SET_INPUT:
158 /** @todo Make buffer size configurable via guest properties/argv! */
159 rc = VBoxServiceControlExecHandleCmdSetInput(g_GuestControlSvcClientID, uNumParms, _1M /* Buffer size */);
160 break;
161
162 case HOST_EXEC_GET_OUTPUT:
163 rc = VBoxServiceControlExecHandleCmdGetOutput(g_GuestControlSvcClientID, uNumParms);
164 break;
165
166 case HOST_DIR_CLOSE:
167 rc = VBoxServiceGCtrlDirClose(g_GuestControlSvcClientID, uNumParms);
168 break;
169
170 case HOST_DIR_OPEN:
171 rc = VBoxServiceGCtrlDirOpen(g_GuestControlSvcClientID, uNumParms);
172 break;
173
174 case HOST_DIR_READ:
175 rc = VBoxServiceGCtrlDirRead(g_GuestControlSvcClientID, uNumParms);
176 break;
177
178 default:
179 VBoxServiceVerbose(3, "Control: Unsupported message from host! Msg=%u\n", uMsg);
180 /* Don't terminate here; just wait for the next message. */
181 break;
182 }
183
184 if (RT_FAILURE(rc))
185 VBoxServiceVerbose(3, "Control: Message was processed with rc=%Rrc\n", rc);
186 }
187
188 /* Do we need to shutdown? */
189 if ( *pfShutdown
190 || uMsg == HOST_CANCEL_PENDING_WAITS)
191 {
192 rc = VINF_SUCCESS;
193 break;
194 }
195
196 /* Let's sleep for a bit and let others run ... */
197 RTThreadYield();
198 }
199
200 RTSemEventMultiDestroy(g_hControlEvent);
201 g_hControlEvent = NIL_RTSEMEVENTMULTI;
202 return rc;
203}
204
205
206/** @copydoc VBOXSERVICE::pfnStop */
207static DECLCALLBACK(void) VBoxServiceControlStop(void)
208{
209 VBoxServiceVerbose(3, "Control: Stopping ...\n");
210
211 /** @todo Later, figure what to do if we're in RTProcWait(). it's a very
212 * annoying call since doesn't support timeouts in the posix world. */
213 RTSemEventMultiSignal(g_hControlEvent);
214
215 /*
216 * Ask the host service to cancel all pending requests so that we can
217 * shutdown properly here.
218 */
219 if (g_GuestControlSvcClientID)
220 {
221 int rc = VbglR3GuestCtrlCancelPendingWaits(g_GuestControlSvcClientID);
222 if (RT_FAILURE(rc))
223 VBoxServiceError("Control: Cancelling pending waits failed; rc=%Rrc\n", rc);
224 }
225}
226
227
228/** @copydoc VBOXSERVICE::pfnTerm */
229static DECLCALLBACK(void) VBoxServiceControlTerm(void)
230{
231 VBoxServiceVerbose(3, "Control: Terminating ...\n");
232
233 VBoxServiceControlExecThreadsShutdown();
234
235 VbglR3GuestCtrlDisconnect(g_GuestControlSvcClientID);
236 g_GuestControlSvcClientID = 0;
237
238 if (g_hControlEvent != NIL_RTSEMEVENTMULTI)
239 {
240 RTSemEventMultiDestroy(g_hControlEvent);
241 g_hControlEvent = NIL_RTSEMEVENTMULTI;
242 }
243}
244
245
246/**
247 * The 'vminfo' service description.
248 */
249VBOXSERVICE g_Control =
250{
251 /* pszName. */
252 "control",
253 /* pszDescription. */
254 "Host-driven Guest Control",
255 /* pszUsage. */
256 " [--control-interval <ms>]"
257 ,
258 /* pszOptions. */
259 " --control-interval Specifies the interval at which to check for\n"
260 " new control commands. The default is 1000 ms.\n"
261 ,
262 /* methods */
263 VBoxServiceControlPreInit,
264 VBoxServiceControlOption,
265 VBoxServiceControlInit,
266 VBoxServiceControlWorker,
267 VBoxServiceControlStop,
268 VBoxServiceControlTerm
269};
270
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