Hypervisor From Scratch - 第 1 部分:基本概念和配置测试环境
Hypervisor From Scratch – 第 2 部分:进入 VMX 操作
Hypervisor From Scratch – 第 3 部分:设置我们的第一个虚拟机
Hypervisor From Scratch – 第 4 部分:使用扩展页表 (EPT) 进行地址转换
Hypervisor From Scratch – 第 5 部分:设置 VMCS 和运行来宾代码
Hypervisor From Scratch – 第 6 部分:虚拟化已经运行的系统
Hypervisor From Scratch – 第 7 部分:使用 EPT 和页面级监控功能
Hypervisor From Scratch – 第 8 部分:如何使用管理程序施展魔法
目录二、进入VMX操作
2.1.介绍
这是名为“ Hypervisor From Scratch ”的多个系列教程的第二部分。首先,请考虑在阅读本部分之前先阅读第一部分(基本概念和配置测试环境),因为它包含您需要了解的基本知识,以便理解本教程的其余部分。在这一部分中,我们将讨论 WDK 驱动程序并最终开始启用 VT-x。
2.2.概述
在本节中,我们将了解检测处理器对虚拟机管理程序的支持,然后我们简单地配置基本操作以启用 VMX、进入 VMX 操作,然后我们将了解有关Window Driver Kit (WDK) 的更多信息。
2.3.IRP主要功能
除了我们的内核模式驱动程序(“ MyHypervisorDriver ”)之外,我们还将创建一个名为“ MyHypervisorApp ”的用户模式应用程序。首先,我应该鼓励您在用户模式而不是内核模式下编写大部分代码(只要可能),这是因为您可能没有正确处理异常。因此,它会导致 BSOD,或者另一方面,在内核模式下运行较少的代码会减少出现一些令人讨厌的内核模式错误的可能性。
如果您还记得上一部分,我们创建了一个 Windows 驱动程序。现在我们想要扩展我们的项目以支持更多 IRP 主要功能。
IRP 主要功能位于为每个设备创建的传统 Windows 表中。一旦我们在 Windows 中注册了一个设备,我们就必须为这些 IRP 主要函数引入一个处理程序。
这就像每个设备都有一个主要功能表。每当用户模式应用程序调用任何这些函数时,Windows 都会找到相应的函数(如果设备驱动程序支持该 MJ 函数),然后将 IRP 传递给内核驱动程序。
2.3.1.什么是IRP
那么,什么是 IRP ? IRP 是表示 I/O 请求数据包的结构。 该数据包包含有关其调用者、参数、数据包状态等的许多详细信息。我们从 IRP 数据包中提取调用者参数。
现在,我们可以根据 IRP 提供的详细信息在内核中处理用户模式请求。
请记住,当内核驱动程序中的函数接收到 IRP 数据包时,我们的代码有责任调查调用者并检查其权限等。
2.3.2.配置IRP主要功能
注册设备后(前面已经介绍过),我们需要介绍设备的主要功能。
以下代码负责配置不同的 IRP MJ 函数并引入自定义内核模式函数作为 IRP 处理程序。
您可以看到我们对所有功能都使用了“ DrvUnsupported ”。 该函数处理所有 MJ 函数并告诉用户它不受支持。
“ DrvUnsupported ”的主体是这样的:
我们还介绍了我们设备所必需的其他主要功能。 我们将在以后的部分中完成其中一些 MJ 函数的实现。
现在让我们看看 IRP MJ 函数列表和其他类型的 Windows 驱动程序工具包处理程序例程。
2.3.3.IRP主要功能列表
我们可以使用此 IRP 主要函数列表在 WDK 驱动程序中执行不同的操作
每个主要函数只有在我们从用户模式调用其相应函数时才会触发。例如,有一个名为CreateFile的函数(在用户模式下) (及其所有变体,例如用于ASCII和Unicode 的CreateFileA和CreateFileW),因此每次我们调用CreateFile时,都会调用注册为IRP_MJ_CREATE的函数 ,如果我们调用ReadFile然后 IRP_MJ_READ或WriteFile 然后IRP_MJ_WRITE 将被触发。
您可以看到 Windows 将其设备视为文件,并且我们需要从用户模式传递到内核模式的所有内容都可以在带有IRP *类型的参数中使用,并且可以用作内核 IRP MJ 函数处理程序的缓冲区。 Windows 负责将用户模式缓冲区复制到内核模式堆栈。
不用担心;我们在项目的其余部分中经常使用它,但我们仅在这部分中支持IRP_MJ_CREATE,而其他部分则在将来的部分中实现。
还有其他术语称为“IRP 次要函数”。我们保留了这些功能,因为本系列中未使用它们。
2.4.在DbgView中查看调试信息
在“regedit.exe”中,添加一个键:
在其下添加一个名为 IHVDRIVER 且值为 0xFFFF 的 DWORD 值。
2.2.检查虚拟机管理程序支持
在启用VT-x之前,我们首先应该考虑发现对VMX的支持。英特尔软件开发人员手册第 3C卷第 23.6 节**“发现对 VMX 的支持”**对此进行了介绍。
如果CPUID.1:ECX.VMX[bit 5] = 1,则支持 VMX 操作,您可以使用CPUID知道 VMX 的存在。
首先,我们需要知道我们是否在基于英特尔的处理器上运行。我们可以使用CPUID指令并找到供应商字符串“ GenuineIntel ”来理解这一点。
以下函数使用指令返回供应商字符串CPUID。
最后一步是检查VMX是否存在 。 我们可以使用以下代码来检查它:
正如你所看到的,它检查 CPUID.1,如果第bit 5位为1,则支持VMX操作。 我们也可以在内核驱动程序中执行相同的操作。
总而言之,我们检测VMX支持的主要代码应该是这样的:
最终结果:
2.6.启用VMX操作
如果处理器支持 VMX 操作,则需要启用它。 正如我上面告诉您的, IRP_MJ_CREATE 是应该用于启动操作的第一个函数。
在进入VMX操作之前,我们应该通过设置CR4.VMXE[bit 13] = 1来使能VMX。然后通过执行VMXON指令进入VMX操作。 如果在 CR4.VMXE = 0 的情况下执行,VMXON 会导致无效操作码异常 (#UD)。一旦进入 VMX 操作,就无法清除 CR4.VMXE。
之后,我们可以通过执行VMXOFF指令来退出VMX操作,此时CR4.VMXE可以被清零。
VMXON 还由 IA32_FEATURE_CONTROL MSR(MSR 地址 3AH)控制。 当逻辑处理器复位时,该 MSR 被清零。
让我们看看这个 MSR 的第一部分:
位 0 是锁定位。 如果该位清零,VMXON 会导致一般保护 (#GP) 异常。 如果锁定位被设置,则对该 MSR 的 WRMSR 会导致一般保护异常; 在上电复位之前,MSR 无法修改。
这是什么意思? 这意味着我们可以禁用 VMX 功能,而无法再次启用。 只有系统重置后,我们才能启用VMX。
系统 BIOS 可以使用该位为 BIOS 提供设置选项以禁用对 VMX 的支持。 要在平台中启用 VMX 支持,BIOS 必须设置位 1、位 2 或两者,以及锁定位。
2.6.1.设置CR4 VMXE位
现在我们应该创建一些函数来在汇编中执行此操作。
只需在头文件(在我的例子中 Source.h )中声明您的函数:
然后在汇编文件(在我的例子中为“SourceAsm.asm”)中,添加此函数(设置 CR4 的bit 13)。
AsmEnableVmxOperation PROC PUBLIC
PUSH RAX ; Save the state
XOR RAX, RAX ; Clear the RAX
MOV RAX, CR4
OR RAX,02000h ; Set the 14th bit
MOV CR4, RAX
POP RAX ; Restore the state
RET
AsmEnableVmxOperation ENDP
另外,在 SourceAsm.asm 的上面声明您的函数。
PUBLIC AsmEnableVmxOperation
此汇编函数 应在DrvCreate中调用 :
最后,我们应该从用户模式调用以下函数:
如果您看到以下结果,则您已成功完成第二部分。
2.7.结论
在这一部分中,我们了解了创建 Windows Driver Kit 程序所需的基本知识,然后进入虚拟环境为其余部分构建基石。
在第三部分中,我们将深入了解 Intel VT-x,并使我们的驱动程序更加先进。
注意:请记住,虚拟机管理程序会随着时间的推移而发生变化,因为新功能会添加到操作系统中或使用新技术。 例如,Meltdown & Spectre 的更新对虚拟机管理程序进行了大量更改,因此,如果您想在项目、研究或其他任何用途中使用虚拟机管理程序 From Scratch,则必须使用这些教程系列最新部分中的驱动程序由于本教程正在积极更新,并且更改已应用于较新的部分(较早的部分保持不变),因此您可能会在较早的部分中遇到错误和不稳定问题,因此请确保在实际项目中使用最新的部分。
2.8.参考
[1] 英特尔® 64 和 IA-32 架构软件开发人员手册合并第 3 卷 ( https://software.intel.com/en-us/articles/intel-sdm )
[2] IRP_MJ_DEVICE_CONTROL( https://docs.microsoft.com/en-us/windows-hardware/drivers/kernel/irp-mj-device-control )
[3] Windows 驱动程序工具包示例 ( https://github.com/Microsoft/Windows-driver-samples/blob/master/general/ioctl/wdm/sys/sioctl.c )
[4] 手动设置单台计算机的本地内核调试 ( https://docs.microsoft.com/en-us/windows-hardware/drivers/debugger/setting-up-local-kernel-debugging-of-a-单机手动 )
[5]使用CPUID获取处理器制造商( https://www.daniweb.com/programming/software-development/threads/112968/obtain-processor-manufacturer-using-cpuid )
[6] 即插即用次要 IRP ( https://docs.microsoft.com/en-us/windows-hardware/drivers/kernel/plug-and-play-minor-irps )
[7] _FAST_IO_DISPATCH 结构( https://docs.microsoft.com/en-us/windows-hardware/drivers/ddi/content/wdm/ns-wdm-_fast_io_dispatch )
[8] 过滤 IRP 和快速 I/O ( https://docs.microsoft.com/en-us/windows-hardware/drivers/ifs/filtering-irps-and-fast-io )
[9] Windows 文件系统过滤器驱动程序开发 ( https://www.apriorit.com/dev-blog/167-file-system-filter-driver )
[竞赛]2024 KCTF 大赛征题截止日期08月10日!