通过 SYSENTER/SYSEXIT指令来学习系统调用

SYSENTER指令---快速系统调用

指令格式没有什么重要的内容,只有opcode ,没有后面的其他字段

指令的作用:

执行快速调用到特权级别0的系统过程或例程。SYSENTER是SYSEXIT的配套指令。该指令经过优化,能够为从运行在特权级别3的用户代码到特权级别0的操作系统或执行过程的系统调用提供最大性能。

IA-32e模式下的 SYSENTER

在IA-32e模式下执行时,SYSENTER指令将逻辑处理器切换到64位模式;否则,逻辑处理器保持在保护模式.

切换的过程

在执行SYSENTER指令之前,先确定

1.特权级别0环的 代码段(CS)和代码入口点(EIP)

2.特权级别0环的 堆栈段(SS)和堆栈指针(ESP)

WRMSR指令 将以上的信息写入以下模型特定寄存器(MSR)中

  • IA32_SYSENTER_CS (MSR地址174H) --- 该MSR的低16位是特权级别0代码段的段选择子(Segment Selector)。该值还用于确定特权级别0堆栈段的段选择符(参见操作部分)。此值不能表示空选择符(Null Selector)。

  • IA32_SYSENTER_EIP (MSR地址176H) --- 该MSR寄存器保存的值被加载到RIP中(因此,该值引用了所选函数的第一条指令)。在保护模式下,仅加载位31:0。

  • IA32_SYSENTER_ESP (MSR地址175H) --- 该MSR的值被加载到RSP中(因此,该值包含特权级别0环的堆栈的ESP)。此值不能表示非规范地址(Non-Canonical Address)。在保护模式下,仅加载位31:0。

虽然 SYSENTER 使用 IA32_SYSENTER_CS MSR 中的值来设置 CSSS 段选择符,但 CSSS 的描述符缓存不会从这些选择符所对应的描述符(位于 GDT 或 LDT)中加载。相反,描述符缓存会加载一组固定值。详细信息请参见操作部分。操作系统有责任确保选择符所对应的描述符(位于 GDT 或 LDT)与加载到描述符缓存中的固定值一致;SYSENTER 指令不保证这种对应性。

可以通过RDMSR指令 读取MSR寄存器的内容 WRMSR/RDMSR指令=都属于特权级别(0环)指令

需要遵循的约定:

  • 特权级别0代码段和堆栈段以及特权级别3代码段和堆栈段的段描述符必须在描述符表中连续。这种约定允许处理器根据SYSENTER_CS_MSR MSR中输入的值计算段选择符。

  • 用户代码执行的快速系统调用"存根"例程(通常位于共享库或DLL中)必须保存所需的返回IP和处理器状态信息,以便在需要返回到调用过程时使用 。同样,通过SYSENTER指令调用的操作系统或执行过程在返回到用户代码时必须能够访问并使用这些保存的返回和状态信息。

Operation(伪代码)

IF CR0.PE = 0 OR IA32_SYSENTER_CS[15:2] = 0 THEN #GP(0); FI;
如果 CR0 寄存器的 PE(保护使能,Protection Enable)位为 0(表示处理器未处于保护模式)或 IA32_SYSENTER_CS MSR(模型特定寄存器)寄存器的位 15 到 2 为 0,处理器将触发一般保护异常 #GP,且错误码为 0

RFLAGS.VM := 0;    (* 确保保护模式执行 *)
RFLAGS.IF := 0;    (* 屏蔽中断 *)
IF in IA-32e mode  
    THEN            (* 如果在IA-32e处理器模式 *)
        RSP := IA32_SYSENTER_ESP;
        RIP := IA32_SYSENTER_EIP;
    ELSE            (*不在IA-32e模式下*)
    ESP := IA32_SYSENTER_ESP[31:0];
    EIP := IA32_SYSENTER_EIP[31:0];
FI;(*if语句结束*)

