Just return null if a user tries to access an invalid argument that wasn't defined when detouring the function.
		
			
				
	
	
		
			218 lines
		
	
	
		
			5.9 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			218 lines
		
	
	
		
			5.9 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /**
 | |
| * =============================================================================
 | |
| * DynamicHooks
 | |
| * Copyright (C) 2015 Robin Gohmert. All rights reserved.
 | |
| * =============================================================================
 | |
| *
 | |
| * This software is provided 'as-is', without any express or implied warranty.
 | |
| * In no event will the authors be held liable for any damages arising from 
 | |
| * the use of this software.
 | |
| * 
 | |
| * Permission is granted to anyone to use this software for any purpose, 
 | |
| * including commercial applications, and to alter it and redistribute it 
 | |
| * freely, subject to the following restrictions:
 | |
| *
 | |
| * 1. The origin of this software must not be misrepresented; you must not 
 | |
| * claim that you wrote the original software. If you use this software in a 
 | |
| * product, an acknowledgment in the product documentation would be 
 | |
| * appreciated but is not required.
 | |
| *
 | |
| * 2. Altered source versions must be plainly marked as such, and must not be
 | |
| * misrepresented as being the original software.
 | |
| *
 | |
| * 3. This notice may not be removed or altered from any source distribution.
 | |
| *
 | |
| * asm.h/cpp from devmaster.net (thanks cybermind) edited by pRED* to handle gcc
 | |
| * -fPIC thunks correctly
 | |
| *
 | |
| * Idea and trampoline code taken from DynDetours (thanks your-name-here).
 | |
| */
 | |
| 
 | |
| // ============================================================================
 | |
| // >> INCLUDES
 | |
| // ============================================================================
 | |
| #include "x86MsThiscall.h"
 | |
| #include <string.h>
 | |
| 
 | |
| 
 | |
| // ============================================================================
 | |
| // >> x86MsThiscall
 | |
| // ============================================================================
 | |
| x86MsThiscall::x86MsThiscall(ke::Vector<DataTypeSized_t> &vecArgTypes, DataTypeSized_t returnType, int iAlignment) :
 | |
| 	ICallingConvention(vecArgTypes, returnType, iAlignment)
 | |
| {
 | |
| 	if (m_returnType.size > 4)
 | |
| 	{
 | |
| 		m_pReturnBuffer = malloc(m_returnType.size);
 | |
| 	}
 | |
| 	else
 | |
| 	{
 | |
| 		m_pReturnBuffer = NULL;
 | |
| 	}
 | |
| 	m_pSavedThisPointer = malloc(sizeof(size_t));
 | |
| }
 | |
| 
 | |
| x86MsThiscall::~x86MsThiscall()
 | |
| {
 | |
| 	if (m_pReturnBuffer)
 | |
| 	{
 | |
| 		free(m_pReturnBuffer);
 | |
| 	}
 | |
| 	free(m_pSavedThisPointer);
 | |
| }
 | |
| 
 | |
| ke::Vector<Register_t> x86MsThiscall::GetRegisters()
 | |
