VirtualBox

source: vbox/trunk/src/VBox/Additions/common/VBoxGuestLib/VBoxGuestR3LibGuestCtrl.cpp@ 40464

Last change on this file since 40464 was 39427, checked in by vboxsync, 13 years ago

r=bird: pre-review comments.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 15.4 KB
Line 
1/* $Id: VBoxGuestR3LibGuestCtrl.cpp 39427 2011-11-25 14:29:43Z vboxsync $ */
2/** @file
3 * VBoxGuestR3Lib - Ring-3 Support Library for VirtualBox guest additions, guest control.
4 */
5
6/*
7 * Copyright (C) 2010-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 * The contents of this file may alternatively be used under the terms
18 * of the Common Development and Distribution License Version 1.0
19 * (CDDL) only, as it comes in the "COPYING.CDDL" file of the
20 * VirtualBox OSE distribution, in which case the provisions of the
21 * CDDL are applicable instead of those of the GPL.
22 *
23 * You may elect to license modified versions of this file under the
24 * terms and conditions of either the GPL or the CDDL or both.
25 */
26
27
28/*******************************************************************************
29* Header Files *
30*******************************************************************************/
31#include <iprt/string.h>
32#include <iprt/mem.h>
33#include <iprt/assert.h>
34#include <iprt/cpp/autores.h>
35#include <iprt/stdarg.h>
36#include <VBox/log.h>
37#include <VBox/HostServices/GuestControlSvc.h>
38
39#include "VBGLR3Internal.h"
40
41
42/*******************************************************************************
43* Structures and Typedefs *
44*******************************************************************************/
45
46using namespace guestControl;
47
48/**
49 * Connects to the guest control service.
50 *
51 * @returns VBox status code
52 * @param pu32ClientId Where to put the client id on success. The client id
53 * must be passed to all the other calls to the service.
54 */
55VBGLR3DECL(int) VbglR3GuestCtrlConnect(uint32_t *pu32ClientId)
56{
57 VBoxGuestHGCMConnectInfo Info;
58 Info.result = VERR_WRONG_ORDER;
59 Info.Loc.type = VMMDevHGCMLoc_LocalHost_Existing;
60 RT_ZERO(Info.Loc.u);
61 strcpy(Info.Loc.u.host.achName, "VBoxGuestControlSvc");
62 Info.u32ClientID = UINT32_MAX; /* try make valgrind shut up. */
63
64 int rc = vbglR3DoIOCtl(VBOXGUEST_IOCTL_HGCM_CONNECT, &Info, sizeof(Info));
65 if (RT_SUCCESS(rc))
66 {
67 rc = Info.result;
68 if (RT_SUCCESS(rc))
69 *pu32ClientId = Info.u32ClientID;
70 }
71 return rc;
72}
73
74
75/**
76 * Disconnect from the guest control service.
77 *
78 * @returns VBox status code.
79 * @param u32ClientId The client id returned by VbglR3GuestCtrlConnect().
80 */
81VBGLR3DECL(int) VbglR3GuestCtrlDisconnect(uint32_t u32ClientId)
82{
83 VBoxGuestHGCMDisconnectInfo Info;
84 Info.result = VERR_WRONG_ORDER;
85 Info.u32ClientID = u32ClientId;
86
87 int rc = vbglR3DoIOCtl(VBOXGUEST_IOCTL_HGCM_DISCONNECT, &Info, sizeof(Info));
88 if (RT_SUCCESS(rc))
89 rc = Info.result;
90 return rc;
91}
92
93
94/**
95 * Waits until a new host message arrives.
96 * This will block until a message becomes available.
97 *
98 * @returns VBox status code.
99 * @param u32ClientId The client id returned by VbglR3GuestCtrlConnect().
100 * @param puMsg Where to store the message id.
101 * @param puNumParms Where to store the number of parameters which will be received
102 * in a second call to the host.
103 */
104VBGLR3DECL(int) VbglR3GuestCtrlWaitForHostMsg(uint32_t u32ClientId, uint32_t *puMsg, uint32_t *puNumParms)
105{
106 AssertPtrReturn(puMsg, VERR_INVALID_PARAMETER);
107 AssertPtrReturn(puNumParms, VERR_INVALID_PARAMETER);
108
109 VBoxGuestCtrlHGCMMsgType Msg;
110
111 Msg.hdr.result = VERR_WRONG_ORDER;
112 Msg.hdr.u32ClientID = u32ClientId;
113 Msg.hdr.u32Function = GUEST_GET_HOST_MSG; /* Tell the host we want our next command. */
114 Msg.hdr.cParms = 2; /* Just peek for the next message! */
115
116 VbglHGCMParmUInt32Set(&Msg.msg, 0);
117 VbglHGCMParmUInt32Set(&Msg.num_parms, 0);
118
119 int rc = vbglR3DoIOCtl(VBOXGUEST_IOCTL_HGCM_CALL(sizeof(Msg)), &Msg, sizeof(Msg));
120 if (RT_SUCCESS(rc))
121 {
122 rc = VbglHGCMParmUInt32Get(&Msg.msg, puMsg);
123 if (RT_SUCCESS(rc))
124 rc = VbglHGCMParmUInt32Get(&Msg.num_parms, puNumParms);
125 if (RT_SUCCESS(rc))
126 rc = Msg.hdr.result;
127 /* Ok, so now we know what message type and how much parameters there are. */
128 }
129 return rc;
130}
131
132
133/**
134 * Asks the host to cancel (release) all pending waits which were deferred.
135 *
136 * @returns VBox status code.
137 * @param u32ClientId The client id returned by VbglR3GuestCtrlConnect().
138 */
139VBGLR3DECL(int) VbglR3GuestCtrlCancelPendingWaits(uint32_t u32ClientId)
140{
141 VBoxGuestCtrlHGCMMsgCancelPendingWaits Msg;
142
143 Msg.hdr.result = VERR_WRONG_ORDER;
144 Msg.hdr.u32ClientID = u32ClientId;
145 Msg.hdr.u32Function = GUEST_CANCEL_PENDING_WAITS;
146 Msg.hdr.cParms = 0;
147
148 int rc = vbglR3DoIOCtl(VBOXGUEST_IOCTL_HGCM_CALL(sizeof(Msg)), &Msg, sizeof(Msg));
149 if (RT_SUCCESS(rc))
150 {
151 int rc2 = Msg.hdr.result;
152 if (RT_FAILURE(rc2))
153 rc = rc2;
154 }
155 return rc;
156}
157
158
159/**
160 * Allocates and gets host data, based on the message id.
161 *
162 * This will block until data becomes available.
163 *
164 * @returns VBox status code.
165 * @param u32ClientId The client id returned by VbglR3GuestCtrlConnect().
166 * @param uNumParms
167 ** @todo Docs!
168 */
169VBGLR3DECL(int) VbglR3GuestCtrlExecGetHostCmdExec(uint32_t u32ClientId, uint32_t cParms,
170 uint32_t *puContext,
171 char *pszCmd, uint32_t cbCmd,
172 uint32_t *puFlags,
173 char *pszArgs, uint32_t cbArgs, uint32_t *pcArgs,
174 char *pszEnv, uint32_t *pcbEnv, uint32_t *pcEnvVars,
175 char *pszUser, uint32_t cbUser,
176 char *pszPassword, uint32_t cbPassword,
177 uint32_t *pcMsTimeLimit)
178{
179 AssertPtrReturn(puContext, VERR_INVALID_PARAMETER);
180 AssertPtrReturn(pszCmd, VERR_INVALID_PARAMETER);
181 AssertPtrReturn(puFlags, VERR_INVALID_PARAMETER);
182 AssertPtrReturn(pszArgs, VERR_INVALID_PARAMETER);
183 AssertPtrReturn(pcArgs, VERR_INVALID_PARAMETER);
184 AssertPtrReturn(pszEnv, VERR_INVALID_PARAMETER);
185 AssertPtrReturn(pcbEnv, VERR_INVALID_PARAMETER);
186 AssertPtrReturn(pcEnvVars, VERR_INVALID_PARAMETER);
187 AssertPtrReturn(pszUser, VERR_INVALID_PARAMETER);
188 AssertPtrReturn(pszPassword, VERR_INVALID_PARAMETER);
189 AssertPtrReturn(pcMsTimeLimit, VERR_INVALID_PARAMETER);
190
191 VBoxGuestCtrlHGCMMsgExecCmd Msg;
192
193 Msg.hdr.result = VERR_WRONG_ORDER;
194 Msg.hdr.u32ClientID = u32ClientId;
195 Msg.hdr.u32Function = GUEST_GET_HOST_MSG;
196 Msg.hdr.cParms = cParms; /** @todo r=bird: This isn't safe/right. The parameter count of this HGCM call
197 * is fixed from our point of view. */
198
199 VbglHGCMParmUInt32Set(&Msg.context, 0);
200 VbglHGCMParmPtrSet(&Msg.cmd, pszCmd, cbCmd);
201 VbglHGCMParmUInt32Set(&Msg.flags, 0);
202 VbglHGCMParmUInt32Set(&Msg.num_args, 0);
203 VbglHGCMParmPtrSet(&Msg.args, pszArgs, cbArgs);
204 VbglHGCMParmUInt32Set(&Msg.num_env, 0);
205 VbglHGCMParmUInt32Set(&Msg.cb_env, 0);
206 VbglHGCMParmPtrSet(&Msg.env, pszEnv, *pcbEnv);
207 VbglHGCMParmPtrSet(&Msg.username, pszUser, cbUser);
208 VbglHGCMParmPtrSet(&Msg.password, pszPassword, cbPassword);
209 VbglHGCMParmUInt32Set(&Msg.timeout, 0);
210
211 int rc = vbglR3DoIOCtl(VBOXGUEST_IOCTL_HGCM_CALL(sizeof(Msg)), &Msg, sizeof(Msg));
212 if (RT_SUCCESS(rc))
213 {
214 int rc2 = Msg.hdr.result;
215 if (RT_FAILURE(rc2))
216 {
217 rc = rc2;
218 }
219 else
220 {
221 Msg.context.GetUInt32(puContext);
222 Msg.flags.GetUInt32(puFlags);
223 Msg.num_args.GetUInt32(pcArgs);
224 Msg.num_env.GetUInt32(pcEnvVars);
225 Msg.cb_env.GetUInt32(pcbEnv);
226 Msg.timeout.GetUInt32(pcMsTimeLimit);
227 }
228 }
229 return rc;
230}
231
232
233/**
234 * Allocates and gets host data, based on the message id.
235 *
236 * This will block until data becomes available.
237 *
238 * @returns VBox status code.
239 * @param u32ClientId The client id returned by VbglR3GuestCtrlConnect().
240 * @param cParms
241 ** @todo Docs!
242 */
243VBGLR3DECL(int) VbglR3GuestCtrlExecGetHostCmdOutput(uint32_t u32ClientId, uint32_t cParms,
244 uint32_t *puContext, uint32_t *puPID,
245 uint32_t *puHandle, uint32_t *puFlags)
246{
247 AssertPtrReturn(puContext, VERR_INVALID_PARAMETER);
248 AssertPtrReturn(puPID, VERR_INVALID_PARAMETER);
249 AssertPtrReturn(puHandle, VERR_INVALID_PARAMETER);
250 AssertPtrReturn(puFlags, VERR_INVALID_PARAMETER);
251
252 VBoxGuestCtrlHGCMMsgExecOut Msg;
253
254 Msg.hdr.result = VERR_WRONG_ORDER;
255 Msg.hdr.u32ClientID = u32ClientId;
256 Msg.hdr.u32Function = GUEST_GET_HOST_MSG;
257 Msg.hdr.cParms = cParms;
258
259 VbglHGCMParmUInt32Set(&Msg.context, 0);
260 VbglHGCMParmUInt32Set(&Msg.pid, 0);
261 VbglHGCMParmUInt32Set(&Msg.handle, 0);
262 VbglHGCMParmUInt32Set(&Msg.flags, 0);
263
264 int rc = vbglR3DoIOCtl(VBOXGUEST_IOCTL_HGCM_CALL(sizeof(Msg)), &Msg, sizeof(Msg));
265 if (RT_SUCCESS(rc))
266 {
267 int rc2 = Msg.hdr.result;
268 if (RT_FAILURE(rc2))
269 {
270 rc = rc2;
271 }
272 else
273 {
274 Msg.context.GetUInt32(puContext);
275 Msg.pid.GetUInt32(puPID);
276 Msg.handle.GetUInt32(puHandle);
277 Msg.flags.GetUInt32(puFlags);
278 }
279 }
280 return rc;
281}
282
283
284/**
285 * Retrieves the input data from host which then gets sent to the
286 * started process.
287 *
288 * This will block until data becomes available.
289 *
290 * @returns VBox status code.
291 * @param u32ClientId The client id returned by VbglR3GuestCtrlConnect().
292 * @param cParms
293 ** @todo Docs!
294 */
295VBGLR3DECL(int) VbglR3GuestCtrlExecGetHostCmdInput(uint32_t u32ClientId, uint32_t cParms,
296 uint32_t *puContext, uint32_t *puPID,
297 uint32_t *puFlags,
298 void *pvData, uint32_t cbData,
299 uint32_t *pcbSize)
300{
301 AssertPtrReturn(puContext, VERR_INVALID_PARAMETER);
302 AssertPtrReturn(puPID, VERR_INVALID_PARAMETER);
303 AssertPtrReturn(puFlags, VERR_INVALID_PARAMETER);
304 AssertPtrReturn(pvData, VERR_INVALID_PARAMETER);
305 AssertPtrReturn(pcbSize, VERR_INVALID_PARAMETER);
306
307 VBoxGuestCtrlHGCMMsgExecIn Msg;
308
309 Msg.hdr.result = VERR_WRONG_ORDER;
310 Msg.hdr.u32ClientID = u32ClientId;
311 Msg.hdr.u32Function = GUEST_GET_HOST_MSG;
312 Msg.hdr.cParms = cParms;
313
314 VbglHGCMParmUInt32Set(&Msg.context, 0);
315 VbglHGCMParmUInt32Set(&Msg.pid, 0);
316 VbglHGCMParmUInt32Set(&Msg.flags, 0);
317 VbglHGCMParmPtrSet(&Msg.data, pvData, cbData);
318 VbglHGCMParmUInt32Set(&Msg.size, 0);
319
320 int rc = vbglR3DoIOCtl(VBOXGUEST_IOCTL_HGCM_CALL(sizeof(Msg)), &Msg, sizeof(Msg));
321 if (RT_SUCCESS(rc))
322 {
323 int rc2 = Msg.hdr.result;
324 if (RT_FAILURE(rc2))
325 {
326 rc = rc2;
327 }
328 else
329 {
330 Msg.context.GetUInt32(puContext);
331 Msg.pid.GetUInt32(puPID);
332 Msg.flags.GetUInt32(puFlags);
333 Msg.size.GetUInt32(pcbSize);
334 }
335 }
336 return rc;
337}
338
339
340/**
341 * Reports the process status (along with some other stuff) to the host.
342 *
343 * @returns VBox status code.
344 ** @todo Docs!
345 */
346VBGLR3DECL(int) VbglR3GuestCtrlExecReportStatus(uint32_t u32ClientId,
347 uint32_t u32Context,
348 uint32_t u32PID,
349 uint32_t u32Status,
350 uint32_t u32Flags,
351 void *pvData,
352 uint32_t cbData)
353{
354 VBoxGuestCtrlHGCMMsgExecStatus Msg;
355
356 Msg.hdr.result = VERR_WRONG_ORDER;
357 Msg.hdr.u32ClientID = u32ClientId;
358 Msg.hdr.u32Function = GUEST_EXEC_SEND_STATUS;
359 Msg.hdr.cParms = 5;
360
361 VbglHGCMParmUInt32Set(&Msg.context, u32Context);
362 VbglHGCMParmUInt32Set(&Msg.pid, u32PID);
363 VbglHGCMParmUInt32Set(&Msg.status, u32Status);
364 VbglHGCMParmUInt32Set(&Msg.flags, u32Flags);
365 VbglHGCMParmPtrSet(&Msg.data, pvData, cbData);
366
367 int rc = vbglR3DoIOCtl(VBOXGUEST_IOCTL_HGCM_CALL(sizeof(Msg)), &Msg, sizeof(Msg));
368 if (RT_SUCCESS(rc))
369 {
370 int rc2 = Msg.hdr.result;
371 if (RT_FAILURE(rc2))
372 rc = rc2;
373 }
374 return rc;
375}
376
377
378/**
379 * Sends output (from stdout/stderr) from a running process.
380 *
381 * @returns VBox status code.
382 ** @todo Docs!
383 */
384VBGLR3DECL(int) VbglR3GuestCtrlExecSendOut(uint32_t u32ClientId,
385 uint32_t u32Context,
386 uint32_t u32PID,
387 uint32_t u32Handle,
388 uint32_t u32Flags,
389 void *pvData,
390 uint32_t cbData)
391{
392 VBoxGuestCtrlHGCMMsgExecOut Msg;
393
394 Msg.hdr.result = VERR_WRONG_ORDER;
395 Msg.hdr.u32ClientID = u32ClientId;
396 Msg.hdr.u32Function = GUEST_EXEC_SEND_OUTPUT;
397 Msg.hdr.cParms = 5;
398
399 VbglHGCMParmUInt32Set(&Msg.context, u32Context);
400 VbglHGCMParmUInt32Set(&Msg.pid, u32PID);
401 VbglHGCMParmUInt32Set(&Msg.handle, u32Handle);
402 VbglHGCMParmUInt32Set(&Msg.flags, u32Flags);
403 VbglHGCMParmPtrSet(&Msg.data, pvData, cbData);
404
405 int rc = vbglR3DoIOCtl(VBOXGUEST_IOCTL_HGCM_CALL(sizeof(Msg)), &Msg, sizeof(Msg));
406 if (RT_SUCCESS(rc))
407 {
408 int rc2 = Msg.hdr.result;
409 if (RT_FAILURE(rc2))
410 rc = rc2;
411 }
412 return rc;
413}
414
415
416/**
417 * Reports back the input status to the host.
418 *
419 * @returns VBox status code.
420 ** @todo Docs!
421 */
422VBGLR3DECL(int) VbglR3GuestCtrlExecReportStatusIn(uint32_t u32ClientId,
423 uint32_t u32Context,
424 uint32_t u32PID,
425 uint32_t u32Status,
426 uint32_t u32Flags,
427 uint32_t cbWritten)
428{
429 VBoxGuestCtrlHGCMMsgExecStatusIn Msg;
430
431 Msg.hdr.result = VERR_WRONG_ORDER;
432 Msg.hdr.u32ClientID = u32ClientId;
433 Msg.hdr.u32Function = GUEST_EXEC_SEND_INPUT_STATUS;
434 Msg.hdr.cParms = 5;
435
436 VbglHGCMParmUInt32Set(&Msg.context, u32Context);
437 VbglHGCMParmUInt32Set(&Msg.pid, u32PID);
438 VbglHGCMParmUInt32Set(&Msg.status, u32Status);
439 VbglHGCMParmUInt32Set(&Msg.flags, u32Flags);
440 VbglHGCMParmUInt32Set(&Msg.written, cbWritten);
441
442 int rc = vbglR3DoIOCtl(VBOXGUEST_IOCTL_HGCM_CALL(sizeof(Msg)), &Msg, sizeof(Msg));
443 if (RT_SUCCESS(rc))
444 {
445 int rc2 = Msg.hdr.result;
446 if (RT_FAILURE(rc2))
447 rc = rc2;
448 }
449 return rc;
450}
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