VirtualBox

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

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

Logging.

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