| {
 | |
| 	ke::Vector<Register_t> registers;
 | |
| 	
 | |
| 	registers.append(ESP);
 | |
| 	// TODO: Allow custom this register.
 | |
| 	registers.append(ECX);
 | |
| 
 | |
| 	if (m_returnType.type == DATA_TYPE_FLOAT || m_returnType.type == DATA_TYPE_DOUBLE)
 | |
| 	{
 | |
| 		registers.append(ST0);
 | |
| 	}
 | |
| 	else
 | |
| 	{
 | |
| 		registers.append(EAX);
 | |
| 		if (m_pReturnBuffer)
 | |
| 		{
 | |
| 			registers.append(EDX);
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	// Save all the custom calling convention registers as well.
 | |
| 	for (unsigned int i = 0; i < m_vecArgTypes.length(); i++)
 | |
| 	{
 | |
| 		if (m_vecArgTypes[i].custom_register == None)
 | |
| 			continue;
 | |
| 
 | |
| 		// TODO: Make sure the list is unique? Set?
 | |
| 		registers.append(m_vecArgTypes[i].custom_register);
 | |
| 	}
 | |
| 
 | |
| 	return registers;
 | |
| }
 | |
| 
 | |
| int x86MsThiscall::GetPopSize()
 | |
| {
 | |
| 	// This pointer.
 | |
| 	// FIXME LINUX
 | |
| 	//int iPopSize = GetDataTypeSize(DATA_TYPE_POINTER, m_iAlignment);
 | |
| 	int iPopSize = 0;
 | |
| 
 | |
| 	for(unsigned int i=0; i < m_vecArgTypes.length(); i++)
 | |
| 	{
 | |
| 		// Only pop arguments that are actually on the stack.
 | |
| 		if (m_vecArgTypes[i].custom_register == None)
 | |
| 			iPopSize += m_vecArgTypes[i].size;
 | |
| 	}
 | |
| 
 | |
| 	return iPopSize;
 | |
| }
 | |
| 
 | |
| int x86MsThiscall::GetArgStackSize()
 | |
| {
 | |
| 	int iArgStackSize = 0;
 | |
| 
 | |
| 	for (unsigned int i = 0; i < m_vecArgTypes.length(); i++)
 | |
| 	{
 | |
| 		if (m_vecArgTypes[i].custom_register == None)
 | |
| 			iArgStackSize += m_vecArgTypes[i].size;
 | |
| 	}
 | |
| 
 | |
| 	return iArgStackSize;
 | |
| }
 | |
| 
 | |
| void** x86MsThiscall::GetStackArgumentPtr(CRegisters* pRegisters)
 | |
| {
 | |
| 	return (void **)(pRegisters->m_esp->GetValue<unsigned long>() + 4);
 | |
| }
 | |
| 
 | |
| int x86MsThiscall::GetArgRegisterSize()
 | |
| {
 | |
| 	int iArgRegisterSize = 0;
 | |
| 
 | |
| 	for (unsigned int i = 0; i < m_vecArgTypes.length(); i++)
 | |
| 	{
 | |
| 		if (m_vecArgTypes[i].custom_register != None)
 | |
| 			iArgRegisterSize += m_vecArgTypes[i].size;
 | |
| 	}
 | |
| 
 | |
| 	return iArgRegisterSize;
 | |
| }
 | |
| 
 | |
| void* x86MsThiscall::GetArgumentPtr(int iIndex, CRegisters* pRegisters)
 | |
| {
 | |
| 	if (iIndex == 0)
 | |
| 	{
 | |
| 		// TODO: Allow custom this register.
 | |
| 		return pRegisters->m_ecx->m_pAddress;
 | |
| 	}
 | |
| 	
 | |
| 	// The this pointer isn't explicitly defined as an argument.
 | |
| 	iIndex--;
 | |
| 	
 | |
| 	if (iIndex < 0 || iIndex >= m_vecArgTypes.length())
 | |
| 		return NULL;
 | |
| 
 | |
| 	// Check if this argument was passed in a register.
 | |
| 	if (m_vecArgTypes[iIndex].custom_register != None)
 | |
| 	{
 | |
| 		CRegister *pRegister = pRegisters->GetRegister(m_vecArgTypes[iIndex].custom_register);
 | |
| 		if (!pRegister)
 | |
| 			return NULL;
 | |
| 
 | |
| 		return pRegister->m_pAddress;
 | |
| 	}
 | |
| 
 | |
| 	int iOffset = 4;
 | |
| 	for(int i=0; i < iIndex; i++)
 | |
| 	{
 | |
| 		if (m_vecArgTypes[i].custom_register == None)
 | |
| 			iOffset += m_vecArgTypes[i].size;
 | |
| 	}
 | |
| 
 | |
| 	return (void *) (pRegisters->m_esp->GetValue<unsigned long>() + iOffset);
 | |
| }
 | |
| 
 | |
| void x86MsThiscall::ArgumentPtrChanged(int iIndex, CRegisters* pRegisters, void* pArgumentPtr)
 | |
| {
 | |
| }
 | |
| 
 | |
| void* x86MsThiscall::GetReturnPtr(CRegisters* pRegisters)
 | |
| {
 | |
| 	if (m_returnType.type == DATA_TYPE_FLOAT || m_returnType.type == DATA_TYPE_DOUBLE)
 | |
| 		return pRegisters->m_st0->m_pAddress;
 | |
| 
 | |
| 	if (m_pReturnBuffer)
 | |
| 	{
 | |
| 		// First half in eax, second half in edx
 | |
| 		memcpy(m_pReturnBuffer, pRegisters->m_eax, 4);
 | |
| 		memcpy((void *) ((unsigned long) m_pReturnBuffer + 4), pRegisters->m_edx, 4);
 | |
| 		return m_pReturnBuffer;
 | |
| 	}
 | |
| 
 | |
| 	return pRegisters->m_eax->m_pAddress;
 | |
| }
 | |
| 
 | |
| void x86MsThiscall::ReturnPtrChanged(CRegisters* pRegisters, void* pReturnPtr)
 | |
| {
 | |
| 	if (m_pReturnBuffer)
 | |
| 	{
 | |
| 		// First half in eax, second half in edx
 | |
| 		memcpy(pRegisters->m_eax, m_pReturnBuffer, 4);
 | |
| 		memcpy(pRegisters->m_edx, (void *) ((unsigned long) m_pReturnBuffer + 4), 4);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void x86MsThiscall::SavePostCallRegisters(CRegisters* pRegisters)
 | |
| {
 | |
| 	memcpy(m_pSavedThisPointer, GetArgumentPtr(0, pRegisters), sizeof(size_t));
 | |
| }
 | |
| 
 | |
| void x86MsThiscall::RestorePostCallRegisters(CRegisters* pRegisters)
 | |
| {
 | |
| 	memcpy(GetArgumentPtr(0, pRegisters), m_pSavedThisPointer, sizeof(size_t));
 | |
| } |