VirtualBox

source: vbox/trunk/src/VBox/HostDrivers/Support/win/SUPDrv-win.cpp@ 47506

Last change on this file since 47506 was 44528, checked in by vboxsync, 12 years ago

header (C) fixes

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 40.4 KB
Line 
1/* $Id: SUPDrv-win.cpp 44528 2013-02-04 14:27:54Z vboxsync $ */
2/** @file
3 * VBoxDrv - The VirtualBox Support Driver - Windows NT specifics.
4 */
5
6/*
7 * Copyright (C) 2006-2012 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* Header Files *
29*******************************************************************************/
30#define LOG_GROUP LOG_GROUP_SUP_DRV
31#include "../SUPDrvInternal.h"
32#include <excpt.h>
33#include <ntimage.h>
34
35#include <iprt/assert.h>
36#include <iprt/initterm.h>
37#include <iprt/mem.h>
38#include <iprt/process.h>
39#include <iprt/power.h>
40#include <iprt/string.h>
41#include <VBox/log.h>
42
43
44/*******************************************************************************
45* Defined Constants And Macros *
46*******************************************************************************/
47/** The support service name. */
48#define SERVICE_NAME "VBoxDrv"
49/** The Pool tag (VBox). */
50#define SUPDRV_NT_POOL_TAG 'xoBV'
51
52/** Win32 device name for system access. */
53#define DEVICE_NAME_SYS "\\\\.\\VBoxDrv"
54/** NT device name for system access. */
55#define DEVICE_NAME_NT_SYS L"\\Device\\VBoxDrv"
56/** Win Symlink name for system access. */
57#define DEVICE_NAME_DOS_SYS L"\\DosDevices\\VBoxDrv"
58
59/** Win32 device name for user access. */
60#define DEVICE_NAME_USR "\\\\.\\VBoxDrvU"
61/** NT device name for user access. */
62#define DEVICE_NAME_NT_USR L"\\Device\\VBoxDrvU"
63/** Win Symlink name for user access. */
64#define DEVICE_NAME_DOS_USR L"\\DosDevices\\VBoxDrvU"
65
66
67/*******************************************************************************
68* Structures and Typedefs *
69*******************************************************************************/
70#if 0 //def RT_ARCH_AMD64
71typedef struct SUPDRVEXECMEM
72{
73 PMDL pMdl;
74 void *pvMapping;
75 void *pvAllocation;
76} SUPDRVEXECMEM, *PSUPDRVEXECMEM;
77#endif
78
79/**
80 * Device extension used by VBoxDrvU.
81 */
82typedef struct SUPDRVDEVEXTUSR
83{
84 /** Global cookie (same location as in SUPDRVDEVEXT, different value). */
85 uint32_t u32Cookie;
86 /** Pointer to the main driver extension. */
87 PSUPDRVDEVEXT pMainDrvExt;
88} SUPDRVDEVEXTUSR;
89AssertCompileMembersAtSameOffset(SUPDRVDEVEXT, u32Cookie, SUPDRVDEVEXTUSR, u32Cookie);
90/** Pointer to the VBoxDrvU device extension. */
91typedef SUPDRVDEVEXTUSR *PSUPDRVDEVEXTUSR;
92/** Value of SUPDRVDEVEXTUSR::u32Cookie. */
93#define SUPDRVDEVEXTUSR_COOKIE UINT32_C(0x12345678)
94
95/** Get the main device extension. */
96#define SUPDRVNT_GET_DEVEXT(pDevObj) \
97 ( pDevObj != g_pDevObjUsr \
98 ? (PSUPDRVDEVEXT)pDevObj->DeviceExtension \
99 : ((PSUPDRVDEVEXTUSR)pDevObj->DeviceExtension)->pMainDrvExt )
100
101
102/*******************************************************************************
103* Internal Functions *
104*******************************************************************************/
105static void _stdcall VBoxDrvNtUnload(PDRIVER_OBJECT pDrvObj);
106static NTSTATUS _stdcall VBoxDrvNtCreate(PDEVICE_OBJECT pDevObj, PIRP pIrp);
107static NTSTATUS _stdcall VBoxDrvNtCleanup(PDEVICE_OBJECT pDevObj, PIRP pIrp);
108static NTSTATUS _stdcall VBoxDrvNtClose(PDEVICE_OBJECT pDevObj, PIRP pIrp);
109static NTSTATUS _stdcall VBoxDrvNtDeviceControl(PDEVICE_OBJECT pDevObj, PIRP pIrp);
110static int VBoxDrvNtDeviceControlSlow(PSUPDRVDEVEXT pDevExt, PSUPDRVSESSION pSession, PIRP pIrp, PIO_STACK_LOCATION pStack);
111static NTSTATUS _stdcall VBoxDrvNtInternalDeviceControl(PDEVICE_OBJECT pDevObj, PIRP pIrp);
112static VOID _stdcall VBoxPowerDispatchCallback(PVOID pCallbackContext, PVOID pArgument1, PVOID pArgument2);
113static NTSTATUS _stdcall VBoxDrvNtNotSupportedStub(PDEVICE_OBJECT pDevObj, PIRP pIrp);
114static NTSTATUS VBoxDrvNtErr2NtStatus(int rc);
115
116
117/*******************************************************************************
118* Exported Functions *
119*******************************************************************************/
120RT_C_DECLS_BEGIN
121ULONG _stdcall DriverEntry(PDRIVER_OBJECT pDrvObj, PUNICODE_STRING pRegPath);
122RT_C_DECLS_END
123
124
125/*******************************************************************************
126* Global Variables *
127*******************************************************************************/
128/** Pointer to the system device instance. */
129static PDEVICE_OBJECT g_pDevObjSys = NULL;
130/** Pointer to the user device instance. */
131static PDEVICE_OBJECT g_pDevObjUsr = NULL;
132
133
134/**
135 * Takes care of creating the devices and their symbolic links.
136 *
137 * @returns NT status code.
138 * @param pDrvObj Pointer to driver object.
139 */
140static NTSTATUS vboxdrvNtCreateDevices(PDRIVER_OBJECT pDrvObj)
141{
142 /*
143 * System device.
144 */
145 UNICODE_STRING DevName;
146 RtlInitUnicodeString(&DevName, DEVICE_NAME_NT_SYS);
147 NTSTATUS rcNt = IoCreateDevice(pDrvObj, sizeof(SUPDRVDEVEXT), &DevName, FILE_DEVICE_UNKNOWN, 0, FALSE, &g_pDevObjSys);
148 if (NT_SUCCESS(rcNt))
149 {
150 UNICODE_STRING DosName;
151 RtlInitUnicodeString(&DosName, DEVICE_NAME_DOS_SYS);
152 rcNt = IoCreateSymbolicLink(&DosName, &DevName);
153 if (NT_SUCCESS(rcNt))
154 {
155 /*
156 * User device.
157 */
158 RtlInitUnicodeString(&DevName, DEVICE_NAME_NT_USR);
159 rcNt = IoCreateDevice(pDrvObj, sizeof(SUPDRVDEVEXTUSR), &DevName, FILE_DEVICE_UNKNOWN, 0, FALSE, &g_pDevObjUsr);
160 if (NT_SUCCESS(rcNt))
161 {
162 UNICODE_STRING DosName;
163 RtlInitUnicodeString(&DosName, DEVICE_NAME_DOS_USR);
164 rcNt = IoCreateSymbolicLink(&DosName, &DevName);
165 if (NT_SUCCESS(rcNt))
166 {
167 PSUPDRVDEVEXTUSR pDevExtUsr = (PSUPDRVDEVEXTUSR)g_pDevObjUsr->DeviceExtension;
168 pDevExtUsr->pMainDrvExt = (PSUPDRVDEVEXT)g_pDevObjSys->DeviceExtension;
169 pDevExtUsr->u32Cookie = SUPDRVDEVEXTUSR_COOKIE;
170
171 /* Done. */
172 return rcNt;
173 }
174
175 /* Bail out. */
176 IoDeleteDevice(g_pDevObjUsr);
177 g_pDevObjUsr = NULL;
178 }
179 IoDeleteSymbolicLink(&DosName);
180 }
181 IoDeleteDevice(g_pDevObjSys);
182 g_pDevObjSys = NULL;
183 }
184 return rcNt;
185}
186
187/**
188 * Destroys the devices and links created by vboxdrvNtCreateDevices.
189 */
190static void vboxdrvNtDestroyDevices(void)
191{
192 UNICODE_STRING DosName;
193 RtlInitUnicodeString(&DosName, DEVICE_NAME_DOS_SYS);
194 NTSTATUS rcNt = IoDeleteSymbolicLink(&DosName);
195
196 RtlInitUnicodeString(&DosName, DEVICE_NAME_DOS_USR);
197 rcNt = IoDeleteSymbolicLink(&DosName);
198
199 PSUPDRVDEVEXTUSR pDevExtUsr = (PSUPDRVDEVEXTUSR)g_pDevObjUsr->DeviceExtension;
200 pDevExtUsr->pMainDrvExt = NULL;
201
202 IoDeleteDevice(g_pDevObjUsr);
203 g_pDevObjUsr = NULL;
204 IoDeleteDevice(g_pDevObjSys);
205 g_pDevObjSys = NULL;
206}
207
208
209/**
210 * Driver entry point.
211 *
212 * @returns appropriate status code.
213 * @param pDrvObj Pointer to driver object.
214 * @param pRegPath Registry base path.
215 */
216ULONG _stdcall DriverEntry(PDRIVER_OBJECT pDrvObj, PUNICODE_STRING pRegPath)
217{
218 /*
219 * Create device.
220 * (That means creating a device object and a symbolic link so the DOS
221 * subsystems (OS/2, win32, ++) can access the device.)
222 */
223 NTSTATUS rcNt = vboxdrvNtCreateDevices(pDrvObj);
224 if (NT_SUCCESS(rcNt))
225 {
226 int vrc = RTR0Init(0);
227 if (RT_SUCCESS(vrc))
228 {
229 Log(("VBoxDrv::DriverEntry\n"));
230
231 /*
232 * Initialize the device extension.
233 */
234 PSUPDRVDEVEXT pDevExt = (PSUPDRVDEVEXT)g_pDevObjSys->DeviceExtension;
235 memset(pDevExt, 0, sizeof(*pDevExt));
236
237 vrc = supdrvInitDevExt(pDevExt, sizeof(SUPDRVSESSION));
238 if (!vrc)
239 {
240 /*
241 * Setup the driver entry points in pDrvObj.
242 */
243 pDrvObj->DriverUnload = VBoxDrvNtUnload;
244 pDrvObj->MajorFunction[IRP_MJ_CREATE] = VBoxDrvNtCreate;
245 pDrvObj->MajorFunction[IRP_MJ_CLEANUP] = VBoxDrvNtCleanup;
246 pDrvObj->MajorFunction[IRP_MJ_CLOSE] = VBoxDrvNtClose;
247 pDrvObj->MajorFunction[IRP_MJ_DEVICE_CONTROL] = VBoxDrvNtDeviceControl;
248 pDrvObj->MajorFunction[IRP_MJ_INTERNAL_DEVICE_CONTROL] = VBoxDrvNtInternalDeviceControl;
249 pDrvObj->MajorFunction[IRP_MJ_READ] = VBoxDrvNtNotSupportedStub;
250 pDrvObj->MajorFunction[IRP_MJ_WRITE] = VBoxDrvNtNotSupportedStub;
251
252 /* more? */
253
254 /* Register ourselves for power state changes. */
255 UNICODE_STRING CallbackName;
256 OBJECT_ATTRIBUTES Attr;
257
258 RtlInitUnicodeString(&CallbackName, L"\\Callback\\PowerState");
259 InitializeObjectAttributes(&Attr, &CallbackName, OBJ_CASE_INSENSITIVE, NULL, NULL);
260
261 rcNt = ExCreateCallback(&pDevExt->pObjPowerCallback, &Attr, TRUE, TRUE);
262 if (rcNt == STATUS_SUCCESS)
263 pDevExt->hPowerCallback = ExRegisterCallback(pDevExt->pObjPowerCallback, VBoxPowerDispatchCallback,
264 g_pDevObjSys);
265
266 Log(("VBoxDrv::DriverEntry returning STATUS_SUCCESS\n"));
267 return STATUS_SUCCESS;
268 }
269
270 Log(("supdrvInitDevExit failed with vrc=%d!\n", vrc));
271 rcNt = VBoxDrvNtErr2NtStatus(vrc);
272
273 RTR0Term();
274 }
275 else
276 {
277 Log(("RTR0Init failed with vrc=%d!\n", vrc));
278 rcNt = VBoxDrvNtErr2NtStatus(vrc);
279 }
280
281 vboxdrvNtDestroyDevices();
282 }
283 if (NT_SUCCESS(rcNt))
284 rcNt = STATUS_INVALID_PARAMETER;
285 return rcNt;
286}
287
288
289/**
290 * Unload the driver.
291 *
292 * @param pDrvObj Driver object.
293 */
294void _stdcall VBoxDrvNtUnload(PDRIVER_OBJECT pDrvObj)
295{
296 PSUPDRVDEVEXT pDevExt = (PSUPDRVDEVEXT)g_pDevObjSys->DeviceExtension;
297
298 Log(("VBoxDrvNtUnload at irql %d\n", KeGetCurrentIrql()));
299
300 /* Clean up the power callback registration. */
301 if (pDevExt->hPowerCallback)
302 ExUnregisterCallback(pDevExt->hPowerCallback);
303 if (pDevExt->pObjPowerCallback)
304 ObDereferenceObject(pDevExt->pObjPowerCallback);
305
306 /*
307 * We ASSUME that it's not possible to unload a driver with open handles.
308 */
309 supdrvDeleteDevExt(pDevExt);
310 RTR0Term();
311 vboxdrvNtDestroyDevices();
312
313 NOREF(pDrvObj);
314}
315
316
317/**
318 * Create (i.e. Open) file entry point.
319 *
320 * @param pDevObj Device object.
321 * @param pIrp Request packet.
322 */
323NTSTATUS _stdcall VBoxDrvNtCreate(PDEVICE_OBJECT pDevObj, PIRP pIrp)
324{
325 Log(("VBoxDrvNtCreate: RequestorMode=%d\n", pIrp->RequestorMode));
326 const bool fUnrestricted = pDevObj == g_pDevObjSys;
327 PIO_STACK_LOCATION pStack = IoGetCurrentIrpStackLocation(pIrp);
328 PFILE_OBJECT pFileObj = pStack->FileObject;
329 PSUPDRVDEVEXT pDevExt = SUPDRVNT_GET_DEVEXT(pDevObj);
330
331 /*
332 * We are not remotely similar to a directory...
333 * (But this is possible.)
334 */
335 if (pStack->Parameters.Create.Options & FILE_DIRECTORY_FILE)
336 {
337 pIrp->IoStatus.Status = STATUS_NOT_A_DIRECTORY;
338 pIrp->IoStatus.Information = 0;
339 IoCompleteRequest(pIrp, IO_NO_INCREMENT);
340 return STATUS_NOT_A_DIRECTORY;
341 }
342
343 /*
344 * Don't create a session for kernel clients, they'll close the handle
345 * immediately and work with the file object via
346 * VBoxDrvNtInternalDeviceControl. The first request will there be one
347 * to create a session.
348 */
349 NTSTATUS rcNt;
350 if (pIrp->RequestorMode == KernelMode)
351 rcNt = STATUS_SUCCESS;
352 else
353 {
354 /*
355 * Call common code for the rest.
356 */
357 pFileObj->FsContext = NULL;
358 PSUPDRVSESSION pSession;
359 int rc = supdrvCreateSession(pDevExt, true /*fUser*/, fUnrestricted, &pSession);
360 if (!rc)
361 pFileObj->FsContext = pSession;
362 rcNt = pIrp->IoStatus.Status = VBoxDrvNtErr2NtStatus(rc);
363 }
364
365 pIrp->IoStatus.Information = 0;
366 IoCompleteRequest(pIrp, IO_NO_INCREMENT);
367
368 return rcNt;
369}
370
371
372/**
373 * Clean up file handle entry point.
374 *
375 * @param pDevObj Device object.
376 * @param pIrp Request packet.
377 */
378NTSTATUS _stdcall VBoxDrvNtCleanup(PDEVICE_OBJECT pDevObj, PIRP pIrp)
379{
380 PSUPDRVDEVEXT pDevExt = SUPDRVNT_GET_DEVEXT(pDevObj);
381 PIO_STACK_LOCATION pStack = IoGetCurrentIrpStackLocation(pIrp);
382 PFILE_OBJECT pFileObj = pStack->FileObject;
383 PSUPDRVSESSION pSession = (PSUPDRVSESSION)pFileObj->FsContext;
384
385 Log(("VBoxDrvNtCleanup: pDevExt=%p pFileObj=%p pSession=%p\n", pDevExt, pFileObj, pSession));
386 if (pSession)
387 {
388 supdrvCloseSession(pDevExt, (PSUPDRVSESSION)pFileObj->FsContext);
389 pFileObj->FsContext = NULL;
390 }
391
392 pIrp->IoStatus.Information = 0;
393 pIrp->IoStatus.Status = STATUS_SUCCESS;
394 IoCompleteRequest(pIrp, IO_NO_INCREMENT);
395
396 return STATUS_SUCCESS;
397}
398
399
400/**
401 * Close file entry point.
402 *
403 * @param pDevObj Device object.
404 * @param pIrp Request packet.
405 */
406NTSTATUS _stdcall VBoxDrvNtClose(PDEVICE_OBJECT pDevObj, PIRP pIrp)
407{
408 PSUPDRVDEVEXT pDevExt = SUPDRVNT_GET_DEVEXT(pDevObj);
409 PIO_STACK_LOCATION pStack = IoGetCurrentIrpStackLocation(pIrp);
410 PFILE_OBJECT pFileObj = pStack->FileObject;
411 PSUPDRVSESSION pSession = (PSUPDRVSESSION)pFileObj->FsContext;
412
413 Log(("VBoxDrvNtClose: pDevExt=%p pFileObj=%p pSession=%p\n", pDevExt, pFileObj, pSession));
414 if (pSession)
415 {
416 supdrvCloseSession(pDevExt, (PSUPDRVSESSION)pFileObj->FsContext);
417 pFileObj->FsContext = NULL;
418 }
419
420 pIrp->IoStatus.Information = 0;
421 pIrp->IoStatus.Status = STATUS_SUCCESS;
422 IoCompleteRequest(pIrp, IO_NO_INCREMENT);
423
424 return STATUS_SUCCESS;
425}
426
427
428/**
429 * Device I/O Control entry point.
430 *
431 * @param pDevObj Device object.
432 * @param pIrp Request packet.
433 */
434NTSTATUS _stdcall VBoxDrvNtDeviceControl(PDEVICE_OBJECT pDevObj, PIRP pIrp)
435{
436 PSUPDRVDEVEXT pDevExt = SUPDRVNT_GET_DEVEXT(pDevObj);
437 PIO_STACK_LOCATION pStack = IoGetCurrentIrpStackLocation(pIrp);
438 PSUPDRVSESSION pSession = (PSUPDRVSESSION)pStack->FileObject->FsContext;
439
440 /*
441 * Deal with the two high-speed IOCtl that takes it's arguments from
442 * the session and iCmd, and only returns a VBox status code.
443 *
444 * Note: The previous method of returning the rc prior to IOC version
445 * 7.4 has been abandond, we're no longer compatible with that
446 * interface.
447 */
448 ULONG ulCmd = pStack->Parameters.DeviceIoControl.IoControlCode;
449 if ( ( ulCmd == SUP_IOCTL_FAST_DO_RAW_RUN
450 || ulCmd == SUP_IOCTL_FAST_DO_HM_RUN
451 || ulCmd == SUP_IOCTL_FAST_DO_NOP)
452 && pSession->fUnrestricted == true)
453 {
454#ifdef VBOX_WITH_VMMR0_DISABLE_PREEMPTION
455 int rc = supdrvIOCtlFast(ulCmd, (unsigned)(uintptr_t)pIrp->UserBuffer /* VMCPU id */, pDevExt, pSession);
456#else
457 /* Raise the IRQL to DISPATCH_LEVEL to prevent Windows from rescheduling us to another CPU/core. */
458 Assert(KeGetCurrentIrql() <= DISPATCH_LEVEL);
459 KIRQL oldIrql;
460 KeRaiseIrql(DISPATCH_LEVEL, &oldIrql);
461 int rc = supdrvIOCtlFast(ulCmd, (unsigned)(uintptr_t)pIrp->UserBuffer /* VMCPU id */, pDevExt, pSession);
462 KeLowerIrql(oldIrql);
463#endif
464
465 /* Complete the I/O request. */
466 NTSTATUS rcNt = pIrp->IoStatus.Status = RT_SUCCESS(rc) ? STATUS_SUCCESS : STATUS_INVALID_PARAMETER;
467 IoCompleteRequest(pIrp, IO_NO_INCREMENT);
468 return rcNt;
469 }
470
471 return VBoxDrvNtDeviceControlSlow(pDevExt, pSession, pIrp, pStack);
472}
473
474
475/**
476 * Worker for VBoxDrvNtDeviceControl that takes the slow IOCtl functions.
477 *
478 * @returns NT status code.
479 *
480 * @param pDevObj Device object.
481 * @param pSession The session.
482 * @param pIrp Request packet.
483 * @param pStack The stack location containing the DeviceControl parameters.
484 */
485static int VBoxDrvNtDeviceControlSlow(PSUPDRVDEVEXT pDevExt, PSUPDRVSESSION pSession, PIRP pIrp, PIO_STACK_LOCATION pStack)
486{
487 NTSTATUS rcNt;
488 unsigned cbOut = 0;
489 int rc = 0;
490 Log2(("VBoxDrvNtDeviceControlSlow(%p,%p): ioctl=%#x pBuf=%p cbIn=%#x cbOut=%#x pSession=%p\n",
491 pDevExt, pIrp, pStack->Parameters.DeviceIoControl.IoControlCode,
492 pIrp->AssociatedIrp.SystemBuffer, pStack->Parameters.DeviceIoControl.InputBufferLength,
493 pStack->Parameters.DeviceIoControl.OutputBufferLength, pSession));
494
495#ifdef RT_ARCH_AMD64
496 /* Don't allow 32-bit processes to do any I/O controls. */
497 if (!IoIs32bitProcess(pIrp))
498#endif
499 {
500 /* Verify that it's a buffered CTL. */
501 if ((pStack->Parameters.DeviceIoControl.IoControlCode & 0x3) == METHOD_BUFFERED)
502 {
503 /* Verify that the sizes in the request header are correct. */
504 PSUPREQHDR pHdr = (PSUPREQHDR)pIrp->AssociatedIrp.SystemBuffer;
505 if ( pStack->Parameters.DeviceIoControl.InputBufferLength >= sizeof(*pHdr)
506 && pStack->Parameters.DeviceIoControl.InputBufferLength == pHdr->cbIn
507 && pStack->Parameters.DeviceIoControl.OutputBufferLength == pHdr->cbOut)
508 {
509 /*
510 * Do the job.
511 */
512 rc = supdrvIOCtl(pStack->Parameters.DeviceIoControl.IoControlCode, pDevExt, pSession, pHdr);
513 if (!rc)
514 {
515 rcNt = STATUS_SUCCESS;
516 cbOut = pHdr->cbOut;
517 if (cbOut > pStack->Parameters.DeviceIoControl.OutputBufferLength)
518 {
519 cbOut = pStack->Parameters.DeviceIoControl.OutputBufferLength;
520 OSDBGPRINT(("VBoxDrvLinuxIOCtl: too much output! %#x > %#x; uCmd=%#x!\n",
521 pHdr->cbOut, cbOut, pStack->Parameters.DeviceIoControl.IoControlCode));
522 }
523 }
524 else
525 rcNt = STATUS_INVALID_PARAMETER;
526 Log2(("VBoxDrvNtDeviceControlSlow: returns %#x cbOut=%d rc=%#x\n", rcNt, cbOut, rc));
527 }
528 else
529 {
530 Log(("VBoxDrvNtDeviceControlSlow: Mismatching sizes (%#x) - Hdr=%#lx/%#lx Irp=%#lx/%#lx!\n",
531 pStack->Parameters.DeviceIoControl.IoControlCode,
532 pStack->Parameters.DeviceIoControl.InputBufferLength >= sizeof(*pHdr) ? pHdr->cbIn : 0,
533 pStack->Parameters.DeviceIoControl.InputBufferLength >= sizeof(*pHdr) ? pHdr->cbOut : 0,
534 pStack->Parameters.DeviceIoControl.InputBufferLength,
535 pStack->Parameters.DeviceIoControl.OutputBufferLength));
536 rcNt = STATUS_INVALID_PARAMETER;
537 }
538 }
539 else
540 {
541 Log(("VBoxDrvNtDeviceControlSlow: not buffered request (%#x) - not supported\n",
542 pStack->Parameters.DeviceIoControl.IoControlCode));
543 rcNt = STATUS_NOT_SUPPORTED;
544 }
545 }
546#ifdef RT_ARCH_AMD64
547 else
548 {
549 Log(("VBoxDrvNtDeviceControlSlow: WOW64 req - not supported\n"));
550 rcNt = STATUS_NOT_SUPPORTED;
551 }
552#endif
553
554 /* complete the request. */
555 pIrp->IoStatus.Status = rcNt;
556 pIrp->IoStatus.Information = cbOut;
557 IoCompleteRequest(pIrp, IO_NO_INCREMENT);
558 return rcNt;
559}
560
561
562/**
563 * Internal Device I/O Control entry point, used for IDC.
564 *
565 * @param pDevObj Device object.
566 * @param pIrp Request packet.
567 */
568NTSTATUS _stdcall VBoxDrvNtInternalDeviceControl(PDEVICE_OBJECT pDevObj, PIRP pIrp)
569{
570 PSUPDRVDEVEXT pDevExt = SUPDRVNT_GET_DEVEXT(pDevObj);
571 PIO_STACK_LOCATION pStack = IoGetCurrentIrpStackLocation(pIrp);
572 PFILE_OBJECT pFileObj = pStack ? pStack->FileObject : NULL;
573 PSUPDRVSESSION pSession = pFileObj ? (PSUPDRVSESSION)pFileObj->FsContext : NULL;
574 NTSTATUS rcNt;
575 unsigned cbOut = 0;
576 int rc = 0;
577 Log2(("VBoxDrvNtInternalDeviceControl(%p,%p): ioctl=%#x pBuf=%p cbIn=%#x cbOut=%#x pSession=%p\n",
578 pDevExt, pIrp, pStack->Parameters.DeviceIoControl.IoControlCode,
579 pIrp->AssociatedIrp.SystemBuffer, pStack->Parameters.DeviceIoControl.InputBufferLength,
580 pStack->Parameters.DeviceIoControl.OutputBufferLength, pSession));
581
582 /* Verify that it's a buffered CTL. */
583 if ((pStack->Parameters.DeviceIoControl.IoControlCode & 0x3) == METHOD_BUFFERED)
584 {
585 /* Verify the pDevExt in the session. */
586 if ( pStack->Parameters.DeviceIoControl.IoControlCode != SUPDRV_IDC_REQ_CONNECT
587 ? VALID_PTR(pSession) && pSession->pDevExt == pDevExt
588 : !pSession
589 )
590 {
591 /* Verify that the size in the request header is correct. */
592 PSUPDRVIDCREQHDR pHdr = (PSUPDRVIDCREQHDR)pIrp->AssociatedIrp.SystemBuffer;
593 if ( pStack->Parameters.DeviceIoControl.InputBufferLength >= sizeof(*pHdr)
594 && pStack->Parameters.DeviceIoControl.InputBufferLength == pHdr->cb
595 && pStack->Parameters.DeviceIoControl.OutputBufferLength == pHdr->cb)
596 {
597 /*
598 * Call the generic code.
599 *
600 * Note! Connect and disconnect requires some extra attention
601 * in order to get the session handling right.
602 */
603 if (pStack->Parameters.DeviceIoControl.IoControlCode == SUPDRV_IDC_REQ_DISCONNECT)
604 pFileObj->FsContext = NULL;
605
606 rc = supdrvIDC(pStack->Parameters.DeviceIoControl.IoControlCode, pDevExt, pSession, pHdr);
607 if (!rc)
608 {
609 if (pStack->Parameters.DeviceIoControl.IoControlCode == SUPDRV_IDC_REQ_CONNECT)
610 pFileObj->FsContext = ((PSUPDRVIDCREQCONNECT)pHdr)->u.Out.pSession;
611
612 rcNt = STATUS_SUCCESS;
613 cbOut = pHdr->cb;
614 }
615 else
616 {
617 rcNt = STATUS_INVALID_PARAMETER;
618 if (pStack->Parameters.DeviceIoControl.IoControlCode == SUPDRV_IDC_REQ_DISCONNECT)
619 pFileObj->FsContext = pSession;
620 }
621 Log2(("VBoxDrvNtInternalDeviceControl: returns %#x/rc=%#x\n", rcNt, rc));
622 }
623 else
624 {
625 Log(("VBoxDrvNtInternalDeviceControl: Mismatching sizes (%#x) - Hdr=%#lx Irp=%#lx/%#lx!\n",
626 pStack->Parameters.DeviceIoControl.IoControlCode,
627 pStack->Parameters.DeviceIoControl.InputBufferLength >= sizeof(*pHdr) ? pHdr->cb : 0,
628 pStack->Parameters.DeviceIoControl.InputBufferLength,
629 pStack->Parameters.DeviceIoControl.OutputBufferLength));
630 rcNt = STATUS_INVALID_PARAMETER;
631 }
632 }
633 else
634 rcNt = STATUS_NOT_SUPPORTED;
635 }
636 else
637 {
638 Log(("VBoxDrvNtInternalDeviceControl: not buffered request (%#x) - not supported\n",
639 pStack->Parameters.DeviceIoControl.IoControlCode));
640 rcNt = STATUS_NOT_SUPPORTED;
641 }
642
643 /* complete the request. */
644 pIrp->IoStatus.Status = rcNt;
645 pIrp->IoStatus.Information = cbOut;
646 IoCompleteRequest(pIrp, IO_NO_INCREMENT);
647 return rcNt;
648}
649
650
651/**
652 * Stub function for functions we don't implemented.
653 *
654 * @returns STATUS_NOT_SUPPORTED
655 * @param pDevObj Device object.
656 * @param pIrp IRP.
657 */
658NTSTATUS _stdcall VBoxDrvNtNotSupportedStub(PDEVICE_OBJECT pDevObj, PIRP pIrp)
659{
660 Log(("VBoxDrvNtNotSupportedStub\n"));
661 NOREF(pDevObj);
662
663 pIrp->IoStatus.Information = 0;
664 pIrp->IoStatus.Status = STATUS_NOT_SUPPORTED;
665 IoCompleteRequest(pIrp, IO_NO_INCREMENT);
666
667 return STATUS_NOT_SUPPORTED;
668}
669
670
671/**
672 * ExRegisterCallback handler for power events
673 *
674 * @param pCallbackContext User supplied parameter (pDevObj)
675 * @param pArgument1 First argument
676 * @param pArgument2 Second argument
677 */
678VOID _stdcall VBoxPowerDispatchCallback(PVOID pCallbackContext, PVOID pArgument1, PVOID pArgument2)
679{
680 PDEVICE_OBJECT pDevObj = (PDEVICE_OBJECT)pCallbackContext;
681
682 Log(("VBoxPowerDispatchCallback: %x %x\n", pArgument1, pArgument2));
683
684 /* Power change imminent? */
685 if ((unsigned)pArgument1 == PO_CB_SYSTEM_STATE_LOCK)
686 {
687 if ((unsigned)pArgument2 == 0)
688 Log(("VBoxPowerDispatchCallback: about to go into suspend mode!\n"));
689 else
690 Log(("VBoxPowerDispatchCallback: resumed!\n"));
691
692 /* Inform any clients that have registered themselves with IPRT. */
693 RTPowerSignalEvent(((unsigned)pArgument2 == 0) ? RTPOWEREVENT_SUSPEND : RTPOWEREVENT_RESUME);
694 }
695}
696
697
698/**
699 * Initializes any OS specific object creator fields.
700 */
701void VBOXCALL supdrvOSObjInitCreator(PSUPDRVOBJ pObj, PSUPDRVSESSION pSession)
702{
703 NOREF(pObj);
704 NOREF(pSession);
705}
706
707
708/**
709 * Checks if the session can access the object.
710 *
711 * @returns true if a decision has been made.
712 * @returns false if the default access policy should be applied.
713 *
714 * @param pObj The object in question.
715 * @param pSession The session wanting to access the object.
716 * @param pszObjName The object name, can be NULL.
717 * @param prc Where to store the result when returning true.
718 */
719bool VBOXCALL supdrvOSObjCanAccess(PSUPDRVOBJ pObj, PSUPDRVSESSION pSession, const char *pszObjName, int *prc)
720{
721 NOREF(pObj);
722 NOREF(pSession);
723 NOREF(pszObjName);
724 NOREF(prc);
725 return false;
726}
727
728
729/**
730 * Force async tsc mode (stub).
731 */
732bool VBOXCALL supdrvOSGetForcedAsyncTscMode(PSUPDRVDEVEXT pDevExt)
733{
734 return false;
735}
736
737
738#define MY_SystemLoadGdiDriverInSystemSpaceInformation 54
739#define MY_SystemUnloadGdiDriverInformation 27
740
741typedef struct MYSYSTEMGDIDRIVERINFO
742{
743 UNICODE_STRING Name; /**< In: image file name. */
744 PVOID ImageAddress; /**< Out: the load address. */
745 PVOID SectionPointer; /**< Out: section object. */
746 PVOID EntryPointer; /**< Out: entry point address. */
747 PVOID ExportSectionPointer; /**< Out: export directory/section. */
748 ULONG ImageLength; /**< Out: SizeOfImage. */
749} MYSYSTEMGDIDRIVERINFO;
750
751extern "C" __declspec(dllimport) NTSTATUS NTAPI ZwSetSystemInformation(ULONG, PVOID, ULONG);
752
753int VBOXCALL supdrvOSLdrOpen(PSUPDRVDEVEXT pDevExt, PSUPDRVLDRIMAGE pImage, const char *pszFilename)
754{
755 pImage->pvNtSectionObj = NULL;
756 pImage->hMemLock = NIL_RTR0MEMOBJ;
757
758#ifdef VBOX_WITHOUT_NATIVE_R0_LOADER
759# ifndef RT_ARCH_X86
760# error "VBOX_WITHOUT_NATIVE_R0_LOADER is only safe on x86."
761# endif
762 NOREF(pDevExt); NOREF(pszFilename); NOREF(pImage);
763 return VERR_NOT_SUPPORTED;
764
765#else
766 /*
767 * Convert the filename from DOS UTF-8 to NT UTF-16.
768 */
769 size_t cwcFilename;
770 int rc = RTStrCalcUtf16LenEx(pszFilename, RTSTR_MAX, &cwcFilename);
771 if (RT_FAILURE(rc))
772 return rc;
773
774 PRTUTF16 pwcsFilename = (PRTUTF16)RTMemTmpAlloc((4 + cwcFilename + 1) * sizeof(RTUTF16));
775 if (!pwcsFilename)
776 return VERR_NO_TMP_MEMORY;
777
778 pwcsFilename[0] = '\\';
779 pwcsFilename[1] = '?';
780 pwcsFilename[2] = '?';
781 pwcsFilename[3] = '\\';
782 PRTUTF16 pwcsTmp = &pwcsFilename[4];
783 rc = RTStrToUtf16Ex(pszFilename, RTSTR_MAX, &pwcsTmp, cwcFilename + 1, NULL);
784 if (RT_SUCCESS(rc))
785 {
786 /*
787 * Try load it.
788 */
789 MYSYSTEMGDIDRIVERINFO Info;
790 RtlInitUnicodeString(&Info.Name, pwcsFilename);
791 Info.ImageAddress = NULL;
792 Info.SectionPointer = NULL;
793 Info.EntryPointer = NULL;
794 Info.ExportSectionPointer = NULL;
795 Info.ImageLength = 0;
796
797 NTSTATUS rcNt = ZwSetSystemInformation(MY_SystemLoadGdiDriverInSystemSpaceInformation, &Info, sizeof(Info));
798 if (NT_SUCCESS(rcNt))
799 {
800 pImage->pvImage = Info.ImageAddress;
801 pImage->pvNtSectionObj = Info.SectionPointer;
802 Log(("ImageAddress=%p SectionPointer=%p ImageLength=%#x cbImageBits=%#x rcNt=%#x '%ls'\n",
803 Info.ImageAddress, Info.SectionPointer, Info.ImageLength, pImage->cbImageBits, rcNt, Info.Name.Buffer));
804# ifdef DEBUG_bird
805 SUPR0Printf("ImageAddress=%p SectionPointer=%p ImageLength=%#x cbImageBits=%#x rcNt=%#x '%ws'\n",
806 Info.ImageAddress, Info.SectionPointer, Info.ImageLength, pImage->cbImageBits, rcNt, Info.Name.Buffer);
807# endif
808 if (pImage->cbImageBits == Info.ImageLength)
809 {
810 /*
811 * Lock down the entire image, just to be on the safe side.
812 */
813 rc = RTR0MemObjLockKernel(&pImage->hMemLock, pImage->pvImage, pImage->cbImageBits, RTMEM_PROT_READ);
814 if (RT_FAILURE(rc))
815 {
816 pImage->hMemLock = NIL_RTR0MEMOBJ;
817 supdrvOSLdrUnload(pDevExt, pImage);
818 }
819 }
820 else
821 {
822 supdrvOSLdrUnload(pDevExt, pImage);
823 rc = VERR_LDR_MISMATCH_NATIVE;
824 }
825 }
826 else
827 {
828 Log(("rcNt=%#x '%ls'\n", rcNt, pwcsFilename));
829 SUPR0Printf("VBoxDrv: rcNt=%x '%ws'\n", rcNt, pwcsFilename);
830 switch (rcNt)
831 {
832 case /* 0xc0000003 */ STATUS_INVALID_INFO_CLASS:
833# ifdef RT_ARCH_AMD64
834 /* Unwind will crash and BSOD, so no fallback here! */
835 rc = VERR_NOT_IMPLEMENTED;
836# else
837 /*
838 * Use the old way of loading the modules.
839 *
840 * Note! We do *NOT* try class 26 because it will probably
841 * not work correctly on terminal servers and such.
842 */
843 rc = VERR_NOT_SUPPORTED;
844# endif
845 break;
846 case /* 0xc0000034 */ STATUS_OBJECT_NAME_NOT_FOUND:
847 rc = VERR_MODULE_NOT_FOUND;
848 break;
849 case /* 0xC0000263 */ STATUS_DRIVER_ENTRYPOINT_NOT_FOUND:
850 rc = VERR_LDR_IMPORTED_SYMBOL_NOT_FOUND;
851 break;
852 case 0xC0000428 /* STATUS_INVALID_IMAGE_HASH */ :
853 rc = VERR_LDR_IMAGE_HASH;
854 break;
855 case 0xC000010E /* STATUS_IMAGE_ALREADY_LOADED */ :
856 Log(("WARNING: see @bugref{4853} for cause of this failure on Windows 7 x64\n"));
857 rc = VERR_ALREADY_LOADED;
858 break;
859 default:
860 rc = VERR_LDR_GENERAL_FAILURE;
861 break;
862 }
863
864 pImage->pvNtSectionObj = NULL;
865 }
866 }
867
868 RTMemTmpFree(pwcsFilename);
869 NOREF(pDevExt);
870 return rc;
871#endif
872}
873
874
875void VBOXCALL supdrvOSLdrNotifyOpened(PSUPDRVDEVEXT pDevExt, PSUPDRVLDRIMAGE pImage)
876{
877 NOREF(pDevExt); NOREF(pImage);
878}
879
880
881int VBOXCALL supdrvOSLdrValidatePointer(PSUPDRVDEVEXT pDevExt, PSUPDRVLDRIMAGE pImage, void *pv, const uint8_t *pbImageBits)
882{
883 NOREF(pDevExt); NOREF(pImage); NOREF(pv); NOREF(pbImageBits);
884 return VINF_SUCCESS;
885}
886
887
888/**
889 * memcmp + log.
890 *
891 * @returns Same as memcmp.
892 * @param pImage The image.
893 * @param pbImageBits The image bits ring-3 uploads.
894 * @param uRva The RVA to start comparing at.
895 * @param cb The number of bytes to compare.
896 */
897static int supdrvNtCompare(PSUPDRVLDRIMAGE pImage, const uint8_t *pbImageBits, uint32_t uRva, uint32_t cb)
898{
899 int iDiff = memcmp((uint8_t const *)pImage->pvImage + uRva, pbImageBits + uRva, cb);
900 if (iDiff)
901 {
902 uint32_t cbLeft = cb;
903 const uint8_t *pbNativeBits = (const uint8_t *)pImage->pvImage;
904 for (size_t off = uRva; cbLeft > 0; off++, cbLeft--)
905 if (pbNativeBits[off] != pbImageBits[off])
906 {
907 char szBytes[128];
908 RTStrPrintf(szBytes, sizeof(szBytes), "native: %.*Rhxs our: %.*Rhxs",
909 RT_MIN(12, cbLeft), &pbNativeBits[off],
910 RT_MIN(12, cbLeft), &pbImageBits[off]);
911 SUPR0Printf("VBoxDrv: Mismatch at %#x of %s: %s\n", off, pImage->szName, szBytes);
912 break;
913 }
914 }
915 return iDiff;
916}
917
918int VBOXCALL supdrvOSLdrLoad(PSUPDRVDEVEXT pDevExt, PSUPDRVLDRIMAGE pImage, const uint8_t *pbImageBits, PSUPLDRLOAD pReq)
919{
920 NOREF(pDevExt); NOREF(pReq);
921 if (pImage->pvNtSectionObj)
922 {
923 /*
924 * Usually, the entire image matches exactly.
925 */
926 if (!memcmp(pImage->pvImage, pbImageBits, pImage->cbImageBits))
927 return VINF_SUCCESS;
928
929 /*
930 * However, on Windows Server 2003 (sp2 x86) both import thunk tables
931 * are fixed up and we typically get a mismatch in the INIT section.
932 *
933 * So, lets see if everything matches when excluding the
934 * OriginalFirstThunk tables. To make life simpler, set the max number
935 * of imports to 16 and just record and sort the locations that needs
936 * to be excluded from the comparison.
937 */
938 IMAGE_NT_HEADERS const *pNtHdrs;
939 pNtHdrs = (IMAGE_NT_HEADERS const *)(pbImageBits
940 + ( *(uint16_t *)pbImageBits == IMAGE_DOS_SIGNATURE
941 ? ((IMAGE_DOS_HEADER const *)pbImageBits)->e_lfanew
942 : 0));
943 if ( pNtHdrs->Signature == IMAGE_NT_SIGNATURE
944 && pNtHdrs->OptionalHeader.Magic == IMAGE_NT_OPTIONAL_HDR_MAGIC
945 && pNtHdrs->OptionalHeader.NumberOfRvaAndSizes > IMAGE_DIRECTORY_ENTRY_IMPORT
946 && pNtHdrs->OptionalHeader.DataDirectory[IMAGE_DIRECTORY_ENTRY_IMPORT].Size >= sizeof(IMAGE_IMPORT_DESCRIPTOR)
947 && pNtHdrs->OptionalHeader.DataDirectory[IMAGE_DIRECTORY_ENTRY_IMPORT].VirtualAddress > sizeof(IMAGE_NT_HEADERS)
948 && pNtHdrs->OptionalHeader.DataDirectory[IMAGE_DIRECTORY_ENTRY_IMPORT].VirtualAddress < pImage->cbImageBits
949 )
950 {
951 struct MyRegion
952 {
953 uint32_t uRva;
954 uint32_t cb;
955 } aExcludeRgns[16];
956 unsigned cExcludeRgns = 0;
957 uint32_t cImpsLeft = pNtHdrs->OptionalHeader.DataDirectory[IMAGE_DIRECTORY_ENTRY_IMPORT].Size
958 / sizeof(IMAGE_IMPORT_DESCRIPTOR);
959 IMAGE_IMPORT_DESCRIPTOR const *pImp;
960 pImp = (IMAGE_IMPORT_DESCRIPTOR const *)(pbImageBits
961 + pNtHdrs->OptionalHeader.DataDirectory[IMAGE_DIRECTORY_ENTRY_IMPORT].VirtualAddress);
962 while ( cImpsLeft-- > 0
963 && cExcludeRgns < RT_ELEMENTS(aExcludeRgns))
964 {
965 uint32_t uRvaThunk = pImp->OriginalFirstThunk;
966 if ( uRvaThunk > sizeof(IMAGE_NT_HEADERS)
967 && uRvaThunk <= pImage->cbImageBits - sizeof(IMAGE_THUNK_DATA)
968 && uRvaThunk != pImp->FirstThunk)
969 {
970 /* Find the size of the thunk table. */
971 IMAGE_THUNK_DATA const *paThunk = (IMAGE_THUNK_DATA const *)(pbImageBits + uRvaThunk);
972 uint32_t cMaxThunks = (pImage->cbImageBits - uRvaThunk) / sizeof(IMAGE_THUNK_DATA);
973 uint32_t cThunks = 0;
974 while (cThunks < cMaxThunks && paThunk[cThunks].u1.Function != 0)
975 cThunks++;
976
977 /* Ordered table insert. */
978 unsigned i = 0;
979 for (; i < cExcludeRgns; i++)
980 if (uRvaThunk < aExcludeRgns[i].uRva)
981 break;
982 if (i != cExcludeRgns)
983 memmove(&aExcludeRgns[i + 1], &aExcludeRgns[i], (cExcludeRgns - i) * sizeof(aExcludeRgns[0]));
984 aExcludeRgns[i].uRva = uRvaThunk;
985 aExcludeRgns[i].cb = cThunks * sizeof(IMAGE_THUNK_DATA);
986 cExcludeRgns++;
987 }
988
989 /* advance */
990 pImp++;
991 }
992
993 /*
994 * Ok, do the comparison.
995 */
996 int iDiff = 0;
997 uint32_t uRvaNext = 0;
998 for (unsigned i = 0; !iDiff && i < cExcludeRgns; i++)
999 {
1000 if (uRvaNext < aExcludeRgns[i].uRva)
1001 iDiff = supdrvNtCompare(pImage, pbImageBits, uRvaNext, aExcludeRgns[i].uRva - uRvaNext);
1002 uRvaNext = aExcludeRgns[i].uRva + aExcludeRgns[i].cb;
1003 }
1004 if (!iDiff && uRvaNext < pImage->cbImageBits)
1005 iDiff = supdrvNtCompare(pImage, pbImageBits, uRvaNext, pImage->cbImageBits - uRvaNext);
1006 if (!iDiff)
1007 return VINF_SUCCESS;
1008 }
1009 else
1010 supdrvNtCompare(pImage, pbImageBits, 0, pImage->cbImageBits);
1011 return VERR_LDR_MISMATCH_NATIVE;
1012 }
1013 return VERR_INTERNAL_ERROR_4;
1014}
1015
1016
1017void VBOXCALL supdrvOSLdrUnload(PSUPDRVDEVEXT pDevExt, PSUPDRVLDRIMAGE pImage)
1018{
1019 if (pImage->pvNtSectionObj)
1020 {
1021 if (pImage->hMemLock != NIL_RTR0MEMOBJ)
1022 {
1023 RTR0MemObjFree(pImage->hMemLock, false /*fFreeMappings*/);
1024 pImage->hMemLock = NIL_RTR0MEMOBJ;
1025 }
1026
1027 NTSTATUS rcNt = ZwSetSystemInformation(MY_SystemUnloadGdiDriverInformation,
1028 &pImage->pvNtSectionObj, sizeof(pImage->pvNtSectionObj));
1029 if (rcNt != STATUS_SUCCESS)
1030 SUPR0Printf("VBoxDrv: failed to unload '%s', rcNt=%#x\n", pImage->szName, rcNt);
1031 pImage->pvNtSectionObj = NULL;
1032 }
1033 NOREF(pDevExt);
1034}
1035
1036
1037/**
1038 * Converts an IPRT error code to an nt status code.
1039 *
1040 * @returns corresponding nt status code.
1041 * @param rc IPRT error status code.
1042 */
1043static NTSTATUS VBoxDrvNtErr2NtStatus(int rc)
1044{
1045 switch (rc)
1046 {
1047 case VINF_SUCCESS: return STATUS_SUCCESS;
1048 case VERR_GENERAL_FAILURE: return STATUS_NOT_SUPPORTED;
1049 case VERR_INVALID_PARAMETER: return STATUS_INVALID_PARAMETER;
1050 case VERR_INVALID_MAGIC: return STATUS_UNKNOWN_REVISION;
1051 case VERR_INVALID_HANDLE: return STATUS_INVALID_HANDLE;
1052 case VERR_INVALID_POINTER: return STATUS_INVALID_ADDRESS;
1053 case VERR_LOCK_FAILED: return STATUS_NOT_LOCKED;
1054 case VERR_ALREADY_LOADED: return STATUS_IMAGE_ALREADY_LOADED;
1055 case VERR_PERMISSION_DENIED: return STATUS_ACCESS_DENIED;
1056 case VERR_VERSION_MISMATCH: return STATUS_REVISION_MISMATCH;
1057 }
1058
1059 return STATUS_UNSUCCESSFUL;
1060}
1061
1062
1063
1064/** @todo use the nocrt stuff? */
1065int VBOXCALL mymemcmp(const void *pv1, const void *pv2, size_t cb)
1066{
1067 const uint8_t *pb1 = (const uint8_t *)pv1;
1068 const uint8_t *pb2 = (const uint8_t *)pv2;
1069 for (; cb > 0; cb--, pb1++, pb2++)
1070 if (*pb1 != *pb2)
1071 return *pb1 - *pb2;
1072 return 0;
1073}
1074
1075
1076#if 0 /* See alternative in SUPDrvA-win.asm */
1077/**
1078 * Alternative version of SUPR0Printf for Windows.
1079 *
1080 * @returns 0.
1081 * @param pszFormat The format string.
1082 */
1083SUPR0DECL(int) SUPR0Printf(const char *pszFormat, ...)
1084{
1085 va_list va;
1086 char szMsg[512];
1087
1088 va_start(va, pszFormat);
1089 size_t cch = RTStrPrintfV(szMsg, sizeof(szMsg) - 1, pszFormat, va);
1090 szMsg[sizeof(szMsg) - 1] = '\0';
1091 va_end(va);
1092
1093 RTLogWriteDebugger(szMsg, cch);
1094 return 0;
1095}
1096#endif
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