@@ -23,42 +23,27 @@ void GameMainThread::RepeatedTaskQueue::Drain()
23
23
++taskIt;
24
24
}
25
25
}
26
- GameMainThread::StateTickOverride::StateTickOverride (uint32_t aHash, const char * acpRealFunctionName)
26
+ GameMainThread::StateTickOverride::StateTickOverride (const char * acpRealFunctionName)
27
27
{
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);
40
28
}
41
29
42
30
GameMainThread::StateTickOverride::~StateTickOverride ()
43
31
{
44
- MH_DisableHook (Location);
45
-
46
- Location = nullptr ;
47
- RealFunction = nullptr ;
48
32
}
49
33
50
34
bool GameMainThread::StateTickOverride::OnTick (RED4ext::IGameState* apThisState, RED4ext::CGameApplication* apGameApplication)
51
35
{
52
36
Tasks.Drain ();
53
37
54
- return RealFunction (apThisState, apGameApplication) ;
38
+ return true ;
55
39
}
56
40
41
+ static GameMainThread* s_gameMainThread = nullptr ;
42
+
57
43
GameMainThread& GameMainThread::Get ()
58
44
{
59
- static GameMainThread s_gameMainThread;
60
-
61
- return s_gameMainThread;
45
+ assert (s_gameMainThread);
46
+ return *s_gameMainThread;
62
47
}
63
48
64
49
void GameMainThread::AddBaseInitializationTask (const std::function<bool ()>& aFunction)
@@ -90,14 +75,76 @@ void GameMainThread::AddGenericTask(const std::function<bool()>& aFunction)
90
75
m_genericQueue.AddTask (aFunction);
91
76
}
92
77
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)
94
105
{
95
106
auto & gmt = Get ();
96
107
97
108
// drain generic tasks
98
109
gmt.m_genericQueue .Drain ();
99
110
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 ;
103
150
}
0 commit comments