sourcemod/extensions/dhooks/util.cpp
2023-08-30 22:08:45 +02:00

237 lines
6.1 KiB
C++

/**
* vim: set ts=4 :
* =============================================================================
* SourceMod Dynamic Hooks Extension
* Copyright (C) 2012-2021 AlliedModders LLC. All rights reserved.
* =============================================================================
*
* This program is free software; you can redistribute it and/or modify it under
* the terms of the GNU General Public License, version 3.0, as published by the
* Free Software Foundation.
*
* This program is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
* FOR A PARTICULAR PURPOSE. See the GNU General Public License for more
* details.
*
* You should have received a copy of the GNU General Public License along with
* this program. If not, see <http://www.gnu.org/licenses/>.
*
* As a special exception, AlliedModders LLC gives you permission to link the
* code of this program (as well as its derivative works) to "Half-Life 2," the
* "Source Engine," the "SourcePawn JIT," and any Game MODs that run on software
* by the Valve Corporation. You must obey the GNU General Public License in
* all respects for all other code used. Additionally, AlliedModders LLC grants
* this exception to all derivative works. AlliedModders LLC defines further
* exceptions, found in LICENSE.txt (as of this writing, version JULY-31-2007),
* or <http://www.sourcemod.net/license.php>.
*
* Version: $Id$
*/
#include "util.h"
void * GetObjectAddr(HookParamType type, unsigned int flags, void **params, size_t offset)
{
#ifdef WIN32
if (type == HookParamType_Object)
return (void *)((intptr_t)params + offset);
#elif POSIX
if (type == HookParamType_Object && !(flags & PASSFLAG_ODTOR)) //Objects are passed by rrefrence if they contain destructors.
return (void *)((intptr_t)params + offset);
#endif
return *(void **)((intptr_t)params + offset);
}
size_t GetStackParamOffset(HookParamsStruct *paramStruct, unsigned int index)
{
assert(paramStruct->dg->params[index].custom_register == None);
size_t offset = 0;
for (unsigned int i = 0; i < index; i++)
{
// Only care for arguments on the stack before us.
if (paramStruct->dg->params[i].custom_register != None)
continue;
#ifndef WIN32
if (paramStruct->dg->params[i].type == HookParamType_Object && (paramStruct->dg->params[i].flags & PASSFLAG_ODTOR)) //Passed by refrence
{
offset += sizeof(void *);
continue;
}
#endif
offset += paramStruct->dg->params[i].size;
}
return offset;
}
size_t GetRegisterParamOffset(HookParamsStruct *paramStruct, unsigned int index)
{
// TODO: Fix this up and get a pointer to the CDetour
assert(paramStruct->dg->params[index].custom_register != None);
// Need to get the size of the stack arguments first. Register arguments are stored after them in the buffer.
size_t stackSize = 0;
for (int i = paramStruct->dg->params.size() - 1; i >= 0; i--)
{
if (paramStruct->dg->params[i].custom_register == None)
stackSize += paramStruct->dg->params[i].size;
}
size_t offset = stackSize;
for (unsigned int i = 0; i < index; i++)
{
// Only care for arguments passed through a register as well before us.
if (paramStruct->dg->params[i].custom_register == None)
continue;
offset += paramStruct->dg->params[i].size;
}
return offset;
}
size_t GetParamOffset(HookParamsStruct *paramStruct, unsigned int index)
{
if (paramStruct->dg->params[index].custom_register == None)
return GetStackParamOffset(paramStruct, index);
else
return GetRegisterParamOffset(paramStruct, index);
}
size_t GetParamTypeSize(HookParamType type)
{
return sizeof(void *);
}
size_t GetParamsSize(DHooksCallback *dg)//Get the full size, this is for creating the STACK.
{
size_t res = 0;
for (int i = dg->params.size() - 1; i >= 0; i--)
{
res += dg->params.at(i).size;
}
return res;
}
DataType_t DynamicHooks_ConvertParamTypeFrom(HookParamType type)
{
switch (type)
{
case HookParamType_Int:
return DATA_TYPE_INT;
case HookParamType_Bool:
return DATA_TYPE_BOOL;
case HookParamType_Float:
return DATA_TYPE_FLOAT;
case HookParamType_StringPtr:
case HookParamType_CharPtr:
case HookParamType_VectorPtr:
case HookParamType_CBaseEntity:
case HookParamType_ObjectPtr:
case HookParamType_Edict:
return DATA_TYPE_POINTER;
case HookParamType_Object:
return DATA_TYPE_OBJECT;
default:
smutils->LogError(myself, "Unhandled parameter type %d!", type);
}
return DATA_TYPE_POINTER;
}
DataType_t DynamicHooks_ConvertReturnTypeFrom(ReturnType type)
{
switch (type)
{
case ReturnType_Void:
return DATA_TYPE_VOID;
case ReturnType_Int:
return DATA_TYPE_INT;
case ReturnType_Bool:
return DATA_TYPE_BOOL;
case ReturnType_Float:
return DATA_TYPE_FLOAT;
case ReturnType_StringPtr:
case ReturnType_CharPtr:
case ReturnType_VectorPtr:
case ReturnType_CBaseEntity:
case ReturnType_Edict:
return DATA_TYPE_POINTER;
case ReturnType_Vector:
return DATA_TYPE_OBJECT;
default:
smutils->LogError(myself, "Unhandled return type %d!", type);
}
return DATA_TYPE_VOID;
}
Register_t DynamicHooks_ConvertRegisterFrom(PluginRegister reg)
{
switch (reg)
{
case DHookRegister_Default:
return None;
case DHookRegister_AL:
return AL;
case DHookRegister_CL:
return CL;
case DHookRegister_DL:
return DL;
case DHookRegister_BL:
return BL;
case DHookRegister_AH:
return AH;
case DHookRegister_CH:
return CH;
case DHookRegister_DH:
return DH;
case DHookRegister_BH:
return BH;
case DHookRegister_EAX:
return EAX;
case DHookRegister_ECX:
return ECX;
case DHookRegister_EDX:
return EDX;
case DHookRegister_EBX:
return EBX;
case DHookRegister_ESP:
return ESP;
case DHookRegister_EBP:
return EBP;
case DHookRegister_ESI:
return ESI;
case DHookRegister_EDI:
return EDI;
case DHookRegister_XMM0:
return XMM0;
case DHookRegister_XMM1:
return XMM1;
case DHookRegister_XMM2:
return XMM2;
case DHookRegister_XMM3:
return XMM3;
case DHookRegister_XMM4:
return XMM4;
case DHookRegister_XMM5:
return XMM5;
case DHookRegister_XMM6:
return XMM6;
case DHookRegister_XMM7:
return XMM7;
case DHookRegister_ST0:
return ST0;
}
return None;
}