百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术文章 > 正文

Intel 处理器虚拟化技术浅析2 VMCS 初始化

nanshan 2024-10-22 12:54 28 浏览 0 评论

VMCS区域:

  1. guest 区域,从真实系统进入虚拟系统时,CPU需要加载的数据,包括很多CPU内部的 寄存器,通用寄存器,控制寄存器,调试寄存器之类。
  2. host区域,就是从虚拟系统返回真实系统时CPU需要加载的寄存器。
  3. 控制区域进入虚拟系统后,处理器的行为由这个区域的字段控制。
  4. 虚拟机退出时的控制区域。
  5. 控制怎样进入虚拟机。
  6. 控制CPU以什么样的方式退出到真实系统。
  7. 从虚拟机退出到真实系统时,虚拟系统退出的原因,或者失败时的错误代码。

上一节提过VMCS区域需要专门的汇编指令才能读取,下面我们填充VMCS区域。

我们看下64位系统段选择子的描述:

_SEGMENT_SELECTOR record Limit:32, \
SEGMENT_GPA:4,SEGMENT_attr_G:1,SEGMENT_attr_D_B:1,SEGMENT_attr_L:1, \
SEGMENT_attr_AVL:1,SEGMENT_attr_P:1,\
SEGMENT_attr_DPL:2,SEGMENT_attr_S:1,SEGMENT_attr_Type_code_or_data:1,SEGMENT_attr_Type_CRA:3,selector_index:13,selector_TI:1,selector_RPL:2

;selector_TI (Table Indicator)描述符表索引位,当TI=0时从GDT查找 =1时从LDT查找。RPL: 权限级别0-3

vmx_vmwrite_seg proc _value_HOST_XX_SELECTOR:qword,_SELECTOR_name:qword
mov rax,_SELECTOR_name
movzx eax,ax
and eax,0f8h ;清低3位,RPL= 0 从GDT(全局描述符表)加载段描述符
cdqe
mov rcx,_value_HOST_XX_SELECTOR
vmwrite rcx, rax
ret
vmx_vmwrite_seg endp
invoke vmx_vmwrite_seg,HOST_ES_SELECTOR,ES
invoke vmx_vmwrite_seg,HOST_CS_SELECTOR,CS
invoke vmx_vmwrite_seg,HOST_SS_SELECTOR,SS
invoke vmx_vmwrite_seg,HOST_DS_SELECTOR,DS
invoke vmx_vmwrite_seg,HOST_FS_SELECTOR,FS
invoke vmx_vmwrite_seg,HOST_GS_SELECTOR,GS

任务寄存器需要单独操作:

str rax
invoke vmx_vmwrite_seg,HOST_TR_SELECTOR,RAX
mov rcx,MSR_IA32_EFER

使能Extended Feature Enable Register;

rdmsr
shl rdx, 20h
or rax, rdx
mov rbx,rax
invoke vmx_vmwrite,HOST_EFER,rbx

cr0 与cr4的固定位:

MSR_IA32_VMX_CR0_FIXED0、MSR_IA32_VMX_CR0_FIXED1 字段里规定了guest里cr0控制寄存器的数值要求。

mov rcx,MSR_IA32_VMX_CR0_FIXED1
rdmsr
shl rdx, 20h
or rax, rdx
mov rbx,cr0
and rbx,rax
mov rcx,MSR_IA32_VMX_CR0_FIXED0
rdmsr
shl rdx, 20h
or rax, rdx
or rbx,rax
vmwrite,HOST_CR0, rbx
vmwrite,HOST_CR3, cr3
mov rcx,MSR_IA32_VMX_CR4_FIXED1
rdmsr
shl rdx, 20h
or rax, rdx
mov rbx,cr4
and rbx,rax
mov rcx,MSR_IA32_VMX_CR4_FIXED0
rdmsr
shl rdx, 20h
or rax, rdx
or rbx,rax
vmwrite,HOST_CR3, rbx

从全局描述符表查找任务寄存器的基址:

sgdt @gdtbase
lea rdi,@gdtbase
mov rbx,[rdi+2];路过Limit
str rdx;tss
invoke GetSegmentDescriptor,addr @_SEGMENT_SELECTOR,rdx,rbx
vmwrite,HOST_TR_BASE, @_SEGMENT_SELECTOR.BASE

填充fs寄存器字段:(64位系统下fs基本是兼容32个应用程序而保留)

