VirtualBox

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

Last change on this file since 61600 was 58875, checked in by vboxsync, 9 years ago

Build fix.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 185.9 KB
Line 
1/* $Id: SUPDrv-win.cpp 58875 2015-11-26 09:18:58Z vboxsync $ */
2/** @file
3 * VBoxDrv - The VirtualBox Support Driver - Windows NT specifics.
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
28/*********************************************************************************************************************************
29* Header Files *
30*********************************************************************************************************************************/
31#define IPRT_NT_MAP_TO_ZW
32#define LOG_GROUP LOG_GROUP_SUP_DRV
33#include "../SUPDrvInternal.h"
34#include <excpt.h>
35#include <ntimage.h>
36
37#include <iprt/assert.h>
38#include <iprt/avl.h>
39#include <iprt/ctype.h>
40#include <iprt/initterm.h>
41#include <iprt/mem.h>
42#include <iprt/process.h>
43#include <iprt/power.h>
44#include <iprt/rand.h>
45#include <iprt/semaphore.h>
46#include <iprt/spinlock.h>
47#include <iprt/string.h>
48#include <VBox/log.h>
49#include <VBox/err.h>
50
51#include <iprt/asm-amd64-x86.h>
52
53#ifdef VBOX_WITH_HARDENING
54# include "SUPHardenedVerify-win.h"
55#endif
56
57
58/*********************************************************************************************************************************
59* Defined Constants And Macros *
60*********************************************************************************************************************************/
61/** The support service name. */
62#define SERVICE_NAME "VBoxDrv"
63/** The Pool tag (VBox). */
64#define SUPDRV_NT_POOL_TAG 'xoBV'
65
66/** NT device name for user access. */
67#define DEVICE_NAME_NT_USR L"\\Device\\VBoxDrvU"
68#ifdef VBOX_WITH_HARDENING
69/** Macro for checking for deflecting calls to the stub device. */
70# define VBOXDRV_COMPLETE_IRP_AND_RETURN_IF_STUB_DEV(a_pDevObj, a_pIrp) \
71 do { if ((a_pDevObj) == g_pDevObjStub) \
72 return supdrvNtCompleteRequest(STATUS_ACCESS_DENIED, a_pIrp); \
73 } while (0)
74/** Macro for checking for deflecting calls to the stub and error info
75 * devices. */
76# define VBOXDRV_COMPLETE_IRP_AND_RETURN_IF_STUB_OR_ERROR_INFO_DEV(a_pDevObj, a_pIrp) \
77 do { if ((a_pDevObj) == g_pDevObjStub || (a_pDevObj) == g_pDevObjErrorInfo) \
78 return supdrvNtCompleteRequest(STATUS_ACCESS_DENIED, a_pIrp); \
79 } while (0)
80#else
81# define VBOXDRV_COMPLETE_IRP_AND_RETURN_IF_STUB_DEV(a_pDevObj, a_pIrp) do {} while (0)
82# define VBOXDRV_COMPLETE_IRP_AND_RETURN_IF_STUB_OR_ERROR_INFO_DEV(a_pDevObj, a_pIrp) do {} while (0)
83#endif
84
85/** Enables the fast I/O control code path. */
86#define VBOXDRV_WITH_FAST_IO
87
88
89/*********************************************************************************************************************************
90* Structures and Typedefs *
91*********************************************************************************************************************************/
92/**
93 * Device extension used by VBoxDrvU.
94 */
95typedef struct SUPDRVDEVEXTUSR
96{
97 /** Global cookie (same location as in SUPDRVDEVEXT, different value). */
98 uint32_t u32Cookie;
99 /** Pointer to the main driver extension. */
100 PSUPDRVDEVEXT pMainDrvExt;
101} SUPDRVDEVEXTUSR;
102AssertCompileMembersAtSameOffset(SUPDRVDEVEXT, u32Cookie, SUPDRVDEVEXTUSR, u32Cookie);
103/** Pointer to the VBoxDrvU device extension. */
104typedef SUPDRVDEVEXTUSR *PSUPDRVDEVEXTUSR;
105/** Value of SUPDRVDEVEXTUSR::u32Cookie. */
106#define SUPDRVDEVEXTUSR_COOKIE UINT32_C(0x12345678)
107
108/** Get the main device extension. */
109#define SUPDRVNT_GET_DEVEXT(pDevObj) \
110 ( pDevObj != g_pDevObjUsr \
111 ? (PSUPDRVDEVEXT)pDevObj->DeviceExtension \
112 : ((PSUPDRVDEVEXTUSR)pDevObj->DeviceExtension)->pMainDrvExt )
113
114#ifdef VBOX_WITH_HARDENING
115
116/**
117 * Device extension used by VBoxDrvStub.
118 */
119typedef struct SUPDRVDEVEXTSTUB
120{
121 /** Common header. */
122 SUPDRVDEVEXTUSR Common;
123} SUPDRVDEVEXTSTUB;
124/** Pointer to the VBoxDrvStub device extension. */
125typedef SUPDRVDEVEXTSTUB *PSUPDRVDEVEXTSTUB;
126/** Value of SUPDRVDEVEXTSTUB::Common.u32Cookie. */
127#define SUPDRVDEVEXTSTUB_COOKIE UINT32_C(0x90abcdef)
128
129
130/**
131 * Device extension used by VBoxDrvErrorInfo.
132 */
133typedef struct SUPDRVDEVEXTERRORINFO
134{
135 /** Common header. */
136 SUPDRVDEVEXTUSR Common;
137} SUPDRVDEVEXTERRORINFO;
138/** Pointer to the VBoxDrvErrorInfo device extension. */
139typedef SUPDRVDEVEXTERRORINFO *PSUPDRVDEVEXTERRORINFO;
140/** Value of SUPDRVDEVEXTERRORINFO::Common.u32Cookie. */
141#define SUPDRVDEVEXTERRORINFO_COOKIE UINT32_C(0xBadC0ca0)
142
143/**
144 * Error info for a failed VBoxDrv or VBoxDrvStub open attempt.
145 */
146typedef struct SUPDRVNTERRORINFO
147{
148 /** The list entry (in g_ErrorInfoHead). */
149 RTLISTNODE ListEntry;
150 /** The ID of the process this error info belongs to. */
151 HANDLE hProcessId;
152 /** The ID of the thread owning this info. */
153 HANDLE hThreadId;
154 /** Milliseconds createion timestamp (for cleaning up). */
155 uint64_t uCreatedMsTs;
156 /** Number of bytes of valid info. */
157 uint32_t cchErrorInfo;
158 /** The error info. */
159 char szErrorInfo[16384 - sizeof(RTLISTNODE) - sizeof(HANDLE)*2 - sizeof(uint64_t) - sizeof(uint32_t) - 0x20];
160} SUPDRVNTERRORINFO;
161/** Pointer to error info. */
162typedef SUPDRVNTERRORINFO *PSUPDRVNTERRORINFO;
163
164
165/**
166 * The kind of process we're protecting.
167 */
168typedef enum SUPDRVNTPROTECTKIND
169{
170 kSupDrvNtProtectKind_Invalid = 0,
171
172 /** Stub process protection while performing process verification.
173 * Next: StubSpawning (or free) */
174 kSupDrvNtProtectKind_StubUnverified,
175 /** Stub process protection before it creates the VM process.
176 * Next: StubParent, StubDead. */
177 kSupDrvNtProtectKind_StubSpawning,
178 /** Stub process protection while having a VM process as child.
179 * Next: StubDead */
180 kSupDrvNtProtectKind_StubParent,
181 /** Dead stub process. */
182 kSupDrvNtProtectKind_StubDead,
183
184 /** Potential VM process.
185 * Next: VmProcessConfirmed, VmProcessDead. */
186 kSupDrvNtProtectKind_VmProcessUnconfirmed,
187 /** Confirmed VM process.
188 * Next: VmProcessDead. */
189 kSupDrvNtProtectKind_VmProcessConfirmed,
190 /** Dead VM process. */
191 kSupDrvNtProtectKind_VmProcessDead,
192
193 /** End of valid protection kinds. */
194 kSupDrvNtProtectKind_End
195} SUPDRVNTPROTECTKIND;
196
197/**
198 * A NT process protection structure.
199 */
200typedef struct SUPDRVNTPROTECT
201{
202 /** The AVL node core structure. The process ID is the pid. */
203 AVLPVNODECORE AvlCore;
204 /** Magic value (SUPDRVNTPROTECT_MAGIC). */
205 uint32_t volatile u32Magic;
206 /** Reference counter. */
207 uint32_t volatile cRefs;
208 /** The kind of process we're protecting. */
209 SUPDRVNTPROTECTKIND volatile enmProcessKind;
210 /** Whether this structure is in the tree. */
211 bool fInTree : 1;
212 /** 7,: Hack to allow the supid themes service duplicate handle privileges to
213 * our process. */
214 bool fThemesFirstProcessCreateHandle : 1;
215 /** Vista, 7 & 8: Hack to allow more rights to the handle returned by
216 * NtCreateUserProcess. Only applicable to VmProcessUnconfirmed. */
217 bool fFirstProcessCreateHandle : 1;
218 /** Vista, 7 & 8: Hack to allow more rights to the handle returned by
219 * NtCreateUserProcess. Only applicable to VmProcessUnconfirmed. */
220 bool fFirstThreadCreateHandle : 1;
221 /** 8.1: Hack to allow more rights to the handle returned by
222 * NtCreateUserProcess. Only applicable to VmProcessUnconfirmed. */
223 bool fCsrssFirstProcessCreateHandle : 1;
224 /** Vista, 7 & 8: Hack to allow more rights to the handle duplicated by CSRSS
225 * during process creation. Only applicable to VmProcessUnconfirmed. On
226 * 32-bit systems we allow two as ZoneAlarm's system call hooks has been
227 * observed to do some seemingly unnecessary duplication work. */
228 int32_t volatile cCsrssFirstProcessDuplicateHandle;
229
230 /** The parent PID for VM processes, otherwise NULL. */
231 HANDLE hParentPid;
232 /** The TID of the thread opening VBoxDrv or VBoxDrvStub, NULL if not opened. */
233 HANDLE hOpenTid;
234 /** The PID of the CSRSS process associated with this process. */
235 HANDLE hCsrssPid;
236 /** Pointer to the CSRSS process structure (referenced). */
237 PEPROCESS pCsrssProcess;
238 /** State dependent data. */
239 union
240 {
241 /** A stub process in the StubParent state will keep a reference to a child
242 * while it's in the VmProcessUnconfirmed state so that it can be cleaned up
243 * correctly if things doesn't work out. */
244 struct SUPDRVNTPROTECT *pChild;
245 /** A process in the VmProcessUnconfirmed state will keep a weak
246 * reference to the parent's protection structure so it can clean up the pChild
247 * reference the parent has to it. */
248 struct SUPDRVNTPROTECT *pParent;
249 } u;
250} SUPDRVNTPROTECT;
251/** Pointer to a NT process protection record. */
252typedef SUPDRVNTPROTECT *PSUPDRVNTPROTECT;
253/** The SUPDRVNTPROTECT::u32Magic value (Robert A. Heinlein). */
254# define SUPDRVNTPROTECT_MAGIC UINT32_C(0x19070707)
255/** The SUPDRVNTPROTECT::u32Magic value of a dead structure. */
256# define SUPDRVNTPROTECT_MAGIC_DEAD UINT32_C(0x19880508)
257
258/** Pointer to ObGetObjectType. */
259typedef POBJECT_TYPE (NTAPI *PFNOBGETOBJECTTYPE)(PVOID);
260/** Pointer to ObRegisterCallbacks. */
261typedef NTSTATUS (NTAPI *PFNOBREGISTERCALLBACKS)(POB_CALLBACK_REGISTRATION, PVOID *);
262/** Pointer to ObUnregisterCallbacks. */
263typedef VOID (NTAPI *PFNOBUNREGISTERCALLBACKS)(PVOID);
264/** Pointer to PsSetCreateProcessNotifyRoutineEx. */
265typedef NTSTATUS (NTAPI *PFNPSSETCREATEPROCESSNOTIFYROUTINEEX)(PCREATE_PROCESS_NOTIFY_ROUTINE_EX, BOOLEAN);
266/** Pointer to PsReferenceProcessFilePointer. */
267typedef NTSTATUS (NTAPI *PFNPSREFERENCEPROCESSFILEPOINTER)(PEPROCESS, PFILE_OBJECT *);
268/** Pointer to PsIsProtectedProcessLight. */
269typedef BOOLEAN (NTAPI *PFNPSISPROTECTEDPROCESSLIGHT)(PEPROCESS);
270/** Pointer to ZwAlpcCreatePort. */
271typedef NTSTATUS (NTAPI *PFNZWALPCCREATEPORT)(PHANDLE, POBJECT_ATTRIBUTES, struct _ALPC_PORT_ATTRIBUTES *);
272
273#endif /* VBOX_WITH_HARDENINIG */
274
275
276/*********************************************************************************************************************************
277* Internal Functions *
278*********************************************************************************************************************************/
279static void _stdcall VBoxDrvNtUnload(PDRIVER_OBJECT pDrvObj);
280static NTSTATUS _stdcall VBoxDrvNtCreate(PDEVICE_OBJECT pDevObj, PIRP pIrp);
281static NTSTATUS _stdcall VBoxDrvNtCleanup(PDEVICE_OBJECT pDevObj, PIRP pIrp);
282static NTSTATUS _stdcall VBoxDrvNtClose(PDEVICE_OBJECT pDevObj, PIRP pIrp);
283#ifdef VBOXDRV_WITH_FAST_IO
284static BOOLEAN _stdcall VBoxDrvNtFastIoDeviceControl(PFILE_OBJECT pFileObj, BOOLEAN fWait, PVOID pvInput, ULONG cbInput,
285 PVOID pvOutput, ULONG cbOutput, ULONG uCmd,
286 PIO_STATUS_BLOCK pIoStatus, PDEVICE_OBJECT pDevObj);
287#endif
288static NTSTATUS _stdcall VBoxDrvNtDeviceControl(PDEVICE_OBJECT pDevObj, PIRP pIrp);
289static int VBoxDrvNtDeviceControlSlow(PSUPDRVDEVEXT pDevExt, PSUPDRVSESSION pSession, PIRP pIrp, PIO_STACK_LOCATION pStack);
290static NTSTATUS _stdcall VBoxDrvNtInternalDeviceControl(PDEVICE_OBJECT pDevObj, PIRP pIrp);
291static VOID _stdcall VBoxPowerDispatchCallback(PVOID pCallbackContext, PVOID pArgument1, PVOID pArgument2);
292static NTSTATUS _stdcall VBoxDrvNtRead(PDEVICE_OBJECT pDevObj, PIRP pIrp);
293static NTSTATUS _stdcall VBoxDrvNtNotSupportedStub(PDEVICE_OBJECT pDevObj, PIRP pIrp);
294static NTSTATUS VBoxDrvNtErr2NtStatus(int rc);
295#ifdef VBOX_WITH_HARDENING
296static NTSTATUS supdrvNtProtectInit(void);
297static void supdrvNtProtectTerm(void);
298static int supdrvNtProtectCreate(PSUPDRVNTPROTECT *ppNtProtect, HANDLE hPid,
299 SUPDRVNTPROTECTKIND enmProcessKind, bool fLink);
300static void supdrvNtProtectRelease(PSUPDRVNTPROTECT pNtProtect);
301static PSUPDRVNTPROTECT supdrvNtProtectLookup(HANDLE hPid);
302static int supdrvNtProtectFindAssociatedCsrss(PSUPDRVNTPROTECT pNtProtect);
303static int supdrvNtProtectVerifyProcess(PSUPDRVNTPROTECT pNtProtect);
304
305static bool supdrvNtIsDebuggerAttached(void);
306static void supdrvNtErrorInfoCleanupProcess(HANDLE hProcessId);
307
308#endif
309
310
311/*********************************************************************************************************************************
312* Exported Functions *
313*********************************************************************************************************************************/
314RT_C_DECLS_BEGIN
315ULONG _stdcall DriverEntry(PDRIVER_OBJECT pDrvObj, PUNICODE_STRING pRegPath);
316RT_C_DECLS_END
317
318
319/*********************************************************************************************************************************
320* Global Variables *
321*********************************************************************************************************************************/
322/** Pointer to the system device instance. */
323static PDEVICE_OBJECT g_pDevObjSys = NULL;
324/** Pointer to the user device instance. */
325static PDEVICE_OBJECT g_pDevObjUsr = NULL;
326#ifdef VBOXDRV_WITH_FAST_IO
327/** Fast I/O dispatch table. */
328static FAST_IO_DISPATCH const g_VBoxDrvFastIoDispatch =
329{
330 /* .SizeOfFastIoDispatch = */ sizeof(g_VBoxDrvFastIoDispatch),
331 /* .FastIoCheckIfPossible = */ NULL,
332 /* .FastIoRead = */ NULL,
333 /* .FastIoWrite = */ NULL,
334 /* .FastIoQueryBasicInfo = */ NULL,
335 /* .FastIoQueryStandardInfo = */ NULL,
336 /* .FastIoLock = */ NULL,
337 /* .FastIoUnlockSingle = */ NULL,
338 /* .FastIoUnlockAll = */ NULL,
339 /* .FastIoUnlockAllByKey = */ NULL,
340 /* .FastIoDeviceControl = */ VBoxDrvNtFastIoDeviceControl,
341 /* .AcquireFileForNtCreateSection = */ NULL,
342 /* .ReleaseFileForNtCreateSection = */ NULL,
343 /* .FastIoDetachDevice = */ NULL,
344 /* .FastIoQueryNetworkOpenInfo = */ NULL,
345 /* .AcquireForModWrite = */ NULL,
346 /* .MdlRead = */ NULL,
347 /* .MdlReadComplete = */ NULL,
348 /* .PrepareMdlWrite = */ NULL,
349 /* .MdlWriteComplete = */ NULL,
350 /* .FastIoReadCompressed = */ NULL,
351 /* .FastIoWriteCompressed = */ NULL,
352 /* .MdlReadCompleteCompressed = */ NULL,
353 /* .MdlWriteCompleteCompressed = */ NULL,
354 /* .FastIoQueryOpen = */ NULL,
355 /* .ReleaseForModWrite = */ NULL,
356 /* .AcquireForCcFlush = */ NULL,
357 /* .ReleaseForCcFlush = */ NULL,
358};
359#endif /* VBOXDRV_WITH_FAST_IO */
360
361#ifdef VBOX_WITH_HARDENING
362/** Pointer to the stub device instance. */
363static PDEVICE_OBJECT g_pDevObjStub = NULL;
364/** Spinlock protecting g_NtProtectTree as well as the releasing of protection
365 * structures. */
366static RTSPINLOCK g_hNtProtectLock = NIL_RTSPINLOCK;
367/** AVL tree of SUPDRVNTPROTECT structures. */
368static AVLPVTREE g_NtProtectTree = NULL;
369/** Cookie returned by ObRegisterCallbacks for the callbacks. */
370static PVOID g_pvObCallbacksCookie = NULL;
371/** Combined windows NT version number. See SUP_MAKE_NT_VER_COMBINED. */
372uint32_t g_uNtVerCombined = 0;
373/** Pointer to ObGetObjectType if available.. */
374static PFNOBGETOBJECTTYPE g_pfnObGetObjectType = NULL;
375/** Pointer to ObRegisterCallbacks if available.. */
376static PFNOBREGISTERCALLBACKS g_pfnObRegisterCallbacks = NULL;
377/** Pointer to ObUnregisterCallbacks if available.. */
378static PFNOBUNREGISTERCALLBACKS g_pfnObUnRegisterCallbacks = NULL;
379/** Pointer to PsSetCreateProcessNotifyRoutineEx if available.. */
380static PFNPSSETCREATEPROCESSNOTIFYROUTINEEX g_pfnPsSetCreateProcessNotifyRoutineEx = NULL;
381/** Pointer to PsReferenceProcessFilePointer if available. */
382static PFNPSREFERENCEPROCESSFILEPOINTER g_pfnPsReferenceProcessFilePointer = NULL;
383/** Pointer to PsIsProtectedProcessLight. */
384static PFNPSISPROTECTEDPROCESSLIGHT g_pfnPsIsProtectedProcessLight = NULL;
385/** Pointer to ZwAlpcCreatePort. */
386static PFNZWALPCCREATEPORT g_pfnZwAlpcCreatePort = NULL;
387
388# ifdef RT_ARCH_AMD64
389extern "C" {
390/** Pointer to KiServiceLinkage (used to fake missing ZwQueryVirtualMemory on
391 * XP64 / W2K3-64). */
392PFNRT g_pfnKiServiceLinkage = NULL;
393/** Pointer to KiServiceInternal (used to fake missing ZwQueryVirtualMemory on
394 * XP64 / W2K3-64) */
395PFNRT g_pfnKiServiceInternal = NULL;
396}
397# endif
398/** The primary ALPC port object type. (LpcPortObjectType at init time.) */
399static POBJECT_TYPE g_pAlpcPortObjectType1 = NULL;
400/** The secondary ALPC port object type. (Sampled at runtime.) */
401static POBJECT_TYPE volatile g_pAlpcPortObjectType2 = NULL;
402
403/** Pointer to the error information device instance. */
404static PDEVICE_OBJECT g_pDevObjErrorInfo = NULL;
405/** Fast mutex semaphore protecting the error info list. */
406static RTSEMMUTEX g_hErrorInfoLock = NIL_RTSEMMUTEX;
407/** Head of the error info (SUPDRVNTERRORINFO). */
408static RTLISTANCHOR g_ErrorInfoHead;
409
410#endif
411
412
413/**
414 * Takes care of creating the devices and their symbolic links.
415 *
416 * @returns NT status code.
417 * @param pDrvObj Pointer to driver object.
418 */
419static NTSTATUS vboxdrvNtCreateDevices(PDRIVER_OBJECT pDrvObj)
420{
421 /*
422 * System device.
423 */
424 UNICODE_STRING DevName;
425 RtlInitUnicodeString(&DevName, SUPDRV_NT_DEVICE_NAME_SYS);
426 NTSTATUS rcNt = IoCreateDevice(pDrvObj, sizeof(SUPDRVDEVEXT), &DevName, FILE_DEVICE_UNKNOWN, 0, FALSE, &g_pDevObjSys);
427 if (NT_SUCCESS(rcNt))
428 {
429 /*
430 * User device.
431 */
432 RtlInitUnicodeString(&DevName, SUPDRV_NT_DEVICE_NAME_USR);
433 rcNt = IoCreateDevice(pDrvObj, sizeof(SUPDRVDEVEXTUSR), &DevName, FILE_DEVICE_UNKNOWN, 0, FALSE, &g_pDevObjUsr);
434 if (NT_SUCCESS(rcNt))
435 {
436 PSUPDRVDEVEXTUSR pDevExtUsr = (PSUPDRVDEVEXTUSR)g_pDevObjUsr->DeviceExtension;
437 pDevExtUsr->pMainDrvExt = (PSUPDRVDEVEXT)g_pDevObjSys->DeviceExtension;
438 pDevExtUsr->u32Cookie = SUPDRVDEVEXTUSR_COOKIE;
439
440#ifdef VBOX_WITH_HARDENING
441 /*
442 * Hardened stub device.
443 */
444 RtlInitUnicodeString(&DevName, SUPDRV_NT_DEVICE_NAME_STUB);
445 rcNt = IoCreateDevice(pDrvObj, sizeof(SUPDRVDEVEXTSTUB), &DevName, FILE_DEVICE_UNKNOWN, 0, FALSE, &g_pDevObjStub);
446 if (NT_SUCCESS(rcNt))
447 {
448 if (NT_SUCCESS(rcNt))
449 {
450 PSUPDRVDEVEXTSTUB pDevExtStub = (PSUPDRVDEVEXTSTUB)g_pDevObjStub->DeviceExtension;
451 pDevExtStub->Common.pMainDrvExt = (PSUPDRVDEVEXT)g_pDevObjSys->DeviceExtension;
452 pDevExtStub->Common.u32Cookie = SUPDRVDEVEXTSTUB_COOKIE;
453
454 /*
455 * Hardened error information device.
456 */
457 RtlInitUnicodeString(&DevName, SUPDRV_NT_DEVICE_NAME_ERROR_INFO);
458 rcNt = IoCreateDevice(pDrvObj, sizeof(SUPDRVDEVEXTERRORINFO), &DevName, FILE_DEVICE_UNKNOWN, 0, FALSE,
459 &g_pDevObjErrorInfo);
460 if (NT_SUCCESS(rcNt))
461 {
462 g_pDevObjErrorInfo->Flags |= DO_BUFFERED_IO;
463
464 if (NT_SUCCESS(rcNt))
465 {
466 PSUPDRVDEVEXTERRORINFO pDevExtStub = (PSUPDRVDEVEXTERRORINFO)g_pDevObjStub->DeviceExtension;
467 pDevExtStub->Common.pMainDrvExt = (PSUPDRVDEVEXT)g_pDevObjSys->DeviceExtension;
468 pDevExtStub->Common.u32Cookie = SUPDRVDEVEXTERRORINFO_COOKIE;
469
470#endif
471 /* Done. */
472 return rcNt;
473#ifdef VBOX_WITH_HARDENING
474 }
475
476 /* Bail out. */
477 IoDeleteDevice(g_pDevObjErrorInfo);
478 g_pDevObjErrorInfo = NULL;
479 }
480 }
481
482 /* Bail out. */
483 IoDeleteDevice(g_pDevObjStub);
484 g_pDevObjUsr = NULL;
485 }
486#endif
487 IoDeleteDevice(g_pDevObjUsr);
488 g_pDevObjUsr = NULL;
489 }
490 IoDeleteDevice(g_pDevObjSys);
491 g_pDevObjSys = NULL;
492 }
493 return rcNt;
494}
495
496/**
497 * Destroys the devices and links created by vboxdrvNtCreateDevices.
498 */
499static void vboxdrvNtDestroyDevices(void)
500{
501 if (g_pDevObjUsr)
502 {
503 PSUPDRVDEVEXTUSR pDevExtUsr = (PSUPDRVDEVEXTUSR)g_pDevObjUsr->DeviceExtension;
504 pDevExtUsr->pMainDrvExt = NULL;
505 }
506#ifdef VBOX_WITH_HARDENING
507 if (g_pDevObjStub)
508 {
509 PSUPDRVDEVEXTSTUB pDevExtStub = (PSUPDRVDEVEXTSTUB)g_pDevObjStub->DeviceExtension;
510 pDevExtStub->Common.pMainDrvExt = NULL;
511 }
512 if (g_pDevObjErrorInfo)
513 {
514 PSUPDRVDEVEXTERRORINFO pDevExtErrorInfo = (PSUPDRVDEVEXTERRORINFO)g_pDevObjStub->DeviceExtension;
515 pDevExtErrorInfo->Common.pMainDrvExt = NULL;
516 }
517#endif
518
519#ifdef VBOX_WITH_HARDENING
520 IoDeleteDevice(g_pDevObjErrorInfo);
521 g_pDevObjErrorInfo = NULL;
522 IoDeleteDevice(g_pDevObjStub);
523 g_pDevObjStub = NULL;
524#endif
525 IoDeleteDevice(g_pDevObjUsr);
526 g_pDevObjUsr = NULL;
527 IoDeleteDevice(g_pDevObjSys);
528 g_pDevObjSys = NULL;
529}
530
531
532/**
533 * Driver entry point.
534 *
535 * @returns appropriate status code.
536 * @param pDrvObj Pointer to driver object.
537 * @param pRegPath Registry base path.
538 */
539ULONG _stdcall DriverEntry(PDRIVER_OBJECT pDrvObj, PUNICODE_STRING pRegPath)
540{
541 /*
542 * Sanity checks.
543 */
544#ifdef VBOXDRV_WITH_FAST_IO
545 if (g_VBoxDrvFastIoDispatch.FastIoDeviceControl != VBoxDrvNtFastIoDeviceControl)
546 {
547 DbgPrint("VBoxDrv: FastIoDeviceControl=%p instead of %p\n",
548 g_VBoxDrvFastIoDispatch.FastIoDeviceControl, VBoxDrvNtFastIoDeviceControl);
549 return STATUS_INTERNAL_ERROR;
550 }
551#endif
552
553 /*
554 * Initialize the runtime (IPRT).
555 */
556 NTSTATUS rcNt;
557 int vrc = RTR0Init(0);
558 if (RT_SUCCESS(vrc))
559 {
560 Log(("VBoxDrv::DriverEntry\n"));
561
562#ifdef VBOX_WITH_HARDENING
563 /*
564 * Initialize process protection.
565 */
566 rcNt = supdrvNtProtectInit();
567 if (NT_SUCCESS(rcNt))
568#endif
569 {
570 /*
571 * Create device.
572 * (That means creating a device object and a symbolic link so the DOS
573 * subsystems (OS/2, win32, ++) can access the device.)
574 */
575 rcNt = vboxdrvNtCreateDevices(pDrvObj);
576 if (NT_SUCCESS(rcNt))
577 {
578 /*
579 * Initialize the device extension.
580 */
581 PSUPDRVDEVEXT pDevExt = (PSUPDRVDEVEXT)g_pDevObjSys->DeviceExtension;
582 memset(pDevExt, 0, sizeof(*pDevExt));
583
584 vrc = supdrvInitDevExt(pDevExt, sizeof(SUPDRVSESSION));
585 if (!vrc)
586 {
587 /*
588 * Setup the driver entry points in pDrvObj.
589 */
590 pDrvObj->DriverUnload = VBoxDrvNtUnload;
591 pDrvObj->MajorFunction[IRP_MJ_CREATE] = VBoxDrvNtCreate;
592 pDrvObj->MajorFunction[IRP_MJ_CLEANUP] = VBoxDrvNtCleanup;
593 pDrvObj->MajorFunction[IRP_MJ_CLOSE] = VBoxDrvNtClose;
594 pDrvObj->MajorFunction[IRP_MJ_DEVICE_CONTROL] = VBoxDrvNtDeviceControl;
595 pDrvObj->MajorFunction[IRP_MJ_INTERNAL_DEVICE_CONTROL] = VBoxDrvNtInternalDeviceControl;
596 pDrvObj->MajorFunction[IRP_MJ_READ] = VBoxDrvNtRead;
597 pDrvObj->MajorFunction[IRP_MJ_WRITE] = VBoxDrvNtNotSupportedStub;
598
599#ifdef VBOXDRV_WITH_FAST_IO
600 /* Fast I/O to speed up guest execution roundtrips. */
601 pDrvObj->FastIoDispatch = (PFAST_IO_DISPATCH)&g_VBoxDrvFastIoDispatch;
602#endif
603
604 /*
605 * Register ourselves for power state changes. We don't
606 * currently care if this fails.
607 */
608 UNICODE_STRING CallbackName;
609 RtlInitUnicodeString(&CallbackName, L"\\Callback\\PowerState");
610
611 OBJECT_ATTRIBUTES Attr;
612 InitializeObjectAttributes(&Attr, &CallbackName, OBJ_CASE_INSENSITIVE, NULL, NULL);
613
614 rcNt = ExCreateCallback(&pDevExt->pObjPowerCallback, &Attr, TRUE, TRUE);
615 if (rcNt == STATUS_SUCCESS)
616 pDevExt->hPowerCallback = ExRegisterCallback(pDevExt->pObjPowerCallback,
617 VBoxPowerDispatchCallback,
618 g_pDevObjSys);
619
620 /*
621 * Done! Returning success!
622 */
623 Log(("VBoxDrv::DriverEntry returning STATUS_SUCCESS\n"));
624 return STATUS_SUCCESS;
625 }
626
627 Log(("supdrvInitDevExit failed with vrc=%d!\n", vrc));
628 rcNt = VBoxDrvNtErr2NtStatus(vrc);
629
630 vboxdrvNtDestroyDevices();
631 }
632#ifdef VBOX_WITH_HARDENING
633 supdrvNtProtectTerm();
634#endif
635 }
636 RTTermRunCallbacks(RTTERMREASON_UNLOAD, 0);
637 RTR0Term();
638 }
639 else
640 {
641 Log(("RTR0Init failed with vrc=%d!\n", vrc));
642 rcNt = VBoxDrvNtErr2NtStatus(vrc);
643 }
644 if (NT_SUCCESS(rcNt))
645 rcNt = STATUS_INVALID_PARAMETER;
646 return rcNt;
647}
648
649
650/**
651 * Unload the driver.
652 *
653 * @param pDrvObj Driver object.
654 */
655void _stdcall VBoxDrvNtUnload(PDRIVER_OBJECT pDrvObj)
656{
657 PSUPDRVDEVEXT pDevExt = (PSUPDRVDEVEXT)g_pDevObjSys->DeviceExtension;
658
659 Log(("VBoxDrvNtUnload at irql %d\n", KeGetCurrentIrql()));
660
661 /* Clean up the power callback registration. */
662 if (pDevExt->hPowerCallback)
663 ExUnregisterCallback(pDevExt->hPowerCallback);
664 if (pDevExt->pObjPowerCallback)
665 ObDereferenceObject(pDevExt->pObjPowerCallback);
666
667 /*
668 * We ASSUME that it's not possible to unload a driver with open handles.
669 */
670 supdrvDeleteDevExt(pDevExt);
671#ifdef VBOX_WITH_HARDENING
672 supdrvNtProtectTerm();
673#endif
674 RTTermRunCallbacks(RTTERMREASON_UNLOAD, 0);
675 RTR0Term();
676 vboxdrvNtDestroyDevices();
677
678 NOREF(pDrvObj);
679}
680
681
682/**
683 * For simplifying request completion into a simple return statement, extended
684 * version.
685 *
686 * @returns rcNt
687 * @param rcNt The status code.
688 * @param uInfo Extra info value.
689 * @param pIrp The IRP.
690 */
691DECLINLINE(NTSTATUS) supdrvNtCompleteRequestEx(NTSTATUS rcNt, ULONG_PTR uInfo, PIRP pIrp)
692{
693 pIrp->IoStatus.Status = rcNt;
694 pIrp->IoStatus.Information = uInfo;
695 IoCompleteRequest(pIrp, IO_NO_INCREMENT);
696 return rcNt;
697}
698
699
700/**
701 * For simplifying request completion into a simple return statement.
702 *
703 * @returns rcNt
704 * @param rcNt The status code.
705 * @param pIrp The IRP.
706 */
707DECLINLINE(NTSTATUS) supdrvNtCompleteRequest(NTSTATUS rcNt, PIRP pIrp)
708{
709 return supdrvNtCompleteRequestEx(rcNt, 0 /*uInfo*/, pIrp);
710}
711
712
713/**
714 * Create (i.e. Open) file entry point.
715 *
716 * @param pDevObj Device object.
717 * @param pIrp Request packet.
718 */
719NTSTATUS _stdcall VBoxDrvNtCreate(PDEVICE_OBJECT pDevObj, PIRP pIrp)
720{
721 Log(("VBoxDrvNtCreate: RequestorMode=%d\n", pIrp->RequestorMode));
722 PIO_STACK_LOCATION pStack = IoGetCurrentIrpStackLocation(pIrp);
723 PFILE_OBJECT pFileObj = pStack->FileObject;
724 PSUPDRVDEVEXT pDevExt = SUPDRVNT_GET_DEVEXT(pDevObj);
725
726 /*
727 * We are not remotely similar to a directory...
728 * (But this is possible.)
729 */
730 if (pStack->Parameters.Create.Options & FILE_DIRECTORY_FILE)
731 return supdrvNtCompleteRequest(STATUS_NOT_A_DIRECTORY, pIrp);
732
733 /*
734 * Don't create a session for kernel clients, they'll close the handle
735 * immediately and work with the file object via
736 * VBoxDrvNtInternalDeviceControl. The first request will be one to
737 * create a session.
738 */
739 NTSTATUS rcNt;
740 if (pIrp->RequestorMode == KernelMode)
741 {
742 if (pDevObj == g_pDevObjSys)
743 return supdrvNtCompleteRequestEx(STATUS_SUCCESS, FILE_OPENED, pIrp);
744
745 rcNt = STATUS_ACCESS_DENIED;
746 }
747#ifdef VBOX_WITH_HARDENING
748 /*
749 * Anyone can open the error device.
750 */
751 else if (pDevObj == g_pDevObjErrorInfo)
752 {
753 pFileObj->FsContext = NULL;
754 return supdrvNtCompleteRequestEx(STATUS_SUCCESS, FILE_OPENED, pIrp);
755 }
756#endif
757 else
758 {
759#if defined(VBOX_WITH_HARDENING) && !defined(VBOX_WITHOUT_DEBUGGER_CHECKS)
760 /*
761 * Make sure no debuggers are attached to non-user processes.
762 */
763 if ( pDevObj != g_pDevObjUsr
764 && supdrvNtIsDebuggerAttached())
765 {
766 LogRel(("vboxdrv: Process %p is being debugged, access to vboxdrv / vboxdrvu declined.\n",
767 PsGetProcessId(PsGetCurrentProcess())));
768 rcNt = STATUS_TRUST_FAILURE;
769 }
770 else
771#endif
772 {
773 int rc = VINF_SUCCESS;
774
775#ifdef VBOX_WITH_HARDENING
776 /*
777 * Access to the stub device is only granted to processes which
778 * passes verification.
779 *
780 * Note! The stub device has no need for a SUPDRVSESSION structure,
781 * so the it uses the SUPDRVNTPROTECT directly instead.
782 */
783 if (pDevObj == g_pDevObjStub)
784 {
785 PSUPDRVNTPROTECT pNtProtect = NULL;
786 rc = supdrvNtProtectCreate(&pNtProtect, PsGetProcessId(PsGetCurrentProcess()),
787 kSupDrvNtProtectKind_StubUnverified, true /*fLink*/);
788 if (RT_SUCCESS(rc))
789 {
790 rc = supdrvNtProtectFindAssociatedCsrss(pNtProtect);
791 if (RT_SUCCESS(rc))
792 rc = supdrvNtProtectVerifyProcess(pNtProtect);
793 if (RT_SUCCESS(rc))
794 {
795 pFileObj->FsContext = pNtProtect; /* Keeps reference. */
796 return supdrvNtCompleteRequestEx(STATUS_SUCCESS, FILE_OPENED, pIrp);
797 }
798
799 supdrvNtProtectRelease(pNtProtect);
800 }
801 LogRel(("vboxdrv: Declined %p access to VBoxDrvStub: rc=%d\n", PsGetProcessId(PsGetCurrentProcess()), rc));
802 }
803 /*
804 * Unrestricted access is only granted to a process in the
805 * VmProcessUnconfirmed state that checks out correctly and is
806 * allowed to transition to VmProcessConfirmed. Again, only one
807 * session per process.
808 */
809 else if (pDevObj != g_pDevObjUsr)
810 {
811 PSUPDRVNTPROTECT pNtProtect = supdrvNtProtectLookup(PsGetProcessId(PsGetCurrentProcess()));
812 if (pNtProtect)
813 {
814 if (pNtProtect->enmProcessKind == kSupDrvNtProtectKind_VmProcessUnconfirmed)
815 {
816 rc = supdrvNtProtectVerifyProcess(pNtProtect);
817 if (RT_SUCCESS(rc))
818 {
819 /* Create a session. */
820 PSUPDRVSESSION pSession;
821 rc = supdrvCreateSession(pDevExt, true /*fUser*/, pDevObj == g_pDevObjSys /*fUnrestricted*/,
822 &pSession);
823 if (RT_SUCCESS(rc))
824 {
825 rc = supdrvSessionHashTabInsert(pDevExt, pSession, (PSUPDRVSESSION *)&pFileObj->FsContext, NULL);
826 supdrvSessionRelease(pSession);
827 if (RT_SUCCESS(rc))
828 {
829 pSession->pNtProtect = pNtProtect; /* Keeps reference. */
830 return supdrvNtCompleteRequestEx(STATUS_SUCCESS, FILE_OPENED, pIrp);
831 }
832 }
833
834 /* No second attempt. */
835 RTSpinlockAcquire(g_hNtProtectLock);
836 if (pNtProtect->enmProcessKind == kSupDrvNtProtectKind_VmProcessConfirmed)
837 pNtProtect->enmProcessKind = kSupDrvNtProtectKind_VmProcessDead;
838 RTSpinlockRelease(g_hNtProtectLock);
839
840 LogRel(("vboxdrv: supdrvCreateSession failed for process %p: rc=%d.\n",
841 PsGetProcessId(PsGetCurrentProcess()), rc));
842 }
843 else
844 LogRel(("vboxdrv: Process %p failed process verification: rc=%d.\n",
845 PsGetProcessId(PsGetCurrentProcess()), rc));
846 }
847 else
848 {
849 LogRel(("vboxdrv: %p is not a budding VM process (enmProcessKind=%d).\n",
850 PsGetProcessId(PsGetCurrentProcess()), pNtProtect->enmProcessKind));
851 rc = VERR_SUPDRV_NOT_BUDDING_VM_PROCESS_2;
852 }
853 supdrvNtProtectRelease(pNtProtect);
854 }
855 else
856 {
857 LogRel(("vboxdrv: %p is not a budding VM process.\n", PsGetProcessId(PsGetCurrentProcess())));
858 rc = VERR_SUPDRV_NOT_BUDDING_VM_PROCESS_1;
859 }
860 }
861 /*
862 * Call common code to create an unprivileged session.
863 */
864 else
865 {
866 PSUPDRVSESSION pSession;
867 rc = supdrvCreateSession(pDevExt, true /*fUser*/, false /*fUnrestricted*/, &pSession);
868 if (RT_SUCCESS(rc))
869 {
870 rc = supdrvSessionHashTabInsert(pDevExt, pSession, (PSUPDRVSESSION *)&pFileObj->FsContext, NULL);
871 supdrvSessionRelease(pSession);
872 if (RT_SUCCESS(rc))
873 {
874 pFileObj->FsContext = pSession; /* Keeps reference. No race. */
875 pSession->pNtProtect = NULL;
876 return supdrvNtCompleteRequestEx(STATUS_SUCCESS, FILE_OPENED, pIrp);
877 }
878 }
879 }
880
881#else /* !VBOX_WITH_HARDENING */
882 /*
883 * Call common code to create a session.
884 */
885 pFileObj->FsContext = NULL;
886 PSUPDRVSESSION pSession;
887 rc = supdrvCreateSession(pDevExt, true /*fUser*/, pDevObj == g_pDevObjSys /*fUnrestricted*/, &pSession);
888 if (RT_SUCCESS(rc))
889 {
890 rc = supdrvSessionHashTabInsert(pDevExt, pSession, (PSUPDRVSESSION *)&pFileObj->FsContext, NULL);
891 supdrvSessionRelease(pSession);
892 if (RT_SUCCESS(rc))
893 return supdrvNtCompleteRequestEx(STATUS_SUCCESS, FILE_OPENED, pIrp);
894
895 }
896#endif /* !VBOX_WITH_HARDENING */
897
898 /* bail out */
899 rcNt = VBoxDrvNtErr2NtStatus(rc);
900 }
901 }
902
903 Assert(!NT_SUCCESS(rcNt));
904 pFileObj->FsContext = NULL;
905 return supdrvNtCompleteRequest(rcNt, pIrp); /* Note. the IoStatus is completely ignored on error. */
906}
907
908
909/**
910 * Clean up file handle entry point.
911 *
912 * @param pDevObj Device object.
913 * @param pIrp Request packet.
914 */
915NTSTATUS _stdcall VBoxDrvNtCleanup(PDEVICE_OBJECT pDevObj, PIRP pIrp)
916{
917 PSUPDRVDEVEXT pDevExt = SUPDRVNT_GET_DEVEXT(pDevObj);
918 PIO_STACK_LOCATION pStack = IoGetCurrentIrpStackLocation(pIrp);
919 PFILE_OBJECT pFileObj = pStack->FileObject;
920
921#ifdef VBOX_WITH_HARDENING
922 if (pDevObj == g_pDevObjStub)
923 {
924 PSUPDRVNTPROTECT pNtProtect = (PSUPDRVNTPROTECT)pFileObj->FsContext;
925 Log(("VBoxDrvNtCleanup: pDevExt=%p pFileObj=%p pNtProtect=%p\n", pDevExt, pFileObj, pNtProtect));
926 if (pNtProtect)
927 {
928 supdrvNtProtectRelease(pNtProtect);
929 pFileObj->FsContext = NULL;
930 }
931 }
932 else if (pDevObj == g_pDevObjErrorInfo)
933 supdrvNtErrorInfoCleanupProcess(PsGetCurrentProcessId());
934 else
935#endif
936 {
937 PSUPDRVSESSION pSession = supdrvSessionHashTabLookup(pDevExt, RTProcSelf(), RTR0ProcHandleSelf(),
938 (PSUPDRVSESSION *)&pFileObj->FsContext);
939 Log(("VBoxDrvNtCleanup: pDevExt=%p pFileObj=%p pSession=%p\n", pDevExt, pFileObj, pSession));
940 if (pSession)
941 {
942 supdrvSessionHashTabRemove(pDevExt, pSession, NULL);
943 supdrvSessionRelease(pSession); /* Drops the reference from supdrvSessionHashTabLookup. */
944 }
945 }
946
947 return supdrvNtCompleteRequest(STATUS_SUCCESS, pIrp);
948}
949
950
951/**
952 * Close file entry point.
953 *
954 * @param pDevObj Device object.
955 * @param pIrp Request packet.
956 */
957NTSTATUS _stdcall VBoxDrvNtClose(PDEVICE_OBJECT pDevObj, PIRP pIrp)
958{
959 PSUPDRVDEVEXT pDevExt = SUPDRVNT_GET_DEVEXT(pDevObj);
960 PIO_STACK_LOCATION pStack = IoGetCurrentIrpStackLocation(pIrp);
961 PFILE_OBJECT pFileObj = pStack->FileObject;
962
963#ifdef VBOX_WITH_HARDENING
964 if (pDevObj == g_pDevObjStub)
965 {
966 PSUPDRVNTPROTECT pNtProtect = (PSUPDRVNTPROTECT)pFileObj->FsContext;
967 Log(("VBoxDrvNtClose: pDevExt=%p pFileObj=%p pNtProtect=%p\n", pDevExt, pFileObj, pNtProtect));
968 if (pNtProtect)
969 {
970 supdrvNtProtectRelease(pNtProtect);
971 pFileObj->FsContext = NULL;
972 }
973 }
974 else if (pDevObj == g_pDevObjErrorInfo)
975 supdrvNtErrorInfoCleanupProcess(PsGetCurrentProcessId());
976 else
977#endif
978 {
979 PSUPDRVSESSION pSession = supdrvSessionHashTabLookup(pDevExt, RTProcSelf(), RTR0ProcHandleSelf(),
980 (PSUPDRVSESSION *)&pFileObj->FsContext);
981 Log(("VBoxDrvNtCleanup: pDevExt=%p pFileObj=%p pSession=%p\n", pDevExt, pFileObj, pSession));
982 if (pSession)
983 {
984 supdrvSessionHashTabRemove(pDevExt, pSession, NULL);
985 supdrvSessionRelease(pSession); /* Drops the reference from supdrvSessionHashTabLookup. */
986 }
987 }
988
989 return supdrvNtCompleteRequest(STATUS_SUCCESS, pIrp);
990}
991
992
993#ifdef VBOXDRV_WITH_FAST_IO
994/**
995 * Fast I/O device control callback.
996 *
997 * This performs no buffering, neither on the way in or out.
998 *
999 * @returns TRUE if handled, FALSE if the normal I/O control routine should be
1000 * called.
1001 * @param pFileObj The file object.
1002 * @param fWait Whether it's a blocking call
1003 * @param pvInput The input buffer as specified by the user.
1004 * @param cbInput The size of the input buffer.
1005 * @param pvOutput The output buffer as specfied by the user.
1006 * @param cbOutput The size of the output buffer.
1007 * @param uCmd The I/O command/function being invoked.
1008 * @param pIoStatus Where to return the status of the operation.
1009 * @param pDevObj The device object..
1010 */
1011static BOOLEAN _stdcall VBoxDrvNtFastIoDeviceControl(PFILE_OBJECT pFileObj, BOOLEAN fWait, PVOID pvInput, ULONG cbInput,
1012 PVOID pvOutput, ULONG cbOutput, ULONG uCmd,
1013 PIO_STATUS_BLOCK pIoStatus, PDEVICE_OBJECT pDevObj)
1014{
1015 /*
1016 * Only the normal devices, not the stub or error info ones.
1017 */
1018 if (pDevObj != g_pDevObjSys && pDevObj != g_pDevObjUsr)
1019 {
1020 pIoStatus->Status = STATUS_NOT_SUPPORTED;
1021 pIoStatus->Information = 0;
1022 return TRUE;
1023 }
1024
1025 /*
1026 * Check the input a little bit and get a the session references.
1027 */
1028 PSUPDRVDEVEXT pDevExt = SUPDRVNT_GET_DEVEXT(pDevObj);
1029 PSUPDRVSESSION pSession = supdrvSessionHashTabLookup(pDevExt, RTProcSelf(), RTR0ProcHandleSelf(),
1030 (PSUPDRVSESSION *)&pFileObj->FsContext);
1031 if (!pSession)
1032 {
1033 pIoStatus->Status = STATUS_TRUST_FAILURE;
1034 pIoStatus->Information = 0;
1035 return TRUE;
1036 }
1037
1038 if (pSession->fUnrestricted)
1039 {
1040#if defined(VBOX_WITH_HARDENING) && !defined(VBOX_WITHOUT_DEBUGGER_CHECKS)
1041 if (supdrvNtIsDebuggerAttached())
1042 {
1043 pIoStatus->Status = STATUS_TRUST_FAILURE;
1044 pIoStatus->Information = 0;
1045 supdrvSessionRelease(pSession);
1046 return TRUE;
1047 }
1048#endif
1049
1050 /*
1051 * Deal with the 2-3 high-speed IOCtl that takes their arguments from
1052 * the session and iCmd, and does not return anything.
1053 */
1054 if ( uCmd == SUP_IOCTL_FAST_DO_RAW_RUN
1055 || uCmd == SUP_IOCTL_FAST_DO_HM_RUN
1056 || uCmd == SUP_IOCTL_FAST_DO_NOP)
1057 {
1058 int rc = supdrvIOCtlFast(uCmd, (unsigned)(uintptr_t)pvOutput/* VMCPU id */, pDevExt, pSession);
1059 pIoStatus->Status = RT_SUCCESS(rc) ? STATUS_SUCCESS : STATUS_INVALID_PARAMETER;
1060 pIoStatus->Information = 0; /* Could be used to pass rc if we liked. */
1061 supdrvSessionRelease(pSession);
1062 return TRUE;
1063 }
1064 }
1065
1066 /*
1067 * The normal path.
1068 */
1069 NTSTATUS rcNt;
1070 unsigned cbOut = 0;
1071 int rc = 0;
1072 Log2(("VBoxDrvNtFastIoDeviceControl(%p): ioctl=%#x pvIn=%p cbIn=%#x pvOut=%p cbOut=%#x pSession=%p\n",
1073 pDevExt, uCmd, pvInput, cbInput, pvOutput, cbOutput, pSession));
1074
1075# ifdef RT_ARCH_AMD64
1076 /* Don't allow 32-bit processes to do any I/O controls. */
1077 if (!IoIs32bitProcess(NULL))
1078# endif
1079 {
1080 /*
1081 * In this fast I/O device control path we have to do our own buffering.
1082 */
1083 /* Verify that the I/O control function matches our pattern. */
1084 if ((uCmd & 0x3) == METHOD_BUFFERED)
1085 {
1086 /* Get the header so we can validate it a little bit against the
1087 parameters before allocating any memory kernel for the reqest. */
1088 SUPREQHDR Hdr;
1089 if (cbInput >= sizeof(Hdr) && cbOutput >= sizeof(Hdr))
1090 {
1091 __try
1092 {
1093 RtlCopyMemory(&Hdr, pvInput, sizeof(Hdr));
1094 rcNt = STATUS_SUCCESS;
1095 }
1096 __except(EXCEPTION_EXECUTE_HANDLER)
1097 {
1098 rcNt = GetExceptionCode();
1099 }
1100 }
1101 else
1102 rcNt = STATUS_INVALID_PARAMETER;
1103 if (NT_SUCCESS(rcNt))
1104 {
1105 /* Verify that the sizes in the request header are correct. */
1106 ULONG cbBuf = RT_MAX(cbInput, cbOutput);
1107 if ( cbInput == Hdr.cbIn
1108 && cbOutput == Hdr.cbOut
1109 && cbBuf < _1M*16)
1110 {
1111 /* Allocate a buffer and copy all the input into it. */
1112 PSUPREQHDR pHdr = (PSUPREQHDR)ExAllocatePoolWithTag(NonPagedPool, cbBuf, 'VBox');
1113 if (pHdr)
1114 {
1115 __try
1116 {
1117 RtlCopyMemory(pHdr, pvInput, cbInput);
1118 if (cbInput < cbBuf)
1119 RtlZeroMemory((uint8_t *)pHdr + cbInput, cbBuf - cbInput);
1120 if (!memcmp(pHdr, &Hdr, sizeof(Hdr)))
1121 rcNt = STATUS_SUCCESS;
1122 else
1123 rcNt = STATUS_INVALID_PARAMETER;
1124 }
1125 __except(EXCEPTION_EXECUTE_HANDLER)
1126 {
1127 rcNt = GetExceptionCode();
1128 }
1129 if (NT_SUCCESS(rcNt))
1130 {
1131 /*
1132 * Now call the common code to do the real work.
1133 */
1134 rc = supdrvIOCtl(uCmd, pDevExt, pSession, pHdr, cbBuf);
1135 if (RT_SUCCESS(rc))
1136 {
1137 /*
1138 * Copy back the result.
1139 */
1140 cbOut = pHdr->cbOut;
1141 if (cbOut > cbOutput)
1142 {
1143 cbOut = cbOutput;
1144 OSDBGPRINT(("VBoxDrvNtFastIoDeviceControl: too much output! %#x > %#x; uCmd=%#x!\n",
1145 pHdr->cbOut, cbOut, uCmd));
1146 }
1147 if (cbOut)
1148 {
1149 __try
1150 {
1151 RtlCopyMemory(pvOutput, pHdr, cbOut);
1152 rcNt = STATUS_SUCCESS;
1153 }
1154 __except(EXCEPTION_EXECUTE_HANDLER)
1155 {
1156 rcNt = GetExceptionCode();
1157 }
1158 }
1159 else
1160 rcNt = STATUS_SUCCESS;
1161 }
1162 else if (rc == VERR_INVALID_PARAMETER)
1163 rcNt = STATUS_INVALID_PARAMETER;
1164 else
1165 rcNt = STATUS_NOT_SUPPORTED;
1166 Log2(("VBoxDrvNtFastIoDeviceControl: returns %#x cbOut=%d rc=%#x\n", rcNt, cbOut, rc));
1167 }
1168 else
1169 Log(("VBoxDrvNtFastIoDeviceControl: Error reading %u bytes of user memory at %p (uCmd=%#x)\n",
1170 cbInput, pvInput, uCmd));
1171 ExFreePoolWithTag(pHdr, 'VBox');
1172 }
1173 else
1174 rcNt = STATUS_NO_MEMORY;
1175 }
1176 else
1177 {
1178 Log(("VBoxDrvNtFastIoDeviceControl: Mismatching sizes (%#x) - Hdr=%#lx/%#lx Irp=%#lx/%#lx!\n",
1179 uCmd, Hdr.cbIn, Hdr.cbOut, cbInput, cbOutput));
1180 rcNt = STATUS_INVALID_PARAMETER;
1181 }
1182 }
1183 }
1184 else
1185 {
1186 Log(("VBoxDrvNtFastIoDeviceControl: not buffered request (%#x) - not supported\n", uCmd));
1187 rcNt = STATUS_NOT_SUPPORTED;
1188 }
1189 }
1190# ifdef RT_ARCH_AMD64
1191 else
1192 {
1193 Log(("VBoxDrvNtFastIoDeviceControl: WOW64 req - not supported\n"));
1194 rcNt = STATUS_NOT_SUPPORTED;
1195 }
1196# endif
1197
1198 /* complete the request. */
1199 pIoStatus->Status = rcNt;
1200 pIoStatus->Information = cbOut;
1201 supdrvSessionRelease(pSession);
1202 return TRUE; /* handled. */
1203}
1204#endif /* VBOXDRV_WITH_FAST_IO */
1205
1206
1207/**
1208 * Device I/O Control entry point.
1209 *
1210 * @param pDevObj Device object.
1211 * @param pIrp Request packet.
1212 */
1213NTSTATUS _stdcall VBoxDrvNtDeviceControl(PDEVICE_OBJECT pDevObj, PIRP pIrp)
1214{
1215 VBOXDRV_COMPLETE_IRP_AND_RETURN_IF_STUB_OR_ERROR_INFO_DEV(pDevObj, pIrp);
1216
1217 PSUPDRVDEVEXT pDevExt = SUPDRVNT_GET_DEVEXT(pDevObj);
1218 PIO_STACK_LOCATION pStack = IoGetCurrentIrpStackLocation(pIrp);
1219 PSUPDRVSESSION pSession = supdrvSessionHashTabLookup(pDevExt, RTProcSelf(), RTR0ProcHandleSelf(),
1220 (PSUPDRVSESSION *)&pStack->FileObject->FsContext);
1221
1222 if (!RT_VALID_PTR(pSession))
1223 return supdrvNtCompleteRequest(STATUS_TRUST_FAILURE, pIrp);
1224
1225 /*
1226 * Deal with the 2-3 high-speed IOCtl that takes their arguments from
1227 * the session and iCmd, and does not return anything.
1228 */
1229 if (pSession->fUnrestricted)
1230 {
1231#if defined(VBOX_WITH_HARDENING) && !defined(VBOX_WITHOUT_DEBUGGER_CHECKS)
1232 if (supdrvNtIsDebuggerAttached())
1233 {
1234 supdrvSessionRelease(pSession);
1235 return supdrvNtCompleteRequest(STATUS_TRUST_FAILURE, pIrp);
1236 }
1237#endif
1238
1239 ULONG ulCmd = pStack->Parameters.DeviceIoControl.IoControlCode;
1240 if ( ulCmd == SUP_IOCTL_FAST_DO_RAW_RUN
1241 || ulCmd == SUP_IOCTL_FAST_DO_HM_RUN
1242 || ulCmd == SUP_IOCTL_FAST_DO_NOP)
1243 {
1244 int rc = supdrvIOCtlFast(ulCmd, (unsigned)(uintptr_t)pIrp->UserBuffer /* VMCPU id */, pDevExt, pSession);
1245
1246 /* Complete the I/O request. */
1247 supdrvSessionRelease(pSession);
1248 return supdrvNtCompleteRequest(RT_SUCCESS(rc) ? STATUS_SUCCESS : STATUS_INVALID_PARAMETER, pIrp);
1249 }
1250 }
1251
1252 return VBoxDrvNtDeviceControlSlow(pDevExt, pSession, pIrp, pStack);
1253}
1254
1255
1256/**
1257 * Worker for VBoxDrvNtDeviceControl that takes the slow IOCtl functions.
1258 *
1259 * @returns NT status code.
1260 *
1261 * @param pDevExt Device extension.
1262 * @param pSession The session.
1263 * @param pIrp Request packet.
1264 * @param pStack The stack location containing the DeviceControl parameters.
1265 */
1266static int VBoxDrvNtDeviceControlSlow(PSUPDRVDEVEXT pDevExt, PSUPDRVSESSION pSession, PIRP pIrp, PIO_STACK_LOCATION pStack)
1267{
1268 NTSTATUS rcNt;
1269 unsigned cbOut = 0;
1270 int rc = 0;
1271 Log2(("VBoxDrvNtDeviceControlSlow(%p,%p): ioctl=%#x pBuf=%p cbIn=%#x cbOut=%#x pSession=%p\n",
1272 pDevExt, pIrp, pStack->Parameters.DeviceIoControl.IoControlCode,
1273 pIrp->AssociatedIrp.SystemBuffer, pStack->Parameters.DeviceIoControl.InputBufferLength,
1274 pStack->Parameters.DeviceIoControl.OutputBufferLength, pSession));
1275
1276#ifdef RT_ARCH_AMD64
1277 /* Don't allow 32-bit processes to do any I/O controls. */
1278 if (!IoIs32bitProcess(pIrp))
1279#endif
1280 {
1281 /* Verify that it's a buffered CTL. */
1282 if ((pStack->Parameters.DeviceIoControl.IoControlCode & 0x3) == METHOD_BUFFERED)
1283 {
1284 /* Verify that the sizes in the request header are correct. */
1285 PSUPREQHDR pHdr = (PSUPREQHDR)pIrp->AssociatedIrp.SystemBuffer;
1286 if ( pStack->Parameters.DeviceIoControl.InputBufferLength >= sizeof(*pHdr)
1287 && pStack->Parameters.DeviceIoControl.InputBufferLength == pHdr->cbIn
1288 && pStack->Parameters.DeviceIoControl.OutputBufferLength == pHdr->cbOut)
1289 {
1290 /* Zero extra output bytes to make sure we don't leak anything. */
1291 if (pHdr->cbIn < pHdr->cbOut)
1292 RtlZeroMemory((uint8_t *)pHdr + pHdr->cbIn, pHdr->cbOut - pHdr->cbIn);
1293
1294 /*
1295 * Do the job.
1296 */
1297 rc = supdrvIOCtl(pStack->Parameters.DeviceIoControl.IoControlCode, pDevExt, pSession, pHdr,
1298 RT_MAX(pHdr->cbIn, pHdr->cbOut));
1299 if (!rc)
1300 {
1301 rcNt = STATUS_SUCCESS;
1302 cbOut = pHdr->cbOut;
1303 if (cbOut > pStack->Parameters.DeviceIoControl.OutputBufferLength)
1304 {
1305 cbOut = pStack->Parameters.DeviceIoControl.OutputBufferLength;
1306 OSDBGPRINT(("VBoxDrvLinuxIOCtl: too much output! %#x > %#x; uCmd=%#x!\n",
1307 pHdr->cbOut, cbOut, pStack->Parameters.DeviceIoControl.IoControlCode));
1308 }
1309 }
1310 else
1311 rcNt = STATUS_INVALID_PARAMETER;
1312 Log2(("VBoxDrvNtDeviceControlSlow: returns %#x cbOut=%d rc=%#x\n", rcNt, cbOut, rc));
1313 }
1314 else
1315 {
1316 Log(("VBoxDrvNtDeviceControlSlow: Mismatching sizes (%#x) - Hdr=%#lx/%#lx Irp=%#lx/%#lx!\n",
1317 pStack->Parameters.DeviceIoControl.IoControlCode,
1318 pStack->Parameters.DeviceIoControl.InputBufferLength >= sizeof(*pHdr) ? pHdr->cbIn : 0,
1319 pStack->Parameters.DeviceIoControl.InputBufferLength >= sizeof(*pHdr) ? pHdr->cbOut : 0,
1320 pStack->Parameters.DeviceIoControl.InputBufferLength,
1321 pStack->Parameters.DeviceIoControl.OutputBufferLength));
1322 rcNt = STATUS_INVALID_PARAMETER;
1323 }
1324 }
1325 else
1326 {
1327 Log(("VBoxDrvNtDeviceControlSlow: not buffered request (%#x) - not supported\n",
1328 pStack->Parameters.DeviceIoControl.IoControlCode));
1329 rcNt = STATUS_NOT_SUPPORTED;
1330 }
1331 }
1332#ifdef RT_ARCH_AMD64
1333 else
1334 {
1335 Log(("VBoxDrvNtDeviceControlSlow: WOW64 req - not supported\n"));
1336 rcNt = STATUS_NOT_SUPPORTED;
1337 }
1338#endif
1339
1340 /* complete the request. */
1341 pIrp->IoStatus.Status = rcNt;
1342 pIrp->IoStatus.Information = cbOut;
1343 supdrvSessionRelease(pSession);
1344 IoCompleteRequest(pIrp, IO_NO_INCREMENT);
1345 return rcNt;
1346}
1347
1348
1349/**
1350 * Internal Device I/O Control entry point, used for IDC.
1351 *
1352 * @param pDevObj Device object.
1353 * @param pIrp Request packet.
1354 */
1355NTSTATUS _stdcall VBoxDrvNtInternalDeviceControl(PDEVICE_OBJECT pDevObj, PIRP pIrp)
1356{
1357 VBOXDRV_COMPLETE_IRP_AND_RETURN_IF_STUB_OR_ERROR_INFO_DEV(pDevObj, pIrp);
1358
1359 PSUPDRVDEVEXT pDevExt = SUPDRVNT_GET_DEVEXT(pDevObj);
1360 PIO_STACK_LOCATION pStack = IoGetCurrentIrpStackLocation(pIrp);
1361 PFILE_OBJECT pFileObj = pStack ? pStack->FileObject : NULL;
1362 PSUPDRVSESSION pSession = pFileObj ? (PSUPDRVSESSION)pFileObj->FsContext : NULL;
1363 NTSTATUS rcNt;
1364 unsigned cbOut = 0;
1365 int rc = 0;
1366 Log2(("VBoxDrvNtInternalDeviceControl(%p,%p): ioctl=%#x pBuf=%p cbIn=%#x cbOut=%#x pSession=%p\n",
1367 pDevExt, pIrp, pStack->Parameters.DeviceIoControl.IoControlCode,
1368 pIrp->AssociatedIrp.SystemBuffer, pStack->Parameters.DeviceIoControl.InputBufferLength,
1369 pStack->Parameters.DeviceIoControl.OutputBufferLength, pSession));
1370
1371 /* Verify that it's a buffered CTL. */
1372 if ((pStack->Parameters.DeviceIoControl.IoControlCode & 0x3) == METHOD_BUFFERED)
1373 {
1374 /* Verify the pDevExt in the session. */
1375 if ( pStack->Parameters.DeviceIoControl.IoControlCode != SUPDRV_IDC_REQ_CONNECT
1376 ? VALID_PTR(pSession) && pSession->pDevExt == pDevExt
1377 : !pSession
1378 )
1379 {
1380 /* Verify that the size in the request header is correct. */
1381 PSUPDRVIDCREQHDR pHdr = (PSUPDRVIDCREQHDR)pIrp->AssociatedIrp.SystemBuffer;
1382 if ( pStack->Parameters.DeviceIoControl.InputBufferLength >= sizeof(*pHdr)
1383 && pStack->Parameters.DeviceIoControl.InputBufferLength == pHdr->cb
1384 && pStack->Parameters.DeviceIoControl.OutputBufferLength == pHdr->cb)
1385 {
1386 /*
1387 * Call the generic code.
1388 *
1389 * Note! Connect and disconnect requires some extra attention
1390 * in order to get the session handling right.
1391 */
1392 if (pStack->Parameters.DeviceIoControl.IoControlCode == SUPDRV_IDC_REQ_DISCONNECT)
1393 pFileObj->FsContext = NULL;
1394
1395 rc = supdrvIDC(pStack->Parameters.DeviceIoControl.IoControlCode, pDevExt, pSession, pHdr);
1396 if (!rc)
1397 {
1398 if (pStack->Parameters.DeviceIoControl.IoControlCode == SUPDRV_IDC_REQ_CONNECT)
1399 pFileObj->FsContext = ((PSUPDRVIDCREQCONNECT)pHdr)->u.Out.pSession;
1400
1401 rcNt = STATUS_SUCCESS;
1402 cbOut = pHdr->cb;
1403 }
1404 else
1405 {
1406 rcNt = STATUS_INVALID_PARAMETER;
1407 if (pStack->Parameters.DeviceIoControl.IoControlCode == SUPDRV_IDC_REQ_DISCONNECT)
1408 pFileObj->FsContext = pSession;
1409 }
1410 Log2(("VBoxDrvNtInternalDeviceControl: returns %#x/rc=%#x\n", rcNt, rc));
1411 }
1412 else
1413 {
1414 Log(("VBoxDrvNtInternalDeviceControl: Mismatching sizes (%#x) - Hdr=%#lx Irp=%#lx/%#lx!\n",
1415 pStack->Parameters.DeviceIoControl.IoControlCode,
1416 pStack->Parameters.DeviceIoControl.InputBufferLength >= sizeof(*pHdr) ? pHdr->cb : 0,
1417 pStack->Parameters.DeviceIoControl.InputBufferLength,
1418 pStack->Parameters.DeviceIoControl.OutputBufferLength));
1419 rcNt = STATUS_INVALID_PARAMETER;
1420 }
1421 }
1422 else
1423 rcNt = STATUS_NOT_SUPPORTED;
1424 }
1425 else
1426 {
1427 Log(("VBoxDrvNtInternalDeviceControl: not buffered request (%#x) - not supported\n",
1428 pStack->Parameters.DeviceIoControl.IoControlCode));
1429 rcNt = STATUS_NOT_SUPPORTED;
1430 }
1431
1432 /* complete the request. */
1433 pIrp->IoStatus.Status = rcNt;
1434 pIrp->IoStatus.Information = cbOut;
1435 IoCompleteRequest(pIrp, IO_NO_INCREMENT);
1436 return rcNt;
1437}
1438
1439
1440/**
1441 * Implementation of the read major function for VBoxDrvErrorInfo.
1442 *
1443 * This is a stub function for the other devices.
1444 *
1445 * @returns NT status code.
1446 * @param pDevObj The device object.
1447 * @param pIrp The I/O request packet.
1448 */
1449NTSTATUS _stdcall VBoxDrvNtRead(PDEVICE_OBJECT pDevObj, PIRP pIrp)
1450{
1451 Log(("VBoxDrvNtRead\n"));
1452
1453 NTSTATUS rcNt;
1454 pIrp->IoStatus.Information = 0;
1455
1456#ifdef VBOX_WITH_HARDENING
1457 /*
1458 * VBoxDrvErrorInfo?
1459 */
1460 if (pDevObj == g_pDevObjErrorInfo)
1461 {
1462 PIO_STACK_LOCATION pStack = IoGetCurrentIrpStackLocation(pIrp);
1463 if ( pStack
1464 && (pIrp->Flags & IRP_BUFFERED_IO))
1465 {
1466 /*
1467 * Look up the process error information.
1468 */
1469 HANDLE hCurThreadId = PsGetCurrentThreadId();
1470 HANDLE hCurProcessId = PsGetCurrentProcessId();
1471 int rc = RTSemMutexRequestNoResume(g_hErrorInfoLock, RT_INDEFINITE_WAIT);
1472 if (RT_SUCCESS(rc))
1473 {
1474 PSUPDRVNTERRORINFO pCur;
1475 RTListForEach(&g_ErrorInfoHead, pCur, SUPDRVNTERRORINFO, ListEntry)
1476 {
1477 if ( pCur->hProcessId == hCurProcessId
1478 && pCur->hThreadId == hCurThreadId)
1479 break;
1480 }
1481
1482 /*
1483 * Did we find error info and is the caller requesting data within it?
1484 * If so, check the destination buffer and copy the data into it.
1485 */
1486 if ( pCur
1487 && pStack->Parameters.Read.ByteOffset.QuadPart < pCur->cchErrorInfo
1488 && pStack->Parameters.Read.ByteOffset.QuadPart >= 0)
1489 {
1490 PVOID pvDstBuf = pIrp->AssociatedIrp.SystemBuffer;
1491 if (pvDstBuf)
1492 {
1493 uint32_t offRead = (uint32_t)pStack->Parameters.Read.ByteOffset.QuadPart;
1494 uint32_t cbToRead = pCur->cchErrorInfo - offRead;
1495 if (cbToRead < pStack->Parameters.Read.Length)
1496 RT_BZERO((uint8_t *)pvDstBuf + cbToRead, pStack->Parameters.Read.Length - cbToRead);
1497 else
1498 cbToRead = pStack->Parameters.Read.Length;
1499 memcpy(pvDstBuf, &pCur->szErrorInfo[offRead], cbToRead);
1500 pIrp->IoStatus.Information = cbToRead;
1501
1502 rcNt = STATUS_SUCCESS;
1503 }
1504 else
1505 rcNt = STATUS_INVALID_ADDRESS;
1506 }
1507 /*
1508 * End of file. Free the info.
1509 */
1510 else if (pCur)
1511 {
1512 RTListNodeRemove(&pCur->ListEntry);
1513 RTMemFree(pCur);
1514 rcNt = STATUS_END_OF_FILE;
1515 }
1516 /*
1517 * We found no error info. Return EOF.
1518 */
1519 else
1520 rcNt = STATUS_END_OF_FILE;
1521
1522 RTSemMutexRelease(g_hErrorInfoLock);
1523 }
1524 else
1525 rcNt = STATUS_UNSUCCESSFUL;
1526
1527 /* Paranoia: Clear the buffer on failure. */
1528 if (!NT_SUCCESS(rcNt))
1529 {
1530 PVOID pvDstBuf = pIrp->AssociatedIrp.SystemBuffer;
1531 if ( pvDstBuf
1532 && pStack->Parameters.Read.Length)
1533 RT_BZERO(pvDstBuf, pStack->Parameters.Read.Length);
1534 }
1535 }
1536 else
1537 rcNt = STATUS_INVALID_PARAMETER;
1538 }
1539 else
1540#endif /* VBOX_WITH_HARDENING */
1541 {
1542 /*
1543 * Stub.
1544 */
1545 rcNt = STATUS_NOT_SUPPORTED;
1546 }
1547
1548 /*
1549 * Complete the request.
1550 */
1551 pIrp->IoStatus.Status = rcNt;
1552 IoCompleteRequest(pIrp, IO_NO_INCREMENT);
1553 return rcNt;
1554}
1555
1556
1557/**
1558 * Stub function for functions we don't implemented.
1559 *
1560 * @returns STATUS_NOT_SUPPORTED
1561 * @param pDevObj Device object.
1562 * @param pIrp IRP.
1563 */
1564NTSTATUS _stdcall VBoxDrvNtNotSupportedStub(PDEVICE_OBJECT pDevObj, PIRP pIrp)
1565{
1566 Log(("VBoxDrvNtNotSupportedStub\n"));
1567 NOREF(pDevObj);
1568
1569 pIrp->IoStatus.Information = 0;
1570 pIrp->IoStatus.Status = STATUS_NOT_SUPPORTED;
1571 IoCompleteRequest(pIrp, IO_NO_INCREMENT);
1572
1573 return STATUS_NOT_SUPPORTED;
1574}
1575
1576
1577/**
1578 * ExRegisterCallback handler for power events
1579 *
1580 * @param pCallbackContext User supplied parameter (pDevObj)
1581 * @param pArgument1 First argument
1582 * @param pArgument2 Second argument
1583 */
1584VOID _stdcall VBoxPowerDispatchCallback(PVOID pCallbackContext, PVOID pArgument1, PVOID pArgument2)
1585{
1586 PDEVICE_OBJECT pDevObj = (PDEVICE_OBJECT)pCallbackContext;
1587
1588 Log(("VBoxPowerDispatchCallback: %x %x\n", pArgument1, pArgument2));
1589
1590 /* Power change imminent? */
1591 if ((unsigned)pArgument1 == PO_CB_SYSTEM_STATE_LOCK)
1592 {
1593 if ((unsigned)pArgument2 == 0)
1594 Log(("VBoxPowerDispatchCallback: about to go into suspend mode!\n"));
1595 else
1596 Log(("VBoxPowerDispatchCallback: resumed!\n"));
1597
1598 /* Inform any clients that have registered themselves with IPRT. */
1599 RTPowerSignalEvent(((unsigned)pArgument2 == 0) ? RTPOWEREVENT_SUSPEND : RTPOWEREVENT_RESUME);
1600 }
1601}
1602
1603
1604/**
1605 * Called to clean up the session structure before it's freed.
1606 *
1607 * @param pDevExt The device globals.
1608 * @param pSession The session that's being cleaned up.
1609 */
1610void VBOXCALL supdrvOSCleanupSession(PSUPDRVDEVEXT pDevExt, PSUPDRVSESSION pSession)
1611{
1612#ifdef VBOX_WITH_HARDENING
1613 if (pSession->pNtProtect)
1614 {
1615 supdrvNtProtectRelease(pSession->pNtProtect);
1616 pSession->pNtProtect = NULL;
1617 }
1618#endif
1619}
1620
1621
1622void VBOXCALL supdrvOSSessionHashTabInserted(PSUPDRVDEVEXT pDevExt, PSUPDRVSESSION pSession, void *pvUser)
1623{
1624 NOREF(pDevExt); NOREF(pSession); NOREF(pvUser);
1625}
1626
1627
1628void VBOXCALL supdrvOSSessionHashTabRemoved(PSUPDRVDEVEXT pDevExt, PSUPDRVSESSION pSession, void *pvUser)
1629{
1630 NOREF(pDevExt); NOREF(pSession); NOREF(pvUser);
1631}
1632
1633
1634/**
1635 * Initializes any OS specific object creator fields.
1636 */
1637void VBOXCALL supdrvOSObjInitCreator(PSUPDRVOBJ pObj, PSUPDRVSESSION pSession)
1638{
1639 NOREF(pObj);
1640 NOREF(pSession);
1641}
1642
1643
1644/**
1645 * Checks if the session can access the object.
1646 *
1647 * @returns true if a decision has been made.
1648 * @returns false if the default access policy should be applied.
1649 *
1650 * @param pObj The object in question.
1651 * @param pSession The session wanting to access the object.
1652 * @param pszObjName The object name, can be NULL.
1653 * @param prc Where to store the result when returning true.
1654 */
1655bool VBOXCALL supdrvOSObjCanAccess(PSUPDRVOBJ pObj, PSUPDRVSESSION pSession, const char *pszObjName, int *prc)
1656{
1657 NOREF(pObj);
1658 NOREF(pSession);
1659 NOREF(pszObjName);
1660 NOREF(prc);
1661 return false;
1662}
1663
1664
1665/**
1666 * Force async tsc mode (stub).
1667 */
1668bool VBOXCALL supdrvOSGetForcedAsyncTscMode(PSUPDRVDEVEXT pDevExt)
1669{
1670 return false;
1671}
1672
1673
1674/**
1675 * Whether the host takes CPUs offline during a suspend/resume operation.
1676 */
1677bool VBOXCALL supdrvOSAreCpusOfflinedOnSuspend(void)
1678{
1679 return false;
1680}
1681
1682
1683/**
1684 * Whether the hardware TSC has been synchronized by the OS.
1685 */
1686bool VBOXCALL supdrvOSAreTscDeltasInSync(void)
1687{
1688 /* If IPRT didn't find KeIpiGenericCall we pretend windows(, the firmware,
1689 or whoever) always configures TSCs perfectly. */
1690 return !RTMpOnPairIsConcurrentExecSupported();
1691}
1692
1693
1694#define MY_SystemLoadGdiDriverInSystemSpaceInformation 54
1695#define MY_SystemUnloadGdiDriverInformation 27
1696
1697typedef struct MYSYSTEMGDIDRIVERINFO
1698{
1699 UNICODE_STRING Name; /**< In: image file name. */
1700 PVOID ImageAddress; /**< Out: the load address. */
1701 PVOID SectionPointer; /**< Out: section object. */
1702 PVOID EntryPointer; /**< Out: entry point address. */
1703 PVOID ExportSectionPointer; /**< Out: export directory/section. */
1704 ULONG ImageLength; /**< Out: SizeOfImage. */
1705} MYSYSTEMGDIDRIVERINFO;
1706
1707extern "C" __declspec(dllimport) NTSTATUS NTAPI ZwSetSystemInformation(ULONG, PVOID, ULONG);
1708
1709int VBOXCALL supdrvOSLdrOpen(PSUPDRVDEVEXT pDevExt, PSUPDRVLDRIMAGE pImage, const char *pszFilename)
1710{
1711 pImage->pvNtSectionObj = NULL;
1712 pImage->hMemLock = NIL_RTR0MEMOBJ;
1713
1714#ifdef VBOX_WITHOUT_NATIVE_R0_LOADER
1715# ifndef RT_ARCH_X86
1716# error "VBOX_WITHOUT_NATIVE_R0_LOADER is only safe on x86."
1717# endif
1718 NOREF(pDevExt); NOREF(pszFilename); NOREF(pImage);
1719 return VERR_NOT_SUPPORTED;
1720
1721#else
1722 /*
1723 * Convert the filename from DOS UTF-8 to NT UTF-16.
1724 */
1725 size_t cwcFilename;
1726 int rc = RTStrCalcUtf16LenEx(pszFilename, RTSTR_MAX, &cwcFilename);
1727 if (RT_FAILURE(rc))
1728 return rc;
1729
1730 PRTUTF16 pwcsFilename = (PRTUTF16)RTMemTmpAlloc((4 + cwcFilename + 1) * sizeof(RTUTF16));
1731 if (!pwcsFilename)
1732 return VERR_NO_TMP_MEMORY;
1733
1734 pwcsFilename[0] = '\\';
1735 pwcsFilename[1] = '?';
1736 pwcsFilename[2] = '?';
1737 pwcsFilename[3] = '\\';
1738 PRTUTF16 pwcsTmp = &pwcsFilename[4];
1739 rc = RTStrToUtf16Ex(pszFilename, RTSTR_MAX, &pwcsTmp, cwcFilename + 1, NULL);
1740 if (RT_SUCCESS(rc))
1741 {
1742 /*
1743 * Try load it.
1744 */
1745 MYSYSTEMGDIDRIVERINFO Info;
1746 RtlInitUnicodeString(&Info.Name, pwcsFilename);
1747 Info.ImageAddress = NULL;
1748 Info.SectionPointer = NULL;
1749 Info.EntryPointer = NULL;
1750 Info.ExportSectionPointer = NULL;
1751 Info.ImageLength = 0;
1752
1753 NTSTATUS rcNt = ZwSetSystemInformation(MY_SystemLoadGdiDriverInSystemSpaceInformation, &Info, sizeof(Info));
1754 if (NT_SUCCESS(rcNt))
1755 {
1756 pImage->pvImage = Info.ImageAddress;
1757 pImage->pvNtSectionObj = Info.SectionPointer;
1758 Log(("ImageAddress=%p SectionPointer=%p ImageLength=%#x cbImageBits=%#x rcNt=%#x '%ls'\n",
1759 Info.ImageAddress, Info.SectionPointer, Info.ImageLength, pImage->cbImageBits, rcNt, Info.Name.Buffer));
1760# ifdef DEBUG_bird
1761 SUPR0Printf("ImageAddress=%p SectionPointer=%p ImageLength=%#x cbImageBits=%#x rcNt=%#x '%ws'\n",
1762 Info.ImageAddress, Info.SectionPointer, Info.ImageLength, pImage->cbImageBits, rcNt, Info.Name.Buffer);
1763# endif
1764 if (pImage->cbImageBits == Info.ImageLength)
1765 {
1766 /*
1767 * Lock down the entire image, just to be on the safe side.
1768 */
1769 rc = RTR0MemObjLockKernel(&pImage->hMemLock, pImage->pvImage, pImage->cbImageBits, RTMEM_PROT_READ);
1770 if (RT_FAILURE(rc))
1771 {
1772 pImage->hMemLock = NIL_RTR0MEMOBJ;
1773 supdrvOSLdrUnload(pDevExt, pImage);
1774 }
1775 }
1776 else
1777 {
1778 supdrvOSLdrUnload(pDevExt, pImage);
1779 rc = VERR_LDR_MISMATCH_NATIVE;
1780 }
1781 }
1782 else
1783 {
1784 Log(("rcNt=%#x '%ls'\n", rcNt, pwcsFilename));
1785 SUPR0Printf("VBoxDrv: rcNt=%x '%ws'\n", rcNt, pwcsFilename);
1786 switch (rcNt)
1787 {
1788 case /* 0xc0000003 */ STATUS_INVALID_INFO_CLASS:
1789# ifdef RT_ARCH_AMD64
1790 /* Unwind will crash and BSOD, so no fallback here! */
1791 rc = VERR_NOT_IMPLEMENTED;
1792# else
1793 /*
1794 * Use the old way of loading the modules.
1795 *
1796 * Note! We do *NOT* try class 26 because it will probably
1797 * not work correctly on terminal servers and such.
1798 */
1799 rc = VERR_NOT_SUPPORTED;
1800# endif
1801 break;
1802 case /* 0xc0000034 */ STATUS_OBJECT_NAME_NOT_FOUND:
1803 rc = VERR_MODULE_NOT_FOUND;
1804 break;
1805 case /* 0xC0000263 */ STATUS_DRIVER_ENTRYPOINT_NOT_FOUND:
1806 rc = VERR_LDR_IMPORTED_SYMBOL_NOT_FOUND;
1807 break;
1808 case /* 0xC0000428 */ STATUS_INVALID_IMAGE_HASH:
1809 rc = VERR_LDR_IMAGE_HASH;
1810 break;
1811 case /* 0xC000010E */ STATUS_IMAGE_ALREADY_LOADED:
1812 Log(("WARNING: see @bugref{4853} for cause of this failure on Windows 7 x64\n"));
1813 rc = VERR_ALREADY_LOADED;
1814 break;
1815 default:
1816 rc = VERR_LDR_GENERAL_FAILURE;
1817 break;
1818 }
1819
1820 pImage->pvNtSectionObj = NULL;
1821 }
1822 }
1823
1824 RTMemTmpFree(pwcsFilename);
1825 NOREF(pDevExt);
1826 return rc;
1827#endif
1828}
1829
1830
1831void VBOXCALL supdrvOSLdrNotifyOpened(PSUPDRVDEVEXT pDevExt, PSUPDRVLDRIMAGE pImage, const char *pszFilename)
1832{
1833 NOREF(pDevExt); NOREF(pImage); NOREF(pszFilename);
1834}
1835
1836void VBOXCALL supdrvOSLdrNotifyUnloaded(PSUPDRVDEVEXT pDevExt, PSUPDRVLDRIMAGE pImage)
1837{
1838 NOREF(pDevExt); NOREF(pImage);
1839}
1840
1841int VBOXCALL supdrvOSLdrValidatePointer(PSUPDRVDEVEXT pDevExt, PSUPDRVLDRIMAGE pImage, void *pv, const uint8_t *pbImageBits)
1842{
1843 NOREF(pDevExt); NOREF(pImage); NOREF(pv); NOREF(pbImageBits);
1844 return VINF_SUCCESS;
1845}
1846
1847
1848/**
1849 * memcmp + errormsg + log.
1850 *
1851 * @returns Same as memcmp.
1852 * @param pImage The image.
1853 * @param pbImageBits The image bits ring-3 uploads.
1854 * @param uRva The RVA to start comparing at.
1855 * @param cb The number of bytes to compare.
1856 * @param pReq The load request.
1857 */
1858static int supdrvNtCompare(PSUPDRVLDRIMAGE pImage, const uint8_t *pbImageBits, uint32_t uRva, uint32_t cb, PSUPLDRLOAD pReq)
1859{
1860 int iDiff = memcmp((uint8_t const *)pImage->pvImage + uRva, pbImageBits + uRva, cb);
1861 if (iDiff)
1862 {
1863 uint32_t cbLeft = cb;
1864 const uint8_t *pbNativeBits = (const uint8_t *)pImage->pvImage;
1865 for (size_t off = uRva; cbLeft > 0; off++, cbLeft--)
1866 if (pbNativeBits[off] != pbImageBits[off])
1867 {
1868 /* Note! We need to copy image bits into a temporary stack buffer here as we'd
1869 otherwise risk overwriting them while formatting the error message. */
1870 uint8_t abBytes[64];
1871 memcpy(abBytes, &pbImageBits[off], RT_MIN(64, cbLeft));
1872 supdrvLdrLoadError(VERR_LDR_MISMATCH_NATIVE, pReq,
1873 "Mismatch at %#x (%p) of %s loaded at %p:\n"
1874 "ntld: %.*Rhxs\n"
1875 "iprt: %.*Rhxs",
1876 off, &pbNativeBits[off], pImage->szName, pImage->pvImage,
1877 RT_MIN(64, cbLeft), &pbNativeBits[off],
1878 RT_MIN(64, cbLeft), &abBytes[0]);
1879 SUPR0Printf("VBoxDrv: %s", pReq->u.Out.szError);
1880 break;
1881 }
1882 }
1883 return iDiff;
1884}
1885
1886
1887int VBOXCALL supdrvOSLdrLoad(PSUPDRVDEVEXT pDevExt, PSUPDRVLDRIMAGE pImage, const uint8_t *pbImageBits, PSUPLDRLOAD pReq)
1888{
1889 NOREF(pDevExt);
1890 if (pImage->pvNtSectionObj)
1891 {
1892 /*
1893 * Usually, the entire image matches exactly.
1894 */
1895 if (!memcmp(pImage->pvImage, pbImageBits, pImage->cbImageBits))
1896 return VINF_SUCCESS;
1897
1898 /*
1899 * On Windows 10 the ImageBase member of the optional header is sometimes
1900 * updated with the actual load address and sometimes not. Try compare
1901 *
1902 */
1903 uint32_t const offNtHdrs = *(uint16_t *)pbImageBits == IMAGE_DOS_SIGNATURE
1904 ? ((IMAGE_DOS_HEADER const *)pbImageBits)->e_lfanew
1905 : 0;
1906 AssertLogRelReturn(offNtHdrs + sizeof(IMAGE_NT_HEADERS) < pImage->cbImageBits, VERR_INTERNAL_ERROR_5);
1907 IMAGE_NT_HEADERS const *pNtHdrsIprt = (IMAGE_NT_HEADERS const *)(pbImageBits + offNtHdrs);
1908 IMAGE_NT_HEADERS const *pNtHdrsNtLd = (IMAGE_NT_HEADERS const *)((uintptr_t)pImage->pvImage + offNtHdrs);
1909
1910 uint32_t const offImageBase = offNtHdrs + RT_OFFSETOF(IMAGE_NT_HEADERS, OptionalHeader.ImageBase);
1911 uint32_t const cbImageBase = RT_SIZEOFMEMB(IMAGE_NT_HEADERS, OptionalHeader.ImageBase);
1912 if ( pNtHdrsNtLd->OptionalHeader.ImageBase != pNtHdrsIprt->OptionalHeader.ImageBase
1913 && ( pNtHdrsNtLd->OptionalHeader.ImageBase == (uintptr_t)pImage->pvImage
1914 || pNtHdrsIprt->OptionalHeader.ImageBase == (uintptr_t)pImage->pvImage)
1915 && pNtHdrsIprt->Signature == IMAGE_NT_SIGNATURE
1916 && pNtHdrsIprt->OptionalHeader.Magic == IMAGE_NT_OPTIONAL_HDR_MAGIC
1917 && !memcmp(pImage->pvImage, pbImageBits, offImageBase)
1918 && !memcmp((uint8_t const *)pImage->pvImage + offImageBase + cbImageBase,
1919 pbImageBits + offImageBase + cbImageBase,
1920 pImage->cbImageBits - offImageBase - cbImageBase))
1921 return VINF_SUCCESS;
1922
1923 /*
1924 * On Windows Server 2003 (sp2 x86) both import thunk tables are fixed
1925 * up and we typically get a mismatch in the INIT section.
1926 *
1927 * So, lets see if everything matches when excluding the
1928 * OriginalFirstThunk tables and (maybe) the ImageBase member.
1929 * For simplicity the max number of exclusion regions is set to 16.
1930 */
1931 if ( pNtHdrsIprt->Signature == IMAGE_NT_SIGNATURE
1932 && pNtHdrsIprt->OptionalHeader.Magic == IMAGE_NT_OPTIONAL_HDR_MAGIC
1933 && pNtHdrsIprt->OptionalHeader.NumberOfRvaAndSizes > IMAGE_DIRECTORY_ENTRY_IMPORT
1934 && pNtHdrsIprt->OptionalHeader.DataDirectory[IMAGE_DIRECTORY_ENTRY_IMPORT].Size >= sizeof(IMAGE_IMPORT_DESCRIPTOR)
1935 && pNtHdrsIprt->OptionalHeader.DataDirectory[IMAGE_DIRECTORY_ENTRY_IMPORT].VirtualAddress > sizeof(IMAGE_NT_HEADERS)
1936 && pNtHdrsIprt->OptionalHeader.DataDirectory[IMAGE_DIRECTORY_ENTRY_IMPORT].VirtualAddress < pImage->cbImageBits
1937 )
1938 {
1939 struct MyRegion
1940 {
1941 uint32_t uRva;
1942 uint32_t cb;
1943 } aExcludeRgns[16];
1944 unsigned cExcludeRgns = 0;
1945
1946 /* ImageBase: */
1947 if ( pNtHdrsNtLd->OptionalHeader.ImageBase != pNtHdrsIprt->OptionalHeader.ImageBase
1948 && ( pNtHdrsNtLd->OptionalHeader.ImageBase == (uintptr_t)pImage->pvImage
1949 || pNtHdrsIprt->OptionalHeader.ImageBase == (uintptr_t)pImage->pvImage) )
1950 {
1951 aExcludeRgns[cExcludeRgns].uRva = offImageBase;
1952 aExcludeRgns[cExcludeRgns].cb = cbImageBase;
1953 cExcludeRgns++;
1954 }
1955
1956 /* Imports: */
1957 uint32_t cImpsLeft = pNtHdrsIprt->OptionalHeader.DataDirectory[IMAGE_DIRECTORY_ENTRY_IMPORT].Size
1958 / sizeof(IMAGE_IMPORT_DESCRIPTOR);
1959 uint32_t offImps = pNtHdrsIprt->OptionalHeader.DataDirectory[IMAGE_DIRECTORY_ENTRY_IMPORT].VirtualAddress;
1960 AssertLogRelReturn(offImps + cImpsLeft * sizeof(IMAGE_IMPORT_DESCRIPTOR) <= pImage->cbImageBits, VERR_INTERNAL_ERROR_3);
1961 IMAGE_IMPORT_DESCRIPTOR const *pImp = (IMAGE_IMPORT_DESCRIPTOR const *)(pbImageBits + offImps);
1962 while ( cImpsLeft-- > 0
1963 && cExcludeRgns < RT_ELEMENTS(aExcludeRgns))
1964 {
1965 uint32_t uRvaThunk = pImp->OriginalFirstThunk;
1966 if ( uRvaThunk > sizeof(IMAGE_NT_HEADERS)
1967 && uRvaThunk <= pImage->cbImageBits - sizeof(IMAGE_THUNK_DATA)
1968 && uRvaThunk != pImp->FirstThunk)
1969 {
1970 /* Find the size of the thunk table. */
1971 IMAGE_THUNK_DATA const *paThunk = (IMAGE_THUNK_DATA const *)(pbImageBits + uRvaThunk);
1972 uint32_t cMaxThunks = (pImage->cbImageBits - uRvaThunk) / sizeof(IMAGE_THUNK_DATA);
1973 uint32_t cThunks = 0;
1974 while (cThunks < cMaxThunks && paThunk[cThunks].u1.Function != 0)
1975 cThunks++;
1976
1977 /* Ordered table insert. */
1978 unsigned i = 0;
1979 for (; i < cExcludeRgns; i++)
1980 if (uRvaThunk < aExcludeRgns[i].uRva)
1981 break;
1982 if (i != cExcludeRgns)
1983 memmove(&aExcludeRgns[i + 1], &aExcludeRgns[i], (cExcludeRgns - i) * sizeof(aExcludeRgns[0]));
1984 aExcludeRgns[i].uRva = uRvaThunk;
1985 aExcludeRgns[i].cb = cThunks * sizeof(IMAGE_THUNK_DATA);
1986 cExcludeRgns++;
1987 }
1988
1989 /* advance */
1990 pImp++;
1991 }
1992
1993 /*
1994 * Ok, do the comparison.
1995 */
1996 int iDiff = 0;
1997 uint32_t uRvaNext = 0;
1998 for (unsigned i = 0; !iDiff && i < cExcludeRgns; i++)
1999 {
2000 if (uRvaNext < aExcludeRgns[i].uRva)
2001 iDiff = supdrvNtCompare(pImage, pbImageBits, uRvaNext, aExcludeRgns[i].uRva - uRvaNext, pReq);
2002 uRvaNext = aExcludeRgns[i].uRva + aExcludeRgns[i].cb;
2003 }
2004 if (!iDiff && uRvaNext < pImage->cbImageBits)
2005 iDiff = supdrvNtCompare(pImage, pbImageBits, uRvaNext, pImage->cbImageBits - uRvaNext, pReq);
2006 if (!iDiff)
2007 return VINF_SUCCESS;
2008 }
2009 else
2010 supdrvNtCompare(pImage, pbImageBits, 0, pImage->cbImageBits, pReq);
2011 return VERR_LDR_MISMATCH_NATIVE;
2012 }
2013 return supdrvLdrLoadError(VERR_INTERNAL_ERROR_4, pReq, "No NT section object! Impossible!");
2014}
2015
2016
2017void VBOXCALL supdrvOSLdrUnload(PSUPDRVDEVEXT pDevExt, PSUPDRVLDRIMAGE pImage)
2018{
2019 if (pImage->pvNtSectionObj)
2020 {
2021 if (pImage->hMemLock != NIL_RTR0MEMOBJ)
2022 {
2023 RTR0MemObjFree(pImage->hMemLock, false /*fFreeMappings*/);
2024 pImage->hMemLock = NIL_RTR0MEMOBJ;
2025 }
2026
2027 NTSTATUS rcNt = ZwSetSystemInformation(MY_SystemUnloadGdiDriverInformation,
2028 &pImage->pvNtSectionObj, sizeof(pImage->pvNtSectionObj));
2029 if (rcNt != STATUS_SUCCESS)
2030 SUPR0Printf("VBoxDrv: failed to unload '%s', rcNt=%#x\n", pImage->szName, rcNt);
2031 pImage->pvNtSectionObj = NULL;
2032 }
2033 NOREF(pDevExt);
2034}
2035
2036
2037#ifdef SUPDRV_WITH_MSR_PROBER
2038
2039#if 1
2040/** @todo make this selectable. */
2041# define AMD_MSR_PASSCODE 0x9c5a203a
2042#else
2043# define ASMRdMsrEx(a, b, c) ASMRdMsr(a)
2044# define ASMWrMsrEx(a, b, c) ASMWrMsr(a,c)
2045#endif
2046
2047
2048/**
2049 * Argument package used by supdrvOSMsrProberRead and supdrvOSMsrProberWrite.
2050 */
2051typedef struct SUPDRVNTMSPROBERARGS
2052{
2053 uint32_t uMsr;
2054 uint64_t uValue;
2055 bool fGp;
2056} SUPDRVNTMSPROBERARGS;
2057
2058/** @callback_method_impl{FNRTMPWORKER, Worker for supdrvOSMsrProberRead.} */
2059static DECLCALLBACK(void) supdrvNtMsProberReadOnCpu(RTCPUID idCpu, void *pvUser1, void *pvUser2)
2060{
2061 /*
2062 * rdmsr and wrmsr faults can be caught even with interrupts disabled.
2063 * (At least on 32-bit XP.)
2064 */
2065 SUPDRVNTMSPROBERARGS *pArgs = (SUPDRVNTMSPROBERARGS *)pvUser1; NOREF(idCpu); NOREF(pvUser2);
2066 RTCCUINTREG fOldFlags = ASMIntDisableFlags();
2067 __try
2068 {
2069 pArgs->uValue = ASMRdMsrEx(pArgs->uMsr, AMD_MSR_PASSCODE);
2070 pArgs->fGp = false;
2071 }
2072 __except(EXCEPTION_EXECUTE_HANDLER)
2073 {
2074 pArgs->fGp = true;
2075 pArgs->uValue = 0;
2076 }
2077 ASMSetFlags(fOldFlags);
2078}
2079
2080
2081int VBOXCALL supdrvOSMsrProberRead(uint32_t uMsr, RTCPUID idCpu, uint64_t *puValue)
2082{
2083 SUPDRVNTMSPROBERARGS Args;
2084 Args.uMsr = uMsr;
2085 Args.uValue = 0;
2086 Args.fGp = true;
2087
2088 if (idCpu == NIL_RTCPUID)
2089 supdrvNtMsProberReadOnCpu(idCpu, &Args, NULL);
2090 else
2091 {
2092 int rc = RTMpOnSpecific(idCpu, supdrvNtMsProberReadOnCpu, &Args, NULL);
2093 if (RT_FAILURE(rc))
2094 return rc;
2095 }
2096
2097 if (Args.fGp)
2098 return VERR_ACCESS_DENIED;
2099 *puValue = Args.uValue;
2100 return VINF_SUCCESS;
2101}
2102
2103
2104/** @callback_method_impl{FNRTMPWORKER, Worker for supdrvOSMsrProberWrite.} */
2105static DECLCALLBACK(void) supdrvNtMsProberWriteOnCpu(RTCPUID idCpu, void *pvUser1, void *pvUser2)
2106{
2107 /*
2108 * rdmsr and wrmsr faults can be caught even with interrupts disabled.
2109 * (At least on 32-bit XP.)
2110 */
2111 SUPDRVNTMSPROBERARGS *pArgs = (SUPDRVNTMSPROBERARGS *)pvUser1; NOREF(idCpu); NOREF(pvUser2);
2112 RTCCUINTREG fOldFlags = ASMIntDisableFlags();
2113 __try
2114 {
2115 ASMWrMsrEx(pArgs->uMsr, AMD_MSR_PASSCODE, pArgs->uValue);
2116 pArgs->fGp = false;
2117 }
2118 __except(EXCEPTION_EXECUTE_HANDLER)
2119 {
2120 pArgs->fGp = true;
2121 }
2122 ASMSetFlags(fOldFlags);
2123}
2124
2125int VBOXCALL supdrvOSMsrProberWrite(uint32_t uMsr, RTCPUID idCpu, uint64_t uValue)
2126{
2127 SUPDRVNTMSPROBERARGS Args;
2128 Args.uMsr = uMsr;
2129 Args.uValue = uValue;
2130 Args.fGp = true;
2131
2132 if (idCpu == NIL_RTCPUID)
2133 supdrvNtMsProberReadOnCpu(idCpu, &Args, NULL);
2134 else
2135 {
2136 int rc = RTMpOnSpecific(idCpu, supdrvNtMsProberReadOnCpu, &Args, NULL);
2137 if (RT_FAILURE(rc))
2138 return rc;
2139 }
2140
2141 if (Args.fGp)
2142 return VERR_ACCESS_DENIED;
2143 return VINF_SUCCESS;
2144}
2145
2146/** @callback_method_impl{FNRTMPWORKER, Worker for supdrvOSMsrProberModify.} */
2147static DECLCALLBACK(void) supdrvNtMsProberModifyOnCpu(RTCPUID idCpu, void *pvUser1, void *pvUser2)
2148{
2149 PSUPMSRPROBER pReq = (PSUPMSRPROBER)pvUser1;
2150 register uint32_t uMsr = pReq->u.In.uMsr;
2151 bool const fFaster = pReq->u.In.enmOp == SUPMSRPROBEROP_MODIFY_FASTER;
2152 uint64_t uBefore = 0;
2153 uint64_t uWritten = 0;
2154 uint64_t uAfter = 0;
2155 bool fBeforeGp = true;
2156 bool fModifyGp = true;
2157 bool fAfterGp = true;
2158 bool fRestoreGp = true;
2159 RTCCUINTREG fOldFlags;
2160
2161 /*
2162 * Do the job.
2163 */
2164 fOldFlags = ASMIntDisableFlags();
2165 ASMCompilerBarrier(); /* paranoia */
2166 if (!fFaster)
2167 ASMWriteBackAndInvalidateCaches();
2168
2169 __try
2170 {
2171 uBefore = ASMRdMsrEx(uMsr, AMD_MSR_PASSCODE);
2172 fBeforeGp = false;
2173 }
2174 __except(EXCEPTION_EXECUTE_HANDLER)
2175 {
2176 fBeforeGp = true;
2177 }
2178 if (!fBeforeGp)
2179 {
2180 register uint64_t uRestore = uBefore;
2181
2182 /* Modify. */
2183 uWritten = uRestore;
2184 uWritten &= pReq->u.In.uArgs.Modify.fAndMask;
2185 uWritten |= pReq->u.In.uArgs.Modify.fOrMask;
2186 __try
2187 {
2188 ASMWrMsrEx(uMsr, AMD_MSR_PASSCODE, uWritten);
2189 fModifyGp = false;
2190 }
2191 __except(EXCEPTION_EXECUTE_HANDLER)
2192 {
2193 fModifyGp = true;
2194 }
2195
2196 /* Read modified value. */
2197 __try
2198 {
2199 uAfter = ASMRdMsrEx(uMsr, AMD_MSR_PASSCODE);
2200 fAfterGp = false;
2201 }
2202 __except(EXCEPTION_EXECUTE_HANDLER)
2203 {
2204 fAfterGp = true;
2205 }
2206
2207 /* Restore original value. */
2208 __try
2209 {
2210 ASMWrMsrEx(uMsr, AMD_MSR_PASSCODE, uRestore);
2211 fRestoreGp = false;
2212 }
2213 __except(EXCEPTION_EXECUTE_HANDLER)
2214 {
2215 fRestoreGp = true;
2216 }
2217
2218 /* Invalid everything we can. */
2219 if (!fFaster)
2220 {
2221 ASMWriteBackAndInvalidateCaches();
2222 ASMReloadCR3();
2223 ASMNopPause();
2224 }
2225 }
2226
2227 ASMCompilerBarrier(); /* paranoia */
2228 ASMSetFlags(fOldFlags);
2229
2230 /*
2231 * Write out the results.
2232 */
2233 pReq->u.Out.uResults.Modify.uBefore = uBefore;
2234 pReq->u.Out.uResults.Modify.uWritten = uWritten;
2235 pReq->u.Out.uResults.Modify.uAfter = uAfter;
2236 pReq->u.Out.uResults.Modify.fBeforeGp = fBeforeGp;
2237 pReq->u.Out.uResults.Modify.fModifyGp = fModifyGp;
2238 pReq->u.Out.uResults.Modify.fAfterGp = fAfterGp;
2239 pReq->u.Out.uResults.Modify.fRestoreGp = fRestoreGp;
2240 RT_ZERO(pReq->u.Out.uResults.Modify.afReserved);
2241}
2242
2243
2244int VBOXCALL supdrvOSMsrProberModify(RTCPUID idCpu, PSUPMSRPROBER pReq)
2245{
2246 if (idCpu == NIL_RTCPUID)
2247 {
2248 supdrvNtMsProberModifyOnCpu(idCpu, pReq, NULL);
2249 return VINF_SUCCESS;
2250 }
2251 return RTMpOnSpecific(idCpu, supdrvNtMsProberModifyOnCpu, pReq, NULL);
2252}
2253
2254#endif /* SUPDRV_WITH_MSR_PROBER */
2255
2256
2257/**
2258 * Converts an IPRT error code to an nt status code.
2259 *
2260 * @returns corresponding nt status code.
2261 * @param rc IPRT error status code.
2262 */
2263static NTSTATUS VBoxDrvNtErr2NtStatus(int rc)
2264{
2265 switch (rc)
2266 {
2267 case VINF_SUCCESS: return STATUS_SUCCESS;
2268 case VERR_GENERAL_FAILURE: return STATUS_NOT_SUPPORTED;
2269 case VERR_INVALID_PARAMETER: return STATUS_INVALID_PARAMETER;
2270 case VERR_INVALID_MAGIC: return STATUS_UNKNOWN_REVISION;
2271 case VERR_INVALID_HANDLE: return STATUS_INVALID_HANDLE;
2272 case VERR_INVALID_POINTER: return STATUS_INVALID_ADDRESS;
2273 case VERR_LOCK_FAILED: return STATUS_NOT_LOCKED;
2274 case VERR_ALREADY_LOADED: return STATUS_IMAGE_ALREADY_LOADED;
2275 case VERR_PERMISSION_DENIED: return STATUS_ACCESS_DENIED;
2276 case VERR_VERSION_MISMATCH: return STATUS_REVISION_MISMATCH;
2277 }
2278
2279 if (rc < 0)
2280 {
2281 if (((uint32_t)rc & UINT32_C(0xffff0000)) == UINT32_C(0xffff0000))
2282 return (NTSTATUS)( ((uint32_t)rc & UINT32_C(0xffff)) | SUP_NT_STATUS_BASE );
2283 }
2284 return STATUS_UNSUCCESSFUL;
2285}
2286
2287
2288#if 0 /* See alternative in SUPDrvA-win.asm */
2289/**
2290 * Alternative version of SUPR0Printf for Windows.
2291 *
2292 * @returns 0.
2293 * @param pszFormat The format string.
2294 */
2295SUPR0DECL(int) SUPR0Printf(const char *pszFormat, ...)
2296{
2297 va_list va;
2298 char szMsg[512];
2299
2300 va_start(va, pszFormat);
2301 size_t cch = RTStrPrintfV(szMsg, sizeof(szMsg) - 1, pszFormat, va);
2302 szMsg[sizeof(szMsg) - 1] = '\0';
2303 va_end(va);
2304
2305 RTLogWriteDebugger(szMsg, cch);
2306 return 0;
2307}
2308#endif
2309
2310
2311SUPR0DECL(uint32_t) SUPR0GetKernelFeatures(void)
2312{
2313 return 0;
2314}
2315
2316
2317#ifdef VBOX_WITH_HARDENING
2318
2319/** @name Identifying Special Processes: CSRSS.EXE
2320 * @{ */
2321
2322
2323/**
2324 * Checks if the process is a system32 process by the given name.
2325 *
2326 * @returns true / false.
2327 * @param pProcess The process to check.
2328 * @param pszName The lower case process name (no path!).
2329 */
2330static bool supdrvNtProtectIsSystem32ProcessMatch(PEPROCESS pProcess, const char *pszName)
2331{
2332 Assert(strlen(pszName) < 16); /* see buffer below */
2333
2334 /*
2335 * This test works on XP+.
2336 */
2337 const char *pszImageFile = (const char *)PsGetProcessImageFileName(pProcess);
2338 if (!pszImageFile)
2339 return false;
2340
2341 if (RTStrICmp(pszImageFile, pszName) != 0)
2342 return false;
2343
2344 /*
2345 * This test requires a Vista+ API.
2346 */
2347 if (g_pfnPsReferenceProcessFilePointer)
2348 {
2349 PFILE_OBJECT pFile = NULL;
2350 NTSTATUS rcNt = g_pfnPsReferenceProcessFilePointer(pProcess, &pFile);
2351 if (!NT_SUCCESS(rcNt))
2352 return false;
2353
2354 union
2355 {
2356 OBJECT_NAME_INFORMATION Info;
2357 uint8_t abBuffer[sizeof(g_System32NtPath) + 16 * sizeof(WCHAR)];
2358 } Buf;
2359 ULONG cbIgn;
2360 rcNt = ObQueryNameString(pFile, &Buf.Info, sizeof(Buf) - sizeof(WCHAR), &cbIgn);
2361 ObDereferenceObject(pFile);
2362 if (!NT_SUCCESS(rcNt))
2363 return false;
2364
2365 /* Terminate the name. */
2366 PRTUTF16 pwszName = Buf.Info.Name.Buffer;
2367 pwszName[Buf.Info.Name.Length / sizeof(RTUTF16)] = '\0';
2368
2369 /* Match the name against the system32 directory path. */
2370 uint32_t cbSystem32 = g_System32NtPath.UniStr.Length;
2371 if (Buf.Info.Name.Length < cbSystem32)
2372 return false;
2373 if (memcmp(pwszName, g_System32NtPath.UniStr.Buffer, cbSystem32))
2374 return false;
2375 pwszName += cbSystem32 / sizeof(RTUTF16);
2376 if (*pwszName++ != '\\')
2377 return false;
2378
2379 /* Compare the name. */
2380 const char *pszRight = pszName;
2381 for (;;)
2382 {
2383 WCHAR wchLeft = *pwszName++;
2384 char chRight = *pszRight++;
2385 Assert(chRight == RT_C_TO_LOWER(chRight));
2386
2387 if ( wchLeft != chRight
2388 && RT_C_TO_LOWER(wchLeft) != chRight)
2389 return false;
2390 if (!chRight)
2391 break;
2392 }
2393 }
2394
2395 return true;
2396}
2397
2398
2399/**
2400 * Checks if the current process is likely to be CSRSS.
2401 *
2402 * @returns true/false.
2403 * @param pProcess The process.
2404 */
2405static bool supdrvNtProtectIsCsrssByProcess(PEPROCESS pProcess)
2406{
2407 /*
2408 * On Windows 8.1 CSRSS.EXE is a protected process.
2409 */
2410 if (g_pfnPsIsProtectedProcessLight)
2411 {
2412 if (!g_pfnPsIsProtectedProcessLight(pProcess))
2413 return false;
2414 }
2415
2416 /*
2417 * The name tests.
2418 */
2419 if (!supdrvNtProtectIsSystem32ProcessMatch(pProcess, "csrss.exe"))
2420 return false;
2421
2422 /** @todo Could extend the CSRSS.EXE check with that the TokenUser of the
2423 * current process must be "NT AUTHORITY\SYSTEM" (S-1-5-18). */
2424
2425 return true;
2426}
2427
2428
2429/**
2430 * Helper for supdrvNtProtectGetAlpcPortObjectType that tries out a name.
2431 *
2432 * @returns true if done, false if not.
2433 * @param pwszPortNm The port path.
2434 * @param ppObjType The object type return variable, updated when
2435 * returning true.
2436 */
2437static bool supdrvNtProtectGetAlpcPortObjectType2(PCRTUTF16 pwszPortNm, POBJECT_TYPE *ppObjType)
2438{
2439 bool fDone = false;
2440
2441 UNICODE_STRING UniStrPortNm;
2442 UniStrPortNm.Buffer = (WCHAR *)pwszPortNm;
2443 UniStrPortNm.Length = (USHORT)(RTUtf16Len(pwszPortNm) * sizeof(WCHAR));
2444 UniStrPortNm.MaximumLength = UniStrPortNm.Length + sizeof(WCHAR);
2445
2446 OBJECT_ATTRIBUTES ObjAttr;
2447 InitializeObjectAttributes(&ObjAttr, &UniStrPortNm, OBJ_CASE_INSENSITIVE | OBJ_KERNEL_HANDLE, NULL, NULL);
2448
2449 HANDLE hPort;
2450 NTSTATUS rcNt = g_pfnZwAlpcCreatePort(&hPort, &ObjAttr, NULL /*pPortAttribs*/);
2451 if (NT_SUCCESS(rcNt))
2452 {
2453 PVOID pvObject;
2454 rcNt = ObReferenceObjectByHandle(hPort, 0 /*DesiredAccess*/, NULL /*pObjectType*/,
2455 KernelMode, &pvObject, NULL /*pHandleInfo*/);
2456 if (NT_SUCCESS(rcNt))
2457 {
2458 POBJECT_TYPE pObjType = g_pfnObGetObjectType(pvObject);
2459 if (pObjType)
2460 {
2461 SUPR0Printf("vboxdrv: ALPC Port Object Type %p (vs %p)\n", pObjType, *ppObjType);
2462 *ppObjType = pObjType;
2463 fDone = true;
2464 }
2465 ObDereferenceObject(pvObject);
2466 }
2467 NtClose(hPort);
2468 }
2469 return fDone;
2470}
2471
2472
2473/**
2474 * Attempts to retrieve the ALPC Port object type.
2475 *
2476 * We've had at least three reports that using LpcPortObjectType when trying to
2477 * get at the ApiPort object results in STATUS_OBJECT_TYPE_MISMATCH errors.
2478 * It's not known who has modified LpcPortObjectType or AlpcPortObjectType (not
2479 * exported) so that it differs from the actual ApiPort type, or maybe this
2480 * unknown entity is intercepting our attempt to reference the port and
2481 * tries to mislead us. The paranoid explanataion is of course that some evil
2482 * root kit like software is messing with the OS, however, it's possible that
2483 * this is valid kernel behavior that 99.8% of our users and 100% of the
2484 * developers are not triggering for some reason.
2485 *
2486 * The code here creates an ALPC port object and gets it's type. It will cache
2487 * the result in g_pAlpcPortObjectType2 on success.
2488 *
2489 * @returns Object type.
2490 * @param uSessionId The session id.
2491 * @param pszSessionId The session id formatted as a string.
2492 */
2493static POBJECT_TYPE supdrvNtProtectGetAlpcPortObjectType(uint32_t uSessionId, const char *pszSessionId)
2494{
2495 POBJECT_TYPE pObjType = *LpcPortObjectType;
2496
2497 if ( g_pfnZwAlpcCreatePort
2498 && g_pfnObGetObjectType)
2499 {
2500 int rc;
2501 ssize_t cchTmp; NOREF(cchTmp);
2502 char szTmp[16];
2503 RTUTF16 wszPortNm[128];
2504 size_t offRand;
2505
2506 /*
2507 * First attempt is in the session directory.
2508 */
2509 rc = RTUtf16CopyAscii(wszPortNm, RT_ELEMENTS(wszPortNm), "\\Sessions\\");
2510 rc |= RTUtf16CatAscii(wszPortNm, RT_ELEMENTS(wszPortNm), pszSessionId);
2511 rc |= RTUtf16CatAscii(wszPortNm, RT_ELEMENTS(wszPortNm), "\\VBoxDrv-");
2512 cchTmp = RTStrFormatU32(szTmp, sizeof(szTmp), (uint32_t)(uintptr_t)PsGetProcessId(PsGetCurrentProcess()), 16, 0, 0, 0);
2513 Assert(cchTmp > 0);
2514 rc |= RTUtf16CatAscii(wszPortNm, RT_ELEMENTS(wszPortNm), szTmp);
2515 rc |= RTUtf16CatAscii(wszPortNm, RT_ELEMENTS(wszPortNm), "-");
2516 offRand = RTUtf16Len(wszPortNm);
2517 cchTmp = RTStrFormatU32(szTmp, sizeof(szTmp), RTRandU32(), 16, 0, 0, 0);
2518 Assert(cchTmp > 0);
2519 rc |= RTUtf16CatAscii(wszPortNm, RT_ELEMENTS(wszPortNm), szTmp);
2520 AssertRCSuccess(rc);
2521
2522 bool fDone = supdrvNtProtectGetAlpcPortObjectType2(wszPortNm, &pObjType);
2523 if (!fDone)
2524 {
2525 wszPortNm[offRand] = '\0';
2526 cchTmp = RTStrFormatU32(szTmp, sizeof(szTmp), RTRandU32(), 16, 0, 0, 0); Assert(cchTmp > 0);
2527 rc |= RTUtf16CatAscii(wszPortNm, RT_ELEMENTS(wszPortNm), szTmp);
2528 AssertRCSuccess(rc);
2529
2530 fDone = supdrvNtProtectGetAlpcPortObjectType2(wszPortNm, &pObjType);
2531 }
2532 if (!fDone)
2533 {
2534 /*
2535 * Try base names.
2536 */
2537 if (uSessionId == 0)
2538 rc = RTUtf16CopyAscii(wszPortNm, RT_ELEMENTS(wszPortNm), "\\BaseNamedObjects\\VBoxDrv-");
2539 else
2540 {
2541 rc = RTUtf16CopyAscii(wszPortNm, RT_ELEMENTS(wszPortNm), "\\Sessions\\");
2542 rc |= RTUtf16CatAscii(wszPortNm, RT_ELEMENTS(wszPortNm), pszSessionId);
2543 rc |= RTUtf16CatAscii(wszPortNm, RT_ELEMENTS(wszPortNm), "\\BaseNamedObjects\\VBoxDrv-");
2544 }
2545 cchTmp = RTStrFormatU32(szTmp, sizeof(szTmp), (uint32_t)(uintptr_t)PsGetProcessId(PsGetCurrentProcess()), 16, 0, 0, 0);
2546 Assert(cchTmp > 0);
2547 rc |= RTUtf16CatAscii(wszPortNm, RT_ELEMENTS(wszPortNm), szTmp);
2548 rc |= RTUtf16CatAscii(wszPortNm, RT_ELEMENTS(wszPortNm), "-");
2549 offRand = RTUtf16Len(wszPortNm);
2550 cchTmp = RTStrFormatU32(szTmp, sizeof(szTmp), RTRandU32(), 16, 0, 0, 0);
2551 Assert(cchTmp > 0);
2552 rc |= RTUtf16CatAscii(wszPortNm, RT_ELEMENTS(wszPortNm), szTmp);
2553 AssertRCSuccess(rc);
2554
2555 bool fDone = supdrvNtProtectGetAlpcPortObjectType2(wszPortNm, &pObjType);
2556 if (!fDone)
2557 {
2558 wszPortNm[offRand] = '\0';
2559 cchTmp = RTStrFormatU32(szTmp, sizeof(szTmp), RTRandU32(), 16, 0, 0, 0);
2560 Assert(cchTmp > 0);
2561 rc |= RTUtf16CatAscii(wszPortNm, RT_ELEMENTS(wszPortNm), szTmp);
2562 AssertRCSuccess(rc);
2563
2564 fDone = supdrvNtProtectGetAlpcPortObjectType2(wszPortNm, &pObjType);
2565 }
2566 }
2567
2568 /* Cache the result in g_pAlpcPortObjectType2. */
2569 if ( g_pAlpcPortObjectType2 == NULL
2570 && pObjType != g_pAlpcPortObjectType1
2571 && fDone)
2572 g_pAlpcPortObjectType2 = pObjType;
2573
2574 }
2575
2576 return pObjType;
2577}
2578
2579
2580/**
2581 * Called in the context of VBoxDrvNtCreate to determin the CSRSS for the
2582 * current process.
2583 *
2584 * The Client/Server Runtime Subsystem (CSRSS) process needs to be allowed some
2585 * additional access right so we need to make 101% sure we correctly identify
2586 * the CSRSS process a process is associated with.
2587 *
2588 * @returns IPRT status code.
2589 * @param pNtProtect The NT protected process structure. The
2590 * hCsrssPid member will be updated on success.
2591 */
2592static int supdrvNtProtectFindAssociatedCsrss(PSUPDRVNTPROTECT pNtProtect)
2593{
2594 Assert(pNtProtect->AvlCore.Key == PsGetCurrentProcessId());
2595 Assert(pNtProtect->pCsrssProcess == NULL);
2596 Assert(pNtProtect->hCsrssPid == NULL);
2597
2598 /*
2599 * We'll try use the ApiPort LPC object for the session we're in to track
2600 * down the CSRSS process. So, we start by constructing a path to it.
2601 */
2602 int rc;
2603 uint32_t uSessionId = PsGetProcessSessionId(PsGetCurrentProcess());
2604 char szSessionId[16];
2605 WCHAR wszApiPort[48];
2606 if (uSessionId == 0)
2607 {
2608 szSessionId[0] = '0';
2609 szSessionId[1] = '\0';
2610 rc = RTUtf16CopyAscii(wszApiPort, RT_ELEMENTS(wszApiPort), "\\Windows\\ApiPort");
2611 }
2612 else
2613 {
2614 ssize_t cchTmp = RTStrFormatU32(szSessionId, sizeof(szSessionId), uSessionId, 10, 0, 0, 0);
2615 AssertReturn(cchTmp > 0, (int)cchTmp);
2616 rc = RTUtf16CopyAscii(wszApiPort, RT_ELEMENTS(wszApiPort), "\\Sessions\\");
2617 if (RT_SUCCESS(rc))
2618 rc = RTUtf16CatAscii(wszApiPort, RT_ELEMENTS(wszApiPort), szSessionId);
2619 if (RT_SUCCESS(rc))
2620 rc = RTUtf16CatAscii(wszApiPort, RT_ELEMENTS(wszApiPort), "\\Windows\\ApiPort");
2621 }
2622 AssertRCReturn(rc, rc);
2623
2624 UNICODE_STRING ApiPortStr;
2625 ApiPortStr.Buffer = wszApiPort;
2626 ApiPortStr.Length = (USHORT)(RTUtf16Len(wszApiPort) * sizeof(RTUTF16));
2627 ApiPortStr.MaximumLength = ApiPortStr.Length + sizeof(RTUTF16);
2628
2629 /*
2630 * The object cannot be opened, but we can reference it by name.
2631 */
2632 void *pvApiPortObj = NULL;
2633 NTSTATUS rcNt = ObReferenceObjectByName(&ApiPortStr,
2634 0,
2635 NULL /*pAccessState*/,
2636 STANDARD_RIGHTS_READ,
2637 g_pAlpcPortObjectType1,
2638 KernelMode,
2639 NULL /*pvParseContext*/,
2640 &pvApiPortObj);
2641 if ( rcNt == STATUS_OBJECT_TYPE_MISMATCH
2642 && g_pAlpcPortObjectType2 != NULL)
2643 rcNt = ObReferenceObjectByName(&ApiPortStr,
2644 0,
2645 NULL /*pAccessState*/,
2646 STANDARD_RIGHTS_READ,
2647 g_pAlpcPortObjectType2,
2648 KernelMode,
2649 NULL /*pvParseContext*/,
2650 &pvApiPortObj);
2651 if ( rcNt == STATUS_OBJECT_TYPE_MISMATCH
2652 && g_pfnObGetObjectType
2653 && g_pfnZwAlpcCreatePort)
2654 rcNt = ObReferenceObjectByName(&ApiPortStr,
2655 0,
2656 NULL /*pAccessState*/,
2657 STANDARD_RIGHTS_READ,
2658 supdrvNtProtectGetAlpcPortObjectType(uSessionId, szSessionId),
2659 KernelMode,
2660 NULL /*pvParseContext*/,
2661 &pvApiPortObj);
2662 if (!NT_SUCCESS(rcNt))
2663 {
2664 SUPR0Printf("vboxdrv: Error opening '%ls': %#x\n", wszApiPort, rcNt);
2665 return rcNt == STATUS_OBJECT_TYPE_MISMATCH ? VERR_SUPDRV_APIPORT_OPEN_ERROR_TYPE : VERR_SUPDRV_APIPORT_OPEN_ERROR;
2666 }
2667
2668 /*
2669 * Query the processes in the system so we can locate CSRSS.EXE candidates.
2670 * Note! Attempts at using SystemSessionProcessInformation failed with
2671 * STATUS_ACCESS_VIOLATION.
2672 * Note! The 32 bytes on the size of to counteract the allocation header
2673 * that rtR0MemAllocEx slaps on everything.
2674 */
2675 ULONG cbNeeded = _64K - 32;
2676 uint32_t cbBuf;
2677 uint8_t *pbBuf = NULL;
2678 do
2679 {
2680 cbBuf = RT_ALIGN(cbNeeded + _4K, _64K) - 32;
2681 pbBuf = (uint8_t *)RTMemAlloc(cbBuf);
2682 if (!pbBuf)
2683 break;
2684
2685 cbNeeded = 0;
2686#if 0 /* doesn't work. */
2687 SYSTEM_SESSION_PROCESS_INFORMATION Req;
2688 Req.SessionId = uSessionId;
2689 Req.BufferLength = cbBuf;
2690 Req.Buffer = pbBuf;
2691 rcNt = NtQuerySystemInformation(SystemSessionProcessInformation, &Req, sizeof(Req), &cbNeeded);
2692#else
2693 rcNt = NtQuerySystemInformation(SystemProcessInformation, pbBuf, cbBuf, &cbNeeded);
2694#endif
2695 if (NT_SUCCESS(rcNt))
2696 break;
2697
2698 RTMemFree(pbBuf);
2699 pbBuf = NULL;
2700 } while ( rcNt == STATUS_INFO_LENGTH_MISMATCH
2701 && cbNeeded > cbBuf
2702 && cbNeeded < 32U*_1M);
2703
2704 if ( pbBuf
2705 && NT_SUCCESS(rcNt)
2706 && cbNeeded >= sizeof(SYSTEM_PROCESS_INFORMATION))
2707 {
2708 /*
2709 * Walk the returned data and look for the process associated with the
2710 * ApiPort object. The ApiPort object keeps the EPROCESS address of
2711 * the owner process (i.e. CSRSS) relatively early in the structure. On
2712 * 64-bit windows 8.1 it's at offset 0x18. So, obtain the EPROCESS
2713 * pointer to likely CSRSS processes and check for a match in the first
2714 * 0x40 bytes of the ApiPort object.
2715 */
2716 rc = VERR_SUPDRV_CSRSS_NOT_FOUND;
2717 for (uint32_t offBuf = 0; offBuf <= cbNeeded - sizeof(SYSTEM_PROCESS_INFORMATION);)
2718 {
2719 PRTNT_SYSTEM_PROCESS_INFORMATION pProcInfo = (PRTNT_SYSTEM_PROCESS_INFORMATION)&pbBuf[offBuf];
2720 if ( pProcInfo->ProcessName.Length == 9 * sizeof(WCHAR)
2721 && pProcInfo->NumberOfThreads > 2 /* Very low guess. */
2722 && pProcInfo->HandleCount > 32 /* Very low guess, I hope. */
2723 && (uintptr_t)pProcInfo->ProcessName.Buffer - (uintptr_t)pbBuf < cbNeeded
2724 && RT_C_TO_LOWER(pProcInfo->ProcessName.Buffer[0]) == 'c'
2725 && RT_C_TO_LOWER(pProcInfo->ProcessName.Buffer[1]) == 's'
2726 && RT_C_TO_LOWER(pProcInfo->ProcessName.Buffer[2]) == 'r'
2727 && RT_C_TO_LOWER(pProcInfo->ProcessName.Buffer[3]) == 's'
2728 && RT_C_TO_LOWER(pProcInfo->ProcessName.Buffer[4]) == 's'
2729 && pProcInfo->ProcessName.Buffer[5] == '.'
2730 && RT_C_TO_LOWER(pProcInfo->ProcessName.Buffer[6]) == 'e'
2731 && RT_C_TO_LOWER(pProcInfo->ProcessName.Buffer[7]) == 'x'
2732 && RT_C_TO_LOWER(pProcInfo->ProcessName.Buffer[8]) == 'e' )
2733 {
2734
2735 /* Get the process structure and perform some more thorough
2736 process checks. */
2737 PEPROCESS pProcess;
2738 rcNt = PsLookupProcessByProcessId(pProcInfo->UniqueProcessId, &pProcess);
2739 if (NT_SUCCESS(rcNt))
2740 {
2741 if (supdrvNtProtectIsCsrssByProcess(pProcess))
2742 {
2743 if (PsGetProcessSessionId(pProcess) == uSessionId)
2744 {
2745 /* Final test, check the ApiPort.
2746 Note! The old LPC (pre Vista) objects has the PID
2747 much earlier in the structure. Might be
2748 worth looking for it instead. */
2749 bool fThatsIt = false;
2750 __try
2751 {
2752 PEPROCESS *ppPortProc = (PEPROCESS *)pvApiPortObj;
2753 uint32_t cTests = g_uNtVerCombined >= SUP_NT_VER_VISTA ? 16 : 38; /* ALPC since Vista. */
2754 do
2755 {
2756 fThatsIt = *ppPortProc == pProcess;
2757 ppPortProc++;
2758 } while (!fThatsIt && --cTests > 0);
2759 }
2760 __except(EXCEPTION_EXECUTE_HANDLER)
2761 {
2762 fThatsIt = false;
2763 }
2764 if (fThatsIt)
2765 {
2766 /* Ok, we found it! Keep the process structure
2767 reference as well as the PID so we can
2768 safely identify it later on. */
2769 pNtProtect->hCsrssPid = pProcInfo->UniqueProcessId;
2770 pNtProtect->pCsrssProcess = pProcess;
2771 rc = VINF_SUCCESS;
2772 break;
2773 }
2774 }
2775 }
2776
2777 ObDereferenceObject(pProcess);
2778 }
2779 }
2780
2781 /* Advance. */
2782 if (!pProcInfo->NextEntryOffset)
2783 break;
2784 offBuf += pProcInfo->NextEntryOffset;
2785 }
2786 }
2787 else
2788 rc = VERR_SUPDRV_SESSION_PROCESS_ENUM_ERROR;
2789 RTMemFree(pbBuf);
2790 ObDereferenceObject(pvApiPortObj);
2791 return rc;
2792}
2793
2794
2795/**
2796 * Checks that the given process is the CSRSS process associated with protected
2797 * process.
2798 *
2799 * @returns true / false.
2800 * @param pNtProtect The NT protection structure.
2801 * @param pCsrss The process structure of the alleged CSRSS.EXE
2802 * process.
2803 */
2804static bool supdrvNtProtectIsAssociatedCsrss(PSUPDRVNTPROTECT pNtProtect, PEPROCESS pCsrss)
2805{
2806 if (pNtProtect->pCsrssProcess == pCsrss)
2807 {
2808 if (pNtProtect->hCsrssPid == PsGetProcessId(pCsrss))
2809 {
2810 return true;
2811 }
2812 }
2813 return false;
2814}
2815
2816
2817/**
2818 * Checks if the given process is the stupid themes service.
2819 *
2820 * The caller does some screening of access masks and what not. We do the rest.
2821 *
2822 * @returns true / false.
2823 * @param pNtProtect The NT protection structure.
2824 * @param pAnnoyingProcess The process structure of an process that might
2825 * happen to be the annoying themes process.
2826 */
2827static bool supdrvNtProtectIsFrigginThemesService(PSUPDRVNTPROTECT pNtProtect, PEPROCESS pAnnoyingProcess)
2828{
2829 /*
2830 * Check the process name.
2831 */
2832 if (!supdrvNtProtectIsSystem32ProcessMatch(pAnnoyingProcess, "svchost.exe"))
2833 return false;
2834
2835 /** @todo Come up with more checks. */
2836
2837 return true;
2838}
2839
2840
2841#ifdef VBOX_WITHOUT_DEBUGGER_CHECKS
2842/**
2843 * Checks if the given process is one of the whitelisted debuggers.
2844 *
2845 * @returns true / false.
2846 * @param pProcess The process to check.
2847 */
2848static bool supdrvNtProtectIsWhitelistedDebugger(PEPROCESS pProcess)
2849{
2850 const char *pszImageFile = (const char *)PsGetProcessImageFileName(pProcess);
2851 if (!pszImageFile)
2852 return false;
2853
2854 if (pszImageFile[0] == 'w' || pszImageFile[0] == 'W')
2855 {
2856 if (RTStrICmp(pszImageFile, "windbg.exe") == 0)
2857 return true;
2858 if (RTStrICmp(pszImageFile, "werfault.exe") == 0)
2859 return true;
2860 if (RTStrICmp(pszImageFile, "werfaultsecure.exe") == 0)
2861 return true;
2862 }
2863 else if (pszImageFile[0] == 'd' || pszImageFile[0] == 'D')
2864 {
2865 if (RTStrICmp(pszImageFile, "drwtsn32.exe") == 0)
2866 return true;
2867 if (RTStrICmp(pszImageFile, "dwwin.exe") == 0)
2868 return true;
2869 }
2870
2871 return false;
2872}
2873#endif /* VBOX_WITHOUT_DEBUGGER_CHECKS */
2874
2875
2876/** @} */
2877
2878
2879/** @name Process Creation Callbacks.
2880 * @{ */
2881
2882
2883/**
2884 * Cleans up VBoxDrv or VBoxDrvStub error info not collected by the dead process.
2885 *
2886 * @param hProcessId The ID of the dead process.
2887 */
2888static void supdrvNtErrorInfoCleanupProcess(HANDLE hProcessId)
2889{
2890 int rc = RTSemMutexRequestNoResume(g_hErrorInfoLock, RT_INDEFINITE_WAIT);
2891 if (RT_SUCCESS(rc))
2892 {
2893 PSUPDRVNTERRORINFO pCur, pNext;
2894 RTListForEachSafe(&g_ErrorInfoHead, pCur, pNext, SUPDRVNTERRORINFO, ListEntry)
2895 {
2896 if (pCur->hProcessId == hProcessId)
2897 {
2898 RTListNodeRemove(&pCur->ListEntry);
2899 RTMemFree(pCur);
2900 }
2901 }
2902 RTSemMutexRelease(g_hErrorInfoLock);
2903 }
2904}
2905
2906
2907/**
2908 * Common worker used by the process creation hooks as well as the process
2909 * handle creation hooks to check if a VM process is being created.
2910 *
2911 * @returns true if likely to be a VM process, false if not.
2912 * @param pNtStub The NT protection structure for the possible
2913 * stub process.
2914 * @param hParentPid The parent pid.
2915 * @param hChildPid The child pid.
2916 */
2917static bool supdrvNtProtectIsSpawningStubProcess(PSUPDRVNTPROTECT pNtStub, HANDLE hParentPid, HANDLE hChildPid)
2918{
2919 bool fRc = false;
2920 if (pNtStub->AvlCore.Key == hParentPid) /* paranoia */
2921 {
2922 if (pNtStub->enmProcessKind == kSupDrvNtProtectKind_StubSpawning)
2923 {
2924 /* Compare short names. */
2925 PEPROCESS pStubProcess;
2926 NTSTATUS rcNt = PsLookupProcessByProcessId(hParentPid, &pStubProcess);
2927 if (NT_SUCCESS(rcNt))
2928 {
2929 PEPROCESS pChildProcess;
2930 rcNt = PsLookupProcessByProcessId(hChildPid, &pChildProcess);
2931 if (NT_SUCCESS(rcNt))
2932 {
2933 const char *pszStub = (const char *)PsGetProcessImageFileName(pStubProcess);
2934 const char *pszChild = (const char *)PsGetProcessImageFileName(pChildProcess);
2935 fRc = pszStub != NULL
2936 && pszChild != NULL
2937 && strcmp(pszStub, pszChild) == 0;
2938
2939 /** @todo check that the full image names matches. */
2940
2941 ObDereferenceObject(pChildProcess);
2942 }
2943 ObDereferenceObject(pStubProcess);
2944 }
2945 }
2946 }
2947 return fRc;
2948}
2949
2950
2951/**
2952 * Common code used by the notifies to protect a child process.
2953 *
2954 * @returns VBox status code.
2955 * @param pNtStub The NT protect structure for the parent.
2956 * @param hChildPid The child pid.
2957 */
2958static int supdrvNtProtectProtectNewStubChild(PSUPDRVNTPROTECT pNtParent, HANDLE hChildPid)
2959{
2960 /*
2961 * Create a child protection struction.
2962 */
2963 PSUPDRVNTPROTECT pNtChild;
2964 int rc = supdrvNtProtectCreate(&pNtChild, hChildPid, kSupDrvNtProtectKind_VmProcessUnconfirmed, false /*fLink*/);
2965 if (RT_SUCCESS(rc))
2966 {
2967 pNtChild->fFirstProcessCreateHandle = true;
2968 pNtChild->fFirstThreadCreateHandle = true;
2969 pNtChild->fCsrssFirstProcessCreateHandle = true;
2970 pNtChild->cCsrssFirstProcessDuplicateHandle = ARCH_BITS == 32 ? 2 : 1;
2971 pNtChild->fThemesFirstProcessCreateHandle = true;
2972 pNtChild->hParentPid = pNtParent->AvlCore.Key;
2973 pNtChild->hCsrssPid = pNtParent->hCsrssPid;
2974 pNtChild->pCsrssProcess = pNtParent->pCsrssProcess;
2975 if (pNtChild->pCsrssProcess)
2976 ObReferenceObject(pNtChild->pCsrssProcess);
2977
2978 /*
2979 * Take the spinlock, recheck parent conditions and link things.
2980 */
2981 RTSpinlockAcquire(g_hNtProtectLock);
2982 if (pNtParent->enmProcessKind == kSupDrvNtProtectKind_StubSpawning)
2983 {
2984 bool fSuccess = RTAvlPVInsert(&g_NtProtectTree, &pNtChild->AvlCore);
2985 if (fSuccess)
2986 {
2987 pNtChild->fInTree = true;
2988 pNtParent->u.pChild = pNtChild; /* Parent keeps the initial reference. */
2989 pNtParent->enmProcessKind = kSupDrvNtProtectKind_StubParent;
2990 pNtChild->u.pParent = pNtParent;
2991
2992 RTSpinlockRelease(g_hNtProtectLock);
2993 return VINF_SUCCESS;
2994 }
2995
2996 rc = VERR_INTERNAL_ERROR_2;
2997 }
2998 else
2999 rc = VERR_WRONG_ORDER;
3000 pNtChild->enmProcessKind = kSupDrvNtProtectKind_VmProcessDead;
3001 RTSpinlockRelease(g_hNtProtectLock);
3002
3003 supdrvNtProtectRelease(pNtChild);
3004 }
3005 return rc;
3006}
3007
3008
3009/**
3010 * Common process termination code.
3011 *
3012 * Transitions protected process to the dead states, protecting against handle
3013 * PID reuse (esp. with unconfirmed VM processes) and handle cleanup issues.
3014 *
3015 * @param hDeadPid The PID of the dead process.
3016 */
3017static void supdrvNtProtectUnprotectDeadProcess(HANDLE hDeadPid)
3018{
3019 PSUPDRVNTPROTECT pNtProtect = supdrvNtProtectLookup(hDeadPid);
3020 if (pNtProtect)
3021 {
3022 PSUPDRVNTPROTECT pNtChild = NULL;
3023
3024 RTSpinlockAcquire(g_hNtProtectLock);
3025
3026 /*
3027 * If this is an unconfirmed VM process, we must release the reference
3028 * the parent structure holds.
3029 */
3030 if (pNtProtect->enmProcessKind == kSupDrvNtProtectKind_VmProcessUnconfirmed)
3031 {
3032 PSUPDRVNTPROTECT pNtParent = pNtProtect->u.pParent;
3033 AssertRelease(pNtParent); AssertRelease(pNtParent->u.pChild == pNtProtect);
3034 pNtParent->u.pChild = NULL;
3035 pNtProtect->u.pParent = NULL;
3036 pNtChild = pNtProtect;
3037 }
3038 /*
3039 * If this is a stub exitting before the VM process gets confirmed,
3040 * release the protection of the potential VM process as this is not
3041 * the prescribed behavior.
3042 */
3043 else if ( pNtProtect->enmProcessKind == kSupDrvNtProtectKind_StubParent
3044 && pNtProtect->u.pChild)
3045 {
3046 pNtChild = pNtProtect->u.pChild;
3047 pNtProtect->u.pChild = NULL;
3048 pNtChild->u.pParent = NULL;
3049 pNtChild->enmProcessKind = kSupDrvNtProtectKind_VmProcessDead;
3050 }
3051
3052 /*
3053 * Transition it to the dead state to prevent it from opening the
3054 * support driver again or be posthumously abused as a vm process parent.
3055 */
3056 if ( pNtProtect->enmProcessKind == kSupDrvNtProtectKind_VmProcessUnconfirmed
3057 || pNtProtect->enmProcessKind == kSupDrvNtProtectKind_VmProcessConfirmed)
3058 pNtProtect->enmProcessKind = kSupDrvNtProtectKind_VmProcessDead;
3059 else if ( pNtProtect->enmProcessKind == kSupDrvNtProtectKind_StubParent
3060 || pNtProtect->enmProcessKind == kSupDrvNtProtectKind_StubSpawning
3061 || pNtProtect->enmProcessKind == kSupDrvNtProtectKind_StubUnverified)
3062 pNtProtect->enmProcessKind = kSupDrvNtProtectKind_StubDead;
3063
3064 RTSpinlockRelease(g_hNtProtectLock);
3065
3066 supdrvNtProtectRelease(pNtProtect);
3067 supdrvNtProtectRelease(pNtChild);
3068
3069 /*
3070 * Do session cleanups.
3071 */
3072 AssertReturnVoid((HANDLE)(uintptr_t)RTProcSelf() == hDeadPid);
3073 if (g_pDevObjSys)
3074 {
3075 PSUPDRVDEVEXT pDevExt = (PSUPDRVDEVEXT)g_pDevObjSys->DeviceExtension;
3076 PSUPDRVSESSION pSession = supdrvSessionHashTabLookup(pDevExt, (RTPROCESS)(uintptr_t)hDeadPid,
3077 RTR0ProcHandleSelf(), NULL);
3078 if (pSession)
3079 {
3080 supdrvSessionHashTabRemove(pDevExt, pSession, NULL);
3081 supdrvSessionRelease(pSession); /* Drops the reference from supdrvSessionHashTabLookup. */
3082 }
3083 }
3084 }
3085}
3086
3087
3088/**
3089 * Common worker for the process creation callback that verifies a new child
3090 * being created by the handle creation callback code.
3091 *
3092 * @param pNtStub The parent.
3093 * @param pNtVm The child.
3094 * @param fCallerChecks The result of any additional tests the caller made.
3095 * This is in order to avoid duplicating the failure
3096 * path code.
3097 */
3098static void supdrvNtProtectVerifyNewChildProtection(PSUPDRVNTPROTECT pNtStub, PSUPDRVNTPROTECT pNtVm, bool fCallerChecks)
3099{
3100 if ( fCallerChecks
3101 && pNtStub->enmProcessKind == kSupDrvNtProtectKind_StubParent
3102 && pNtVm->enmProcessKind == kSupDrvNtProtectKind_VmProcessUnconfirmed
3103 && pNtVm->u.pParent == pNtStub
3104 && pNtStub->u.pChild == pNtVm)
3105 {
3106 /* Fine, reset the CSRSS hack (fixes ViRobot APT Shield 2.0 issue). */
3107 pNtVm->fFirstProcessCreateHandle = true;
3108 return;
3109 }
3110
3111 LogRel(("vboxdrv: Misdetected vm stub; hParentPid=%p hChildPid=%p\n", pNtStub->AvlCore.Key, pNtVm->AvlCore.Key));
3112 if (pNtStub->enmProcessKind != kSupDrvNtProtectKind_VmProcessConfirmed)
3113 supdrvNtProtectUnprotectDeadProcess(pNtVm->AvlCore.Key);
3114}
3115
3116
3117/**
3118 * Old style callback (since forever).
3119 *
3120 * @param hParentPid The parent PID.
3121 * @param hNewPid The PID of the new child.
3122 * @param fCreated TRUE if it's a creation notification,
3123 * FALSE if termination.
3124 * @remarks ASSUMES this arrives before the handle creation callback.
3125 */
3126static VOID __stdcall
3127supdrvNtProtectCallback_ProcessCreateNotify(HANDLE hParentPid, HANDLE hNewPid, BOOLEAN fCreated)
3128{
3129 /*
3130 * Is it a new process that needs protection?
3131 */
3132 if (fCreated)
3133 {
3134 PSUPDRVNTPROTECT pNtStub = supdrvNtProtectLookup(hParentPid);
3135 if (pNtStub)
3136 {
3137 PSUPDRVNTPROTECT pNtVm = supdrvNtProtectLookup(hNewPid);
3138 if (!pNtVm)
3139 {
3140 if (supdrvNtProtectIsSpawningStubProcess(pNtStub, hParentPid, hNewPid))
3141 supdrvNtProtectProtectNewStubChild(pNtStub, hNewPid);
3142 }
3143 else
3144 {
3145 supdrvNtProtectVerifyNewChildProtection(pNtStub, pNtVm, true);
3146 supdrvNtProtectRelease(pNtVm);
3147 }
3148 supdrvNtProtectRelease(pNtStub);
3149 }
3150 }
3151 /*
3152 * Process termination, do clean ups.
3153 */
3154 else
3155 {
3156 supdrvNtProtectUnprotectDeadProcess(hNewPid);
3157 supdrvNtErrorInfoCleanupProcess(hNewPid);
3158 }
3159}
3160
3161
3162/**
3163 * New style callback (Vista SP1+ / w2k8).
3164 *
3165 * @param pNewProcess The new process.
3166 * @param hNewPid The PID of the new process.
3167 * @param pInfo Process creation details. NULL if process
3168 * termination notification.
3169 * @remarks ASSUMES this arrives before the handle creation callback.
3170 */
3171static VOID __stdcall
3172supdrvNtProtectCallback_ProcessCreateNotifyEx(PEPROCESS pNewProcess, HANDLE hNewPid, PPS_CREATE_NOTIFY_INFO pInfo)
3173{
3174 /*
3175 * Is it a new process that needs protection?
3176 */
3177 if (pInfo)
3178 {
3179 PSUPDRVNTPROTECT pNtStub = supdrvNtProtectLookup(pInfo->CreatingThreadId.UniqueProcess);
3180
3181 Log(("vboxdrv/NewProcessEx: ctx=%04zx/%p pid=%04zx ppid=%04zx ctor=%04zx/%04zx rcNt=%#x %.*ls\n",
3182 PsGetProcessId(PsGetCurrentProcess()), PsGetCurrentProcess(),
3183 hNewPid, pInfo->ParentProcessId,
3184 pInfo->CreatingThreadId.UniqueProcess, pInfo->CreatingThreadId.UniqueThread, pInfo->CreationStatus,
3185 pInfo->FileOpenNameAvailable && pInfo->ImageFileName ? (size_t)pInfo->ImageFileName->Length / 2 : 0,
3186 pInfo->FileOpenNameAvailable && pInfo->ImageFileName ? pInfo->ImageFileName->Buffer : NULL));
3187
3188 if (pNtStub)
3189 {
3190 PSUPDRVNTPROTECT pNtVm = supdrvNtProtectLookup(hNewPid);
3191 if (!pNtVm)
3192 {
3193 /* Parent must be creator. */
3194 if (pInfo->CreatingThreadId.UniqueProcess == pInfo->ParentProcessId)
3195 {
3196 if (supdrvNtProtectIsSpawningStubProcess(pNtStub, pInfo->ParentProcessId, hNewPid))
3197 supdrvNtProtectProtectNewStubChild(pNtStub, hNewPid);
3198 }
3199 }
3200 else
3201 {
3202 /* Parent must be creator (as above). */
3203 supdrvNtProtectVerifyNewChildProtection(pNtStub, pNtVm,
3204 pInfo->CreatingThreadId.UniqueProcess == pInfo->ParentProcessId);
3205 supdrvNtProtectRelease(pNtVm);
3206 }
3207 supdrvNtProtectRelease(pNtStub);
3208 }
3209 }
3210 /*
3211 * Process termination, do clean ups.
3212 */
3213 else
3214 {
3215 supdrvNtProtectUnprotectDeadProcess(hNewPid);
3216 supdrvNtErrorInfoCleanupProcess(hNewPid);
3217 }
3218}
3219
3220/** @} */
3221
3222
3223/** @name Process Handle Callbacks.
3224 * @{ */
3225
3226/** Process rights that we allow for handles to stub and VM processes. */
3227# define SUPDRV_NT_ALLOW_PROCESS_RIGHTS \
3228 ( PROCESS_TERMINATE \
3229 | PROCESS_VM_READ \
3230 | PROCESS_QUERY_INFORMATION \
3231 | PROCESS_QUERY_LIMITED_INFORMATION \
3232 | PROCESS_SUSPEND_RESUME \
3233 | DELETE \
3234 | READ_CONTROL \
3235 | SYNCHRONIZE)
3236
3237/** Evil process rights. */
3238# define SUPDRV_NT_EVIL_PROCESS_RIGHTS \
3239 ( PROCESS_CREATE_THREAD \
3240 | PROCESS_SET_SESSIONID /*?*/ \
3241 | PROCESS_VM_OPERATION \
3242 | PROCESS_VM_WRITE \
3243 | PROCESS_DUP_HANDLE \
3244 | PROCESS_CREATE_PROCESS /*?*/ \
3245 | PROCESS_SET_QUOTA /*?*/ \
3246 | PROCESS_SET_INFORMATION \
3247 | PROCESS_SET_LIMITED_INFORMATION /*?*/ \
3248 | 0)
3249AssertCompile((SUPDRV_NT_ALLOW_PROCESS_RIGHTS & SUPDRV_NT_EVIL_PROCESS_RIGHTS) == 0);
3250
3251
3252static OB_PREOP_CALLBACK_STATUS __stdcall
3253supdrvNtProtectCallback_ProcessHandlePre(PVOID pvUser, POB_PRE_OPERATION_INFORMATION pOpInfo)
3254{
3255 Assert(pvUser == NULL);
3256 Assert(pOpInfo->Operation == OB_OPERATION_HANDLE_CREATE || pOpInfo->Operation == OB_OPERATION_HANDLE_DUPLICATE);
3257 Assert(pOpInfo->ObjectType == *PsProcessType);
3258
3259 /*
3260 * Protected? Kludge required for NtOpenProcess calls comming in before
3261 * the create process hook triggers on Windows 8.1 (possibly others too).
3262 */
3263 HANDLE hObjPid = PsGetProcessId((PEPROCESS)pOpInfo->Object);
3264 PSUPDRVNTPROTECT pNtProtect = supdrvNtProtectLookup(hObjPid);
3265 if (!pNtProtect)
3266 {
3267 HANDLE hParentPid = PsGetProcessInheritedFromUniqueProcessId((PEPROCESS)pOpInfo->Object);
3268 PSUPDRVNTPROTECT pNtStub = supdrvNtProtectLookup(hParentPid);
3269 if (pNtStub)
3270 {
3271 if (supdrvNtProtectIsSpawningStubProcess(pNtStub, hParentPid, hObjPid))
3272 {
3273 supdrvNtProtectProtectNewStubChild(pNtStub, hObjPid);
3274 pNtProtect = supdrvNtProtectLookup(hObjPid);
3275 }
3276 supdrvNtProtectRelease(pNtStub);
3277 }
3278 }
3279 pOpInfo->CallContext = pNtProtect; /* Just for reference. */
3280 if (pNtProtect)
3281 {
3282 /*
3283 * Ok, it's a protected process. Strip rights as required or possible.
3284 */
3285 static ACCESS_MASK const s_fCsrssStupidDesires = 0x1fffff;
3286 ACCESS_MASK fAllowedRights = SUPDRV_NT_ALLOW_PROCESS_RIGHTS;
3287
3288 if (pOpInfo->Operation == OB_OPERATION_HANDLE_CREATE)
3289 {
3290 /* Don't restrict the process accessing itself. */
3291 if ((PEPROCESS)pOpInfo->Object == PsGetCurrentProcess())
3292 {
3293 pOpInfo->CallContext = NULL; /* don't assert */
3294 pNtProtect->fFirstProcessCreateHandle = false;
3295
3296 Log(("vboxdrv/ProcessHandlePre: %sctx=%04zx/%p wants %#x to %p in pid=%04zx [%d] %s\n",
3297 pOpInfo->KernelHandle ? "k" : "", PsGetProcessId(PsGetCurrentProcess()), PsGetCurrentProcess(),
3298 pOpInfo->Parameters->CreateHandleInformation.DesiredAccess,
3299 pOpInfo->Object, pNtProtect->AvlCore.Key, pNtProtect->enmProcessKind,
3300 PsGetProcessImageFileName(PsGetCurrentProcess()) ));
3301 }
3302#ifdef VBOX_WITHOUT_DEBUGGER_CHECKS
3303 /* Allow debuggers full access. */
3304 else if (supdrvNtProtectIsWhitelistedDebugger(PsGetCurrentProcess()))
3305 {
3306 pOpInfo->CallContext = NULL; /* don't assert */
3307 pNtProtect->fFirstProcessCreateHandle = false;
3308
3309 Log(("vboxdrv/ProcessHandlePre: %sctx=%04zx/%p wants %#x to %p in pid=%04zx [%d] %s [debugger]\n",
3310 pOpInfo->KernelHandle ? "k" : "", PsGetProcessId(PsGetCurrentProcess()), PsGetCurrentProcess(),
3311 pOpInfo->Parameters->CreateHandleInformation.DesiredAccess,
3312 pOpInfo->Object, pNtProtect->AvlCore.Key, pNtProtect->enmProcessKind,
3313 PsGetProcessImageFileName(PsGetCurrentProcess()) ));
3314 }
3315#endif
3316 else
3317 {
3318 ACCESS_MASK const fDesiredAccess = pOpInfo->Parameters->CreateHandleInformation.DesiredAccess;
3319
3320 /* Special case 1 on Vista, 7 & 8:
3321 The CreateProcess code passes the handle over to CSRSS.EXE
3322 and the code inBaseSrvCreateProcess will duplicate the
3323 handle with 0x1fffff as access mask. NtDuplicateObject will
3324 fail this call before it ever gets down here.
3325
3326 Special case 2 on 8.1:
3327 The CreateProcess code requires additional rights for
3328 something, we'll drop these in the stub code. */
3329 if ( pNtProtect->enmProcessKind == kSupDrvNtProtectKind_VmProcessUnconfirmed
3330 && pNtProtect->fFirstProcessCreateHandle
3331 && pOpInfo->KernelHandle == 0
3332 && pNtProtect->hParentPid == PsGetProcessId(PsGetCurrentProcess())
3333 && ExGetPreviousMode() != KernelMode)
3334 {
3335 if ( !pOpInfo->KernelHandle
3336 && fDesiredAccess == s_fCsrssStupidDesires)
3337 {
3338 if (g_uNtVerCombined < SUP_MAKE_NT_VER_SIMPLE(6, 3))
3339 fAllowedRights |= s_fCsrssStupidDesires;
3340 else
3341 fAllowedRights = fAllowedRights
3342 | PROCESS_VM_OPERATION
3343 | PROCESS_VM_WRITE
3344 | PROCESS_SET_INFORMATION
3345 | PROCESS_SET_LIMITED_INFORMATION
3346 | 0;
3347 pOpInfo->CallContext = NULL; /* don't assert this. */
3348 }
3349 pNtProtect->fFirstProcessCreateHandle = false;
3350 }
3351
3352 /* Special case 3 on 8.1:
3353 The interaction between the CreateProcess code and CSRSS.EXE
3354 has changed to the better with Windows 8.1. CSRSS.EXE no
3355 longer duplicates the process (thread too) handle, but opens
3356 it, thus allowing us to do our job. */
3357 if ( g_uNtVerCombined >= SUP_MAKE_NT_VER_SIMPLE(6, 3)
3358 && pNtProtect->enmProcessKind == kSupDrvNtProtectKind_VmProcessUnconfirmed
3359 && pNtProtect->fCsrssFirstProcessCreateHandle
3360 && pOpInfo->KernelHandle == 0
3361 && ExGetPreviousMode() == UserMode
3362 && supdrvNtProtectIsAssociatedCsrss(pNtProtect, PsGetCurrentProcess()) )
3363 {
3364 pNtProtect->fCsrssFirstProcessCreateHandle = false;
3365 if (fDesiredAccess == s_fCsrssStupidDesires)
3366 {
3367 /* Not needed: PROCESS_CREATE_THREAD, PROCESS_SET_SESSIONID,
3368 PROCESS_CREATE_PROCESS */
3369 fAllowedRights = fAllowedRights
3370 | PROCESS_VM_OPERATION
3371 | PROCESS_VM_WRITE
3372 | PROCESS_DUP_HANDLE /* Needed for CreateProcess/VBoxTestOGL. */
3373 | 0;
3374 pOpInfo->CallContext = NULL; /* don't assert this. */
3375 }
3376 }
3377
3378 /* Special case 4, Windows 7, Vista, possibly 8, but not 8.1:
3379 The Themes service requires PROCESS_DUP_HANDLE access to our
3380 process or we won't get any menus and dialogs will be half
3381 unreadable. This is _very_ unfortunate and more work will
3382 go into making this more secure. */
3383 if ( g_uNtVerCombined >= SUP_MAKE_NT_VER_SIMPLE(6, 0)
3384 && g_uNtVerCombined < SUP_MAKE_NT_VER_SIMPLE(6, 2)
3385 && fDesiredAccess == 0x1478 /* 6.1.7600.16385 (win7_rtm.090713-1255) */
3386 && pNtProtect->fThemesFirstProcessCreateHandle
3387 && pOpInfo->KernelHandle == 0
3388 && ExGetPreviousMode() == UserMode
3389 && supdrvNtProtectIsFrigginThemesService(pNtProtect, PsGetCurrentProcess()) )
3390 {
3391 pNtProtect->fThemesFirstProcessCreateHandle = true; /* Only once! */
3392 fAllowedRights |= PROCESS_DUP_HANDLE;
3393 pOpInfo->CallContext = NULL; /* don't assert this. */
3394 }
3395
3396 /* Special case 6a, Windows 10+: AudioDG.exe opens the process with the
3397 PROCESS_SET_LIMITED_INFORMATION right. It seems like it need it for
3398 some myserious and weirdly placed cpu set management of our process.
3399 I'd love to understand what that's all about...
3400 Currently playing safe and only grand this right, however limited, to
3401 audiodg.exe. */
3402 if ( g_uNtVerCombined >= SUP_MAKE_NT_VER_SIMPLE(10, 0)
3403 && ( fDesiredAccess == PROCESS_SET_LIMITED_INFORMATION
3404 || fDesiredAccess == (PROCESS_SET_LIMITED_INFORMATION | PROCESS_QUERY_LIMITED_INFORMATION) /* expected fix #1 */
3405 || fDesiredAccess == (PROCESS_SET_LIMITED_INFORMATION | PROCESS_QUERY_INFORMATION) /* expected fix #2 */
3406 )
3407 && pOpInfo->KernelHandle == 0
3408 && ExGetPreviousMode() == UserMode
3409 && supdrvNtProtectIsSystem32ProcessMatch(PsGetCurrentProcess(), "audiodg.exe") )
3410 {
3411 fAllowedRights |= PROCESS_SET_LIMITED_INFORMATION;
3412 pOpInfo->CallContext = NULL; /* don't assert this. */
3413 }
3414
3415 Log(("vboxdrv/ProcessHandlePre: %sctx=%04zx/%p wants %#x to %p/pid=%04zx [%d], allow %#x => %#x; %s [prev=%#x]\n",
3416 pOpInfo->KernelHandle ? "k" : "", PsGetProcessId(PsGetCurrentProcess()), PsGetCurrentProcess(),
3417 fDesiredAccess, pOpInfo->Object, pNtProtect->AvlCore.Key, pNtProtect->enmProcessKind,
3418 fAllowedRights, fDesiredAccess & fAllowedRights,
3419 PsGetProcessImageFileName(PsGetCurrentProcess()), ExGetPreviousMode() ));
3420
3421 pOpInfo->Parameters->CreateHandleInformation.DesiredAccess &= fAllowedRights;
3422 }
3423 }
3424 else
3425 {
3426 /* Don't restrict the process accessing itself. */
3427 if ( (PEPROCESS)pOpInfo->Object == PsGetCurrentProcess()
3428 && pOpInfo->Parameters->DuplicateHandleInformation.TargetProcess == pOpInfo->Object)
3429 {
3430 Log(("vboxdrv/ProcessHandlePre: ctx=%04zx/%p[%p] dup from %04zx/%p with %#x to %p in pid=%04zx [%d] %s\n",
3431 PsGetProcessId(PsGetCurrentProcess()), PsGetCurrentProcess(),
3432 pOpInfo->Parameters->DuplicateHandleInformation.TargetProcess,
3433 PsGetProcessId((PEPROCESS)pOpInfo->Parameters->DuplicateHandleInformation.SourceProcess),
3434 pOpInfo->Parameters->DuplicateHandleInformation.SourceProcess,
3435 pOpInfo->Parameters->DuplicateHandleInformation.DesiredAccess,
3436 pOpInfo->Object, pNtProtect->AvlCore.Key, pNtProtect->enmProcessKind,
3437 PsGetProcessImageFileName(PsGetCurrentProcess()) ));
3438
3439 pOpInfo->CallContext = NULL; /* don't assert */
3440 }
3441 else
3442 {
3443 ACCESS_MASK const fDesiredAccess = pOpInfo->Parameters->DuplicateHandleInformation.DesiredAccess;
3444
3445 /* Special case 5 on Vista, 7 & 8:
3446 This is the CSRSS.EXE end of special case #1. */
3447 if ( g_uNtVerCombined < SUP_MAKE_NT_VER_SIMPLE(6, 3)
3448 && pNtProtect->enmProcessKind == kSupDrvNtProtectKind_VmProcessUnconfirmed
3449 && pNtProtect->cCsrssFirstProcessDuplicateHandle > 0
3450 && pOpInfo->KernelHandle == 0
3451 && fDesiredAccess == s_fCsrssStupidDesires
3452 && pNtProtect->hParentPid
3453 == PsGetProcessId((PEPROCESS)pOpInfo->Parameters->DuplicateHandleInformation.SourceProcess)
3454 && pOpInfo->Parameters->DuplicateHandleInformation.TargetProcess == PsGetCurrentProcess()
3455 && ExGetPreviousMode() == UserMode
3456 && supdrvNtProtectIsAssociatedCsrss(pNtProtect, PsGetCurrentProcess()))
3457 {
3458 if (ASMAtomicDecS32(&pNtProtect->cCsrssFirstProcessDuplicateHandle) >= 0)
3459 {
3460 /* Not needed: PROCESS_CREATE_THREAD, PROCESS_SET_SESSIONID,
3461 PROCESS_CREATE_PROCESS, PROCESS_DUP_HANDLE */
3462 fAllowedRights = fAllowedRights
3463 | PROCESS_VM_OPERATION
3464 | PROCESS_VM_WRITE
3465 | PROCESS_DUP_HANDLE /* Needed for launching VBoxTestOGL. */
3466 | 0;
3467 pOpInfo->CallContext = NULL; /* don't assert this. */
3468 }
3469 }
3470
3471 /* Special case 6b, Windows 10+: AudioDG.exe duplicates the handle it opened above. */
3472 if ( g_uNtVerCombined >= SUP_MAKE_NT_VER_SIMPLE(10, 0)
3473 && ( fDesiredAccess == PROCESS_SET_LIMITED_INFORMATION
3474 || fDesiredAccess == (PROCESS_SET_LIMITED_INFORMATION | PROCESS_QUERY_LIMITED_INFORMATION) /* expected fix #1 */
3475 || fDesiredAccess == (PROCESS_SET_LIMITED_INFORMATION | PROCESS_QUERY_INFORMATION) /* expected fix #2 */
3476 )
3477 && pOpInfo->KernelHandle == 0
3478 && ExGetPreviousMode() == UserMode
3479 && supdrvNtProtectIsSystem32ProcessMatch(PsGetCurrentProcess(), "audiodg.exe") )
3480 {
3481 fAllowedRights |= PROCESS_SET_LIMITED_INFORMATION;
3482 pOpInfo->CallContext = NULL; /* don't assert this. */
3483 }
3484
3485 Log(("vboxdrv/ProcessHandlePre: %sctx=%04zx/%p[%p] dup from %04zx/%p with %#x to %p in pid=%04zx [%d] %s\n",
3486 pOpInfo->KernelHandle ? "k" : "", PsGetProcessId(PsGetCurrentProcess()), PsGetCurrentProcess(),
3487 pOpInfo->Parameters->DuplicateHandleInformation.TargetProcess,
3488 PsGetProcessId((PEPROCESS)pOpInfo->Parameters->DuplicateHandleInformation.SourceProcess),
3489 pOpInfo->Parameters->DuplicateHandleInformation.SourceProcess,
3490 fDesiredAccess,
3491 pOpInfo->Object, pNtProtect->AvlCore.Key, pNtProtect->enmProcessKind,
3492 PsGetProcessImageFileName(PsGetCurrentProcess()) ));
3493
3494 pOpInfo->Parameters->DuplicateHandleInformation.DesiredAccess &= fAllowedRights;
3495 }
3496 }
3497 supdrvNtProtectRelease(pNtProtect);
3498 }
3499
3500 return OB_PREOP_SUCCESS;
3501}
3502
3503
3504static VOID __stdcall
3505supdrvNtProtectCallback_ProcessHandlePost(PVOID pvUser, POB_POST_OPERATION_INFORMATION pOpInfo)
3506{
3507 Assert(pvUser == NULL);
3508 Assert(pOpInfo->Operation == OB_OPERATION_HANDLE_CREATE || pOpInfo->Operation == OB_OPERATION_HANDLE_DUPLICATE);
3509 Assert(pOpInfo->ObjectType == *PsProcessType);
3510
3511 if ( pOpInfo->CallContext
3512 && NT_SUCCESS(pOpInfo->ReturnStatus))
3513 {
3514 ACCESS_MASK const fGrantedAccess = pOpInfo->Operation == OB_OPERATION_HANDLE_CREATE
3515 ? pOpInfo->Parameters->CreateHandleInformation.GrantedAccess
3516 : pOpInfo->Parameters->DuplicateHandleInformation.GrantedAccess;
3517 AssertReleaseMsg( !(fGrantedAccess & ~( SUPDRV_NT_ALLOW_PROCESS_RIGHTS
3518 | WRITE_OWNER | WRITE_DAC /* these two might be forced upon us */
3519 | PROCESS_UNKNOWN_4000 /* Seen set on win 8.1 */
3520 /*| PROCESS_UNKNOWN_8000 */ ) )
3521 || pOpInfo->KernelHandle,
3522 ("GrantedAccess=%#x - we allow %#x - we did not allow %#x\n",
3523 fGrantedAccess, SUPDRV_NT_ALLOW_PROCESS_RIGHTS, fGrantedAccess & ~SUPDRV_NT_ALLOW_PROCESS_RIGHTS));
3524 }
3525}
3526
3527# undef SUPDRV_NT_ALLOW_PROCESS_RIGHTS
3528
3529/** @} */
3530
3531
3532/** @name Thread Handle Callbacks
3533 * @{ */
3534
3535/* From ntifs.h */
3536extern "C" NTKERNELAPI PEPROCESS __stdcall IoThreadToProcess(PETHREAD);
3537
3538/** Thread rights that we allow for handles to stub and VM processes. */
3539# define SUPDRV_NT_ALLOWED_THREAD_RIGHTS \
3540 ( THREAD_TERMINATE \
3541 | THREAD_GET_CONTEXT \
3542 | THREAD_QUERY_INFORMATION \
3543 | THREAD_QUERY_LIMITED_INFORMATION \
3544 | DELETE \
3545 | READ_CONTROL \
3546 | SYNCHRONIZE)
3547/** @todo consider THREAD_SET_LIMITED_INFORMATION & THREAD_RESUME */
3548
3549/** Evil thread rights.
3550 * @remarks THREAD_RESUME is not included as it seems to be forced upon us by
3551 * Windows 8.1, at least for some processes. We dont' actively
3552 * allow it though, just tollerate it when forced to. */
3553# define SUPDRV_NT_EVIL_THREAD_RIGHTS \
3554 ( THREAD_SUSPEND_RESUME \
3555 | THREAD_SET_CONTEXT \
3556 | THREAD_SET_INFORMATION \
3557 | THREAD_SET_LIMITED_INFORMATION /*?*/ \
3558 | THREAD_SET_THREAD_TOKEN /*?*/ \
3559 | THREAD_IMPERSONATE /*?*/ \
3560 | THREAD_DIRECT_IMPERSONATION /*?*/ \
3561 /*| THREAD_RESUME - see remarks. */ \
3562 | 0)
3563AssertCompile((SUPDRV_NT_EVIL_THREAD_RIGHTS & SUPDRV_NT_ALLOWED_THREAD_RIGHTS) == 0);
3564
3565
3566static OB_PREOP_CALLBACK_STATUS __stdcall
3567supdrvNtProtectCallback_ThreadHandlePre(PVOID pvUser, POB_PRE_OPERATION_INFORMATION pOpInfo)
3568{
3569 Assert(pvUser == NULL);
3570 Assert(pOpInfo->Operation == OB_OPERATION_HANDLE_CREATE || pOpInfo->Operation == OB_OPERATION_HANDLE_DUPLICATE);
3571 Assert(pOpInfo->ObjectType == *PsThreadType);
3572
3573 PEPROCESS pProcess = IoThreadToProcess((PETHREAD)pOpInfo->Object);
3574 PSUPDRVNTPROTECT pNtProtect = supdrvNtProtectLookup(PsGetProcessId(pProcess));
3575 pOpInfo->CallContext = pNtProtect; /* Just for reference. */
3576 if (pNtProtect)
3577 {
3578 static ACCESS_MASK const s_fCsrssStupidDesires = 0x1fffff;
3579 ACCESS_MASK fAllowedRights = SUPDRV_NT_ALLOWED_THREAD_RIGHTS;
3580
3581 if (pOpInfo->Operation == OB_OPERATION_HANDLE_CREATE)
3582 {
3583 /* Don't restrict the process accessing its own threads. */
3584 if (pProcess == PsGetCurrentProcess())
3585 {
3586 Log(("vboxdrv/ThreadHandlePre: %sctx=%04zx/%p wants %#x to %p in pid=%04zx [%d] self\n",
3587 pOpInfo->KernelHandle ? "k" : "", PsGetProcessId(PsGetCurrentProcess()), PsGetCurrentProcess(),
3588 pOpInfo->Parameters->CreateHandleInformation.DesiredAccess,
3589 pOpInfo->Object, pNtProtect->AvlCore.Key, pNtProtect->enmProcessKind));
3590 pOpInfo->CallContext = NULL; /* don't assert */
3591 pNtProtect->fFirstThreadCreateHandle = false;
3592 }
3593#ifdef VBOX_WITHOUT_DEBUGGER_CHECKS
3594 /* Allow debuggers full access. */
3595 else if (supdrvNtProtectIsWhitelistedDebugger(PsGetCurrentProcess()))
3596 {
3597 Log(("vboxdrv/ThreadHandlePre: %sctx=%04zx/%p wants %#x to %p in pid=%04zx [%d] %s [debugger]\n",
3598 pOpInfo->KernelHandle ? "k" : "", PsGetProcessId(PsGetCurrentProcess()), PsGetCurrentProcess(),
3599 pOpInfo->Parameters->CreateHandleInformation.DesiredAccess,
3600 pOpInfo->Object, pNtProtect->AvlCore.Key, pNtProtect->enmProcessKind,
3601 PsGetProcessImageFileName(PsGetCurrentProcess()) ));
3602 pOpInfo->CallContext = NULL; /* don't assert */
3603 }
3604#endif
3605 else
3606 {
3607 /* Special case 1 on Vista, 7, 8:
3608 The CreateProcess code passes the handle over to CSRSS.EXE
3609 and the code inBaseSrvCreateProcess will duplicate the
3610 handle with 0x1fffff as access mask. NtDuplicateObject will
3611 fail this call before it ever gets down here. */
3612 if ( g_uNtVerCombined < SUP_MAKE_NT_VER_SIMPLE(6, 3)
3613 && pNtProtect->enmProcessKind == kSupDrvNtProtectKind_VmProcessUnconfirmed
3614 && pNtProtect->fFirstThreadCreateHandle
3615 && pOpInfo->KernelHandle == 0
3616 && ExGetPreviousMode() == UserMode
3617 && pNtProtect->hParentPid == PsGetProcessId(PsGetCurrentProcess()) )
3618 {
3619 if ( !pOpInfo->KernelHandle
3620 && pOpInfo->Parameters->CreateHandleInformation.DesiredAccess == s_fCsrssStupidDesires)
3621 {
3622 fAllowedRights |= s_fCsrssStupidDesires;
3623 pOpInfo->CallContext = NULL; /* don't assert this. */
3624 }
3625 pNtProtect->fFirstThreadCreateHandle = false;
3626 }
3627
3628 /* Special case 2 on 8.1, possibly also Vista, 7, 8:
3629 When creating a process like VBoxTestOGL from the VM process,
3630 CSRSS.EXE will try talk to the calling thread and, it
3631 appears, impersonate it. We unfortunately need to allow
3632 this or there will be no 3D support. Typical DbgPrint:
3633 "SXS: BasepCreateActCtx() Calling csrss server failed. Status = 0xc00000a5" */
3634 SUPDRVNTPROTECTKIND enmProcessKind;
3635 if ( g_uNtVerCombined >= SUP_MAKE_NT_VER_COMBINED(6, 0, 0, 0, 0)
3636 && ( (enmProcessKind = pNtProtect->enmProcessKind) == kSupDrvNtProtectKind_VmProcessConfirmed
3637 || enmProcessKind == kSupDrvNtProtectKind_VmProcessUnconfirmed)
3638 && pOpInfo->KernelHandle == 0
3639 && ExGetPreviousMode() == UserMode
3640 && supdrvNtProtectIsAssociatedCsrss(pNtProtect, PsGetCurrentProcess()) )
3641 {
3642 fAllowedRights |= THREAD_IMPERSONATE;
3643 fAllowedRights |= THREAD_DIRECT_IMPERSONATION;
3644 //fAllowedRights |= THREAD_SET_LIMITED_INFORMATION; - try without this one
3645 pOpInfo->CallContext = NULL; /* don't assert this. */
3646 }
3647
3648 Log(("vboxdrv/ThreadHandlePre: %sctx=%04zx/%p wants %#x to %p in pid=%04zx [%d], allow %#x => %#x; %s [prev=%#x]\n",
3649 pOpInfo->KernelHandle ? "k" : "", PsGetProcessId(PsGetCurrentProcess()), PsGetCurrentProcess(),
3650 pOpInfo->Parameters->CreateHandleInformation.DesiredAccess,
3651 pOpInfo->Object, pNtProtect->AvlCore.Key, pNtProtect->enmProcessKind, fAllowedRights,
3652 pOpInfo->Parameters->CreateHandleInformation.DesiredAccess & fAllowedRights,
3653 PsGetProcessImageFileName(PsGetCurrentProcess()), ExGetPreviousMode()));
3654
3655 pOpInfo->Parameters->CreateHandleInformation.DesiredAccess &= fAllowedRights;
3656 }
3657 }
3658 else
3659 {
3660 /* Don't restrict the process accessing its own threads. */
3661 if ( pProcess == PsGetCurrentProcess()
3662 && (PEPROCESS)pOpInfo->Parameters->DuplicateHandleInformation.TargetProcess == pProcess)
3663 {
3664 Log(("vboxdrv/ThreadHandlePre: %sctx=%04zx/%p[%p] dup from %04zx/%p with %#x to %p in pid=%04zx [%d] self\n",
3665 pOpInfo->KernelHandle ? "k" : "", PsGetProcessId(PsGetCurrentProcess()), PsGetCurrentProcess(),
3666 pOpInfo->Parameters->DuplicateHandleInformation.TargetProcess,
3667 PsGetProcessId((PEPROCESS)pOpInfo->Parameters->DuplicateHandleInformation.SourceProcess),
3668 pOpInfo->Parameters->DuplicateHandleInformation.SourceProcess,
3669 pOpInfo->Parameters->DuplicateHandleInformation.DesiredAccess,
3670 pOpInfo->Object, pNtProtect->AvlCore.Key, pNtProtect->enmProcessKind,
3671 PsGetProcessImageFileName(PsGetCurrentProcess()) ));
3672 pOpInfo->CallContext = NULL; /* don't assert */
3673 }
3674 else
3675 {
3676 /* Special case 3 on Vista, 7, 8:
3677 This is the follow up to special case 1. */
3678 SUPDRVNTPROTECTKIND enmProcessKind;
3679 if ( g_uNtVerCombined >= SUP_MAKE_NT_VER_COMBINED(6, 0, 0, 0, 0)
3680 && ( (enmProcessKind = pNtProtect->enmProcessKind) == kSupDrvNtProtectKind_VmProcessConfirmed
3681 || enmProcessKind == kSupDrvNtProtectKind_VmProcessUnconfirmed)
3682 && pOpInfo->Parameters->DuplicateHandleInformation.TargetProcess == PsGetCurrentProcess()
3683 && pOpInfo->KernelHandle == 0
3684 && ExGetPreviousMode() == UserMode
3685 && supdrvNtProtectIsAssociatedCsrss(pNtProtect, PsGetCurrentProcess()) )
3686 {
3687 fAllowedRights |= THREAD_IMPERSONATE;
3688 fAllowedRights |= THREAD_DIRECT_IMPERSONATION;
3689 //fAllowedRights |= THREAD_SET_LIMITED_INFORMATION; - try without this one
3690 pOpInfo->CallContext = NULL; /* don't assert this. */
3691 }
3692
3693 Log(("vboxdrv/ThreadHandlePre: %sctx=%04zx/%p[%p] dup from %04zx/%p with %#x to %p in pid=%04zx [%d], allow %#x => %#x; %s\n",
3694 pOpInfo->KernelHandle ? "k" : "", PsGetProcessId(PsGetCurrentProcess()), PsGetCurrentProcess(),
3695 pOpInfo->Parameters->DuplicateHandleInformation.TargetProcess,
3696 PsGetProcessId((PEPROCESS)pOpInfo->Parameters->DuplicateHandleInformation.SourceProcess),
3697 pOpInfo->Parameters->DuplicateHandleInformation.SourceProcess,
3698 pOpInfo->Parameters->DuplicateHandleInformation.DesiredAccess,
3699 pOpInfo->Object, pNtProtect->AvlCore.Key, pNtProtect->enmProcessKind, fAllowedRights,
3700 pOpInfo->Parameters->DuplicateHandleInformation.DesiredAccess & fAllowedRights,
3701 PsGetProcessImageFileName(PsGetCurrentProcess()) ));
3702
3703 pOpInfo->Parameters->DuplicateHandleInformation.DesiredAccess &= fAllowedRights;
3704 }
3705 }
3706
3707 supdrvNtProtectRelease(pNtProtect);
3708 }
3709
3710 return OB_PREOP_SUCCESS;
3711}
3712
3713
3714static VOID __stdcall
3715supdrvNtProtectCallback_ThreadHandlePost(PVOID pvUser, POB_POST_OPERATION_INFORMATION pOpInfo)
3716{
3717 Assert(pvUser == NULL);
3718 Assert(pOpInfo->Operation == OB_OPERATION_HANDLE_CREATE || pOpInfo->Operation == OB_OPERATION_HANDLE_DUPLICATE);
3719 Assert(pOpInfo->ObjectType == *PsThreadType);
3720
3721 if ( pOpInfo->CallContext
3722 && NT_SUCCESS(pOpInfo->ReturnStatus))
3723 {
3724 ACCESS_MASK const fGrantedAccess = pOpInfo->Parameters->CreateHandleInformation.GrantedAccess;
3725 AssertReleaseMsg( !(fGrantedAccess & ~( SUPDRV_NT_ALLOWED_THREAD_RIGHTS
3726 | WRITE_OWNER | WRITE_DAC /* these two might be forced upon us */
3727 | THREAD_RESUME /* This seems to be force upon us too with 8.1. */
3728 ) )
3729 || pOpInfo->KernelHandle,
3730 ("GrantedAccess=%#x - we allow %#x - we did not allow %#x\n",
3731 fGrantedAccess, SUPDRV_NT_ALLOWED_THREAD_RIGHTS, fGrantedAccess & ~SUPDRV_NT_ALLOWED_THREAD_RIGHTS));
3732 }
3733}
3734
3735# undef SUPDRV_NT_ALLOWED_THREAD_RIGHTS
3736
3737/** @} */
3738
3739
3740/**
3741 * Creates a new process protection structure.
3742 *
3743 * @returns VBox status code.
3744 * @param ppNtProtect Where to return the pointer to the structure
3745 * on success.
3746 * @param hPid The process ID of the process to protect.
3747 * @param enmProcessKind The kind of process we're protecting.
3748 * @param fLink Whether to link the structure into the tree.
3749 */
3750static int supdrvNtProtectCreate(PSUPDRVNTPROTECT *ppNtProtect, HANDLE hPid, SUPDRVNTPROTECTKIND enmProcessKind, bool fLink)
3751{
3752 AssertReturn(g_hNtProtectLock != NIL_RTSPINLOCK, VERR_WRONG_ORDER);
3753
3754 PSUPDRVNTPROTECT pNtProtect = (PSUPDRVNTPROTECT)RTMemAllocZ(sizeof(*pNtProtect));
3755 if (!pNtProtect)
3756 return VERR_NO_MEMORY;
3757
3758 pNtProtect->AvlCore.Key = hPid;
3759 pNtProtect->u32Magic = SUPDRVNTPROTECT_MAGIC;
3760 pNtProtect->cRefs = 1;
3761 pNtProtect->enmProcessKind = enmProcessKind;
3762 pNtProtect->hParentPid = NULL;
3763 pNtProtect->hOpenTid = NULL;
3764 pNtProtect->hCsrssPid = NULL;
3765 pNtProtect->pCsrssProcess = NULL;
3766
3767 if (fLink)
3768 {
3769 RTSpinlockAcquire(g_hNtProtectLock);
3770 bool fSuccess = RTAvlPVInsert(&g_NtProtectTree, &pNtProtect->AvlCore);
3771 pNtProtect->fInTree = fSuccess;
3772 RTSpinlockRelease(g_hNtProtectLock);
3773
3774 if (!fSuccess)
3775 {
3776 /* Duplicate entry, fail. */
3777 pNtProtect->u32Magic = SUPDRVNTPROTECT_MAGIC_DEAD;
3778 LogRel(("supdrvNtProtectCreate: Duplicate (%#x).\n", pNtProtect->AvlCore.Key));
3779 RTMemFree(pNtProtect);
3780 return VERR_DUPLICATE;
3781 }
3782 }
3783
3784 *ppNtProtect = pNtProtect;
3785 return VINF_SUCCESS;
3786}
3787
3788
3789/**
3790 * Releases a reference to a NT protection structure.
3791 *
3792 * @param pNtProtect The NT protection structure.
3793 */
3794static void supdrvNtProtectRelease(PSUPDRVNTPROTECT pNtProtect)
3795{
3796 if (!pNtProtect)
3797 return;
3798 AssertReturnVoid(pNtProtect->u32Magic == SUPDRVNTPROTECT_MAGIC);
3799
3800 RTSpinlockAcquire(g_hNtProtectLock);
3801 uint32_t cRefs = ASMAtomicDecU32(&pNtProtect->cRefs);
3802 if (cRefs != 0)
3803 RTSpinlockRelease(g_hNtProtectLock);
3804 else
3805 {
3806 /*
3807 * That was the last reference. Remove it from the tree, invalidate it
3808 * and free the resources associated with it. Also, release any
3809 * child/parent references related to this protection structure.
3810 */
3811 ASMAtomicWriteU32(&pNtProtect->u32Magic, SUPDRVNTPROTECT_MAGIC_DEAD);
3812 if (pNtProtect->fInTree)
3813 {
3814 PSUPDRVNTPROTECT pRemoved = (PSUPDRVNTPROTECT)RTAvlPVRemove(&g_NtProtectTree, pNtProtect->AvlCore.Key);
3815 Assert(pRemoved == pNtProtect);
3816 pNtProtect->fInTree = false;
3817 }
3818
3819 PSUPDRVNTPROTECT pChild = NULL;
3820 if (pNtProtect->enmProcessKind == kSupDrvNtProtectKind_StubParent)
3821 {
3822 pChild = pNtProtect->u.pChild;
3823 if (pChild)
3824 {
3825 pNtProtect->u.pChild = NULL;
3826 pChild->u.pParent = NULL;
3827 pChild->enmProcessKind = kSupDrvNtProtectKind_VmProcessDead;
3828 uint32_t cChildRefs = ASMAtomicDecU32(&pChild->cRefs);
3829 if (!cChildRefs)
3830 {
3831 Assert(pChild->fInTree);
3832 if (pChild->fInTree)
3833 {
3834 PSUPDRVNTPROTECT pRemovedChild = (PSUPDRVNTPROTECT)RTAvlPVRemove(&g_NtProtectTree, pChild->AvlCore.Key);
3835 Assert(pRemovedChild == pChild);
3836 pChild->fInTree = false;
3837 }
3838 }
3839 else
3840 pChild = NULL;
3841 }
3842 }
3843 else
3844 AssertRelease(pNtProtect->enmProcessKind != kSupDrvNtProtectKind_VmProcessUnconfirmed);
3845
3846 RTSpinlockRelease(g_hNtProtectLock);
3847
3848 if (pNtProtect->pCsrssProcess)
3849 {
3850 ObDereferenceObject(pNtProtect->pCsrssProcess);
3851 pNtProtect->pCsrssProcess = NULL;
3852 }
3853
3854 RTMemFree(pNtProtect);
3855 if (pChild)
3856 RTMemFree(pChild);
3857 }
3858}
3859
3860
3861/**
3862 * Looks up a PID in the NT protect tree.
3863 *
3864 * @returns Pointer to a NT protection structure (with a referenced) on success,
3865 * NULL if not found.
3866 * @param hPid The process ID.
3867 */
3868static PSUPDRVNTPROTECT supdrvNtProtectLookup(HANDLE hPid)
3869{
3870 RTSpinlockAcquire(g_hNtProtectLock);
3871 PSUPDRVNTPROTECT pFound = (PSUPDRVNTPROTECT)RTAvlPVGet(&g_NtProtectTree, hPid);
3872 if (pFound)
3873 ASMAtomicIncU32(&pFound->cRefs);
3874 RTSpinlockRelease(g_hNtProtectLock);
3875 return pFound;
3876}
3877
3878
3879/**
3880 * Validates a few facts about the stub process when the VM process opens
3881 * vboxdrv.
3882 *
3883 * This makes sure the stub process is still around and that it has neither
3884 * debugger nor extra threads in it.
3885 *
3886 * @returns VBox status code.
3887 * @param pNtProtect The unconfirmed VM process currently trying to
3888 * open vboxdrv.
3889 * @param pErrInfo Additional error information.
3890 */
3891static int supdrvNtProtectVerifyStubForVmProcess(PSUPDRVNTPROTECT pNtProtect, PRTERRINFO pErrInfo)
3892{
3893 /*
3894 * Grab a reference to the parent stub process.
3895 */
3896 SUPDRVNTPROTECTKIND enmStub = kSupDrvNtProtectKind_Invalid;
3897 PSUPDRVNTPROTECT pNtStub = NULL;
3898 RTSpinlockAcquire(g_hNtProtectLock);
3899 if (pNtProtect->enmProcessKind == kSupDrvNtProtectKind_VmProcessUnconfirmed)
3900 {
3901 pNtStub = pNtProtect->u.pParent; /* weak reference. */
3902 if (pNtStub)
3903 {
3904 enmStub = pNtStub->enmProcessKind;
3905 if (enmStub == kSupDrvNtProtectKind_StubParent)
3906 {
3907 uint32_t cRefs = ASMAtomicIncU32(&pNtStub->cRefs);
3908 Assert(cRefs > 0 && cRefs < 1024);
3909 }
3910 else
3911 pNtStub = NULL;
3912 }
3913 }
3914 RTSpinlockRelease(g_hNtProtectLock);
3915
3916 /*
3917 * We require the stub process to be present.
3918 */
3919 if (!pNtStub)
3920 return RTErrInfoSetF(pErrInfo, VERR_SUP_VP_STUB_NOT_FOUND, "Missing stub process (enmStub=%d).", enmStub);
3921
3922 /*
3923 * Open the parent process and thread so we can check for debuggers and unwanted threads.
3924 */
3925 int rc;
3926 PEPROCESS pStubProcess;
3927 NTSTATUS rcNt = PsLookupProcessByProcessId(pNtStub->AvlCore.Key, &pStubProcess);
3928 if (NT_SUCCESS(rcNt))
3929 {
3930 HANDLE hStubProcess;
3931 rcNt = ObOpenObjectByPointer(pStubProcess, OBJ_KERNEL_HANDLE, NULL /*PassedAccessState*/,
3932 0 /*DesiredAccess*/, *PsProcessType, KernelMode, &hStubProcess);
3933 if (NT_SUCCESS(rcNt))
3934 {
3935 PETHREAD pStubThread;
3936 rcNt = PsLookupThreadByThreadId(pNtStub->hOpenTid, &pStubThread);
3937 if (NT_SUCCESS(rcNt))
3938 {
3939 HANDLE hStubThread;
3940 rcNt = ObOpenObjectByPointer(pStubThread, OBJ_KERNEL_HANDLE, NULL /*PassedAccessState*/,
3941 0 /*DesiredAccess*/, *PsThreadType, KernelMode, &hStubThread);
3942 if (NT_SUCCESS(rcNt))
3943 {
3944 /*
3945 * Do some simple sanity checking.
3946 */
3947 rc = supHardNtVpDebugger(hStubProcess, pErrInfo);
3948 if (RT_SUCCESS(rc))
3949 rc = supHardNtVpThread(hStubProcess, hStubThread, pErrInfo);
3950
3951 /* Clean up. */
3952 rcNt = NtClose(hStubThread); AssertMsg(NT_SUCCESS(rcNt), ("%#x\n", rcNt));
3953 }
3954 else
3955 rc = RTErrInfoSetF(pErrInfo, VERR_SUP_VP_STUB_THREAD_OPEN_ERROR,
3956 "Error opening stub thread %p (tid %p, pid %p): %#x",
3957 pStubThread, pNtStub->hOpenTid, pNtStub->AvlCore.Key, rcNt);
3958 }
3959 else
3960 rc = RTErrInfoSetF(pErrInfo, VERR_SUP_VP_STUB_THREAD_NOT_FOUND,
3961 "Failed to locate thread %p in %p: %#x", pNtStub->hOpenTid, pNtStub->AvlCore.Key, rcNt);
3962 rcNt = NtClose(hStubProcess); AssertMsg(NT_SUCCESS(rcNt), ("%#x\n", rcNt));
3963 }
3964 else
3965 rc = RTErrInfoSetF(pErrInfo, VERR_SUP_VP_STUB_OPEN_ERROR,
3966 "Error opening stub process %p (pid %p): %#x", pStubProcess, pNtStub->AvlCore.Key, rcNt);
3967 ObDereferenceObject(pStubProcess);
3968 }
3969 else
3970 rc = RTErrInfoSetF(pErrInfo, VERR_SUP_VP_STUB_NOT_FOUND,
3971 "Failed to locate stub process %p: %#x", pNtStub->AvlCore.Key, rcNt);
3972
3973 supdrvNtProtectRelease(pNtStub);
3974 return rc;
3975}
3976
3977
3978/**
3979 * Worker for supdrvNtProtectVerifyProcess that verifies the handles to a VM
3980 * process and its thread.
3981 *
3982 * @returns VBox status code.
3983 * @param pNtProtect The NT protect structure for getting information
3984 * about special processes.
3985 * @param pErrInfo Where to return additional error details.
3986 */
3987static int supdrvNtProtectRestrictHandlesToProcessAndThread(PSUPDRVNTPROTECT pNtProtect, PRTERRINFO pErrInfo)
3988{
3989 /*
3990 * What to protect.
3991 */
3992 PEPROCESS pProtectedProcess = PsGetCurrentProcess();
3993 HANDLE hProtectedPid = PsGetProcessId(pProtectedProcess);
3994 PETHREAD pProtectedThread = PsGetCurrentThread();
3995 AssertReturn(pNtProtect->AvlCore.Key == hProtectedPid, VERR_INTERNAL_ERROR_5);
3996
3997 /*
3998 * Take a snapshot of all the handles in the system.
3999 * Note! The 32 bytes on the size of to counteract the allocation header
4000 * that rtR0MemAllocEx slaps on everything.
4001 */
4002 uint32_t cbBuf = _256K - 32;
4003 uint8_t *pbBuf = (uint8_t *)RTMemAlloc(cbBuf);
4004 ULONG cbNeeded = cbBuf;
4005 NTSTATUS rcNt = NtQuerySystemInformation(SystemExtendedHandleInformation, pbBuf, cbBuf, &cbNeeded);
4006 if (!NT_SUCCESS(rcNt))
4007 {
4008 while ( rcNt == STATUS_INFO_LENGTH_MISMATCH
4009 && cbNeeded > cbBuf
4010 && cbBuf <= 32U*_1M)
4011 {
4012 cbBuf = RT_ALIGN_32(cbNeeded + _4K, _64K) - 32;
4013 RTMemFree(pbBuf);
4014 pbBuf = (uint8_t *)RTMemAlloc(cbBuf);
4015 if (!pbBuf)
4016 return RTErrInfoSetF(pErrInfo, VERR_NO_MEMORY, "Error allocating %zu bytes for querying handles.", cbBuf);
4017 rcNt = NtQuerySystemInformation(SystemExtendedHandleInformation, pbBuf, cbBuf, &cbNeeded);
4018 }
4019 if (!NT_SUCCESS(rcNt))
4020 {
4021 RTMemFree(pbBuf);
4022 return RTErrInfoSetF(pErrInfo, RTErrConvertFromNtStatus(rcNt),
4023 "NtQuerySystemInformation/SystemExtendedHandleInformation failed: %#x\n", rcNt);
4024 }
4025 }
4026
4027 /*
4028 * Walk the information and look for handles to the two objects we're protecting.
4029 */
4030 int rc = VINF_SUCCESS;
4031# ifdef VBOX_WITHOUT_DEBUGGER_CHECKS
4032 HANDLE idLastDebugger = (HANDLE)~(uintptr_t)0;
4033# endif
4034
4035 uint32_t cCsrssProcessHandles = 0;
4036 uint32_t cSystemProcessHandles = 0;
4037 uint32_t cEvilProcessHandles = 0;
4038 uint32_t cBenignProcessHandles = 0;
4039
4040 uint32_t cCsrssThreadHandles = 0;
4041 uint32_t cEvilThreadHandles = 0;
4042 uint32_t cBenignThreadHandles = 0;
4043
4044 SYSTEM_HANDLE_INFORMATION_EX const *pInfo = (SYSTEM_HANDLE_INFORMATION_EX const *)pbBuf;
4045 ULONG_PTR i = pInfo->NumberOfHandles;
4046 AssertRelease(RT_OFFSETOF(SYSTEM_HANDLE_INFORMATION_EX, Handles[i]) == cbNeeded);
4047 while (i-- > 0)
4048 {
4049 const char *pszType;
4050 SYSTEM_HANDLE_ENTRY_INFO_EX const *pHandleInfo = &pInfo->Handles[i];
4051 if (pHandleInfo->Object == pProtectedProcess)
4052 {
4053 /* Handles within the protected process are fine. */
4054 if ( !(pHandleInfo->GrantedAccess & SUPDRV_NT_EVIL_PROCESS_RIGHTS)
4055 || pHandleInfo->UniqueProcessId == hProtectedPid)
4056 {
4057 cBenignProcessHandles++;
4058 continue;
4059 }
4060
4061 /* CSRSS is allowed to have one evil process handle.
4062 See the special cases in the hook code. */
4063 if ( cCsrssProcessHandles < 1
4064 && pHandleInfo->UniqueProcessId == pNtProtect->hCsrssPid)
4065 {
4066 cCsrssProcessHandles++;
4067 continue;
4068 }
4069
4070 /* The system process is allowed having two open process handle in
4071 Windows 8.1 and later, and one in earlier. This is probably a
4072 little overly paranoid as I think we can safely trust the
4073 system process... */
4074 if ( cSystemProcessHandles < (g_uNtVerCombined >= SUP_MAKE_NT_VER_SIMPLE(6, 3) ? UINT32_C(2) : UINT32_C(1))
4075 && pHandleInfo->UniqueProcessId == PsGetProcessId(PsInitialSystemProcess))
4076 {
4077 cSystemProcessHandles++;
4078 continue;
4079 }
4080
4081 cEvilProcessHandles++;
4082 pszType = "process";
4083 }
4084 else if (pHandleInfo->Object == pProtectedThread)
4085 {
4086 /* Handles within the protected process is fine. */
4087 if ( !(pHandleInfo->GrantedAccess & SUPDRV_NT_EVIL_THREAD_RIGHTS)
4088 || pHandleInfo->UniqueProcessId == hProtectedPid)
4089 {
4090 cBenignThreadHandles++;
4091 continue;
4092 }
4093
4094 /* CSRSS is allowed to have one evil handle to the primary thread
4095 for LPC purposes. See the hook for special case. */
4096 if ( cCsrssThreadHandles < 1
4097 && pHandleInfo->UniqueProcessId == pNtProtect->hCsrssPid)
4098 {
4099 cCsrssThreadHandles++;
4100 continue;
4101 }
4102
4103 cEvilThreadHandles++;
4104 pszType = "thread";
4105 }
4106 else
4107 continue;
4108
4109# ifdef VBOX_WITHOUT_DEBUGGER_CHECKS
4110 /* Ignore whitelisted debuggers. */
4111 if (pHandleInfo->UniqueProcessId == idLastDebugger)
4112 continue;
4113 PEPROCESS pDbgProc;
4114 NTSTATUS rcNt = PsLookupProcessByProcessId(pHandleInfo->UniqueProcessId, &pDbgProc);
4115 if (NT_SUCCESS(rcNt))
4116 {
4117 bool fIsDebugger = supdrvNtProtectIsWhitelistedDebugger(pDbgProc);
4118 ObDereferenceObject(pDbgProc);
4119 if (fIsDebugger)
4120 {
4121 idLastDebugger = pHandleInfo->UniqueProcessId;
4122 continue;
4123 }
4124 }
4125# endif
4126
4127 /* Found evil handle. Currently ignoring on pre-Vista. */
4128# ifndef VBOX_WITH_VISTA_NO_SP
4129 if ( g_uNtVerCombined >= SUP_NT_VER_VISTA
4130# else
4131 if ( g_uNtVerCombined >= SUP_MAKE_NT_VER_COMBINED(6, 0, 6001, 0, 0)
4132# endif
4133 || g_pfnObRegisterCallbacks)
4134 {
4135 LogRel(("vboxdrv: Found evil handle to budding VM process: pid=%p h=%p acc=%#x attr=%#x type=%s\n",
4136 pHandleInfo->UniqueProcessId, pHandleInfo->HandleValue,
4137 pHandleInfo->GrantedAccess, pHandleInfo->HandleAttributes, pszType));
4138 rc = RTErrInfoAddF(pErrInfo, VERR_SUPDRV_HARDENING_EVIL_HANDLE,
4139 *pErrInfo->pszMsg
4140 ? "\nFound evil handle to budding VM process: pid=%p h=%p acc=%#x attr=%#x type=%s"
4141 : "Found evil handle to budding VM process: pid=%p h=%p acc=%#x attr=%#x type=%s",
4142 pHandleInfo->UniqueProcessId, pHandleInfo->HandleValue,
4143 pHandleInfo->GrantedAccess, pHandleInfo->HandleAttributes, pszType);
4144
4145 /* Try add the process name. */
4146 PEPROCESS pOffendingProcess;
4147 rcNt = PsLookupProcessByProcessId(pHandleInfo->UniqueProcessId, &pOffendingProcess);
4148 if (NT_SUCCESS(rcNt))
4149 {
4150 const char *pszName = (const char *)PsGetProcessImageFileName(pOffendingProcess);
4151 if (pszName && *pszName)
4152 rc = RTErrInfoAddF(pErrInfo, rc, " [%s]", pszName);
4153
4154 ObDereferenceObject(pOffendingProcess);
4155 }
4156 }
4157 }
4158
4159 RTMemFree(pbBuf);
4160 return rc;
4161}
4162
4163
4164/**
4165 * Checks if the current process checks out as a VM process stub.
4166 *
4167 * @returns VBox status code.
4168 * @param pNtProtect The NT protect structure. This is upgraded to a
4169 * final protection kind (state) on success.
4170 */
4171static int supdrvNtProtectVerifyProcess(PSUPDRVNTPROTECT pNtProtect)
4172{
4173 AssertReturn(PsGetProcessId(PsGetCurrentProcess()) == pNtProtect->AvlCore.Key, VERR_INTERNAL_ERROR_3);
4174
4175 /*
4176 * Do the verification. The handle restriction checks are only preformed
4177 * on VM processes.
4178 */
4179 int rc = VINF_SUCCESS;
4180 PSUPDRVNTERRORINFO pErrorInfo = (PSUPDRVNTERRORINFO)RTMemAllocZ(sizeof(*pErrorInfo));
4181 if (RT_SUCCESS(rc))
4182 {
4183 pErrorInfo->hProcessId = PsGetCurrentProcessId();
4184 pErrorInfo->hThreadId = PsGetCurrentThreadId();
4185 RTERRINFO ErrInfo;
4186 RTErrInfoInit(&ErrInfo, pErrorInfo->szErrorInfo, sizeof(pErrorInfo->szErrorInfo));
4187
4188 if (pNtProtect->enmProcessKind >= kSupDrvNtProtectKind_VmProcessUnconfirmed)
4189 rc = supdrvNtProtectRestrictHandlesToProcessAndThread(pNtProtect, &ErrInfo);
4190 if (RT_SUCCESS(rc))
4191 {
4192 rc = supHardenedWinVerifyProcess(NtCurrentProcess(), NtCurrentThread(), SUPHARDNTVPKIND_VERIFY_ONLY, 0 /*fFlags*/,
4193 NULL /*pcFixes*/, &ErrInfo);
4194 if (RT_SUCCESS(rc) && pNtProtect->enmProcessKind >= kSupDrvNtProtectKind_VmProcessUnconfirmed)
4195 rc = supdrvNtProtectVerifyStubForVmProcess(pNtProtect, &ErrInfo);
4196 }
4197 }
4198 else
4199 rc = VERR_NO_MEMORY;
4200
4201 /*
4202 * Upgrade and return.
4203 */
4204 HANDLE hOpenTid = PsGetCurrentThreadId();
4205 RTSpinlockAcquire(g_hNtProtectLock);
4206
4207 /* Stub process verficiation is pretty much straight forward. */
4208 if (pNtProtect->enmProcessKind == kSupDrvNtProtectKind_StubUnverified)
4209 {
4210 pNtProtect->enmProcessKind = RT_SUCCESS(rc) ? kSupDrvNtProtectKind_StubSpawning : kSupDrvNtProtectKind_StubDead;
4211 pNtProtect->hOpenTid = hOpenTid;
4212 }
4213 /* The VM process verification is a little bit more complicated
4214 because we need to drop the parent process reference as well. */
4215 else if (pNtProtect->enmProcessKind == kSupDrvNtProtectKind_VmProcessUnconfirmed)
4216 {
4217 AssertRelease(pNtProtect->cRefs >= 2); /* Parent + Caller */
4218 PSUPDRVNTPROTECT pParent = pNtProtect->u.pParent;
4219 AssertRelease(pParent);
4220 AssertRelease(pParent->u.pParent == pNtProtect);
4221 AssertRelease(pParent->enmProcessKind == kSupDrvNtProtectKind_StubParent);
4222 pParent->u.pParent = NULL;
4223
4224 pNtProtect->u.pParent = NULL;
4225 ASMAtomicDecU32(&pNtProtect->cRefs);
4226
4227 if (RT_SUCCESS(rc))
4228 {
4229 pNtProtect->enmProcessKind = kSupDrvNtProtectKind_VmProcessConfirmed;
4230 pNtProtect->hOpenTid = hOpenTid;
4231 }
4232 else
4233 pNtProtect->enmProcessKind = kSupDrvNtProtectKind_VmProcessDead;
4234 }
4235
4236 /* Since the stub and VM processes are only supposed to have one thread,
4237 we're not supposed to be subject to any races from within the processes.
4238
4239 There is a race between VM process verification and the stub process
4240 exiting, though. We require the stub process to be alive until the new
4241 VM process has made it thru the validation. So, when the stub
4242 terminates the notification handler will change the state of both stub
4243 and VM process to dead.
4244
4245 Also, I'm not entirely certain where the process
4246 termination notification is triggered from, so that can theorically
4247 create a race in both cases. */
4248 else
4249 {
4250 AssertReleaseMsg( pNtProtect->enmProcessKind == kSupDrvNtProtectKind_StubDead
4251 || pNtProtect->enmProcessKind == kSupDrvNtProtectKind_VmProcessDead,
4252 ("enmProcessKind=%d rc=%Rrc\n", pNtProtect->enmProcessKind, rc));
4253 if (RT_SUCCESS(rc))
4254 rc = VERR_INVALID_STATE; /* There should be no races here. */
4255 }
4256
4257 RTSpinlockRelease(g_hNtProtectLock);
4258
4259 /*
4260 * Free error info on success, keep it on failure.
4261 */
4262 if (RT_SUCCESS(rc))
4263 RTMemFree(pErrorInfo);
4264 else if (pErrorInfo)
4265 {
4266 pErrorInfo->cchErrorInfo = (uint32_t)strlen(pErrorInfo->szErrorInfo);
4267 if (!pErrorInfo->cchErrorInfo)
4268 pErrorInfo->cchErrorInfo = (uint32_t)RTStrPrintf(pErrorInfo->szErrorInfo, sizeof(pErrorInfo->szErrorInfo),
4269 "supdrvNtProtectVerifyProcess: rc=%d", rc);
4270 RTLogWriteDebugger(pErrorInfo->szErrorInfo, pErrorInfo->cchErrorInfo);
4271
4272 int rc2 = RTSemMutexRequest(g_hErrorInfoLock, RT_INDEFINITE_WAIT);
4273 if (RT_SUCCESS(rc2))
4274 {
4275 pErrorInfo->uCreatedMsTs = RTTimeMilliTS();
4276
4277 /* Free old entries. */
4278 PSUPDRVNTERRORINFO pCur;
4279 while ( (pCur = RTListGetFirst(&g_ErrorInfoHead, SUPDRVNTERRORINFO, ListEntry)) != NULL
4280 && (int64_t)(pErrorInfo->uCreatedMsTs - pCur->uCreatedMsTs) > 60000 /*60sec*/)
4281 {
4282 RTListNodeRemove(&pCur->ListEntry);
4283 RTMemFree(pCur);
4284 }
4285
4286 /* Insert our new entry. */
4287 RTListAppend(&g_ErrorInfoHead, &pErrorInfo->ListEntry);
4288
4289 RTSemMutexRelease(g_hErrorInfoLock);
4290 }
4291 else
4292 RTMemFree(pErrorInfo);
4293 }
4294
4295 return rc;
4296}
4297
4298
4299# ifndef VBOX_WITHOUT_DEBUGGER_CHECKS
4300
4301/**
4302 * Checks if the current process is being debugged.
4303 * @return @c true if debugged, @c false if not.
4304 */
4305static bool supdrvNtIsDebuggerAttached(void)
4306{
4307 return PsIsProcessBeingDebugged(PsGetCurrentProcess()) != FALSE;
4308}
4309
4310# endif /* !VBOX_WITHOUT_DEBUGGER_CHECKS */
4311
4312
4313/**
4314 * Terminates the hardening bits.
4315 */
4316static void supdrvNtProtectTerm(void)
4317{
4318 /*
4319 * Stop intercepting process and thread handle creation calls.
4320 */
4321 if (g_pvObCallbacksCookie)
4322 {
4323 g_pfnObUnRegisterCallbacks(g_pvObCallbacksCookie);
4324 g_pvObCallbacksCookie = NULL;
4325 }
4326
4327 /*
4328 * Stop intercepting process creation and termination notifications.
4329 */
4330 NTSTATUS rcNt;
4331 if (g_pfnPsSetCreateProcessNotifyRoutineEx)
4332 rcNt = g_pfnPsSetCreateProcessNotifyRoutineEx(supdrvNtProtectCallback_ProcessCreateNotifyEx, TRUE /*fRemove*/);
4333 else
4334 rcNt = PsSetCreateProcessNotifyRoutine(supdrvNtProtectCallback_ProcessCreateNotify, TRUE /*fRemove*/);
4335 AssertMsg(NT_SUCCESS(rcNt), ("rcNt=%#x\n", rcNt));
4336
4337 Assert(g_NtProtectTree == NULL);
4338
4339 /*
4340 * Clean up globals.
4341 */
4342 RTSpinlockDestroy(g_hNtProtectLock);
4343 g_NtProtectTree = NIL_RTSPINLOCK;
4344
4345 RTSemMutexDestroy(g_hErrorInfoLock);
4346 g_hErrorInfoLock = NIL_RTSEMMUTEX;
4347
4348 PSUPDRVNTERRORINFO pCur;
4349 while ((pCur = RTListGetFirst(&g_ErrorInfoHead, SUPDRVNTERRORINFO, ListEntry)) != NULL)
4350 {
4351 RTListNodeRemove(&pCur->ListEntry);
4352 RTMemFree(pCur);
4353 }
4354
4355 supHardenedWinTermImageVerifier();
4356}
4357
4358# ifdef RT_ARCH_X86
4359DECLASM(void) supdrvNtQueryVirtualMemory_0xAF(void);
4360DECLASM(void) supdrvNtQueryVirtualMemory_0xB0(void);
4361DECLASM(void) supdrvNtQueryVirtualMemory_0xB1(void);
4362DECLASM(void) supdrvNtQueryVirtualMemory_0xB2(void);
4363DECLASM(void) supdrvNtQueryVirtualMemory_0xB3(void);
4364DECLASM(void) supdrvNtQueryVirtualMemory_0xB4(void);
4365DECLASM(void) supdrvNtQueryVirtualMemory_0xB5(void);
4366DECLASM(void) supdrvNtQueryVirtualMemory_0xB6(void);
4367DECLASM(void) supdrvNtQueryVirtualMemory_0xB7(void);
4368DECLASM(void) supdrvNtQueryVirtualMemory_0xB8(void);
4369DECLASM(void) supdrvNtQueryVirtualMemory_0xB9(void);
4370DECLASM(void) supdrvNtQueryVirtualMemory_0xBA(void);
4371DECLASM(void) supdrvNtQueryVirtualMemory_0xBB(void);
4372DECLASM(void) supdrvNtQueryVirtualMemory_0xBC(void);
4373DECLASM(void) supdrvNtQueryVirtualMemory_0xBD(void);
4374DECLASM(void) supdrvNtQueryVirtualMemory_0xBE(void);
4375# elif defined(RT_ARCH_AMD64)
4376DECLASM(void) supdrvNtQueryVirtualMemory_0x1F(void);
4377DECLASM(void) supdrvNtQueryVirtualMemory_0x20(void);
4378DECLASM(void) supdrvNtQueryVirtualMemory_0x21(void);
4379DECLASM(void) supdrvNtQueryVirtualMemory_0x22(void);
4380DECLASM(void) supdrvNtQueryVirtualMemory_0x23(void);
4381extern "C" NTSYSAPI NTSTATUS NTAPI ZwRequestWaitReplyPort(HANDLE, PVOID, PVOID);
4382# endif
4383
4384/**
4385 * Initalizes the hardening bits.
4386 *
4387 * @returns NT status code.
4388 */
4389static NTSTATUS supdrvNtProtectInit(void)
4390{
4391 /*
4392 * Initialize the globals.
4393 */
4394
4395 /* The NT version. */
4396 ULONG uMajor, uMinor, uBuild;
4397 PsGetVersion(&uMajor, &uMinor, &uBuild, NULL);
4398 g_uNtVerCombined = SUP_MAKE_NT_VER_COMBINED(uMajor, uMinor, uBuild, 0, 0);
4399
4400 /* Resolve methods we want but isn't available everywhere. */
4401 UNICODE_STRING RoutineName;
4402
4403 RtlInitUnicodeString(&RoutineName, L"ObGetObjectType");
4404 g_pfnObGetObjectType = (PFNOBGETOBJECTTYPE)MmGetSystemRoutineAddress(&RoutineName);
4405
4406 RtlInitUnicodeString(&RoutineName, L"ObRegisterCallbacks");
4407 g_pfnObRegisterCallbacks = (PFNOBREGISTERCALLBACKS)MmGetSystemRoutineAddress(&RoutineName);
4408
4409 RtlInitUnicodeString(&RoutineName, L"ObUnRegisterCallbacks");
4410 g_pfnObUnRegisterCallbacks = (PFNOBUNREGISTERCALLBACKS)MmGetSystemRoutineAddress(&RoutineName);
4411
4412 RtlInitUnicodeString(&RoutineName, L"PsSetCreateProcessNotifyRoutineEx");
4413 g_pfnPsSetCreateProcessNotifyRoutineEx = (PFNPSSETCREATEPROCESSNOTIFYROUTINEEX)MmGetSystemRoutineAddress(&RoutineName);
4414
4415 RtlInitUnicodeString(&RoutineName, L"PsReferenceProcessFilePointer");
4416 g_pfnPsReferenceProcessFilePointer = (PFNPSREFERENCEPROCESSFILEPOINTER)MmGetSystemRoutineAddress(&RoutineName);
4417
4418 RtlInitUnicodeString(&RoutineName, L"PsIsProtectedProcessLight");
4419 g_pfnPsIsProtectedProcessLight = (PFNPSISPROTECTEDPROCESSLIGHT)MmGetSystemRoutineAddress(&RoutineName);
4420
4421 RtlInitUnicodeString(&RoutineName, L"ZwAlpcCreatePort");
4422 g_pfnZwAlpcCreatePort = (PFNZWALPCCREATEPORT)MmGetSystemRoutineAddress(&RoutineName);
4423
4424 RtlInitUnicodeString(&RoutineName, L"ZwQueryVirtualMemory"); /* Yes, using Zw version here. */
4425 g_pfnNtQueryVirtualMemory = (PFNNTQUERYVIRTUALMEMORY)MmGetSystemRoutineAddress(&RoutineName);
4426 if (!g_pfnNtQueryVirtualMemory && g_uNtVerCombined < SUP_NT_VER_VISTA)
4427 {
4428 /* XP & W2K3 doesn't have this function exported, so we've cooked up a
4429 few alternative in the assembly helper file that uses the code in
4430 ZwReadFile with a different eax value. We figure the syscall number
4431 by inspecting ZwQueryVolumeInformationFile as it's the next number. */
4432# ifdef RT_ARCH_X86
4433 uint8_t const *pbCode = (uint8_t const *)(uintptr_t)ZwQueryVolumeInformationFile;
4434 if (*pbCode == 0xb8) /* mov eax, dword */
4435 switch (*(uint32_t const *)&pbCode[1])
4436 {
4437 case 0xb0: g_pfnNtQueryVirtualMemory = (PFNNTQUERYVIRTUALMEMORY)supdrvNtQueryVirtualMemory_0xAF; break; /* just in case */
4438 case 0xb1: g_pfnNtQueryVirtualMemory = (PFNNTQUERYVIRTUALMEMORY)supdrvNtQueryVirtualMemory_0xB0; break; /* just in case */
4439 case 0xb2: g_pfnNtQueryVirtualMemory = (PFNNTQUERYVIRTUALMEMORY)supdrvNtQueryVirtualMemory_0xB1; break; /* just in case */
4440 case 0xb3: g_pfnNtQueryVirtualMemory = (PFNNTQUERYVIRTUALMEMORY)supdrvNtQueryVirtualMemory_0xB2; break; /* XP SP3 */
4441 case 0xb4: g_pfnNtQueryVirtualMemory = (PFNNTQUERYVIRTUALMEMORY)supdrvNtQueryVirtualMemory_0xB2; break; /* just in case */
4442 case 0xb5: g_pfnNtQueryVirtualMemory = (PFNNTQUERYVIRTUALMEMORY)supdrvNtQueryVirtualMemory_0xB3; break; /* just in case */
4443 case 0xb6: g_pfnNtQueryVirtualMemory = (PFNNTQUERYVIRTUALMEMORY)supdrvNtQueryVirtualMemory_0xB4; break; /* just in case */
4444 case 0xb7: g_pfnNtQueryVirtualMemory = (PFNNTQUERYVIRTUALMEMORY)supdrvNtQueryVirtualMemory_0xB5; break; /* just in case */
4445 case 0xb8: g_pfnNtQueryVirtualMemory = (PFNNTQUERYVIRTUALMEMORY)supdrvNtQueryVirtualMemory_0xB6; break; /* just in case */
4446 case 0xb9: g_pfnNtQueryVirtualMemory = (PFNNTQUERYVIRTUALMEMORY)supdrvNtQueryVirtualMemory_0xB7; break; /* just in case */
4447 case 0xba: g_pfnNtQueryVirtualMemory = (PFNNTQUERYVIRTUALMEMORY)supdrvNtQueryVirtualMemory_0xB8; break; /* just in case */
4448 case 0xbb: g_pfnNtQueryVirtualMemory = (PFNNTQUERYVIRTUALMEMORY)supdrvNtQueryVirtualMemory_0xBA; break; /* W2K3 R2 SP2 */
4449 case 0xbc: g_pfnNtQueryVirtualMemory = (PFNNTQUERYVIRTUALMEMORY)supdrvNtQueryVirtualMemory_0xBB; break; /* just in case */
4450 case 0xbd: g_pfnNtQueryVirtualMemory = (PFNNTQUERYVIRTUALMEMORY)supdrvNtQueryVirtualMemory_0xBC; break; /* just in case */
4451 case 0xbe: g_pfnNtQueryVirtualMemory = (PFNNTQUERYVIRTUALMEMORY)supdrvNtQueryVirtualMemory_0xBD; break; /* just in case */
4452 case 0xbf: g_pfnNtQueryVirtualMemory = (PFNNTQUERYVIRTUALMEMORY)supdrvNtQueryVirtualMemory_0xBE; break; /* just in case */
4453 }
4454# elif defined(RT_ARCH_AMD64)
4455 uint8_t const *pbCode = (uint8_t const *)(uintptr_t)ZwRequestWaitReplyPort;
4456 if ( pbCode[ 0] == 0x48 /* mov rax, rsp */
4457 && pbCode[ 1] == 0x8b
4458 && pbCode[ 2] == 0xc4
4459 && pbCode[ 3] == 0xfa /* cli */
4460 && pbCode[ 4] == 0x48 /* sub rsp, 10h */
4461 && pbCode[ 5] == 0x83
4462 && pbCode[ 6] == 0xec
4463 && pbCode[ 7] == 0x10
4464 && pbCode[ 8] == 0x50 /* push rax */
4465 && pbCode[ 9] == 0x9c /* pushfq */
4466 && pbCode[10] == 0x6a /* push 10 */
4467 && pbCode[11] == 0x10
4468 && pbCode[12] == 0x48 /* lea rax, [nt!KiServiceLinkage] */
4469 && pbCode[13] == 0x8d
4470 && pbCode[14] == 0x05
4471 && pbCode[19] == 0x50 /* push rax */
4472 && pbCode[20] == 0xb8 /* mov eax,1fh <- the syscall no. */
4473 /*&& pbCode[21] == 0x1f*/
4474 && pbCode[22] == 0x00
4475 && pbCode[23] == 0x00
4476 && pbCode[24] == 0x00
4477 && pbCode[25] == 0xe9 /* jmp KiServiceInternal */
4478 )
4479 {
4480 uint8_t const *pbKiServiceInternal = &pbCode[30] + *(int32_t const *)&pbCode[26];
4481 uint8_t const *pbKiServiceLinkage = &pbCode[19] + *(int32_t const *)&pbCode[15];
4482 if (*pbKiServiceLinkage == 0xc3)
4483 {
4484 g_pfnKiServiceInternal = (PFNRT)pbKiServiceInternal;
4485 g_pfnKiServiceLinkage = (PFNRT)pbKiServiceLinkage;
4486 switch (pbCode[21])
4487 {
4488 case 0x1e: g_pfnNtQueryVirtualMemory = (PFNNTQUERYVIRTUALMEMORY)supdrvNtQueryVirtualMemory_0x1F; break;
4489 case 0x1f: g_pfnNtQueryVirtualMemory = (PFNNTQUERYVIRTUALMEMORY)supdrvNtQueryVirtualMemory_0x20; break;
4490 case 0x20: g_pfnNtQueryVirtualMemory = (PFNNTQUERYVIRTUALMEMORY)supdrvNtQueryVirtualMemory_0x21; break;
4491 case 0x21: g_pfnNtQueryVirtualMemory = (PFNNTQUERYVIRTUALMEMORY)supdrvNtQueryVirtualMemory_0x22; break;
4492 case 0x22: g_pfnNtQueryVirtualMemory = (PFNNTQUERYVIRTUALMEMORY)supdrvNtQueryVirtualMemory_0x23; break;
4493 }
4494 }
4495 }
4496# endif
4497 }
4498 if (!g_pfnNtQueryVirtualMemory)
4499 {
4500 LogRel(("vboxdrv: Cannot locate ZwQueryVirtualMemory in ntoskrnl, nor were we able to cook up a replacement.\n"));
4501 return STATUS_PROCEDURE_NOT_FOUND;
4502 }
4503
4504# ifdef VBOX_STRICT
4505 if ( g_uNtVerCombined >= SUP_NT_VER_W70
4506 && ( g_pfnObGetObjectType == NULL
4507 || g_pfnZwAlpcCreatePort == NULL) )
4508 {
4509 LogRel(("vboxdrv: g_pfnObGetObjectType=%p g_pfnZwAlpcCreatePort=%p.\n", g_pfnObGetObjectType, g_pfnZwAlpcCreatePort));
4510 return STATUS_PROCEDURE_NOT_FOUND;
4511 }
4512# endif
4513
4514 /* LPC object type. */
4515 g_pAlpcPortObjectType1 = *LpcPortObjectType;
4516
4517 /* The spinlock protecting our structures. */
4518 int rc = RTSpinlockCreate(&g_hNtProtectLock, RTSPINLOCK_FLAGS_INTERRUPT_UNSAFE, "NtProtectLock");
4519 if (RT_FAILURE(rc))
4520 return VBoxDrvNtErr2NtStatus(rc);
4521 g_NtProtectTree = NULL;
4522
4523 NTSTATUS rcNt;
4524
4525 /* The mutex protecting the error information. */
4526 RTListInit(&g_ErrorInfoHead);
4527 rc = RTSemMutexCreate(&g_hErrorInfoLock);
4528 if (RT_SUCCESS(rc))
4529 {
4530 /* Image stuff + certificates. */
4531 rc = supHardenedWinInitImageVerifier(NULL);
4532 if (RT_SUCCESS(rc))
4533 {
4534 /*
4535 * Intercept process creation and termination.
4536 */
4537 if (g_pfnPsSetCreateProcessNotifyRoutineEx)
4538 rcNt = g_pfnPsSetCreateProcessNotifyRoutineEx(supdrvNtProtectCallback_ProcessCreateNotifyEx, FALSE /*fRemove*/);
4539 else
4540 rcNt = PsSetCreateProcessNotifyRoutine(supdrvNtProtectCallback_ProcessCreateNotify, FALSE /*fRemove*/);
4541 if (NT_SUCCESS(rcNt))
4542 {
4543 /*
4544 * Intercept process and thread handle creation calls.
4545 * The preferred method is only available on Vista SP1+.
4546 */
4547 if (g_pfnObRegisterCallbacks && g_pfnObUnRegisterCallbacks)
4548 {
4549 static OB_OPERATION_REGISTRATION s_aObOperations[] =
4550 {
4551 {
4552 PsProcessType,
4553 OB_OPERATION_HANDLE_CREATE | OB_OPERATION_HANDLE_DUPLICATE,
4554 supdrvNtProtectCallback_ProcessHandlePre,
4555 supdrvNtProtectCallback_ProcessHandlePost,
4556 },
4557 {
4558 PsThreadType,
4559 OB_OPERATION_HANDLE_CREATE | OB_OPERATION_HANDLE_DUPLICATE,
4560 supdrvNtProtectCallback_ThreadHandlePre,
4561 supdrvNtProtectCallback_ThreadHandlePost,
4562 },
4563 };
4564 static OB_CALLBACK_REGISTRATION s_ObCallbackReg =
4565 {
4566 /* .Version = */ OB_FLT_REGISTRATION_VERSION,
4567 /* .OperationRegistrationCount = */ RT_ELEMENTS(s_aObOperations),
4568 /* .Altitude.Length = */ 0,
4569 /* .Altitude.MaximumLength = */ 0,
4570 /* .Altitude.Buffer = */ NULL,
4571 /* .RegistrationContext = */ NULL,
4572 /* .OperationRegistration = */ &s_aObOperations[0]
4573 };
4574 static WCHAR const *s_apwszAltitudes[] = /** @todo get a valid number */
4575 {
4576 L"48596.98940", L"46935.19485", L"49739.39704", L"40334.74976",
4577 L"66667.98940", L"69888.19485", L"69889.39704", L"60364.74976",
4578 L"85780.98940", L"88978.19485", L"89939.39704", L"80320.74976",
4579 L"329879.98940", L"326787.19485", L"328915.39704", L"320314.74976",
4580 };
4581
4582 rcNt = STATUS_FLT_INSTANCE_ALTITUDE_COLLISION;
4583 for (uint32_t i = 0; i < RT_ELEMENTS(s_apwszAltitudes) && rcNt == STATUS_FLT_INSTANCE_ALTITUDE_COLLISION; i++)
4584 {
4585 s_ObCallbackReg.Altitude.Buffer = (WCHAR *)s_apwszAltitudes[i];
4586 s_ObCallbackReg.Altitude.Length = (uint16_t)RTUtf16Len(s_apwszAltitudes[i]) * sizeof(WCHAR);
4587 s_ObCallbackReg.Altitude.MaximumLength = s_ObCallbackReg.Altitude.Length + sizeof(WCHAR);
4588
4589 rcNt = g_pfnObRegisterCallbacks(&s_ObCallbackReg, &g_pvObCallbacksCookie);
4590 if (NT_SUCCESS(rcNt))
4591 {
4592 /*
4593 * Happy ending.
4594 */
4595 return STATUS_SUCCESS;
4596 }
4597 }
4598 LogRel(("vboxdrv: ObRegisterCallbacks failed with rcNt=%#x\n", rcNt));
4599 g_pvObCallbacksCookie = NULL;
4600 }
4601 else
4602 {
4603 /*
4604 * For the time being, we do not implement extra process
4605 * protection on pre-Vista-SP1 systems as they are lacking
4606 * necessary KPIs. XP is end of life, we do not wish to
4607 * spend more time on it, so we don't put up a fuss there.
4608 * Vista users without SP1 can install SP1 (or later), darn it,
4609 * so refuse to load.
4610 */
4611 /** @todo Hack up an XP solution - will require hooking kernel APIs or doing bad
4612 * stuff to a couple of object types. */
4613# ifndef VBOX_WITH_VISTA_NO_SP
4614 if (g_uNtVerCombined >= SUP_NT_VER_VISTA)
4615# else
4616 if (g_uNtVerCombined >= SUP_MAKE_NT_VER_COMBINED(6, 0, 6001, 0, 0))
4617# endif
4618 {
4619 DbgPrint("vboxdrv: ObRegisterCallbacks was not found. Please make sure you got the latest updates and service packs installed\n");
4620 rcNt = STATUS_SXS_VERSION_CONFLICT;
4621 }
4622 else
4623 {
4624 Log(("vboxdrv: ObRegisterCallbacks was not found; ignored pre-Vista\n"));
4625 return rcNt = STATUS_SUCCESS;
4626 }
4627 g_pvObCallbacksCookie = NULL;
4628 }
4629
4630 /*
4631 * Drop process create/term notifications.
4632 */
4633 if (g_pfnPsSetCreateProcessNotifyRoutineEx)
4634 g_pfnPsSetCreateProcessNotifyRoutineEx(supdrvNtProtectCallback_ProcessCreateNotifyEx, TRUE /*fRemove*/);
4635 else
4636 PsSetCreateProcessNotifyRoutine(supdrvNtProtectCallback_ProcessCreateNotify, TRUE /*fRemove*/);
4637 }
4638 else
4639 LogRel(("vboxdrv: PsSetCreateProcessNotifyRoutine%s failed with rcNt=%#x\n",
4640 g_pfnPsSetCreateProcessNotifyRoutineEx ? "Ex" : "", rcNt));
4641 supHardenedWinTermImageVerifier();
4642 }
4643 else
4644 rcNt = VBoxDrvNtErr2NtStatus(rc);
4645
4646 RTSemMutexDestroy(g_hErrorInfoLock);
4647 g_hErrorInfoLock = NIL_RTSEMMUTEX;
4648 }
4649 else
4650 rcNt = VBoxDrvNtErr2NtStatus(rc);
4651
4652 RTSpinlockDestroy(g_hNtProtectLock);
4653 g_NtProtectTree = NIL_RTSPINLOCK;
4654 return rcNt;
4655}
4656
4657#endif /* VBOX_WITH_HARDENING */
4658
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