2020-09-30 16:53:55 +02:00

473 lines
16 KiB
NASM

title "Capture and Restore Context"
;++
;
; Copyright (c) 2000 Microsoft Corporation
;
; Module Name:
;
; capture.asm
;
; Abstract:
;
; This module implements the platform specific code to capture and restore
; the context of the caller.
;
; Author:
;
; David N. Cutler (davec) 4-Jul-2000
;
; Environment:
;
; Any mode.
;
;--
include ksamd64.inc
altentry RcConsolidateFrames
subttl "Capture Context"
;++
;
; VOID
; RtlCaptureContext (
; IN PCONTEXT ContextRecord
; )
;
; Routine Description:
;
; This function captures the context of the caller in the specified
; context record.
;
; N.B. The stored value of registers rcx and rsp will be a side effect of
; having made this call. All other registers will be stored as they
; were when the call to this function was made.
;
; Arguments:
;
; ContextRecord (rcx) - Supplies a pointer to a context record.
;
; Return Value:
;
; None.
;
;--
CcFrame struct
EFlags dd ? ; saved procssor flags
Fill dd ? ; fill
CcFrame ends
NESTED_ENTRY RtlCaptureContext, _TEXT$00
push_eflags ; save processor flags
END_PROLOGUE
mov CxSegCs[rcx], cs ; save segment registers
mov CxSegDs[rcx], ds ;
mov CxSegEs[rcx], es ;
mov CxSegSs[rcx], ss ;
mov CxSegFs[rcx], fs ;
mov CxSegGs[rcx], gs ;
mov CxRax[rcx], rax ; save integer registers
mov CxRcx[rcx], rcx ;
mov CxRdx[rcx], rdx ;
mov CxRbx[rcx], rbx ;
lea rax, 16[rsp] ;
mov CxRsp[rcx], rax ;
mov CxRbp[rcx], rbp ;
mov CxRsi[rcx], rsi ;
mov CxRdi[rcx], rdi ;
mov CxR8[rcx], r8 ;
mov CxR9[rcx], r9 ;
mov CxR10[rcx], r10 ;
mov CxR11[rcx], r11 ;
mov CxR12[rcx], r12 ;
mov CxR13[rcx], r13 ;
mov CxR14[rcx], r14 ;
mov CxR15[rcx], r15 ;
movdqa CxXmm0[rcx], xmm0 ; save xmm floating registers
movdqa CxXmm1[rcx], xmm1 ;
movdqa CxXmm2[rcx], xmm2 ;
movdqa CxXmm3[rcx], xmm3 ;
movdqa CxXmm4[rcx], xmm4 ;
movdqa CxXmm5[rcx], xmm5 ;
movdqa CxXmm6[rcx], xmm6 ;
movdqa CxXmm7[rcx], xmm7 ;
movdqa CxXmm8[rcx], xmm8 ;
movdqa CxXmm9[rcx], xmm9 ;
movdqa CxXmm10[rcx], xmm10 ;
movdqa CxXmm11[rcx], xmm11 ;
movdqa CxXmm12[rcx], xmm12 ;
movdqa CxXmm13[rcx], xmm13 ;
movdqa CxXmm14[rcx], xmm14 ;
movdqa CxXmm15[rcx], xmm15 ;
stmxcsr CxMxCsr[rcx] ; save xmm floating state
ifndef NTOS_KERNEL_RUNTIME
fnsaved CxFltSave[rcx] ; save legacy floating state
endif
mov rax, 8[rsp] ; set return address
mov CxRip[rcx], rax ;
mov eax, Ccframe.EFlags[rsp] ; set processor flags
mov CxEFlags[rcx], eax ;
mov dword ptr CxContextFlags[rcx], CONTEXT_FULL or CONTEXT_SEGMENTS ; set context flags
add rsp, sizeof CcFrame ; deallocate stack frame
ret ; return
NESTED_END RtlCaptureContext, _TEXT$00
subttl "Restore Context"
;++
;
; VOID
; RtlRestoreContext (
; IN PCONTEXT ContextRecord,
; IN PEXCEPTION_RECORD ExceptionRecord OPTIONAL
; )
;
; Routine Description:
;
; This function restores the context of the caller to the specified
; context.
;
; Arguments:
;
; ContextRecord (rcx) - Supplies a pointer to a context record.
;
; ExceptionRecord (rdx) - Supplies an optional pointer to an exception
; record.
;
; Return Value:
;
; None - there is no return from this function.
;
;--
RcFrame struct
Mframe db MachineFrameLength dup (?) ; machine frame
Fill dq ? ; fill to 0 mod 16
RcFrame ends
NESTED_ENTRY RtlRestoreContext, _TEXT$00
push_reg rbp ; save nonvolatile registers
push_reg rsi ;
push_reg rdi ;
alloc_stack (sizeof RcFrame) ; allocate stack frame
set_frame rbp, 0 ; set frame pointer
END_PROLOGUE
;
; If an exception record is specified and the exception status is the unwind
; consolidation code and there is at least one parameter, then consolidate
; all the frames that have been unwound and call back to a language specified
; routine.
;
test rdx, rdx ; test if exception record specified
jz Rc10 ; if z, no exception record specified
cmp dword ptr ErExceptionCode[rdx], STATUS_UNWIND_CONSOLIDATE ; check call back
jne short Rc05 ; if ne, not C++ unwind
cmp dword ptr ErNumberParameters[rdx], 1 ; check number parameters
jae Rc20 ; if ae, unwind consolidation
;
; If an exception record is specified and the exception status is long jump,
; then restore the nonvolatile registers to their state at the call to set
; jump before restoring the context record.
;
Rc05: cmp dword ptr ErExceptionCode[rdx], STATUS_LONGJUMP ; check for long jump
jne Rc10 ; if ne, not a long jump
;
; Long jump unwind.
;
; Copy register values from the jump buffer to the context record.
;
mov rax, ErExceptionInformation[rdx] ; get jump buffer address
mov r8, JbRbx[rax] ; move nonvolatile integer registers
mov CxRbx[rcx], r8 ; to context record
mov r8, JbRsp[rax] ;
mov CxRsp[rcx], r8 ;
mov r8, JbRbp[rax] ;
mov CxRbp[rcx], r8 ;
mov r8, JbRsi[rax] ;
mov CxRsi[rcx], r8 ;
mov r8, JbRdi[rax] ;
mov CxRdi[rcx], r8 ;
mov r8, JbR12[rax] ;
mov CxR12[rcx], r8 ;
mov r8, JbR13[rax] ;
mov CxR13[rcx], r8 ;
mov r8, JbR14[rax] ;
mov CxR14[rcx], r8 ;
mov r8, JbR15[rax] ;
mov CxR15[rcx], r8 ;
mov r8, JbRip[rax] ;
mov CxRip[rcx], r8 ;
movdqa xmm0, JbXmm6[rax] ; move nonvolatile floating register
movdqa CxXmm6[rcx], xmm0 ; to context record
movdqa xmm0, JbXmm7[rax] ;
movdqa CxXmm7[rcx], xmm0 ;
movdqa xmm0, JbXmm8[rax] ;
movdqa CxXmm8[rcx], xmm0 ;
movdqa xmm0, JbXmm9[rax] ;
movdqa CxXmm9[rcx], xmm0 ;
movdqa xmm0, JbXmm10[rax] ;
movdqa CxXmm10[rcx], xmm0 ;
movdqa xmm0, JbXmm11[rax] ;
movdqa CxXmm11[rcx], xmm0 ;
movdqa xmm0, JbXmm12[rax] ;
movdqa CxXmm12[rcx], xmm0 ;
movdqa xmm0, JbXmm13[rax] ;
movdqa CxXmm13[rcx], xmm0 ;
movdqa xmm0, JbXmm14[rax] ;
movdqa CxXmm14[rcx], xmm0 ;
movdqa xmm0, JbXmm15[rax] ;
movdqa CxXmm15[rcx], xmm0 ;
;
; Restore context and continue.
;
Rc10: movdqa xmm0, CxXmm0[rcx] ; restore floating registers
movdqa xmm1, CxXmm1[rcx] ;
movdqa xmm2, CxXmm2[rcx] ;
movdqa xmm3, CxXmm3[rcx] ;
movdqa xmm4, CxXmm4[rcx] ;
movdqa xmm5, CxXmm5[rcx] ;
movdqa xmm6, CxXmm6[rcx] ;
movdqa xmm7, CxXmm7[rcx] ;
movdqa xmm8, CxXmm8[rcx] ;
movdqa xmm9, CxXmm9[rcx] ;
movdqa xmm10, CxXmm10[rcx] ;
movdqa xmm11, CxXmm11[rcx] ;
movdqa xmm12, CxXmm12[rcx] ;
movdqa xmm13, CxXmm13[rcx] ;
movdqa xmm14, CxXmm14[rcx] ;
movdqa xmm15, CxXmm15[rcx] ;
ldmxcsr CxMxCsr[rcx] ; restore floating state
mov ax, CxSegSs[rcx] ; set SS segment
mov MfSegSs[rsp], ax ;
mov rax, CxRsp[rcx] ; set stack address
mov MfRsp[rsp], rax ;
mov eax, CxEFlags[rcx] ; set processor flags
mov MfEFlags[rsp], eax ;
mov ax, CxSegCs[rcx] ; set CS segment
mov MfSegCs[rsp], ax ;
mov rax, CxRip[rcx] ; set return address
mov MfRip[rsp], rax ;
mov rax, CxRax[rcx] ; restore volatile integer registers
mov rdx, CxRdx[rcx] ;
mov r8, CxR8[rcx] ;
mov r9, CxR9[rcx] ;
mov r10, CxR10[rcx] ;
mov r11, CxR11[rcx] ;
ifdef NTOS_KERNEL_RUNTIME
cli ; disable interrupts
endif
mov rbx, CxRbx[rcx] ; restore nonvolatile integer registers
mov rsi, CxRsi[rcx] ;
mov rdi, CxRdi[rcx] ;
mov rbp, CxRbp[rcx] ;
mov r12, CxR12[rcx] ;
mov r13, CxR13[rcx] ;
mov r14, CxR14[rcx] ;
mov r15, CxR15[rcx] ;
mov rcx, CxRcx[rcx] ; restore integer register
iretq ; return
;
; Frame consoldation and language specific unwind call back.
;
Rc20: sub rsp, MachineFrameLength + 8; allocate machine frame
mov r8, rsp ; save machine frame address
sub rsp, CONTEXT_FRAME_LENGTH ; allocate context frame
mov rsi, rcx ; set source copy address
mov rdi, rsp ; set destination copy address
mov ecx, CONTEXT_FRAME_LENGTH / 8 ; set length of copy
rep movsq ; copy context frame
mov rax, CxRsp[rsp] ; set destination stack address in
mov MfRsp[r8], rax ; machine frame
mov rax, CxRip[rsp] ; set destination address in machine
mov MfRip[r8], rax ; frame
mov rcx, rdx ; set address of exception record
jmp RcConsolidateFrames ; consolidate frames - no return
NESTED_END RtlRestoreContext, _TEXT$00
subttl "Frame Consolidation"
;++
;
; Ths following code is never executed. Its purpose is to provide the dummy
; prologue necessary to consolidate stack frames for unwind call back processing
; at the end of an unwind operation.
;
;--
NESTED_ENTRY RcFrameConsolidation, _TEXT$00
.pushframe ;
.allocstack CONTEXT_FRAME_LENGTH ; allocate stack frame
.savereg rbx, CxRbx ; save nonvolatile integer registers
.savereg rbp, CxRbp ;
.savereg rsi, CxRsi ;
.savereg rdi, CxRdi ;
.savereg r12, CxR12 ;
.savereg r13, CxR13 ;
.savereg r14, CxR14 ;
.savereg r15, CxR15 ;
.savexmm128 xmm6, CxXmm6 ; save nonvolatile floating register
.savexmm128 xmm7, CxXmm7 ;
.savexmm128 xmm8, CxXmm8 ;
.savexmm128 xmm9, CxXmm9 ;
.savexmm128 xmm10, CxXmm10 ;
.savexmm128 xmm11, CxXmm11 ;
.savexmm128 xmm12, CxXmm12 ;
.savexmm128 xmm13, CxXmm13 ;
.savexmm128 xmm14, CxXmm14 ;
.savexmm128 xmm15, CxXmm15 ;
END_PROLOGUE
;++
;
; VOID
; RcConsolidateFrames (
; IN PEXCEPTION_RECORD ExceptionRecord
; )
;
; Routine Description:
;
; This routine is called at the end of a unwind operation to logically
; remove unwound frames from the stack. This is accomplished by building a
; call frame using a machine frame and a context record and then calling
; the alternate entry of this function.
;
; The following code calls the language call back function specified in the
; exception record. If the function returns, then the destination frame
; context is restored and control transfered to the address returned by the
; language call back function. If control does not return, then another
; exception must be raised.
;
; Arguments:
;
; ExceptionRecord (rdx) - Supplies a pointer to an exception record.
;
; Implicit Arguments:
;
; ContextRecord (rsp) - Supplies a pointer to a context record.
;
; Return Value:
;
; None.
;
;--
ALTERNATE_ENTRY RcConsolidateFrames
;
; At this point all call frames from the dispatching of the an exception to
; a destination language specific handler have been logically unwound and
; consolidated into a single large frame.
;
; The first parameter in the exception record is the address of a callback
; routine that performs language specific operations. This routine is called
; with the specified exception record as a parameter.
;
call qword ptr ErExceptionInformation[rcx] ; call back to handler
;
; Restore context and continue.
;
mov rcx, rsp ; set address of context record
mov CxRip[rcx], rax ; set destination address
movdqa xmm0, CxXmm0[rcx] ; restore floating registers
movdqa xmm1, CxXmm1[rcx] ;
movdqa xmm2, CxXmm2[rcx] ;
movdqa xmm3, CxXmm3[rcx] ;
movdqa xmm4, CxXmm4[rcx] ;
movdqa xmm5, CxXmm5[rcx] ;
movdqa xmm6, CxXmm6[rcx] ;
movdqa xmm7, CxXmm7[rcx] ;
movdqa xmm8, CxXmm8[rcx] ;
movdqa xmm9, CxXmm9[rcx] ;
movdqa xmm10, CxXmm10[rcx] ;
movdqa xmm11, CxXmm11[rcx] ;
movdqa xmm12, CxXmm12[rcx] ;
movdqa xmm13, CxXmm13[rcx] ;
movdqa xmm14, CxXmm14[rcx] ;
movdqa xmm15, CxXmm15[rcx] ;
ldmxcsr CxMxCsr[rcx] ; restore floating state
;
; Contruct a machine frame of the stack using information from the context
; record.
;
; N.B. The machine frame overlays the parameter area in the context record.
;
mov ax, CxSegSs[rcx] ; set SS segment
mov MfSegSs[rsp], ax ;
mov rax, CxRsp[rcx] ; set stack address
mov MfRsp[rsp], rax ;
mov eax, CxEFlags[rcx] ; set processor flags
mov MfEFlags[rsp], eax ;
mov ax, CxSegCs[rcx] ; set CS segment
mov MfSegCs[rsp], ax ;
mov rax, CxRip[rcx] ; set return address
mov MfRip[rsp], rax ;
mov rax, CxRax[rcx] ; restore volatile integer registers
mov rdx, CxRdx[rcx] ;
mov r8, CxR8[rcx] ;
mov r9, CxR9[rcx] ;
mov r10, CxR10[rcx] ;
mov r11, CxR11[rcx] ;
ifdef NTOS_KERNEL_RUNTIME
cli ; disable interrupts
endif
mov rbx, CxRbx[rcx] ; restore nonvolatile integer registers
mov rsi, CxRsi[rcx] ;
mov rdi, CxRdi[rcx] ;
mov rbp, CxRbp[rcx] ;
mov r12, CxR12[rcx] ;
mov r13, CxR13[rcx] ;
mov r14, CxR14[rcx] ;
mov r15, CxR15[rcx] ;
mov rcx, CxRcx[rcx] ; restore integer register
iretq ; return
NESTED_END RcFrameConsolidation, _TEXT$00
end