240 lines
6.6 KiB
C++
240 lines
6.6 KiB
C++
|
/**
|
||
|
* =============================================================================
|
||
|
* DynamicHooks
|
||
|
* Copyright (C) 2015 Robin Gohmert. All rights reserved.
|
||
|
* Copyright (C) 2018-2021 AlliedModders LLC. 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).
|
||
|
*
|
||
|
* Adopted to provide similar features to SourceHook by AlliedModders LLC.
|
||
|
*/
|
||
|
|
||
|
// ============================================================================
|
||
|
// >> INCLUDES
|
||
|
// ============================================================================
|
||
|
#include "x86MsThiscall.h"
|
||
|
#include <string.h>
|
||
|
|
||
|
|
||
|
// ============================================================================
|
||
|
// >> x86MsThiscall
|
||
|
// ============================================================================
|
||
|
x86MsThiscall::x86MsThiscall(std::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;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
x86MsThiscall::~x86MsThiscall()
|
||
|
{
|
||
|
if (m_pReturnBuffer)
|
||
|
{
|
||
|
free(m_pReturnBuffer);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
std::vector<Register_t> x86MsThiscall::GetRegisters()
|
||
|
{
|
||
|
std::vector<Register_t> registers;
|
||
|
|
||
|
registers.push_back(ESP);
|
||
|
// TODO: Allow custom this register.
|
||
|
registers.push_back(ECX);
|
||
|
|
||
|
if (m_returnType.type == DATA_TYPE_FLOAT || m_returnType.type == DATA_TYPE_DOUBLE)
|
||
|
{
|
||
|
registers.push_back(ST0);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
registers.push_back(EAX);
|
||
|
if (m_pReturnBuffer)
|
||
|
{
|
||
|
registers.push_back(EDX);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Save all the custom calling convention registers as well.
|
||
|
for (size_t i = 0; i < m_vecArgTypes.size(); i++)
|
||
|
{
|
||
|
if (m_vecArgTypes[i].custom_register == None)
|
||
|
continue;
|
||
|
|
||
|
// TODO: Make sure the list is unique? Set?
|
||
|
registers.push_back(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(size_t i=0; i < m_vecArgTypes.size(); 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 (size_t i = 0; i < m_vecArgTypes.size(); 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 (size_t i = 0; i < m_vecArgTypes.size(); i++)
|
||
|
{
|
||
|
if (m_vecArgTypes[i].custom_register != None)
|
||
|
iArgRegisterSize += m_vecArgTypes[i].size;
|
||
|
}
|
||
|
|
||
|
return iArgRegisterSize;
|
||
|
}
|
||
|
|
||
|
void* x86MsThiscall::GetArgumentPtr(unsigned 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 >= m_vecArgTypes.size())
|
||
|
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;
|
||
|
}
|
||
|
|
||
|
size_t iOffset = 4;
|
||
|
for(unsigned 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(unsigned 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::SaveCallArguments(CRegisters* pRegisters)
|
||
|
{
|
||
|
// Account for implicit this-pointer in ecx.
|
||
|
int size = GetArgStackSize() + GetArgRegisterSize() + sizeof(void *);
|
||
|
std::unique_ptr<uint8_t[]> pSavedCallArguments = std::make_unique<uint8_t[]>(size);
|
||
|
memcpy(pSavedCallArguments.get(), GetArgumentPtr(0, pRegisters), sizeof(void *));
|
||
|
|
||
|
size_t offset = sizeof(void *);
|
||
|
for (size_t i = 0; i < m_vecArgTypes.size(); i++) {
|
||
|
DataTypeSized_t &type = m_vecArgTypes[i];
|
||
|
memcpy((void *)((unsigned long)pSavedCallArguments.get() + offset), GetArgumentPtr(i + 1, pRegisters), type.size);
|
||
|
offset += type.size;
|
||
|
}
|
||
|
m_pSavedCallArguments.push_back(std::move(pSavedCallArguments));
|
||
|
}
|
||
|
|
||
|
void x86MsThiscall::RestoreCallArguments(CRegisters* pRegisters)
|
||
|
{
|
||
|
uint8_t* pSavedCallArguments = m_pSavedCallArguments.back().get();
|
||
|
memcpy(GetArgumentPtr(0, pRegisters), pSavedCallArguments, sizeof(void *));
|
||
|
|
||
|
size_t offset = sizeof(void *);
|
||
|
for (size_t i = 0; i < m_vecArgTypes.size(); i++) {
|
||
|
DataTypeSized_t &type = m_vecArgTypes[i];
|
||
|
memcpy(GetArgumentPtr(i + 1, pRegisters), (void *)((unsigned long)pSavedCallArguments + offset), type.size);
|
||
|
offset += type.size;
|
||
|
}
|
||
|
m_pSavedCallArguments.pop_back();
|
||
|
}
|