32位和64位系统内核函数调用从ZwProtectVirtualMemory到NtProtectVirtualMemory
0x01 前言
我们知道R3层中,Zw系列函数和Nt系列函数函数是一样的,但是在内核Zw系列函数调用了Nt系列函数,但是为什么要在内核设置一个Zw系列函数而不是直接调用Nt函数呢?Zw系列函数又是怎么调用Nt系列函数的呢?我们利用IDA分析NtosKrnl.exe文件。
0x02 ZwProtectVirtualMemory
我们先看看ZwProtectVirtualMemory的实现
.text: ; NTSTATUS __stdcall ZwProtectVirtualMemory(HANDLE ProcessHandle, PVOID *BaseAddress, PULONG ProtectSize, ULONG NewProtect, PULONG OldProtect)
.text: _ZwProtectVirtualMemory@20 proc near ; CODE XREF: RtlpCreateStack(x,x,x,x,x)+FAp
.text:
.text: ProcessHandle = dword ptr
.text: BaseAddress = dword ptr
.text: ProtectSize = dword ptr 0Ch
.text: NewProtect = dword ptr 10h
.text: OldProtect = dword ptr 14h
.text:
.text: mov eax, 89h ;Nt函数的系统调用号
.text: lea edx, [esp+ProcessHandle] ;使用EDX指向堆栈上的参数块
.text: pushf ;EFLAGS
.text:0040617A push ;CS KGDT_R0_CODE
.text:0040617C call _KiSystemService
.text: retn 14h ;5个参数,20字节
.text: _ZwProtectVirtualMemory@20 endp
这里89h为NtProtectVirtualMemory函数在SSDT函数中的调用号,CS寄存器,最后位为0表示当前处于内核态,然后调用KiSystemService函数
0x03 KiSystemService
我们接着看KiSystemService的函数实现
.text: _KiSystemService proc near ; CODE XREF: ZwAcceptConnectPort(x,x,x,x,x,x)+Cp
.text: ; ZwAccessCheck(x,x,x,x,x,x,x,x)+Cp ...
.text:
.text: arg_0 = dword ptr
.text:
.text: push
.text: push ebp
.text: push ebx
.text: push esi
.text: push edi
.text: push fs ; 保存用户空间的fs
.text: mov ebx, 30h ; KGDT_R0_PCR
.text:0040763E mov fs, ebx ; 使FS段的起点与KPCR数据结构对齐
.text: push dword ptr ds:0FFDFF000h
.text: mov dword ptr ds:0FFDFF000h, 0FFFFFFFFh
.text: mov esi, ds:0FFDFF124h ; #define KPCR_CURRENT_THREAD 0x124
.text: ; 指向当前cpu正在运行的线程
.text: ; FS:0x124
.text: ; PCR的大小只有0x54,这里偏移到了KPRCB中的CurrentThread
.text: push dword ptr [esi+140h]
.text:0040765C sub esp, 48h
.text:0040765F mov ebx, [esp+68h+arg_0] ; 系统调用前夕的CS映像
.text: and ebx, ; 0环的最低位为0,3环的最低位为1
.text: mov [esi+140h], bl ; 新的"先前模式" [esi+KTHREAD_PREVIOUS_MODE]
.text:0040766C mov ebp, esp
.text:0040766E mov ebx, [esi+134h] ; KTHREAD结构中的指针TrapFrame [esi+KTHREAD_TRAP_FRAME]
.text: mov [ebp+3Ch], ebx ; 暂时保存在这里 [ebp+KTRAP_FRAME_EDX]
.text: mov [esi+134h], ebp ; 新的TrapFrame,指向堆栈上的框架 [esi+KTHREAD_TRAP_FRAME]
...
.text:0040769E sti
.text:0040769F jmp loc_407781
.text:0040769F _KiSystemService endp
这里首先要在系统态堆栈上构建一个系统调用"框架Frame",或称为"自陷框架",其作用主要是用来保存发生自陷时CPU中各寄存器的"现场",或者说"上下文",以备返回用户空间时予以恢复。
Windows内核有个特殊的基本要求,只要CPU在内核运行,FS寄存器就指向一个KPCR的数据结构,FS的值为0x30,其0-1位为0,表示0环,第2位为0,表示GDT表,为1则表示LDT表,3-15位为6,表示在GDT的下标为6的表项中的地址即为KPCR的地址。KPCR是处理器控制块,在单处理器中只有一个KPCR,在多CPU的系统中,每个CPU都有自己的KPCR结构。
CPU从用户空间进入系统空间时会将当时寄存器CS的内容压入系统态堆栈,CS的最低位就可以说明当时运行于何种模式的标志位。这里取出CS最低位保存在ETHREAD的PREVIOUS_MODE上。
更新ETHREAD中的TrapFrame框架,保存旧的框架。
0x04 KiFastCallEntry
KiSystemService中的jmp loc_407781跳转到KiFastCallEntry函数中,代码如下:
.text:004076F0 _KiFastCallEntry proc near ; DATA XREF: _KiTrap01+6Fo
.text:004076F0 ; KiLoadFastSyscallMachineSpecificRegisters(x)+24o
.text:004076F0
.text:004076F0 var_B = byte ptr -0Bh
.text:004076F0
.text:004076F0 ; FUNCTION CHUNK AT .text:004076C8 SIZE 00000023 BYTES
.text:004076F0 ; FUNCTION CHUNK AT .text:00407990 SIZE 00000014 BYTES
.text:004076F0
.text:004076F0 mov ecx, 23h ; KGDT_R3_DATA OR RPL_MASK
.text:004076F5 push 30h
.text:004076F7 pop fs ; 只要进入内核..fs->KPCR(Kernel's Processor Control Region,内核进程控制区域)
.text:004076F9 mov ds, ecx ; 使用23h选择子
.text:004076FB mov es, ecx
.text:004076FD mov ecx, ds:0FFDFF040h ; _KPCR->_KTSS
.text: mov esp, [ecx+] ; 取出_KTSS->esp0
.text: ; 这里是模拟自陷框架,以形成和中断, 异常统一的框架 _KTRAP_FRAME
.text: push 23h ; KGDT_R3_DATA OR RPL_MASK
.text: push edx ; R3 ss:esp
.text: pushf ; R3 Eflags
.text:0040770A
.text:0040770A loc_40770A: ; CODE XREF: _KiFastCallEntry2+22j
.text:0040770A push
.text:0040770C add edx, ; edx -> args 用户态参数
.text:0040770F popf ; Eflags = 2 中断已关闭
.text: or [esp+0Ch+var_B], ; 开启R3 Eflags的中断标记
.text: push 1Bh ; R3 cs:eip
.text: push dword ptr ds:0FFDF0304h ; ntdll!KiFastSystemCallRet
.text:0040771D push ; 为了和中断保持一致, 中断会有错误码, 同时用于返回值
.text:0040771F push ebp
.text: push ebx
.text: push esi
.text: push edi
.text: mov ebx, ds:0FFDFF01Ch ; ebx<-_KPCR.SelfPcr 这是pcr的指针
.text: push 3Bh
.text:0040772B mov esi, [ebx+124h] ; esi=_KPCR.PrcbData.CurrentThread _KTHREAD
.text: push dword ptr [ebx] ; 异常链表
.text: mov dword ptr [ebx], 0FFFFFFFFh ; 初始化链表
.text: mov ebp, [esi+18h] ; 获取线程堆栈
.text:0040773C push ; MODE_MASK = User Mode
.text:0040773E sub esp, 48h ; 分配剩余 _KTRAP_FRAME 框架
.text: sub ebp, 29Ch ; (_FX_SAVE_AREA)NPX_FRAME_LENGTH=210h, (_KTRAP_FRAME)KTRAP_FRAME_LENGTH=8C
.text: mov byte ptr [esi+140h], ; MODE_MASK = 1 设置线程模式
.text: ; 现在_KTRAP_FRAME已经建立完成
.text: ;
.text: ; 计算初始堆栈线程的初始堆栈指针,包含NPX和_KTRAP_FRAME
.text: ;
.text: ; 如果 ebp 和 esp 不相等, 那么这是一个V86模式的线程. 拒绝调用.
.text:0040774E cmp ebp, esp
.text: jnz loc_4076C8 ; 处理V86模式的代码不看了.
.text: and dword ptr [ebp+2Ch], ; 清空 Dr7 调试寄存器
.text:0040775A test byte ptr [esi+2Ch], 0FFh ; 线程是否被调试状态
.text:0040775E mov [esi+134h], ebp ; ebp = _KTRAP_FRAME 保存新的 TrapFrame
.text: jnz Dr_FastCallDrSave ; 如果线程被调试, 那么还要做些处理, 这里先不管.
.text:0040776A
.text:0040776A loc_40776A: ; CODE XREF: Dr_FastCallDrSave+10j
.text:0040776A ; Dr_FastCallDrSave+7Cj
.text:0040776A mov ebx, [ebp+60h] ; ebx = _KTRAP_FRAME->Ebp
.text:0040776D mov edi, [ebp+68h] ; edi = _KTRAP_FRAME->Eip
.text: mov [ebp+0Ch], edx ; edx = 参数指针
.text: mov dword ptr [ebp+], 0BADB0D00h
.text:0040777A mov [ebp+], ebx ; _KTRAP_FRAME.DbgEbp = _KTRAP_FRAME->Ebp
.text:0040777D mov [ebp+], edi ; _KTRAP_FRAME.DbgEip = _KTRAP_FRAME->Eip
.text: sti
.text:
.text: loc_407781: ; CODE XREF: _KiBBTUnexpectedRange+18j
.text: ; _KiSystemService+6Ej
.text: mov edi, eax ; 系统调用号
.text: shr edi, ; NtProtectVirtualMemory 89h = 10001001
.text: ; shr右移8位为0
.text: ; Shadow SSDT函数索引都在0x1000以上
.text: and edi, 30h
.text: mov ecx, edi ; 如果是shadow ecx = 10h,否则ecx =0h (bit11 bit12)
.text:0040778B add edi, [esi+0E0h] ; 确定是哪个表
.text:0040778B ; 本线程的系统调用表
.text:0040778B ; EDI指向描述块0或描述块1
.text: mov ebx, eax ; 将eax中的索引值,赋值给ebx
.text: and eax, 0FFFh ; SERVICE_NUMBER_MASK定义为0xFFF
.text: cmp eax, [edi+] ; 检查系统调用号是否越界
.text: ; SERVICE_DESCRIPTOR_LIMIT定义为8
.text:0040779B jnb _KiBBTUnexpectedRange ; 系统调用号越界,超过SSDT表中的Number
我们这里是直接跳转到loc_407781的地方,再此之前的代码是通过用户层调用Native API的时候进行的处理。
这里eax保存着系统调用号,在KTHREAD中有一个指针ServiceTable,如果是gui线程则指向KeServiceDescriptorTableShadow[],如果不是则指向KeServiceDescriptor[]。这里检查了系统调用号是否越界。多数情况下不会越界,我们继续往下看:
.text:004077A1 cmp ecx, 10h ; 测试是否调用 Shadow Ssdt
.text:004077A4 jnz short NotWin32K ; 不跳则是shadow
.text:004077A6 mov ecx, ds:0FFDFF018h ; ecx = _KPCR->_NT_TIB->Self 指向 _TEB
.text:004077AC xor ebx, ebx
.text:004077AE
.text:004077AE loc_4077AE: ; DATA XREF: _KiTrap0E+110o
.text:004077AE or ebx, [ecx+0F70h] ; _TEB.GdiBatchCount
.text:004077B4 jz short NotWin32K
.text:004077B6 push edx ; edx = argc
.text:004077B7 push eax ; eax = Index
.text:004077B8 call ds:_KeGdiFlushUserBatch
.text:004077BE pop eax ; eax = Index
.text:004077BF pop edx ; edx = argc
.text:004077C0
.text:004077C0 NotWin32K: ; CODE XREF: _KiFastCallEntry+B4j
.text:004077C0 ; _KiFastCallEntry+C4j
.text:004077C0 inc dword ptr ds:0FFDFF638h ; _KPRCB->KeSystemCalls++, 记录系统调用次数
.text:004077C6 mov esi, edx ; 使ESI指向用户空间堆栈上的参数块
.text:004077C8 mov ebx, [edi+0Ch] ; ebx = ssdt->ParamTableBase
.text:004077C8 ; [edi+SERVICE_DESCRIPTOR_NUMBER]
.text:004077CB xor ecx, ecx
.text:004077CD mov cl, [eax+ebx] ; 寄存器ECX cl = 参数总共占得字节大小
.text:004077D0 mov edi, [edi] ; edi=ssdt->ServiceTableBase
.text:004077D0 ; EDI指向具体的系统调用表
.text:004077D0 ; [edi+SERVICE_DESCRIPTOR_BASE]
.text:004077D2 mov ebx, [edi+eax*] ; 函数指针
.text:004077D5 sub esp, ecx ; 系统堆栈上留出空间
.text:004077D7 shr ecx, ; 除4,参数个数
.text:004077DA mov edi, esp ; edi = 内核栈的参数位置
.text:004077DC cmp esi, ds:_MmUserProbeAddress ; 参数块的位置不得高于MmSystemRangeStart-0x10000
.text:004077E2 jnb AccessViolation
.text:004077E8
.text:004077E8 loc_4077E8: ; CODE XREF: _KiFastCallEntry+2A4j
.text:004077E8 ; DATA XREF: _KiTrap0E+106o
.text:004077E8 rep movsd ; 复制参数,以ESI为源,EDI为目标,ECX为循环次数
.text:004077E8 ; ecx是参数个数,从用户栈复制参数到内核栈,原来SSDT所有参数都是4个字节为单位的.
.text:004077EA call ebx ; 调用目标函数
这里将ECX与0x10比较,如果不是0x10则为基本调用表(SSDT函数),转到NotWin32K处。这里ecx的cl保存着KSERVICE_TABLE_DESCRIPTOR结构体中的Number,将cl右移2位就是参数的个数,后面重复执行的movsd的次数就是参数的个数,不过复制之前要调整堆栈指针,将ESP与移位前的ECX相减,在系统空间堆栈上 空出相应的字节数。注意movsd指令以ESI所指处为源,以EDI所指处为目标,另一方面,指令获得函数的指针赋值为ebx,最后call ebx实现了对目标函数的调用。
一些安全软件对KiFastCallEntry通过Hook实现过滤SSDT框架的时候,通常是在ebx完成赋值之后,在call ebx之前,替换这中间的地方,进入fake1函数,将保存好的参数push,比如edi保存的SSDT表地址,ebx保存函数地址,eax保存调用号,ecx保存参数个数,在这中间hook,可以直接利用系统初始化好的寄存器,然后调用filter函数,通过寄存器的值,过滤指定的SSDT函数,替换ebx的值,然后继续执行KiFastCallEntry中的call ebx,这样就可以过滤整个SSDT系统调用了。
当执行完成call ebx,从目标函数返回时我们继续看下面的指令:
.text:004077EC
.text:004077EC loc_4077EC: ; CODE XREF: _KiFastCallEntry+2AFj
.text:004077EC ; DATA XREF: _KiTrap0E+126o ...
.text:004077EC mov esp, ebp ; 恢复栈顶,此时栈顶是KTRAP_FRAME
.text:004077EE
.text:004077EE KeReturnFromSystemCall: ; CODE XREF: _KiBBTUnexpectedRange+38j
.text:004077EE ; _KiBBTUnexpectedRange+43j
.text:004077EE mov ecx, ds:0FFDFF124h ; ecx = _KTHREAD
.text:004077F4 mov edx, [ebp+3Ch] ; edx = KTRAP_FRAME->Edx
.text:004077F4 ; 从堆栈中取出保存着的框架指针
.text:004077F4 ; [ebp+KTRAP_FRAME_EDX]
.text:004077F7 mov [ecx+134h], edx ; KThread->TrapFrame = KTRAP_FRAME->Edx 恢复ring3 陷阱帧.
.text:004077F7 _KiFastCallEntry endp ; sp-analysis failed
首先将堆栈指针恢复指向系统调用框架即自陷框架的底部,因为这些参数已经失去意义,然后把原先保存在堆栈上的先前自陷框架指针恢复到当前线程的控制块中。
0x05 KiServiceExit
然后继续执行KiServiceExit函数
.text:004077FD _KiServiceExit proc near ; CODE XREF: _KiSetLowWaitHighThread+7Cj
.text:004077FD ; NtContinue(x,x)+42j ...
.text:004077FD
.text:004077FD arg_C = dword ptr 10h
.text:004077FD arg_10 = dword ptr 14h
.text:004077FD arg_40 = dword ptr 44h
.text:004077FD arg_44 = dword ptr 48h
.text:004077FD arg_48 = dword ptr 4Ch
.text:004077FD arg_60 = dword ptr 64h
.text:004077FD arg_64 = dword ptr 68h
.text:004077FD arg_68 = dword ptr 6Ch
.text:004077FD arg_6C = dword ptr 70h
.text:004077FD
.text:004077FD ; FUNCTION CHUNK AT .text:00407908 SIZE 00000088 BYTES
.text:004077FD
.text:004077FD cli ; 关中断
.text:004077FE test dword ptr [ebp+70h], 20000h ; _KTRAP_FRAME->EFlags is this a V86 frame
.text: jnz short CHECK_FOR_APC_DELIVER ; 跳则不是V86
.text: test byte ptr [ebp+6Ch], ; KTRAP_FRAME->SegCs 测试CS是否是R3选择子
.text:0040780B jz short loc_407864 ; 如果CPL非0则跳.
.text:0040780D
.text:0040780D CHECK_FOR_APC_DELIVER: ; CODE XREF: _KiServiceExit+8j
.text:0040780D ; _KiServiceExit+63j
.text:0040780D mov ebx, ds:0FFDFF124h ; ebx->_KTHREAD
.text: mov byte ptr [ebx+2Eh], ; 清除线程警觉位. APC有关.
.text: cmp byte ptr [ebx+4Ah], ; 这里判断是否有APC挂起
.text:0040781B jz short loc_407864 ; 没有APC挂起
.text:0040781B ; 如果先前模式是内核模式,就往前跳转到下面,不递交APC请求
.text:0040781D mov ebx, ebp
.text:0040781F mov [ebx+44h], eax ; 保存调用例程的返回值
.text: mov dword ptr [ebx+50h], 3Bh
.text: mov dword ptr [ebx+38h], 23h
.text: mov dword ptr [ebx+34h], 23h
.text: mov dword ptr [ebx+30h],
.text:0040783E mov ecx, ; APC_LEVEL 将当前线程IRQL调整到APC_LEVEL
.text: call ds:__imp_@KfRaiseIrql@4 ; 这是快速调用模式的函数,通过寄存器传递参数
.text: push eax ; 保存旧的IRQL.
.text:0040784A sti ; 开中断以后, 有可能带来线程切换
.text:0040784B push ebx ; _KTRAP_FRAME
.text:0040784C push ; Null exception frame
.text:0040784E push ; Previous mode = User Mode
.text: call _KiDeliverApc@12 ; 执行内核APC,并未用户空间APC的执行进行准备
.text: pop ecx ; 从堆栈恢复老的运行级别
.text: call ds:__imp_@KfLowerIrql@4 ; 恢复原来的运行级别,在这里应该是PASSIVE_LEVEL
.text:0040785C mov eax, [ebx+44h] ; 重新读出Eax
.text:0040785F cli
.text: jmp short CHECK_FOR_APC_DELIVER ; 这是一个循环, 循环的处理APC
.text: ; ---------------------------------------------------------------------------
.text: align
.text:
.text: loc_407864: ; CODE XREF: _KiServiceExit+Ej
.text: ; _KiServiceExit+1Ej
.text: mov edx, [esp+arg_48] ; ExceptionList arg_48 = 0x4C
.text: mov ebx, large fs:50h
.text:0040786F mov large fs:, edx ; 还原线程seh
.text: mov ecx, [esp+arg_44]
.text:0040787A mov esi, large fs:124h ; esi-->_KTHREAD
.text: mov [esi+140h], cl ; _KTHREAD.PreviousMode = _KTRAP_FRAME.PreviousPreviousMode
.text: test ebx, 0FFh ; 当前线程是否在调试
.text:0040788D jnz short loc_407908 ; 是被调试, 则跳走
.text:0040788F
.text:0040788F loc_40788F: ; CODE XREF: _KiServiceExit+11Bj
.text:0040788F ; _KiServiceExit+14Aj
.text:0040788F test [esp+arg_6C], 20000h ; 判断当前是否是V86模式.
.text:0040788F ; arg_6c = 0x70
.text: jnz loc_408188 ; 是, 则跳走
.text:0040789D test word ptr [esp+arg_68], 0FFF8h ; FRAME_EDITED
.text:004078A4 jz loc_40795E
.text:004078AA cmp word ptr [esp+arg_68], 1Bh ; set/clear ZF
.text:004078B0 bt word ptr [esp+arg_68], ; test MODE_MASK set/clear CF
.text:004078B7 cmc
.text:004078B8 ja loc_40794C ; jmp if CF=0 and ZF=0
.text:004078BE cmp word ptr [ebp+6Ch], ; _KTRAP_FRAME.Cs 选择子的合法性
.text:004078C3 jz short loc_4078CA ; 如果CS是内核模式, 那么我们直接就可以跳到恢复通用寄存器的地方
.text:004078C5
.text:004078C5 loc_4078C5: ; CODE XREF: _KiServiceExit+15Cj
.text:004078C5 lea esp, [ebp+50h] ; 恢复FS
.text:004078C8 pop fs
.text:004078CA assume fs:nothing
.text:004078CA
.text:004078CA loc_4078CA: ; CODE XREF: _KiServiceExit+C6j
.text:004078CA lea esp, [ebp+54h] ; 获取edi的值
.text:004078CD pop edi
.text:004078CE pop esi
.text:004078CF pop ebx
.text:004078D0 pop ebp
.text:004078D1 cmp word ptr [esp-60h+arg_64], 80h
.text:004078D8 ja loc_4081A4
.text:004078DE add esp,
.text:004078E1 test [esp-64h+arg_64], ; 是从用户空间发起的调用
.text:004078E1 _KiServiceExit endp ; sp-analysis failed
.text:004078E1
.text:004078E9
.text:004078E9 ; =============== S U B R O U T I N E =======================================
.text:004078E9
.text:004078E9
.text:004078E9 _KiSystemCallExitBranch proc near ; DATA XREF: KiEnableFastSyscallReturn():loc_439CBBr
.text:004078E9 ; KiEnableFastSyscallReturn()+26w ...
.text:004078E9 jnz short _KiSystemCallExit ; 测试是否是从内核种发起的调用
.text:004078EB pop edx
.text:004078EC pop ecx
.text:004078ED popf
.text:004078EE jmp edx ; 从内核中发起的调用, 在这里返回
.text:004078F0 ; ---------------------------------------------------------------------------
.text:004078F0
.text:004078F0 _KiSystemCallExit: ; CODE XREF: _KiSystemCallExitBranchj
.text:004078F0 ; _KiSystemCallExit2+5j
.text:004078F0 ; DATA XREF: ...
.text:004078F0 iret
.text:004078F0 _KiSystemCallExitBranch endp ; sp-analysis failed
.text:004078F0
.text:004078F1
.text:004078F1 ; =============== S U B R O U T I N E =======================================
.text:004078F1
.text:004078F1
.text:004078F1 _KiSystemCallExit2 proc near ; DATA XREF: KiRestoreFastSyscallReturnState()+16o
.text:004078F1
.text:004078F1 arg_5 = byte ptr
.text:004078F1
.text:004078F1 test [esp+arg_5],
.text:004078F6 jnz short _KiSystemCallExit ; 不为0是则是通过自陷指令进入内核的
.text:004078F8 pop edx ; New R3 EIP
.text:004078F9 add esp, ; Skip R3 DS
.text:004078FC and [esp-+arg_5], 0FDh ; NOT EFLAGS_INTERRUPT_MASK ; 关闭中断标记位
.text: popf ; 还原eflag
.text: pop ecx ; ecx = _KTRAP_FRAME.esp r3 的栈顶
.text: sti ; 开中断
.text: sysexit ; 退出内核模式.
.text: iret
.text: _KiSystemCallExit2 endp ; sp-analysis failed
在KiServiceExit执行的时候,首先关闭中断,然后检查是否有APC请求,如果有就通过KiDeliverApc递交APC请求(插入线程apc队列)。
最后会通过TrapFrame返回r3或者返回内核调用Zw函数的地方。
0x06 NtProtectVirtualMemory
我们再看看call ebx之后调用Nt函数的情况,NtProtectVirtualMemory代码如下:
PAGE:0049ACB1 ; NTSTATUS __stdcall NtProtectVirtualMemory(HANDLE ProcessHandle, PVOID *BaseAddress, PULONG ProtectSize, ULONG NewProtect, PULONG OldProtect)
PAGE:0049ACB1 _NtProtectVirtualMemory@20 proc near ; DATA XREF: .text:0040B8CCo
PAGE:0049ACB1
PAGE:0049ACB1 var_54 = dword ptr -54h
PAGE:0049ACB1 Status = dword ptr -3Ch
PAGE:0049ACB1 LastProtect = dword ptr -38h
PAGE:0049ACB1 CurrentProcess = dword ptr -34h
PAGE:0049ACB1 var_30 = dword ptr -30h
PAGE:0049ACB1 AccessMode = byte ptr -2Ch
PAGE:0049ACB1 Attached = dword ptr -28h
PAGE:0049ACB1 CapturedBase = dword ptr -24h
PAGE:0049ACB1 CapturedRegionSize= dword ptr -20h
PAGE:0049ACB1 Object = dword ptr -1Ch
PAGE:0049ACB1 ms_exc = CPPEH_RECORD ptr -18h
PAGE:0049ACB1 ProcessHandle = dword ptr
PAGE:0049ACB1 BaseAddress = dword ptr 0Ch
PAGE:0049ACB1 ProtectSize = dword ptr 10h
PAGE:0049ACB1 NewProtect = dword ptr 14h
PAGE:0049ACB1 OldProtect = dword ptr 18h
PAGE:0049ACB1
PAGE:0049ACB1 ; FUNCTION CHUNK AT PAGE:004B9A0D SIZE 00000024 BYTES
PAGE:0049ACB1 ; FUNCTION CHUNK AT PAGE:004E7866 SIZE 00000018 BYTES
PAGE:0049ACB1 ; FUNCTION CHUNK AT PAGE:0051C445 SIZE 00000023 BYTES
PAGE:0049ACB1 ; FUNCTION CHUNK AT PAGE:0051C46D SIZE 0000000E BYTES
PAGE:0049ACB1 ; FUNCTION CHUNK AT PAGE:0051C480 SIZE 00000044 BYTES
PAGE:0049ACB1 ; FUNCTION CHUNK AT PAGE:0051C4C9 SIZE 00000004 BYTES
PAGE:0049ACB1 ; FUNCTION CHUNK AT PAGE:0051C4D2 SIZE 00000008 BYTES
PAGE:0049ACB1
PAGE:0049ACB1 push 44h
PAGE:0049ACB3 push offset stru_413468
PAGE:0049ACB8 call __SEH_prolog
PAGE:0049ACBD xor ebx, ebx
PAGE:0049ACBF mov [ebp+Attached], ebx
PAGE:0049ACC2 mov ecx, [ebp+NewProtect] ; NewProtect
PAGE:0049ACC5 call @MiMakeProtectionMask@4 ; ProtectionMask = MiMakeProtectionMask (NewProtect);
PAGE:0049ACCA cmp eax, 0FFFFFFFFh ; MM_INVALID_PROTECTION
PAGE:0049ACCD jz loc_51C445 ; STATUS_INVALID_PAGE_PROTECTION
PAGE:0049ACD3 mov eax, large fs:124h ; PsGetCurrentThread()
PAGE:0049ACD9 mov ecx, [eax+44h] ; EHTREAD中的Process指针
PAGE:0049ACDC mov [ebp+CurrentProcess], ecx ; PsGetCurrentProcessByThread (CurrentThread)
PAGE:0049ACDF mov al, [eax+140h] ; PreviousMode
PAGE:0049ACE5 mov [ebp+AccessMode], al
PAGE:0049ACE8 test al, al
PAGE:0049ACEA jz loc_4E7866 ; PreviousMode = 0(jz为0 跳转)
PAGE:0049ACEA ; 用户层调用则不跳转
PAGE:0049ACF0 mov [ebp+ms_exc.registration.TryLevel], ebx
PAGE:0049ACF3 mov edi, [ebp+BaseAddress]
PAGE:0049ACF6 mov eax, _MmUserProbeAddress ;ProbeForWritePointer (BaseAddress);
PAGE:0049ACFB cmp edi, eax
PAGE:0049ACFD jnb loc_51C44F
PAGE:0049AD03
PAGE:0049AD03 loc_49AD03: ; CODE XREF: NtProtectVirtualMemory(x,x,x,x,x)+817A0j
PAGE:0049AD03 mov eax, [edi]
PAGE:0049AD05 mov [edi], eax
PAGE:0049AD07 mov esi, [ebp+ProtectSize]
PAGE:0049AD0A mov eax, _MmUserProbeAddress ;ProbeForWriteUlong_ptr (RegionSize);
PAGE:0049AD0F cmp esi, eax
PAGE:0049AD11 jnb loc_51C456
PAGE:0049AD17
PAGE:0049AD17 loc_49AD17: ; CODE XREF: NtProtectVirtualMemory(x,x,x,x,x)+817A7j
PAGE:0049AD17 mov eax, [esi]
PAGE:0049AD19 mov [esi], eax
PAGE:0049AD1B mov ebx, [ebp+OldProtect]
PAGE:0049AD1E mov eax, _MmUserProbeAddress ;ProbeForWriteUlong (OldProtect);
PAGE:0049AD23 cmp ebx, eax
PAGE:0049AD25 jnb loc_51C45D
PAGE:0049AD2B
PAGE:0049AD2B loc_49AD2B: ; CODE XREF: NtProtectVirtualMemory(x,x,x,x,x)+817B2j
PAGE:0049AD2B mov eax, [ebx]
PAGE:0049AD2D mov [ebx], eax
PAGE:0049AD2F mov ecx, [edi]
PAGE:0049AD31 mov [ebp+CapturedBase], ecx ;CapturedBase = *BaseAddress;
PAGE:0049AD34 mov edx, [esi]
PAGE:0049AD36 mov [ebp+CapturedRegionSize], edx ;CapturedRegionSize = *RegionSize;
PAGE:0049AD39 or [ebp+ms_exc.registration.TryLevel], 0FFFFFFFFh
PAGE:0049AD3D
PAGE:0049AD3D loc_49AD3D: ; CODE XREF: NtProtectVirtualMemory(x,x,x,x,x)+4CBC8j
PAGE:0049AD3D mov eax, _MmHighestUserAddress
PAGE:0049AD42 cmp ecx, eax ; CapturedBase>MM_HIGHEST_USER_ADDRESS
PAGE:0049AD44 ja RETURN_STATUS_INVALID_PARAMETER_2 ; 跳转无效的参数
PAGE:0049AD4A sub eax, ecx ; MM_HIGHEST_USER_ADDRESS-CapturedBase
PAGE:0049AD4C cmp eax, edx ; >CapturedRegionSize
PAGE:0049AD4E jb RETURN_STATUS_INVALID_PARAMETER_3 ; 跳转无效的参数
PAGE:0049AD54 test edx, edx ; edx=0
PAGE:0049AD56 jz RETURN_STATUS_INVALID_PARAMETER_3 ; 跳转无效的参数
PAGE:0049AD5C push ; HandleInformation
PAGE:0049AD5E lea eax, [ebp+Object]
PAGE:0049AD61 push eax ; Object
PAGE:0049AD62 push dword ptr [ebp+AccessMode] ; AccessMode
PAGE:0049AD65 push _PsProcessType ; ObjectType
PAGE:0049AD6B push ; DesiredAccess PROCESS_VM_OPERATION
PAGE:0049AD6D push [ebp+ProcessHandle] ; Handle
PAGE:0049AD70 call _ObReferenceObjectByHandle@24 ; ObReferenceObjectByHandle(x,x,x,x,x,x)
PAGE:0049AD75 test eax, eax
PAGE:0049AD77 jl loc_49AE0B ; 返回0则跳走
PAGE:0049AD7D mov eax, [ebp+Object]
PAGE:0049AD80 cmp [ebp+CurrentProcess], eax ; 比较ObRef..得到的Process是否等于CurrentProcess
PAGE:0049AD83 jnz loc_4B9A1B ; 不等于,调用KeStackAttachProcess附加到进程空间
PAGE:0049AD89
PAGE:0049AD89 loc_49AD89: ; CODE XREF: NtProtectVirtualMemory(x,x,x,x,x)+1ED7Bj
PAGE:0049AD89 lea eax, [ebp+LastProtect]
PAGE:0049AD8C push eax ; &LastProtect
PAGE:0049AD8D push [ebp+NewProtect] ; NewProtect
PAGE:0049AD90 lea eax, [ebp+CapturedRegionSize]
PAGE:0049AD93 push eax ; &CapturedRegionSize
PAGE:0049AD94 lea eax, [ebp+CapturedBase]
PAGE:0049AD97 push eax ; &CapturedBase
PAGE:0049AD98 push [ebp+Object] ; Process
PAGE:0049AD9B call _MiProtectVirtualMemory@20 ; MiProtectVirtualMemory(x,x,x,x,x)
PAGE:0049ADA0 mov [ebp+Status], eax
PAGE:0049ADA3 cmp [ebp+Attached], ; 是否附加了,之前KeStackDetachProcess调用的时候修改的
PAGE:0049ADA7 jnz loc_4B9A0D ; 调用KeUnStackDetachProcess
PAGE:0049ADAD
PAGE:0049ADAD loc_49ADAD: ; CODE XREF: NtProtectVirtualMemory(x,x,x,x,x)+1ED65j
PAGE:0049ADAD mov ecx, [ebp+Object] ; Object
PAGE:0049ADB0 call @ObfDereferenceObject@4 ; ObfDereferenceObject(x)
PAGE:0049ADB5 mov [ebp+ms_exc.registration.TryLevel],
PAGE:0049ADBC cmp [ebp+AccessMode],
PAGE:0049ADC0 jz short loc_49ADF5 ; 内核模式跳转
PAGE:0049ADC0 ; *RegionSize = CapturedRegionSize;
PAGE:0049ADC0 ; *BaseAddress = CapturedBase;
PAGE:0049ADC0 ; *OldProtect = LastProtect;
PAGE:0049ADC2 mov eax, _MmUserProbeAddress
PAGE:0049ADC7 cmp edi, eax
PAGE:0049ADC9 jnb loc_51C4A3
PAGE:0049ADCF
PAGE:0049ADCF loc_49ADCF: ; CODE XREF: NtProtectVirtualMemory(x,x,x,x,x)+817F8j
PAGE:0049ADCF mov eax, [edi]
PAGE:0049ADD1 mov [edi], eax
PAGE:0049ADD3 mov eax, _MmUserProbeAddress
PAGE:0049ADD8 cmp esi, eax
PAGE:0049ADDA jnb loc_51C4AE
PAGE:0049ADE0
PAGE:0049ADE0 loc_49ADE0: ; CODE XREF: NtProtectVirtualMemory(x,x,x,x,x)+81803j
PAGE:0049ADE0 mov eax, [esi]
PAGE:0049ADE2 mov [esi], eax
PAGE:0049ADE4 mov eax, _MmUserProbeAddress
PAGE:0049ADE9 cmp ebx, eax
PAGE:0049ADEB jnb loc_51C4B9
PAGE:0049ADF1
PAGE:0049ADF1 loc_49ADF1: ; CODE XREF: NtProtectVirtualMemory(x,x,x,x,x)+8180Ej
PAGE:0049ADF1 mov eax, [ebx]
PAGE:0049ADF3 mov [ebx], eax
PAGE:0049ADF5
PAGE:0049ADF5 loc_49ADF5: ; CODE XREF: NtProtectVirtualMemory(x,x,x,x,x)+10Fj
PAGE:0049ADF5 mov eax, [ebp+CapturedRegionSize]
PAGE:0049ADF8 mov [esi], eax
PAGE:0049ADFA mov eax, [ebp+CapturedBase]
PAGE:0049ADFD mov [edi], eax
PAGE:0049ADFF mov eax, [ebp+LastProtect]
PAGE:0049AE02 mov [ebx], eax
PAGE:0049AE04
PAGE:0049AE04 loc_49AE04: ; CODE XREF: NtProtectVirtualMemory(x,x,x,x,x)+81824j
PAGE:0049AE04 or [ebp+ms_exc.registration.TryLevel], 0FFFFFFFFh
PAGE:0049AE08 mov eax, [ebp+Status]
PAGE:0049AE0B
PAGE:0049AE0B loc_49AE0B: ; CODE XREF: NtProtectVirtualMemory(x,x,x,x,x)+C6j
PAGE:0049AE0B ; NtProtectVirtualMemory(x,x,x,x,x)+81799j ...
PAGE:0049AE0B call __SEH_epilog
PAGE:0049AE10 retn 14h
PAGE:0049AE10 _NtProtectVirtualMemory@20 endp
我们看到中间有个地方对于PreviousMode有一个判断,如果PreviousMode为用户模式则不跳转,为内核模式则跳转,我们看看跳转的代码:
PAGE:0049ACEA jz loc_4E7866 ; PreviousMode = 0(jz为0 跳转)
PAGE:0049ACEA ; 用户层调用则不跳转
PAGE:004E7866 loc_4E7866: ; CODE XREF: NtProtectVirtualMemory(x,x,x,x,x)+39j
PAGE:004E7866 mov esi, [ebp+ProtectSize]
PAGE:004E7869 mov edx, [esi]
PAGE:004E786B mov [ebp+CapturedRegionSize], edx
PAGE:004E786E mov edi, [ebp+BaseAddress]
PAGE:004E7871 mov ecx, [edi]
PAGE:004E7873 mov [ebp+CapturedBase], ecx
PAGE:004E7876 mov ebx, [ebp+OldProtect]
PAGE:004E7879 jmp loc_49AD3D
PAGE:0049AD3D loc_49AD3D: ; CODE XREF: NtProtectVirtualMemory(x,x,x,x,x)+4CBC8j
PAGE:0049AD3D mov eax, _MmHighestUserAddress
PAGE:0049AD42 cmp ecx, eax ; CapturedBase>MM_HIGHEST_USER_ADDRESS
PAGE:0049AD44 ja RETURN_STATUS_INVALID_PARAMETER_2 ; 跳转无效的参数
PAGE:0049AD4A sub eax, ecx ; MM_HIGHEST_USER_ADDRESS-CapturedBase
PAGE:0049AD4C cmp eax, edx ; >CapturedRegionSize
PAGE:0049AD4E jb RETURN_STATUS_INVALID_PARAMETER_3 ; 跳转无效的参数
PAGE:0049AD54 test edx, edx ; edx=0
PAGE:0049AD56 jz RETURN_STATUS_INVALID_PARAMETER_3 ; 跳转无效的参数
PAGE:0049AD5C push ; HandleInformation
PAGE:0049AD5E lea eax, [ebp+Object]
PAGE:0049AD61 push eax ; Object
PAGE:0049AD62 push dword ptr [ebp+AccessMode] ; AccessMode
PAGE:0049AD65 push _PsProcessType ; ObjectType
PAGE:0049AD6B push ; DesiredAccess PROCESS_VM_OPERATION
PAGE:0049AD6D push [ebp+ProcessHandle] ; Handle
PAGE:0049AD70 call _ObReferenceObjectByHandle@24 ; ObReferenceObjectByHandle(x,x,x,x,x,x)
可以看到如果是内核模式,直接跳过了校验参数合法性的部分,直接走入下面的ObRegerenceObjectByHandle,跳过的代码如下所示:
PAGE:0049ACF0 mov [ebp+ms_exc.registration.TryLevel], ebx
PAGE:0049ACF3 mov edi, [ebp+BaseAddress]
PAGE:0049ACF6 mov eax, _MmUserProbeAddress ;ProbeForWritePointer (BaseAddress);
PAGE:0049ACFB cmp edi, eax
PAGE:0049ACFD jnb loc_51C44F
PAGE:0049AD03
PAGE:0049AD03 loc_49AD03: ; CODE XREF: NtProtectVirtualMemory(x,x,x,x,x)+817A0j
PAGE:0049AD03 mov eax, [edi]
PAGE:0049AD05 mov [edi], eax
PAGE:0049AD07 mov esi, [ebp+ProtectSize]
PAGE:0049AD0A mov eax, _MmUserProbeAddress ;ProbeForWriteUlong_ptr (RegionSize);
PAGE:0049AD0F cmp esi, eax
PAGE:0049AD11 jnb loc_51C456
PAGE:0049AD17
PAGE:0049AD17 loc_49AD17: ; CODE XREF: NtProtectVirtualMemory(x,x,x,x,x)+817A7j
PAGE:0049AD17 mov eax, [esi]
PAGE:0049AD19 mov [esi], eax
PAGE:0049AD1B mov ebx, [ebp+OldProtect]
PAGE:0049AD1E mov eax, _MmUserProbeAddress ;ProbeForWriteUlong (OldProtect);
PAGE:0049AD23 cmp ebx, eax
PAGE:0049AD25 jnb loc_51C45D
所以我们可以看出PreviousMode为内核模式的时候会比用户模式检查的地方少,效率会快一些。
0x07 x64位下的内核Zw调用Nt函数
.text:00000001400795E0 ZwProtectVirtualMemory proc near ; CODE XREF: KiOpPatchCode+C9p
.text:00000001400795E0 ; KiOpPatchCode+1E2p ...
.text:00000001400795E0 8B C4 mov rax, rsp
.text:00000001400795E3 FA cli ; 关中断
.text:00000001400795E4 EC sub rsp, 10h ; 开辟栈区
.text:00000001400795E8 push rax ; 保存栈顶
.text:00000001400795E9 9C pushfq ; ELFALGS
.text:00000001400795EA 6A push 10h
.text:00000001400795EC 8D 7D + lea rax, KiServiceLinkage
.text:00000001400795F3 push rax
.text:00000001400795F4 B8 4D mov eax, 4Dh ; 函数索引0x4D
.text:00000001400795F9 E9 C2 5F jmp KiServiceInternal
.text:00000001400795F9 ZwProtectVirtualMemory endp
.text:000000014007F5C0 KiServiceInternal proc near ; CODE XREF: ZwMapUserPhysicalPagesScatter+19j
.text:000000014007F5C0 ; ZwWaitForSingleObject+19j ...
.text:000000014007F5C0
.text:000000014007F5C0 var_140 = byte ptr -140h
.text:000000014007F5C0 var_30 = qword ptr -30h
.text:000000014007F5C0 var_28 = qword ptr -28h
.text:000000014007F5C0 var_20 = qword ptr -20h
.text:000000014007F5C0 var_18 = qword ptr -18h
.text:000000014007F5C0
.text:000000014007F5C0 EC sub rsp,
.text:000000014007F5C4 push rbp
.text:000000014007F5C5 EC + sub rsp, 158h ; 栈区
.text:000000014007F5CC 8D AC + lea rbp, [rsp+80h] ; 栈底
.text:000000014007F5D4 9D C0 + mov [rbp+0E8h+var_28], rbx ; mov TrRbx[rbp], rbx
.text:000000014007F5DB BD C8 + mov [rbp+0E8h+var_20], rdi ; mov TrRdi[rbp], rdi
.text:000000014007F5E2 B5 D0 + mov [rbp+0E8h+var_18], rsi ; mov TrRsi[rbp], rsi
.text:000000014007F5E9 FB sti ; 开中断
.text:000000014007F5EA 8B 1C + mov rbx, gs:188h ; PcCurrentThread get current thread address
.text:000000014007F5F3 0F 0D 8B D8 + prefetchw byte ptr [rbx+1D8h] ; prefetch with write intent
.text:000000014007F5FA 0F B6 BB F6 + movzx edi, byte ptr [rbx+1F6h] ; save previous mode in trap frame
.text:000000014007F601 7D A8 mov [rbp+0E8h+var_140], dil
.text:000000014007F605 C6 F6 + mov byte ptr [rbx+1F6h], ; set thread previous mode
.text:000000014007F60C 4C 8B D8 + mov r10, [rbx+1D8h] ; save previous frame pointer address
.text:000000014007F613 4C B8 + mov [rbp+0E8h+var_30], r10
.text:000000014007F61A 4C 8D 1D 3D + lea r11, KiSystemServiceStart ; get address of service start
.text:000000014007F621 FF E3 jmp r11 ; finish in common code
.text:000000014007F621 KiServiceInternal endp
ZwProtectVirtualMemory调用了KiServiceLinkage,把系统服务序号放进eax后又调用了KiServiceInternal,KiServiceInternal又调用了KiSystemServiceStart。KiServiceLinkage和KiServiceInternal是初始化系统服务的,KiSystemServiceStart则是开始执行系统服务。我们再看看KiSystemServiceStart干了些什么:
.text:000000014007F640 KiSystemCall64 proc near ; DATA XREF: KiInitializeBootStructures+26Eo
.text:000000014007F640
...
.text:000000014007F75E KiSystemServiceStart: ; DATA XREF: KiServiceInternal+5Ao
.text:000000014007F75E ; .data:00000001401EE648o
.text:000000014007F75E A3 D8 + mov [rbx+1D8h], rsp ; ThTrapFrame[rbx] set current frame pointer address
.text:000000014007F765 8B F8 mov edi, eax
.text:000000014007F767 C1 EF shr edi, ; SERVICE_TABLE_SHIFT
.text:000000014007F76A E7 and edi, 20h ; SERVICE_TABLE_MASK
.text:000000014007F76D FF 0F and eax, 0FFFh ; SERVICE_NUMBER_MASK
.text:000000014007F772
.text:000000014007F772 KiSystemServiceRepeat: ; CODE XREF: KiSystemCall64+47Bj
.text:000000014007F772 4C 8D C7 + lea r10, KeServiceDescriptorTable ; get table base address
.text:000000014007F779 4C 8D 1D + lea r11, KeServiceDescriptorTableShadow
.text:000000014007F780 F7 + test dword ptr [rbx+100h], 80h
.text:000000014007F78A 4D 0F D3 cmovnz r10, r11
.text:000000014007F78E 3B cmp eax, [rdi+r10+10h]
.text:000000014007F793 0F E9 + jnb loc_14007FA82
.text:000000014007F799 4E 8B mov r10, [rdi+r10]
.text:000000014007F79D 4D 1C movsxd r11, dword ptr [r10+rax*] ; get system service offset
.text:000000014007F7A1 8B C3 mov rax, r11
.text:000000014007F7A4 C1 FB sar r11,
.text:000000014007F7A8 4D D3 add r10, r11 ; add table base to
.text:000000014007F7AB FF cmp edi, 20h ; check if GUI service
.text:000000014007F7AE jnz short loc_14007F800 ; if ne,not GUI service
.text:000000014007F7B0 4C 8B 9B B8 + mov r11, [rbx+0B8h] ; get user TEB address
.text:000000014007F7B7
.text:000000014007F7B7 KiSystemServiceGdiTebAccess: ; DATA XREF: KiSystemServiceHandler+Do
.text:000000014007F7B7 BB + cmp dword ptr [r11+1740h], ; check batch queue depth
.text:000000014007F7BF 3F jz short loc_14007F800 ; if e,batch queue empty
.text:000000014007F7C1 B0 mov [rbp-50h], rax
.text:000000014007F7C5 4D B8 mov [rbp-48h], rcx ; mov TrRcx[rbp],rcx save system service arguments
.text:000000014007F7C9 C0 mov [rbp-40h], rdx
.text:000000014007F7CD 8B D8 mov rbx, r8
.text:000000014007F7D0 8B F9 mov rdi, r9
.text:000000014007F7D3 8B F2 mov rsi, r10 ; save system service address
.text:000000014007F7D6 FF 1F + call cs:KeGdiFlushUserBatch ; call flush GDI user batch routine
.text:000000014007F7DC 8B B0 mov rax, [rbp-50h] ; restore system service arguments
.text:000000014007F7E0 8B 4D B8 mov rcx, [rbp-48h]
.text:000000014007F7E4 8B C0 mov rdx, [rbp-40h]
.text:000000014007F7E8 4C 8B C3 mov r8, rbx
.text:000000014007F7EB 4C 8B CF mov r9, rdi
.text:000000014007F7EE 4C 8B D6 mov r10, rsi ; restore system service address
.text:000000014007F7F1 db 66h, 66h, 66h, 66h, 66h, 66h
.text:000000014007F7F1 + nop word ptr [rax+rax+00000000h]
.text:000000014007F800
.text:000000014007F800 loc_14007F800: ; CODE XREF: KiSystemCall64+16Ej
.text:000000014007F800 ; KiSystemCall64+17Fj
.text:000000014007F800 E0 0F and eax, 0Fh ; ; Check if system service has any in memory arguments.
.text:000000014007F803 0F B7 + jz KiSystemServiceCopyEnd ; if z, no in memory arguments
.text:000000014007F809 C1 E0 shl eax, ; compute argument bytes for dispatch
.text:000000014007F80C 8D lea rsp, [rsp-70h] ; allocate stack argument area
.text:000000014007F811 8D 7C lea rdi, [rsp+190h+var_178] ; compute copy destination address
.text:000000014007F816 8B B5 + mov rsi, [rbp+100h] ; get previous stack address TrRsp[rbp]
.text:000000014007F81D 8D lea rsi, [rsi+20h] ; compute copy source address
.text:000000014007F821 F6 F0 + test byte ptr [rbp+0F0h], ; check if previous mode user TrSegCs[rbp]
.text:000000014007F828 jz short loc_14007F840 ; if z, previous mode kernel
.text:000000014007F82A 3B CF + cmp rsi, cs:MmUserProbeAddress ; check if source address in range
.text:000000014007F831 0F C7+ cmovnb rsi, cs:MmUserProbeAddress ; if ae, reset copy source address
.text:000000014007F839 0F 1F + nop dword ptr [rax+00000000h]
.text:000000014007F840
.text:000000014007F840 loc_14007F840: ; CODE XREF: KiSystemCall64+1E8j
.text:000000014007F840 4C 8D 1D + lea r11, KiSystemServiceCopyEnd ; get copy ending address
.text:000000014007F847 4C 2B D8 sub r11, rax ; substract number of bytes to copy
.text:000000014007F84A FF E3 jmp r11
.text:000000014007F84A ; ---------------------------------------------------------------------------
.text:000000014007F84D 0F 1F align 10h
.text:000000014007F850
.text:000000014007F850 KiSystemServiceCopyStart: ; DATA XREF: KiSystemServiceHandler+1Ao
.text:000000014007F850 8B mov rax, [rsi+70h] ; copy fourteenth argument
.text:000000014007F854 mov [rdi+70h], rax
.text:000000014007F858 8B mov rax, [rsi+68h] ; copy thirteenth argument
.text:000000014007F85C mov [rdi+68h], rax
.text:000000014007F860 8B mov rax, [rsi+60h] ; copy twelfth argument
.text:000000014007F864 mov [rdi+60h], rax
.text:000000014007F868 8B mov rax, [rsi+58h] ; copy eleventh argument
.text:000000014007F86C mov [rdi+58h], rax
.text:000000014007F870 8B mov rax, [rsi+50h] ; copy tenth argument
.text:000000014007F874 mov [rdi+50h], rax
.text:000000014007F878 8B mov rax, [rsi+48h] ; copy nineth argument
.text:000000014007F87C mov [rdi+48h], rax
.text:000000014007F880 8B mov rax, [rsi+40h] ; copy eighth argument
.text:000000014007F884 mov [rdi+40h], rax
.text:000000014007F888 8B mov rax, [rsi+38h] ; copy seventh argument
.text:000000014007F88C mov [rdi+38h], rax
.text:000000014007F890 8B mov rax, [rsi+30h] ; copy sixth argument
.text:000000014007F894 mov [rdi+30h], rax
.text:000000014007F898 8B mov rax, [rsi+28h] ; copy fifth argument
.text:000000014007F89C mov [rdi+28h], rax
.text:000000014007F8A0 8B mov rax, [rsi+20h] ; copy fourth argument
.text:000000014007F8A4 mov [rdi+20h], rax
.text:000000014007F8A8 8B mov rax, [rsi+18h] ; copy third argument
.text:000000014007F8AC mov [rdi+18h], rax
.text:000000014007F8B0 8B mov rax, [rsi+10h] ; copy second argument
.text:000000014007F8B4 mov [rdi+10h], rax
.text:000000014007F8B8 8B mov rax, [rsi+] ; copy first argument
.text:000000014007F8BC mov [rdi+], rax
.text:000000014007F8C0
.text:000000014007F8C0 KiSystemServiceCopyEnd: ; CODE XREF: KiSystemCall64+1C3j
.text:000000014007F8C0 ; DATA XREF: KiSystemServiceHandler+27o ...
.text:000000014007F8C0 F7 BE 7D + test cs:dword_140207688, 40h
.text:000000014007F8CA 0F + jnz loc_14007FB20
.text:000000014007F8D0 FF D2 call r10 ; call system service
.text:000000014007F8D3
.text:000000014007F8D3 loc_14007F8D3: ; CODE XREF: KiSystemCall64+535j
.text:000000014007F8D3 FF + inc dword ptr gs:2238h ; increment number of system calls gs:[PcSystemCalls]
.text:000000014007F8DB
.text:000000014007F8DB KiSystemServiceExit: ; CODE XREF: KiSystemCall64+49Cj
...
.text:000000014007FB75 KiSystemCall64 endp
KiSystemServiceStart调用了KiSystemServiceRepeat,KiSystemServiceRepeat根据系统服务序号来选择SSDT还是ShadowSSDT(到了KiSystemServiceRepeat才真正调用Nt函数,通过call r11调用了Nt函数)。KiSystemServiceRepeat执行完成之后,会调用KiSystemServiceExit(系统服务调用完毕,会有返回信息)
我们接下来再看看NtProtectVirtualMemory函数的实现
PAGE:0000000140398B2C NtProtectVirtualMemory proc near ; DATA XREF: .text:0000000140081568o
PAGE:0000000140398B2C
PAGE:0000000140398B2C var_78 = qword ptr -78h
PAGE:0000000140398B2C var_70 = qword ptr -70h
PAGE:0000000140398B2C var_68 = qword ptr -68h
PAGE:0000000140398B2C LastProtect = dword ptr -58h
PAGE:0000000140398B2C var_54 = dword ptr -54h
PAGE:0000000140398B2C Object = qword ptr -50h
PAGE:0000000140398B2C CaptureRegionSize= qword ptr -48h
PAGE:0000000140398B2C CapturedBase = qword ptr -40h
PAGE:0000000140398B2C ApcState = byte ptr -38h
PAGE:0000000140398B2C var_8 = byte ptr -
PAGE:0000000140398B2C OldProtect = qword ptr 28h
PAGE:0000000140398B2C
PAGE:0000000140398B2C ; FUNCTION CHUNK AT PAGE:00000001403C8D10 SIZE 0000001E BYTES
PAGE:0000000140398B2C
PAGE:0000000140398B2C 8B C4 mov rax, rsp
PAGE:0000000140398B2F mov [rax+], rsi
PAGE:0000000140398B33 mov [rax+10h], rdi
PAGE:0000000140398B37 4C mov [rax+18h], r12
PAGE:0000000140398B3B 4C mov [rax+20h], r13
PAGE:0000000140398B3F push r14
PAGE:0000000140398B41 EC + sub rsp, 90h
PAGE:0000000140398B48 8B F9 mov edi, r9d ; NewProtect
PAGE:0000000140398B4B 4D 8B E8 mov r13, r8 ; RegionSize
PAGE:0000000140398B4E 4C 8B E2 mov r12, rdx ; *BaseAddress
PAGE:0000000140398B51 4C 8B D1 mov r10, rcx ; ProcessHandle
PAGE:0000000140398B54 8B C9 mov ecx, r9d
PAGE:0000000140398B57 E8 A4 AE CF FF call MiMakeProtectionMask ; MiMakeProtectionMask (NewProtect);
PAGE:0000000140398B5C F8 FF cmp eax, 0FFFFFFFFh
PAGE:0000000140398B5F 0F AB + jz loc_1403C8D10 ; error
PAGE:0000000140398B65 8B + mov rax, gs:188h ; 获得线程体
PAGE:0000000140398B6E 8B mov rsi, [rax+70h] ; 线程所属的Process
PAGE:0000000140398B72 8A F6 + mov r9b, [rax+1F6h] ; PreviousMode
PAGE:0000000140398B79 C9 test r9b, r9b
PAGE:0000000140398B7C 0F 3E + jz loc_140398CC0 ; 如果是KernelMode就跳转
PAGE:0000000140398B7C ; 如果是UserMode就继续执行
PAGE:0000000140398B82 8B CC mov rcx, r12 ; BaseAddress
PAGE:0000000140398B85 8B + mov rax, cs:MmUserProbeAddress ; ProbeForWrite (BaseAddress, sizeof(PVOID64), sizeof(PVOID64));
PAGE:0000000140398B8C 4C 3B E0 cmp r12, rax
PAGE:0000000140398B8F 0F C8 cmovnb rcx, rax ; 大于等于时传送
PAGE:0000000140398B93 8B mov rax, [rcx]
PAGE:0000000140398B96 mov [rcx], rax
PAGE:0000000140398B99 8B C8 mov rcx, r8 ; RegionSize
PAGE:0000000140398B9C 8B 5D + mov rax, cs:MmUserProbeAddress ; ProbeForWrite (RegionSize, sizeof(ULONGLONG), sizeof(ULONGLONG));
PAGE:0000000140398BA3 4C 3B C0 cmp r8, rax
PAGE:0000000140398BA6 0F C8 cmovnb rcx, rax ; 大于等于时传送
PAGE:0000000140398BAA 8B mov rax, [rcx]
PAGE:0000000140398BAD mov [rcx], rax
PAGE:0000000140398BB0 4C 8B B4 C0+ mov r14, [rsp+98h+OldProtect]
PAGE:0000000140398BB8 8B CE mov rcx, r14
PAGE:0000000140398BBB 8B 3E + mov rax, cs:MmUserProbeAddress ; ProbeForWriteUlong (OldProtect);
PAGE:0000000140398BC2 4C 3B F0 cmp r14, rax
PAGE:0000000140398BC5 0F C8 cmovnb rcx, rax
PAGE:0000000140398BC9 8B mov eax, [rcx]
PAGE:0000000140398BCB mov [rcx], eax
PAGE:0000000140398BCD 8B mov rdx, [r12]
PAGE:0000000140398BD1 mov [rsp+98h+CapturedBase], rdx ; CapturedBase = *BaseAddress;
PAGE:0000000140398BD6 8B mov rcx, [r8]
PAGE:0000000140398BD9 4C mov [rsp+98h+CaptureRegionSize], rcx ; CapturedRegionSize = *RegionSize;
PAGE:0000000140398BDE EB jmp short loc_140398BE5
PAGE:0000000140398BE0 ; ---------------------------------------------------------------------------
PAGE:0000000140398BE0 E9 BD jmp loc_140398CA2
PAGE:0000000140398BE5 ; ---------------------------------------------------------------------------
PAGE:0000000140398BE5
PAGE:0000000140398BE5 loc_140398BE5: ; CODE XREF: NtProtectVirtualMemory+B2j
PAGE:0000000140398BE5 ; NtProtectVirtualMemory+1ADj
PAGE:0000000140398BE5 8B + mov rax, cs:MmHighestUserAddress ; 合法性校验
PAGE:0000000140398BEC 3B D0 cmp rdx, rax ; CapturedBase>MM_HIGHEST_USER_ADDRESS
PAGE:0000000140398BEF 0F + ja RETURN_STATUS_INVALID_PARAMETER_2
PAGE:0000000140398BF5 2B C2 sub rax, rdx ; MM_HIGHEST_USER_ADDRESS64-CapturedBase
PAGE:0000000140398BF8 3B C1 cmp rax, rcx
PAGE:0000000140398BFB 0F + jb RETURN_STATUS_INVALID_PARAMETER_3 ; 小于跳转
PAGE:0000000140398C01 C9 test rcx, rcx ; CapturedRegionSize是否为0
PAGE:0000000140398C04 0F 1A + jz RETURN_STATUS_INVALID_PARAMETER_4
PAGE:0000000140398C0A + and [rsp+98h+var_68], ; POBJECT_HANDLE_INFORMATION NULL
PAGE:0000000140398C10 8D lea rax, [rsp+98h+Object]
PAGE:0000000140398C15 mov [rsp+98h+var_70], rax ; Process
PAGE:0000000140398C1A C7 + mov dword ptr [rsp+98h+var_78], 746C6644h ; Tag
PAGE:0000000140398C22 4C 8B F7 + mov r8, cs:PsProcessType ; PsProcessType
PAGE:0000000140398C29 BA mov edx, ; PROCESS_VM_OPERATION
PAGE:0000000140398C2E 8B CA mov rcx, r10 ; ProcessHandle
PAGE:0000000140398C31 E8 AA D8 FD FF call ObReferenceObjectByHandleWithTag ; NTSTATUS ObReferenceObjectByHandleWithTag(
PAGE:0000000140398C31 ; _In_ HANDLE Handle,
PAGE:0000000140398C31 ; _In_ ACCESS_MASK DesiredAccess,
PAGE:0000000140398C31 ; _In_opt_ POBJECT_TYPE ObjectType,
PAGE:0000000140398C31 ; _In_ KPROCESSOR_MODE AccessMode,
PAGE:0000000140398C31 ; _In_ ULONG Tag,
PAGE:0000000140398C31 ; _Out_ PVOID *Object,
PAGE:0000000140398C31 ; _Out_opt_ POBJECT_HANDLE_INFORMATION HandleInformation
PAGE:0000000140398C31 ; );
PAGE:0000000140398C36 C0 test eax, eax
PAGE:0000000140398C38 js short loc_140398CA2
PAGE:0000000140398C3A 3B cmp rsi, [rsp+98h+Object] ; 是否是当前进程
PAGE:0000000140398C3F 0F + jnz loc_140398CDE ; 不是则跳走调用KeAttachProcess
PAGE:0000000140398C45 F6 xor esi, esi
PAGE:0000000140398C47
PAGE:0000000140398C47 loc_140398C47: ; CODE XREF: NtProtectVirtualMemory+1C6j
PAGE:0000000140398C47 8D lea rax, [rsp+98h+LastProtect] ; &LastProtect
PAGE:0000000140398C4C mov [rsp+98h+var_78], rax ; &LastProtect
PAGE:0000000140398C51 8B CF mov r9d, edi ; NewProtect
PAGE:0000000140398C54 4C 8D lea r8, [rsp+98h+CaptureRegionSize] ; CapturedRegionSize
PAGE:0000000140398C59 8D lea rdx, [rsp+98h+CapturedBase] ; CapturedBase
PAGE:0000000140398C5E 8B 4C mov rcx, [rsp+98h+Object] ; Process
PAGE:0000000140398C63 E8 B8 F9 FF FF call MiProtectVirtualMemory
PAGE:0000000140398C68 8B F8 mov edi, eax
PAGE:0000000140398C6A mov [rsp+98h+var_54], eax
PAGE:0000000140398C6E F6 test esi, esi ; 为0则是 调用过KeStackAttachProcess,需要恢复
PAGE:0000000140398C70 0F + jnz loc_140398CF7 ; 调用KeUnStackDetachProcess
PAGE:0000000140398C76
PAGE:0000000140398C76 loc_140398C76: ; CODE XREF: NtProtectVirtualMemory+1D5j
PAGE:0000000140398C76 8B 4C mov rcx, [rsp+98h+Object] ; Object
PAGE:0000000140398C7B E8 C0 CF FF call ObfDereferenceObject ; 减少引用计数
PAGE:0000000140398C80 nop
PAGE:0000000140398C81 8B mov rax, [rsp+98h+CaptureRegionSize] ; CapturedRegionSize
PAGE:0000000140398C86 mov [r13+], rax
PAGE:0000000140398C8A 8B mov rax, [rsp+98h+CapturedBase] ; CapturedBase
PAGE:0000000140398C8F mov [r12], rax
PAGE:0000000140398C93 8B mov eax, [rsp+98h+LastProtect]
PAGE:0000000140398C97 mov [r14], eax
PAGE:0000000140398C9A EB jmp short loc_140398CA0
PAGE:0000000140398C9C ; ---------------------------------------------------------------------------
PAGE:0000000140398C9C 8B 7C mov edi, [rsp+98h+var_54]
PAGE:0000000140398CA0
PAGE:0000000140398CA0 loc_140398CA0: ; CODE XREF: NtProtectVirtualMemory+16Ej
PAGE:0000000140398CA0 8B C7 mov eax, edi
PAGE:0000000140398CA2
PAGE:0000000140398CA2 loc_140398CA2: ; CODE XREF: NtProtectVirtualMemory+B4j
PAGE:0000000140398CA2 ; NtProtectVirtualMemory+10Cj ...
PAGE:0000000140398CA2 4C 8D 9C + lea r11, [rsp+98h+var_8]
PAGE:0000000140398CAA 8B mov rsi, [r11+10h]
PAGE:0000000140398CAE 8B 7B mov rdi, [r11+18h]
PAGE:0000000140398CB2 4D 8B mov r12, [r11+20h]
PAGE:0000000140398CB6 4D 8B 6B mov r13, [r11+28h]
PAGE:0000000140398CBA 8B E3 mov rsp, r11
PAGE:0000000140398CBD 5E pop r14
PAGE:0000000140398CBF C3 retn
PAGE:0000000140398CC0 ; ---------------------------------------------------------------------------
PAGE:0000000140398CC0
PAGE:0000000140398CC0 loc_140398CC0: ; CODE XREF: NtProtectVirtualMemory+50j
PAGE:0000000140398CC0 8B mov rcx, [r8]
PAGE:0000000140398CC3 4C mov [rsp+98h+CaptureRegionSize], rcx ; CapturedRegionSize
PAGE:0000000140398CC8 8B mov rdx, [r12]
PAGE:0000000140398CCC mov [rsp+98h+CapturedBase], rdx ; CapturedBase
PAGE:0000000140398CD1 4C 8B B4 C0+ mov r14, [rsp+98h+OldProtect]
PAGE:0000000140398CD9 E9 FF FF FF jmp loc_140398BE5
PAGE:0000000140398CDE ; ---------------------------------------------------------------------------
PAGE:0000000140398CDE
PAGE:0000000140398CDE loc_140398CDE: ; CODE XREF: NtProtectVirtualMemory+113j
PAGE:0000000140398CDE 8D lea rdx, [rsp+98h+ApcState] ; ApcState
PAGE:0000000140398CE3 8B 4C mov rcx, [rsp+98h+Object] ; Process
PAGE:0000000140398CE8 E8 D1 FF call KeStackAttachProcess ; VOID KeStackAttachProcess(
PAGE:0000000140398CE8 ; _Inout_ PRKPROCESS Process,
PAGE:0000000140398CE8 ; _Out_ PRKAPC_STATE ApcState
PAGE:0000000140398CE8 ; );
PAGE:0000000140398CED BE mov esi,
PAGE:0000000140398CF2 E9 FF FF FF jmp loc_140398C47
PAGE:0000000140398CF7 ; ---------------------------------------------------------------------------
PAGE:0000000140398CF7
PAGE:0000000140398CF7 loc_140398CF7: ; CODE XREF: NtProtectVirtualMemory+144j
PAGE:0000000140398CF7 8D 4C lea rcx, [rsp+98h+ApcState]
PAGE:0000000140398CFC E8 1F D1 FF call KeUnstackDetachProcess ;
PAGE:0000000140398CFC ; VOID KeUnstackDetachProcess(
PAGE:0000000140398CFC ; _In_ PRKAPC_STATE ApcState
PAGE:0000000140398CFC ; );
PAGE:0000000140398D01 E9 FF FF FF jmp loc_140398C76
PAGE:0000000140398D06 ; ---------------------------------------------------------------------------
PAGE:0000000140398D06
PAGE:0000000140398D06 RETURN_STATUS_INVALID_PARAMETER_2: ; CODE XREF: NtProtectVirtualMemory+C3j
PAGE:0000000140398D06 B8 F0 C0 mov eax, 0C00000F0h
PAGE:0000000140398D0B EB jmp short loc_140398CA2
PAGE:0000000140398D0B ; ---------------------------------------------------------------------------
PAGE:0000000140398D0D + align 20h
PAGE:0000000140398D0D +NtProtectVirtualMemory endp
这里也是根据KernelMode和UserMode的不同而选择性的验证BaseAddress,如果是KernelMode这个地方就没有进行ProbeForWrite验证,直接跳过验证步骤。
0x08 总结
1.Zw函数会在KiSystemService中将ETHREAD中的PreviousMode改为KernelMode,最后在Nt函数中如果是KernelMode就会跳过对参数是否可写的验证,如果是UserMode就会验证。如果是UserMode,访问内核地址会报错,所以如果内核中直接调用Nt函数,需要手动将PreviousMode修改为KernelMode否则无法访问内核地址,而修改PreviousMode并且通过系统服务表获取SSDT函数这个过程是很复杂的,直接调用内核导出的Zw函数就行,不过在调用Zw函数的时候需要自己对地址的可写性验证。而且通过Zw函数调用会在系统空间堆栈上有个属于本次调用的自陷框架。
2.32位下Zw函数会将内核模式保存在CS最后一位上,调用KiSystemService修改PreviousMode为KernelMode,接着跳转到KiFastCallEntry中间的地方,初始化一些寄存器,最后通过call ebx的方式调用Nt函数,最后通过KiSystemExit返回。
64位下Zw函数会调用KiServiceInternal,在这个函数中修改PerviousMode为KernelMode,然后跳转到KiSystemCall64中的KiSystemServiceStart部分,接着在KiSystemServiceRepeat部分通过jmp r11调用Nt函数,最后通过KiSystemServiceExit函数返回。
32位和64位系统内核函数调用从ZwProtectVirtualMemory到NtProtectVirtualMemory的更多相关文章
- 查看Unix系统是32位还是64位
#getconf查看OS位数 以下经过测试了HP: getconf KERNEL_BITSLinux: getconf LONG_BITAIX: getconf KERNEL_BITMODE #AIX ...
- 64位主机64位oracle下装32位客户端ODAC(NFPACS版)
64位主机64位oracle下装32位客户端ODAC(NFPACS版) by dd 1.下载Oracle Data Access Components(ODAC) Xcopy的两个版本: x86:(我 ...
- Linux系统查看系统是32位还是64位方法总结
这篇博客是总结.归纳查看Linux系统是32位还是64位的一些方法,很多内容来自网上网友的博客.本篇只是整理.梳理这方面的知识,方便自己忘记的时候随时查看. 方法1:getconf LONG_BIT ...
- oracle 32位导64位
oracle 32位导64位 SHUTDOWN IMMEDIATE; STARTUP MOUNT; ALTER SYSTEM ENABLE RESTRICTED SESSION; ; ; ALTER ...
- Shell脚本中,如何判断Linux系统是32位还是64位?
一行就能搞定,输出32或者64 可以用“和. 参考代码如下: ldconfig if [ $(getconf WORD_BIT) = '32' ] && [ $(getconf LON ...
- 【转】Tomcat版本是32位、64位问题
转载地址:http://www.cnblogs.com/greensleeves/p/3168541.html 最近遇到一个Tomcat windows安装版本是32位还是64位问题.由于一系列原因, ...
- 查看Linux是32位还是64位
最直接简洁的办法: 在linux终端输入getconf LONG_BIT命令 如果是32位机器,则结果为32 [root@localhost ~]# getconf LONG_BIT 32 如果是64 ...
- Windows2003 IIS6.0支持32位和64位两种模式的设置方法
IIS 6.0 可支持 32 位和 64 位两种模式.但是,IIS 6.0 不支持在 64 位版本的 Windows 上同时运行这两种模式.ASP.NET 1.1 只在 32 位模式下运行.而 ASP ...
- linux-查看系统是32位还是64位
可以用命令“getconf LONG_BIT”查看, 如果返回的结果是32则说明是32位,返回的结果是64则说明是64位. 此外还可以使用命令“uname -a”查看, 输出的结果中,如果有x86_6 ...
随机推荐
- vmware获取主机、数据中心等对象ManagedObjectReference
在vmware的api中提供以下列表中的对象,称作ManagedObjectReference,包括虚拟机信息.主机.数据中心等等一些信息,我们可以通过vcenter的web api得到. 下面我们来 ...
- solr的schema.xml配置属性解释
schema.xml做什么? SOLR加载数据,创建索引和数据时,核心数据结构的配置文件是schema.xml,该配置文件主要用于配置数据源,字段类型定义,搜索类型定义等.schema.xml的配置直 ...
- C#和JAVA 访问修饰符
JAVA ----------------------------------------------- 访问修饰符 描述 ------------------------------- ...
- CentOS下Docker与.netcore(一) 之 安装
CentOS下Docker与.netcore(一) 之 安装 CentOS下Docker与.netcore(二) 之 Dockerfile CentOS下Docker与.netcore(三)之 三剑客 ...
- ajax 多个setInterval进行ajax请求的页面长时间打开会出现页面卡死问题
多个setInterval进行ajax请求的页面长时间打开会出现页面卡死问题 浏览器的渲染(UI)线程和js线程是互斥的,在执行js耗时操作时,页面渲染会被阻塞掉.当我们执行异步ajax的时候没有问 ...
- 《Beginning Java 7》 - 5 - Hash Codes 哈希码
哈希码和 equals() 都是用来比较的. 1. 哈希码的作用是用来提高比较的效率.因为当比较的对象比较复杂时,equals() 可能很耗时,但哈希码只需要比较一个 int .哈希码常用于集 (se ...
- [BZOJ5212][ZJOI2018]历史
传送门(洛谷) 人生第一道九条可怜……神仙操作…… 看着题解理解了一个早上才勉强看懂怎么回事…… 简化一下题目就是:已知每一个点access的总次数,求一个顺序使虚实边转化的次数最多 考虑一下,对于x ...
- Logstash IIS日志采集
Logstash IIS 日志采集,跟Linux上运行差不多,都需要java运行环境,装个jdk就好了,对于IIS日志暂时未处理X-forward-for,纠结怎么弄当中,貌似要装个插件,慢慢研究. ...
- [Swift]扩展String类:extension String
请参考本博客另一篇技术博文: <[Swift]字符串(String类.NSString类)常用操作> extension String { //获取字符串首字符 var first: St ...
- 动态sql语句和动态传入参数个数
1.可以将要传入的几个参数封装成一个实体类,然后将实体类作为一个参数传入到相应的方法中,这时候就需要这sqlMapper.xml文件中对传入的字段利用<if test=""& ...