Intel 处理器虚拟化技术浅析2 VMCS 初始化
nanshan 2024-10-22 12:54 14 浏览 0 评论
VMCS区域:
- guest 区域,从真实系统进入虚拟系统时,CPU需要加载的数据,包括很多CPU内部的 寄存器,通用寄存器,控制寄存器,调试寄存器之类。
- host区域,就是从虚拟系统返回真实系统时CPU需要加载的寄存器。
- 控制区域进入虚拟系统后,处理器的行为由这个区域的字段控制。
- 虚拟机退出时的控制区域。
- 控制怎样进入虚拟机。
- 控制CPU以什么样的方式退出到真实系统。
- 从虚拟机退出到真实系统时,虚拟系统退出的原因,或者失败时的错误代码。
上一节提过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
某些处理函数没有给出,会在最后的源码中展示。
来自网络的图片:
相关推荐
- 实战派 | Java项目中玩转Redis6.0客户端缓存
-
铺垫首先介绍一下今天要使用到的工具Lettuce,它是一个可伸缩线程安全的redis客户端。多个线程可以共享同一个RedisConnection,利用nio框架Netty来高效地管理多个连接。放眼望向...
- 轻松掌握redis缓存穿透、击穿、雪崩问题解决方案(20230529版)
-
1、缓存穿透所谓缓存穿透就是非法传输了一个在数据库中不存在的条件,导致查询redis和数据库中都没有,并且有大量的请求进来,就会导致对数据库产生压力,解决这一问题的方法如下:1、使用空缓存解决对查询到...
- Redis与本地缓存联手:多级缓存架构的奥秘
-
多级缓存(如Redis+本地缓存)是一种在系统架构中广泛应用的提高系统性能和响应速度的技术手段,它综合利用了不同类型缓存的优势,以下为你详细介绍:基本概念本地缓存:指的是在应用程序所在的服务器内...
- 腾讯云国际站:腾讯云服务器如何配置Redis缓存?
-
本文由【云老大】TG@yunlaoda360撰写一、安装Redis使用包管理器安装(推荐)在CentOS系统中,可以通过yum包管理器安装Redis:sudoyumupdate-...
- Spring Boot3 整合 Redis 实现数据缓存,你做对了吗?
-
你是否在开发互联网大厂后端项目时,遇到过系统响应速度慢的问题?当高并发请求涌入,数据库压力剧增,响应时间拉长,用户体验直线下降。相信不少后端开发同行都被这个问题困扰过。其实,通过在SpringBo...
- 【Redis】Redis应用问题-缓存穿透缓存击穿、缓存雪崩及解决方案
-
在我们使用redis时,也会存在一些问题,导致请求直接打到数据库上,导致数据库挂掉。下面我们来说说这些问题及解决方案。1、缓存穿透1.1场景一个请求进来后,先去redis进行查找,redis存在,则...
- Spring boot 整合Redis缓存你了解多少
-
在前一篇里面讲到了Redis缓存击穿、缓存穿透、缓存雪崩这三者区别,接下来我们讲解Springboot整合Redis中的一些知识点:之前遇到过,有的了四五年,甚至更长时间的后端Java开发,并且...
- 揭秘!Redis 缓存与数据库一致性问题的终极解决方案
-
在现代软件开发中,Redis作为一款高性能的缓存数据库,被广泛应用于提升系统的响应速度和吞吐量。然而,缓存与数据库之间的数据一致性问题,一直是开发者们面临的一大挑战。本文将深入探讨Redis缓存...
- 高并发下Spring Cache缓存穿透?我用Caffeine+Redis破局
-
一、什么是缓存穿透?缓存穿透是指查询一个根本不存在的数据,导致请求直接穿透缓存层到达数据库,可能压垮数据库的现象。在高并发场景下,这尤其危险。典型场景:恶意攻击:故意查询不存在的ID(如负数或超大数值...
- Redis缓存三剑客:穿透、雪崩、击穿—手把手教你解决
-
缓存穿透菜小弟:我先问问什么是缓存穿透?我听说是缓存查不到,直接去查数据库了。表哥:没错。缓存穿透是指查询一个缓存中不存在且数据库中也不存在的数据,导致每次请求都直接访问数据库的行为。这种行为会让缓存...
- Redis中缓存穿透问题与解决方法
-
缓存穿透问题概述在Redis作为缓存使用时,缓存穿透是常见问题。正常查询流程是先从Redis缓存获取数据,若有则直接使用;若没有则去数据库查询,查到后存入缓存。但当请求的数据在缓存和数据库中都...
- Redis客户端缓存的几种实现方式
-
前言:Redis作为当今最流行的内存数据库和缓存系统,被广泛应用于各类应用场景。然而,即使Redis本身性能卓越,在高并发场景下,应用于Redis服务器之间的网络通信仍可能成为性能瓶颈。所以客户端缓存...
- Nginx合集-常用功能指导
-
1)启动、重启以及停止nginx进入sbin目录之后,输入以下命令#启动nginx./nginx#指定配置文件启动nginx./nginx-c/usr/local/nginx/conf/n...
- 腾讯云国际站:腾讯云怎么提升服务器速度?
-
本文由【云老大】TG@yunlaoda360撰写升级服务器规格选择更高性能的CPU、内存和带宽,以提供更好的处理能力和网络性能。优化网络配置调整网络接口卡(NIC)驱动,优化TCP/IP参数...
- 雷霆一击服务器管理员教程
-
本文转载莱卡云游戏服务器雷霆一击管理员教程(搜索莱卡云面版可搜到)首先你需要给服务器设置管理员密码,默认是空的管理员密码在启动页面进行设置设置完成后你需要重启服务器才可生效加入游戏后,点击键盘左上角E...
你 发表评论:
欢迎- 一周热门
-
-
爱折腾的特斯拉车主必看!手把手教你TESLAMATE的备份和恢复
-
如何在安装前及安装后修改黑群晖的Mac地址和Sn系列号
-
[常用工具] OpenCV_contrib库在windows下编译使用指南
-
WindowsServer2022|配置NTP服务器的命令
-
Ubuntu系统Daphne + Nginx + supervisor部署Django项目
-
WIN11 安装配置 linux 子系统 Ubuntu 图形界面 桌面系统
-
解决Linux终端中“-bash: nano: command not found”问题
-
Linux 中的文件描述符是什么?(linux 打开文件表 文件描述符)
-
NBA 2K25虚拟内存不足/爆内存/内存占用100% 一文速解
-
K3s禁用Service Load Balancer,解决获取浏览器IP不正确问题
-
- 最近发表
-
- 实战派 | Java项目中玩转Redis6.0客户端缓存
- 轻松掌握redis缓存穿透、击穿、雪崩问题解决方案(20230529版)
- Redis与本地缓存联手:多级缓存架构的奥秘
- 腾讯云国际站:腾讯云服务器如何配置Redis缓存?
- Spring Boot3 整合 Redis 实现数据缓存,你做对了吗?
- 【Redis】Redis应用问题-缓存穿透缓存击穿、缓存雪崩及解决方案
- Spring boot 整合Redis缓存你了解多少
- 揭秘!Redis 缓存与数据库一致性问题的终极解决方案
- 高并发下Spring Cache缓存穿透?我用Caffeine+Redis破局
- Redis缓存三剑客:穿透、雪崩、击穿—手把手教你解决
- 标签列表
-
- linux 查询端口号 (58)
- docker映射容器目录到宿主机 (66)
- 杀端口 (60)
- yum更换阿里源 (62)
- internet explorer 增强的安全配置已启用 (65)
- linux自动挂载 (56)
- 禁用selinux (55)
- sysv-rc-conf (69)
- ubuntu防火墙状态查看 (64)
- windows server 2022激活密钥 (56)
- 无法与服务器建立安全连接是什么意思 (74)
- 443/80端口被占用怎么解决 (56)
- ping无法访问目标主机怎么解决 (58)
- fdatasync (59)
- 405 not allowed (56)
- 免备案虚拟主机zxhost (55)
- linux根据pid查看进程 (60)
- dhcp工具 (62)
- mysql 1045 (57)
- 宝塔远程工具 (56)
- ssh服务器拒绝了密码 请再试一次 (56)
- ubuntu卸载docker (56)
- linux查看nginx状态 (63)
- tomcat 乱码 (76)
- 2008r2激活序列号 (65)