2022-02-27 07:53:44 -05:00
|
|
|
#include <STDInclude.hpp>
|
2023-01-03 07:16:44 -05:00
|
|
|
#include <Utils/Compression.hpp>
|
|
|
|
|
2022-12-26 07:07:24 -05:00
|
|
|
#include "QuickPatch.hpp"
|
2016-10-19 03:01:01 -04:00
|
|
|
|
|
|
|
namespace Components
|
|
|
|
{
|
2023-01-11 14:37:35 -05:00
|
|
|
Dvar::Var QuickPatch::UIMousePitch;
|
|
|
|
|
2021-11-10 14:48:00 -05:00
|
|
|
Dvar::Var QuickPatch::r_customAspectRatio;
|
2016-10-19 03:01:01 -04:00
|
|
|
|
|
|
|
void QuickPatch::UnlockStats()
|
|
|
|
{
|
2017-02-21 15:49:42 -05:00
|
|
|
if (Dedicated::IsEnabled()) return;
|
2017-02-21 15:40:59 -05:00
|
|
|
|
2016-11-04 15:24:47 -04:00
|
|
|
if (Game::CL_IsCgameInitialized())
|
|
|
|
{
|
|
|
|
Toast::Show("cardicon_locked", "^1Error", "Not allowed while ingame.", 3000);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2016-10-19 03:01:01 -04:00
|
|
|
Command::Execute("setPlayerData prestige 10");
|
|
|
|
Command::Execute("setPlayerData experience 2516000");
|
|
|
|
Command::Execute("setPlayerData iconUnlocked cardicon_prestige10_02 1");
|
|
|
|
|
|
|
|
// Unlock challenges
|
|
|
|
Game::StringTable* challengeTable = Game::DB_FindXAssetHeader(Game::XAssetType::ASSET_TYPE_STRINGTABLE, "mp/allchallengestable.csv").stringTable;
|
|
|
|
|
|
|
|
if (challengeTable)
|
|
|
|
{
|
|
|
|
for (int i = 0; i < challengeTable->rowCount; ++i)
|
|
|
|
{
|
|
|
|
// Find challenge
|
|
|
|
const char* challenge = Game::TableLookup(challengeTable, i, 0);
|
|
|
|
|
|
|
|
int maxState = 0;
|
|
|
|
int maxProgress = 0;
|
|
|
|
|
|
|
|
// Find correct tier and progress
|
|
|
|
for (int j = 0; j < 10; ++j)
|
|
|
|
{
|
|
|
|
int progress = atoi(Game::TableLookup(challengeTable, i, 6 + j * 2));
|
|
|
|
if (!progress) break;
|
|
|
|
|
|
|
|
maxState = j + 2;
|
|
|
|
maxProgress = progress;
|
|
|
|
}
|
|
|
|
|
2017-01-06 09:27:35 -05:00
|
|
|
Command::Execute(Utils::String::VA("setPlayerData challengeState %s %d", challenge, maxState));
|
|
|
|
Command::Execute(Utils::String::VA("setPlayerData challengeProgress %s %d", challenge, maxProgress));
|
2016-10-19 03:01:01 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-12-22 23:15:48 -05:00
|
|
|
Game::dvar_t* QuickPatch::g_antilag;
|
2022-08-03 12:27:07 -04:00
|
|
|
__declspec(naked) void QuickPatch::ClientEventsFireWeapon_Stub()
|
2020-12-22 23:15:48 -05:00
|
|
|
{
|
|
|
|
__asm
|
|
|
|
{
|
|
|
|
// check g_antilag dvar value
|
|
|
|
mov eax, g_antilag;
|
2022-11-10 11:35:57 -05:00
|
|
|
cmp byte ptr [eax + 16], 1;
|
2020-12-22 23:15:48 -05:00
|
|
|
|
|
|
|
// do antilag if 1
|
|
|
|
je fireWeapon
|
|
|
|
|
|
|
|
// do not do antilag if 0
|
|
|
|
mov eax, 0x1A83554 // level.time
|
|
|
|
mov ecx, [eax]
|
|
|
|
|
|
|
|
fireWeapon:
|
2022-08-02 07:24:22 -04:00
|
|
|
push edx
|
|
|
|
push ecx
|
|
|
|
push edi
|
|
|
|
mov eax, 0x4A4D50 // FireWeapon
|
|
|
|
call eax
|
|
|
|
add esp, 0Ch
|
|
|
|
pop edi
|
|
|
|
pop ecx
|
2020-12-22 23:15:48 -05:00
|
|
|
retn
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-08-03 12:27:07 -04:00
|
|
|
__declspec(naked) void QuickPatch::ClientEventsFireWeaponMelee_Stub()
|
2020-12-22 23:15:48 -05:00
|
|
|
{
|
|
|
|
__asm
|
|
|
|
{
|
|
|
|
// check g_antilag dvar value
|
|
|
|
mov eax, g_antilag;
|
2022-11-10 11:35:57 -05:00
|
|
|
cmp byte ptr [eax + 16], 1;
|
2020-12-22 23:15:48 -05:00
|
|
|
|
|
|
|
// do antilag if 1
|
|
|
|
je fireWeaponMelee
|
|
|
|
|
|
|
|
// do not do antilag if 0
|
|
|
|
mov eax, 0x1A83554 // level.time
|
|
|
|
mov edx, [eax]
|
|
|
|
|
|
|
|
fireWeaponMelee:
|
2022-08-02 07:24:22 -04:00
|
|
|
push edx
|
|
|
|
push edi
|
|
|
|
mov eax, 0x4F2470 // FireWeaponMelee
|
|
|
|
call eax
|
|
|
|
add esp, 8
|
|
|
|
pop edi
|
|
|
|
pop ecx
|
2020-12-22 23:15:48 -05:00
|
|
|
retn
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-03-08 07:20:28 -05:00
|
|
|
Game::dvar_t* QuickPatch::Dvar_RegisterAspectRatioDvar(const char* dvarName, const char** /*valueList*/, int defaultIndex, unsigned __int16 flags, const char* description)
|
2019-01-22 19:36:31 -05:00
|
|
|
{
|
2021-11-11 11:47:42 -05:00
|
|
|
static const char* r_aspectRatioEnum[] =
|
2019-01-22 19:36:31 -05:00
|
|
|
{
|
2021-11-10 14:48:00 -05:00
|
|
|
"auto",
|
|
|
|
"standard",
|
|
|
|
"wide 16:10",
|
|
|
|
"wide 16:9",
|
|
|
|
"custom",
|
|
|
|
nullptr
|
2019-01-22 19:36:31 -05:00
|
|
|
};
|
|
|
|
|
|
|
|
// register custom aspect ratio dvar
|
2021-11-10 14:48:00 -05:00
|
|
|
QuickPatch::r_customAspectRatio = Dvar::Register<float>("r_customAspectRatio",
|
|
|
|
16.0f / 9.0f, 4.0f / 3.0f, 63.0f / 9.0f, flags,
|
|
|
|
"Screen aspect ratio. Divide the width by the height in order to get the aspect ratio value. For example: 16 / 9 = 1,77");
|
2019-01-22 19:36:31 -05:00
|
|
|
|
|
|
|
// register enumeration dvar
|
2022-03-08 07:20:28 -05:00
|
|
|
return Game::Dvar_RegisterEnum(dvarName, r_aspectRatioEnum, defaultIndex, flags, description);
|
2019-01-22 19:36:31 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
void QuickPatch::SetAspectRatio()
|
|
|
|
{
|
|
|
|
// set the aspect ratio
|
2021-11-10 14:48:00 -05:00
|
|
|
Utils::Hook::Set<float>(0x66E1C78, r_customAspectRatio.get<float>());
|
2019-01-22 19:36:31 -05:00
|
|
|
}
|
|
|
|
|
2022-08-03 12:27:07 -04:00
|
|
|
__declspec(naked) void QuickPatch::SetAspectRatio_Stub()
|
2019-01-22 19:36:31 -05:00
|
|
|
{
|
|
|
|
__asm
|
|
|
|
{
|
|
|
|
cmp eax, 4;
|
|
|
|
ja goToDefaultCase;
|
|
|
|
je useCustomRatio;
|
|
|
|
|
|
|
|
// execute switch statement code
|
2022-08-02 07:24:22 -04:00
|
|
|
push 0x5063FC;
|
2019-01-22 19:36:31 -05:00
|
|
|
retn;
|
|
|
|
|
|
|
|
goToDefaultCase:
|
2022-08-02 07:24:22 -04:00
|
|
|
push 0x5064FC;
|
2019-01-22 19:36:31 -05:00
|
|
|
retn;
|
|
|
|
|
|
|
|
useCustomRatio:
|
|
|
|
// set custom resolution
|
|
|
|
pushad;
|
|
|
|
call SetAspectRatio;
|
|
|
|
popad;
|
|
|
|
|
|
|
|
// set widescreen to 1
|
|
|
|
mov eax, 1;
|
|
|
|
|
|
|
|
// continue execution
|
2022-08-02 07:24:22 -04:00
|
|
|
push 0x506495;
|
2019-01-22 19:36:31 -05:00
|
|
|
retn;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-08-03 12:27:07 -04:00
|
|
|
BOOL QuickPatch::IsDynClassname_Stub(const char* classname)
|
2021-07-05 15:23:57 -04:00
|
|
|
{
|
2022-07-23 16:16:26 -04:00
|
|
|
const auto version = Zones::Version();
|
2021-08-03 15:36:57 -04:00
|
|
|
|
2021-07-28 13:15:47 -04:00
|
|
|
if (version >= VERSION_LATEST_CODO)
|
|
|
|
{
|
2021-07-28 12:49:39 -04:00
|
|
|
for (auto i = 0; i < Game::spawnVars->numSpawnVars; i++)
|
2021-07-05 15:23:57 -04:00
|
|
|
{
|
2021-07-28 12:49:39 -04:00
|
|
|
char** kvPair = Game::spawnVars->spawnVars[i];
|
2022-07-23 16:16:26 -04:00
|
|
|
const auto* key = kvPair[0];
|
|
|
|
const auto* val = kvPair[1];
|
2021-07-28 12:49:39 -04:00
|
|
|
|
2022-07-23 16:16:26 -04:00
|
|
|
auto isSpecOps = std::strncmp(key, "script_specialops", 17) == 0;
|
|
|
|
auto isSpecOpsOnly = (val[0] == '1') && (val[1] == '\0');
|
2021-07-28 12:49:39 -04:00
|
|
|
|
|
|
|
if (isSpecOps && isSpecOpsOnly)
|
|
|
|
{
|
|
|
|
// This will prevent spawning of any entity that contains "script_specialops: '1'"
|
|
|
|
// It removes extra hitboxes / meshes on 461+ CODO multiplayer maps
|
2021-08-03 15:36:57 -04:00
|
|
|
return TRUE;
|
2021-07-28 12:49:39 -04:00
|
|
|
}
|
2021-07-02 05:43:04 -04:00
|
|
|
}
|
2021-07-05 15:23:57 -04:00
|
|
|
}
|
2021-07-02 05:43:04 -04:00
|
|
|
|
2022-07-23 16:16:26 -04:00
|
|
|
return Utils::Hook::Call<BOOL(const char*)>(0x444810)(classname); // IsDynClassname
|
2021-07-02 05:43:04 -04:00
|
|
|
}
|
2021-09-21 16:04:19 -04:00
|
|
|
|
|
|
|
void QuickPatch::CL_KeyEvent_OnEscape()
|
2022-08-01 10:30:31 -04:00
|
|
|
{
|
2021-09-21 16:04:19 -04:00
|
|
|
if (Game::Con_CancelAutoComplete())
|
|
|
|
return;
|
|
|
|
|
2021-09-21 16:07:30 -04:00
|
|
|
if (TextRenderer::HandleFontIconAutocompleteKey(0, TextRenderer::FONT_ICON_ACI_CONSOLE, Game::K_ESCAPE))
|
|
|
|
return;
|
|
|
|
|
2021-09-21 16:04:19 -04:00
|
|
|
// Close console
|
|
|
|
Game::Key_RemoveCatcher(0, ~Game::KEYCATCH_CONSOLE);
|
2022-08-01 10:30:31 -04:00
|
|
|
}
|
2021-09-21 16:04:19 -04:00
|
|
|
|
|
|
|
__declspec(naked) void QuickPatch::CL_KeyEvent_ConsoleEscape_Stub()
|
|
|
|
{
|
2022-08-01 10:30:31 -04:00
|
|
|
__asm
|
2021-09-21 16:04:19 -04:00
|
|
|
{
|
|
|
|
pushad
|
|
|
|
call CL_KeyEvent_OnEscape
|
|
|
|
popad
|
|
|
|
|
|
|
|
// Exit CL_KeyEvent function
|
|
|
|
mov ebx, 0x4F66F2
|
|
|
|
jmp ebx
|
|
|
|
}
|
|
|
|
}
|
2022-01-25 13:07:31 -05:00
|
|
|
|
2022-08-01 10:30:31 -04:00
|
|
|
void QuickPatch::R_AddImageToList_Hk(Game::XAssetHeader header, void* data)
|
|
|
|
{
|
|
|
|
auto* imageList = static_cast<Game::ImageList*>(data);
|
|
|
|
|
2022-08-01 13:15:03 -04:00
|
|
|
assert(imageList->count < ARRAYSIZE(imageList->image));
|
2022-08-01 10:30:31 -04:00
|
|
|
|
|
|
|
if (header.image->texture.basemap)
|
|
|
|
{
|
|
|
|
imageList->image[imageList->count++] = header.image;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-08-02 07:24:22 -04:00
|
|
|
void QuickPatch::Sys_SpawnQuitProcess_Hk()
|
|
|
|
{
|
|
|
|
if (*Game::sys_exitCmdLine[0] == '\0')
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto workingDir = std::filesystem::current_path().string();
|
2022-08-03 12:27:07 -04:00
|
|
|
auto binary = FileSystem::GetAppdataPath() / "data" / "iw4x" / *Game::sys_exitCmdLine;
|
2022-08-02 07:24:22 -04:00
|
|
|
|
|
|
|
SetEnvironmentVariableA("XLABS_MW2_INSTALL", workingDir.data());
|
2022-08-03 12:27:07 -04:00
|
|
|
Utils::Library::LaunchProcess(binary.string(), "-singleplayer", workingDir);
|
|
|
|
}
|
|
|
|
|
|
|
|
__declspec(naked) void QuickPatch::SND_GetAliasOffset_Stub()
|
|
|
|
{
|
|
|
|
using namespace Game;
|
|
|
|
|
|
|
|
static const char* msg = "SND_GetAliasOffset: Could not find sound alias '%s'";
|
2023-01-11 14:37:35 -05:00
|
|
|
using namespace Game;
|
2022-08-03 12:27:07 -04:00
|
|
|
|
|
|
|
__asm
|
|
|
|
{
|
|
|
|
// Check if snd_alias_t* is null immediately after call to Com_FindSoundAlias_FastFile
|
|
|
|
test eax, eax
|
|
|
|
jz error
|
|
|
|
|
|
|
|
// Game code hook skipped
|
|
|
|
mov ecx, eax
|
|
|
|
mov edx, dword ptr [ecx + 0x4]
|
|
|
|
|
|
|
|
// Resume function
|
|
|
|
push 0x437CB2
|
|
|
|
ret
|
|
|
|
|
|
|
|
error:
|
|
|
|
add esp, 0x4 // Com_FindSoundAlias_FastFile takes one argument
|
|
|
|
|
|
|
|
push [esi] // alias->aliasName
|
|
|
|
push msg
|
|
|
|
push ERR_DROP
|
2023-01-11 14:37:35 -05:00
|
|
|
call Com_Error // Going to longjmp back to safety
|
2022-08-03 12:27:07 -04:00
|
|
|
add esp, 0xC
|
|
|
|
|
|
|
|
xor eax, eax
|
|
|
|
pop esi
|
|
|
|
ret
|
|
|
|
}
|
2022-08-02 07:24:22 -04:00
|
|
|
}
|
|
|
|
|
2022-06-04 13:07:45 -04:00
|
|
|
Game::dvar_t* QuickPatch::Dvar_RegisterConMinicon(const char* dvarName, [[maybe_unused]] bool value, unsigned __int16 flags, const char* description)
|
|
|
|
{
|
|
|
|
#ifdef _DEBUG
|
|
|
|
constexpr auto value_ = true;
|
|
|
|
#else
|
|
|
|
constexpr auto value_ = false;
|
|
|
|
#endif
|
|
|
|
return Game::Dvar_RegisterBool(dvarName, value_, flags, description);
|
|
|
|
}
|
|
|
|
|
2016-10-19 03:01:01 -04:00
|
|
|
QuickPatch::QuickPatch()
|
|
|
|
{
|
2021-07-05 15:23:57 -04:00
|
|
|
// Filtering any mapents that is intended for Spec:Ops gamemode (CODO) and prevent them from spawning
|
2022-08-03 12:27:07 -04:00
|
|
|
Utils::Hook(0x5FBD6E, QuickPatch::IsDynClassname_Stub, HOOK_CALL).install()->quick();
|
2021-07-02 05:43:04 -04:00
|
|
|
|
2021-09-21 16:04:19 -04:00
|
|
|
// Hook escape handling on open console to change behaviour to close the console instead of only canceling autocomplete
|
|
|
|
Utils::Hook(0x4F66A3, CL_KeyEvent_ConsoleEscape_Stub, HOOK_JUMP).install()->quick();
|
|
|
|
|
2019-12-23 10:58:42 -05:00
|
|
|
// Intermission time dvar
|
2022-07-02 13:52:57 -04:00
|
|
|
Game::Dvar_RegisterFloat("scr_intermissionTime", 10, 0, 120, Game::DVAR_NONE, "Time in seconds before match server loads the next map");
|
2019-12-23 10:58:42 -05:00
|
|
|
|
2022-03-08 07:20:28 -05:00
|
|
|
g_antilag = Game::Dvar_RegisterBool("g_antilag", true, Game::DVAR_CODINFO, "Perform antilag");
|
2022-08-03 12:27:07 -04:00
|
|
|
Utils::Hook(0x5D6D56, QuickPatch::ClientEventsFireWeapon_Stub, HOOK_JUMP).install()->quick();
|
|
|
|
Utils::Hook(0x5D6D6A, QuickPatch::ClientEventsFireWeaponMelee_Stub, HOOK_JUMP).install()->quick();
|
2020-12-22 23:15:48 -05:00
|
|
|
|
2019-01-22 19:36:31 -05:00
|
|
|
// Add ultrawide support
|
2021-11-10 14:48:00 -05:00
|
|
|
Utils::Hook(0x51B13B, QuickPatch::Dvar_RegisterAspectRatioDvar, HOOK_CALL).install()->quick();
|
2022-08-03 12:27:07 -04:00
|
|
|
Utils::Hook(0x5063F3, QuickPatch::SetAspectRatio_Stub, HOOK_JUMP).install()->quick();
|
2019-01-22 19:36:31 -05:00
|
|
|
|
2022-06-04 13:07:45 -04:00
|
|
|
Utils::Hook(0x4FA448, QuickPatch::Dvar_RegisterConMinicon, HOOK_CALL).install()->quick();
|
|
|
|
|
2022-08-02 07:24:22 -04:00
|
|
|
Utils::Hook::Set<void(*)(Game::XAssetHeader, void*)>(0x51FCDD, QuickPatch::R_AddImageToList_Hk);
|
|
|
|
|
|
|
|
Utils::Hook::Set<const char*>(0x41DB8C, "iw4x-sp.exe");
|
|
|
|
Utils::Hook(0x4D6989, QuickPatch::Sys_SpawnQuitProcess_Hk, HOOK_CALL).install()->quick();
|
2022-08-01 10:30:31 -04:00
|
|
|
|
2022-08-03 12:27:07 -04:00
|
|
|
// Fix crash as nullptr goes unchecked
|
|
|
|
Utils::Hook(0x437CAD, QuickPatch::SND_GetAliasOffset_Stub, HOOK_JUMP).install()->quick();
|
|
|
|
|
2016-10-19 03:01:01 -04:00
|
|
|
// remove system pre-init stuff (improper quit, disk full)
|
|
|
|
Utils::Hook::Set<BYTE>(0x411350, 0xC3);
|
|
|
|
|
|
|
|
// Don't delete config files if corrupted
|
|
|
|
Utils::Hook::Set<BYTE>(0x47DCB3, 0xEB);
|
|
|
|
Utils::Hook::Set<BYTE>(0x4402B6, 0);
|
|
|
|
|
|
|
|
// hopefully allow alt-tab during game, used at least in alt-enter handling
|
|
|
|
Utils::Hook::Set<DWORD>(0x45ACE0, 0xC301B0);
|
|
|
|
|
|
|
|
// fs_basegame
|
2020-12-09 14:13:34 -05:00
|
|
|
Utils::Hook::Set<const char*>(0x6431D1, BASEGAME);
|
2016-10-19 03:01:01 -04:00
|
|
|
|
|
|
|
// window title
|
2020-12-09 14:13:34 -05:00
|
|
|
Utils::Hook::Set<const char*>(0x5076A0, "IW4x: Multiplayer");
|
2016-10-19 03:01:01 -04:00
|
|
|
|
|
|
|
// sv_hostname
|
2020-12-09 14:13:34 -05:00
|
|
|
Utils::Hook::Set<const char*>(0x4D378B, "IW4Host");
|
2016-10-19 03:01:01 -04:00
|
|
|
|
|
|
|
// console logo
|
2020-12-09 14:13:34 -05:00
|
|
|
Utils::Hook::Set<const char*>(0x428A66, BASEGAME "/images/logo.bmp");
|
2016-10-19 03:01:01 -04:00
|
|
|
|
|
|
|
// splash logo
|
2020-12-09 14:13:34 -05:00
|
|
|
Utils::Hook::Set<const char*>(0x475F9E, BASEGAME "/images/splash.bmp");
|
2016-10-19 03:01:01 -04:00
|
|
|
|
2020-12-09 14:13:34 -05:00
|
|
|
Utils::Hook::Set<const char*>(0x4876C6, "Successfully read stats data\n");
|
2017-01-02 16:14:07 -05:00
|
|
|
|
2016-10-19 03:01:01 -04:00
|
|
|
// Numerical ping (cg_scoreboardPingText 1)
|
|
|
|
Utils::Hook::Set<BYTE>(0x45888E, 1);
|
2022-07-02 13:52:57 -04:00
|
|
|
Utils::Hook::Set<BYTE>(0x45888C, Game::DVAR_CHEAT);
|
2016-10-19 03:01:01 -04:00
|
|
|
|
|
|
|
// increase font sizes for chat on higher resolutions
|
|
|
|
static float float13 = 13.0f;
|
|
|
|
static float float10 = 10.0f;
|
|
|
|
Utils::Hook::Set<float*>(0x5814AE, &float13);
|
|
|
|
Utils::Hook::Set<float*>(0x5814C8, &float10);
|
|
|
|
|
|
|
|
// Enable commandline arguments
|
|
|
|
Utils::Hook::Set<BYTE>(0x464AE4, 0xEB);
|
|
|
|
|
|
|
|
// remove limit on IWD file loading
|
|
|
|
Utils::Hook::Set<BYTE>(0x642BF3, 0xEB);
|
|
|
|
|
2017-06-07 15:52:02 -04:00
|
|
|
// dont run UPNP stuff on main thread
|
|
|
|
Utils::Hook::Set<BYTE>(0x48A135, 0xC3);
|
|
|
|
Utils::Hook::Set<BYTE>(0x48A151, 0xC3);
|
2017-06-11 15:25:31 -04:00
|
|
|
Utils::Hook::Nop(0x684080, 5); // Don't spam the console
|
2017-06-07 15:52:02 -04:00
|
|
|
|
|
|
|
// spawn upnp thread when UPNP_init returns
|
2018-10-09 04:53:04 -04:00
|
|
|
Utils::Hook::Hook(0x47982B, []()
|
2017-06-07 15:52:02 -04:00
|
|
|
{
|
2022-08-01 10:30:31 -04:00
|
|
|
std::thread([]
|
2017-06-07 15:52:02 -04:00
|
|
|
{
|
2023-01-30 07:43:24 -05:00
|
|
|
Com_InitThreadData();
|
|
|
|
|
2017-06-07 15:52:02 -04:00
|
|
|
// check natpmpstate
|
|
|
|
// state 4 is no more devices to query
|
2017-06-08 06:22:45 -04:00
|
|
|
while (Utils::Hook::Get<int>(0x66CE200) < 4)
|
2017-06-07 15:52:02 -04:00
|
|
|
{
|
2017-06-08 06:22:45 -04:00
|
|
|
Utils::Hook::Call<void()>(0x4D7030)();
|
2023-01-30 07:43:24 -05:00
|
|
|
Game::Sys_Sleep(500);
|
2017-06-07 15:52:02 -04:00
|
|
|
}
|
2017-06-08 06:22:45 -04:00
|
|
|
}).detach();
|
2018-10-09 04:53:04 -04:00
|
|
|
}, HOOK_JUMP).install()->quick();
|
2016-10-19 03:01:01 -04:00
|
|
|
|
|
|
|
// disable the IWNet IP detection (default 'got ipdetect' flag to 1)
|
2018-10-09 04:53:04 -04:00
|
|
|
Utils::Hook::Set<BYTE>(0x649D6F0, 1);
|
2016-10-19 03:01:01 -04:00
|
|
|
|
|
|
|
// Fix stats sleeping
|
|
|
|
Utils::Hook::Set<BYTE>(0x6832BA, 0xEB);
|
|
|
|
Utils::Hook::Set<BYTE>(0x4BD190, 0xC3);
|
|
|
|
|
|
|
|
// remove 'impure stats' checking
|
|
|
|
Utils::Hook::Set<BYTE>(0x4BB250, 0x33);
|
|
|
|
Utils::Hook::Set<BYTE>(0x4BB251, 0xC0);
|
|
|
|
Utils::Hook::Set<DWORD>(0x4BB252, 0xC3909090);
|
|
|
|
|
|
|
|
// default sv_pure to 0
|
|
|
|
Utils::Hook::Set<BYTE>(0x4D3A74, 0);
|
|
|
|
|
|
|
|
// remove activeAction execution (exploit in mods)
|
|
|
|
Utils::Hook::Set<BYTE>(0x5A1D43, 0xEB);
|
|
|
|
|
|
|
|
// disable bind protection
|
|
|
|
Utils::Hook::Set<BYTE>(0x4DACA2, 0xEB);
|
|
|
|
|
2017-07-02 05:12:17 -04:00
|
|
|
// require Windows 5
|
|
|
|
Utils::Hook::Set<BYTE>(0x467ADF, 5);
|
|
|
|
Utils::Hook::Set<char>(0x6DF5D6, '5');
|
2016-10-19 03:01:01 -04:00
|
|
|
|
|
|
|
// disable 'ignoring asset' notices
|
|
|
|
Utils::Hook::Nop(0x5BB902, 5);
|
|
|
|
|
|
|
|
// disable migration_dvarErrors
|
|
|
|
Utils::Hook::Set<BYTE>(0x60BDA7, 0);
|
|
|
|
|
|
|
|
// allow joining 'developer 1' servers
|
|
|
|
Utils::Hook::Set<BYTE>(0x478BA2, 0xEB);
|
|
|
|
|
|
|
|
// fs_game fixes
|
|
|
|
Utils::Hook::Nop(0x4A5D74, 2); // remove fs_game profiles
|
|
|
|
Utils::Hook::Set<BYTE>(0x4081FD, 0xEB); // defaultweapon
|
|
|
|
|
|
|
|
// filesystem init default_mp.cfg check
|
|
|
|
Utils::Hook::Nop(0x461A9E, 5);
|
|
|
|
Utils::Hook::Nop(0x461AAA, 5);
|
|
|
|
Utils::Hook::Set<BYTE>(0x461AB4, 0xEB);
|
|
|
|
|
|
|
|
// vid_restart when ingame
|
|
|
|
Utils::Hook::Nop(0x4CA1FA, 6);
|
|
|
|
|
2023-01-16 18:32:00 -05:00
|
|
|
// Filter log (initially com_logFilter, but I don't see why that dvar print is needed)
|
2017-01-14 06:32:44 -05:00
|
|
|
// Seems like it's needed for B3, so there is a separate handling for dedicated servers in Dedicated.cpp
|
|
|
|
if (!Dedicated::IsEnabled())
|
|
|
|
{
|
|
|
|
Utils::Hook::Nop(0x647466, 5); // 'dvar set' lines
|
|
|
|
Utils::Hook::Nop(0x5DF4F2, 5); // 'sending splash open' lines
|
|
|
|
}
|
2016-10-19 03:01:01 -04:00
|
|
|
|
2017-02-21 15:49:42 -05:00
|
|
|
// intro stuff
|
|
|
|
Utils::Hook::Nop(0x60BEE9, 5); // Don't show legals
|
|
|
|
Utils::Hook::Nop(0x60BEF6, 5); // Don't reset the intro dvar
|
2020-12-09 14:13:34 -05:00
|
|
|
Utils::Hook::Set<const char*>(0x60BED2, "unskippablecinematic IW_logo\n");
|
|
|
|
Utils::Hook::Set<const char*>(0x51C2A4, "%s\\" BASEGAME "\\video\\%s.bik");
|
2017-02-21 15:49:42 -05:00
|
|
|
Utils::Hook::Set<DWORD>(0x51C2C2, 0x78A0AC);
|
2016-10-19 03:01:01 -04:00
|
|
|
|
|
|
|
// Redirect logs
|
2020-12-09 14:13:34 -05:00
|
|
|
Utils::Hook::Set<const char*>(0x5E44D8, "logs/games_mp.log");
|
|
|
|
Utils::Hook::Set<const char*>(0x60A90C, "logs/console_mp.log");
|
|
|
|
Utils::Hook::Set<const char*>(0x60A918, "logs/console_mp.log");
|
2016-10-19 03:01:01 -04:00
|
|
|
|
|
|
|
// Rename config
|
2020-12-09 14:13:34 -05:00
|
|
|
Utils::Hook::Set<const char*>(0x461B4B, CLIENT_CONFIG);
|
|
|
|
Utils::Hook::Set<const char*>(0x47DCBB, CLIENT_CONFIG);
|
|
|
|
Utils::Hook::Set<const char*>(0x6098F8, CLIENT_CONFIG);
|
|
|
|
Utils::Hook::Set<const char*>(0x60B279, CLIENT_CONFIG);
|
|
|
|
Utils::Hook::Set<const char*>(0x60BBD4, CLIENT_CONFIG);
|
2016-10-19 03:01:01 -04:00
|
|
|
|
|
|
|
// Disable profile system
|
2022-08-01 10:30:31 -04:00
|
|
|
// Utils::Hook::Nop(0x60BEB1, 5); // GamerProfile_InitAllProfiles - Causes an error, when calling a harrier killstreak.
|
|
|
|
// Utils::Hook::Nop(0x60BEB8, 5); // GamerProfile_LogInProfile
|
|
|
|
// Utils::Hook::Nop(0x4059EA, 5); // GamerProfile_RegisterCommands
|
|
|
|
Utils::Hook::Nop(0x4059EF, 5); // GamerProfile_RegisterDvars
|
|
|
|
Utils::Hook::Nop(0x47DF9A, 5); // GamerProfile_UpdateSystemDvars
|
2016-10-19 03:01:01 -04:00
|
|
|
Utils::Hook::Set<BYTE>(0x5AF0D0, 0xC3); // GamerProfile_SaveProfile
|
|
|
|
Utils::Hook::Set<BYTE>(0x4E6870, 0xC3); // GamerProfile_UpdateSystemVarsFromProfile
|
|
|
|
Utils::Hook::Set<BYTE>(0x4C37F0, 0xC3); // GamerProfile_UpdateProfileAndSaveIfNeeded
|
|
|
|
Utils::Hook::Set<BYTE>(0x633CA0, 0xC3); // GamerProfile_SetPercentCompleteMP
|
|
|
|
|
2016-12-05 11:50:34 -05:00
|
|
|
Utils::Hook::Nop(0x5AF1EC, 5); // Profile loading error
|
2016-11-05 09:45:38 -04:00
|
|
|
Utils::Hook::Set<BYTE>(0x5AE212, 0xC3); // Profile reading
|
|
|
|
|
2016-10-19 03:01:01 -04:00
|
|
|
// GamerProfile_RegisterCommands
|
|
|
|
// Some random function used as nullsub :P
|
|
|
|
Utils::Hook::Set<DWORD>(0x45B868, 0x5188FB); // profile_menuDvarsSetup
|
|
|
|
Utils::Hook::Set<DWORD>(0x45B87E, 0x5188FB); // profile_menuDvarsFinish
|
|
|
|
Utils::Hook::Set<DWORD>(0x45B894, 0x5188FB); // profile_toggleInvertedPitch
|
|
|
|
Utils::Hook::Set<DWORD>(0x45B8AA, 0x5188FB); // profile_setViewSensitivity
|
|
|
|
Utils::Hook::Set<DWORD>(0x45B8C3, 0x5188FB); // profile_setButtonsConfig
|
|
|
|
Utils::Hook::Set<DWORD>(0x45B8D9, 0x5188FB); // profile_setSticksConfig
|
|
|
|
Utils::Hook::Set<DWORD>(0x45B8EF, 0x5188FB); // profile_toggleAutoAim
|
|
|
|
Utils::Hook::Set<DWORD>(0x45B905, 0x5188FB); // profile_SetHasEverPlayed_MainMenu
|
|
|
|
Utils::Hook::Set<DWORD>(0x45B91E, 0x5188FB); // profile_SetHasEverPlayed_SP
|
|
|
|
Utils::Hook::Set<DWORD>(0x45B934, 0x5188FB); // profile_SetHasEverPlayed_SO
|
|
|
|
Utils::Hook::Set<DWORD>(0x45B94A, 0x5188FB); // profile_SetHasEverPlayed_MP
|
|
|
|
Utils::Hook::Set<DWORD>(0x45B960, 0x5188FB); // profile_setVolume
|
|
|
|
Utils::Hook::Set<DWORD>(0x45B979, 0x5188FB); // profile_setGamma
|
|
|
|
Utils::Hook::Set<DWORD>(0x45B98F, 0x5188FB); // profile_setBlacklevel
|
|
|
|
Utils::Hook::Set<DWORD>(0x45B9A5, 0x5188FB); // profile_toggleCanSkipOffensiveMissions
|
|
|
|
|
|
|
|
// Patch SV_IsClientUsingOnlineStatsOffline
|
|
|
|
Utils::Hook::Set<DWORD>(0x46B710, 0x90C3C033);
|
|
|
|
|
2019-03-10 17:01:56 -04:00
|
|
|
// Fix mouse lag
|
|
|
|
Utils::Hook::Nop(0x4731F5, 8);
|
2022-05-05 10:03:14 -04:00
|
|
|
Scheduler::Loop([]
|
2019-03-10 17:01:56 -04:00
|
|
|
{
|
|
|
|
SetThreadExecutionState(ES_DISPLAY_REQUIRED);
|
2022-05-05 10:03:14 -04:00
|
|
|
}, Scheduler::Pipeline::RENDERER);
|
2019-03-10 17:01:56 -04:00
|
|
|
|
2017-02-21 15:49:42 -05:00
|
|
|
// Fix mouse pitch adjustments
|
2023-01-11 14:37:35 -05:00
|
|
|
UIMousePitch = Dvar::Register<bool>("ui_mousePitch", false, Game::DVAR_ARCHIVE, "");
|
2022-08-24 10:38:14 -04:00
|
|
|
UIScript::Add("updateui_mousePitch", []([[maybe_unused]] const UIScript::Token& token, [[maybe_unused]] const Game::uiInfo_s* info)
|
2016-10-19 03:01:01 -04:00
|
|
|
{
|
2023-01-11 14:37:35 -05:00
|
|
|
if (UIMousePitch.get<bool>())
|
2016-10-19 03:01:01 -04:00
|
|
|
{
|
2023-01-11 14:37:35 -05:00
|
|
|
Game::Dvar_SetFloatByName("m_pitch", -0.022f);
|
2017-02-21 15:49:42 -05:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2023-01-11 14:37:35 -05:00
|
|
|
Game::Dvar_SetFloatByName("m_pitch", 0.022f);
|
2017-02-21 15:49:42 -05:00
|
|
|
}
|
|
|
|
});
|
2016-10-19 03:01:01 -04:00
|
|
|
|
2022-06-12 18:02:20 -04:00
|
|
|
Command::Add("unlockstats", QuickPatch::UnlockStats);
|
2016-10-19 03:01:01 -04:00
|
|
|
|
2017-03-26 20:44:37 -04:00
|
|
|
Command::Add("dumptechsets", [](Command::Params* param)
|
|
|
|
{
|
2022-03-17 14:50:20 -04:00
|
|
|
if (param->size() != 2)
|
2017-03-26 20:44:37 -04:00
|
|
|
{
|
|
|
|
Logger::Print("usage: dumptechsets <fastfile> | all\n");
|
|
|
|
return;
|
|
|
|
}
|
2022-06-12 18:02:20 -04:00
|
|
|
|
|
|
|
std::vector<std::string> fastFiles;
|
2023-01-06 07:51:41 -05:00
|
|
|
if (std::strcmp(param->get(1), "all") == 0)
|
2017-03-26 20:44:37 -04:00
|
|
|
{
|
2023-03-17 08:46:29 -04:00
|
|
|
for (const auto& entry : Utils::IO::ListFiles("zone/english", false))
|
2023-01-06 07:51:41 -05:00
|
|
|
{
|
2023-03-17 08:46:29 -04:00
|
|
|
const auto& f = entry.path().string();
|
2022-06-12 18:02:20 -04:00
|
|
|
fastFiles.emplace_back(f.substr(7, f.length() - 10));
|
2023-01-06 07:51:41 -05:00
|
|
|
}
|
2022-06-12 18:02:20 -04:00
|
|
|
|
2023-03-17 08:46:29 -04:00
|
|
|
for (const auto& entry : Utils::IO::ListFiles("zone/dlc", false))
|
2023-01-06 07:51:41 -05:00
|
|
|
{
|
2023-03-17 08:46:29 -04:00
|
|
|
const auto& f = entry.path().string();
|
2022-06-12 18:02:20 -04:00
|
|
|
fastFiles.emplace_back(f.substr(3, f.length() - 6));
|
2023-01-06 07:51:41 -05:00
|
|
|
}
|
2022-06-12 18:02:20 -04:00
|
|
|
|
2023-03-17 08:46:29 -04:00
|
|
|
for (const auto& entry : Utils::IO::ListFiles("zone/patch", false))
|
2023-01-06 07:51:41 -05:00
|
|
|
{
|
2023-03-17 08:46:29 -04:00
|
|
|
const auto& f = entry.path().string();
|
2022-06-12 18:02:20 -04:00
|
|
|
fastFiles.emplace_back(f.substr(5, f.length() - 8));
|
2023-01-06 07:51:41 -05:00
|
|
|
}
|
2017-03-26 20:44:37 -04:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2022-06-12 18:02:20 -04:00
|
|
|
fastFiles.emplace_back(param->get(1));
|
2017-03-26 20:44:37 -04:00
|
|
|
}
|
|
|
|
|
2022-06-12 18:02:20 -04:00
|
|
|
auto count = 0;
|
2017-03-26 20:44:37 -04:00
|
|
|
|
2018-12-17 08:29:18 -05:00
|
|
|
AssetHandler::OnLoad([](Game::XAssetType type, Game::XAssetHeader asset, const std::string& name, bool* /*restrict*/)
|
2017-03-26 20:44:37 -04:00
|
|
|
{
|
|
|
|
// they're basically the same right?
|
2017-03-27 11:48:52 -04:00
|
|
|
if (type == Game::ASSET_TYPE_PIXELSHADER || type == Game::ASSET_TYPE_VERTEXSHADER)
|
2017-03-26 20:44:37 -04:00
|
|
|
{
|
|
|
|
Utils::IO::CreateDir("userraw/shader_bin");
|
|
|
|
|
|
|
|
const char* formatString;
|
|
|
|
if (type == Game::ASSET_TYPE_PIXELSHADER)
|
|
|
|
{
|
|
|
|
formatString = "userraw/shader_bin/%.ps";
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
formatString = "userraw/shader_bin/%.vs";
|
|
|
|
}
|
|
|
|
|
2022-12-11 12:54:24 -05:00
|
|
|
const auto path = std::format("{}{}", formatString, name);
|
|
|
|
if (Utils::IO::FileExists(path)) return;
|
2017-03-26 20:44:37 -04:00
|
|
|
|
2017-04-22 15:47:04 -04:00
|
|
|
Utils::Stream buffer(0x1000);
|
2022-06-12 18:02:20 -04:00
|
|
|
auto* dest = buffer.dest<Game::MaterialPixelShader>();
|
2017-04-22 15:47:04 -04:00
|
|
|
buffer.save(asset.pixelShader);
|
2017-03-26 20:44:37 -04:00
|
|
|
|
2017-04-22 15:47:04 -04:00
|
|
|
if (asset.pixelShader->prog.loadDef.program)
|
2017-03-26 20:44:37 -04:00
|
|
|
{
|
2017-04-22 15:47:04 -04:00
|
|
|
buffer.saveArray(asset.pixelShader->prog.loadDef.program, asset.pixelShader->prog.loadDef.programSize);
|
|
|
|
Utils::Stream::ClearPointer(&dest->prog.loadDef.program);
|
2017-03-26 20:44:37 -04:00
|
|
|
}
|
|
|
|
|
2022-12-11 12:54:24 -05:00
|
|
|
Utils::IO::WriteFile(path, buffer.toBuffer());
|
2017-03-26 20:44:37 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
if (type == Game::ASSET_TYPE_TECHNIQUE_SET)
|
|
|
|
{
|
|
|
|
Utils::IO::CreateDir("userraw/techsets");
|
2017-05-01 07:08:34 -04:00
|
|
|
Utils::Stream buffer(0x1000);
|
2023-01-06 07:51:41 -05:00
|
|
|
auto* dest = buffer.dest<Game::MaterialTechniqueSet>();
|
2017-05-01 07:08:34 -04:00
|
|
|
buffer.save(asset.techniqueSet);
|
2017-03-26 20:44:37 -04:00
|
|
|
|
|
|
|
if (asset.techniqueSet->name)
|
|
|
|
{
|
2017-05-01 07:08:34 -04:00
|
|
|
buffer.saveString(asset.techniqueSet->name);
|
2017-03-26 20:44:37 -04:00
|
|
|
Utils::Stream::ClearPointer(&dest->name);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (int i = 0; i < ARRAYSIZE(Game::MaterialTechniqueSet::techniques); ++i)
|
|
|
|
{
|
2023-03-09 20:03:50 -05:00
|
|
|
auto* technique = asset.techniqueSet->techniques[i];
|
2017-03-26 20:44:37 -04:00
|
|
|
|
|
|
|
if (technique)
|
|
|
|
{
|
2022-06-12 18:02:20 -04:00
|
|
|
// Size-check is obsolete, as the structure is dynamic
|
|
|
|
buffer.align(Utils::Stream::ALIGN_4);
|
2017-03-26 20:44:37 -04:00
|
|
|
|
2023-03-09 20:03:50 -05:00
|
|
|
auto* destTechnique = buffer.dest<Game::MaterialTechnique>();
|
2022-06-12 18:02:20 -04:00
|
|
|
buffer.save(technique, 8);
|
2017-03-26 20:44:37 -04:00
|
|
|
|
2022-06-12 18:02:20 -04:00
|
|
|
// Save_MaterialPassArray
|
2023-03-09 20:03:50 -05:00
|
|
|
auto* destPasses = buffer.dest<Game::MaterialPass>();
|
2022-06-12 18:02:20 -04:00
|
|
|
buffer.saveArray(technique->passArray, technique->passCount);
|
2017-03-26 20:44:37 -04:00
|
|
|
|
2022-06-12 18:02:20 -04:00
|
|
|
for (std::uint16_t j = 0; j < technique->passCount; ++j)
|
|
|
|
{
|
|
|
|
AssertSize(Game::MaterialPass, 20);
|
2017-03-26 20:44:37 -04:00
|
|
|
|
2022-06-12 18:02:20 -04:00
|
|
|
Game::MaterialPass* destPass = &destPasses[j];
|
|
|
|
Game::MaterialPass* pass = &technique->passArray[j];
|
2017-03-26 20:44:37 -04:00
|
|
|
|
2022-06-12 18:02:20 -04:00
|
|
|
if (pass->vertexDecl)
|
|
|
|
{
|
2017-03-26 20:44:37 -04:00
|
|
|
|
|
|
|
}
|
|
|
|
|
2022-06-12 18:02:20 -04:00
|
|
|
if (pass->args)
|
2017-03-26 20:44:37 -04:00
|
|
|
{
|
2022-06-12 18:02:20 -04:00
|
|
|
buffer.align(Utils::Stream::ALIGN_4);
|
|
|
|
buffer.saveArray(pass->args, pass->perPrimArgCount + pass->perObjArgCount + pass->stableArgCount);
|
|
|
|
Utils::Stream::ClearPointer(&destPass->args);
|
2017-03-26 20:44:37 -04:00
|
|
|
}
|
2022-06-12 18:02:20 -04:00
|
|
|
}
|
2017-03-26 20:44:37 -04:00
|
|
|
|
2022-06-12 18:02:20 -04:00
|
|
|
if (technique->name)
|
|
|
|
{
|
|
|
|
buffer.saveString(technique->name);
|
|
|
|
Utils::Stream::ClearPointer(&destTechnique->name);
|
2017-03-26 20:44:37 -04:00
|
|
|
}
|
2022-06-12 18:02:20 -04:00
|
|
|
|
|
|
|
Utils::Stream::ClearPointer(&dest->techniques[i]);
|
2017-03-26 20:44:37 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2022-06-12 18:02:20 -04:00
|
|
|
for (const auto& fastFile : fastFiles)
|
2017-03-26 20:44:37 -04:00
|
|
|
{
|
2022-06-12 18:02:20 -04:00
|
|
|
if (!Game::DB_IsZoneLoaded(fastFile.data()))
|
2017-03-26 20:44:37 -04:00
|
|
|
{
|
|
|
|
Game::XZoneInfo info;
|
2022-06-12 18:02:20 -04:00
|
|
|
info.name = fastFile.data();
|
2017-03-26 20:44:37 -04:00
|
|
|
info.allocFlags = 0x20;
|
|
|
|
info.freeFlags = 0;
|
|
|
|
|
|
|
|
Game::DB_LoadXAssets(&info, 1, true);
|
|
|
|
}
|
|
|
|
|
|
|
|
// unload the fastfiles so we don't run out of memory or asset pools
|
|
|
|
if (count % 5)
|
|
|
|
{
|
|
|
|
Game::XZoneInfo info;
|
|
|
|
info.name = nullptr;
|
|
|
|
info.allocFlags = 0x0;
|
|
|
|
info.freeFlags = 0x20;
|
|
|
|
Game::DB_LoadXAssets(&info, 1, true);
|
|
|
|
}
|
|
|
|
|
|
|
|
count++;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2023-02-01 05:10:24 -05:00
|
|
|
#ifdef DEBUG_MAT_LOG
|
2018-12-17 08:29:18 -05:00
|
|
|
AssetHandler::OnLoad([](Game::XAssetType type, Game::XAssetHeader asset, const std::string& /*name*/, bool* /*restrict*/)
|
2017-03-27 21:25:58 -04:00
|
|
|
{
|
|
|
|
if (type == Game::XAssetType::ASSET_TYPE_GFXWORLD)
|
|
|
|
{
|
2017-03-28 08:09:24 -04:00
|
|
|
std::string buffer;
|
|
|
|
|
|
|
|
for (unsigned int i = 0; i < asset.gfxWorld->dpvs.staticSurfaceCount; ++i)
|
2017-03-27 21:25:58 -04:00
|
|
|
{
|
2018-05-09 08:33:52 -04:00
|
|
|
buffer.append(Utils::String::VA("%s\n", asset.gfxWorld->dpvs.surfaces[asset.gfxWorld->dpvs.sortedSurfIndex[i]].material->info.name));
|
2017-03-27 21:25:58 -04:00
|
|
|
}
|
|
|
|
|
2017-03-28 08:09:24 -04:00
|
|
|
Utils::IO::WriteFile("userraw/logs/matlog.txt", buffer);
|
2017-03-27 21:25:58 -04:00
|
|
|
}
|
|
|
|
});
|
2022-03-20 05:05:45 -04:00
|
|
|
#endif
|
2017-03-27 21:25:58 -04:00
|
|
|
|
2016-10-19 03:01:01 -04:00
|
|
|
// Debug patches
|
|
|
|
#ifdef DEBUG
|
|
|
|
// ui_debugMode 1
|
|
|
|
//Utils::Hook::Set<bool>(0x6312E0, true);
|
|
|
|
|
|
|
|
// fs_debug 1
|
|
|
|
Utils::Hook::Set<bool>(0x643172, true);
|
|
|
|
|
|
|
|
// developer 2
|
|
|
|
Utils::Hook::Set<BYTE>(0x4FA425, 2);
|
|
|
|
Utils::Hook::Set<BYTE>(0x51B087, 2);
|
|
|
|
Utils::Hook::Set<BYTE>(0x60AE13, 2);
|
|
|
|
|
|
|
|
// developer_Script 1
|
|
|
|
Utils::Hook::Set<bool>(0x60AE2B, true);
|
|
|
|
|
|
|
|
// Disable cheat protection for dvars
|
2022-07-01 13:08:35 -04:00
|
|
|
Utils::Hook::Set<BYTE>(0x646515, 0xEB); // Dvar_IsCheatProtected
|
2017-01-18 10:04:24 -05:00
|
|
|
#else
|
|
|
|
// Remove missing tag message
|
|
|
|
Utils::Hook::Nop(0x4EBF1A, 5);
|
2016-10-19 03:01:01 -04:00
|
|
|
#endif
|
2017-04-13 16:04:01 -04:00
|
|
|
|
2017-06-14 06:06:04 -04:00
|
|
|
if (Flags::HasFlag("nointro"))
|
2017-04-13 16:04:01 -04:00
|
|
|
{
|
|
|
|
Utils::Hook::Set<BYTE>(0x60BECF, 0xEB);
|
|
|
|
}
|
2016-10-19 03:01:01 -04:00
|
|
|
}
|
|
|
|
|
2016-11-20 08:09:07 -05:00
|
|
|
bool QuickPatch::unitTest()
|
2016-10-19 03:01:01 -04:00
|
|
|
{
|
|
|
|
uint32_t randIntCount = 4'000'000;
|
2022-06-22 04:58:51 -04:00
|
|
|
Logger::Debug("Generating {} random integers...", randIntCount);
|
2016-10-19 03:01:01 -04:00
|
|
|
|
2022-06-13 11:32:45 -04:00
|
|
|
const auto startTime = std::chrono::high_resolution_clock::now();
|
2016-10-19 03:01:01 -04:00
|
|
|
|
|
|
|
for (uint32_t i = 0; i < randIntCount; ++i)
|
|
|
|
{
|
|
|
|
Utils::Cryptography::Rand::GenerateInt();
|
|
|
|
}
|
|
|
|
|
|
|
|
auto duration = std::chrono::duration_cast<std::chrono::milliseconds>(std::chrono::high_resolution_clock::now() - startTime).count();
|
2022-06-22 04:58:51 -04:00
|
|
|
Logger::Debug("took {}ms", duration);
|
2016-10-19 03:01:01 -04:00
|
|
|
|
2022-06-13 11:32:45 -04:00
|
|
|
Logger::Debug("Testing ZLib compression...");
|
2016-10-19 03:01:01 -04:00
|
|
|
|
2017-01-06 09:27:35 -05:00
|
|
|
std::string test = Utils::String::VA("%c", Utils::Cryptography::Rand::GenerateInt());
|
2016-10-19 03:01:01 -04:00
|
|
|
|
|
|
|
for (int i = 0; i < 21; ++i)
|
|
|
|
{
|
2017-07-03 10:32:10 -04:00
|
|
|
std::string compressed = Utils::Compression::ZLib::Compress(test);
|
|
|
|
std::string decompressed = Utils::Compression::ZLib::Decompress(compressed);
|
2016-10-19 03:01:01 -04:00
|
|
|
|
|
|
|
if (test != decompressed)
|
|
|
|
{
|
2022-06-13 11:32:45 -04:00
|
|
|
Logger::PrintError(Game::CON_CHANNEL_ERROR, "Compressing {} bytes and decompressing failed!\n", test.size());
|
2016-10-19 03:01:01 -04:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2022-06-13 11:32:45 -04:00
|
|
|
const auto size = test.size();
|
2016-10-19 03:01:01 -04:00
|
|
|
for (unsigned int j = 0; j < size; ++j)
|
|
|
|
{
|
2017-01-06 09:27:35 -05:00
|
|
|
test.append(Utils::String::VA("%c", Utils::Cryptography::Rand::GenerateInt()));
|
2016-10-19 03:01:01 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-06-13 11:32:45 -04:00
|
|
|
Logger::Debug("Success");
|
2017-06-13 09:35:12 -04:00
|
|
|
|
2022-06-13 11:32:45 -04:00
|
|
|
Logger::Debug("Testing trimming...");
|
2017-06-13 09:35:12 -04:00
|
|
|
std::string trim1 = " 1 ";
|
|
|
|
std::string trim2 = " 1";
|
|
|
|
std::string trim3 = "1 ";
|
|
|
|
|
|
|
|
Utils::String::Trim(trim1);
|
|
|
|
Utils::String::LTrim(trim2);
|
|
|
|
Utils::String::RTrim(trim3);
|
|
|
|
|
|
|
|
if (trim1 != "1") return false;
|
|
|
|
if (trim2 != "1") return false;
|
|
|
|
if (trim3 != "1") return false;
|
|
|
|
|
2022-06-13 11:32:45 -04:00
|
|
|
Logger::Debug("Success");
|
2016-10-19 03:01:01 -04:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|