mirror of
https://github.com/hedge-dev/UnleashedRecomp.git
synced 2025-12-31 16:31:30 -06:00
* Implemented guest-to-host function pointers (WIP) Co-Authored-By: Skyth (Asilkan) <19259897+blueskythlikesclouds@users.noreply.github.com> * function: support more types for function pointers * api: ported BlueBlur headers and misc. research * Move over function-pointers changes from options-menu branch. --------- Co-authored-by: Skyth (Asilkan) <19259897+blueskythlikesclouds@users.noreply.github.com>
361 lines
11 KiB
C++
361 lines
11 KiB
C++
#pragma once
|
|
|
|
#include <cpu/ppc_context.h>
|
|
#include <array>
|
|
#include "xbox.h"
|
|
#include "memory.h"
|
|
|
|
template <typename R, typename... T>
|
|
constexpr std::tuple<T...> function_args(R(*)(T...)) noexcept
|
|
{
|
|
return std::tuple<T...>();
|
|
}
|
|
|
|
template<auto V>
|
|
static constexpr decltype(V) constant_v = V;
|
|
|
|
template<typename T>
|
|
static constexpr bool is_precise_v = std::is_same_v<T, float> || std::is_same_v<T, double>;
|
|
|
|
template<auto Func>
|
|
struct arg_count_t
|
|
{
|
|
static constexpr size_t value = std::tuple_size_v<decltype(function_args(Func))>;
|
|
};
|
|
|
|
template<typename TCallable, int I = 0, typename ...TArgs>
|
|
std::enable_if_t<(I >= sizeof...(TArgs)), void> _tuple_for(std::tuple<TArgs...>&, const TCallable& callable) noexcept
|
|
{
|
|
|
|
}
|
|
|
|
template<typename TCallable, int I = 0, typename ...TArgs>
|
|
std::enable_if_t<(I < sizeof...(TArgs)), void> _tuple_for(std::tuple<TArgs...>& tpl, const TCallable& callable) noexcept
|
|
{
|
|
callable(std::get<I>(tpl), I);
|
|
|
|
_tuple_for<TCallable, I + 1>(tpl, callable);
|
|
}
|
|
|
|
struct ArgTranslator
|
|
{
|
|
FORCEINLINE constexpr static uint64_t GetIntegerArgumentValue(const PPCContext& ctx, uint8_t* base, size_t arg) noexcept
|
|
{
|
|
if (arg <= 7)
|
|
{
|
|
switch (arg)
|
|
{
|
|
case 0: return ctx.r3.u32;
|
|
case 1: return ctx.r4.u32;
|
|
case 2: return ctx.r5.u32;
|
|
case 3: return ctx.r6.u32;
|
|
case 4: return ctx.r7.u32;
|
|
case 5: return ctx.r8.u32;
|
|
case 6: return ctx.r9.u32;
|
|
case 7: return ctx.r10.u32;
|
|
default: break;
|
|
}
|
|
}
|
|
|
|
return *reinterpret_cast<XLPDWORD>(base + ctx.r1.u32 + 0x54 + ((arg - 8) * 8));
|
|
}
|
|
|
|
FORCEINLINE static double GetPrecisionArgumentValue(const PPCContext& ctx, uint8_t* base, size_t arg) noexcept
|
|
{
|
|
switch (arg)
|
|
{
|
|
case 0: return ctx.f1.f64;
|
|
case 1: return ctx.f2.f64;
|
|
case 2: return ctx.f3.f64;
|
|
case 3: return ctx.f4.f64;
|
|
case 4: return ctx.f5.f64;
|
|
case 5: return ctx.f6.f64;
|
|
case 6: return ctx.f7.f64;
|
|
case 7: return ctx.f8.f64;
|
|
case 8: return ctx.f9.f64;
|
|
case 9: return ctx.f10.f64;
|
|
case 10: return ctx.f11.f64;
|
|
case 11: return ctx.f12.f64;
|
|
case 12: return ctx.f13.f64;
|
|
[[unlikely]] default: break;
|
|
}
|
|
|
|
// TODO: get value from stack.
|
|
return 0;
|
|
}
|
|
|
|
FORCEINLINE constexpr static void SetIntegerArgumentValue(PPCContext& ctx, uint8_t* base, size_t arg, uint64_t value) noexcept
|
|
{
|
|
if (arg <= 7)
|
|
{
|
|
switch (arg)
|
|
{
|
|
case 0: ctx.r3.u64 = value; return;
|
|
case 1: ctx.r4.u64 = value; return;
|
|
case 2: ctx.r5.u64 = value; return;
|
|
case 3: ctx.r6.u64 = value; return;
|
|
case 4: ctx.r7.u64 = value; return;
|
|
case 5: ctx.r8.u64 = value; return;
|
|
case 6: ctx.r9.u64 = value; return;
|
|
case 7: ctx.r10.u64 = value; return;
|
|
[[unlikely]] default: break;
|
|
}
|
|
}
|
|
|
|
assert(arg < 7 && "Pushing to stack memory is not yet supported.");
|
|
}
|
|
|
|
FORCEINLINE static void SetPrecisionArgumentValue(PPCContext& ctx, uint8_t* base, size_t arg, double value) noexcept
|
|
{
|
|
switch (arg)
|
|
{
|
|
case 0: ctx.f1.f64 = value; return;
|
|
case 1: ctx.f2.f64 = value; return;
|
|
case 2: ctx.f3.f64 = value; return;
|
|
case 3: ctx.f4.f64 = value; return;
|
|
case 4: ctx.f5.f64 = value; return;
|
|
case 5: ctx.f6.f64 = value; return;
|
|
case 6: ctx.f7.f64 = value; return;
|
|
case 7: ctx.f8.f64 = value; return;
|
|
case 8: ctx.f9.f64 = value; return;
|
|
case 9: ctx.f10.f64 = value; return;
|
|
case 10: ctx.f11.f64 = value; return;
|
|
case 11: ctx.f12.f64 = value; return;
|
|
case 12: ctx.f13.f64 = value; return;
|
|
[[unlikely]] default: break;
|
|
}
|
|
|
|
assert(arg < 12 && "Pushing to stack memory is not yet supported.");
|
|
}
|
|
|
|
template<typename T>
|
|
FORCEINLINE constexpr static std::enable_if_t<!std::is_pointer_v<T>, T> GetValue(PPCContext& ctx, uint8_t* base, size_t idx) noexcept
|
|
{
|
|
if constexpr (is_precise_v<T>)
|
|
{
|
|
return static_cast<T>(GetPrecisionArgumentValue(ctx, base, idx));
|
|
}
|
|
else
|
|
{
|
|
return static_cast<T>(GetIntegerArgumentValue(ctx, base, idx));
|
|
}
|
|
}
|
|
|
|
template<typename T>
|
|
FORCEINLINE constexpr static std::enable_if_t<std::is_pointer_v<T>, T> GetValue(PPCContext& ctx, uint8_t* base, size_t idx) noexcept
|
|
{
|
|
const auto v = GetIntegerArgumentValue(ctx, base, idx);
|
|
if (!v)
|
|
{
|
|
return nullptr;
|
|
}
|
|
|
|
return reinterpret_cast<T>(base + static_cast<uint32_t>(v));
|
|
}
|
|
|
|
template<typename T>
|
|
FORCEINLINE constexpr static std::enable_if_t<!std::is_pointer_v<T>, void> SetValue(PPCContext& ctx, uint8_t* base, size_t idx, T value) noexcept
|
|
{
|
|
if constexpr (is_precise_v<T>)
|
|
{
|
|
SetPrecisionArgumentValue(ctx, base, idx, value);
|
|
}
|
|
else if constexpr (std::is_null_pointer_v<T>)
|
|
{
|
|
SetIntegerArgumentValue(ctx, base, idx, 0);
|
|
}
|
|
else if constexpr (std::is_pointer_v<T>)
|
|
{
|
|
SetIntegerArgumentValue(ctx, base, idx, g_memory.MapVirtual(value));
|
|
}
|
|
else
|
|
{
|
|
SetIntegerArgumentValue(ctx, base, idx, value);
|
|
}
|
|
}
|
|
|
|
template<typename T>
|
|
FORCEINLINE constexpr static std::enable_if_t<std::is_pointer_v<T>, void> SetValue(PPCContext& ctx, uint8_t* base, size_t idx, T value) noexcept
|
|
{
|
|
const auto v = g_memory.MapVirtual((void*)value);
|
|
if (!v)
|
|
{
|
|
return;
|
|
}
|
|
|
|
SetValue(ctx, base, idx, v);
|
|
}
|
|
};
|
|
|
|
struct Argument
|
|
{
|
|
int type{};
|
|
int ordinal{};
|
|
};
|
|
|
|
template<typename T1>
|
|
constexpr std::array<Argument, std::tuple_size_v<T1>> GatherFunctionArguments(const T1& tpl)
|
|
{
|
|
std::array<Argument, std::tuple_size_v<T1>> args{};
|
|
|
|
int floatOrdinal{};
|
|
size_t i{};
|
|
|
|
if constexpr (!args.empty())
|
|
{
|
|
std::apply([&](const auto& first, const auto&... rest)
|
|
{
|
|
auto append = [&]<typename T2>(const T2& v)
|
|
{
|
|
if constexpr (is_precise_v<T2>)
|
|
{
|
|
args[i] = { 1, floatOrdinal++ };
|
|
}
|
|
else
|
|
{
|
|
args[i] = { 0, static_cast<int>(i) }; // what the fuck
|
|
}
|
|
|
|
i++;
|
|
};
|
|
|
|
append(first);
|
|
(append(rest), ...);
|
|
}, tpl);
|
|
}
|
|
|
|
return args;
|
|
}
|
|
|
|
template<auto Func>
|
|
constexpr std::array<Argument, arg_count_t<Func>::value> GatherFunctionArguments()
|
|
{
|
|
return GatherFunctionArguments(function_args(Func));
|
|
}
|
|
|
|
template<auto Func, size_t I>
|
|
struct arg_ordinal_t
|
|
{
|
|
static constexpr size_t value = GatherFunctionArguments<Func>()[I].ordinal;
|
|
};
|
|
|
|
template<auto Func, int I = 0, typename ...TArgs>
|
|
FORCEINLINE void _translate_args_to_host(PPCContext& ctx, uint8_t* base, std::tuple<TArgs...>&) noexcept
|
|
requires (I >= sizeof...(TArgs))
|
|
{
|
|
}
|
|
|
|
template <auto Func, int I = 0, typename ...TArgs>
|
|
FORCEINLINE std::enable_if_t<(I < sizeof...(TArgs)), void> _translate_args_to_host(PPCContext& ctx, uint8_t* base, std::tuple<TArgs...>& tpl) noexcept
|
|
{
|
|
using T = std::tuple_element_t<I, std::remove_reference_t<decltype(tpl)>>;
|
|
std::get<I>(tpl) = ArgTranslator::GetValue<T>(ctx, base, arg_ordinal_t<Func, I>::value);
|
|
|
|
_translate_args_to_host<Func, I + 1>(ctx, base, tpl);
|
|
}
|
|
|
|
template<int I = 0, typename ...TArgs>
|
|
FORCEINLINE void _translate_args_to_guest(PPCContext& ctx, uint8_t* base, std::tuple<TArgs...>&) noexcept
|
|
requires (I >= sizeof...(TArgs))
|
|
{
|
|
}
|
|
|
|
template <int I = 0, typename ...TArgs>
|
|
FORCEINLINE std::enable_if_t<(I < sizeof...(TArgs)), void> _translate_args_to_guest(PPCContext& ctx, uint8_t* base, std::tuple<TArgs...>& tpl) noexcept
|
|
{
|
|
using T = std::tuple_element_t<I, std::remove_reference_t<decltype(tpl)>>;
|
|
ArgTranslator::SetValue<T>(ctx, base, GatherFunctionArguments(std::tuple<TArgs...>{})[I].ordinal, std::get<I>(tpl));
|
|
|
|
_translate_args_to_guest<I + 1>(ctx, base, tpl);
|
|
}
|
|
|
|
template<auto Func>
|
|
FORCEINLINE PPC_FUNC(HostToGuestFunction)
|
|
{
|
|
using ret_t = decltype(std::apply(Func, function_args(Func)));
|
|
|
|
auto args = function_args(Func);
|
|
_translate_args_to_host<Func>(ctx, base, args);
|
|
|
|
if constexpr (std::is_same_v<ret_t, void>)
|
|
{
|
|
std::apply(Func, args);
|
|
}
|
|
else
|
|
{
|
|
auto v = std::apply(Func, args);
|
|
|
|
if constexpr (std::is_pointer<ret_t>())
|
|
{
|
|
if (v != nullptr)
|
|
{
|
|
ctx.r3.u64 = static_cast<uint32_t>(reinterpret_cast<size_t>(v) - reinterpret_cast<size_t>(base));
|
|
}
|
|
else
|
|
{
|
|
ctx.r3.u64 = NULL;
|
|
}
|
|
}
|
|
else if constexpr (is_precise_v<ret_t>)
|
|
{
|
|
ctx.f1.f64 = v;
|
|
}
|
|
else
|
|
{
|
|
ctx.r3.u64 = (uint64_t)v;
|
|
}
|
|
}
|
|
}
|
|
|
|
template<typename T, typename TFunction, typename... TArgs>
|
|
FORCEINLINE T GuestToHostFunction(const TFunction& func, TArgs&&... argv)
|
|
{
|
|
auto args = std::make_tuple(std::forward<TArgs>(argv)...);
|
|
auto& currentCtx = *GetPPCContext();
|
|
|
|
PPCContext newCtx; // NOTE: No need for zero initialization, has lots of unnecessary code generation.
|
|
newCtx.fn = currentCtx.fn;
|
|
newCtx.r1 = currentCtx.r1;
|
|
newCtx.r13 = currentCtx.r13;
|
|
newCtx.fpscr = currentCtx.fpscr;
|
|
|
|
_translate_args_to_guest(newCtx, (uint8_t*)g_memory.base, args);
|
|
|
|
SetPPCContext(newCtx);
|
|
|
|
if constexpr (std::is_function_v<TFunction>)
|
|
func(newCtx, (uint8_t*)g_memory.base);
|
|
else
|
|
(*(PPCFunc**)(newCtx.fn + uint64_t(func) * 2))(newCtx, (uint8_t*)g_memory.base);
|
|
|
|
currentCtx.fpscr = newCtx.fpscr;
|
|
SetPPCContext(currentCtx);
|
|
|
|
if constexpr (std::is_void_v<T>)
|
|
{
|
|
return;
|
|
}
|
|
else if constexpr (std::is_pointer_v<T>)
|
|
{
|
|
return reinterpret_cast<T>((uint64_t)g_memory.Translate(newCtx.r3.u32));
|
|
}
|
|
else if constexpr (is_precise_v<T>)
|
|
{
|
|
return static_cast<T>(newCtx.f1.f64);
|
|
}
|
|
else if constexpr (std::is_integral_v<T>)
|
|
{
|
|
return static_cast<T>(newCtx.r3.u64);
|
|
}
|
|
else
|
|
{
|
|
static_assert(false, "Unsupported return type.");
|
|
}
|
|
}
|
|
|
|
#define GUEST_FUNCTION_HOOK(subroutine, function) \
|
|
PPC_FUNC(subroutine) { HostToGuestFunction<function>(ctx, base); }
|
|
|
|
#define GUEST_FUNCTION_STUB(subroutine) \
|
|
PPC_FUNC(subroutine) { }
|