CS.Selector := IA32_SYSENTER_CS[15:0] AND FFFCH; (* 操作系统提供 CS;强制 RPL 为 0 *)

(* 将 CS 的其余部分设置为固定值 *)

CS.Base := 0; (* 平坦段 *)
CS.Limit := FFFFFH; (* 具有4KB粒度,表示4GB限制 *)
CS.S := 1;     (*数据或代码段*)
CS.Type := 11; (* 执行/读取已访问 代码段*)
CS.DPL := 0;
CS.P := 1;


IF in IA-32e mode
    THEN        (*如果在IA32-e架构下)
        CS.L := 1; (* 进入64位模式 *)
        CS.D := 0; (* 若 CS.L = 1 时必需设置 *)
    ELSE        (X86架构下)
        CS.L := 0;
        CS.D := 1; (* 32位代码段 *)
FI; (*if语句结束*)

CS.G := 1;(* 4KB粒度 *)


CPL := 0;
SS.Selector := CS.Selector + 8;(* SS 紧接在 CS 之上 *)

(*将SS的剩余部分设置为固定值*)

SS.Base := 0; (* 平坦段 *)
SS.Limit := FFFFFH; (* 具有4KB粒度,表示4GB限制 *)
SS.Type := 3;(* 读/写数据,已访问 *)
SS.S := 1;
SS.DPL := 0;
SS.P := 1;
SS.B := 1; (* 32位堆栈段 *)
SS.G := 1; (* 4KB粒度 *)

SYSEXIT指令--快速系统调用的快速返回

1.快速返回到特权级别3的用户代码。

2.快速返回到64位模式特权级别3的用户代码。

指令的作用:

执行快速返回到特权级别3的用户代码。SYSEXIT 是 SYSENTER 指令的配套指令。此指令经过优化,旨在为从特权级别0的系统过程返回到特权级别3的用户过程提供最高性能。它必须从特权级别0的代码执行。

不同模式下指令的行为:

1.如果 SYSEXIT 指令在64位模式下并使用64位操作数(REX.W前缀),它将继续保持在64位模式下运行。这意味着当 SYSEXIT 执行返回时,处理器将保持在64位模式下。

2.如果没有使用64位操作数大小,那么 SYSEXIT 的行为会根据逻辑处理器的模式而不同:

  • 如果逻辑处理器处于 IA-32e 模式 :即如果处理器当前在 IA-32e 模式下(即启用了64位模式),但 SYSEXIT 指令没有使用64位操作数,它将返回到兼容模式(Compatibility Mode)。兼容模式允许32位应用在64位操作系统上运行,这样操作系统可以在保持64位模式的情况下,执行32位代码。

  • 如果处理器不在 IA-32e 模式下 :即处理器在普通的保护模式 (Protected Mode) 下,则 SYSEXIT 返回时保持在保护模式。

指令的执行流程:

在执行 SYSEXIT 之前,软件必须通过写入以下 MSR 和通用寄存器来指定特权级别3代码段和代码入口点,以及特权级别3的堆栈段和堆栈指针:

  • IA32_SYSENTER_CS (MSR 地址174H) ------ 包含一个32位值,用于确定特权级别3代码段和堆栈段的段选择符(参见操作部分)。
  • RDX ------ 此寄存器中的规范地址加载到 RIP 中(因此该值指向将在用户代码中执行的第一条指令)。如果返回时不处于64位模式,仅加载位31:0。
  • ECX ------ 此寄存器中的规范地址加载到 RSP 中(因此该值包含特权级别3堆栈的堆栈指针)。如果返回时不处于64位模式,仅加载位31:0。

IA32_SYSENTER_CS MSR 可以使用 RDMSR 和 WRMSR 进行读写。

伪代码

IF IA32_SYSENTER_CS[15:2] = 0 OR CR0.PE = 0 OR CPL ≠ 0 THEN #GP(0); FI;

