VirtualBox

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

Last change on this file since 3968 was 3672, checked in by vboxsync, 18 years ago

RT_OS_* and RT_ARCH_* for Runtime/ and Support/

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 38.3 KB
Line 
1/* $Id: SUPDrv-win.cpp 3672 2007-07-17 12:39:30Z vboxsync $ */
2/** @file
3 * VirtualBox Support Driver - Windows NT specific parts.
4 */
5
6/*
7 * Copyright (C) 2006-2007 innotek GmbH
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 as published by the Free Software Foundation,
13 * in version 2 as it comes in the "COPYING" file of the VirtualBox OSE
14 * distribution. VirtualBox OSE is distributed in the hope that it will
15 * be useful, but WITHOUT ANY WARRANTY of any kind.
16 *
17 * If you received this file as part of a commercial VirtualBox
18 * distribution, then only the terms of your commercial VirtualBox
19 * license agreement apply instead of the previous paragraph.
20 */
21
22
23
24/*******************************************************************************
25* Header Files *
26*******************************************************************************/
27#include "SUPDRV.h"
28#include <excpt.h>
29#include <iprt/assert.h>
30#include <iprt/process.h>
31
32
33/*******************************************************************************
34* Defined Constants And Macros *
35*******************************************************************************/
36/** The support service name. */
37#define SERVICE_NAME "VBoxDrv"
38/** Win32 Device name. */
39#define DEVICE_NAME "\\\\.\\VBoxDrv"
40/** NT Device name. */
41#define DEVICE_NAME_NT L"\\Device\\VBoxDrv"
42/** Win Symlink name. */
43#define DEVICE_NAME_DOS L"\\DosDevices\\VBoxDrv"
44/** The Pool tag (VBox). */
45#define SUPDRV_NT_POOL_TAG 'xoBV'
46
47
48/*******************************************************************************
49* Structures and Typedefs *
50*******************************************************************************/
51#if 0 //def RT_ARCH_AMD64
52typedef struct SUPDRVEXECMEM
53{
54 PMDL pMdl;
55 void *pvMapping;
56 void *pvAllocation;
57} SUPDRVEXECMEM, *PSUPDRVEXECMEM;
58#endif
59
60
61/*******************************************************************************
62* Internal Functions *
63*******************************************************************************/
64static void _stdcall VBoxSupDrvUnload(PDRIVER_OBJECT pDrvObj);
65static NTSTATUS _stdcall VBoxSupDrvCreate(PDEVICE_OBJECT pDevObj, PIRP pIrp);
66static NTSTATUS _stdcall VBoxSupDrvClose(PDEVICE_OBJECT pDevObj, PIRP pIrp);
67static NTSTATUS _stdcall VBoxSupDrvDeviceControl(PDEVICE_OBJECT pDevObj, PIRP pIrp);
68static int VBoxSupDrvDeviceControlSlow(PSUPDRVDEVEXT pDevExt, PSUPDRVSESSION pSession, PIRP pIrp, PIO_STACK_LOCATION pStack);
69static NTSTATUS _stdcall VBoxSupDrvNotSupportedStub(PDEVICE_OBJECT pDevObj, PIRP pIrp);
70static NTSTATUS VBoxSupDrvErr2NtStatus(int rc);
71static NTSTATUS VBoxSupDrvGipInit(PSUPDRVDEVEXT pDevExt);
72static void VBoxSupDrvGipTerm(PSUPDRVDEVEXT pDevExt);
73static void _stdcall VBoxSupDrvGipTimer(IN PKDPC pDpc, IN PVOID pvUser, IN PVOID SystemArgument1, IN PVOID SystemArgument2);
74static void _stdcall VBoxSupDrvGipPerCpuDpc(IN PKDPC pDpc, IN PVOID pvUser, IN PVOID SystemArgument1, IN PVOID SystemArgument2);
75
76
77/*******************************************************************************
78* Exported Functions *
79*******************************************************************************/
80__BEGIN_DECLS
81ULONG _stdcall DriverEntry(PDRIVER_OBJECT pDrvObj, PUNICODE_STRING pRegPath);
82__END_DECLS
83
84
85/**
86 * Driver entry point.
87 *
88 * @returns appropriate status code.
89 * @param pDrvObj Pointer to driver object.
90 * @param pRegPath Registry base path.
91 */
92ULONG _stdcall DriverEntry(PDRIVER_OBJECT pDrvObj, PUNICODE_STRING pRegPath)
93{
94 NTSTATUS rc;
95 dprintf(("VBoxDrv::DriverEntry\n"));
96
97 /*
98 * Create device.
99 * (That means creating a device object and a symbolic link so the DOS
100 * subsystems (OS/2, win32, ++) can access the device.)
101 */
102 UNICODE_STRING DevName;
103 RtlInitUnicodeString(&DevName, DEVICE_NAME_NT);
104 PDEVICE_OBJECT pDevObj;
105 rc = IoCreateDevice(pDrvObj, sizeof(SUPDRVDEVEXT), &DevName, FILE_DEVICE_UNKNOWN, 0, FALSE, &pDevObj);
106 if (NT_SUCCESS(rc))
107 {
108 UNICODE_STRING DosName;
109 RtlInitUnicodeString(&DosName, DEVICE_NAME_DOS);
110 rc = IoCreateSymbolicLink(&DosName, &DevName);
111 if (NT_SUCCESS(rc))
112 {
113 /*
114 * Initialize the device extension.
115 */
116 PSUPDRVDEVEXT pDevExt = (PSUPDRVDEVEXT)pDevObj->DeviceExtension;
117 memset(pDevExt, 0, sizeof(*pDevExt));
118 int vrc = supdrvInitDevExt(pDevExt);
119 if (!vrc)
120 {
121 /*
122 * Inititalize the GIP.
123 */
124 rc = VBoxSupDrvGipInit(pDevExt);
125 if (NT_SUCCESS(rc))
126 {
127 /*
128 * Setup the driver entry points in pDrvObj.
129 */
130 pDrvObj->DriverUnload = VBoxSupDrvUnload;
131 pDrvObj->MajorFunction[IRP_MJ_CREATE] = VBoxSupDrvCreate;
132 pDrvObj->MajorFunction[IRP_MJ_CLOSE] = VBoxSupDrvClose;
133 pDrvObj->MajorFunction[IRP_MJ_DEVICE_CONTROL] = VBoxSupDrvDeviceControl;
134 pDrvObj->MajorFunction[IRP_MJ_READ] = VBoxSupDrvNotSupportedStub;
135 pDrvObj->MajorFunction[IRP_MJ_WRITE] = VBoxSupDrvNotSupportedStub;
136 /* more? */
137 dprintf(("VBoxDrv::DriverEntry returning STATUS_SUCCESS\n"));
138 return STATUS_SUCCESS;
139 }
140 dprintf(("VBoxSupDrvGipInit failed with rc=%#x!\n", rc));
141
142 supdrvDeleteDevExt(pDevExt);
143 }
144 else
145 {
146 dprintf(("supdrvInitDevExit failed with vrc=%d!\n", vrc));
147 rc = VBoxSupDrvErr2NtStatus(vrc);
148 }
149
150 IoDeleteSymbolicLink(&DosName);
151 }
152 else
153 dprintf(("IoCreateSymbolicLink failed with rc=%#x!\n", rc));
154
155 IoDeleteDevice(pDevObj);
156 }
157 else
158 dprintf(("IoCreateDevice failed with rc=%#x!\n", rc));
159
160 if (NT_SUCCESS(rc))
161 rc = STATUS_INVALID_PARAMETER;
162 dprintf(("VBoxDrv::DriverEntry returning %#x\n", rc));
163 return rc;
164}
165
166
167/**
168 * Unload the driver.
169 *
170 * @param pDrvObj Driver object.
171 */
172void _stdcall VBoxSupDrvUnload(PDRIVER_OBJECT pDrvObj)
173{
174 dprintf(("VBoxSupDrvUnload\n"));
175 PSUPDRVDEVEXT pDevExt = (PSUPDRVDEVEXT)pDrvObj->DeviceObject->DeviceExtension;
176
177 /*
178 * We ASSUME that it's not possible to unload a driver with open handles.
179 * Start by deleting the symbolic link
180 */
181 UNICODE_STRING DosName;
182 RtlInitUnicodeString(&DosName, DEVICE_NAME_DOS);
183 NTSTATUS rc = IoDeleteSymbolicLink(&DosName);
184
185 /*
186 * Terminate the GIP page and delete the device extension.
187 */
188 VBoxSupDrvGipTerm(pDevExt);
189 supdrvDeleteDevExt(pDevExt);
190 IoDeleteDevice(pDrvObj->DeviceObject);
191}
192
193
194/**
195 * Create (i.e. Open) file entry point.
196 *
197 * @param pDevObj Device object.
198 * @param pIrp Request packet.
199 */
200NTSTATUS _stdcall VBoxSupDrvCreate(PDEVICE_OBJECT pDevObj, PIRP pIrp)
201{
202 dprintf(("VBoxSupDrvCreate\n"));
203 PIO_STACK_LOCATION pStack = IoGetCurrentIrpStackLocation(pIrp);
204 PFILE_OBJECT pFileObj = pStack->FileObject;
205 PSUPDRVDEVEXT pDevExt = (PSUPDRVDEVEXT)pDevObj->DeviceExtension;
206
207 /*
208 * We are not remotely similar to a directory...
209 * (But this is possible.)
210 */
211 if (pStack->Parameters.Create.Options & FILE_DIRECTORY_FILE)
212 {
213 pIrp->IoStatus.Status = STATUS_NOT_A_DIRECTORY;
214 pIrp->IoStatus.Information = 0;
215 IoCompleteRequest(pIrp, IO_NO_INCREMENT);
216 return STATUS_NOT_A_DIRECTORY;
217 }
218
219 /*
220 * Call common code for the rest.
221 */
222 pFileObj->FsContext = NULL;
223 PSUPDRVSESSION pSession;
224 int rc = supdrvCreateSession(pDevExt, &pSession);
225 if (!rc)
226 {
227 pSession->Uid = NIL_RTUID;
228 pSession->Gid = NIL_RTGID;
229 pSession->Process = RTProcSelf();
230 pSession->R0Process = RTR0ProcHandleSelf();
231 pFileObj->FsContext = pSession;
232 }
233
234 NTSTATUS rcNt = pIrp->IoStatus.Status = VBoxSupDrvErr2NtStatus(rc);
235 pIrp->IoStatus.Information = 0;
236 IoCompleteRequest(pIrp, IO_NO_INCREMENT);
237
238 return rcNt;
239}
240
241
242/**
243 * Close file entry point.
244 *
245 * @param pDevObj Device object.
246 * @param pIrp Request packet.
247 */
248NTSTATUS _stdcall VBoxSupDrvClose(PDEVICE_OBJECT pDevObj, PIRP pIrp)
249{
250 PSUPDRVDEVEXT pDevExt = (PSUPDRVDEVEXT)pDevObj->DeviceExtension;
251 PIO_STACK_LOCATION pStack = IoGetCurrentIrpStackLocation(pIrp);
252 PFILE_OBJECT pFileObj = pStack->FileObject;
253 dprintf(("VBoxSupDrvClose: pDevExt=%p pFileObj=%p pSession=%p\n",
254 pDevExt, pFileObj, pFileObj->FsContext));
255 supdrvCloseSession(pDevExt, (PSUPDRVSESSION)pFileObj->FsContext);
256 pFileObj->FsContext = NULL;
257 pIrp->IoStatus.Information = 0;
258 pIrp->IoStatus.Status = STATUS_SUCCESS;
259 IoCompleteRequest(pIrp, IO_NO_INCREMENT);
260
261 return STATUS_SUCCESS;
262}
263
264
265/**
266 * Device I/O Control entry point.
267 *
268 * @param pDevObj Device object.
269 * @param pIrp Request packet.
270 */
271NTSTATUS _stdcall VBoxSupDrvDeviceControl(PDEVICE_OBJECT pDevObj, PIRP pIrp)
272{
273 PSUPDRVDEVEXT pDevExt = (PSUPDRVDEVEXT)pDevObj->DeviceExtension;
274 PIO_STACK_LOCATION pStack = IoGetCurrentIrpStackLocation(pIrp);
275 PSUPDRVSESSION pSession = (PSUPDRVSESSION)pStack->FileObject->FsContext;
276
277#ifdef VBOX_WITHOUT_IDT_PATCHING
278 /*
279 * Deal with the two high-speed IOCtl that takes it's arguments from
280 * the session and iCmd, and only returns a VBox status code.
281 */
282 ULONG ulCmd = pStack->Parameters.DeviceIoControl.IoControlCode;
283 if ( ulCmd == SUP_IOCTL_FAST_DO_RAW_RUN
284 || ulCmd == SUP_IOCTL_FAST_DO_HWACC_RUN
285 || ulCmd == SUP_IOCTL_FAST_DO_NOP)
286 {
287 int rc = supdrvIOCtlFast(ulCmd, pDevExt, pSession);
288
289 /* Complete the I/O request. */
290 NTSTATUS rcNt = pIrp->IoStatus.Status = STATUS_SUCCESS;
291 pIrp->IoStatus.Information = sizeof(rc);
292 __try
293 {
294 *(int *)pIrp->UserBuffer = rc;
295 }
296 __except(EXCEPTION_EXECUTE_HANDLER)
297 {
298 rcNt = pIrp->IoStatus.Status = GetExceptionCode();
299 dprintf(("VBoxSupDrvDeviceContorl: Exception Code %#x\n", rcNt));
300 }
301 IoCompleteRequest(pIrp, IO_NO_INCREMENT);
302 return rcNt;
303 }
304#endif /* VBOX_WITHOUT_IDT_PATCHING */
305
306 return VBoxSupDrvDeviceControlSlow(pDevExt, pSession, pIrp, pStack);
307}
308
309
310/**
311 * Worker for VBoxSupDrvDeviceControl that takes the slow IOCtl functions.
312 *
313 * @returns NT status code.
314 *
315 * @param pDevObj Device object.
316 * @param pSession The session.
317 * @param pIrp Request packet.
318 * @param pStack The stack location containing the DeviceControl parameters.
319 */
320static int VBoxSupDrvDeviceControlSlow(PSUPDRVDEVEXT pDevExt, PSUPDRVSESSION pSession, PIRP pIrp, PIO_STACK_LOCATION pStack)
321{
322 NTSTATUS rcNt = STATUS_NOT_SUPPORTED;
323 unsigned cbOut = 0;
324 int rc = 0;
325 dprintf2(("VBoxSupDrvDeviceControlSlow(%p,%p): ioctl=%#x pBuf=%p cbIn=%#x cbOut=%#x pSession=%p\n",
326 pDevObj, pIrp, pStack->Parameters.DeviceIoControl.IoControlCode,
327 pBuf, pStack->Parameters.DeviceIoControl.InputBufferLength,
328 pStack->Parameters.DeviceIoControl.OutputBufferLength, pSession));
329
330#ifdef RT_ARCH_AMD64
331 /* Don't allow 32-bit processes to do any I/O controls. */
332 if (!IoIs32bitProcess(pIrp))
333#endif
334 {
335 /* Verify that it's a buffered CTL. */
336 if ((pStack->Parameters.DeviceIoControl.IoControlCode & 0x3) == METHOD_BUFFERED)
337 {
338 char *pBuf = (char *)pIrp->AssociatedIrp.SystemBuffer;
339
340 /*
341 * Do the job.
342 */
343 rc = supdrvIOCtl(pStack->Parameters.DeviceIoControl.IoControlCode, pDevExt, pSession,
344 pBuf, pStack->Parameters.DeviceIoControl.InputBufferLength,
345 pBuf, pStack->Parameters.DeviceIoControl.OutputBufferLength,
346 &cbOut);
347 rcNt = VBoxSupDrvErr2NtStatus(rc);
348
349 /* sanity check. */
350 AssertMsg(cbOut <= pStack->Parameters.DeviceIoControl.OutputBufferLength,
351 ("cbOut is too large! cbOut=%d max=%d! ioctl=%#x\n",
352 cbOut, pStack->Parameters.DeviceIoControl.OutputBufferLength,
353 pStack->Parameters.DeviceIoControl.IoControlCode));
354 if (cbOut > pStack->Parameters.DeviceIoControl.OutputBufferLength)
355 cbOut = pStack->Parameters.DeviceIoControl.OutputBufferLength;
356 dprintf2(("VBoxSupDrvDeviceControlSlow: returns %#x cbOut=%d rc=%#x\n", rcNt, cbOut, rc));
357 }
358 else
359 dprintf(("VBoxSupDrvDeviceControlSlow: not buffered request (%#x) - not supported\n",
360 pStack->Parameters.DeviceIoControl.IoControlCode));
361 }
362#ifdef RT_ARCH_AMD64
363 else
364 dprintf(("VBoxSupDrvDeviceControlSlow: WOW64 req - not supported\n"));
365#endif
366
367 /* complete the request. */
368 pIrp->IoStatus.Status = rcNt;
369 pIrp->IoStatus.Information = NT_SUCCESS(rcNt) ? cbOut : rc; /* does this rc passing actually work?!? */
370 IoCompleteRequest(pIrp, IO_NO_INCREMENT);
371 return rcNt;
372}
373
374
375/**
376 * Stub function for functions we don't implemented.
377 *
378 * @returns STATUS_NOT_SUPPORTED
379 * @param pDevObj Device object.
380 * @param pIrp IRP.
381 */
382NTSTATUS _stdcall VBoxSupDrvNotSupportedStub(PDEVICE_OBJECT pDevObj, PIRP pIrp)
383{
384 dprintf(("VBoxSupDrvNotSupportedStub\n"));
385 pDevObj = pDevObj;
386
387 pIrp->IoStatus.Information = 0;
388 pIrp->IoStatus.Status = STATUS_NOT_SUPPORTED;
389 IoCompleteRequest(pIrp, IO_NO_INCREMENT);
390
391 return STATUS_NOT_SUPPORTED;
392}
393
394
395/**
396 * Initializes any OS specific object creator fields.
397 */
398void VBOXCALL supdrvOSObjInitCreator(PSUPDRVOBJ pObj, PSUPDRVSESSION pSession)
399{
400 NOREF(pObj);
401 NOREF(pSession);
402}
403
404
405/**
406 * Checks if the session can access the object.
407 *
408 * @returns true if a decision has been made.
409 * @returns false if the default access policy should be applied.
410 *
411 * @param pObj The object in question.
412 * @param pSession The session wanting to access the object.
413 * @param pszObjName The object name, can be NULL.
414 * @param prc Where to store the result when returning true.
415 */
416bool VBOXCALL supdrvOSObjCanAccess(PSUPDRVOBJ pObj, PSUPDRVSESSION pSession, const char *pszObjName, int *prc)
417{
418 NOREF(pObj);
419 NOREF(pSession);
420 NOREF(pszObjName);
421 NOREF(prc);
422 return false;
423}
424
425
426/**
427 * OS Specific code for locking down memory.
428 *
429 * @returns 0 on success.
430 * @returns SUPDRV_ERR_* on failure.
431 * @param pMem Pointer to memory.
432 * This is not linked in anywhere.
433 * @param paPages Array which should be filled with the address of the physical pages.
434 */
435int VBOXCALL supdrvOSLockMemOne(PSUPDRVMEMREF pMem, PSUPPAGE paPages)
436{
437 /* paranoia */
438 if (!pMem->cb)
439 {
440 AssertMsgFailed(("Fool! No memory to lock!\n"));
441 return SUPDRV_ERR_INVALID_PARAM;
442 }
443 Assert(RT_ALIGN(pMem->cb, PAGE_SIZE) == pMem->cb);
444
445 /*
446 * Calc the number of MDLs we need to allocate.
447 */
448 unsigned cMdls = pMem->cb / MAX_LOCK_MEM_SIZE;
449 if ((pMem->cb % MAX_LOCK_MEM_SIZE) > 0)
450 cMdls++;
451
452 /*
453 * Allocate memory for the MDL pointer array.
454 */
455 pMem->u.locked.papMdl = (PMDL *)ExAllocatePoolWithTag(NonPagedPool, sizeof(*pMem->u.locked.papMdl) * cMdls, SUPDRV_NT_POOL_TAG);
456 if (!pMem->u.locked.papMdl)
457 {
458 AssertMsgFailed(("shit, couldn't allocated %d bytes for the mdl pointer array!\n", sizeof(*pMem->u.locked.papMdl) * cMdls));
459 return SUPDRV_ERR_NO_MEMORY;
460 }
461
462 /*
463 * Loop locking down the sub parts of the memory.
464 */
465 PSUPPAGE pPage = paPages;
466 unsigned cbTotal = 0;
467 uint8_t *pu8 = (uint8_t *)pMem->pvR3;
468 for (unsigned i = 0; i < cMdls; i++)
469 {
470 /*
471 * Calc the number of bytes to lock this time.
472 */
473 unsigned cbCur = pMem->cb - cbTotal;
474 if (cbCur > MAX_LOCK_MEM_SIZE)
475 cbCur = MAX_LOCK_MEM_SIZE;
476
477 if (cbCur == 0)
478 AssertMsgFailed(("cbCur: 0!\n"));
479
480 /*
481 * Allocate pMdl.
482 */
483 PMDL pMdl = IoAllocateMdl(pu8, cbCur, FALSE, FALSE, NULL);
484 if (!pMdl)
485 {
486 AssertMsgFailed(("Ops! IoAllocateMdl failed for pu8=%p and cb=%d\n", pu8, cbCur));
487 return SUPDRV_ERR_NO_MEMORY;
488 }
489
490 /*
491 * Lock the pages.
492 */
493 NTSTATUS rc = STATUS_SUCCESS;
494 __try
495 {
496 MmProbeAndLockPages(pMdl, UserMode, IoModifyAccess);
497 }
498 __except(EXCEPTION_EXECUTE_HANDLER)
499 {
500 rc = GetExceptionCode();
501 dprintf(("supdrvOSLockMemOne: Exception Code %#x\n", rc));
502 }
503
504 if (!NT_SUCCESS(rc))
505 {
506 /*
507 * Cleanup and fail.
508 */
509 IoFreeMdl(pMdl);
510 while (i-- > 0)
511 {
512 MmUnlockPages(pMem->u.locked.papMdl[i]);
513 IoFreeMdl(pMem->u.locked.papMdl[i]);
514 }
515 ExFreePool(pMem->u.locked.papMdl);
516 pMem->u.locked.papMdl = NULL;
517 return SUPDRV_ERR_LOCK_FAILED;
518 }
519
520 /*
521 * Add MDL to array and update the pages.
522 */
523 pMem->u.locked.papMdl[i] = pMdl;
524
525 const uintptr_t *pauPFNs = (uintptr_t *)(pMdl + 1); /* ASSUMES ULONG_PTR == uintptr_t, NTDDK4 doesn't have ULONG_PTR. */
526 for (unsigned iPage = 0, cPages = cbCur >> PAGE_SHIFT; iPage < cPages; iPage++)
527 {
528 pPage->Phys = (RTHCPHYS)pauPFNs[iPage] << PAGE_SHIFT;
529 pPage->uReserved = 0;
530 pPage++;
531 }
532
533 /* next */
534 cbTotal += cbCur;
535 pu8 += cbCur;
536 }
537
538 /*
539 * Finish structure and return succesfully.
540 */
541 pMem->u.locked.cMdls = cMdls;
542
543 dprintf2(("supdrvOSLockMemOne: pvR3=%p cb=%d cMdls=%d\n",
544 pMem->pvR3, pMem->cb, cMdls));
545 return 0;
546}
547
548
549/**
550 * Unlocks the memory pointed to by pv.
551 *
552 * @param pv Memory to unlock.
553 * @param cb Size of the memory (debug).
554 */
555void VBOXCALL supdrvOSUnlockMemOne(PSUPDRVMEMREF pMem)
556{
557 dprintf2(("supdrvOSUnlockMemOne: pvR3=%p cb=%d cMdl=%p papMdl=%p\n",
558 pMem->pvR3, pMem->cb, pMem->u.locked.cMdls, pMem->u.locked.papMdl));
559
560 for (unsigned i = 0; i < pMem->u.locked.cMdls; i++)
561 {
562 MmUnlockPages(pMem->u.locked.papMdl[i]);
563 IoFreeMdl(pMem->u.locked.papMdl[i]);
564 }
565
566 ExFreePool(pMem->u.locked.papMdl);
567 pMem->u.locked.papMdl = NULL;
568}
569
570
571/**
572 * OS Specific code for allocating page aligned memory with continuous fixed
573 * physical paged backing.
574 *
575 * @returns 0 on success.
576 * @returns SUPDRV_ERR_* on failure.
577 * @param pMem Memory reference record of the memory to be allocated.
578 * (This is not linked in anywhere.)
579 * @param ppvR0 Where to store the virtual address of the ring-0 mapping. (optional)
580 * @param ppvR3 Where to store the virtual address of the ring-3 mapping.
581 * @param pHCPhys Where to store the physical address.
582 */
583int VBOXCALL supdrvOSContAllocOne(PSUPDRVMEMREF pMem, PRTR0PTR ppvR0, PRTR3PTR ppvR3, PRTHCPHYS pHCPhys)
584{
585 Assert(ppvR3);
586 Assert(pHCPhys);
587
588 /*
589 * Try allocate the memory.
590 */
591 PHYSICAL_ADDRESS Phys;
592 Phys.HighPart = 0;
593 Phys.LowPart = ~0;
594 unsigned cbAligned = RT_ALIGN(pMem->cb, PAGE_SIZE);
595 pMem->pvR0 = MmAllocateContiguousMemory(cbAligned, Phys);
596 if (!pMem->pvR0)
597 return SUPDRV_ERR_NO_MEMORY;
598
599 /*
600 * Map into user space.
601 */
602 int rc = SUPDRV_ERR_NO_MEMORY;
603 pMem->u.cont.pMdl = IoAllocateMdl(pMem->pvR0, cbAligned, FALSE, FALSE, NULL);
604 if (pMem->u.cont.pMdl)
605 {
606 MmBuildMdlForNonPagedPool(pMem->u.cont.pMdl);
607 __try
608 {
609 pMem->pvR3 = (RTR3PTR)MmMapLockedPagesSpecifyCache(pMem->u.cont.pMdl, UserMode, MmCached, NULL, FALSE, NormalPagePriority);
610 if (pMem->pvR3)
611 {
612 /*
613 * Done, setup pMem and return values.
614 */
615#ifdef RT_ARCH_AMD64
616 MmProtectMdlSystemAddress(pMem->u.cont.pMdl, PAGE_EXECUTE_READWRITE);
617#endif
618 *ppvR3 = pMem->pvR3;
619 if (ppvR0)
620 *ppvR0 = pMem->pvR0;
621 const uintptr_t *pauPFNs = (const uintptr_t *)(pMem->u.cont.pMdl + 1); /* ASSUMES ULONG_PTR == uintptr_t, NTDDK4 doesn't have ULONG_PTR. */
622 *pHCPhys = (RTHCPHYS)pauPFNs[0] << PAGE_SHIFT;
623 dprintf2(("supdrvOSContAllocOne: pvR0=%p pvR3=%p cb=%d pMdl=%p *pHCPhys=%VHp\n",
624 pMem->pvR0, pMem->pvR3, pMem->cb, pMem->u.mem.pMdl, *pHCPhys));
625 return 0;
626 }
627 }
628 __except(EXCEPTION_EXECUTE_HANDLER)
629 {
630 NTSTATUS rc = GetExceptionCode();
631 dprintf(("supdrvOSContAllocOne: Exception Code %#x\n", rc));
632 }
633 IoFreeMdl(pMem->u.cont.pMdl);
634 rc = SUPDRV_ERR_LOCK_FAILED;
635 }
636 MmFreeContiguousMemory(pMem->pvR0);
637 pMem->pvR0 = NULL;
638 return rc;
639}
640
641
642/**
643 * Frees contiguous memory.
644 *
645 * @param pMem Memory reference record of the memory to be freed.
646 */
647void VBOXCALL supdrvOSContFreeOne(PSUPDRVMEMREF pMem)
648{
649 __try
650 {
651 dprintf2(("supdrvOSContFreeOne: pvR0=%p pvR3=%p cb=%d pMdl=%p\n",
652 pMem->pvR0, pMem->pvR3, pMem->cb, pMem->u.cont.pMdl));
653 if (pMem->pvR3)
654 {
655 MmUnmapLockedPages((void *)pMem->pvR3, pMem->u.cont.pMdl);
656 dprintf2(("MmUnmapLockedPages ok!\n"));
657 pMem->pvR3 = NULL;
658 }
659
660 IoFreeMdl(pMem->u.cont.pMdl);
661 dprintf2(("IoFreeMdl ok!\n"));
662 pMem->u.cont.pMdl = NULL;
663
664 MmFreeContiguousMemory(pMem->pvR0);
665 dprintf2(("MmFreeContiguousMemory ok!\n"));
666 pMem->pvR0 = NULL;
667 }
668 __except(EXCEPTION_EXECUTE_HANDLER)
669 {
670 NTSTATUS rc = GetExceptionCode();
671 dprintf(("supdrvOSContFreeOne: Exception Code %#x\n", rc));
672 }
673}
674
675
676/**
677 * Allocates memory which mapped into both kernel and user space.
678 * The returned memory is page aligned and so is the allocation.
679 *
680 * @returns 0 on success.
681 * @returns SUPDRV_ERR_* on failure.
682 * @param pMem Memory reference record of the memory to be allocated.
683 * (This is not linked in anywhere.)
684 * @param ppvR0 Where to store the address of the Ring-0 mapping.
685 * @param ppvR3 Where to store the address of the Ring-3 mapping.
686 */
687int VBOXCALL supdrvOSMemAllocOne(PSUPDRVMEMREF pMem, PRTR0PTR ppvR0, PRTR3PTR ppvR3)
688{
689 Assert(ppvR0);
690 Assert(ppvR3);
691
692 /*
693 * Try allocate the memory.
694 */
695 unsigned cbAligned = RT_ALIGN(RT_MAX(pMem->cb, PAGE_SIZE * 2), PAGE_SIZE);
696 pMem->pvR0 = ExAllocatePoolWithTag(NonPagedPool, cbAligned, SUPDRV_NT_POOL_TAG);
697 if (!pMem->pvR0)
698 return SUPDRV_ERR_NO_MEMORY;
699
700 /*
701 * Map into user space.
702 */
703 int rc = SUPDRV_ERR_NO_MEMORY;
704 pMem->u.mem.pMdl = IoAllocateMdl(pMem->pvR0, cbAligned, FALSE, FALSE, NULL);
705 if (pMem->u.mem.pMdl)
706 {
707 MmBuildMdlForNonPagedPool(pMem->u.mem.pMdl);
708 __try
709 {
710 pMem->pvR3 = (RTR3PTR)MmMapLockedPagesSpecifyCache(pMem->u.mem.pMdl, UserMode, MmCached, NULL, FALSE, NormalPagePriority);
711 if (pMem->pvR3)
712 {
713 /*
714 * Done, setup pMem and return values.
715 */
716 *ppvR3 = pMem->pvR3;
717 *ppvR0 = pMem->pvR0;
718 dprintf2(("supdrvOSContAllocOne: pvR0=%p pvR3=%p cb=%d pMdl=%p\n",
719 pMem->pvR0, pMem->pvR3, pMem->cb, pMem->u.mem.pMdl));
720 return 0;
721 }
722 }
723 __except(EXCEPTION_EXECUTE_HANDLER)
724 {
725 NTSTATUS rc = GetExceptionCode();
726 dprintf(("supdrvOSContAllocOne: Exception Code %#x\n", rc));
727 }
728 rc = SUPDRV_ERR_LOCK_FAILED;
729
730 IoFreeMdl(pMem->u.mem.pMdl);
731 pMem->u.mem.pMdl = NULL;
732 pMem->pvR3 = NULL;
733 }
734
735 MmFreeContiguousMemory(pMem->pvR0);
736 pMem->pvR0 = NULL;
737 return rc;
738}
739
740
741/**
742 * Get the physical addresses of the pages in the allocation.
743 * This is called while inside bundle the spinlock.
744 *
745 * @param pMem Memory reference record of the memory.
746 * @param paPages Where to store the page addresses.
747 */
748void VBOXCALL supdrvOSMemGetPages(PSUPDRVMEMREF pMem, PSUPPAGE paPages)
749{
750 const unsigned cPages = RT_ALIGN(pMem->cb, PAGE_SIZE) >> PAGE_SHIFT;
751 const uintptr_t *pauPFNs = (const uintptr_t *)(pMem->u.mem.pMdl + 1); /* ASSUMES ULONG_PTR == uintptr_t, NTDDK doesn't have ULONG_PTR. */
752 for (unsigned iPage = 0; iPage < cPages; iPage++)
753 {
754 paPages[iPage].Phys = (RTHCPHYS)pauPFNs[iPage] << PAGE_SHIFT;
755 paPages[iPage].uReserved = 0;
756 }
757}
758
759
760/**
761 * Frees memory allocated by supdrvOSMemAllocOne().
762 *
763 * @param pMem Memory reference record of the memory to be free.
764 */
765void VBOXCALL supdrvOSMemFreeOne(PSUPDRVMEMREF pMem)
766{
767 __try
768 {
769 dprintf2(("supdrvOSContFreeOne: pvR0=%p pvR3=%p cb=%d pMdl=%p\n",
770 pMem->pvR0, pMem->pvR3, pMem->cb, pMem->u.mem.pMdl));
771 if (pMem->pvR3)
772 {
773 MmUnmapLockedPages((void *)pMem->pvR3, pMem->u.mem.pMdl);
774 pMem->pvR3 = NULL;
775 dprintf2(("MmUnmapLockedPages ok!\n"));
776 }
777
778 IoFreeMdl(pMem->u.mem.pMdl);
779 pMem->u.mem.pMdl = NULL;
780 dprintf2(("IoFreeMdl ok!\n"));
781
782 ExFreePool(pMem->pvR0);
783 pMem->pvR0 = NULL;
784 dprintf2(("MmFreeContiguousMemory ok!\n"));
785 }
786 __except(EXCEPTION_EXECUTE_HANDLER)
787 {
788 NTSTATUS rc = GetExceptionCode();
789 dprintf(("supdrvOSContFreeOne: Exception Code %#x\n", rc));
790 }
791}
792
793
794/**
795 * Gets the monotone timestamp (nano seconds).
796 * @returns NanoTS.
797 */
798static inline uint64_t supdrvOSMonotime(void)
799{
800 return (uint64_t)KeQueryInterruptTime() * 100;
801}
802
803
804/**
805 * Initializes the GIP.
806 *
807 * @returns NT status code.
808 * @param pDevExt Instance data. GIP stuff may be updated.
809 */
810static NTSTATUS VBoxSupDrvGipInit(PSUPDRVDEVEXT pDevExt)
811{
812 dprintf2(("VBoxSupDrvTermGip:\n"));
813
814 /*
815 * Try allocate the memory.
816 * Make sure it's below 4GB for 32-bit GC support
817 */
818 NTSTATUS rc;
819 PHYSICAL_ADDRESS Phys;
820 Phys.HighPart = 0;
821 Phys.LowPart = ~0;
822 PSUPGLOBALINFOPAGE pGip = (PSUPGLOBALINFOPAGE)MmAllocateContiguousMemory(PAGE_SIZE, Phys);
823 if (pGip)
824 {
825 if (!((uintptr_t)pGip & (PAGE_SIZE - 1)))
826 {
827 pDevExt->pGipMdl = IoAllocateMdl(pGip, PAGE_SIZE, FALSE, FALSE, NULL);
828 if (pDevExt->pGipMdl)
829 {
830 MmBuildMdlForNonPagedPool(pDevExt->pGipMdl);
831
832 /*
833 * Figure the timer interval and frequency.
834 * It turns out trying 1023Hz doesn't work. So, we'll set the max Hz at 128 for now.
835 */
836 ExSetTimerResolution(156250, TRUE);
837 ULONG ulClockIntervalActual = ExSetTimerResolution(0, FALSE);
838 ULONG ulClockInterval = RT_MAX(ulClockIntervalActual, 78125); /* 1/128 */
839 ULONG ulClockFreq = 10000000 / ulClockInterval;
840 pDevExt->ulGipTimerInterval = ulClockInterval / 10000; /* ms */
841
842 /*
843 * Call common initialization routine.
844 */
845 Phys = MmGetPhysicalAddress(pGip); /* could perhaps use the Mdl, not that it looks much better */
846 supdrvGipInit(pDevExt, pGip, (RTHCPHYS)Phys.QuadPart, supdrvOSMonotime(), ulClockFreq);
847
848 /*
849 * Initialize the timer.
850 */
851 KeInitializeTimerEx(&pDevExt->GipTimer, SynchronizationTimer);
852 KeInitializeDpc(&pDevExt->GipDpc, VBoxSupDrvGipTimer, pDevExt);
853
854 /*
855 * Initialize the DPCs we're using to update the per-cpu GIP data.
856 * (Not sure if we need to be this careful with KeSetTargetProcessorDpc...)
857 */
858 UNICODE_STRING RoutineName;
859 RtlInitUnicodeString(&RoutineName, L"KeSetTargetProcessorDpc");
860 VOID (*pfnKeSetTargetProcessorDpc)(IN PRKDPC, IN CCHAR) = (VOID (*)(IN PRKDPC, IN CCHAR))MmGetSystemRoutineAddress(&RoutineName);
861
862 for (unsigned i = 0; i < RT_ELEMENTS(pDevExt->aGipCpuDpcs); i++)
863 {
864 KeInitializeDpc(&pDevExt->aGipCpuDpcs[i], VBoxSupDrvGipPerCpuDpc, pGip);
865 KeSetImportanceDpc(&pDevExt->aGipCpuDpcs[i], HighImportance);
866 if (pfnKeSetTargetProcessorDpc)
867 pfnKeSetTargetProcessorDpc(&pDevExt->aGipCpuDpcs[i], i);
868 }
869
870 dprintf(("VBoxSupDrvGipInit: ulClockFreq=%ld ulClockInterval=%ld ulClockIntervalActual=%ld Phys=%x%08x\n",
871 ulClockFreq, ulClockInterval, ulClockIntervalActual, Phys.HighPart, Phys.LowPart));
872 return STATUS_SUCCESS;
873 }
874
875 dprintf(("VBoxSupDrvInitGip: IoAllocateMdl failed for %p/PAGE_SIZE\n", pGip));
876 rc = STATUS_NO_MEMORY;
877 }
878 else
879 {
880 dprintf(("VBoxSupDrvInitGip: GIP memory is not page aligned! pGip=%p\n", pGip));
881 rc = STATUS_INVALID_ADDRESS;
882 }
883 MmFreeContiguousMemory(pGip);
884 }
885 else
886 {
887 dprintf(("VBoxSupDrvInitGip: no cont memory.\n"));
888 rc = STATUS_NO_MEMORY;
889 }
890 return rc;
891}
892
893
894/**
895 * Terminates the GIP.
896 *
897 * @returns negative errno.
898 * @param pDevExt Instance data. GIP stuff may be updated.
899 */
900static void VBoxSupDrvGipTerm(PSUPDRVDEVEXT pDevExt)
901{
902 dprintf(("VBoxSupDrvTermGip:\n"));
903 PSUPGLOBALINFOPAGE pGip;
904
905 /*
906 * Cancel the timer and wait on DPCs if it was still pending.
907 */
908 if (KeCancelTimer(&pDevExt->GipTimer))
909 {
910 UNICODE_STRING RoutineName;
911 RtlInitUnicodeString(&RoutineName, L"KeFlushQueuedDpcs");
912 VOID (*pfnKeFlushQueuedDpcs)(VOID) = (VOID (*)(VOID))MmGetSystemRoutineAddress(&RoutineName);
913 if (pfnKeFlushQueuedDpcs)
914 pfnKeFlushQueuedDpcs();
915 }
916
917 /*
918 * Uninitialize the content.
919 */
920 pGip = pDevExt->pGip;
921 pDevExt->pGip = NULL;
922 if (pGip)
923 {
924 supdrvGipTerm(pGip);
925
926 /*
927 * Free the page.
928 */
929 if (pDevExt->pGipMdl)
930 {
931 IoFreeMdl(pDevExt->pGipMdl);
932 pDevExt->pGipMdl = NULL;
933 }
934 MmFreeContiguousMemory(pGip);
935 }
936}
937
938
939/**
940 * Timer callback function.
941 * The pvUser parameter is the pDevExt pointer.
942 */
943static void _stdcall VBoxSupDrvGipTimer(IN PKDPC pDpc, IN PVOID pvUser, IN PVOID SystemArgument1, IN PVOID SystemArgument2)
944{
945 PSUPDRVDEVEXT pDevExt = (PSUPDRVDEVEXT)pvUser;
946 PSUPGLOBALINFOPAGE pGip = pDevExt->pGip;
947 if (pGip)
948 {
949 if (pGip->u32Mode != SUPGIPMODE_ASYNC_TSC)
950 supdrvGipUpdate(pGip, supdrvOSMonotime());
951 else
952 {
953 RTCCUINTREG xFL = ASMGetFlags();
954 ASMIntDisable();
955
956 /*
957 * We cannot do other than assume a 1:1 relation ship between the
958 * affinity mask and the process despite the warnings in the docs.
959 * If someone knows a better way to get this done, please let bird know.
960 */
961 unsigned iSelf = KeGetCurrentProcessorNumber();
962 KAFFINITY Mask = KeQueryActiveProcessors();
963
964 for (unsigned i = 0; i < RT_ELEMENTS(pDevExt->aGipCpuDpcs); i++)
965 {
966 if ( i != iSelf
967 && (Mask & RT_BIT_64(i)))
968 KeInsertQueueDpc(&pDevExt->aGipCpuDpcs[i], 0, 0);
969 }
970
971 /* Run the normal update. */
972 supdrvGipUpdate(pGip, supdrvOSMonotime());
973
974 ASMSetFlags(xFL);
975 }
976 }
977}
978
979
980/**
981 * Per cpu callback callback function.
982 * The pvUser parameter is the pGip pointer.
983 */
984static void _stdcall VBoxSupDrvGipPerCpuDpc(IN PKDPC pDpc, IN PVOID pvUser, IN PVOID SystemArgument1, IN PVOID SystemArgument2)
985{
986 PSUPGLOBALINFOPAGE pGip = (PSUPGLOBALINFOPAGE)pvUser;
987 supdrvGipUpdatePerCpu(pGip, supdrvOSMonotime(), ASMGetApicId());
988}
989
990
991/**
992 * Maps the GIP into user space.
993 *
994 * @returns negative errno.
995 * @param pDevExt Instance data.
996 */
997int VBOXCALL supdrvOSGipMap(PSUPDRVDEVEXT pDevExt, PSUPGLOBALINFOPAGE *ppGip)
998{
999 dprintf2(("supdrvOSGipMap: ppGip=%p (pDevExt->pGipMdl=%p)\n", ppGip, pDevExt->pGipMdl));
1000
1001 /*
1002 * Map into user space.
1003 */
1004 int rc = 0;
1005 void *pv = NULL;
1006 __try
1007 {
1008 *ppGip = (PSUPGLOBALINFOPAGE)MmMapLockedPagesSpecifyCache(pDevExt->pGipMdl, UserMode, MmCached, NULL, FALSE, NormalPagePriority);
1009 }
1010 __except(EXCEPTION_EXECUTE_HANDLER)
1011 {
1012 NTSTATUS rcNt = GetExceptionCode();
1013 dprintf(("supdrvOsGipMap: Exception Code %#x\n", rcNt));
1014 rc = SUPDRV_ERR_LOCK_FAILED;
1015 }
1016
1017 dprintf2(("supdrvOSGipMap: returns %d, *ppGip=%p\n", rc, *ppGip));
1018 return 0;
1019}
1020
1021
1022/**
1023 * Maps the GIP into user space.
1024 *
1025 * @returns negative errno.
1026 * @param pDevExt Instance data.
1027 */
1028int VBOXCALL supdrvOSGipUnmap(PSUPDRVDEVEXT pDevExt, PSUPGLOBALINFOPAGE pGip)
1029{
1030 dprintf2(("supdrvOSGipUnmap: pGip=%p (pGipMdl=%p)\n", pGip, pDevExt->pGipMdl));
1031
1032 int rc = 0;
1033 __try
1034 {
1035 MmUnmapLockedPages((void *)pGip, pDevExt->pGipMdl);
1036 }
1037 __except(EXCEPTION_EXECUTE_HANDLER)
1038 {
1039 NTSTATUS rcNt = GetExceptionCode();
1040 dprintf(("supdrvOSGipUnmap: Exception Code %#x\n", rcNt));
1041 rc = SUPDRV_ERR_GENERAL_FAILURE;
1042 }
1043 dprintf2(("supdrvOSGipUnmap: returns %d\n", rc));
1044 return rc;
1045}
1046
1047
1048/**
1049 * Resumes the GIP updating.
1050 *
1051 * @param pDevExt Instance data.
1052 */
1053void VBOXCALL supdrvOSGipResume(PSUPDRVDEVEXT pDevExt)
1054{
1055 dprintf2(("supdrvOSGipResume:\n"));
1056 LARGE_INTEGER DueTime;
1057 DueTime.QuadPart = -10000; /* 1ms, relative */
1058 KeSetTimerEx(&pDevExt->GipTimer, DueTime, pDevExt->ulGipTimerInterval, &pDevExt->GipDpc);
1059}
1060
1061
1062/**
1063 * Suspends the GIP updating.
1064 *
1065 * @param pDevExt Instance data.
1066 */
1067void VBOXCALL supdrvOSGipSuspend(PSUPDRVDEVEXT pDevExt)
1068{
1069 dprintf2(("supdrvOSGipSuspend:\n"));
1070 KeCancelTimer(&pDevExt->GipTimer);
1071#ifdef RT_ARCH_AMD64
1072 ExSetTimerResolution(0, FALSE);
1073#endif
1074}
1075
1076
1077/**
1078 * Allocate small amounts of memory which is does not have the NX bit set.
1079 *
1080 * @returns Pointer to the allocated memory
1081 * @returns NULL if out of memory.
1082 * @param cb Size of the memory block.
1083 */
1084void *VBOXCALL supdrvOSExecAlloc(size_t cb)
1085{
1086#if 0 //def RT_ARCH_AMD64
1087 cb = RT_ALIGN_Z(cb, PAGE_SIZE);
1088 void *pv = ExAllocatePoolWithTag(NonPagedPool, cb, SUPDRV_NT_POOL_TAG);
1089 if (pv)
1090 {
1091 /*
1092 * Create a kernel mapping which we make PAGE_EXECUTE_READWRITE using
1093 * the MmProtectMdlSystemAddress API.
1094 */
1095 int rc = SUPDRV_ERR_NO_MEMORY;
1096 PMDL pMdl = IoAllocateMdl(pv, cb, FALSE, FALSE, NULL);
1097 if (pMdl)
1098 {
1099 MmBuildMdlForNonPagedPool(pMdl);
1100 __try
1101 {
1102 void *pvMapping = MmMapLockedPagesSpecifyCache(pMdl, KernelMode, MmCached, NULL, FALSE, NormalPagePriority);
1103 if (pvMapping)
1104 {
1105 NTSTATUS rc = MmProtectMdlSystemAddress(pMdl, PAGE_EXECUTE_READWRITE);
1106 if (NT_SUCCESS(rc))
1107 {
1108 /*
1109 * Create tracking structure and insert it into the list.
1110 */
1111
1112
1113 return pvMapping;
1114 }
1115
1116 MmUnmapLockedPages(pvMapping, pMdl);
1117 }
1118 }
1119 __except(EXCEPTION_EXECUTE_HANDLER)
1120 {
1121 NTSTATUS rc = GetExceptionCode();
1122 dprintf(("supdrvOSExecAlloc: Exception Code %#x\n", rc));
1123 }
1124 IoFreeMdl(pMem->u.mem.pMdl);
1125 }
1126 ExFreePool(pv);
1127 }
1128 dprintf2(("supdrvOSExecAlloc(%d): returns NULL\n", cb));
1129 return NULL;
1130#else
1131 void *pv = ExAllocatePoolWithTag(NonPagedPool, cb, SUPDRV_NT_POOL_TAG);
1132 dprintf2(("supdrvOSExecAlloc(%d): returns %p\n", cb, pv));
1133 return pv;
1134#endif
1135}
1136
1137
1138/**
1139 * Get the current CPU count.
1140 * @returns Number of cpus.
1141 */
1142unsigned VBOXCALL supdrvOSGetCPUCount(void)
1143{
1144 KAFFINITY Mask = KeQueryActiveProcessors();
1145 unsigned cCpus = 0;
1146 unsigned iBit;
1147 for (iBit = 0; iBit < sizeof(Mask) * 8; iBit++)
1148 if (Mask & RT_BIT_64(iBit))
1149 cCpus++;
1150 if (cCpus == 0) /* paranoia */
1151 cCpus = 1;
1152 return cCpus;
1153}
1154
1155
1156/**
1157 * Force async tsc mode (stub).
1158 */
1159bool VBOXCALL supdrvOSGetForcedAsyncTscMode(void)
1160{
1161 return false;
1162}
1163
1164
1165/**
1166 * Converts a supdrv error code to an nt status code.
1167 *
1168 * @returns corresponding nt status code.
1169 * @param rc supdrv error code (SUPDRV_ERR_* defines).
1170 */
1171static NTSTATUS VBoxSupDrvErr2NtStatus(int rc)
1172{
1173 switch (rc)
1174 {
1175 case 0: return STATUS_SUCCESS;
1176 case SUPDRV_ERR_GENERAL_FAILURE: return STATUS_NOT_SUPPORTED;
1177 case SUPDRV_ERR_INVALID_PARAM: return STATUS_INVALID_PARAMETER;
1178 case SUPDRV_ERR_INVALID_MAGIC: return STATUS_UNKNOWN_REVISION;
1179 case SUPDRV_ERR_INVALID_HANDLE: return STATUS_INVALID_HANDLE;
1180 case SUPDRV_ERR_INVALID_POINTER: return STATUS_INVALID_ADDRESS;
1181 case SUPDRV_ERR_LOCK_FAILED: return STATUS_NOT_LOCKED;
1182 case SUPDRV_ERR_ALREADY_LOADED: return STATUS_IMAGE_ALREADY_LOADED;
1183 case SUPDRV_ERR_PERMISSION_DENIED: return STATUS_ACCESS_DENIED;
1184 case SUPDRV_ERR_VERSION_MISMATCH: return STATUS_REVISION_MISMATCH;
1185 }
1186
1187 return STATUS_UNSUCCESSFUL;
1188}
1189
1190
1191/** Runtime assert implementation for Native Win32 Ring-0. */
1192RTDECL(void) AssertMsg1(const char *pszExpr, unsigned uLine, const char *pszFile, const char *pszFunction)
1193{
1194 DbgPrint("\n!!Assertion Failed!!\n"
1195 "Expression: %s\n"
1196 "Location : %s(%d) %s\n",
1197 pszExpr, pszFile, uLine, pszFunction);
1198}
1199
1200int VBOXCALL mymemcmp(const void *pv1, const void *pv2, size_t cb)
1201{
1202 const uint8_t *pb1 = (const uint8_t *)pv1;
1203 const uint8_t *pb2 = (const uint8_t *)pv2;
1204 for (; cb > 0; cb--, pb1++, pb2++)
1205 if (*pb1 != *pb2)
1206 return *pb1 - *pb2;
1207 return 0;
1208}
1209
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