2022-02-21 14:39:44 -05:00
|
|
|
#include <std_include.hpp>
|
|
|
|
#include "loader/component_loader.hpp"
|
|
|
|
|
2022-02-28 14:36:32 -05:00
|
|
|
#include "dvars.hpp"
|
2022-03-01 15:38:16 -05:00
|
|
|
#include "version.h"
|
|
|
|
#include "command.hpp"
|
|
|
|
#include "console.hpp"
|
|
|
|
#include "network.hpp"
|
|
|
|
#include "scheduler.hpp"
|
2022-03-03 07:56:46 -05:00
|
|
|
#include "filesystem.hpp"
|
2022-02-28 14:36:32 -05:00
|
|
|
|
2022-02-21 14:39:44 -05:00
|
|
|
#include "game/game.hpp"
|
|
|
|
#include "game/dvars.hpp"
|
|
|
|
|
|
|
|
#include <utils/hook.hpp>
|
|
|
|
#include <utils/string.hpp>
|
2022-03-01 15:38:16 -05:00
|
|
|
#include <utils/flags.hpp>
|
2022-02-26 10:10:35 -05:00
|
|
|
|
2022-02-21 14:39:44 -05:00
|
|
|
namespace patches
|
|
|
|
{
|
|
|
|
namespace
|
|
|
|
{
|
2022-02-26 10:10:35 -05:00
|
|
|
const char* live_get_local_client_name()
|
|
|
|
{
|
|
|
|
return game::Dvar_FindVar("name")->current.string;
|
|
|
|
}
|
|
|
|
|
|
|
|
utils::hook::detour sv_kick_client_num_hook;
|
|
|
|
|
|
|
|
void sv_kick_client_num(const int client_num, const char* reason)
|
|
|
|
{
|
|
|
|
// Don't kick bot to equalize team balance.
|
|
|
|
if (reason == "EXE_PLAYERKICKED_BOT_BALANCE"s)
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
2022-02-28 20:39:51 -05:00
|
|
|
return sv_kick_client_num_hook.invoke<void>(client_num, reason);
|
2022-02-21 14:39:44 -05:00
|
|
|
}
|
|
|
|
|
2022-02-26 10:10:35 -05:00
|
|
|
std::string get_login_username()
|
|
|
|
{
|
|
|
|
char username[UNLEN + 1];
|
|
|
|
DWORD username_len = UNLEN + 1;
|
|
|
|
if (!GetUserNameA(username, &username_len))
|
|
|
|
{
|
|
|
|
return "Unknown Soldier";
|
|
|
|
}
|
|
|
|
|
|
|
|
return std::string{ username, username_len - 1 };
|
|
|
|
}
|
|
|
|
|
|
|
|
utils::hook::detour com_register_dvars_hook;
|
|
|
|
|
|
|
|
void com_register_dvars_stub()
|
|
|
|
{
|
|
|
|
if (game::environment::is_mp())
|
|
|
|
{
|
|
|
|
// Make name save
|
|
|
|
dvars::register_string("name", get_login_username().data(), game::DVAR_FLAG_SAVED, true);
|
|
|
|
|
|
|
|
// Disable data validation error popup
|
|
|
|
dvars::register_int("data_validation_allow_drop", 0, 0, 0, game::DVAR_FLAG_NONE, true);
|
|
|
|
}
|
|
|
|
|
|
|
|
return com_register_dvars_hook.invoke<void>();
|
|
|
|
}
|
|
|
|
|
|
|
|
int is_item_unlocked()
|
|
|
|
{
|
|
|
|
return 0; // 0 == yes
|
|
|
|
}
|
|
|
|
|
2022-02-27 14:35:21 -05:00
|
|
|
void set_client_dvar_from_server_stub(void* a1, void* a2, const char* dvar, const char* value)
|
|
|
|
{
|
|
|
|
if (dvar == "cg_fov"s)
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// CG_SetClientDvarFromServer
|
2022-03-03 07:56:46 -05:00
|
|
|
utils::hook::invoke<void>(0x140236120, a1, a2, dvar, value);
|
2022-02-27 14:35:21 -05:00
|
|
|
}
|
|
|
|
|
2022-03-03 07:56:46 -05:00
|
|
|
const char* db_read_raw_file_stub(const char* filename, char* buf, const int size)
|
2022-02-26 10:10:35 -05:00
|
|
|
{
|
2022-03-03 07:56:46 -05:00
|
|
|
std::string file_name = filename;
|
|
|
|
if (file_name.find(".cfg") == std::string::npos)
|
|
|
|
{
|
|
|
|
file_name.append(".cfg");
|
|
|
|
}
|
2022-02-26 10:10:35 -05:00
|
|
|
|
2022-03-03 07:56:46 -05:00
|
|
|
const auto file = filesystem::file(file_name);
|
|
|
|
if (file.exists())
|
|
|
|
{
|
|
|
|
snprintf(buf, size, "%s\n", file.get_buffer().data());
|
|
|
|
return buf;
|
|
|
|
}
|
|
|
|
|
|
|
|
// DB_ReadRawFile
|
|
|
|
return utils::hook::invoke<const char*>(SELECT_VALUE(0x1401CD4F0, 0x1402BEF10), filename, buf, size);
|
|
|
|
}
|
2022-02-26 10:10:35 -05:00
|
|
|
|
|
|
|
void bsp_sys_error_stub(const char* error, const char* arg1)
|
|
|
|
{
|
|
|
|
if (game::environment::is_dedi())
|
|
|
|
{
|
|
|
|
game::Sys_Error(error, arg1);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
scheduler::once([]()
|
2022-02-27 01:36:54 -05:00
|
|
|
{
|
|
|
|
command::execute("reconnect");
|
|
|
|
}, scheduler::pipeline::main, 1s);
|
2022-02-26 10:10:35 -05:00
|
|
|
game::Com_Error(game::ERR_DROP, error, arg1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
utils::hook::detour cmd_lui_notify_server_hook;
|
|
|
|
void cmd_lui_notify_server_stub(game::mp::gentity_s* ent)
|
|
|
|
{
|
|
|
|
command::params_sv params{};
|
|
|
|
const auto menu_id = atoi(params.get(1));
|
|
|
|
const auto client = &game::mp::svs_clients[ent->s.entityNum];
|
|
|
|
|
|
|
|
// 22 => "end_game"
|
|
|
|
if (menu_id == 22 && client->header.remoteAddress.type != game::NA_LOOPBACK)
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
cmd_lui_notify_server_hook.invoke<void>(ent);
|
|
|
|
}
|
|
|
|
|
|
|
|
void sv_execute_client_message_stub(game::mp::client_t* client, game::msg_t* msg)
|
|
|
|
{
|
|
|
|
if (client->reliableAcknowledge < 0)
|
|
|
|
{
|
|
|
|
client->reliableAcknowledge = client->reliableSequence;
|
|
|
|
console::info("Negative reliableAcknowledge from %s - cl->reliableSequence is %i, reliableAcknowledge is %i\n",
|
|
|
|
client->name, client->reliableSequence, client->reliableAcknowledge);
|
|
|
|
network::send(client->header.remoteAddress, "error", "EXE_LOSTRELIABLECOMMANDS", '\n');
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2022-03-03 07:56:46 -05:00
|
|
|
utils::hook::invoke<void>(0x140481A00, client, msg);
|
2022-02-26 10:10:35 -05:00
|
|
|
}
|
2022-02-28 15:49:10 -05:00
|
|
|
|
|
|
|
void aim_assist_add_to_target_list(void* a1, void* a2)
|
|
|
|
{
|
|
|
|
if (!dvars::aimassist_enabled->current.enabled)
|
|
|
|
return;
|
|
|
|
|
|
|
|
game::AimAssist_AddToTargetList(a1, a2);
|
|
|
|
}
|
2022-02-21 14:39:44 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
class component final : public component_interface
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
void post_unpack() override
|
|
|
|
{
|
2022-02-26 10:10:35 -05:00
|
|
|
// Register dvars
|
|
|
|
com_register_dvars_hook.create(SELECT_VALUE(0x140351B80, 0x1400D9320), &com_register_dvars_stub); // H1(1.4)
|
|
|
|
|
2022-02-21 14:39:44 -05:00
|
|
|
// Unlock fps in main menu
|
|
|
|
utils::hook::set<BYTE>(SELECT_VALUE(0x14018D47B, 0x14025B86B), 0xEB); // H1(1.4)
|
|
|
|
|
2022-02-28 20:39:51 -05:00
|
|
|
if (!game::environment::is_dedi())
|
2022-02-26 14:12:05 -05:00
|
|
|
{
|
2022-02-28 20:39:51 -05:00
|
|
|
// Fix mouse lag
|
|
|
|
utils::hook::nop(SELECT_VALUE(0x1403E3C05, 0x1404DB1AF), 6);
|
|
|
|
scheduler::loop([]()
|
|
|
|
{
|
|
|
|
SetThreadExecutionState(ES_DISPLAY_REQUIRED);
|
|
|
|
}, scheduler::pipeline::main);
|
|
|
|
}
|
2022-02-21 14:39:44 -05:00
|
|
|
|
|
|
|
// Make cg_fov and cg_fovscale saved dvars
|
2022-02-28 14:36:32 -05:00
|
|
|
dvars::override::register_float("cg_fov", 65.f, 40.f, 200.f, game::DvarFlags::DVAR_FLAG_SAVED);
|
|
|
|
dvars::override::register_float("cg_fovScale", 1.f, 0.1f, 2.f, game::DvarFlags::DVAR_FLAG_SAVED);
|
2022-02-28 19:16:28 -05:00
|
|
|
|
|
|
|
// Allow kbam input when gamepad is enabled
|
|
|
|
utils::hook::nop(SELECT_VALUE(0x14018797E, 0x14024EF60), 2);
|
|
|
|
utils::hook::nop(SELECT_VALUE(0x1401856DC, 0x14024C6B0), 6);
|
2022-02-26 10:10:35 -05:00
|
|
|
|
2022-03-03 07:56:46 -05:00
|
|
|
// Allow executing custom cfg files with the "exec" command
|
|
|
|
utils::hook::call(SELECT_VALUE(0x140343855, 0x140403E28), db_read_raw_file_stub);
|
|
|
|
|
2022-03-01 09:40:37 -05:00
|
|
|
if (!game::environment::is_sp())
|
2022-02-26 10:10:35 -05:00
|
|
|
{
|
|
|
|
patch_mp();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void patch_mp()
|
|
|
|
{
|
|
|
|
// Use name dvar
|
2022-02-26 17:53:05 -05:00
|
|
|
utils::hook::jump(0x14050FF90, &live_get_local_client_name); // H1(1.4)
|
2022-02-26 10:10:35 -05:00
|
|
|
|
|
|
|
// Patch SV_KickClientNum
|
|
|
|
sv_kick_client_num_hook.create(0x14047ED00, &sv_kick_client_num); // H1(1.4)
|
|
|
|
|
|
|
|
// block changing name in-game
|
|
|
|
utils::hook::set<uint8_t>(0x14047FC90, 0xC3); // H1(1.4)
|
|
|
|
|
|
|
|
// patch "Couldn't find the bsp for this map." error to not be fatal in mp
|
|
|
|
utils::hook::call(0x1402BA26B, bsp_sys_error_stub); // H1(1.4)
|
|
|
|
|
|
|
|
// client side aim assist dvar
|
2022-02-28 15:49:10 -05:00
|
|
|
dvars::aimassist_enabled = dvars::register_bool("aimassist_enabled", true,
|
|
|
|
game::DvarFlags::DVAR_FLAG_SAVED,
|
|
|
|
true);
|
|
|
|
utils::hook::call(0x14009EE9E, aim_assist_add_to_target_list);
|
2022-02-26 10:10:35 -05:00
|
|
|
|
|
|
|
// unlock all items
|
|
|
|
utils::hook::jump(0x140413E60, is_item_unlocked); // LiveStorage_IsItemUnlockedFromTable_LocalClient H1(1.4)
|
|
|
|
utils::hook::jump(0x140413860, is_item_unlocked); // LiveStorage_IsItemUnlockedFromTable H1(1.4)
|
|
|
|
utils::hook::jump(0x140412B70, is_item_unlocked); // idk ( unlocks loot etc ) H1(1.4)
|
|
|
|
|
|
|
|
// isProfanity
|
|
|
|
utils::hook::set(0x1402877D0, 0xC3C033); // MAY BE WRONG H1(1.4)
|
|
|
|
|
|
|
|
// disable emblems
|
2022-02-28 14:36:32 -05:00
|
|
|
dvars::override::register_int("emblems_active", 0, 0, 0, game::DVAR_FLAG_NONE);
|
|
|
|
utils::hook::set<uint8_t>(0x140479590, 0xC3); // don't register commands
|
2022-02-26 10:10:35 -05:00
|
|
|
|
|
|
|
// disable elite_clan
|
|
|
|
dvars::override::register_int("elite_clan_active", 0, 0, 0, game::DVAR_FLAG_NONE);
|
|
|
|
utils::hook::set<uint8_t>(0x140585680, 0xC3); // don't register commands H1(1.4)
|
|
|
|
|
2022-02-27 01:36:54 -05:00
|
|
|
// disable codPointStore
|
2022-02-28 14:36:32 -05:00
|
|
|
dvars::override::register_int("codPointStore_enabled", 0, 0, 0, game::DVAR_FLAG_NONE);
|
2022-02-26 10:10:35 -05:00
|
|
|
|
|
|
|
// don't register every replicated dvar as a network dvar
|
|
|
|
utils::hook::nop(0x14039E58E, 5); // dvar_foreach H1(1.4)
|
|
|
|
|
|
|
|
// patch "Server is different version" to show the server client version
|
|
|
|
utils::hook::inject(0x140480952, VERSION); // H1(1.4)
|
|
|
|
|
2022-03-01 09:40:37 -05:00
|
|
|
// prevent servers overriding our fov
|
|
|
|
utils::hook::call(0x14023279E, set_client_dvar_from_server_stub);
|
|
|
|
utils::hook::nop(0x1400DAF69, 5);
|
|
|
|
utils::hook::nop(0x140190C16, 5);
|
|
|
|
utils::hook::set<uint8_t>(0x14021D22A, 0xEB);
|
2022-02-26 10:10:35 -05:00
|
|
|
|
|
|
|
// unlock safeArea_*
|
|
|
|
utils::hook::jump(0x1402624F5, 0x140262503); // H1(1.4)
|
|
|
|
utils::hook::jump(0x14026251C, 0x140262547); // H1(1.4)
|
2022-02-28 14:36:32 -05:00
|
|
|
dvars::override::register_int("safeArea_adjusted_horizontal", 1, 0, 1, game::DVAR_FLAG_SAVED);
|
|
|
|
dvars::override::register_int("safeArea_adjusted_vertical", 1, 0, 1, game::DVAR_FLAG_SAVED);
|
|
|
|
dvars::override::register_int("safeArea_horizontal", 1, 0, 1, game::DVAR_FLAG_SAVED);
|
|
|
|
dvars::override::register_int("safeArea_vertical", 1, 0, 1, game::DVAR_FLAG_SAVED);
|
2022-02-26 10:10:35 -05:00
|
|
|
|
|
|
|
// move chat position on the screen above menu splashes
|
2022-02-28 14:36:32 -05:00
|
|
|
dvars::override::register_vec2("cg_hudChatPosition", 5, 170, 0, 640, game::DVAR_FLAG_SAVED);
|
2022-02-26 10:10:35 -05:00
|
|
|
|
|
|
|
// allow servers to check for new packages more often
|
2022-02-28 14:36:32 -05:00
|
|
|
dvars::override::register_int("sv_network_fps", 1000, 20, 1000, game::DVAR_FLAG_SAVED);
|
2022-02-26 10:10:35 -05:00
|
|
|
|
|
|
|
// Massively increate timeouts
|
2022-02-28 14:36:32 -05:00
|
|
|
dvars::override::register_int("cl_timeout", 90, 90, 1800, game::DVAR_FLAG_NONE); // Seems unused
|
|
|
|
dvars::override::register_int("sv_timeout", 90, 90, 1800, game::DVAR_FLAG_NONE); // 30 - 0 - 1800
|
|
|
|
dvars::override::register_int("cl_connectTimeout", 120, 120, 1800, game::DVAR_FLAG_NONE); // Seems unused
|
|
|
|
dvars::override::register_int("sv_connectTimeout", 120, 120, 1800, game::DVAR_FLAG_NONE); // 60 - 0 - 1800
|
|
|
|
|
|
|
|
dvars::register_int("scr_game_spectatetype", 1, 0, 99, game::DVAR_FLAG_REPLICATED);
|
2022-02-26 10:10:35 -05:00
|
|
|
|
2022-02-28 14:36:32 -05:00
|
|
|
dvars::override::register_int("com_maxfps", 0, 10, 1000, game::DVAR_FLAG_SAVED);
|
2022-02-26 10:10:35 -05:00
|
|
|
|
|
|
|
// Prevent clients from ending the game as non host by sending 'end_game' lui notification
|
2022-02-28 09:25:30 -05:00
|
|
|
// cmd_lui_notify_server_hook.create(0x140335A70, cmd_lui_notify_server_stub); // H1(1.4)
|
2022-02-26 10:10:35 -05:00
|
|
|
|
|
|
|
// Prevent clients from sending invalid reliableAcknowledge
|
2022-02-27 10:10:04 -05:00
|
|
|
// utils::hook::call(0x1404899C6, sv_execute_client_message_stub); // H1(1.4)
|
2022-03-01 15:38:16 -05:00
|
|
|
|
|
|
|
// "fix" for rare 'Out of memory error' error
|
|
|
|
if (utils::flags::has_flag("memoryfix"))
|
|
|
|
{
|
|
|
|
utils::hook::jump(0x140578BE0, malloc);
|
|
|
|
utils::hook::jump(0x140578B00, _aligned_malloc);
|
|
|
|
utils::hook::jump(0x140578C40, free);
|
|
|
|
utils::hook::jump(0x140578D30, realloc);
|
|
|
|
utils::hook::jump(0x140578B60, _aligned_realloc);
|
|
|
|
}
|
2022-03-01 19:00:27 -05:00
|
|
|
|
|
|
|
// Change default hostname and make it replicated
|
|
|
|
dvars::override::register_string("sv_hostname", "^2H1-Mod^7 Default Server", game::DVAR_FLAG_REPLICATED);
|
2022-02-21 14:39:44 -05:00
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2022-02-23 15:23:00 -05:00
|
|
|
REGISTER_COMPONENT(patches::component)
|