Files
native_invoker_v2/include/fxn/invoker.hpp

148 lines
4.8 KiB
C++

#pragma once
#include <cstdint>
#include <string_view>
#include <fxn/context.hpp>
namespace fxn
{
struct invoker_base
{
typedef void*(*get_native_handler_t)(std::uint64_t hash);
static get_native_handler_t _get_native_handler;
typedef void(*schedule_native_t)(fxn::NativeContext* context, bool wait);
static schedule_native_t _schedule_native;
typedef bool(*native_hook_t)(fxn::NativeContext* context, bool& call_original);
typedef size_t(*register_native_hook_t)(std::uint64_t hash, native_hook_t hook);
static register_native_hook_t _register_native_hook;
typedef void(*unregister_native_hook_t)(std::uint64_t hash, size_t index);
static unregister_native_hook_t _unregister_native_hook;
static void initialize(std::string_view module_name);
static void get_native_handler(std::uint64_t hash);
static void schedule_native(fxn::NativeContext* context, bool wait);
static size_t register_native_hook(std::uint64_t hash, native_hook_t hook);
static void unregister_native_hook(std::uint64_t hash, size_t index);
};
template <std::uint64_t Hash, typename Ret, typename... Args>
struct invoker
{
using return_type = Ret;
using native_hash = std::integral_constant<std::uint64_t, Hash>;
return_type operator()(Args... args);
static return_type invoke(Args... args)
{
invoker<Hash, Ret, Args...> inv;
return inv(args...);
}
};
template<std::uint64_t Hash>
struct native_hooker
{
private:
std::size_t _hook;
public:
using native_hash = std::integral_constant<std::uint64_t, Hash>;
native_hooker(fxn::invoker_base::native_hook_t hook)
{
_hook = fxn::invoker_base::register_native_hook(native_hash::value, hook);
}
~native_hooker()
{
fxn::invoker_base::unregister_native_hook(native_hash::value, _hook);
}
};
}
#ifdef FXN_IMPORTS
#include <Windows.h>
namespace fxn
{
inline invoker_base::get_native_handler_t invoker_base::_get_native_handler = nullptr;
inline invoker_base::schedule_native_t invoker_base::_schedule_native = nullptr;
inline invoker_base::register_native_hook_t invoker_base::_register_native_hook = nullptr;
inline invoker_base::unregister_native_hook_t invoker_base::_unregister_native_hook = nullptr;
inline void invoker_base::initialize(std::string_view module_name)
{
HMODULE mod = GetModuleHandleA(module_name.data());
if (!mod)
{
exit(EXIT_FAILURE);
}
invoker_base::_get_native_handler = reinterpret_cast<get_native_handler_t>(
GetProcAddress(mod, "fxn_native_invoker_get_handler")
);
invoker_base::_schedule_native = reinterpret_cast<schedule_native_t>(
GetProcAddress(mod, "fxn_schedule_native")
);
invoker_base::_register_native_hook = reinterpret_cast<register_native_hook_t>(
GetProcAddress(mod, "fxn_register_native_hook")
);
invoker_base::_unregister_native_hook = reinterpret_cast<unregister_native_hook_t>(
GetProcAddress(mod, "fxn_unregister_native_hook")
);
if (!invoker_base::_get_native_handler ||
!invoker_base::_schedule_native ||
!invoker_base::_register_native_hook ||
!invoker_base::_unregister_native_hook)
{
exit(EXIT_FAILURE);
}
}
inline void invoker_base::get_native_handler(std::uint64_t hash)
{
invoker_base::_get_native_handler(hash);
}
inline void invoker_base::schedule_native(fxn::NativeContext* context, bool wait)
{
invoker_base::_schedule_native(context, wait);
}
inline size_t invoker_base::register_native_hook(std::uint64_t hash, native_hook_t hook)
{
return invoker_base::_register_native_hook(hash, hook);
}
inline void invoker_base::unregister_native_hook(std::uint64_t hash, size_t index)
{
invoker_base::_unregister_native_hook(hash, index);
}
template <std::uint64_t Hash, typename Ret, typename... Args>
inline typename invoker<Hash, Ret, Args...>::return_type invoker<Hash, Ret, Args...>::operator()(Args... args)
{
fxn::NativeContext context(Hash);
// Push arguments
(context.PushArgument<Args>(args), ...);
// Schedule native call
invoker_base::schedule_native(&context, true);
if constexpr (!std::is_same_v<Ret, void>)
{
return context.GetResult<Ret>();
}
}
}
#endif