IF operand size is 64-bit
    THEN (* Return to 64-bit mode *)
        RSP := RCX;
        RIP := RDX;
    ELSE (* Return to protected mode or compatibility mode *)
        RSP := ECX;
        RIP := EDX;
FI;

IF operand size is 64-bit (* Operating system provides CS; RPL forced to 3 *)
    THEN CS.Selector := IA32_SYSENTER_CS[15:0] + 32;
    ELSE CS.Selector := IA32_SYSENTER_CS[15:0] + 16;
FI;

CS.Selector := CS.Selector OR 3; (* RPL forced to 3 *)
(* Set rest of CS to a fixed value *)
CS.Base := 0; (* Flat segment *)
CS.Limit := FFFFFH; (* With 4-KByte granularity, implies a 4-GByte limit *)
CS.Type := 11; (* Execute/read code, accessed *)
CS.S := 1;
CS.DPL := 3;
CS.P := 1;

IF operand size is 64-bit
    THEN (* return to 64-bit mode *)
        CS.L := 1; (* 64-bit code segment *)
        CS.D := 0; (* Required if CS.L = 1 *)
    ELSE (* return to protected mode or compatibility mode *)
        CS.L := 0;
        CS.D := 1; (* 32-bit code segment*)
FI;

CS.G := 1; (* 4-KByte granularity *)
CPL := 3;
IF ShadowStackEnabled(CPL)
    THEN SSP := IA32_PL3_SSP;
FI;
SS.Selector := CS.Selector + 8; (* SS just above CS *)
(* Set rest of SS to a fixed value *)
SS.Base := 0; (* Flat segment *)
SS.Limit := FFFFFH; (* With 4-KByte granularity, implies a 4-GByte limit *)
SS.Type := 3; (* Read/write data, accessed *)
SS.S := 1;
SS.DPL := 3;
SS.P := 1;
SS.B := 1; (* 32-bit stack segment*)
SS.G := 1; (* 4-KByte granularity *)

解释:

特权级别0代码段和堆栈段以及特权级别3代码段和堆栈段的段描述符必须在描述符表中连续

总结:

通过sysenter指令,切换特权级别时,CPU不会保存任何寄存器的值,并且会修改elfags寄存器的IF位.

通过sysexit指令返回时,将读取msr174的 3环选择子,设置cs段寄存器,通过cs段选择子+8的位置就是3环的ss段选择子.RDX 的值加载到EIP 中**,RCX** 的值加载到ESP 中.

所有其他的寄存器 需要操作系统软件保存,并且在特权级别0 能够访问到这块内存

相关推荐
会写代码的孙悟空5 小时前
windows下解决端口被占用,但是找不到占用端口的应用程序;以一种访问权限不允许的方式做了一个访问套接字的尝试;搜索可用端口
运维·网络·windows
sukalot6 小时前
windows 驱动实例分析系列: NDIS 6.0的Filter 驱动改造(二)
windows
喜欢打篮球的普通人7 小时前
2024 Rust现代实用教程:1.3获取rust的库国内源以及windows下的操作
开发语言·windows·rust
sukalot8 小时前
windows 驱动实例分析系列: NDIS 6.0的Filter 驱动改造(四)
windows·单片机·嵌入式硬件
微雨盈萍cbb10 小时前
Windows 上更新OpenSSL 到 1.1.1
windows
askah664410 小时前
无法启动此程序win10玩游戏找不到d3dx9_43.dll缺失的五种常用有效解决方法
windows·游戏·电脑·dll丢失·1024程序员节
charlie11451419113 小时前
STM32 从0开始系统学习2
stm32·嵌入式硬件·c·1024程序员节·arm架构·寄存器映射
admin_23320 小时前
rsync部署 附报错解决、配置详解及Windows脚本
windows
shylyly_21 小时前
string类的模拟实现
开发语言·数据结构·c++·c·string类的模拟实现·手搓