VirtualBox

source: vbox/trunk/src/VBox/Additions/common/VBoxGuestLib/SysHlp.cpp@ 61439

Last change on this file since 61439 was 58183, checked in by vboxsync, 9 years ago

VBoxGuestLib: u32ClientI[Dd] -> idClient. Some cleanups.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 10.1 KB
Line 
1/* $Id: SysHlp.cpp 58183 2015-10-12 13:46:16Z vboxsync $ */
2/** @file
3 * VBoxGuestLibR0 - IDC with VBoxGuest and HGCM helpers.
4 */
5
6/*
7 * Copyright (C) 2006-2015 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#define LOG_GROUP LOG_GROUP_HGCM
28#include <VBox/log.h>
29
30#include <VBox/VBoxGuestLib.h>
31#include "SysHlp.h"
32
33#include <iprt/assert.h>
34
35#ifdef VBGL_VBOXGUEST
36
37#if !defined (RT_OS_WINDOWS)
38# include <iprt/memobj.h>
39# include <iprt/mem.h>
40#endif
41
42
43/**
44 * Internal worker for locking a range of linear addresses.
45 *
46 * @returns VBox status code.
47 * @param ppvCtx Where to store context data.
48 * @param pv The start of the range.
49 * @param u32Size The size of the range.
50 * @param fWriteAccess Lock for read-write (true) or readonly (false).
51 * @param fFlags HGCM call flags, VBGLR0_HGCM_F_XXX.
52 */
53int vbglLockLinear(void **ppvCtx, void *pv, uint32_t u32Size, bool fWriteAccess, uint32_t fFlags)
54{
55 int rc = VINF_SUCCESS;
56#ifndef RT_OS_WINDOWS
57 RTR0MEMOBJ MemObj = NIL_RTR0MEMOBJ;
58 uint32_t fAccess = RTMEM_PROT_READ | (fWriteAccess ? RTMEM_PROT_WRITE : 0);
59#endif
60
61 /* Zero size buffers shouldn't be locked. */
62 if (u32Size == 0)
63 {
64 Assert(pv == NULL);
65#ifdef RT_OS_WINDOWS
66 *ppvCtx = NULL;
67#else
68 *ppvCtx = NIL_RTR0MEMOBJ;
69#endif
70 return VINF_SUCCESS;
71 }
72
73 /** @todo just use IPRT here. the extra allocation shouldn't matter much...
74 * Then we can move all this up one level even. */
75#ifdef RT_OS_WINDOWS
76 PMDL pMdl = IoAllocateMdl(pv, u32Size, FALSE, FALSE, NULL);
77
78 if (pMdl == NULL)
79 {
80 rc = VERR_NOT_SUPPORTED;
81 AssertMsgFailed(("IoAllocateMdl %p %x failed!!\n", pv, u32Size));
82 }
83 else
84 {
85 __try {
86 /* Calls to MmProbeAndLockPages must be enclosed in a try/except block. */
87 MmProbeAndLockPages(pMdl,
88 /** @todo (fFlags & VBGLR0_HGCMCALL_F_MODE_MASK) == VBGLR0_HGCMCALL_F_USER? UserMode: KernelMode */
89 KernelMode,
90 (fWriteAccess) ? IoModifyAccess : IoReadAccess);
91
92 *ppvCtx = pMdl;
93
94 } __except(EXCEPTION_EXECUTE_HANDLER) {
95
96 IoFreeMdl(pMdl);
97 /** @todo */
98 rc = VERR_INVALID_PARAMETER;
99 AssertMsgFailed(("MmProbeAndLockPages %p %x failed!!\n", pv, u32Size));
100 }
101 }
102
103#else
104 /*
105 * Lock depending on context.
106 *
107 * Note: We will later use the memory object here to convert the HGCM
108 * linear buffer parameter into a physical page list. This is why
109 * we lock both kernel pages on all systems, even those where we
110 * know they aren't pageable.
111 */
112 if ((fFlags & VBGLR0_HGCMCALL_F_MODE_MASK) == VBGLR0_HGCMCALL_F_USER)
113 rc = RTR0MemObjLockUser(&MemObj, (RTR3PTR)pv, u32Size, fAccess, NIL_RTR0PROCESS);
114 else
115 rc = RTR0MemObjLockKernel(&MemObj, pv, u32Size, fAccess);
116 if (RT_SUCCESS(rc))
117 *ppvCtx = MemObj;
118 else
119 *ppvCtx = NIL_RTR0MEMOBJ;
120
121#endif
122
123 return rc;
124}
125
126void vbglUnlockLinear(void *pvCtx, void *pv, uint32_t u32Size)
127{
128#ifdef RT_OS_WINDOWS
129 PMDL pMdl = (PMDL)pvCtx;
130
131 Assert(pMdl);
132 if (pMdl != NULL)
133 {
134 MmUnlockPages(pMdl);
135 IoFreeMdl(pMdl);
136 }
137
138#else
139 RTR0MEMOBJ MemObj = (RTR0MEMOBJ)pvCtx;
140 int rc = RTR0MemObjFree(MemObj, false);
141 AssertRC(rc);
142
143#endif
144
145 NOREF(pv);
146 NOREF(u32Size);
147}
148
149#else /* !VBGL_VBOXGUEST */
150
151# ifdef RT_OS_OS2
152# include <VBox/VBoxGuest.h> /* for VBOXGUESTOS2IDCCONNECT */
153RT_C_DECLS_BEGIN
154/*
155 * On OS/2 we'll do the connecting in the assembly code of the
156 * client driver, exporting a g_VBoxGuestIDC symbol containing
157 * the connection information obtained from the 16-bit IDC.
158 */
159extern VBOXGUESTOS2IDCCONNECT g_VBoxGuestIDC;
160RT_C_DECLS_END
161# endif
162
163# if !defined(RT_OS_OS2) \
164 && !defined(RT_OS_WINDOWS)
165RT_C_DECLS_BEGIN
166extern DECLVBGL(void *) VBoxGuestIDCOpen(uint32_t *pu32Version);
167extern DECLVBGL(void) VBoxGuestIDCClose(void *pvOpaque);
168extern DECLVBGL(int) VBoxGuestIDCCall(void *pvOpaque, unsigned int iCmd, void *pvData, size_t cbSize, size_t *pcbReturn);
169RT_C_DECLS_END
170# endif
171
172bool vbglDriverIsOpened(VBGLDRIVER *pDriver)
173{
174# ifdef RT_OS_WINDOWS
175 return pDriver->pFileObject != NULL;
176# elif defined (RT_OS_OS2)
177 return pDriver->u32Session != UINT32_MAX && pDriver->u32Session != 0;
178# else
179 return pDriver->pvOpaque != NULL;
180# endif
181}
182
183int vbglDriverOpen(VBGLDRIVER *pDriver)
184{
185# ifdef RT_OS_WINDOWS
186 UNICODE_STRING uszDeviceName;
187 RtlInitUnicodeString(&uszDeviceName, L"\\Device\\VBoxGuest");
188
189 PDEVICE_OBJECT pDeviceObject = NULL;
190 PFILE_OBJECT pFileObject = NULL;
191
192 NTSTATUS rc = IoGetDeviceObjectPointer(&uszDeviceName, FILE_ALL_ACCESS, &pFileObject, &pDeviceObject);
193 if (NT_SUCCESS(rc))
194 {
195 Log(("vbglDriverOpen VBoxGuest successful pDeviceObject=%x\n", pDeviceObject));
196 pDriver->pDeviceObject = pDeviceObject;
197 pDriver->pFileObject = pFileObject;
198 return VINF_SUCCESS;
199 }
200 /** @todo return RTErrConvertFromNtStatus(rc)! */
201 Log(("vbglDriverOpen VBoxGuest failed with ntstatus=%x\n", rc));
202 return rc;
203
204# elif defined (RT_OS_OS2)
205 /*
206 * Just check whether the connection was made or not.
207 */
208 if ( g_VBoxGuestIDC.u32Version == VMMDEV_VERSION
209 && RT_VALID_PTR(g_VBoxGuestIDC.u32Session)
210 && RT_VALID_PTR(g_VBoxGuestIDC.pfnServiceEP))
211 {
212 pDriver->u32Session = g_VBoxGuestIDC.u32Session;
213 return VINF_SUCCESS;
214 }
215 pDriver->u32Session = UINT32_MAX;
216 Log(("vbglDriverOpen: failed\n"));
217 return VERR_FILE_NOT_FOUND;
218
219# else
220 uint32_t u32VMMDevVersion;
221 pDriver->pvOpaque = VBoxGuestIDCOpen(&u32VMMDevVersion);
222 if ( pDriver->pvOpaque
223 && u32VMMDevVersion == VMMDEV_VERSION)
224 return VINF_SUCCESS;
225
226 Log(("vbglDriverOpen: failed\n"));
227 return VERR_FILE_NOT_FOUND;
228# endif
229}
230
231# ifdef RT_OS_WINDOWS
232static NTSTATUS vbglDriverIOCtlCompletion(IN PDEVICE_OBJECT DeviceObject, IN PIRP Irp, IN PVOID Context)
233{
234 Log(("VBGL completion %x\n", Irp));
235
236 KEVENT *pEvent = (KEVENT *)Context;
237 KeSetEvent(pEvent, IO_NO_INCREMENT, FALSE);
238
239 return STATUS_MORE_PROCESSING_REQUIRED;
240}
241# endif
242
243int vbglDriverIOCtl(VBGLDRIVER *pDriver, uint32_t u32Function, void *pvData, uint32_t cbData)
244{
245 Log(("vbglDriverIOCtl: pDriver: %p, Func: %x, pvData: %p, cbData: %d\n", pDriver, u32Function, pvData, cbData));
246
247# ifdef RT_OS_WINDOWS
248 KEVENT Event;
249
250 KeInitializeEvent(&Event, NotificationEvent, FALSE);
251
252 /* Have to use the IoAllocateIRP method because this code is generic and
253 * must work in any thread context.
254 * The IoBuildDeviceIoControlRequest, which was used here, does not work
255 * when APCs are disabled, for example.
256 */
257 PIRP irp = IoAllocateIrp(pDriver->pDeviceObject->StackSize, FALSE);
258
259 Log(("vbglDriverIOCtl: irp %p, IRQL = %d\n", irp, KeGetCurrentIrql()));
260
261 if (irp == NULL)
262 {
263 Log(("vbglDriverIOCtl: IRP allocation failed!\n"));
264 return VERR_NO_MEMORY;
265 }
266
267 /*
268 * Setup the IRP_MJ_DEVICE_CONTROL IRP.
269 */
270
271 PIO_STACK_LOCATION nextStack = IoGetNextIrpStackLocation(irp);
272
273 nextStack->MajorFunction = IRP_MJ_DEVICE_CONTROL;
274 nextStack->MinorFunction = 0;
275 nextStack->DeviceObject = pDriver->pDeviceObject;
276 nextStack->Parameters.DeviceIoControl.OutputBufferLength = cbData;
277 nextStack->Parameters.DeviceIoControl.InputBufferLength = cbData;
278 nextStack->Parameters.DeviceIoControl.IoControlCode = u32Function;
279 nextStack->Parameters.DeviceIoControl.Type3InputBuffer = pvData;
280
281 irp->AssociatedIrp.SystemBuffer = pvData; /* Output buffer. */
282 irp->MdlAddress = NULL;
283
284 /* A completion routine is required to signal the Event. */
285 IoSetCompletionRoutine(irp, vbglDriverIOCtlCompletion, &Event, TRUE, TRUE, TRUE);
286
287 NTSTATUS rc = IoCallDriver(pDriver->pDeviceObject, irp);
288
289 if (NT_SUCCESS (rc))
290 {
291 /* Wait the event to be signalled by the completion routine. */
292 KeWaitForSingleObject(&Event, Executive, KernelMode, FALSE, NULL);
293
294 rc = irp->IoStatus.Status;
295
296 Log(("vbglDriverIOCtl: wait completed IRQL = %d\n", KeGetCurrentIrql()));
297 }
298
299 IoFreeIrp(irp);
300
301 if (rc != STATUS_SUCCESS)
302 Log(("vbglDriverIOCtl: ntstatus=%x\n", rc));
303
304 if (NT_SUCCESS(rc))
305 return VINF_SUCCESS;
306 if (rc == STATUS_INVALID_PARAMETER)
307 return VERR_INVALID_PARAMETER;
308 if (rc == STATUS_INVALID_BUFFER_SIZE)
309 return VERR_OUT_OF_RANGE;
310 return VERR_VBGL_IOCTL_FAILED;
311
312# elif defined (RT_OS_OS2)
313 if ( pDriver->u32Session
314 && pDriver->u32Session == g_VBoxGuestIDC.u32Session)
315 return g_VBoxGuestIDC.pfnServiceEP(pDriver->u32Session, u32Function, pvData, cbData, NULL);
316
317 Log(("vbglDriverIOCtl: No connection\n"));
318 return VERR_WRONG_ORDER;
319
320# else
321 return VBoxGuestIDCCall(pDriver->pvOpaque, u32Function, pvData, cbData, NULL);
322# endif
323}
324
325void vbglDriverClose(VBGLDRIVER *pDriver)
326{
327# ifdef RT_OS_WINDOWS
328 Log(("vbglDriverClose pDeviceObject=%x\n", pDriver->pDeviceObject));
329 ObDereferenceObject(pDriver->pFileObject);
330 pDriver->pFileObject = NULL;
331 pDriver->pDeviceObject = NULL;
332
333# elif defined (RT_OS_OS2)
334 pDriver->u32Session = 0;
335
336# else
337 VBoxGuestIDCClose(pDriver->pvOpaque);
338 pDriver->pvOpaque = NULL;
339# endif
340}
341
342#endif /* !VBGL_VBOXGUEST */
343
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