mov ecx, MSR_FS_BASE
rdmsr
shl rdx, 20h
or rax, rdx
vmwrite,HOST_FS_BASE,rax

填充gs寄存器字段:(64位系统非常重要的寄存器,相当于32个系统里的fs寄存器)

mov ecx, MSR_GS_BASE
rdmsr
shl rdx, 20h
or rax, rdx
vmwrite,HOST_GS_BASE, rax

填充全局描述符表字段:

sgdt @gdtbase
lea rdi,@gdtbase
mov rbx,[rdi+2]
vmwrite,HOST_GDTR_BASE,rbx

填充中断描述符表字段:

sidt @idtbase
lea rdi,@idtbase
mov rbx,[rdi+2]
vmwrite,HOST_IDTR_BASE,rbx
vmwrite,VMCS_LINK_POINTER,-1;不启用SMM双重监控处理机制
vmwrite,EXCEPTION_BITMAP,0

填充guest 段选择子、段限、访问权限、段基址。

sgdt tbyte ptr @gdtbase
lea rdi,@gdtbase
mov rbx,[rdi+2]
invoke FillGuestSelectorData,rbx,0,es
invoke FillGuestSelectorData,rbx,1,cs
invoke FillGuestSelectorData,rbx,2,ss
invoke FillGuestSelectorData,rbx,3,ds
invoke FillGuestSelectorData,rbx,4,fs
invoke FillGuestSelectorData,rbx,5,gs
sldt rax
invoke FillGuestSelectorData,rbx,6,rax
str rax
invoke FillGuestSelectorData,rbx,7,rax

填充fs寄存器字段:

mov ecx, MSR_FS_BASE
rdmsr
shl rdx, 20h
or rax, rdx
vmwrite,GUEST_FS_BASE,rax

填充gs寄存器字段:

mov ecx, MSR_GS_BASE
rdmsr
shl rdx, 20h
or rax, rdx
vmwrite,GUEST_GS_BASE,rax

填充控制字段:

mov ecx, MSR_IA32_VMX_TRUE_PROCBASED_CTLS
Rdmsr
;eax存入的是允许为零的位 "允许为零的位"已经被标0(其余的位必须为1)edx存入的是允 许为1的位“允许为1的位”已经被标1(其余的位必须为0)
Mov ecx,CPU_BASED_ACTIVATE_MSR_BITMAP+\
CPU_BASED_ACTIVATE_SECONDARY_CONTROLS
mov rbx,rcx
or ecx,eax
and ecx,edx
vmwrite,CPU_BASED_VM_EXEC_CONTROL,rcx
mov ecx, MSR_IA32_VMX_PROCBASED_CTLS2
rdmsr
mov rcx,CPU_BASED_CTL2_RDTSCP
; RDTSCP这个指令的用途,它是RDTSC的升级版,在一些比较新的处理器中用于获 ;得CPU时间计数器。
;SECONDARY_EXEC_ENABLE_RDTSCP | SECONDARY_EXEC_ENABLE_INVPCID | ;SECONDARY_EXEC_XSAVES;
;如果不设置 rdtsc exiting 和 use tsc offsetting 的话,那么 rdtscp 会正常执行,因此这 ;里其实可以只设置 SECONDARY_EXEC_ENABLE_RDTSCP ,其他不设置。
;这样的话 rdtscp 指令的执行并不会导致vm-exit事件的发生。这里我尝试了只将 ;SECONDARY_EXEC_ENABLE_RDTSCP 控制位置位,其他位不动,发现完全不会进入到 ;vm-exit事件中。
;因此如果图省事的话其实可以只将 SECONDARY_EXEC_ENABLE_RDTSCP 置位,其他位不 ;进行操作即可。
or rcx,CPU_BASED_CTL2_ENABLE_INVPCID
;如果 invlpg exiting 没有被设置,那么也不会导致vm-exit事件的发生。因此这里其实 ;也可以只设置 SECONDARY_EXEC_ENABLE_INVPCID ,不设置 invlpg exiting 。这样可以 ;让其正常执行,不用在vm-exit处理函数中对其进行复杂的处理。

or rcx,CPU_BASED_CTL2_ENABLE_XSAVE_XRSTORS

or rcx,CPU_BASED_CTL2_ENABLE_EPT

or rcx,CPU_BASED_CTL2_ENABLE_VPID

mov rbx,rcx
or ecx,eax
and ecx,edx
and ebx,edx
or ebx,eax

