From ec25e19f467edd7c0819a58b3525525ed58796b2 Mon Sep 17 00:00:00 2001
From: BlueAmulet <43395286+BlueAmulet@users.noreply.github.com>
Date: Sun, 4 Aug 2024 16:43:15 -0600
Subject: [PATCH] Add support for EGS version of EDF6
Pointer scan to locate EDF6 patch points
Adjust current patches to work for EDF6 EGS
Switch initialization to CPP_OnBoot, initterm is in EDF.dll's DllMain
Use C++17 for LightningScanner
Don't add Mods folder hook if std::wstring::assign is missing
Fix debug builds failing
Fix game crashing if a problem occurs in the ModLoader
Should only crash now if system winmm.dll fails to load
Workaround LightningScanner bug if AOB scan fails
Fix missing labels for aob/alloc commands failure
---
EDFModLoader/EDFModLoader.vcxproj | 8 +-
EDFModLoader/dllmain.cpp | 160 +++++++++++++-----
Patcher/EDF6/ExtraPatches/ChangeFOV.txt | 2 +-
.../EDF6/ExtraPatches/ChangePickupLimits.txt | 10 +-
Patcher/EDF6/Patches/EOSLauncherBypass.txt | 3 +-
Patcher/EDF6/Patches/RemoveChatCensor.txt | 3 +-
Patcher/dllmain.cpp | 10 +-
7 files changed, 139 insertions(+), 57 deletions(-)
diff --git a/EDFModLoader/EDFModLoader.vcxproj b/EDFModLoader/EDFModLoader.vcxproj
index 5396b55..2de580a 100644
--- a/EDFModLoader/EDFModLoader.vcxproj
+++ b/EDFModLoader/EDFModLoader.vcxproj
@@ -113,7 +113,8 @@
true
pch.h
.\;..\HookLib\HookLib\HookLib;%(AdditionalIncludeDirectories)
- MultiThreadedDebug
+ MultiThreadedDebugDLL
+ stdcpp17
Windows
@@ -132,7 +133,8 @@
true
pch.h
.\;..\HookLib\HookLib\HookLib;%(AdditionalIncludeDirectories)
- MultiThreadedDebug
+ MultiThreadedDebugDLL
+ stdcpp17
Windows
@@ -153,6 +155,7 @@
pch.h
.\;..\HookLib\HookLib\HookLib;%(AdditionalIncludeDirectories)
true
+ stdcpp17
Windows
@@ -176,6 +179,7 @@
pch.h
.\;..\HookLib\HookLib\HookLib;%(AdditionalIncludeDirectories)
true
+ stdcpp17
Windows
diff --git a/EDFModLoader/dllmain.cpp b/EDFModLoader/dllmain.cpp
index f4a5fba..9f214b5 100644
--- a/EDFModLoader/dllmain.cpp
+++ b/EDFModLoader/dllmain.cpp
@@ -12,6 +12,7 @@
#include
#include
#include
+#include
#include "proxy.h"
#include "PluginAPI.h"
@@ -19,6 +20,7 @@
#include
#include
+#include
typedef struct {
@@ -36,8 +38,8 @@ static std::vector hooks; // Holds all original hooked functions
typedef void* (__fastcall *initterm_func)(void*, void*);
static initterm_func initterm_orig;
// Handles opening files from disk or through CRI File System
-typedef void* (__fastcall *fnk244d0_func)(void*, void*, void*);
-static fnk244d0_func fnk244d0_orig;
+typedef void* (__fastcall *crifsio_func)(void*, void*, void*);
+static crifsio_func crifsio_orig;
// Puts wchar_t* in wstring
typedef void* (__fastcall *wstrassign_func)(void*, const wchar_t*, size_t);
static wstrassign_func wstrassign_orig;
@@ -143,12 +145,7 @@ typedef struct {
int version;
} PointerSet;
-int pointerSet = -1;
-PointerSet psets[3] = {
- {0xaa36d0, L"EarthDefenceForce 4.1 for Windows", "EDF41", "EML4_Load", {0x667102, 0x8ed80, 0x91580, 0x91790}, 41}, // EDF 4.1
- {0xebcbd0, L"EarthDefenceForce 5 for PC", "EDF5", "EML5_Load", {0x9c835a, 0x244d0, 0x27380, 0x27680}, 5}, // EDF 5
- {0x17e4210, L"EarthDefenceForce 6 for PC", "EDF6", "EML6_Load", {0x12d6f26, 0x748d0, 0x3d440, 0x3e490}, 6}, // EDF 6
-};
+static const char *plugFunc;
// Search and load all *.dll files in Mods\Plugins\ folder
static void LoadPluginsFromPath(LPCWSTR path, bool asi) {
@@ -166,7 +163,7 @@ static void LoadPluginsFromPath(LPCWSTR path, bool asi) {
PLOG_INFO << "Loading Plugin: " << plugpath;
HMODULE plugin = LoadLibraryW(plugpath);
if (plugin != NULL) {
- LoadDef loadfunc = (LoadDef)GetProcAddress(plugin, psets[pointerSet].plugfunc);
+ LoadDef loadfunc = (LoadDef)GetProcAddress(plugin, plugFunc);
bool unload = false;
if (loadfunc != NULL) {
PluginInfo *pluginInfo = new PluginInfo();
@@ -208,7 +205,7 @@ static void LoadPluginsFromPath(LPCWSTR path, bool asi) {
procedure();
}
} else {
- PLOG_WARNING << "Plugin does not contain " << psets[pointerSet].plugfunc << " function";
+ PLOG_WARNING << "Plugin does not contain " << plugFunc << " function";
unload = true;
}
if (unload) {
@@ -270,7 +267,7 @@ struct oddstr {
};
static_assert(sizeof(oddstr) == 32, "Weird string structure should have a length of 32");
-static void *__fastcall fnk244d0_hook(void *unk1, oddstr *str, void *unk2) {
+static void *__fastcall crifsio_hook(void *unk1, oddstr *str, void *unk2) {
// First 8 bytes are a pointer if length >= 8
// Otherwise string is stored where pointer will be?
wchar_t *path = (wchar_t*)str;
@@ -290,7 +287,7 @@ static void *__fastcall fnk244d0_hook(void *unk1, oddstr *str, void *unk2) {
}
delete[] modpath;
}
- return fnk244d0_orig(unk1, str, unk2);
+ return crifsio_orig(unk1, str, unk2);
}
// Internal logging hook
@@ -361,6 +358,21 @@ static const char ModLoaderStr[] = "ModLoader";
PBYTE hmodEXE;
char hmodName[MAX_PATH];
+PointerSet psets[] = {
+ {0xaa36d0, L"EarthDefenceForce 4.1 for Windows", "EDF41", "EML4_Load", {0x667102, 0x8ed80, 0x91580, 0x91790}, 41}, // EDF 4.1
+ {0xebcbd0, L"EarthDefenceForce 5 for PC", "EDF5", "EML5_Load", {0x9c835a, 0x244d0, 0x27380, 0x27680}, 5}, // EDF 5
+};
+
+static uintptr_t ScanPtr(LightningScanner::ScanResult result) {
+ void *addr = result.Get();
+ if (addr != NULL) {
+ // Return the result as an offset to the executable
+ return (uintptr_t)addr - (uintptr_t)hmodEXE;
+ } else {
+ return NULL;
+ }
+}
+
void SetupHook(uintptr_t offset, void **func, void* hook, const char *reason, BOOL active) {
if (!offset) {
PLOG_INFO << "Skipping unsupported " << hmodName << " hook (" << reason << ")";
@@ -412,62 +424,114 @@ BOOL APIENTRY DllMain(HMODULE hModule, DWORD ul_reason_for_call, LPVOID lpReserv
PluginInfo *selfInfo = new PluginInfo;
selfInfo->infoVersion = PluginInfo::MaxInfoVer;
selfInfo->name = "EDFModLoader";
- selfInfo->version = PLUG_VER(1, 0, 8, 0);
+ selfInfo->version = PLUG_VER(1, 0, 9, 0);
PluginData *selfData = new PluginData;
selfData->info = selfInfo;
selfData->module = hModule;
plugins.push_back(selfData);
+ PluginVersion v = selfInfo->version;
+ PLOG_INFO.printf("EDFModLoader v%u.%u.%u Initializing\n", v.major, v.minor, v.patch);
+
+ // Setup DLL proxy
+ wchar_t path[MAX_PATH];
+ if (!GetWindowsDirectoryW(path, _countof(path))) {
+ DWORD dwError = GetLastError();
+ PLOG_ERROR << "Failed to get windows directory path: error " << dwError;
+ return FALSE;
+ }
+
+ wcscat_s(path, L"\\System32\\winmm.dll");
+
+ PLOG_INFO << "Loading real winmm.dll";
+ HMODULE syswinmm = LoadLibraryW(path);
+ if (syswinmm == NULL) {
+ DWORD dwError = GetLastError();
+ PLOG_ERROR << "Failed to load system winmm.dll: error " << dwError;
+ return FALSE;
+ }
+ PLOG_INFO << "Setting up dll proxy functions";
+ setupFunctions(syswinmm);
+
// Determine what game is hosting us
hmodEXE = (PBYTE)GetModuleHandleW(NULL);
if (hmodEXE == NULL) {
PLOG_ERROR << "Failed to retrieve a handle to the currently running game";
- return FALSE;
+ break;
}
GetModuleFileNameA((HMODULE)hmodEXE, hmodName, _countof(hmodName));
char *hmodFName = PathFindFileNameA(hmodName);
memmove(hmodName, hmodFName, strlen(hmodFName) + 1);
+
+ int pointerSet = -1;
for (int i = 0; i < _countof(psets); i++) {
- // TODO: Hack to support EDF6
- if (lstrcmpiA(hmodName, "EDF6.exe") == 0 && psets[i].version == 6)
- {
- hmodEXE = (PBYTE)GetModuleHandleW(L"EDF.dll");
- if (hmodEXE == NULL) {
- PLOG_ERROR << "Failed to retrieve a handle to EDF.dll";
- return FALSE;
- }
- GetModuleFileNameA((HMODULE)hmodEXE, hmodName, _countof(hmodName));
- char *hmodFName = PathFindFileNameA(hmodName);
- memmove(hmodName, hmodFName, strlen(hmodFName) + 1);
- }
size_t search_len = wcslen(psets[i].search);
if (!IsBadReadPtr(hmodEXE + psets[i].offset, search_len+1) && !wcsncmp((wchar_t*)(hmodEXE + psets[i].offset), psets[i].search, search_len)) {
pointerSet = i;
break;
}
}
- if (pointerSet == -1) {
- PLOG_ERROR << "Failed to determine what exe is running";
- return FALSE;
- }
- FuncOffsets pointers = psets[pointerSet].pointers;
+ FuncOffsets pointers = {};
+ if (pointerSet != -1) {
+ PLOG_INFO << "Running under " << psets[pointerSet].ident;
+ pointers = psets[pointerSet].pointers;
+ plugFunc = psets[pointerSet].plugfunc;
+ } else if (lstrcmpiA(hmodName, "EDF6.exe") == 0) {
+ PLOG_INFO << "Running under EDF6";
+ plugFunc = "EML6_Load";
+
+ // Get a handle to EDF.dll
+ hmodEXE = (PBYTE)GetModuleHandleW(L"EDF.dll");
+ if (hmodEXE == NULL) {
+ PLOG_ERROR << "Failed to retrieve a handle to EDF.dll";
+ break;
+ }
- PluginVersion v = selfInfo->version;
- PLOG_INFO.printf("EDFModLoader (%s) v%u.%u.%u Initializing\n", psets[pointerSet].ident, v.major, v.minor, v.patch);
+ // Determine bounds of executable
+ MODULEINFO modInfo = {};
+ if (!GetModuleInformation(GetCurrentProcess(), (HMODULE)hmodEXE, &modInfo, sizeof(MODULEINFO))) {
+ PLOG_ERROR << "Failed to fetch size information for EDF.dll";
+ break;
+ }
+ size_t ScanRange = modInfo.SizeOfImage - 32; // Reduce range by 32 to workaround overrun bug in LightningScanner
+
+ // Update the module name
+ GetModuleFileNameA((HMODULE)hmodEXE, hmodName, _countof(hmodName));
+ char *hmodFName = PathFindFileNameA(hmodName);
+ memmove(hmodName, hmodFName, strlen(hmodFName) + 1);
+
+ // Hook the DLLs secondary entrypoint for additional initialization
+ pointers.initterm = (uintptr_t)GetProcAddress((HMODULE)hmodEXE, "CPP_OnBoot");
+ if (pointers.initterm != NULL) {
+ pointers.initterm -= (uintptr_t)hmodEXE;
+ } else {
+ PLOG_ERROR << "Failed to locate CPP_OnBoot function";
+ }
- // Setup DLL proxy
- wchar_t path[MAX_PATH];
- if (!GetWindowsDirectoryW(path, _countof(path))) {
- DWORD dwError = GetLastError();
- PLOG_ERROR << "Failed to get windows directory path: error " << dwError;
- return FALSE;
- }
+ // Find patch points via pointer scanning
+ LightningScanner::Scanner scanner("");
- wcscat_s(path, L"\\System32\\winmm.dll");
+ scanner = LightningScanner::Scanner("48895C2418488974242055574156488BEC4883EC70488B05????F7");
+ pointers.redirect = ScanPtr(scanner.Find(hmodEXE, ScanRange));
+ if (pointers.redirect == NULL) {
+ PLOG_ERROR << "Failed to locate CriFsIo function";
+ }
- PLOG_INFO << "Loading real winmm.dll";
- PLOG_INFO << "Setting up dll proxy functions";
- setupFunctions(LoadLibraryW(path));
+ scanner = LightningScanner::Scanner("48895C240848896C2410488974241857415641574883EC20488B71");
+ pointers.wstrassign = ScanPtr(scanner.Find(hmodEXE, ScanRange));
+ if (pointers.wstrassign == NULL) {
+ PLOG_ERROR << "Failed to locate std::wstring::assign function";
+ }
+
+ scanner = LightningScanner::Scanner("48895424104C894424184C894C2420C3488D");
+ pointers.gamelog = ScanPtr(scanner.Find(hmodEXE, ScanRange));
+ if (pointers.gamelog == NULL) {
+ PLOG_ERROR << "Failed to locate game debug logging function";
+ }
+ } else {
+ PLOG_ERROR << "Failed to determine what exe is running";
+ break;
+ }
// Create ModLoader folders
CreateDirectoryW(L"Mods", NULL);
@@ -476,7 +540,7 @@ BOOL APIENTRY DllMain(HMODULE hModule, DWORD ul_reason_for_call, LPVOID lpReserv
// Initialize MinHook
if (MH_Initialize() != MH_OK) {
PLOG_ERROR << "Failed to initialize MinHook";
- return FALSE;
+ break;
}
// Hook function for additional ModLoader initialization
@@ -484,7 +548,11 @@ BOOL APIENTRY DllMain(HMODULE hModule, DWORD ul_reason_for_call, LPVOID lpReserv
// Add Mods folder redirector hook
wstrassign_orig = (wstrassign_func)((PBYTE)hmodEXE + pointers.wstrassign);
- SetupHook(pointers.redirect, (PVOID*)&fnk244d0_orig, fnk244d0_hook, "Mods folder redirector", Redirect);
+ if (pointers.wstrassign) {
+ SetupHook(pointers.redirect, (PVOID *)&crifsio_orig, crifsio_hook, "Mods folder redirector", Redirect);
+ } else {
+ PLOG_INFO << "Skipping unsupported " << hmodName << " hook (Mods folder redirector, no std::wstring::assign)";
+ }
// Add internal logging hook
SetupHook(pointers.gamelog, (PVOID*)&gamelog_orig, gamelog_hook, "Interal logging hook", GameLog);
diff --git a/Patcher/EDF6/ExtraPatches/ChangeFOV.txt b/Patcher/EDF6/ExtraPatches/ChangeFOV.txt
index b299392..faa0220 100644
--- a/Patcher/EDF6/ExtraPatches/ChangeFOV.txt
+++ b/Patcher/EDF6/ExtraPatches/ChangeFOV.txt
@@ -11,5 +11,5 @@ f32! 0.7853982 ; 45 vFOV (Default)
;f32! 1.5707964 ; 90 vFOV
;f32! 2.9670596 ; Quake Pro
-aob vFOVoff F3 0F 10 05 ?? ?? ?? ?? F3 0F 5E 87 10 04 00 00
+aob vFOVoff F30F1005????????F30F5E87
vFOVoff+4: rel32! vFOV
\ No newline at end of file
diff --git a/Patcher/EDF6/ExtraPatches/ChangePickupLimits.txt b/Patcher/EDF6/ExtraPatches/ChangePickupLimits.txt
index 9cf5521..f26f45a 100644
--- a/Patcher/EDF6/ExtraPatches/ChangePickupLimits.txt
+++ b/Patcher/EDF6/ExtraPatches/ChangePickupLimits.txt
@@ -1,5 +1,9 @@
; Author: demonized and redone
-2C7709: u32! 128 ; Maximum pickups on a map at once. Default 128
+aob maxOnMap 288000000089
+aob maxMission1 00040000735B
+aob maxMission2 00040000730EFF
-2C737A: u32! 1024 ;Maximum pickups per mission. Default 1024
-2C73C7: u32! 1024 ;Synchronize with above value
\ No newline at end of file
+maxOnMap+1: u32! 128 ; Maximum pickups on a map at once. Default 128
+
+maxMission1: u32! 1024 ;Maximum pickups per mission. Default 1024
+maxMission2: u32! 1024 ;Synchronize with above value
\ No newline at end of file
diff --git a/Patcher/EDF6/Patches/EOSLauncherBypass.txt b/Patcher/EDF6/Patches/EOSLauncherBypass.txt
index 70b57d4..627b2a9 100644
--- a/Patcher/EDF6/Patches/EOSLauncherBypass.txt
+++ b/Patcher/EDF6/Patches/EOSLauncherBypass.txt
@@ -1,2 +1,3 @@
; Author: BlueAmulet
-702CC2: 90 E9
\ No newline at end of file
+aob EOSCheck 200F845E
+EOSCheck+1: 90 E9
\ No newline at end of file
diff --git a/Patcher/EDF6/Patches/RemoveChatCensor.txt b/Patcher/EDF6/Patches/RemoveChatCensor.txt
index 179ff10..18fe312 100644
--- a/Patcher/EDF6/Patches/RemoveChatCensor.txt
+++ b/Patcher/EDF6/Patches/RemoveChatCensor.txt
@@ -1,2 +1,3 @@
; Author: Souzooka
-71E4A2: 90 E9
\ No newline at end of file
+aob ChatFilter 604885C90F84BF
+ChatFilter+4: 90 E9
\ No newline at end of file
diff --git a/Patcher/dllmain.cpp b/Patcher/dllmain.cpp
index 53d829c..1fd7bdf 100644
--- a/Patcher/dllmain.cpp
+++ b/Patcher/dllmain.cpp
@@ -161,6 +161,7 @@ void EMLCommon_Load(BOOL EDF6) {
ltprintf("Failed to fetch size information for %s", hmodName);
return;
}
+ size_t ScanRange = modInfo.SizeOfImage - 32; // Reduce range by 32 to workaround overrun bug in LightningScanner
SYSTEM_INFO sysInfo = {0};
GetSystemInfo(&sysInfo);
@@ -289,14 +290,16 @@ void EMLCommon_Load(BOOL EDF6) {
// TODO: Validate pattern
if (valid) {
- const auto scanner = LightningScanner::Scanner(LightningScanner::Pattern(pattern));
- void *result = scanner.Find(hmodEXE, modInfo.SizeOfImage).Get();
+ const LightningScanner::Scanner scanner = LightningScanner::Scanner(LightningScanner::Pattern(pattern));
+ void *result = scanner.Find(hmodEXE, ScanRange).Get();
if (result != NULL) {
labels[symbol] = (uintptr_t)result - (uintptr_t)hmodEXE;
ltprintf("AOB scan found %s at %I64X", symbol.c_str(), labels[symbol]);
} else {
ltprintf("AOB scan failed: %s: %s", symbol.c_str(), pattern.c_str());
patch = false;
+ // Create dummy label anyway to reduce future errors
+ labels[symbol] = (uintptr_t)hmodEXE;
}
} else {
patch = false;
@@ -401,7 +404,8 @@ void EMLCommon_Load(BOOL EDF6) {
ltprintf("Allocation failed: %s bytes for %s", sizeStr, arguments[0].c_str());
patch = false;
}
- } else {
+ }
+ if (!patch) {
// Patch failed, but create dummy label anyway to reduce future errors
labels[arguments[0]] = (uintptr_t)target - (uintptr_t)hmodEXE;
}