2022-02-27 07:53:44 -05:00
# include <STDInclude.hpp>
2022-12-26 07:07:24 -05:00
# include "Weapon.hpp"
2022-07-06 11:48:40 -04:00
# include "GSC/Script.hpp"
2017-01-19 16:23:59 -05:00
namespace Components
{
2022-12-06 09:42:09 -05:00
const Game : : dvar_t * Weapon : : BGWeaponOffHandFix ;
2022-07-02 12:30:35 -04:00
Game : : WeaponCompleteDef * Weapon : : LoadWeaponCompleteDef ( const char * name )
2017-01-19 16:23:59 -05:00
{
2022-07-02 12:30:35 -04:00
if ( auto * rawWeaponFile = Game : : BG_LoadWeaponCompleteDefInternal ( " mp " , name ) )
2017-01-19 16:23:59 -05:00
{
2022-07-02 12:30:35 -04:00
return rawWeaponFile ;
2017-01-19 16:23:59 -05:00
}
2022-07-02 12:30:35 -04:00
auto * zoneWeaponFile = Game : : DB_FindXAssetHeader ( Game : : ASSET_TYPE_WEAPON , name ) . weapon ;
return Game : : DB_IsXAssetDefault ( Game : : ASSET_TYPE_WEAPON , name ) ? nullptr : zoneWeaponFile ;
2017-01-19 16:23:59 -05:00
}
2017-02-02 18:00:08 -05:00
const char * Weapon : : GetWeaponConfigString ( int index )
{
2017-02-03 15:53:03 -05:00
if ( index > = ( 1200 + 2804 ) ) index + = ( 2939 - 2804 ) ;
return Game : : CL_GetConfigString ( index ) ;
2017-02-02 18:00:08 -05:00
}
void Weapon : : SaveRegisteredWeapons ( )
{
* reinterpret_cast < DWORD * > ( 0x1A86098 ) = 0 ;
if ( Game : : BG_GetNumWeapons ( ) > 1u )
{
2017-02-03 15:53:03 -05:00
for ( unsigned int i = 1 ; i < Game : : BG_GetNumWeapons ( ) ; + + i )
2017-02-02 18:00:08 -05:00
{
2017-06-14 06:06:04 -04:00
Game : : SV_SetConfigstring ( i + ( i > = 1200 ? 2939 : 2804 ) , Game : : BG_GetWeaponName ( i ) ) ;
2017-02-02 18:00:08 -05:00
}
}
2017-02-03 15:53:03 -05:00
}
int Weapon : : ParseWeaponConfigStrings ( )
{
Command : : ClientParams params ;
2022-03-17 14:50:20 -04:00
if ( params . size ( ) < = 1 )
return 0 ;
2022-04-12 08:34:51 -04:00
char * end ;
const auto * input = params . get ( 1 ) ;
auto index = std : : strtol ( input , & end , 10 ) ;
if ( input = = end )
{
2022-06-12 17:07:53 -04:00
Logger : : Warning ( Game : : CON_CHANNEL_DONT_FILTER , " {} is not a valid input \n Usage: {} <weapon index> \n " ,
2022-04-12 08:34:51 -04:00
input , params . get ( 0 ) ) ;
return 0 ;
}
2017-02-02 18:00:08 -05:00
2017-06-14 06:06:04 -04:00
if ( index > = 4139 )
2017-02-02 18:00:08 -05:00
{
2017-02-03 15:53:03 -05:00
index - = 2939 ;
}
2017-06-14 06:06:04 -04:00
else if ( index > 2804 & & index < = 2804 + 1200 )
2017-02-03 15:53:03 -05:00
{
index - = 2804 ;
}
else
{
return 0 ;
}
2022-04-12 08:34:51 -04:00
Game : : CG_SetupWeaponDef ( 0 , index ) ;
2017-02-03 15:53:03 -05:00
return 1 ;
}
__declspec ( naked ) void Weapon : : ParseConfigStrings ( )
{
__asm
{
push eax
pushad
push edi
call Weapon : : ParseWeaponConfigStrings
pop edi
mov [ esp + 20 h ] , eax
popad
pop eax
test eax , eax
jz continueParsing
retn
continueParsing :
push 592960 h
retn
2017-02-02 18:00:08 -05:00
}
}
2017-02-05 16:25:38 -05:00
int Weapon : : ClearConfigStrings ( void * dest , int value , int size )
{
memset ( Utils : : Hook : : Get < void * > ( 0x405B72 ) , value , MAX_CONFIGSTRINGS * 2 ) ;
return Utils : : Hook : : Call < int ( void * , int , int ) > ( 0x4C98D0 ) ( dest , value , size ) ; // Com_Memset
}
2017-02-03 04:26:19 -05:00
void Weapon : : PatchConfigStrings ( )
{
Utils : : Hook : : Set < DWORD > ( 0x4347A7 , MAX_CONFIGSTRINGS ) ;
Utils : : Hook : : Set < DWORD > ( 0x4982F4 , MAX_CONFIGSTRINGS ) ;
Utils : : Hook : : Set < DWORD > ( 0x4F88B6 , MAX_CONFIGSTRINGS ) ; // Save file
Utils : : Hook : : Set < DWORD > ( 0x5A1FA7 , MAX_CONFIGSTRINGS ) ;
Utils : : Hook : : Set < DWORD > ( 0x5A210D , MAX_CONFIGSTRINGS ) ; // Game state
Utils : : Hook : : Set < DWORD > ( 0x5A840E , MAX_CONFIGSTRINGS ) ;
Utils : : Hook : : Set < DWORD > ( 0x5A853C , MAX_CONFIGSTRINGS ) ;
Utils : : Hook : : Set < DWORD > ( 0x5AC392 , MAX_CONFIGSTRINGS ) ;
Utils : : Hook : : Set < DWORD > ( 0x5AC3F5 , MAX_CONFIGSTRINGS ) ;
Utils : : Hook : : Set < DWORD > ( 0x5AC542 , MAX_CONFIGSTRINGS ) ; // Game state
Utils : : Hook : : Set < DWORD > ( 0x5EADF0 , MAX_CONFIGSTRINGS ) ;
Utils : : Hook : : Set < DWORD > ( 0x625388 , MAX_CONFIGSTRINGS ) ;
Utils : : Hook : : Set < DWORD > ( 0x625516 , MAX_CONFIGSTRINGS ) ;
2017-02-03 15:53:03 -05:00
// Adjust weapon count index
// Actually this has to stay the way it is!
//Utils::Hook::Set<DWORD>(0x4EB7B3, MAX_CONFIGSTRINGS - 1);
//Utils::Hook::Set<DWORD>(0x5929BA, MAX_CONFIGSTRINGS - 1);
//Utils::Hook::Set<DWORD>(0x5E2FAA, MAX_CONFIGSTRINGS - 1);
2017-02-03 09:02:43 -05:00
static short configStrings [ MAX_CONFIGSTRINGS ] ;
2017-02-03 15:53:03 -05:00
ZeroMemory ( & configStrings , sizeof ( configStrings ) ) ;
2017-02-03 04:26:19 -05:00
Utils : : Hook : : Set ( 0x405B72 , configStrings ) ;
Utils : : Hook : : Set ( 0x468508 , configStrings ) ;
Utils : : Hook : : Set ( 0x47FD7C , configStrings ) ;
Utils : : Hook : : Set ( 0x49830E , configStrings ) ;
Utils : : Hook : : Set ( 0x498371 , configStrings ) ;
Utils : : Hook : : Set ( 0x4983D5 , configStrings ) ;
Utils : : Hook : : Set ( 0x4A74AD , configStrings ) ;
2017-02-03 09:02:43 -05:00
Utils : : Hook : : Set ( 0x4BAE7C , configStrings ) ;
Utils : : Hook : : Set ( 0x4BAEC3 , configStrings ) ;
2017-02-03 04:26:19 -05:00
Utils : : Hook : : Set ( 0x6252F5 , configStrings ) ;
Utils : : Hook : : Set ( 0x625372 , configStrings ) ;
Utils : : Hook : : Set ( 0x6253D3 , configStrings ) ;
Utils : : Hook : : Set ( 0x625480 , configStrings ) ;
Utils : : Hook : : Set ( 0x6254CB , configStrings ) ;
2017-02-05 12:21:09 -05:00
// This has nothing to do with configstrings
2017-02-03 15:53:03 -05:00
//Utils::Hook::Set(0x608095, configStrings[4139 - 0x16]);
//Utils::Hook::Set(0x6080BC, configStrings[4139 - 0x16]);
//Utils::Hook::Set(0x6082AC, configStrings[4139 - 0x16]);
//Utils::Hook::Set(0x6082B3, configStrings[4139 - 0x16]);
//Utils::Hook::Set(0x608856, configStrings[4139 - 0x14]);
2017-02-03 04:26:19 -05:00
// TODO: Check if all of these actually mark the end of the array
2017-02-05 12:21:09 -05:00
// Only 2 actually mark the end, the rest is header data or so
2017-02-05 05:22:08 -05:00
Utils : : Hook : : Set ( 0x405B8F , & configStrings [ ARRAYSIZE ( configStrings ) ] ) ;
//Utils::Hook::Set(0x459121, &configStrings[ARRAYSIZE(configStrings)]);
//Utils::Hook::Set(0x45A476, &configStrings[ARRAYSIZE(configStrings)]);
//Utils::Hook::Set(0x49FD56, &configStrings[ARRAYSIZE(configStrings)]);
Utils : : Hook : : Set ( 0x4A74C9 , & configStrings [ ARRAYSIZE ( configStrings ) ] ) ;
//Utils::Hook::Set(0x4C8196, &configStrings[ARRAYSIZE(configStrings)]);
//Utils::Hook::Set(0x4EBCE6, &configStrings[ARRAYSIZE(configStrings)]);
//Utils::Hook::Set(0x4F36D6, &configStrings[ARRAYSIZE(configStrings)]);
//Utils::Hook::Set(0x60807C, &configStrings[ARRAYSIZE(configStrings)]);
//Utils::Hook::Set(0x6080A9, &configStrings[ARRAYSIZE(configStrings)]);
//Utils::Hook::Set(0x6080D0, &configStrings[ARRAYSIZE(configStrings)]);
//Utils::Hook::Set(0x6081C4, &configStrings[ARRAYSIZE(configStrings)]);
//Utils::Hook::Set(0x608211, &configStrings[ARRAYSIZE(configStrings)]);
//Utils::Hook::Set(0x608274, &configStrings[ARRAYSIZE(configStrings)]);
//Utils::Hook::Set(0x6083D6, &configStrings[ARRAYSIZE(configStrings)]);
//Utils::Hook::Set(0x60848E, &configStrings[ARRAYSIZE(configStrings)]);
2017-02-03 09:02:43 -05:00
2017-02-05 16:25:38 -05:00
Utils : : Hook ( 0x405BBE , Weapon : : ClearConfigStrings , HOOK_CALL ) . install ( ) - > quick ( ) ;
2017-02-03 15:53:03 -05:00
Utils : : Hook ( 0x593CA4 , Weapon : : ParseConfigStrings , HOOK_CALL ) . install ( ) - > quick ( ) ;
Utils : : Hook ( 0x4BD52D , Weapon : : GetWeaponConfigString , HOOK_CALL ) . install ( ) - > quick ( ) ;
Utils : : Hook ( 0x45D170 , Weapon : : SaveRegisteredWeapons , HOOK_JUMP ) . install ( ) - > quick ( ) ;
2017-02-03 09:02:43 -05:00
2017-02-03 15:53:03 -05:00
// Patch game state
// The structure below is our own implementation of the gameState_t structure
2017-02-05 05:22:08 -05:00
static struct newGameState_t
2017-02-03 09:02:43 -05:00
{
2017-02-03 15:53:03 -05:00
int stringOffsets [ MAX_CONFIGSTRINGS ] ;
char stringData [ 131072 ] ; // MAX_GAMESTATE_CHARS
int dataCount ;
} gameState ;
ZeroMemory ( & gameState , sizeof ( gameState ) ) ;
Utils : : Hook : : Set < DWORD > ( 0x44A333 , sizeof ( gameState ) ) ;
Utils : : Hook : : Set < DWORD > ( 0x5A1F56 , sizeof ( gameState ) ) ;
Utils : : Hook : : Set < DWORD > ( 0x5A2043 , sizeof ( gameState ) ) ;
Utils : : Hook : : Set < DWORD > ( 0x5A2053 , sizeof ( gameState . stringOffsets ) ) ;
Utils : : Hook : : Set < DWORD > ( 0x5A2098 , sizeof ( gameState . stringOffsets ) ) ;
Utils : : Hook : : Set < DWORD > ( 0x5AC32C , sizeof ( gameState . stringOffsets ) ) ;
Utils : : Hook : : Set ( 0x4235AC , & gameState . stringOffsets ) ;
Utils : : Hook : : Set ( 0x434783 , & gameState . stringOffsets ) ;
Utils : : Hook : : Set ( 0x44A339 , & gameState . stringOffsets ) ;
Utils : : Hook : : Set ( 0x44ADB7 , & gameState . stringOffsets ) ;
Utils : : Hook : : Set ( 0x5A1FE6 , & gameState . stringOffsets ) ;
Utils : : Hook : : Set ( 0x5A2048 , & gameState . stringOffsets ) ;
Utils : : Hook : : Set ( 0x5A205A , & gameState . stringOffsets ) ;
Utils : : Hook : : Set ( 0x5A2077 , & gameState . stringOffsets ) ;
Utils : : Hook : : Set ( 0x5A2091 , & gameState . stringOffsets ) ;
Utils : : Hook : : Set ( 0x5A20D7 , & gameState . stringOffsets ) ;
Utils : : Hook : : Set ( 0x5A83FF , & gameState . stringOffsets ) ;
Utils : : Hook : : Set ( 0x5A84B0 , & gameState . stringOffsets ) ;
Utils : : Hook : : Set ( 0x5A84E5 , & gameState . stringOffsets ) ;
Utils : : Hook : : Set ( 0x5AC333 , & gameState . stringOffsets ) ;
Utils : : Hook : : Set ( 0x5AC44A , & gameState . stringOffsets ) ;
Utils : : Hook : : Set ( 0x5AC4F3 , & gameState . stringOffsets ) ;
Utils : : Hook : : Set ( 0x5AC57A , & gameState . stringOffsets ) ;
Utils : : Hook : : Set ( 0x4235B7 , & gameState . stringData ) ;
Utils : : Hook : : Set ( 0x43478D , & gameState . stringData ) ;
Utils : : Hook : : Set ( 0x44ADBC , & gameState . stringData ) ;
Utils : : Hook : : Set ( 0x5A1FEF , & gameState . stringData ) ;
Utils : : Hook : : Set ( 0x5A20E6 , & gameState . stringData ) ;
Utils : : Hook : : Set ( 0x5AC457 , & gameState . stringData ) ;
Utils : : Hook : : Set ( 0x5AC502 , & gameState . stringData ) ;
Utils : : Hook : : Set ( 0x5AC586 , & gameState . stringData ) ;
Utils : : Hook : : Set ( 0x5A2071 , & gameState . dataCount ) ;
Utils : : Hook : : Set ( 0x5A20CD , & gameState . dataCount ) ;
Utils : : Hook : : Set ( 0x5A20DC , & gameState . dataCount ) ;
Utils : : Hook : : Set ( 0x5A20F3 , & gameState . dataCount ) ;
Utils : : Hook : : Set ( 0x5A2104 , & gameState . dataCount ) ;
Utils : : Hook : : Set ( 0x5AC33F , & gameState . dataCount ) ;
Utils : : Hook : : Set ( 0x5AC43B , & gameState . dataCount ) ;
Utils : : Hook : : Set ( 0x5AC450 , & gameState . dataCount ) ;
Utils : : Hook : : Set ( 0x5AC463 , & gameState . dataCount ) ;
Utils : : Hook : : Set ( 0x5AC471 , & gameState . dataCount ) ;
Utils : : Hook : : Set ( 0x5AC4C3 , & gameState . dataCount ) ;
Utils : : Hook : : Set ( 0x5AC4E8 , & gameState . dataCount ) ;
Utils : : Hook : : Set ( 0x5AC4F8 , & gameState . dataCount ) ;
Utils : : Hook : : Set ( 0x5AC50F , & gameState . dataCount ) ;
Utils : : Hook : : Set ( 0x5AC528 , & gameState . dataCount ) ;
Utils : : Hook : : Set ( 0x5AC56F , & gameState . dataCount ) ;
Utils : : Hook : : Set ( 0x5AC580 , & gameState . dataCount ) ;
Utils : : Hook : : Set ( 0x5AC592 , & gameState . dataCount ) ;
Utils : : Hook : : Set ( 0x5AC59F , & gameState . dataCount ) ;
2017-02-03 04:26:19 -05:00
}
2017-02-01 18:01:46 -05:00
void Weapon : : PatchLimit ( )
{
2017-02-02 18:00:08 -05:00
Utils : : Hook : : Set < DWORD > ( 0x403783 , WEAPON_LIMIT ) ;
Utils : : Hook : : Set < DWORD > ( 0x403E8C , WEAPON_LIMIT ) ;
Utils : : Hook : : Set < DWORD > ( 0x41BC34 , WEAPON_LIMIT ) ;
Utils : : Hook : : Set < DWORD > ( 0x42EB42 , WEAPON_LIMIT ) ;
Utils : : Hook : : Set < DWORD > ( 0x44FA7B , WEAPON_LIMIT ) ;
Utils : : Hook : : Set < DWORD > ( 0x474E0D , WEAPON_LIMIT ) ;
Utils : : Hook : : Set < DWORD > ( 0x48E8F2 , WEAPON_LIMIT ) ;
Utils : : Hook : : Set < DWORD > ( 0x492647 , WEAPON_LIMIT ) ;
Utils : : Hook : : Set < DWORD > ( 0x494585 , WEAPON_LIMIT ) ;
Utils : : Hook : : Set < DWORD > ( 0x4945DB , WEAPON_LIMIT ) ;
Utils : : Hook : : Set < DWORD > ( 0x4B1F96 , WEAPON_LIMIT ) ;
Utils : : Hook : : Set < DWORD > ( 0x4D4A99 , WEAPON_LIMIT ) ;
Utils : : Hook : : Set < DWORD > ( 0x4DD566 , WEAPON_LIMIT ) ;
2017-02-03 15:53:03 -05:00
Utils : : Hook : : Set < DWORD > ( 0x4E3683 , WEAPON_LIMIT ) ; // Configstring
2017-02-02 18:00:08 -05:00
Utils : : Hook : : Set < DWORD > ( 0x58609F , WEAPON_LIMIT ) ;
Utils : : Hook : : Set < DWORD > ( 0x586CAE , WEAPON_LIMIT ) ;
Utils : : Hook : : Set < DWORD > ( 0x58F7BE , WEAPON_LIMIT ) ;
Utils : : Hook : : Set < DWORD > ( 0x58F7D9 , WEAPON_LIMIT ) ;
Utils : : Hook : : Set < DWORD > ( 0x58F82D , WEAPON_LIMIT ) ;
Utils : : Hook : : Set < DWORD > ( 0x5D6C8B , WEAPON_LIMIT ) ;
Utils : : Hook : : Set < DWORD > ( 0x5D6CF7 , WEAPON_LIMIT ) ;
Utils : : Hook : : Set < DWORD > ( 0x5E24D5 , WEAPON_LIMIT ) ;
Utils : : Hook : : Set < DWORD > ( 0x5E2604 , WEAPON_LIMIT ) ;
Utils : : Hook : : Set < DWORD > ( 0x5E2828 , WEAPON_LIMIT ) ;
Utils : : Hook : : Set < DWORD > ( 0x5E2B4F , WEAPON_LIMIT ) ;
Utils : : Hook : : Set < DWORD > ( 0x5E366C , WEAPON_LIMIT ) ;
Utils : : Hook : : Set < DWORD > ( 0x5F2614 , WEAPON_LIMIT ) ;
Utils : : Hook : : Set < DWORD > ( 0x5F7187 , WEAPON_LIMIT ) ;
Utils : : Hook : : Set < DWORD > ( 0x5FECF9 , WEAPON_LIMIT ) ;
2017-02-06 13:29:49 -05:00
Utils : : Hook : : Set < int > ( 0x586CC3 , - ( WEAPON_LIMIT ) ) ;
2017-02-05 11:26:54 -05:00
// Reference: https://courses.engr.illinois.edu/ece390/books/artofasm/CH09/CH09-6.html (See 9.5.2 Division Without DIV and IDIV)
// And http://reverseengineering.stackexchange.com/questions/1397/how-can-i-reverse-optimized-integer-division-modulo-by-constant-operations
// The game's magic number is computed using this formula: (1 / 1200) * (2 ^ (32 + 7)
// I'm too lazy to generate the new magic number, so we can make use of the fact that using powers of 2 as scales allows to change the compensating shift
static_assert ( ( ( WEAPON_LIMIT / 1200 ) * 1200 ) = = WEAPON_LIMIT & & ( WEAPON_LIMIT / 1200 ) ! = 0 & & ! ( ( WEAPON_LIMIT / 1200 ) & ( ( WEAPON_LIMIT / 1200 ) - 1 ) ) , " WEAPON_LIMIT / 1200 is not a power of 2! " ) ;
const unsigned char compensation = 7 + static_cast < unsigned char > ( log2 ( WEAPON_LIMIT / 1200 ) ) ; // 7 is the compensation the game uses
Utils : : Hook : : Set < BYTE > ( 0x49263D , compensation ) ;
Utils : : Hook : : Set < BYTE > ( 0x5E250C , compensation ) ;
Utils : : Hook : : Set < BYTE > ( 0x5E2B43 , compensation ) ;
Utils : : Hook : : Set < BYTE > ( 0x5E352F , compensation ) ;
Utils : : Hook : : Set < BYTE > ( 0x5FECEF , compensation ) ;
2017-02-01 18:01:46 -05:00
static int bg_weaponCompleteDefs [ WEAPON_LIMIT ] ;
Utils : : Hook : : Set < DWORD > ( 0x4B35E1 , sizeof ( bg_weaponCompleteDefs ) ) ;
Utils : : Hook : : Set ( 0x44CE07 , bg_weaponCompleteDefs ) ;
Utils : : Hook : : Set ( 0x45C671 , bg_weaponCompleteDefs ) ;
Utils : : Hook : : Set ( 0x45C6A4 , bg_weaponCompleteDefs ) ;
Utils : : Hook : : Set ( 0x4B35E8 , bg_weaponCompleteDefs ) ;
Utils : : Hook : : Set ( 0x4B3643 , bg_weaponCompleteDefs ) ;
Utils : : Hook : : Set ( 0x4CE903 , bg_weaponCompleteDefs ) ;
Utils : : Hook : : Set ( 0x4CE927 , bg_weaponCompleteDefs ) ;
Utils : : Hook : : Set ( 0x4E6EC7 , bg_weaponCompleteDefs ) ;
Utils : : Hook : : Set ( 0x57B69A , bg_weaponCompleteDefs ) ;
Utils : : Hook : : Set ( 0x57B910 , bg_weaponCompleteDefs ) ;
Utils : : Hook : : Set ( 0x57B925 , bg_weaponCompleteDefs ) ;
Utils : : Hook : : Set ( 0x57BA22 , bg_weaponCompleteDefs ) ;
Utils : : Hook : : Set ( 0x57BA51 , bg_weaponCompleteDefs ) ;
Utils : : Hook : : Set ( 0x57BA78 , bg_weaponCompleteDefs ) ;
static int bg_weaponDefs [ WEAPON_LIMIT ] ;
Utils : : Hook : : Set < DWORD > ( 0x4B35F2 , sizeof ( bg_weaponDefs ) ) ;
Utils : : Hook : : Set ( 0x42F697 , bg_weaponDefs ) ;
Utils : : Hook : : Set ( 0x440EB7 , bg_weaponDefs ) ;
Utils : : Hook : : Set ( 0x45C67D , bg_weaponDefs ) ;
Utils : : Hook : : Set ( 0x45C685 , bg_weaponDefs ) ;
Utils : : Hook : : Set ( 0x45C6B0 , bg_weaponDefs ) ;
Utils : : Hook : : Set ( 0x4B35F9 , bg_weaponDefs ) ;
Utils : : Hook : : Set ( 0x4B364B , bg_weaponDefs ) ;
Utils : : Hook : : Set ( 0x57B693 , bg_weaponDefs ) ;
Utils : : Hook : : Set ( 0x57B75D , bg_weaponDefs ) ;
Utils : : Hook : : Set ( 0x57B809 , bg_weaponDefs ) ;
Utils : : Hook : : Set ( 0x57B8D3 , bg_weaponDefs ) ;
Utils : : Hook : : Set ( 0x57B96D , bg_weaponDefs ) ;
Utils : : Hook : : Set ( 0x57BA32 , bg_weaponDefs ) ;
static int bg_weapClips [ WEAPON_LIMIT ] ;
Utils : : Hook : : Set < DWORD > ( 0x4B3603 , sizeof ( bg_weapClips ) ) ;
Utils : : Hook : : Set ( 0x4B360A , bg_weapClips ) ;
Utils : : Hook : : Set ( 0x4B3666 , bg_weapClips ) ;
Utils : : Hook : : Set ( 0x57B993 , bg_weapClips ) ;
Utils : : Hook : : Set ( 0x57B9E1 , bg_weapClips ) ;
Utils : : Hook : : Set ( 0x57B9EA , bg_weapClips ) ;
static int bg_sharedAmmoCaps [ WEAPON_LIMIT ] ;
Utils : : Hook : : Set < DWORD > ( 0x4B3614 , sizeof ( bg_sharedAmmoCaps ) ) ;
Utils : : Hook : : Set ( 0x414D27 , bg_sharedAmmoCaps ) ;
Utils : : Hook : : Set ( 0x4B361B , bg_sharedAmmoCaps ) ;
Utils : : Hook : : Set ( 0x4B365B , bg_sharedAmmoCaps ) ;
Utils : : Hook : : Set ( 0x57B83B , bg_sharedAmmoCaps ) ;
Utils : : Hook : : Set ( 0x57B870 , bg_sharedAmmoCaps ) ;
Utils : : Hook : : Set ( 0x57B87D , bg_sharedAmmoCaps ) ;
Utils : : Hook : : Set ( 0x57B89C , bg_sharedAmmoCaps ) ;
Utils : : Hook : : Set ( 0x57B8E0 , bg_sharedAmmoCaps ) ;
Utils : : Hook : : Set ( 0x57B901 , bg_sharedAmmoCaps ) ;
static int bg_weapAmmoTypes [ WEAPON_LIMIT ] ;
Utils : : Hook : : Set < DWORD > ( 0x4B3625 , sizeof ( bg_weapAmmoTypes ) ) ;
Utils : : Hook : : Set ( 0x4B362C , bg_weapAmmoTypes ) ;
Utils : : Hook : : Set ( 0x4B3650 , bg_weapAmmoTypes ) ;
2017-02-02 16:18:43 -05:00
Utils : : Hook : : Set ( 0x57B783 , bg_weapAmmoTypes ) ;
2017-02-01 18:01:46 -05:00
Utils : : Hook : : Set ( 0x57B7D1 , bg_weapAmmoTypes ) ;
Utils : : Hook : : Set ( 0x57B7DA , bg_weapAmmoTypes ) ;
2017-02-01 18:19:18 -05:00
static int weaponStrings [ WEAPON_LIMIT * 2 ] ; // string + hash
Utils : : Hook : : Set < DWORD > ( 0x504E01 , sizeof ( weaponStrings ) ) ;
Utils : : Hook : : Set < DWORD > ( 0x4C77DC , sizeof ( weaponStrings ) ) ;
Utils : : Hook : : Set ( 0x4B72DC , weaponStrings ) ;
Utils : : Hook : : Set ( 0x4C77E2 , weaponStrings ) ;
Utils : : Hook : : Set ( 0x504E08 , weaponStrings ) ;
Utils : : Hook : : Set ( 0x795584 , weaponStrings ) ;
Utils : : Hook : : Set ( 0x7955FC , weaponStrings ) ;
Utils : : Hook : : Set ( 0x4B72E8 , & weaponStrings [ 1 ] ) ;
2017-02-01 18:01:46 -05:00
2017-02-02 11:47:11 -05:00
static char cg_weaponsArray [ 32 * WEAPON_LIMIT ] ;
Utils : : Hook : : Set < DWORD > ( 0x4E3300 , sizeof ( cg_weaponsArray ) ) ;
Utils : : Hook : : Set < DWORD > ( 0x4E3305 , sizeof ( cg_weaponsArray ) ) ;
Utils : : Hook : : Set < DWORD > ( 0x4F1927 , sizeof ( cg_weaponsArray ) ) ;
Utils : : Hook : : Set < DWORD > ( 0x4F192C , sizeof ( cg_weaponsArray ) ) ;
Utils : : Hook : : Set ( 0x4172D8 , cg_weaponsArray ) ;
Utils : : Hook : : Set ( 0x45C96F , cg_weaponsArray ) ;
Utils : : Hook : : Set ( 0x45C9C7 , cg_weaponsArray ) ;
Utils : : Hook : : Set ( 0x47F417 , cg_weaponsArray ) ;
Utils : : Hook : : Set ( 0x4BA571 , cg_weaponsArray ) ;
Utils : : Hook : : Set ( 0x4E330B , cg_weaponsArray ) ;
Utils : : Hook : : Set ( 0x4EF56C , cg_weaponsArray ) ;
Utils : : Hook : : Set ( 0x4F1934 , cg_weaponsArray ) ;
Utils : : Hook : : Set ( 0x58B879 , cg_weaponsArray ) ;
Utils : : Hook : : Set ( 0x58B98A , cg_weaponsArray ) ;
Utils : : Hook : : Set ( 0x58CCF7 , cg_weaponsArray ) ;
Utils : : Hook : : Set ( 0x59BF05 , cg_weaponsArray ) ;
Utils : : Hook : : Set ( 0x59C39B , cg_weaponsArray ) ;
Utils : : Hook : : Set ( 0x446D30 , & cg_weaponsArray [ 8 ] ) ;
Utils : : Hook : : Set ( 0x59BD68 , & cg_weaponsArray [ 13 ] ) ;
Utils : : Hook : : Set ( 0x58D0AE , & cg_weaponsArray [ 20 ] ) ;
static int cg_weaponsStaticArray [ 3 * WEAPON_LIMIT ] ;
Utils : : Hook : : Set < DWORD > ( 0x4E3322 , sizeof ( cg_weaponsStaticArray ) ) ;
Utils : : Hook : : Set < DWORD > ( 0x4F1912 , sizeof ( cg_weaponsStaticArray ) ) ;
Utils : : Hook : : Set ( 0x4548DE , cg_weaponsStaticArray ) ;
Utils : : Hook : : Set ( 0x4E3328 , cg_weaponsStaticArray ) ;
2017-02-02 12:51:20 -05:00
Utils : : Hook : : Set ( 0x4EF57A , cg_weaponsStaticArray ) ;
2017-02-02 11:47:11 -05:00
Utils : : Hook : : Set ( 0x4F1919 , cg_weaponsStaticArray ) ;
Utils : : Hook : : Set ( 0x59C095 , cg_weaponsStaticArray ) ;
2017-02-02 12:51:20 -05:00
Utils : : Hook : : Set ( 0x59C09D , cg_weaponsStaticArray ) ;
2017-02-06 09:23:29 -05:00
static int unknownMaterialArray [ WEAPON_LIMIT + 4 ] ;
2017-02-06 07:32:30 -05:00
Utils : : Hook : : Set ( 0x58D003 , unknownMaterialArray ) ;
2017-02-06 09:23:29 -05:00
Utils : : Hook : : Set ( 0x58969A , & unknownMaterialArray [ 3 ] ) ;
Utils : : Hook : : Set ( 0x4EF619 , & unknownMaterialArray [ 4 ] ) ;
Utils : : Hook : : Set ( 0x5896AB , & unknownMaterialArray [ 4 ] ) ;
2017-02-02 12:51:20 -05:00
// Has to do with fx, but somehow lies within the material array
2017-02-11 18:56:46 -05:00
//Utils::Hook::Set(0x402069, &unknownMaterialArray[32]);
//Utils::Hook::Set(0x4E05D9, &unknownMaterialArray[32]);
2017-02-01 18:01:46 -05:00
// Patch bg_weaponDefs on the stack
Utils : : Hook : : Set < DWORD > ( 0x40C31D , sizeof ( bg_weaponDefs ) ) ;
Utils : : Hook : : Set < DWORD > ( 0x40C32F , sizeof ( bg_weaponDefs ) ) ;
Utils : : Hook : : Set < DWORD > ( 0x40C311 , 0x258C + ( ( sizeof ( bg_weaponDefs ) * 2 ) - ( 1200 * 4 * 2 ) ) ) ;
Utils : : Hook : : Set < DWORD > ( 0x40C45F , 0x258C + ( ( sizeof ( bg_weaponDefs ) * 2 ) - ( 1200 * 4 * 2 ) ) ) ;
Utils : : Hook : : Set < DWORD > ( 0x40C478 , 0x258C + ( ( sizeof ( bg_weaponDefs ) * 2 ) - ( 1200 * 4 * 2 ) ) ) ;
Utils : : Hook : : Set < DWORD > ( 0x40C434 , 0x258C + ( ( sizeof ( bg_weaponDefs ) * 2 ) - ( 1200 * 4 * 2 ) ) ) ;
Utils : : Hook : : Set < DWORD > ( 0x40C434 , 0x258C + ( ( sizeof ( bg_weaponDefs ) * 2 ) - ( 1200 * 4 * 2 ) ) ) ;
// Move second buffer pointers
Utils : : Hook : : Set < DWORD > ( 0x40C336 , 0x12E4 + ( ( sizeof ( bg_weaponDefs ) ) - ( 1200 * 4 ) ) ) ;
Utils : : Hook : : Set < DWORD > ( 0x40C3C6 , 0x12DC + ( ( sizeof ( bg_weaponDefs ) ) - ( 1200 * 4 ) ) ) ;
Utils : : Hook : : Set < DWORD > ( 0x40C3CE , 0x12DC + ( ( sizeof ( bg_weaponDefs ) ) - ( 1200 * 4 ) ) ) ;
// Move arg0 pointers
Utils : : Hook : : Set < DWORD > ( 0x40C365 , 0x259C + ( ( sizeof ( bg_weaponDefs ) * 2 ) - ( 1200 * 4 * 2 ) ) ) ;
Utils : : Hook : : Set < DWORD > ( 0x40C44E , 0x259C + ( ( sizeof ( bg_weaponDefs ) * 2 ) - ( 1200 * 4 * 2 ) ) ) ;
Utils : : Hook : : Set < DWORD > ( 0x40C467 , 0x259C + ( ( sizeof ( bg_weaponDefs ) * 2 ) - ( 1200 * 4 * 2 ) ) ) ;
// Move arg4 pointers
Utils : : Hook : : Set < DWORD > ( 0x40C344 , 0x25B4 + ( ( sizeof ( bg_weaponDefs ) * 2 ) - ( 1200 * 4 * 2 ) ) ) ;
// Patch bg_sharedAmmoCaps on the stack
Utils : : Hook : : Set < DWORD > ( 0x4F76E6 , sizeof ( bg_sharedAmmoCaps ) ) ;
Utils : : Hook : : Set < DWORD > ( 0x4F7621 , 0x12C8 + ( sizeof ( bg_sharedAmmoCaps ) - ( 1200 * 4 ) ) ) ;
Utils : : Hook : : Set < DWORD > ( 0x4F76AF , 0x12C8 + ( sizeof ( bg_sharedAmmoCaps ) - ( 1200 * 4 ) ) ) ;
Utils : : Hook : : Set < DWORD > ( 0x4F76DA , 0x12C8 + ( sizeof ( bg_sharedAmmoCaps ) - ( 1200 * 4 ) ) ) ;
Utils : : Hook : : Set < DWORD > ( 0x4F77C5 , 0x12C8 + ( sizeof ( bg_sharedAmmoCaps ) - ( 1200 * 4 ) ) ) ;
// Move arg0 pointers
Utils : : Hook : : Set < DWORD > ( 0x4F766D , 0x12DC + ( sizeof ( bg_sharedAmmoCaps ) - ( 1200 * 4 ) ) ) ;
Utils : : Hook : : Set < DWORD > ( 0x4F76B7 , 0x12DC + ( sizeof ( bg_sharedAmmoCaps ) - ( 1200 * 4 ) ) ) ;
Utils : : Hook : : Set < DWORD > ( 0x4F76FB , 0x12EC + ( sizeof ( bg_sharedAmmoCaps ) - ( 1200 * 4 ) ) ) ;
// Move arg4 pointers
Utils : : Hook : : Set < DWORD > ( 0x4F7630 , 0x12DC + ( sizeof ( bg_sharedAmmoCaps ) - ( 1200 * 4 ) ) ) ;
}
2017-05-30 09:58:39 -04:00
void * Weapon : : LoadNoneWeaponHook ( )
{
// load anim scripts now, rather than a bit later on
2017-05-30 10:32:17 -04:00
Utils : : Hook : : Call < void ( ) > ( 0x4E46A0 ) ( ) ;
2017-05-30 09:58:39 -04:00
return Game : : DB_FindXAssetHeader ( Game : : XAssetType : : ASSET_TYPE_WEAPON , " none " ) . data ;
}
void __declspec ( naked ) Weapon : : LoadNoneWeaponHookStub ( )
{
__asm
{
jmp LoadNoneWeaponHook
}
}
2022-06-30 16:24:40 -04:00
void __declspec ( naked ) Weapon : : CG_UpdatePrimaryForAltModeWeapon_Stub ( )
{
__asm
{
mov eax , 0x440EB0 // BG_GetWeaponDef
call eax
add esp , 0x4
test eax , eax
jz null
// Game code
push 0x59E349
retn
null :
mov al , 1
ret
}
}
void __declspec ( naked ) Weapon : : CG_SelectWeaponIndex_Stub ( )
{
__asm
{
mov eax , 0x440EB0 // BG_GetWeaponDef
call eax
add esp , 0x4
test eax , eax
jz null
// Game code
push 0x48BB2D
2022-12-06 09:42:09 -05:00
ret
2022-06-30 16:24:40 -04:00
null :
push 0x48BB1F // Exit function
ret
}
}
2022-06-25 14:22:13 -04:00
void __declspec ( naked ) Weapon : : WeaponEntCanBeGrabbed_Stub ( )
{
using namespace Game ;
__asm
{
cmp dword ptr [ esp + 0x8 ] , 0x0
jz touched
push 0x56E82C
2022-12-06 09:42:09 -05:00
ret
2022-06-25 14:22:13 -04:00
touched :
test dword ptr [ edi + 0x2BC ] , PWF_DISABLE_WEAPON_PICKUP
jnz exit_func
// Game code
test eax , eax
jz continue_func
exit_func :
xor eax , eax
ret
continue_func :
push 0x56E84C
2022-12-06 09:42:09 -05:00
ret
}
}
__declspec ( naked ) void Weapon : : JavelinResetHook_Stub ( )
{
static DWORD PM_Weapon_OffHandEnd_t = 0x577A10 ;
__asm
{
call PM_Weapon_OffHandEnd_t
push eax
mov eax , BGWeaponOffHandFix
cmp byte ptr [ eax + 0x10 ] , 1
pop eax
jne safeReturn
mov dword ptr [ esi + 0x34 ] , 0 // playerState_s.grenadeTimeLeft
safeReturn :
pop edi
pop esi
pop ebx
ret
2022-06-25 14:22:13 -04:00
}
}
2023-03-03 18:35:30 -05:00
void Weapon : : PlayerCmd_initialWeaponRaise ( Game : : scr_entref_t entref )
{
2023-03-05 08:14:47 -05:00
auto * ent = GSC : : Script : : Scr_GetPlayerEntity ( entref ) ;
2023-03-03 18:35:30 -05:00
const auto * weapon = Game : : Scr_GetString ( 0 ) ;
const auto index = Game : : G_GetWeaponIndexForName ( weapon ) ;
auto * ps = & ent - > client - > ps ;
if ( ! Game : : BG_IsWeaponValid ( ps , index ) )
{
2023-03-04 12:40:28 -05:00
Game : : Scr_Error ( Utils : : String : : VA ( " invalid InitialWeaponRaise: %s " , weapon ) ) ;
2023-03-03 18:35:30 -05:00
return ;
}
assert ( ps ) ;
if ( ! index )
{
return ;
}
auto * equippedWeapon = Game : : BG_GetEquippedWeaponState ( ps , index ) ;
if ( ! equippedWeapon )
{
return ;
}
equippedWeapon - > usedBefore = false ;
Game : : Player_SwitchToWeapon ( ent ) ;
}
2022-06-25 14:22:13 -04:00
void Weapon : : AddScriptMethods ( )
{
2023-03-05 08:14:47 -05:00
GSC : : Script : : AddMethod ( " DisableWeaponPickup " , [ ] ( Game : : scr_entref_t entref )
2022-06-25 14:22:13 -04:00
{
2023-03-10 15:55:22 -05:00
const auto * ent = GSC : : Script : : Scr_GetPlayerEntity ( entref ) ;
2022-06-25 14:22:13 -04:00
ent - > client - > ps . weapCommon . weapFlags | = Game : : PWF_DISABLE_WEAPON_PICKUP ;
} ) ;
2023-03-05 08:14:47 -05:00
GSC : : Script : : AddMethod ( " EnableWeaponPickup " , [ ] ( Game : : scr_entref_t entref )
2022-06-25 14:22:13 -04:00
{
2023-03-10 15:55:22 -05:00
const auto * ent = GSC : : Script : : Scr_GetPlayerEntity ( entref ) ;
2022-06-25 14:22:13 -04:00
ent - > client - > ps . weapCommon . weapFlags & = ~ Game : : PWF_DISABLE_WEAPON_PICKUP ;
} ) ;
2023-03-03 18:35:30 -05:00
2023-03-05 08:14:47 -05:00
GSC : : Script : : AddMethod ( " InitialWeaponRaise " , PlayerCmd_initialWeaponRaise ) ;
2022-06-25 14:22:13 -04:00
}
2017-01-19 16:23:59 -05:00
Weapon : : Weapon ( )
{
2022-06-25 14:22:13 -04:00
PatchLimit ( ) ;
PatchConfigStrings ( ) ;
2017-02-01 18:01:46 -05:00
2022-07-02 12:30:35 -04:00
// BG_LoadWEaponCompleteDef_FastFile
Utils : : Hook ( 0x57B650 , LoadWeaponCompleteDef , HOOK_JUMP ) . install ( ) - > quick ( ) ;
// Disable warning if raw weapon file cannot be found
Utils : : Hook : : Nop ( 0x57AF60 , 5 ) ;
2017-01-19 16:23:59 -05:00
// weapon asset existence check
Utils : : Hook : : Nop ( 0x408228 , 5 ) ; // find asset header
Utils : : Hook : : Nop ( 0x408230 , 5 ) ; // is asset default
Utils : : Hook : : Nop ( 0x40823A , 2 ) ; // jump
// Skip double loading for fs_game
Utils : : Hook : : Set < BYTE > ( 0x4081FD , 0xEB ) ;
2017-02-12 11:16:01 -05:00
2017-05-30 10:32:17 -04:00
// Weapon swap fix
Utils : : Hook : : Nop ( 0x4B3670 , 5 ) ;
2022-06-25 14:22:13 -04:00
Utils : : Hook ( 0x57B4F0 , LoadNoneWeaponHookStub , HOOK_JUMP ) . install ( ) - > quick ( ) ;
2017-05-30 09:58:39 -04:00
2017-02-12 11:16:01 -05:00
// Clear weapons independently from fs_game
2022-07-19 08:02:53 -04:00
Utils : : Hook : : Nop ( 0x452C1D , 2 ) ;
Utils : : Hook : : Nop ( 0x452C24 , 5 ) ;
2022-06-25 14:22:13 -04:00
2022-06-30 16:24:40 -04:00
Utils : : Hook ( 0x59E341 , CG_UpdatePrimaryForAltModeWeapon_Stub , HOOK_JUMP ) . install ( ) - > quick ( ) ;
Utils : : Hook ( 0x48BB25 , CG_SelectWeaponIndex_Stub , HOOK_JUMP ) . install ( ) - > quick ( ) ;
2022-06-25 14:22:13 -04:00
AssertOffset ( Game : : playerState_s , Game : : playerState_s : : weapCommon . weapFlags , 0x2BC ) ;
Utils : : Hook ( 0x56E825 , WeaponEntCanBeGrabbed_Stub , HOOK_JUMP ) . install ( ) - > quick ( ) ;
2022-12-06 09:42:09 -05:00
// Javelin fix (PM_Weapon_OffHandEnd)
AssertOffset ( Game : : playerState_s , grenadeTimeLeft , 0x34 ) ;
BGWeaponOffHandFix = Game : : Dvar_RegisterBool ( " bg_weaponOffHandFix " , true , Game : : DVAR_CODINFO , " Reset grenadeTimeLeft after using off hand weapon " ) ;
Utils : : Hook ( 0x578F52 , JavelinResetHook_Stub , HOOK_JUMP ) . install ( ) - > quick ( ) ;
AddScriptMethods ( ) ;
2017-01-19 16:23:59 -05:00
}
}