/** * ============================================================================= * 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 // ============================================================================ // >> x86MsThiscall // ============================================================================ x86MsThiscall::x86MsThiscall(std::vector &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 x86MsThiscall::GetRegisters() { std::vector 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() + 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() + 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 pSavedCallArguments = std::make_unique(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(); }