88 lines
		
	
	
		
			3.2 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			88 lines
		
	
	
		
			3.2 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 "x86GccThiscall.h"
 | |
| 
 | |
| 
 | |
| // ============================================================================
 | |
| // >> CLASSES
 | |
| // ============================================================================
 | |
| 
 | |
| x86GccThiscall::x86GccThiscall(std::vector<DataTypeSized_t> &vecArgTypes, DataTypeSized_t returnType, int iAlignment) :
 | |
| 	x86GccCdecl(vecArgTypes, returnType, iAlignment)
 | |
| {
 | |
| 	// Always add the |this| pointer.
 | |
| 	DataTypeSized_t type;
 | |
| 	type.type = DATA_TYPE_POINTER;
 | |
| 	type.size = GetDataTypeSize(type, iAlignment);
 | |
| 	type.custom_register = None;
 | |
| 	m_vecArgTypes.insert(m_vecArgTypes.begin(), type);
 | |
| }
 | |
| 
 | |
| x86GccThiscall::~x86GccThiscall()
 | |
| {
 | |
| }
 | |
| 
 | |
| int x86GccThiscall::GetArgStackSize()
 | |
| {
 | |
| 	// Remove the this pointer from the arguments size.
 | |
| 	DataTypeSized_t type;
 | |
| 	type.type = DATA_TYPE_POINTER;
 | |
| 	return x86GccCdecl::GetArgStackSize() - GetDataTypeSize(type, m_iAlignment);
 | |
| }
 | |
| 
 | |
| void** x86GccThiscall::GetStackArgumentPtr(CRegisters* pRegisters)
 | |
| {
 | |
| 	// Skip return address and this pointer.
 | |
| 	DataTypeSized_t type;
 | |
| 	type.type = DATA_TYPE_POINTER;
 | |
| 	return (void **)(pRegisters->m_esp->GetValue<unsigned long>() + 4 + GetDataTypeSize(type, m_iAlignment));
 | |
| }
 | |
| 
 | |
| void x86GccThiscall::SaveCallArguments(CRegisters* pRegisters)
 | |
| {
 | |
| 	// Count the this pointer.
 | |
| 	int size = x86GccCdecl::GetArgStackSize() + GetArgRegisterSize();
 | |
| 	std::unique_ptr<uint8_t[]> pSavedCallArguments = std::make_unique<uint8_t[]>(size);
 | |
| 	size_t offset = 0;
 | |
| 	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, pRegisters), type.size);
 | |
| 		offset += type.size;
 | |
| 	}
 | |
| 	m_pSavedCallArguments.push_back(std::move(pSavedCallArguments));
 | |
| }
 |