vmwrite,SECONDARY_VM_EXEC_CONTROL,rcx
vmwrite,PIN_BASED_VM_EXEC_CONTROL,0
mov ecx, MSR_IA32_VMX_TRUE_EXIT_CTLS
Rdmsr
mov rcx,VM_EXIT_IA32E_MODE + VM_EXIT_ACK_INTR_ON_EXIT
;On processors that support Intel 64 architecture, this control determines whether a logical
;processor is in 64-bit mode after the next VM exit. Its value is loaded into CS.L,
and ecx,edx
or ecx,eax
vmwrite,VM_EXIT_CONTROLS,rcx
mov ecx, MSR_IA32_VMX_TRUE_EXIT_CTLS
rdmsr
mov rcx,VM_EXIT_IA32E_MODE + VM_EXIT_ACK_INTR_ON_EXIT ;On processors that ;support Intel 64 architecture, this control determines whether a logical
;processor is in 64-bit mode after the next VM exit. Its value is loaded into CS.L,
and ecx,edx
or ecx,eax
vmwrite,VM_EXIT_CONTROLS,rcx

;. 配置vm-entry控制域

mov ecx, MSR_IA32_VMX_TRUE_ENTRY_CTLS
rdmsr
mov rcx,VM_ENTRY_IA32E_MODE ;64系统必须填, 参考【处理器虚拟化技术】(第212页)
and ecx,edx
or ecx,eax
vmwrite,VM_ENTRY_CONTROLS,rcx ;

以下为填充guest区域和填充host区域基本相同

mov rcx,MSR_IA32_EFER
rdmsr
shl rdx, 20h
or rax, rdx
mov rbx,rax
vmwrite,GUEST_EFER,rbx

mov rcx,MSR_IA32_VMX_CR0_FIXED1
rdmsr
shl rdx, 20h
or rax, rdx
mov rbx,cr0
and rbx,rax

mov rcx,MSR_IA32_VMX_CR0_FIXED0
rdmsr
shl rdx, 20h
or rax, rdx
or rbx,rax
vmwrite,GUEST_CR0,rbx

mov rcx,MSR_IA32_VMX_CR4_FIXED1
rdmsr
shl rdx, 20h
or rax, rdx
mov rbx,cr4
and rbx,rax

mov rcx,MSR_IA32_VMX_CR4_FIXED0
rdmsr
shl rdx, 20h
or rax, rdx

or rbx,rax

vmwrite,GUEST_CR4,rbx


vmwrite,GUEST_CR3,cr3
invoke vmx_vmwrite,GUEST_DR7, 0400h

sgdt @gdtbase
lea rdi,@gdtbase
mov rbx,[rdi+2]
vmwrite,GUEST_GDTR_BASE,rbx;Get_GDT_Base());
movzx eax, word ptr [rdi]
vmwrite,GUEST_GDTR_LIMIT, rax;Get_GDT_Limit());
sidt @idtbase
lea rdi,@idtbase
mov rbx,[rdi+2]
vmwrite,GUEST_IDTR_BASE,rbx;Get_IDT_Base());
movzx eax,word ptr [rdi]

vmwrite,GUEST_IDTR_LIMIT,rax; Get_IDT_Limit());
pushfq
pop rax
vmwrite,GUEST_RFLAGS, rax;

mov rsi,_current_vmState
mov rdx,[rsi+VirtualMachineState.MSRBitMapPhysical]

vmwrite,MSR_BITMAP, rdx;vmState->MSRBitMapPhysical);MSR Bitmap的某位为0时访问 ;该位所对应的MSR不会产生VM-exit
vmwrite,GUEST_RSP, _GuestStack;(ULONG64)GuestStack); //setup guest sp
mov rax,[rsi+VirtualMachineState.VmxGuestRip]
vmwrite,GUEST_RIP, rax;addr VMXRestoreState; //setup guest ip

mov rsi,_current_vmState
mov rdx,[rsi+VirtualMachineState.VMM_Stack]

mov rax,VMM_STACK_SIZE;
add rdx,4000h;;堆栈向下增长,所以要增加,空间是0-VMM_STACK_SIZE 所以 要减1
vmwrite,HOST_RSP, rdx;((ULONG64)vmState->VMM_Stack + VMM_STACK_SIZE - 1)); ;host的rsp必须使用自己申请的一块内存。如果还是使用guest退出时的rsp,一定会 ;导致guest中堆栈被破坏从而导致不可预知的结果
lea rax,VMExitHandler
vmwrite,HOST_RIP, rax;addr VMExitHandler

