From e55ef3b510faa60821b9ea64609f132531411657 Mon Sep 17 00:00:00 2001 From: IcePixelx <41352111+PixieCore@users.noreply.github.com> Date: Mon, 17 Jan 2022 03:28:16 +0100 Subject: [PATCH] Implemented Factories/Interface Classes. --- r5dev/r5dev.vcxproj | 1 + r5dev/r5dev.vcxproj.filters | 3 ++ r5dev/vpc/interfaces.cpp | 60 +++++++++++++++++++++++++++++++++++++ r5dev/vpc/interfaces.h | 35 +++++++++++++++++++++- 4 files changed, 98 insertions(+), 1 deletion(-) create mode 100644 r5dev/vpc/interfaces.cpp diff --git a/r5dev/r5dev.vcxproj b/r5dev/r5dev.vcxproj index 957cc6ca..eee9e822 100644 --- a/r5dev/r5dev.vcxproj +++ b/r5dev/r5dev.vcxproj @@ -203,6 +203,7 @@ + diff --git a/r5dev/r5dev.vcxproj.filters b/r5dev/r5dev.vcxproj.filters index 5eb0bd54..6bcf85ed 100644 --- a/r5dev/r5dev.vcxproj.filters +++ b/r5dev/r5dev.vcxproj.filters @@ -408,6 +408,9 @@ core + + sdk\vpc + diff --git a/r5dev/vpc/interfaces.cpp b/r5dev/vpc/interfaces.cpp new file mode 100644 index 00000000..6f17ddc2 --- /dev/null +++ b/r5dev/vpc/interfaces.cpp @@ -0,0 +1,60 @@ +#include "core/stdafx.h" +#include "interfaces.h" + +/* Might wanna move this and rename a few things? +* I'm not sure how Amos wants to structure this part of the SDK. +* - Pix +*/ + +//--------------------------------------------------------------------------------- +// Purpose: get all factory registered in the global s_pInterfacesRegs +//--------------------------------------------------------------------------------- +void IFactory::GetFactoriesFromRegister() +{ + for (InterfaceGlobals_t* it = s_pInterfacesRegs; it; it = it->m_pNextInterfacePtr) // Loop till we go out of scope. + { + std::string interfaceName = it->m_pInterfaceName; // Get copy of the name. + int indexOfVersionStart = 0; + for (int i = 0; i < interfaceName.length(); i++) // Loop through each charater to find the start of interface version. + { + if (std::isdigit(interfaceName[i])) + { + indexOfVersionStart = i; + break; + } + } + + // Push back the interface. + AddFactory(FactoryInfo(interfaceName, interfaceName.substr(0, indexOfVersionStart), interfaceName.substr(indexOfVersionStart), reinterpret_cast(it->m_pInterfacePtr()))); + } +} + +//--------------------------------------------------------------------------------- +// Purpose: get factory pointer from factoryName from factories vector +//--------------------------------------------------------------------------------- +ADDRESS IFactory::GetFactoryPtr(const std::string& factoryName, bool versionLess) +{ + for (auto& it : factories) // Loop through the whole vector. + { + if (versionLess) + { + if (it.m_szFactoryName == factoryName) // Name match? + return it.m_pFactoryPtr; // Return factory. + } + else + { + if (it.m_szFactoryFullName == factoryName) // Name match? + return it.m_pFactoryPtr; // Return factory. + } + } + + return ADDRESS(); +} + +//--------------------------------------------------------------------------------- +// Purpose: add a factory to the factories vector +//--------------------------------------------------------------------------------- +void IFactory::AddFactory(FactoryInfo factoryInfo) +{ + factories.push_back(factoryInfo); // Push factory info back into the vector. +} \ No newline at end of file diff --git a/r5dev/vpc/interfaces.h b/r5dev/vpc/interfaces.h index 55982391..b14b7dbb 100644 --- a/r5dev/vpc/interfaces.h +++ b/r5dev/vpc/interfaces.h @@ -8,10 +8,43 @@ // Mapping of interface string to globals //----------------------------------------------------------------------------- typedef void* (*InstantiateInterfaceFn)(); - struct InterfaceGlobals_t { InstantiateInterfaceFn m_pInterfacePtr; const char* m_pInterfaceName; InterfaceGlobals_t* m_pNextInterfacePtr; }; + +struct FactoryInfo +{ + ADDRESS m_pFactoryPtr; + std::string m_szFactoryFullName; + std::string m_szFactoryName; + std::string m_szFactoryVersion; + + FactoryInfo() : m_szFactoryFullName(std::string()), m_szFactoryName(std::string()), m_szFactoryVersion(std::string()), m_pFactoryPtr(nullptr) {} + FactoryInfo(std::string factoryFullName, std::string factoryName, std::string factoryVersion, std::uintptr_t factoryPtr) : m_szFactoryFullName(factoryFullName), m_szFactoryName(factoryName), m_szFactoryVersion(factoryVersion), m_pFactoryPtr(factoryPtr) {} + FactoryInfo(std::string factoryFullName, std::uintptr_t factoryPtr) : m_szFactoryFullName(factoryFullName), m_szFactoryName(std::string()), m_szFactoryVersion(std::string()), m_pFactoryPtr(factoryPtr) {} +}; + +//----------------------------------------------------------------------------- +// Class to hold all factories (interfaces) +//----------------------------------------------------------------------------- +class IFactory +{ +public: + + void GetFactoriesFromRegister(); + void AddFactory(FactoryInfo factoryInfo);; + ADDRESS GetFactoryPtr(const std::string& factoryName, bool versionLess = true); + + std::vector factories = {}; +}; + +namespace +{ + /* ==== s_pInterfaceRegs ==================================================================================================================================================== */ + // Check pattern viability for all seasons. + // Make sure it properly gets the interface factory pointer. + InterfaceGlobals_t* s_pInterfacesRegs = nullptr; /* g_mGameDll.FindPatternSIMD((std::uint8_t*)"\xE9\x00\x00\x00\x00\xCC\xCC\x89\x91\x00\x00\x00\x00", "x????xxxx????").FollowNearCallSelf().FindPatternSelf("48 8B 1D", ADDRESS::Direction::DOWN).ResolveRelativeAddressSelf(0x3, 0x7).DerefSelf().RCast(); */ +} \ No newline at end of file