Skip to content

Commit 7e08372

Browse files
committed
Use red4ext state hooks
1 parent 6d779f3 commit 7e08372

File tree

5 files changed

+95
-43
lines changed

5 files changed

+95
-43
lines changed

src/CET.cpp

+3-3
Original file line numberDiff line numberDiff line change
@@ -8,9 +8,9 @@ using namespace std::chrono_literals;
88
static std::unique_ptr<CET> s_pInstance{nullptr};
99
static bool s_isRunning{true};
1010

11-
void CET::Initialize()
11+
void CET::Initialize(const RED4ext::Sdk* aSdk)
1212
{
13-
s_pInstance.reset(new CET);
13+
s_pInstance.reset(new CET(aSdk));
1414

1515
s_pInstance->GetVM().Prepare();
1616
}
@@ -67,7 +67,7 @@ bool CET::IsRunning() noexcept
6767
return s_isRunning;
6868
}
6969

70-
CET::CET()
70+
CET::CET(const RED4ext::Sdk* aSdk)
7171
: m_options(m_paths)
7272
, m_persistentState(m_paths, m_options)
7373
, m_bindings(m_paths, m_options)

src/CET.h

+3-2
Original file line numberDiff line numberDiff line change
@@ -6,13 +6,14 @@
66
#include "VKBindings.h"
77
#include "d3d12/D3D12.h"
88
#include "overlay/Overlay.h"
9+
#include "RED4ext/Api/Sdk.hpp"
910
#include "scripting/LuaVM.h"
1011

1112
struct CET
1213
{
1314
~CET();
1415

15-
static void Initialize();
16+
static void Initialize(const RED4ext::Sdk* aSdk);
1617
static void Shutdown();
1718
static CET& Get();
1819

@@ -27,7 +28,7 @@ struct CET
2728
static bool IsRunning() noexcept;
2829

2930
private:
30-
CET();
31+
CET(const RED4ext::Sdk* aSdk);
3132

3233
Paths m_paths;
3334
Options m_options;

src/dllmain.cpp

+5-4
Original file line numberDiff line numberDiff line change
@@ -21,13 +21,15 @@ static HANDLE s_modInstanceMutex = nullptr;
2121

2222
using namespace std::chrono_literals;
2323

24-
static bool Initialize()
24+
static bool Initialize(RED4ext::PluginHandle aHandle, const RED4ext::Sdk* aSdk)
2525
{
2626
try
2727
{
2828
MH_Initialize();
2929

30-
CET::Initialize();
30+
GameMainThread::Create(aHandle, aSdk);
31+
32+
CET::Initialize(aSdk);
3133

3234
const auto& options = CET::Get().GetOptions();
3335

@@ -90,13 +92,12 @@ static void Shutdown()
9092
RED4EXT_C_EXPORT bool RED4EXT_CALL Main(RED4ext::PluginHandle aHandle, RED4ext::EMainReason aReason, const RED4ext::Sdk* aSdk)
9193
{
9294
RED4EXT_UNUSED_PARAMETER(aHandle);
93-
RED4EXT_UNUSED_PARAMETER(aSdk);
9495

9596
switch (aReason)
9697
{
9798
case RED4ext::EMainReason::Load:
9899
{
99-
return Initialize();
100+
return Initialize(aHandle, aSdk);
100101
break;
101102
}
102103
case RED4ext::EMainReason::Unload:

src/scripting/GameHooks.cpp

+72-25
Original file line numberDiff line numberDiff line change
@@ -23,42 +23,27 @@ void GameMainThread::RepeatedTaskQueue::Drain()
2323
++taskIt;
2424
}
2525
}
26-
GameMainThread::StateTickOverride::StateTickOverride(uint32_t aHash, const char* acpRealFunctionName)
26+
GameMainThread::StateTickOverride::StateTickOverride(const char* acpRealFunctionName)
2727
{
28-
const RED4ext::UniversalRelocPtr<uint8_t> func(aHash);
29-
Location = func.GetAddr();
30-
31-
if (Location)
32-
{
33-
if (MH_CreateHook(Location, reinterpret_cast<void*>(&GameMainThread::HookStateTick), reinterpret_cast<void**>(&RealFunction)) != MH_OK || MH_EnableHook(Location) != MH_OK)
34-
Log::Error("Could not hook main thread function {}! Main thread is not completely hooked!", acpRealFunctionName);
35-
else
36-
Log::Info("Main thread function {} hook complete!", acpRealFunctionName);
37-
}
38-
else
39-
Log::Error("Could not locate {}! Main thread is not completely hooked!", acpRealFunctionName);
4028
}
4129

4230
GameMainThread::StateTickOverride::~StateTickOverride()
4331
{
44-
MH_DisableHook(Location);
45-
46-
Location = nullptr;
47-
RealFunction = nullptr;
4832
}
4933

5034
bool GameMainThread::StateTickOverride::OnTick(RED4ext::IGameState* apThisState, RED4ext::CGameApplication* apGameApplication)
5135
{
5236
Tasks.Drain();
5337

54-
return RealFunction(apThisState, apGameApplication);
38+
return true;
5539
}
5640

41+
static GameMainThread* s_gameMainThread = nullptr;
42+
5743
GameMainThread& GameMainThread::Get()
5844
{
59-
static GameMainThread s_gameMainThread;
60-
61-
return s_gameMainThread;
45+
assert(s_gameMainThread);
46+
return *s_gameMainThread;
6247
}
6348

6449
void GameMainThread::AddBaseInitializationTask(const std::function<bool()>& aFunction)
@@ -90,14 +75,76 @@ void GameMainThread::AddGenericTask(const std::function<bool()>& aFunction)
9075
m_genericQueue.AddTask(aFunction);
9176
}
9277

93-
bool GameMainThread::HookStateTick(RED4ext::IGameState* apThisState, RED4ext::CGameApplication* apGameApplication)
78+
void GameMainThread::Create(RED4ext::PluginHandle aHandle, const RED4ext::Sdk* aSdk)
79+
{
80+
if (!s_gameMainThread)
81+
s_gameMainThread = new GameMainThread(aHandle, aSdk);
82+
}
83+
84+
GameMainThread::GameMainThread(RED4ext::PluginHandle aHandle, const RED4ext::Sdk* sdk)
85+
{
86+
{
87+
RED4ext::GameState state{nullptr, &HookBaseInitStateTick, nullptr};
88+
sdk->gameStates->Add(aHandle, RED4ext::EGameStateType::BaseInitialization, &state);
89+
}
90+
{
91+
RED4ext::GameState state{nullptr, &HookInitStateTick, nullptr};
92+
sdk->gameStates->Add(aHandle, RED4ext::EGameStateType::Initialization, &state);
93+
}
94+
{
95+
RED4ext::GameState state{nullptr, &HookRunningStateTick, nullptr};
96+
sdk->gameStates->Add(aHandle, RED4ext::EGameStateType::Running, &state);
97+
}
98+
{
99+
RED4ext::GameState state{nullptr, &HookShutdownStateTick, nullptr};
100+
sdk->gameStates->Add(aHandle, RED4ext::EGameStateType::Shutdown, &state);
101+
}
102+
}
103+
104+
bool GameMainThread::HookBaseInitStateTick(RED4ext::CGameApplication* apGameApplication)
94105
{
95106
auto& gmt = Get();
96107

97108
// drain generic tasks
98109
gmt.m_genericQueue.Drain();
99110

100-
// execute specific state tasks, including original function
101-
const auto cStateIndex = static_cast<size_t>(apThisState->GetType());
102-
return gmt.m_stateTickOverrides[cStateIndex].OnTick(apThisState, apGameApplication);
111+
gmt.m_stateTickOverrides[0].OnTick(apGameApplication->currState, apGameApplication);
112+
113+
return true;
114+
}
115+
116+
bool GameMainThread::HookInitStateTick(RED4ext::CGameApplication* apGameApplication)
117+
{
118+
auto& gmt = Get();
119+
120+
// drain generic tasks
121+
gmt.m_genericQueue.Drain();
122+
123+
gmt.m_stateTickOverrides[1].OnTick(apGameApplication->currState, apGameApplication);
124+
125+
return true;
126+
}
127+
128+
bool GameMainThread::HookRunningStateTick(RED4ext::CGameApplication* apGameApplication)
129+
{
130+
auto& gmt = Get();
131+
132+
// drain generic tasks
133+
gmt.m_genericQueue.Drain();
134+
135+
gmt.m_stateTickOverrides[2].OnTick(apGameApplication->currState, apGameApplication);
136+
137+
return false;
138+
}
139+
140+
bool GameMainThread::HookShutdownStateTick(RED4ext::CGameApplication* apGameApplication)
141+
{
142+
auto& gmt = Get();
143+
144+
// drain generic tasks
145+
gmt.m_genericQueue.Drain();
146+
147+
gmt.m_stateTickOverrides[3].OnTick(apGameApplication->currState, apGameApplication);
148+
149+
return true;
103150
}

src/scripting/GameHooks.h

+12-9
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,9 @@
11
#pragma once
2+
#include "RED4ext/Api/Sdk.hpp"
23

34
struct GameMainThread
45
{
6+
static void Create(RED4ext::PluginHandle aHandle, const RED4ext::Sdk* aSdk);
57
static GameMainThread& Get();
68

79
void AddBaseInitializationTask(const std::function<bool()>& aFunction);
@@ -11,11 +13,14 @@ struct GameMainThread
1113
void AddGenericTask(const std::function<bool()>& aFunction);
1214

1315
private:
14-
GameMainThread() = default;
16+
GameMainThread(RED4ext::PluginHandle aHandle, const RED4ext::Sdk* sdk);
1517

1618
using TStateTick = bool(RED4ext::IGameState*, RED4ext::CGameApplication*);
1719

18-
static bool HookStateTick(RED4ext::IGameState* apThisState, RED4ext::CGameApplication* apGameApplication);
20+
static bool HookBaseInitStateTick(RED4ext::CGameApplication* apGameApplication);
21+
static bool HookInitStateTick(RED4ext::CGameApplication* apGameApplication);
22+
static bool HookRunningStateTick(RED4ext::CGameApplication* apGameApplication);
23+
static bool HookShutdownStateTick(RED4ext::CGameApplication* apGameApplication);
1924

2025
// helper task queue which executes added tasks each drain until they are finished
2126
struct RepeatedTaskQueue
@@ -30,21 +35,19 @@ struct GameMainThread
3035

3136
struct StateTickOverride
3237
{
33-
StateTickOverride(uint32_t aHash, const char* acpRealFunctionName);
38+
StateTickOverride(const char* acpRealFunctionName);
3439
~StateTickOverride();
3540

3641
bool OnTick(RED4ext::IGameState*, RED4ext::CGameApplication*);
3742

38-
uint8_t* Location = nullptr;
39-
TStateTick* RealFunction = nullptr;
4043
RepeatedTaskQueue Tasks;
4144
};
4245

4346
std::array<StateTickOverride, 4> m_stateTickOverrides{
44-
StateTickOverride(CyberEngineTweaks::AddressHashes::CBaseInitializationState_OnTick, "CBaseInitializationState::OnTick"),
45-
StateTickOverride(CyberEngineTweaks::AddressHashes::CInitializationState_OnTick, "CInitializationState::OnTick"),
46-
StateTickOverride(CyberEngineTweaks::AddressHashes::CRunningState_OnTick, "CRunningState::OnTick"),
47-
StateTickOverride(CyberEngineTweaks::AddressHashes::CShutdownState_OnTick, "CShutdownState::OnTick")};
47+
StateTickOverride("CBaseInitializationState::OnTick"),
48+
StateTickOverride("CInitializationState::OnTick"),
49+
StateTickOverride( "CRunningState::OnTick"),
50+
StateTickOverride("CShutdownState::OnTick")};
4851

4952
RepeatedTaskQueue m_genericQueue;
5053
};

0 commit comments

Comments
 (0)