某些处理函数没有给出,会在最后的源码中展示。

来自网络的图片:

相关推荐

如何为MySQL服务器和客户机启用SSL?

用户想要与MySQL服务器建立一条安全连接时,常常依赖VPN隧道或SSH隧道。不过,获得MySQL连接的另一个办法是,启用MySQL服务器上的SSL封装器(SSLwrapper)。这每一种方法各有其...

Mysql5.7 出现大量 unauthenticated user

线上环境mysql5.7突然出现大量unauthenticateduser,进mysql,showprocesslist;解决办法有:在/etc/hosts中添加客户端ip,如192.16...

MySQL 在 Windows 系统下的安装(mysql安装教程windows)

更多技术文章MySQL在Windows系统下的安装1.下载mysql和Framework链接链接:百度网盘请输入提取码提取码:6w3p双击mysql-installer-communit...

MySql5.7.21.zip绿色版安装(mysql数据库绿色版安装)

1、去网上下载满足系统要求的版本(mysql-5.7.21-winx64.zip)2、直接解压3、mysql的初始化(1)以管理员身份运行cmd,在mysql中的bin目录下shift+右键-在...

MySQL(8.0)中文全文检索 (亲测有效)

在一堆文字中找到含有关键字的应用。当然也可以用以下语句实现:SELECT*FROM<表名>WHERE<字段名>like‘%ABC%’但是它的效率太低,是全盘扫描。...

新手教程,Linux系统下MySQL的安装

看了两三个教程。终于在哔哩哔哩找到一个简单高效的教程,成功安装,up主名叫bili逍遥bili,感兴趣可以去看看。下面这个是我总结的安装方法环境:CentOS764位1.下载安装包,个人觉得在...

麒麟服务器操作系统安装 MySQL 8 实战指南

原文连接:「链接」Hello,大家好啊,今天给大家带来一篇麒麟服务器操作系统上安装MySQL8的文章,欢迎大家分享点赞,点个在看和关注吧!MySQL作为主流开源数据库之一,被广泛应用于各种业务...

用Python玩转MySQL的全攻略,从环境搭建到项目实战全解析

这是一篇关于“MySQL数据库入门实战-Python版”的教程,结合了案例实战分析,帮助初学者快速掌握如何使用Python操作MySQL数据库。一、环境准备1.安装Python访问Pytho...

安装MySQL(中标麒麟 安装mysql)

安装MySQL注意:一定要用root用户操作如下步骤;先卸载MySQL再安装1.安装包准备(1)查看MySQL是否安装rpm-qa|grepmysql(2)如果安装了MySQL,就先卸载rpm-...

Mysql最全笔记,快速入门,干货满满,爆肝

目录一、MySQL的重要性二、MySQL介绍三、软件的服务架构四、MySQL的安装五、SQL语句六、数据库相关(DDL)七、表相关八、DML相关(表中数据)九、DQL(重点)十、数据完...

MAC电脑安装MySQL操作步骤(mac安装mysqldb)

1、在官网下载MySQL:https://dev.mysql.com/downloads/mysql/根据自己的macOS版本,选择适配的MySQL版本根据自己需求选择相应的安装包,我这里选择macO...

mysql主从(mysql主从切换)

1、本章面试题什么是mysql主从,主从有什么好处什么是读写分离,有什么好处,使用mycat如何实现2、知识点2.1、课程回顾dubboORM->MVC->RPC->SOApro...

【linux学习】以MySQL为例,带你了解数据库

做运维的小伙伴在日常工作中难免需要接触到数据库,不管是MySQL,mariadb,达梦还是瀚高等其实命令都差不多,下面我就以MySQL为例带大家一起来了解下数据库。有兴趣的小伙伴不妨评论区一起交流下...

玩玩WordPress - 环境简介(0)(玩玩网络科技有限公司)

简介提到开源博客系统,一般都会直接想到WordPress!WordPress是使用PHP开发的,数据库使用的是MySQL,一般会在Linux上运行,Nginx作为前端。这时候就需要有一套LNMP(Li...

服务器常用端口都有哪些?(服务器端使用的端口号范围)

下面为大家介绍一下,服务器常用的一些默认端口,以及他们的作用:  21:FTP服务所开放的端口,用于上传、下载文件。  22:SSH端口,用于通过命令行模式远程连接Linux服务器或vps。  23:...

取消回复欢迎 发表评论: