geode/loader/launcher/windows/proxyLoader.cpp

129 lines
4.2 KiB
C++
Raw Normal View History

2024-06-02 16:46:01 -04:00
#include <Windows.h>
#include <string>
#include <array>
#include <vector>
#include <filesystem>
2024-06-02 16:46:01 -04:00
2024-06-03 12:36:51 -04:00
struct XINPUT_STATE;
struct XINPUT_CAPABILITIES;
struct XINPUT_VIBRATION;
2024-06-02 16:46:01 -04:00
2024-06-03 12:36:51 -04:00
constexpr static auto MAX_PATH_CHARS = 32768u;
2024-06-02 16:46:01 -04:00
2024-06-03 12:36:51 -04:00
static HMODULE getXInput() {
static auto xinput = []() -> HMODULE {
std::wstring path(MAX_PATH_CHARS, L'\0');
2024-09-12 12:02:30 -04:00
auto size = GetSystemDirectoryW(path.data(), path.size());
2024-06-03 12:36:51 -04:00
if (size) {
path.resize(size);
return LoadLibraryW((path + L"\\XInput1_4.dll").c_str());
}
return NULL;
}();
2024-06-02 16:46:01 -04:00
2024-06-03 12:36:51 -04:00
return xinput;
}
static FARPROC getFP(const std::string& sym) {
if (auto xinput = getXInput())
return GetProcAddress(xinput, sym.c_str());
return NULL;
}
2024-06-02 16:46:01 -04:00
#pragma comment(linker, "/export:XInputGetState,@2")
extern "C" DWORD XInputGetState(DWORD dwUserIndex, XINPUT_STATE *pState) {
2024-06-03 12:36:51 -04:00
static auto fp = getFP("XInputGetState");
if (fp) {
using FPType = decltype(&XInputGetState);
return reinterpret_cast<FPType>(fp)(dwUserIndex, pState);
2024-06-02 16:46:01 -04:00
}
return ERROR_DEVICE_NOT_CONNECTED;
}
#pragma comment(linker, "/export:XInputSetState,@3")
extern "C" DWORD XInputSetState(DWORD dwUserIndex, XINPUT_VIBRATION* pVibration) {
static auto fp = getFP("XInputSetState");
if (fp) {
using FPType = decltype(&XInputSetState);
return reinterpret_cast<FPType>(fp)(dwUserIndex, pVibration);
}
return ERROR_DEVICE_NOT_CONNECTED;
}
2024-06-03 10:03:37 -04:00
#pragma comment(linker, "/export:XInputGetCapabilities,@4")
extern "C" DWORD XInputGetCapabilities(DWORD dwUserIndex, DWORD dwFlags, XINPUT_CAPABILITIES *pCapabilities) {
2024-06-03 12:36:51 -04:00
static auto fp = getFP("XInputGetCapabilities");
if (fp) {
using FPType = decltype(&XInputGetCapabilities);
return reinterpret_cast<FPType>(fp)(dwUserIndex, dwFlags, pCapabilities);
2024-06-03 10:03:37 -04:00
}
return ERROR_DEVICE_NOT_CONNECTED;
}
2024-06-10 14:36:00 -04:00
static std::wstring getErrorString(DWORD error) {
return L"Could not load Geode! Error code: " + std::to_wstring(error);
}
2024-06-11 09:04:15 -04:00
static DWORD errorThread(LPVOID param) {
constexpr wchar_t REDIST_ERROR[] = L"Could not load Geode!\n"
"This is likely due to an outdated redist package.\n"
"Do you want to update Microsoft Visual C++ Redistributable 2022 to try to fix this issue?";
constexpr wchar_t ALT_REDIST_ERROR[] = L"Could not load Geode!\n\n"
"Please **delete** the following files from your Geometry Dash directory and try again: ";
2024-06-11 09:04:15 -04:00
const DWORD error = reinterpret_cast<DWORD64>(param);
if (error == ERROR_DLL_INIT_FAILED) {
std::array<std::wstring, 4> msvcpDlls = {
L"msvcp140.dll",
L"msvcp140d.dll",
L"vcruntime140.dll",
L"vcruntime140d.dll"
};
std::vector<std::wstring> foundDlls;
for(auto dll : msvcpDlls) {
if(std::filesystem::exists(dll)) {
foundDlls.push_back(dll);
}
}
if(foundDlls.empty()) {
const auto choice = MessageBoxW(NULL, REDIST_ERROR, L"Load failed", MB_YESNO | MB_ICONWARNING);
2024-06-11 09:04:15 -04:00
if (choice == IDYES)
ShellExecuteW(NULL, L"open", L"https://aka.ms/vs/17/release/vc_redist.x64.exe", NULL, NULL, SW_SHOWNORMAL);
} else {
std::wstring files = ALT_REDIST_ERROR;
bool first = true;
for(auto dll : foundDlls) {
if(!first) files += L", ";
files += dll;
first = false;
}
const auto choice = MessageBoxW(NULL, files.c_str(), L"Load failed", MB_OK | MB_ICONWARNING);
}
2024-06-11 09:04:15 -04:00
} else {
MessageBoxW(NULL, getErrorString(error).c_str(), L"Load failed" , MB_OK | MB_ICONWARNING);
}
return 0u;
2024-06-02 16:46:01 -04:00
}
2024-06-03 12:36:51 -04:00
BOOL WINAPI DllMain(HINSTANCE module, DWORD reason, LPVOID _) {
2024-06-02 16:46:01 -04:00
if (reason == DLL_PROCESS_ATTACH) {
DisableThreadLibraryCalls(module);
// This is UB.
2024-06-10 14:36:00 -04:00
if (LoadLibraryW(L"Geode.dll") == NULL) {
2024-06-11 09:04:15 -04:00
const auto param = reinterpret_cast<LPVOID>(static_cast<DWORD64>(GetLastError()));
CreateThread(NULL, 0, &errorThread, param, 0, NULL);
2024-06-10 14:36:00 -04:00
}
2024-06-02 16:46:01 -04:00
}
return TRUE;
}