SUPR3HardenedMainA-win.asm revision 4b54bc4334bafab3a2fb62746e675703dcb19f00
; $Id$
;; @file
; VirtualBox Support Library - Hardened main(), Windows assembly bits.
;
;
; Copyright (C) 2012-2014 Oracle Corporation
;
; This file is part of VirtualBox Open Source Edition (OSE), as
; available from http://www.virtualbox.org. This file is free software;
; you can redistribute it and/or modify it under the terms of the GNU
; General Public License (GPL) as published by the Free Software
; Foundation, in version 2 as it comes in the "COPYING" file of the
; VirtualBox OSE distribution. VirtualBox OSE is distributed in the
; hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
;
; The contents of this file may alternatively be used under the terms
; of the Common Development and Distribution License Version 1.0
; (CDDL) only, as it comes in the "COPYING.CDDL" file of the
; VirtualBox OSE distribution, in which case the provisions of the
; CDDL are applicable instead of those of the GPL.
;
; You may elect to license modified versions of this file under the
; terms and conditions of either the GPL or the CDDL or both.
;
;*******************************************************************************
;* Header Files *
;*******************************************************************************
%define RT_ASM_WITH_SEH64
%include "iprt/asmdefs.mac"
; External data.
extern NAME(g_pfnNtCreateSectionJmpBack)
; External code.
extern NAME(supR3HardenedEarlyProcessInit)
BEGINCODE
;
; 64-bit
;
%ifdef RT_ARCH_AMD64
%macro supR3HardenedJmpBack_NtCreateSection_Xxx 1
BEGINPROC supR3HardenedJmpBack_NtCreateSection_ %+ %1
SEH64_END_PROLOGUE
; The code we replaced.
mov r10, rcx
mov eax, %1
; Jump back to the original code.
jmp [NAME(g_pfnNtCreateSectionJmpBack) wrt RIP]
ENDPROC supR3HardenedJmpBack_NtCreateSection_ %+ %1
%endm
%define SYSCALL(a_Num) supR3HardenedJmpBack_NtCreateSection_Xxx a_Num
%endif
;
; 32-bit.
;
%ifdef RT_ARCH_X86
%macro supR3HardenedJmpBack_NtCreateSection_Xxx 1
BEGINPROC supR3HardenedJmpBack_NtCreateSection_ %+ %1
; The code we replaced.
mov eax, %1
; Jump back to the original code.
jmp [NAME(g_pfnNtCreateSectionJmpBack)]
ENDPROC supR3HardenedJmpBack_NtCreateSection_ %+ %1
%endm
%define SYSCALL(a_Num) supR3HardenedJmpBack_NtCreateSection_Xxx a_Num
%endif
;;
; Alternative code for LdrInitializeThunk that performs the early process startup
; for the Stub and VM processes.
;
; This does not concern itself with any arguments on stack or in registers that
; may be passed to the LdrIntializeThunk routine as we just save and restore
; them all before we restart the restored LdrInitializeThunk routine.
;
; @sa supR3HardenedEarlyProcessInit
;
BEGINPROC supR3HardenedEarlyProcessInitThunk
;
; Prologue.
;
; Reserve space for the "return" address.
push 0
; Create a stack frame, saving xBP.
push xBP
SEH64_PUSH_xBP
mov xBP, xSP
SEH64_SET_FRAME_xBP 0 ; probably wrong...
; Save all volatile registers.
push xAX
push xCX
push xDX
%ifdef RT_ARCH_AMD64
push r8
push r9
push r10
push r11
%endif
; Reserve spill space and align the stack.
sub xSP, 20h
and xSP, ~0fh
SEH64_END_PROLOGUE
;
; Call the C/C++ code that does the actual work. This returns the
; resume address in xAX, which we put in the "return" stack position.
;
call NAME(supR3HardenedEarlyProcessInit)
mov [xBP + xCB], xAX
;
; Restore volatile registers.
;
mov xAX, [xBP - xCB*1]
mov xCX, [xBP - xCB*2]
mov xDX, [xBP - xCB*3]
%ifdef RT_ARCH_AMD64
mov r8, [xBP - xCB*4]
mov r9, [xBP - xCB*5]
mov r10, [xBP - xCB*6]
mov r11, [xBP - xCB*7]
%endif
;
; Use the leave instruction to restore xBP and set up xSP to point at
; the resume address. Then use the 'ret' instruction to resume process
; initializaton.
;
leave
ret
ENDPROC supR3HardenedEarlyProcessInitThunk
;;
; Composes a standard call name.
%ifdef RT_ARCH_X86
%define SUPHNTIMP_STDCALL_NAME(a,b) _ %+ a %+ @ %+ b
%else
%define SUPHNTIMP_STDCALL_NAME(a,b) NAME(a)
%endif
;; Concats two litterals.
%define SUPHNTIMP_CONCAT(a,b) a %+ b
;;
; Import data and code for an API call.
;
; @param 1 The plain API name.
; @param 2 The parameter frame size on x86. Multiple of dword.
; @param 3 Non-zero expression if system call.
; @param 4 Non-zero expression if early available call
;
%define SUPHNTIMP_SYSCALL 1
%macro SupHardNtImport 4
;
; The data.
;
BEGINDATA
global __imp_ %+ SUPHNTIMP_STDCALL_NAME(%1,%2) ; The import name used via dllimport.
__imp_ %+ SUPHNTIMP_STDCALL_NAME(%1,%2):
GLOBALNAME g_pfn %+ %1 ; The name we like to refer to.
RTCCPTR_DEF 0
%if %3
GLOBALNAME g_uApiNo %+ %1
RTCCPTR_DEF 0
%endif
;
; The code: First a call stub.
;
BEGINCODE
global SUPHNTIMP_STDCALL_NAME(%1, %2)
SUPHNTIMP_STDCALL_NAME(%1, %2):
jmp RTCCPTR_PRE [NAME(g_pfn %+ %1) xWrtRIP]
%if %3
;
; Make system calls.
;
%ifdef RT_ARCH_AMD64
BEGINPROC %1 %+ _SyscallType1
SEH64_END_PROLOGUE
mov eax, [NAME(g_uApiNo %+ %1) xWrtRIP]
mov r10, rcx
syscall
ret
ENDPROC %1 %+ _SyscallType1
%else
BEGINPROC %1 %+ _SyscallType1
mov edx, 07ffe0300h ; SharedUserData!SystemCallStub
mov eax, [NAME(g_uApiNo %+ %1) xWrtRIP]
call dword [edx]
ret %2
ENDPROC %1 %+ _SyscallType1
BEGINPROC %1 %+ _SyscallType2
push .return
mov edx, esp
mov eax, [NAME(g_uApiNo %+ %1) xWrtRIP]
sysenter
add esp, 4
.return:
ret %2
ENDPROC %1 %+ _SyscallType2
%endif
%endif
%if %4 == 0
global NAME(SUPHNTIMP_CONCAT(%1,_Early))
NAME(SUPHNTIMP_CONCAT(%1,_Early)):
int3
%ifdef RT_ARCH_AMD64
ret
%else
ret %2
%endif
%endif
%endmacro
%define SUPHARNT_COMMENT(a_Comment)
%define SUPHARNT_IMPORT_SYSCALL(a_Name, a_cbParamsX86) SupHardNtImport a_Name, a_cbParamsX86, SUPHNTIMP_SYSCALL, 1
%define SUPHARNT_IMPORT_STDCALL(a_Name, a_cbParamsX86) SupHardNtImport a_Name, a_cbParamsX86, 0, 0
%define SUPHARNT_IMPORT_STDCALL_EARLY(a_Name, a_cbParamsX86) SupHardNtImport a_Name, a_cbParamsX86, 0, 1
%define SUPHARNT_IMPORT_STDCALL_EARLY_OPTIONAL(a_Name, a_cbParamsX86) SUPHARNT_IMPORT_STDCALL_EARLY(a_Name, a_cbParamsX86)
%include "import-template-ntdll.h"
%include "import-template-kernel32.h"
;
; For simplified LdrLoadDll patching we define a special writable, readable and
; exectuable section of 4KB where we can put jump back code.
;
section .rwxpg bss execute read write align=4096
GLOBALNAME g_abSupHardReadWriteExecPage
resb 4096