217 lines
5.9 KiB
C++
217 lines
5.9 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.
|
||
|
*/
|
||
|
|
||
|
#ifndef _HOOK_H
|
||
|
#define _HOOK_H
|
||
|
|
||
|
// ============================================================================
|
||
|
// >> INCLUDES
|
||
|
// ============================================================================
|
||
|
#include "registers.h"
|
||
|
#include "convention.h"
|
||
|
#include <am-hashmap.h>
|
||
|
#include <am-hashset.h>
|
||
|
|
||
|
// ============================================================================
|
||
|
// >> HookType_t
|
||
|
// ============================================================================
|
||
|
enum HookType_t
|
||
|
{
|
||
|
// Callback will be executed before the original function.
|
||
|
HOOKTYPE_PRE,
|
||
|
|
||
|
// Callback will be executed after the original function.
|
||
|
HOOKTYPE_POST
|
||
|
};
|
||
|
|
||
|
enum ReturnAction_t
|
||
|
{
|
||
|
ReturnAction_Ignored, // handler didn't take any action
|
||
|
ReturnAction_Handled, // plugin did something, but real function should still be called
|
||
|
ReturnAction_Override, // call real function, but use my return value
|
||
|
ReturnAction_Supercede // skip real function; use my return value
|
||
|
};
|
||
|
|
||
|
|
||
|
// ============================================================================
|
||
|
// >> TYPEDEFS
|
||
|
// ============================================================================
|
||
|
class CHook;
|
||
|
typedef ReturnAction_t (*HookHandlerFn)(HookType_t, CHook*);
|
||
|
|
||
|
#ifdef __linux__
|
||
|
#define __cdecl
|
||
|
#endif
|
||
|
|
||
|
struct IntegerPolicy
|
||
|
{
|
||
|
static inline uint32_t hash(size_t i) {
|
||
|
return ke::HashInteger<sizeof(size_t)>(i);
|
||
|
}
|
||
|
static inline bool matches(size_t i1, size_t i2) {
|
||
|
return i1 == i2;
|
||
|
}
|
||
|
};
|
||
|
|
||
|
typedef ke::HashSet<HookHandlerFn*, ke::PointerPolicy<HookHandlerFn>> HookHandlerSet;
|
||
|
typedef ke::HashMap<HookType_t, HookHandlerSet, IntegerPolicy> HookTypeMap;
|
||
|
typedef ke::HashMap<void*, std::vector<void*>, ke::PointerPolicy<void>> ReturnAddressMap;
|
||
|
|
||
|
namespace sp
|
||
|
{
|
||
|
class MacroAssembler;
|
||
|
}
|
||
|
|
||
|
|
||
|
// ============================================================================
|
||
|
// >> CLASSES
|
||
|
// ============================================================================
|
||
|
|
||
|
class CHook
|
||
|
{
|
||
|
private:
|
||
|
friend class CHookManager;
|
||
|
|
||
|
/*
|
||
|
Creates a new function hook.
|
||
|
|
||
|
@param <pFunc>:
|
||
|
The address of the function to hook
|
||
|
|
||
|
@param <pConvention>:
|
||
|
The calling convention of <pFunc>.
|
||
|
*/
|
||
|
CHook(void* pFunc, ICallingConvention* pConvention);
|
||
|
~CHook();
|
||
|
|
||
|
public:
|
||
|
/*
|
||
|
Adds a hook handler to the hook.
|
||
|
|
||
|
@param type The hook type.
|
||
|
@param pFunc The hook handler that should be added.
|
||
|
*/
|
||
|
void AddCallback(HookType_t type, HookHandlerFn* pFunc);
|
||
|
|
||
|
/*
|
||
|
Removes a hook handler to the hook.
|
||
|
|
||
|
@param type The hook type.
|
||
|
@param pFunc The hook handler that should be removed.
|
||
|
*/
|
||
|
void RemoveCallback(HookType_t type, HookHandlerFn* pFunc);
|
||
|
|
||
|
/*
|
||
|
Checks if a hook handler is already added.
|
||
|
|
||
|
@param type The hook type.
|
||
|
@param pFunc The hook handler that should be checked.
|
||
|
*/
|
||
|
bool IsCallbackRegistered(HookType_t type, HookHandlerFn* pFunc);
|
||
|
|
||
|
/*
|
||
|
Checks if there are any hook handlers added to this hook.
|
||
|
*/
|
||
|
bool AreCallbacksRegistered();
|
||
|
|
||
|
template<class T>
|
||
|
T GetArgument(int iIndex)
|
||
|
{
|
||
|
return *(T *) m_pCallingConvention->GetArgumentPtr(iIndex, m_pRegisters);
|
||
|
}
|
||
|
|
||
|
template<class T>
|
||
|
void SetArgument(int iIndex, T value)
|
||
|
{
|
||
|
void* pPtr = m_pCallingConvention->GetArgumentPtr(iIndex, m_pRegisters);
|
||
|
*(T *) pPtr = value;
|
||
|
m_pCallingConvention->ArgumentPtrChanged(iIndex, m_pRegisters, pPtr);
|
||
|
}
|
||
|
|
||
|
template<class T>
|
||
|
T GetReturnValue()
|
||
|
{
|
||
|
return *(T *) m_pCallingConvention->GetReturnPtr(m_pRegisters);
|
||
|
}
|
||
|
|
||
|
template<class T>
|
||
|
void SetReturnValue(T value)
|
||
|
{
|
||
|
void* pPtr = m_pCallingConvention->GetReturnPtr(m_pRegisters);
|
||
|
*(T *) pPtr = value;
|
||
|
m_pCallingConvention->ReturnPtrChanged(m_pRegisters, pPtr);
|
||
|
}
|
||
|
|
||
|
private:
|
||
|
void* CreateBridge();
|
||
|
|
||
|
void Write_ModifyReturnAddress(sp::MacroAssembler& masm);
|
||
|
void Write_CallHandler(sp::MacroAssembler& masm, HookType_t type);
|
||
|
void Write_SaveRegisters(sp::MacroAssembler& masm, HookType_t type);
|
||
|
void Write_RestoreRegisters(sp::MacroAssembler& masm, HookType_t type);
|
||
|
|
||
|
void* CreatePostCallback();
|
||
|
|
||
|
ReturnAction_t __cdecl HookHandler(HookType_t type);
|
||
|
|
||
|
void* __cdecl GetReturnAddress(void* pESP);
|
||
|
void __cdecl SetReturnAddress(void* pRetAddr, void* pESP);
|
||
|
|
||
|
public:
|
||
|
|
||
|
HookTypeMap m_hookHandler;
|
||
|
|
||
|
// Address of the original function
|
||
|
void* m_pFunc;
|
||
|
|
||
|
ICallingConvention* m_pCallingConvention;
|
||
|
|
||
|
// Address of the bridge
|
||
|
void* m_pBridge;
|
||
|
|
||
|
// Address of the trampoline
|
||
|
void* m_pTrampoline;
|
||
|
|
||
|
// Register storage
|
||
|
CRegisters* m_pRegisters;
|
||
|
|
||
|
// New return address
|
||
|
void* m_pNewRetAddr;
|
||
|
|
||
|
ReturnAddressMap m_RetAddr;
|
||
|
|
||
|
// Save the last return action of the pre HookHandler for use in the post handler.
|
||
|
std::vector<ReturnAction_t> m_LastPreReturnAction;
|
||
|
};
|
||
|
|
||
|
#endif // _HOOK_H
|