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