2
0
mirror of https://github.com/rehlds/rehlds.git synced 2025-01-19 10:08:04 +03:00

Merge branch 'master' into ipconnlimit

This commit is contained in:
Dmitry Novikov 2018-05-07 23:07:02 +07:00 committed by GitHub
commit 1328b28304
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
25 changed files with 1027 additions and 280 deletions

View File

@ -52,7 +52,7 @@ Bugfixed version of rehlds contains an additional cvars:
<li>sv_rehlds_stringcmdrate_avg_punish // Time in minutes for which the player will be banned (0 - Permanent, use a negative number for a kick). Default: 5 <li>sv_rehlds_stringcmdrate_avg_punish // Time in minutes for which the player will be banned (0 - Permanent, use a negative number for a kick). Default: 5
<li>sv_rehlds_stringcmdrate_max_burst // Max burst level of 'string' cmds for ban. Default: 400 <li>sv_rehlds_stringcmdrate_max_burst // Max burst level of 'string' cmds for ban. Default: 400
<li>sv_rehlds_stringcmdrate_burst_punish // Time in minutes for which the player will be banned (0 - Permanent, use a negative number for a kick). Default: 5 <li>sv_rehlds_stringcmdrate_burst_punish // Time in minutes for which the player will be banned (0 - Permanent, use a negative number for a kick). Default: 5
<li>sv_rehlds_userinfo_transmitted_fields // Userinfo fields only with these keys will be transmitted to clients via network. If not set then all fields will be transmitted (except prefixed with underscore). Each key must be prefixed by backslash, for example "\name\model\*sid\*hltv\bottomcolor\topcolor". Default: "" <li>sv_rehlds_userinfo_transmitted_fields // Userinfo fields only with these keys will be transmitted to clients via network. If not set then all fields will be transmitted (except prefixed with underscore). Each key must be prefixed by backslash, for example "\name\model\*sid\*hltv\bottomcolor\topcolor". See [wiki](https://github.com/dreamstalker/rehlds/wiki/Userinfo-keys) to collect sufficient set of keys for your server. Default: ""
<li>sv_rehlds_attachedentities_playeranimationspeed_fix // Fixes bug with gait animation speed increase when player has some attached entities (aiments). Can cause animation lags when cl_updaterate is low. Default: 0 <li>sv_rehlds_attachedentities_playeranimationspeed_fix // Fixes bug with gait animation speed increase when player has some attached entities (aiments). Can cause animation lags when cl_updaterate is low. Default: 0
<li>sv_rehlds_maxclients_from_single_ip // Limit number of connections from the single ip address. Default: 5 <li>sv_rehlds_maxclients_from_single_ip // Limit number of connections from the single ip address. Default: 5
</ul> </ul>

View File

@ -159,7 +159,7 @@
<PrecompiledHeader>Use</PrecompiledHeader> <PrecompiledHeader>Use</PrecompiledHeader>
<WarningLevel>Level3</WarningLevel> <WarningLevel>Level3</WarningLevel>
<Optimization>Disabled</Optimization> <Optimization>Disabled</Optimization>
<PreprocessorDefinitions>HLTV;WIN32;_DEBUG;_CONSOLE;_CRT_SECURE_NO_WARNINGS;%(PreprocessorDefinitions)</PreprocessorDefinitions> <PreprocessorDefinitions>HLTV;HLTV_FIXES;WIN32;_DEBUG;_CONSOLE;_CRT_SECURE_NO_WARNINGS;%(PreprocessorDefinitions)</PreprocessorDefinitions>
<AdditionalIncludeDirectories>$(ProjectDir)\..\src;$(ProjectDir)\..\..\;$(ProjectDir)\..\..\..\;$(ProjectDir)\..\..\..\common;$(ProjectDir)\..\..\..\engine;$(ProjectDir)\..\..\..\public;$(ProjectDir)\..\..\..\public\rehlds;%(AdditionalIncludeDirectories)</AdditionalIncludeDirectories> <AdditionalIncludeDirectories>$(ProjectDir)\..\src;$(ProjectDir)\..\..\;$(ProjectDir)\..\..\..\;$(ProjectDir)\..\..\..\common;$(ProjectDir)\..\..\..\engine;$(ProjectDir)\..\..\..\public;$(ProjectDir)\..\..\..\public\rehlds;%(AdditionalIncludeDirectories)</AdditionalIncludeDirectories>
<PrecompiledHeaderFile>precompiled.h</PrecompiledHeaderFile> <PrecompiledHeaderFile>precompiled.h</PrecompiledHeaderFile>
<RuntimeLibrary>MultiThreadedDebug</RuntimeLibrary> <RuntimeLibrary>MultiThreadedDebug</RuntimeLibrary>
@ -190,7 +190,7 @@
<Optimization>Disabled</Optimization> <Optimization>Disabled</Optimization>
<FunctionLevelLinking>true</FunctionLevelLinking> <FunctionLevelLinking>true</FunctionLevelLinking>
<IntrinsicFunctions>true</IntrinsicFunctions> <IntrinsicFunctions>true</IntrinsicFunctions>
<PreprocessorDefinitions>HLTV;WIN32;NDEBUG;_CONSOLE;_CRT_SECURE_NO_WARNINGS;%(PreprocessorDefinitions)</PreprocessorDefinitions> <PreprocessorDefinitions>HLTV;HLTV_FIXES;WIN32;NDEBUG;_CONSOLE;_CRT_SECURE_NO_WARNINGS;%(PreprocessorDefinitions)</PreprocessorDefinitions>
<AdditionalIncludeDirectories>$(ProjectDir)\..\src;$(ProjectDir)\..\..\;$(ProjectDir)\..\..\..\;$(ProjectDir)\..\..\..\common;$(ProjectDir)\..\..\..\engine;$(ProjectDir)\..\..\..\public;$(ProjectDir)\..\..\..\public\rehlds;%(AdditionalIncludeDirectories)</AdditionalIncludeDirectories> <AdditionalIncludeDirectories>$(ProjectDir)\..\src;$(ProjectDir)\..\..\;$(ProjectDir)\..\..\..\;$(ProjectDir)\..\..\..\common;$(ProjectDir)\..\..\..\engine;$(ProjectDir)\..\..\..\public;$(ProjectDir)\..\..\..\public\rehlds;%(AdditionalIncludeDirectories)</AdditionalIncludeDirectories>
<PrecompiledHeaderFile>precompiled.h</PrecompiledHeaderFile> <PrecompiledHeaderFile>precompiled.h</PrecompiledHeaderFile>
<RuntimeLibrary>MultiThreaded</RuntimeLibrary> <RuntimeLibrary>MultiThreaded</RuntimeLibrary>

View File

@ -20,7 +20,7 @@ void setupToolchain(NativeBinarySpec b) {
cfg.projectInclude(project, '/..', '/../..', '/src', '/../../common', '/../../engine', '/../../public', '/../../public/rehlds', '/../../pm_shared'); cfg.projectInclude(project, '/..', '/../..', '/src', '/../../common', '/../../engine', '/../../public', '/../../public/rehlds', '/../../pm_shared');
cfg.projectInclude(dep_bzip2, '/include') cfg.projectInclude(dep_bzip2, '/include')
cfg.singleDefines 'USE_BREAKPAD_HANDLER', 'HLTV', 'CORE_MODULE' cfg.singleDefines 'USE_BREAKPAD_HANDLER', 'HLTV', 'HLTV_FIXES', 'CORE_MODULE'
if (cfg instanceof MsvcToolchainConfig) { if (cfg instanceof MsvcToolchainConfig) {
cfg.compilerOptions.pchConfig = new MsvcToolchainConfig.PrecompiledHeadersConfig( cfg.compilerOptions.pchConfig = new MsvcToolchainConfig.PrecompiledHeadersConfig(

View File

@ -77,7 +77,7 @@
<PrecompiledHeader>Use</PrecompiledHeader> <PrecompiledHeader>Use</PrecompiledHeader>
<WarningLevel>Level3</WarningLevel> <WarningLevel>Level3</WarningLevel>
<Optimization>Disabled</Optimization> <Optimization>Disabled</Optimization>
<PreprocessorDefinitions>HLTV;WIN32;_DEBUG;_WINDOWS;_USRDLL;CORE_MODULE;_CRT_SECURE_NO_WARNINGS;%(PreprocessorDefinitions)</PreprocessorDefinitions> <PreprocessorDefinitions>HLTV;HLTV_FIXES;WIN32;_DEBUG;_WINDOWS;_USRDLL;CORE_MODULE;_CRT_SECURE_NO_WARNINGS;%(PreprocessorDefinitions)</PreprocessorDefinitions>
<SDLCheck> <SDLCheck>
</SDLCheck> </SDLCheck>
<AdditionalIncludeDirectories>$(ProjectDir)\..\src;$(ProjectDir)\..\..\;$(ProjectDir)\..\..\..\;$(ProjectDir)\..\..\..\common;$(ProjectDir)\..\..\..\engine;$(ProjectDir)\..\..\..\public;$(ProjectDir)\..\..\..\public\rehlds;$(ProjectDir)\..\..\..\pm_shared;$(ProjectDir)\..\..\..\..\dep\bzip2\include;%(AdditionalIncludeDirectories)</AdditionalIncludeDirectories> <AdditionalIncludeDirectories>$(ProjectDir)\..\src;$(ProjectDir)\..\..\;$(ProjectDir)\..\..\..\;$(ProjectDir)\..\..\..\common;$(ProjectDir)\..\..\..\engine;$(ProjectDir)\..\..\..\public;$(ProjectDir)\..\..\..\public\rehlds;$(ProjectDir)\..\..\..\pm_shared;$(ProjectDir)\..\..\..\..\dep\bzip2\include;%(AdditionalIncludeDirectories)</AdditionalIncludeDirectories>
@ -112,7 +112,7 @@
<Optimization>MaxSpeed</Optimization> <Optimization>MaxSpeed</Optimization>
<FunctionLevelLinking>true</FunctionLevelLinking> <FunctionLevelLinking>true</FunctionLevelLinking>
<IntrinsicFunctions>true</IntrinsicFunctions> <IntrinsicFunctions>true</IntrinsicFunctions>
<PreprocessorDefinitions>HLTV;WIN32;NDEBUG;_WINDOWS;_USRDLL;CORE_MODULE;_CRT_SECURE_NO_WARNINGS;%(PreprocessorDefinitions)</PreprocessorDefinitions> <PreprocessorDefinitions>HLTV;HLTV_FIXES;WIN32;NDEBUG;_WINDOWS;_USRDLL;CORE_MODULE;_CRT_SECURE_NO_WARNINGS;%(PreprocessorDefinitions)</PreprocessorDefinitions>
<SDLCheck> <SDLCheck>
</SDLCheck> </SDLCheck>
<AdditionalIncludeDirectories>$(ProjectDir)\..\src;$(ProjectDir)\..\..\;$(ProjectDir)\..\..\..\;$(ProjectDir)\..\..\..\common;$(ProjectDir)\..\..\..\engine;$(ProjectDir)\..\..\..\public;$(ProjectDir)\..\..\..\public\rehlds;$(ProjectDir)\..\..\..\pm_shared;$(ProjectDir)\..\..\..\..\dep\bzip2\include;%(AdditionalIncludeDirectories)</AdditionalIncludeDirectories> <AdditionalIncludeDirectories>$(ProjectDir)\..\src;$(ProjectDir)\..\..\;$(ProjectDir)\..\..\..\;$(ProjectDir)\..\..\..\common;$(ProjectDir)\..\..\..\engine;$(ProjectDir)\..\..\..\public;$(ProjectDir)\..\..\..\public\rehlds;$(ProjectDir)\..\..\..\pm_shared;$(ProjectDir)\..\..\..\..\dep\bzip2\include;%(AdditionalIncludeDirectories)</AdditionalIncludeDirectories>

View File

@ -297,6 +297,14 @@ int NetSocket::GetLong(unsigned char *pData, int size)
{ {
m_NetSplitPacket.currentSequence = sequenceNumber; m_NetSplitPacket.currentSequence = sequenceNumber;
m_NetSplitPacket.splitCount = packetCount; m_NetSplitPacket.splitCount = packetCount;
#ifdef HLTV_FIXES
m_NetSplitPacket.totalSize = 0;
// clear part's sequence
for (int i = 0; i < MAX_SPLIT_FRAGMENTS; i++)
netSplitFlags[i] = -1;
#endif
} }
unsigned int packetPayloadSize = size - sizeof(SPLITPACKET); unsigned int packetPayloadSize = size - sizeof(SPLITPACKET);
@ -310,16 +318,47 @@ int NetSocket::GetLong(unsigned char *pData, int size)
m_NetSplitPacket.totalSize = packetPayloadSize + SPLIT_SIZE * (packetCount - 1); m_NetSplitPacket.totalSize = packetPayloadSize + SPLIT_SIZE * (packetCount - 1);
} }
--m_NetSplitPacket.splitCount; m_NetSplitPacket.splitCount--;
netSplitFlags[packetNumber] = sequenceNumber; netSplitFlags[packetNumber] = sequenceNumber;
#ifdef HLTV_FIXES
if (SPLIT_SIZE * packetNumber + packetPayloadSize > MAX_UDP_PACKET)
{
m_System->DPrintf("WARNING! NetSocket::GetLong: Malformed packet size (%i, %i)\n", SPLIT_SIZE * packetNumber, packetPayloadSize);
m_NetSplitPacket.currentSequence = -1;
return -1;
} }
Q_memcpy(&m_NetSplitPacket.buffer[SPLIT_SIZE * packetNumber], pHeader + 1, packetPayloadSize); Q_memcpy(&m_NetSplitPacket.buffer[SPLIT_SIZE * packetNumber], pHeader + 1, packetPayloadSize);
#endif
}
#ifndef HLTV_FIXES
Q_memcpy(&m_NetSplitPacket.buffer[SPLIT_SIZE * packetNumber], pHeader + 1, packetPayloadSize);
#endif
if (m_NetSplitPacket.splitCount > 0) { if (m_NetSplitPacket.splitCount > 0) {
return 0; return 0;
} }
#ifdef HLTV_FIXES
for (unsigned int i = 0; i < packetCount; i++)
{
if (netSplitFlags[i] != m_NetSplitPacket.currentSequence)
{
m_System->DPrintf("WARNING! NetSocket::GetLong: Split packet without all %i parts, part %i had wrong sequence %i/%i\n",
packetCount,
i + 1,
netSplitFlags[i],
m_NetSplitPacket.currentSequence);
m_NetSplitPacket.currentSequence = -1;
return -1;
}
}
#endif
m_NetSplitPacket.currentSequence = -1; m_NetSplitPacket.currentSequence = -1;
if (m_NetSplitPacket.totalSize > MAX_UDP_PACKET) if (m_NetSplitPacket.totalSize > MAX_UDP_PACKET)
{ {

View File

@ -252,11 +252,11 @@ protected:
int m_MaxInstanced_BaseLine; int m_MaxInstanced_BaseLine;
#ifdef HOOK_HLTV #if defined(HLTV_FIXES) && !defined(HOOK_HLTV)
char m_Lightstyles[MAX_LIGHTSTYLES][65];
#else
char m_Lightstyles[MAX_LIGHTSTYLES][64]; char m_Lightstyles[MAX_LIGHTSTYLES][64];
#endif // HOOK_HLTV #else
char m_Lightstyles[MAX_LIGHTSTYLES][65];
#endif
movevars_t m_MoveVars; movevars_t m_MoveVars;
BSPModel m_WorldModel; BSPModel m_WorldModel;

View File

@ -16,7 +16,7 @@ void setupToolchain(NativeBinarySpec b) {
boolean useGcc = project.hasProperty("useGcc") boolean useGcc = project.hasProperty("useGcc")
def cfg = rootProject.createToolchainConfig(b); def cfg = rootProject.createToolchainConfig(b);
cfg.projectInclude(project, '/..', '/../..', '/src', '/../common', '/../../common', '/../../engine', '/../../public', '/../../public/rehlds', '/../../pm_shared'); cfg.projectInclude(project, '/..', '/../..', '/src', '/../common', '/../../common', '/../../engine', '/../../public', '/../../public/rehlds', '/../../pm_shared');
cfg.singleDefines 'USE_BREAKPAD_HANDLER', 'HLTV', 'DEMOPLAYER_MODULE' cfg.singleDefines 'USE_BREAKPAD_HANDLER', 'HLTV', 'HLTV_FIXES', 'DEMOPLAYER_MODULE'
if (cfg instanceof MsvcToolchainConfig) { if (cfg instanceof MsvcToolchainConfig) {
cfg.compilerOptions.pchConfig = new MsvcToolchainConfig.PrecompiledHeadersConfig( cfg.compilerOptions.pchConfig = new MsvcToolchainConfig.PrecompiledHeadersConfig(

View File

@ -125,7 +125,7 @@
<PrecompiledHeader>Use</PrecompiledHeader> <PrecompiledHeader>Use</PrecompiledHeader>
<WarningLevel>Level3</WarningLevel> <WarningLevel>Level3</WarningLevel>
<Optimization>Disabled</Optimization> <Optimization>Disabled</Optimization>
<PreprocessorDefinitions>HLTV;WIN32;_DEBUG;_WINDOWS;_USRDLL;DEMOPLAYER_MODULE;_CRT_SECURE_NO_WARNINGS;%(PreprocessorDefinitions)</PreprocessorDefinitions> <PreprocessorDefinitions>HLTV;HLTV_FIXES;WIN32;_DEBUG;_WINDOWS;_USRDLL;DEMOPLAYER_MODULE;_CRT_SECURE_NO_WARNINGS;%(PreprocessorDefinitions)</PreprocessorDefinitions>
<PrecompiledHeaderFile>precompiled.h</PrecompiledHeaderFile> <PrecompiledHeaderFile>precompiled.h</PrecompiledHeaderFile>
<AdditionalIncludeDirectories>$(ProjectDir)\..\src;$(ProjectDir)\..\..\;$(ProjectDir)\..\..\..\;$(ProjectDir)\..\..\..\common;$(ProjectDir)\..\..\..\engine;$(ProjectDir)\..\..\..\public;$(ProjectDir)\..\..\..\public\rehlds;$(ProjectDir)\..\..\..\pm_shared;$(ProjectDir)\..\..\..\game_shared;$(ProjectDir)\..\..\..\..\dep\bzip2\include;%(AdditionalIncludeDirectories)</AdditionalIncludeDirectories> <AdditionalIncludeDirectories>$(ProjectDir)\..\src;$(ProjectDir)\..\..\;$(ProjectDir)\..\..\..\;$(ProjectDir)\..\..\..\common;$(ProjectDir)\..\..\..\engine;$(ProjectDir)\..\..\..\public;$(ProjectDir)\..\..\..\public\rehlds;$(ProjectDir)\..\..\..\pm_shared;$(ProjectDir)\..\..\..\game_shared;$(ProjectDir)\..\..\..\..\dep\bzip2\include;%(AdditionalIncludeDirectories)</AdditionalIncludeDirectories>
<RuntimeLibrary>MultiThreadedDebug</RuntimeLibrary> <RuntimeLibrary>MultiThreadedDebug</RuntimeLibrary>
@ -155,7 +155,7 @@
<Optimization>MaxSpeed</Optimization> <Optimization>MaxSpeed</Optimization>
<FunctionLevelLinking>true</FunctionLevelLinking> <FunctionLevelLinking>true</FunctionLevelLinking>
<IntrinsicFunctions>true</IntrinsicFunctions> <IntrinsicFunctions>true</IntrinsicFunctions>
<PreprocessorDefinitions>HLTV;WIN32;NDEBUG;_WINDOWS;_USRDLL;DEMOPLAYER_MODULE;_CRT_SECURE_NO_WARNINGS;%(PreprocessorDefinitiHOOons)</PreprocessorDefinitions> <PreprocessorDefinitions>HLTV;HLTV_FIXES;WIN32;NDEBUG;_WINDOWS;_USRDLL;DEMOPLAYER_MODULE;_CRT_SECURE_NO_WARNINGS;%(PreprocessorDefinitiHOOons)</PreprocessorDefinitions>
<PrecompiledHeaderFile>precompiled.h</PrecompiledHeaderFile> <PrecompiledHeaderFile>precompiled.h</PrecompiledHeaderFile>
<AdditionalIncludeDirectories>$(ProjectDir)\..\src;$(ProjectDir)\..\..\;$(ProjectDir)\..\..\..\;$(ProjectDir)\..\..\..\common;$(ProjectDir)\..\..\..\engine;$(ProjectDir)\..\..\..\public;$(ProjectDir)\..\..\..\public\rehlds;$(ProjectDir)\..\..\..\pm_shared;%(AdditionalIncludeDirectories)</AdditionalIncludeDirectories> <AdditionalIncludeDirectories>$(ProjectDir)\..\src;$(ProjectDir)\..\..\;$(ProjectDir)\..\..\..\;$(ProjectDir)\..\..\..\common;$(ProjectDir)\..\..\..\engine;$(ProjectDir)\..\..\..\public;$(ProjectDir)\..\..\..\public\rehlds;$(ProjectDir)\..\..\..\pm_shared;%(AdditionalIncludeDirectories)</AdditionalIncludeDirectories>
<RuntimeLibrary>MultiThreaded</RuntimeLibrary> <RuntimeLibrary>MultiThreaded</RuntimeLibrary>

View File

@ -16,7 +16,7 @@ void setupToolchain(NativeBinarySpec b) {
boolean useGcc = project.hasProperty("useGcc") boolean useGcc = project.hasProperty("useGcc")
def cfg = rootProject.createToolchainConfig(b); def cfg = rootProject.createToolchainConfig(b);
cfg.projectInclude(project, '/..', '/../..', '/src', '/../../common', '/../../engine', '/../../public', '/../../public/rehlds', '/../../pm_shared'); cfg.projectInclude(project, '/..', '/../..', '/src', '/../../common', '/../../engine', '/../../public', '/../../public/rehlds', '/../../pm_shared');
cfg.singleDefines 'USE_BREAKPAD_HANDLER', 'HLTV', 'DIRECTOR_MODULE' cfg.singleDefines 'USE_BREAKPAD_HANDLER', 'HLTV', 'HLTV_FIXES', 'DIRECTOR_MODULE'
if (cfg instanceof MsvcToolchainConfig) { if (cfg instanceof MsvcToolchainConfig) {
cfg.compilerOptions.pchConfig = new MsvcToolchainConfig.PrecompiledHeadersConfig( cfg.compilerOptions.pchConfig = new MsvcToolchainConfig.PrecompiledHeadersConfig(

View File

@ -76,7 +76,7 @@
<PrecompiledHeader>Use</PrecompiledHeader> <PrecompiledHeader>Use</PrecompiledHeader>
<WarningLevel>Level3</WarningLevel> <WarningLevel>Level3</WarningLevel>
<Optimization>Disabled</Optimization> <Optimization>Disabled</Optimization>
<PreprocessorDefinitions>HLTV;WIN32;_DEBUG;_WINDOWS;_USRDLL;DIRECTOR_MODULE;_CRT_SECURE_NO_WARNINGS;%(PreprocessorDefinitions)</PreprocessorDefinitions> <PreprocessorDefinitions>HLTV;HLTV_FIXES;WIN32;_DEBUG;_WINDOWS;_USRDLL;DIRECTOR_MODULE;_CRT_SECURE_NO_WARNINGS;%(PreprocessorDefinitions)</PreprocessorDefinitions>
<AdditionalIncludeDirectories>$(ProjectDir)\..\..\;$(ProjectDir)\..\src;$(ProjectDir)\..\..\..\common;$(ProjectDir)\..\..\..\engine;$(ProjectDir)\..\..\..\public;$(ProjectDir)\..\..\..\public\rehlds;$(ProjectDir)\..\..\..\pm_shared;%(AdditionalIncludeDirectories)</AdditionalIncludeDirectories> <AdditionalIncludeDirectories>$(ProjectDir)\..\..\;$(ProjectDir)\..\src;$(ProjectDir)\..\..\..\common;$(ProjectDir)\..\..\..\engine;$(ProjectDir)\..\..\..\public;$(ProjectDir)\..\..\..\public\rehlds;$(ProjectDir)\..\..\..\pm_shared;%(AdditionalIncludeDirectories)</AdditionalIncludeDirectories>
<PrecompiledHeaderFile>precompiled.h</PrecompiledHeaderFile> <PrecompiledHeaderFile>precompiled.h</PrecompiledHeaderFile>
<RuntimeLibrary>MultiThreadedDebug</RuntimeLibrary> <RuntimeLibrary>MultiThreadedDebug</RuntimeLibrary>
@ -107,7 +107,7 @@
<Optimization>MaxSpeed</Optimization> <Optimization>MaxSpeed</Optimization>
<FunctionLevelLinking>true</FunctionLevelLinking> <FunctionLevelLinking>true</FunctionLevelLinking>
<IntrinsicFunctions>true</IntrinsicFunctions> <IntrinsicFunctions>true</IntrinsicFunctions>
<PreprocessorDefinitions>HLTV;WIN32;NDEBUG;_WINDOWS;_USRDLL;DIRECTOR_MODULE;_CRT_SECURE_NO_WARNINGS;%(PreprocessorDefinitions)</PreprocessorDefinitions> <PreprocessorDefinitions>HLTV;HLTV_FIXES;WIN32;NDEBUG;_WINDOWS;_USRDLL;DIRECTOR_MODULE;_CRT_SECURE_NO_WARNINGS;%(PreprocessorDefinitions)</PreprocessorDefinitions>
<AdditionalIncludeDirectories>$(ProjectDir)\..\..\;$(ProjectDir)\..\src;$(ProjectDir)\..\..\..\common;$(ProjectDir)\..\..\..\engine;$(ProjectDir)\..\..\..\public;$(ProjectDir)\..\..\..\public\rehlds;$(ProjectDir)\..\..\..\pm_shared;%(AdditionalIncludeDirectories)</AdditionalIncludeDirectories> <AdditionalIncludeDirectories>$(ProjectDir)\..\..\;$(ProjectDir)\..\src;$(ProjectDir)\..\..\..\common;$(ProjectDir)\..\..\..\engine;$(ProjectDir)\..\..\..\public;$(ProjectDir)\..\..\..\public\rehlds;$(ProjectDir)\..\..\..\pm_shared;%(AdditionalIncludeDirectories)</AdditionalIncludeDirectories>
<PrecompiledHeaderFile>precompiled.h</PrecompiledHeaderFile> <PrecompiledHeaderFile>precompiled.h</PrecompiledHeaderFile>
<RuntimeLibrary>MultiThreaded</RuntimeLibrary> <RuntimeLibrary>MultiThreaded</RuntimeLibrary>

View File

@ -16,11 +16,12 @@ project.ext.dep_bzip2 = project(':dep/bzip2')
void setupToolchain(NativeBinarySpec b) { void setupToolchain(NativeBinarySpec b) {
boolean useGcc = project.hasProperty("useGcc") boolean useGcc = project.hasProperty("useGcc")
def cfg = rootProject.createToolchainConfig(b); def cfg = rootProject.createToolchainConfig(b);
cfg.projectInclude(project, '/..', '/../..', '/src', '/../Director/src', '/../../common', '/../../engine', '/../../public', '/../../public/rehlds', '/../../pm_shared'); cfg.projectInclude(project, '/..', '/../..', '/src', '/../Director/src', '/../../common', '/../../engine', '/../../public', '/../../public/rehlds', '/../../pm_shared');
cfg.projectInclude(dep_bzip2, '/include') cfg.projectInclude(dep_bzip2, '/include')
cfg.singleDefines 'USE_BREAKPAD_HANDLER', 'HLTV', 'CORE_MODULE' cfg.singleDefines 'USE_BREAKPAD_HANDLER', 'HLTV', 'HLTV_FIXES', 'PROXY_MODULE'
if (cfg instanceof MsvcToolchainConfig) { if (cfg instanceof MsvcToolchainConfig) {
cfg.compilerOptions.pchConfig = new MsvcToolchainConfig.PrecompiledHeadersConfig( cfg.compilerOptions.pchConfig = new MsvcToolchainConfig.PrecompiledHeadersConfig(

View File

@ -384,7 +384,7 @@
<PrecompiledHeader>Use</PrecompiledHeader> <PrecompiledHeader>Use</PrecompiledHeader>
<WarningLevel>Level3</WarningLevel> <WarningLevel>Level3</WarningLevel>
<Optimization>Disabled</Optimization> <Optimization>Disabled</Optimization>
<PreprocessorDefinitions>HLTV;WIN32;_DEBUG;_WINDOWS;_USRDLL;PROXY_MODULE;_CRT_SECURE_NO_WARNINGS;%(PreprocessorDefinitions)</PreprocessorDefinitions> <PreprocessorDefinitions>HLTV;HLTV_FIXES;WIN32;_DEBUG;_WINDOWS;_USRDLL;PROXY_MODULE;_CRT_SECURE_NO_WARNINGS;%(PreprocessorDefinitions)</PreprocessorDefinitions>
<AdditionalIncludeDirectories>$(ProjectDir)\..\src;$(ProjectDir)\..\..\;$(ProjectDir)\..\..\..\;$(ProjectDir)\..\..\Director\src;$(ProjectDir)\..\..\..\common;$(ProjectDir)\..\..\..\engine;$(ProjectDir)\..\..\..\public;$(ProjectDir)\..\..\..\public\rehlds;$(ProjectDir)\..\..\..\pm_shared;$(ProjectDir)\..\..\..\..\dep\bzip2\include;%(AdditionalIncludeDirectories)</AdditionalIncludeDirectories> <AdditionalIncludeDirectories>$(ProjectDir)\..\src;$(ProjectDir)\..\..\;$(ProjectDir)\..\..\..\;$(ProjectDir)\..\..\Director\src;$(ProjectDir)\..\..\..\common;$(ProjectDir)\..\..\..\engine;$(ProjectDir)\..\..\..\public;$(ProjectDir)\..\..\..\public\rehlds;$(ProjectDir)\..\..\..\pm_shared;$(ProjectDir)\..\..\..\..\dep\bzip2\include;%(AdditionalIncludeDirectories)</AdditionalIncludeDirectories>
<PrecompiledHeaderFile>precompiled.h</PrecompiledHeaderFile> <PrecompiledHeaderFile>precompiled.h</PrecompiledHeaderFile>
<RuntimeLibrary>MultiThreadedDebug</RuntimeLibrary> <RuntimeLibrary>MultiThreadedDebug</RuntimeLibrary>
@ -418,7 +418,7 @@
<Optimization>MaxSpeed</Optimization> <Optimization>MaxSpeed</Optimization>
<FunctionLevelLinking>true</FunctionLevelLinking> <FunctionLevelLinking>true</FunctionLevelLinking>
<IntrinsicFunctions>true</IntrinsicFunctions> <IntrinsicFunctions>true</IntrinsicFunctions>
<PreprocessorDefinitions>HLTV;WIN32;NDEBUG;_WINDOWS;_USRDLL;PROXY_MODULE;_CRT_SECURE_NO_WARNINGS;%(PreprocessorDefinitions)</PreprocessorDefinitions> <PreprocessorDefinitions>HLTV;HLTV_FIXES;WIN32;NDEBUG;_WINDOWS;_USRDLL;PROXY_MODULE;_CRT_SECURE_NO_WARNINGS;%(PreprocessorDefinitions)</PreprocessorDefinitions>
<AdditionalIncludeDirectories>$(ProjectDir)\..\src;$(ProjectDir)\..\..\;$(ProjectDir)\..\..\..\;$(ProjectDir)\..\..\Director\src;$(ProjectDir)\..\..\..\common;$(ProjectDir)\..\..\..\engine;$(ProjectDir)\..\..\..\public;$(ProjectDir)\..\..\..\public\rehlds;$(ProjectDir)\..\..\..\pm_shared;$(ProjectDir)\..\..\..\..\dep\bzip2\include;%(AdditionalIncludeDirectories)</AdditionalIncludeDirectories> <AdditionalIncludeDirectories>$(ProjectDir)\..\src;$(ProjectDir)\..\..\;$(ProjectDir)\..\..\..\;$(ProjectDir)\..\..\Director\src;$(ProjectDir)\..\..\..\common;$(ProjectDir)\..\..\..\engine;$(ProjectDir)\..\..\..\public;$(ProjectDir)\..\..\..\public\rehlds;$(ProjectDir)\..\..\..\pm_shared;$(ProjectDir)\..\..\..\..\dep\bzip2\include;%(AdditionalIncludeDirectories)</AdditionalIncludeDirectories>
<PrecompiledHeaderFile>precompiled.h</PrecompiledHeaderFile> <PrecompiledHeaderFile>precompiled.h</PrecompiledHeaderFile>
<RuntimeLibrary>MultiThreaded</RuntimeLibrary> <RuntimeLibrary>MultiThreaded</RuntimeLibrary>

View File

@ -69,7 +69,7 @@ const uint32 INVBITTABLE[] =
BitBuffer::BitBuffer() : m_Data(nullptr), BitBuffer::BitBuffer() : m_Data(nullptr),
m_CurByte(nullptr), m_CurByte(nullptr),
m_CurSize(0), m_CurBit(0),
m_MaxSize(0), m_MaxSize(0),
m_Overflowed(false), m_Overflowed(false),
m_LittleEndian(false), m_LittleEndian(false),
@ -83,7 +83,7 @@ BitBuffer::BitBuffer(void *newData, unsigned int size)
m_Data = (unsigned char *)newData; m_Data = (unsigned char *)newData;
m_CurByte = m_Data; m_CurByte = m_Data;
m_CurSize = 0; m_CurBit = 0;
m_MaxSize = size; m_MaxSize = size;
m_Overflowed = false; m_Overflowed = false;
m_LittleEndian = true; m_LittleEndian = true;
@ -100,7 +100,7 @@ BitBuffer::BitBuffer(unsigned int size)
m_Data = nullptr; m_Data = nullptr;
m_CurByte = nullptr; m_CurByte = nullptr;
m_CurSize = 0; m_CurBit = 0;
m_MaxSize = size; m_MaxSize = size;
m_Overflowed = false; m_Overflowed = false;
m_LittleEndian = false; m_LittleEndian = false;
@ -114,7 +114,7 @@ bool BitBuffer::Resize(unsigned int size)
Free(); Free();
m_Data = (unsigned char *)Mem_ZeroMalloc(size + 4); m_Data = (unsigned char *)Mem_ZeroMalloc(size + 4);
m_CurSize = 0; m_CurBit = 0;
m_Overflowed = false; m_Overflowed = false;
if (m_Data) if (m_Data)
@ -139,7 +139,7 @@ void BitBuffer::Clear()
Q_memset(m_Data, 0, m_MaxSize); Q_memset(m_Data, 0, m_MaxSize);
m_CurByte = m_Data; m_CurByte = m_Data;
m_CurSize = 0; m_CurBit = 0;
m_Overflowed = false; m_Overflowed = false;
m_LittleEndian = true; m_LittleEndian = true;
@ -147,13 +147,13 @@ void BitBuffer::Clear()
int BitBuffer::CurrentBit() int BitBuffer::CurrentBit()
{ {
return m_CurSize + 8 * (m_CurByte - m_Data); return m_CurBit + 8 * (m_CurByte - m_Data);
} }
void BitBuffer::Reset() void BitBuffer::Reset()
{ {
m_CurByte = m_Data; m_CurByte = m_Data;
m_CurSize = 0; m_CurBit = 0;
m_Overflowed = false; m_Overflowed = false;
m_LittleEndian = true; m_LittleEndian = true;
@ -168,7 +168,7 @@ void BitBuffer::Free()
m_Data = nullptr; m_Data = nullptr;
m_CurByte = nullptr; m_CurByte = nullptr;
m_CurSize = 0; m_CurBit = 0;
m_MaxSize = 0; m_MaxSize = 0;
m_OwnData = false; m_OwnData = false;
@ -187,26 +187,26 @@ unsigned int BitBuffer::ReadBits(int numbits)
return -1; return -1;
} }
int bits = m_CurSize + numbits; int bits = m_CurBit + numbits;
if (bits <= 32) if (bits <= 32)
{ {
result = (*(unsigned int *)m_CurByte >> m_CurSize) & ROWBITTABLE[numbits]; result = (*(unsigned int *)m_CurByte >> m_CurBit) & ROWBITTABLE[numbits];
m_CurByte += numbits >> 3; m_CurByte += numbits >> 3;
m_CurSize += numbits & 7; m_CurBit += numbits & 7;
if (m_CurSize > 7) if (m_CurBit > 7)
{ {
m_CurSize &= 7; m_CurBit &= 7;
m_CurByte++; m_CurByte++;
} }
} }
else else
{ {
unsigned int data = *(unsigned int *)m_CurByte >> m_CurSize; unsigned int data = *(unsigned int *)m_CurByte >> m_CurBit;
m_CurByte += 4; m_CurByte += 4;
result = ((ROWBITTABLE[bits & 7] & *(unsigned int *)m_CurByte) << (32 - m_CurSize)) | data; result = ((ROWBITTABLE[bits & 7] & *(unsigned int *)m_CurByte) << (32 - m_CurBit)) | data;
m_CurSize = bits & 7; m_CurBit = bits & 7;
} }
} }
else else
@ -235,26 +235,26 @@ int BitBuffer::ReadBit()
{ {
if (m_LittleEndian) if (m_LittleEndian)
{ {
if (m_CurSize == 7) if (m_CurBit == 7)
{ {
m_CurSize = 0; m_CurBit = 0;
result = (*m_CurByte++ >> 7) & 1; result = (*m_CurByte++ >> 7) & 1;
} }
else else
{ {
result = ((unsigned int)*m_CurByte >> m_CurSize++) & 1; result = ((unsigned int)*m_CurByte >> m_CurBit++) & 1;
} }
} }
else else
{ {
if (m_CurSize == 7) if (m_CurBit == 7)
{ {
m_CurSize = 0; m_CurBit = 0;
result = *m_CurByte++ & 1; result = *m_CurByte++ & 1;
} }
else else
{ {
result = ((unsigned int)*m_CurByte >> (7 - m_CurSize++)) & 1; result = ((unsigned int)*m_CurByte >> (7 - m_CurBit++)) & 1;
} }
} }
} }
@ -264,11 +264,11 @@ int BitBuffer::ReadBit()
unsigned int BitBuffer::PeekBits(int numbits) unsigned int BitBuffer::PeekBits(int numbits)
{ {
int oldcurrentBit = m_CurSize; int oldcurrentBit = m_CurBit;
unsigned char *oldcurrentByte = m_CurByte; unsigned char *oldcurrentByte = m_CurByte;
unsigned int data = ReadBits(numbits); unsigned int data = ReadBits(numbits);
m_CurSize = oldcurrentBit; m_CurBit = oldcurrentBit;
m_CurByte = oldcurrentByte; m_CurByte = oldcurrentByte;
return data; return data;
} }
@ -316,7 +316,7 @@ bool BitBuffer::ReadBuf(int iSize, void *pbuf)
return false; return false;
} }
if (m_CurSize) if (m_CurBit)
{ {
int i, j; int i, j;
unsigned int *p = (unsigned int *)pbuf; unsigned int *p = (unsigned int *)pbuf;
@ -402,7 +402,7 @@ void BitBuffer::WriteBit(int c)
if (m_LittleEndian) if (m_LittleEndian)
{ {
if (m_CurSize == 7) if (m_CurBit == 7)
{ {
if (c) if (c)
{ {
@ -414,20 +414,20 @@ void BitBuffer::WriteBit(int c)
} }
m_CurByte++; m_CurByte++;
m_CurSize = 0; m_CurBit = 0;
} }
else else
{ {
if (c) if (c)
{ {
m_CurByte[0] |= BITTABLE[ m_CurSize ]; m_CurByte[0] |= BITTABLE[ m_CurBit ];
} }
else else
{ {
m_CurByte[0] &= INVBITTABLE[ m_CurSize ]; m_CurByte[0] &= INVBITTABLE[ m_CurBit ];
} }
m_CurSize++; m_CurBit++;
} }
} }
else else
@ -436,13 +436,13 @@ void BitBuffer::WriteBit(int c)
static unsigned char inv_masks[] = { 0x7Fu, 0xBFu, 0xDFu, 0xEFu, 0xF7u, 0xFBu, 0xFDu, 0xFEu }; static unsigned char inv_masks[] = { 0x7Fu, 0xBFu, 0xDFu, 0xEFu, 0xF7u, 0xFBu, 0xFDu, 0xFEu };
if (c) if (c)
m_CurByte[0] |= masks[ m_CurSize ]; m_CurByte[0] |= masks[ m_CurBit ];
else else
m_CurByte[0] &= inv_masks[ m_CurSize ]; m_CurByte[0] &= inv_masks[ m_CurBit ];
if (++m_CurSize == 8) if (++m_CurBit == 8)
{ {
m_CurSize = 0; m_CurBit = 0;
m_CurByte++; m_CurByte++;
} }
} }
@ -461,26 +461,26 @@ void BitBuffer::WriteBits(unsigned int data, int numbits)
return; return;
} }
int bits = numbits + m_CurSize; int bits = numbits + m_CurBit;
if (bits <= 32) if (bits <= 32)
{ {
*(uint32 *)m_CurByte |= (ROWBITTABLE[numbits] & data) << m_CurSize; *(uint32 *)m_CurByte |= (ROWBITTABLE[numbits] & data) << m_CurBit;
m_CurByte = &m_CurByte[numbits >> 3]; m_CurByte = &m_CurByte[numbits >> 3];
m_CurSize = m_CurSize + (numbits & 7); m_CurBit = m_CurBit + (numbits & 7);
if (m_CurSize > 7) if (m_CurBit > 7)
{ {
m_CurSize = m_CurSize & 7; m_CurBit = m_CurBit & 7;
m_CurByte = m_CurByte + 1; m_CurByte = m_CurByte + 1;
} }
} }
else else
{ {
*(uint32 *)m_CurByte |= (ROWBITTABLE[numbits] & data) << m_CurSize; *(uint32 *)m_CurByte |= (ROWBITTABLE[numbits] & data) << m_CurBit;
int leftBits = (32 - m_CurSize); int leftBits = (32 - m_CurBit);
m_CurSize = (m_CurSize + numbits) & 7; m_CurBit = (m_CurBit + numbits) & 7;
m_CurByte += 4; m_CurByte += 4;
*(uint32 *)m_CurByte |= (ROWBITTABLE[numbits] & data) >> leftBits; *(uint32 *)m_CurByte |= (ROWBITTABLE[numbits] & data) >> leftBits;
@ -581,7 +581,7 @@ void BitBuffer::WriteBuf(const void *buf, int iSize)
return; return;
} }
if (m_CurSize) if (m_CurBit)
{ {
int i, j; int i, j;
unsigned int *p = (unsigned int *)buf; unsigned int *p = (unsigned int *)buf;
@ -623,7 +623,7 @@ void BitBuffer::WriteHiresAngle(float f)
int BitBuffer::CurrentSize() int BitBuffer::CurrentSize()
{ {
return (m_CurSize != 0) + m_CurByte - m_Data; return (m_CurBit != 0) + m_CurByte - m_Data;
} }
unsigned char *BitBuffer::CurrentByte() unsigned char *BitBuffer::CurrentByte()
@ -638,10 +638,10 @@ int BitBuffer::SpaceLeft()
void BitBuffer::AlignByte() void BitBuffer::AlignByte()
{ {
if (m_CurSize) if (m_CurBit)
{ {
m_CurByte++; m_CurByte++;
m_CurSize = 0; m_CurBit = 0;
} }
} }
@ -712,7 +712,7 @@ void BitBuffer::WriteBitString(const char *p)
void BitBuffer::StartBitMode() void BitBuffer::StartBitMode()
{ {
if (m_CurSize) { if (m_CurBit) {
m_Overflowed = true; m_Overflowed = true;
} }
} }
@ -739,7 +739,7 @@ void BitBuffer::SetBuffer(void *buffer, int size)
m_Data = (unsigned char *)buffer; m_Data = (unsigned char *)buffer;
m_CurByte = (unsigned char *)buffer; m_CurByte = (unsigned char *)buffer;
m_MaxSize = size; m_MaxSize = size;
m_CurSize = 0; m_CurBit = 0;
m_OwnData = false; m_OwnData = false;
m_Overflowed = false; m_Overflowed = false;
@ -811,22 +811,22 @@ void BitBuffer::SkipBits(int numbits)
return; return;
} }
int bits = m_CurSize + numbits; int bits = m_CurBit + numbits;
if (bits <= 32) if (bits <= 32)
{ {
m_CurByte += numbits >> 3; m_CurByte += numbits >> 3;
m_CurSize += numbits & 7; m_CurBit += numbits & 7;
if (m_CurSize > 7) if (m_CurBit > 7)
{ {
m_CurSize &= 7; m_CurBit &= 7;
m_CurByte++; m_CurByte++;
} }
} }
else else
{ {
m_CurByte += 4; m_CurByte += 4;
m_CurSize = bits & 7; m_CurBit = bits & 7;
} }
} }
else else
@ -835,14 +835,14 @@ void BitBuffer::SkipBits(int numbits)
while (d > 0) while (d > 0)
{ {
--d; --d;
if (m_CurSize == 7) if (m_CurBit == 7)
{ {
m_CurByte++; m_CurByte++;
m_CurSize = 0; m_CurBit = 0;
} }
else else
{ {
m_CurSize++; m_CurBit++;
} }
} }
} }
@ -870,7 +870,7 @@ void BitBuffer::FastClear()
Q_memset(m_Data, 0, iSize); Q_memset(m_Data, 0, iSize);
m_CurByte = m_Data; m_CurByte = m_Data;
m_CurSize = 0; m_CurBit = 0;
m_Overflowed = false; m_Overflowed = false;
m_LittleEndian = true; m_LittleEndian = true;

View File

@ -41,7 +41,6 @@ public:
unsigned char *CurrentByte(); unsigned char *CurrentByte();
int GetMaxSize() const { return m_MaxSize; } int GetMaxSize() const { return m_MaxSize; }
unsigned int GetCurSize() const { return m_CurSize; }
unsigned char *GetData() const { return m_Data; } unsigned char *GetData() const { return m_Data; }
bool IsOverflowed() const { return m_Overflowed; } bool IsOverflowed() const { return m_Overflowed; }
@ -108,7 +107,7 @@ public:
bool m_Overflowed; bool m_Overflowed;
unsigned char *m_Data; unsigned char *m_Data;
unsigned char *m_CurByte; unsigned char *m_CurByte;
int m_CurSize; int m_CurBit;
int m_MaxSize; int m_MaxSize;
protected: protected:

View File

@ -81,7 +81,7 @@ bool InfoString::SetString(char *string)
return false; return false;
} }
Q_strnlcpy(m_String, string, m_MaxSize); Q_strlcpy(m_String, string, m_MaxSize);
return true; return true;
} }
@ -95,7 +95,7 @@ void InfoString::SetMaxSize(unsigned int maxSize)
if (m_String) if (m_String)
{ {
if (maxSize > Q_strlen(m_String)) { if (maxSize > Q_strlen(m_String)) {
Q_strnlcpy(newBuffer, m_String, maxSize); Q_strlcpy(newBuffer, m_String, maxSize);
} }
Mem_Free(m_String); Mem_Free(m_String);

View File

@ -310,7 +310,12 @@ void NetChannel::UpdateFlow(int stream)
void NetChannel::TransmitOutgoing() void NetChannel::TransmitOutgoing()
{ {
#ifdef HLTV_FIXES
byte send_buf[MAX_UDP_PACKET];
#else
byte send_buf[NET_MAX_MESSAGE]; byte send_buf[NET_MAX_MESSAGE];
#endif
BitBuffer data(send_buf, sizeof(send_buf)); BitBuffer data(send_buf, sizeof(send_buf));
bool send_reliable; bool send_reliable;
@ -336,7 +341,7 @@ void NetChannel::TransmitOutgoing()
// check for reliable message overflow // check for reliable message overflow
if (m_reliableStream.IsOverflowed()) if (m_reliableStream.IsOverflowed())
{ {
m_System->DPrintf("Transmit:Outgoing m_reliableStream overflow (%s)\n", m_remote_address.ToString()); m_System->DPrintf("NetChannel::Transmit:Outgoing m_reliableStream overflow (%s)\n", m_remote_address.ToString());
m_reliableStream.Clear(); m_reliableStream.Clear();
return; return;
} }
@ -344,7 +349,7 @@ void NetChannel::TransmitOutgoing()
// check for unreliable message overflow // check for unreliable message overflow
if (m_unreliableStream.IsOverflowed()) if (m_unreliableStream.IsOverflowed())
{ {
m_System->DPrintf("Transmit:Outgoing m_unreliableStream overflow (%s)\n", m_remote_address.ToString()); m_System->DPrintf("NetChannel::Transmit:Outgoing m_unreliableStream overflow (%s)\n", m_remote_address.ToString());
m_unreliableStream.Clear(); m_unreliableStream.Clear();
} }
@ -653,6 +658,46 @@ bool NetChannel::CheckForCompletion(int stream, int intotalbuffers)
return false; return false;
} }
bool NetChannel::ValidateFragments(BitBuffer &buf, bool *frag_message, unsigned int *fragid, int *frag_offset, int *frag_length)
{
for (int i = 0; i < MAX_STREAMS; i++)
{
if (!frag_message[i])
continue;
// total fragments should be <= MAX_FRAGMENTS and current fragment can't be > total fragments
if (i == FRAG_NORMAL_STREAM && FRAG_GETCOUNT(fragid[i]) > MAX_NORMAL_FRAGMENTS)
return false;
if (i == FRAG_FILE_STREAM && FRAG_GETCOUNT(fragid[i]) > MAX_FILE_FRAGMENTS)
return false;
if (FRAG_GETID(fragid[i]) > FRAG_GETCOUNT(fragid[i]))
return false;
if (!frag_length[i])
return false;
if ((size_t)frag_length[i] > FRAGMENT_MAX_SIZE || (size_t)frag_offset[i] > MAX_POSSIBLE_MSG - 1)
return false;
int frag_end = frag_offset[i] + frag_length[i];
// end of fragment is out of the packet
if (frag_end + buf.CurrentSize() > buf.GetMaxSize())
return false;
// fragment overlaps next stream's fragment or placed after it
for (int j = i + 1; j < MAX_STREAMS; j++)
{
if (frag_end > frag_offset[j] && frag_message[j]) // don't add buf.CurrentSize() for comparison
return false;
}
}
return true;
}
void NetChannel::ProcessIncoming(unsigned char *data, int size) void NetChannel::ProcessIncoming(unsigned char *data, int size)
{ {
BitBuffer message(data, size); BitBuffer message(data, size);
@ -721,6 +766,11 @@ void NetChannel::ProcessIncoming(unsigned char *data, int size)
frag_length[i] = message.ReadShort(); frag_length[i] = message.ReadShort();
} }
} }
#ifdef HLTV_FIXES
if (!ValidateFragments(message, frag_message, fragid, frag_offset, frag_length))
return;
#endif
} }
sequence &= ~(1 << 31); sequence &= ~(1 << 31);
@ -758,7 +808,12 @@ void NetChannel::ProcessIncoming(unsigned char *data, int size)
// clear the buffer to make way for the next // clear the buffer to make way for the next
if (reliable_ack == (unsigned int)m_reliable_sequence) if (reliable_ack == (unsigned int)m_reliable_sequence)
{ {
// Make sure we actually could have ack'd this message
#ifdef HLTV_FIXES
if (sequence_ack >= (unsigned)m_last_reliable_sequence)
#else
if (m_incoming_acknowledged + 1 >= m_last_reliable_sequence) if (m_incoming_acknowledged + 1 >= m_last_reliable_sequence)
#endif
{ {
// it has been received // it has been received
m_reliableOutSize = 0; m_reliableOutSize = 0;
@ -1104,6 +1159,24 @@ void NetChannel::CopyNormalFragments()
p = n; p = n;
} }
#ifdef HLTV_FIXES
if (packet->data.IsOverflowed())
{
if (packet->address.IsValid())
{
m_System->Printf("WARNING! NetChannel::CopyNormalFragments: Incoming overflowed from %s\n", packet->address.ToString());
}
else
{
m_System->Printf("WARNING! NetChannel::CopyNormalFragments: Incoming overflowed\n");
}
packet->data.Clear();
m_incomingbufs[FRAG_NORMAL_STREAM] = nullptr;
return;
}
#endif
if (*(uint32 *)packet->data.GetData() == MAKEID('B', 'Z', '2', '\0')) if (*(uint32 *)packet->data.GetData() == MAKEID('B', 'Z', '2', '\0'))
{ {
char uncompressed[65536]; char uncompressed[65536];

View File

@ -53,6 +53,15 @@ enum
#define MAX_LATENT 32 #define MAX_LATENT 32
#define FRAGMENT_MAX_SIZE 1400 // Size of fragmentation buffer internal buffers #define FRAGMENT_MAX_SIZE 1400 // Size of fragmentation buffer internal buffers
#define CLIENT_FRAGMENT_SIZE_ONCONNECT 128
#define CUSTOMIZATION_MAX_SIZE 20480
// Client sends normal fragments only while connecting
#define MAX_NORMAL_FRAGMENTS (MAX_POSSIBLE_MSG / CLIENT_FRAGMENT_SIZE_ONCONNECT)
// While client is connecting it sending fragments with minimal size, also it transfers sprays with minimal fragments...
// But with sv_delayed_spray_upload it sends with cl_dlmax fragment size
#define MAX_FILE_FRAGMENTS (CUSTOMIZATION_MAX_SIZE / FRAGMENT_C2S_MIN_SIZE)
#define UDP_HEADER_SIZE 28 #define UDP_HEADER_SIZE 28
#define MAX_RELIABLE_PAYLOAD 1200 #define MAX_RELIABLE_PAYLOAD 1200
@ -152,6 +161,7 @@ public:
fragbuf_t *fragbufs; // The actual buffers fragbuf_t *fragbufs; // The actual buffers
} fragbufwaiting_t; } fragbufwaiting_t;
bool ValidateFragments(BitBuffer &buf, bool *frag_message, unsigned int *fragid, int *frag_offset, int *frag_length);
bool CreateFragmentsFromFile(char *fileName); bool CreateFragmentsFromFile(char *fileName);
bool CopyFileFragments(); bool CopyFileFragments();
void GetFlowStats(float *avgInKBSec, float *avgOutKBSec); void GetFlowStats(float *avgInKBSec, float *avgOutKBSec);

View File

@ -2365,17 +2365,56 @@ void Host_Version_f(void)
void Host_FullInfo_f(void) void Host_FullInfo_f(void)
{ {
char key[512];
char value[512];
char *o;
char *s;
if (Cmd_Argc() != 2) if (Cmd_Argc() != 2)
{ {
Con_Printf("fullinfo <complete info string>\n"); Con_Printf("fullinfo <complete info string>\n");
return; return;
} }
#ifdef REHLDS_FIXES
char copy[MAX_INFO_STRING];
Q_strlcpy(copy, Cmd_Argv(1));
char* s = copy;
if (*s != '\\')
return;
bool eos = false;
while (!eos) {
const char* key = ++s;
// key
while (*s != '\\')
{
// key should end with a '\', not a NULL
if (*s == '\0') {
Con_Printf("MISSING VALUE\n");
return;
}
s++;
}
*s = '\0';
const char* value = ++s;
// value
while (*s != '\\') {
if (*s == '\0') {
eos = true;
break;
}
s++;
}
*s = '\0';
#else
char key[512];
char value[512];
char *o;
char *s;
s = (char *)Cmd_Argv(1); s = (char *)Cmd_Argv(1);
if (*s == '\\') if (*s == '\\')
s++; s++;
@ -2400,6 +2439,7 @@ void Host_FullInfo_f(void)
*o = 0; *o = 0;
if (*s) if (*s)
s++; s++;
#endif
if (cmd_source == src_command) if (cmd_source == src_command)
{ {

View File

@ -31,10 +31,85 @@
// NOTE: This file contains a lot of fixes that are not covered by REHLDS_FIXES define. // NOTE: This file contains a lot of fixes that are not covered by REHLDS_FIXES define.
// TODO: Most of the Info_ functions can be speedup via removing unneded copy of key and values. // TODO: Most of the Info_ functions can be speedup via removing unneded copy of key and values.
struct info_field_t
{
char* name;
bool integer;
};
info_field_t g_info_important_fields[] =
{
// name integer
{ "name", false },
{ "model", false },
// model colors
{ "topcolor", true },
{ "bottomcolor", true },
// network
{ "rate", true },
{ "cl_updaterate", true },
{ "cl_lw", true },
{ "cl_lc", true },
// hltv flag
{ "*hltv", true },
// avatars
{ "*sid", false }, // transmit as string because it's int64
// gui/text menus
{ "_vgui_menus", true }
};
std::vector<info_field_t *> g_info_transmitted_fields;
// Searches the string for the given // Searches the string for the given
// key and returns the associated value, or an empty string. // key and returns the associated value, or an empty string.
const char* EXT_FUNC Info_ValueForKey(const char *s, const char *key) const char* EXT_FUNC Info_ValueForKey(const char *s, const char *lookup)
{ {
#ifdef REHLDS_FIXES
static char valueBuf[INFO_MAX_BUFFER_VALUES][MAX_KV_LEN];
static int valueIndex;
size_t lookupLen = Q_strlen(lookup);
while (*s == '\\')
{
// skip starting slash
const char* key = ++s;
// skip key
while (*s != '\\') {
// Add some sanity checks because it's external function
if (*s == '\0')
return "";
s++;
}
size_t keyLen = s - key;
const char* value = ++s; // skip separating slash
// skip value
while (*s != '\\' && *s != '\0')
s++;
size_t valueLen = Q_min(s - value, MAX_KV_LEN - 1);
if (keyLen == lookupLen && !Q_strncmp(key, lookup, lookupLen))
{
char* dest = valueBuf[valueIndex];
Q_memcpy(dest, value, valueLen);
dest[valueLen] = '\0';
valueIndex = (valueIndex + 1) % INFO_MAX_BUFFER_VALUES;
return dest;
}
}
return "";
#else
// use few (two?) buffers so compares work without stomping on each other // use few (two?) buffers so compares work without stomping on each other
static char value[INFO_MAX_BUFFER_VALUES][MAX_KV_LEN]; static char value[INFO_MAX_BUFFER_VALUES][MAX_KV_LEN];
static int valueindex; static int valueindex;
@ -88,7 +163,7 @@ const char* EXT_FUNC Info_ValueForKey(const char *s, const char *key)
} }
*c = 0; *c = 0;
if (!Q_strcmp(key, pkey)) if (!Q_strcmp(lookup, pkey))
{ {
c = value[valueindex]; c = value[valueindex];
valueindex = (valueindex + 1) % INFO_MAX_BUFFER_VALUES; valueindex = (valueindex + 1) % INFO_MAX_BUFFER_VALUES;
@ -97,10 +172,47 @@ const char* EXT_FUNC Info_ValueForKey(const char *s, const char *key)
} }
return ""; return "";
#endif
} }
void Info_RemoveKey(char *s, const char *key) void Info_RemoveKey(char *s, const char *lookup)
{ {
#ifdef REHLDS_FIXES
size_t lookupLen = Q_strlen(lookup);
while (*s == '\\')
{
char* start = s;
// skip starting slash
const char* key = ++s;
// skip key
while (*s != '\\') {
if (*s == '\0')
return;
s++;
}
size_t keyLen = s - key;
++s; // skip separating slash
// skip value
while (*s != '\\' && *s != '\0')
s++;
if (keyLen != lookupLen)
continue;
if (!Q_memcmp(key, lookup, lookupLen))
{
// cut key and value
Q_memmove(start, s, Q_strlen(s) + 1);
break;
}
}
#else
char pkey[MAX_KV_LEN]; char pkey[MAX_KV_LEN];
char value[MAX_KV_LEN]; char value[MAX_KV_LEN];
char *start; char *start;
@ -108,13 +220,13 @@ void Info_RemoveKey(char *s, const char *key)
int cmpsize; int cmpsize;
int nCount; int nCount;
if (Q_strstr(key, "\\")) if (Q_strstr(lookup, "\\"))
{ {
Con_Printf("Can't use a key with a \\\n"); Con_Printf("Can't use a key with a \\\n");
return; return;
} }
cmpsize = Q_strlen(key); cmpsize = Q_strlen(lookup);
if (cmpsize > MAX_KV_LEN - 1) if (cmpsize > MAX_KV_LEN - 1)
cmpsize = MAX_KV_LEN - 1; cmpsize = MAX_KV_LEN - 1;
@ -168,16 +280,47 @@ void Info_RemoveKey(char *s, const char *key)
*c = 0; *c = 0;
// Compare keys // Compare keys
if (!Q_strncmp(key, pkey, cmpsize)) if (!Q_strncmp(lookup, pkey, cmpsize))
{ {
Q_strcpy_s(start, s); // remove this part Q_strcpy_s(start, s); // remove this part
s = start; // continue searching s = start; // continue searching
} }
} }
#endif
} }
void Info_RemovePrefixedKeys(char *s, const char prefix) void Info_RemovePrefixedKeys(char *s, const char prefix)
{ {
#ifdef REHLDS_FIXES
while (*s == '\\')
{
char* start = s;
// skip starting slash
const char* key = ++s;
// skip key
while (*s != '\\') {
if (*s == '\0')
return;
s++;
}
// skip separating slash
++s;
// skip value
while (*s != '\\' && *s != '\0')
s++;
if (key[0] == prefix)
{
Q_memmove(start, s, Q_strlen(s) + 1);
s = start;
}
}
#else
char pkey[MAX_KV_LEN]; char pkey[MAX_KV_LEN];
char value[MAX_KV_LEN]; char value[MAX_KV_LEN];
char *start; char *start;
@ -239,12 +382,37 @@ void Info_RemovePrefixedKeys(char *s, const char prefix)
s = start; // continue searching s = start; // continue searching
} }
} }
#endif
} }
#ifdef REHLDS_FIXES
qboolean Info_IsKeyImportant(const char *key) qboolean Info_IsKeyImportant(const char *key)
{ {
if (key[0] == '*') if (key[0] == '*')
return true; return true;
for (auto& field : g_info_important_fields) {
if (!Q_strcmp(key, field.name))
return true;
}
return false;
}
qboolean Info_IsKeyImportant(const char *key, size_t keyLen)
{
char copy[MAX_KV_LEN];
keyLen = min(keyLen, sizeof(copy) - 1);
Q_memcpy(copy, key, keyLen);
copy[keyLen] = '\0';
return Info_IsKeyImportant(copy);
}
#else
qboolean Info_IsKeyImportant(const char *key)
{
if (key[0] == '*')
return true;
if (!Q_strcmp(key, "name")) if (!Q_strcmp(key, "name"))
return true; return true;
if (!Q_strcmp(key, "model")) if (!Q_strcmp(key, "model"))
@ -261,19 +429,53 @@ qboolean Info_IsKeyImportant(const char *key)
return true; return true;
if (!Q_strcmp(key, "cl_lc")) if (!Q_strcmp(key, "cl_lc"))
return true; return true;
#ifndef REHLDS_FIXES
// keys starts from '*' already checked
if (!Q_strcmp(key, "*hltv")) if (!Q_strcmp(key, "*hltv"))
return true; return true;
if (!Q_strcmp(key, "*sid")) if (!Q_strcmp(key, "*sid"))
return true; return true;
#endif
return false; return false;
} }
#endif
char *Info_FindLargestKey(char *s, int maxsize) const char *Info_FindLargestKey(const char *s, int maxsize)
{ {
#ifdef REHLDS_FIXES
static char largestKey[MAX_KV_LEN];
size_t largestLen = 0;
while (*s == '\\')
{
// skip starting slash
const char* key = ++s;
// skip key
while (*s != '\\') {
if (*s == '\0')
return "";
s++;
}
size_t keyLen = s - key;
const char* value = ++s; // skip separating slash
// skip value
while (*s != '\\' && *s != '\0')
s++;
size_t valueLen = s - value;
size_t totalLen = keyLen + valueLen;
if (totalLen > largestLen && !Info_IsKeyImportant(key, keyLen)) {
largestLen = totalLen;
Q_memcpy(largestKey, key, keyLen);
largestKey[keyLen] = '\0';
}
}
return largestLen ? largestKey : "";
#else
static char largest_key[MAX_KV_LEN]; static char largest_key[MAX_KV_LEN];
char key[MAX_KV_LEN]; char key[MAX_KV_LEN];
char value[MAX_KV_LEN]; char value[MAX_KV_LEN];
@ -347,8 +549,104 @@ char *Info_FindLargestKey(char *s, int maxsize)
} }
return largest_key; return largest_key;
#endif
} }
#ifdef REHLDS_FIXES
qboolean Info_SetValueForStarKey(char *s, const char *key, const char *value, size_t maxsize)
{
char newArray[MAX_INFO_STRING], valueBuf[MAX_KV_LEN];
if (!key || !value)
{
Con_Printf("Keys and values can't be null\n");
return FALSE;
}
if (key[0] == '\0')
{
Con_Printf("Keys can't be an empty string\n");
return FALSE;
}
if (Q_strchr(key, '\\') || Q_strchr(value, '\\'))
{
Con_Printf("Can't use keys or values with a \\\n");
return FALSE;
}
if (Q_strchr(key, '\"') || Q_strchr(value, '\"'))
{
Con_Printf("Can't use keys or values with a \"\n");
return FALSE;
}
if (Q_strstr(key, "..") || Q_strstr(value, ".."))
{
Con_Printf("Can't use keys or values with a ..\n");
return FALSE;
}
int keyLen = Q_strlen(key);
int valueLen = Q_strlen(value);
if (keyLen >= MAX_KV_LEN || valueLen >= MAX_KV_LEN)
{
Con_Printf("Keys and values must be < %i characters\n", MAX_KV_LEN);
return FALSE;
}
if (!Q_UnicodeValidate(key) || !Q_UnicodeValidate(value))
{
Con_Printf("Keys and values must be valid utf8 text\n");
return FALSE;
}
// Remove current key/value and return if we doesn't specified to set a value
Info_RemoveKey(s, key);
if (value[0] == '\0')
{
return TRUE;
}
// auto lowercase team
if (!Q_strcmp(key, "team")) {
value = Q_strcpy(valueBuf, value);
Q_strlwr(valueBuf);
}
// Create key/value pair
size_t neededLength = Q_snprintf(newArray, sizeof newArray, "\\%s\\%s", key, value);
if (Q_strlen(s) + neededLength >= maxsize)
{
// no more room in the buffer to add key/value
if (!Info_IsKeyImportant(key))
{
// no room to add setting
Con_Printf("Info string length exceeded\n");
return FALSE;
}
// keep removing the largest key/values until we have a room
do
{
const char* largekey = Info_FindLargestKey(s, maxsize);
if (largekey[0] == '\0')
{
// no room to add setting
Con_Printf("Info string length exceeded\n");
return FALSE;
}
Info_RemoveKey(s, largekey);
} while ((int)Q_strlen(s) + neededLength >= maxsize);
}
Q_strcat(s, newArray);
return TRUE;
}
#else
void Info_SetValueForStarKey(char *s, const char *key, const char *value, int maxsize) void Info_SetValueForStarKey(char *s, const char *key, const char *value, int maxsize)
{ {
char newArray[MAX_INFO_STRING]; char newArray[MAX_INFO_STRING];
@ -424,7 +722,7 @@ void Info_SetValueForStarKey(char *s, const char *key, const char *value, int ma
} }
// keep removing the largest key/values until we have a room // keep removing the largest key/values until we have a room
char *largekey; const char *largekey;
do do
{ {
largekey = Info_FindLargestKey(s, maxsize); largekey = Info_FindLargestKey(s, maxsize);
@ -453,6 +751,7 @@ void Info_SetValueForStarKey(char *s, const char *key, const char *value, int ma
} }
*s = 0; *s = 0;
} }
#endif
void Info_SetValueForKey(char *s, const char *key, const char *value, int maxsize) void Info_SetValueForKey(char *s, const char *key, const char *value, int maxsize)
{ {
@ -541,6 +840,89 @@ void Info_Print(const char *s)
qboolean Info_IsValid(const char *s) qboolean Info_IsValid(const char *s)
{ {
#ifdef REHLDS_FIXES
struct {
const char* start;
size_t len;
} existingKeys[MAX_INFO_STRING * 2 / 4];
size_t existingKeysNum = 0;
auto isAlreadyExists = [&](const char* key, size_t len)
{
for (size_t i = 0; i < existingKeysNum; i++) {
if (len == existingKeys[i].len && !Q_memcmp(key, existingKeys[i].start, existingKeys[i].len))
return true;
}
return false;
};
while (*s == '\\')
{
const char* key = ++s;
// keys and values are separated by another slash
while (*s != '\\')
{
// key should end with a '\', not a NULL
if (*s == '\0')
return FALSE;
// quotes are deprecated
if (*s == '"')
return FALSE;
// ".." deprecated. don't know why. model path?
if (*s == '.' && *(s + 1) == '.')
return FALSE;
s++;
}
size_t keyLen = s - key;
if (keyLen == 0 || keyLen >= MAX_KV_LEN)
return FALSE;
if (isAlreadyExists(key, keyLen))
return FALSE;
const char* value = ++s; // skip the slash
// values should be ended by eos or slash
while (*s != '\\' && *s != '\0')
{
// quotes are deprecated
if (*s == '"')
return FALSE;
// ".." deprecated
if (*s == '.' && *(s + 1) == '.')
return FALSE;
s++;
}
size_t valueLen = s - value;
if (valueLen == 0 || valueLen >= MAX_KV_LEN)
return FALSE;
if (*s == '\0')
return TRUE;
if (existingKeysNum == ARRAYSIZE(existingKeys))
return FALSE;
existingKeys[existingKeysNum].start = key;
existingKeys[existingKeysNum].len = keyLen;
existingKeysNum++;
}
return FALSE;
#else
char key[MAX_KV_LEN];
char value[MAX_KV_LEN];
char *c;
int nCount;
while (*s) while (*s)
{ {
if (*s == '\\') if (*s == '\\')
@ -548,48 +930,46 @@ qboolean Info_IsValid(const char *s)
s++; // skip the slash s++; // skip the slash
} }
// Returns character count // Copy a key
// -1 - error nCount = 0;
// 0 - string size is zero c = key;
enum class AllowNull { while (*s != '\\')
Yes,
No,
};
auto validate = [&s](AllowNull allowNull) -> int
{
int nCount = 0;
for(; *s != '\\'; nCount++, s++)
{ {
if (!*s) if (!*s)
{ {
return (allowNull == AllowNull::Yes) ? nCount : -1; return FALSE; // key should end with a \, not a NULL
} }
if (nCount >= MAX_KV_LEN) if (nCount >= MAX_KV_LEN)
{ {
return -1; // string length should be less then MAX_KV_LEN return FALSE; // key length should be less then MAX_KV_LEN
} }
*c++ = *s++;
nCount++;
}
*c = 0;
s++; // skip the slash
#ifdef REHLDS_FIXES // Copy a value
if (*s == '\"') nCount = 0;
c = value;
while (*s != '\\')
{ {
return -1; // string should not contain " if (!*s)
{
break; // allow value to be ended with NULL
} }
#endif if (nCount >= MAX_KV_LEN)
{
return FALSE; // value length should be less then MAX_KV_LEN
} }
return nCount; *c++ = *s++;
}; nCount++;
}
*c = 0;
if (validate(AllowNull::No) == -1) if (value[0] == 0)
{ {
return FALSE; return FALSE; // empty values are not valid
}
s++; // Skip slash
if (validate(AllowNull::Yes) <= 0)
{
return FALSE;
} }
if (!*s) if (!*s)
@ -599,52 +979,89 @@ qboolean Info_IsValid(const char *s)
} }
return FALSE; return FALSE;
#endif
} }
#ifdef REHLDS_FIXES #ifdef REHLDS_FIXES
void Info_CollectFields(char *destInfo, const char *srcInfo, const char *collectedKeysOfFields) void Info_SetFieldsToTransmit()
{ {
char keys[MAX_INFO_STRING]; // clean all
Q_strcpy(keys, collectedKeysOfFields); for (auto field : g_info_transmitted_fields) {
free(field->name);
delete field;
}
g_info_transmitted_fields.clear();
char keys[512];
Q_strlcpy(keys, sv_rehlds_userinfo_transmitted_fields.string);
auto isIntegerField = [](const char* key)
{
for (auto& x : g_info_important_fields) {
if (!Q_strcmp(key, x.name))
return x.integer;
}
return false;
};
for (char *key = Q_strtok(keys, "\\"); key; key = Q_strtok(nullptr, "\\"))
{
if (key[0] == '_') {
Con_Printf("%s: private key '%s' couldn't be transmitted.\n", __FUNCTION__, key);
continue;
}
if (Q_strlen(key) >= MAX_KV_LEN) {
Con_Printf("%s: key '%s' is too long (should be < %i characters)\n", __FUNCTION__, key, MAX_KV_LEN);
continue;
}
if (std::find_if(g_info_transmitted_fields.begin(), g_info_transmitted_fields.end(), [key](info_field_t* field) { return !Q_strcmp(key, field->name); }) == g_info_transmitted_fields.end()) {
auto field = new info_field_t;
field->name = Q_strdup(key);
field->integer = isIntegerField(key);
g_info_transmitted_fields.push_back(field);
}
}
}
void Info_CollectFields(char *destInfo, const char *srcInfo, size_t maxsize)
{
if (g_info_transmitted_fields.empty()) {
Q_strlcpy(destInfo, srcInfo, maxsize);
Info_RemovePrefixedKeys(destInfo, '_');
return;
}
char add[512], valueBuf[32];
size_t userInfoLength = 0; size_t userInfoLength = 0;
for (const char *key = strtok(keys, "\\"); key; key = strtok(nullptr, "\\"))
{
const char *value = Info_ValueForKey(srcInfo, key);
for (auto field : g_info_transmitted_fields)
{
const char *value = Info_ValueForKey(srcInfo, field->name);
if (value[0] == '\0') if (value[0] == '\0')
continue; continue;
// Integer fields // clean garbage from integer fields
if (!Q_strcmp(key, "*hltv") if (field->integer)
|| !Q_strcmp(key, "bottomcolor")
|| !Q_strcmp(key, "topcolor"))
{ {
// don't send zero fields
int intValue = Q_atoi(value); int intValue = Q_atoi(value);
if (!intValue) if (!intValue)
continue; continue;
destInfo[userInfoLength++] = '\\'; Q_sprintf(valueBuf, "%i", intValue);
Q_strcpy(&destInfo[userInfoLength], key); value = valueBuf;
userInfoLength += Q_strlen(key);
destInfo[userInfoLength++] = '\\';
userInfoLength += Q_sprintf(&destInfo[userInfoLength], "%d", intValue);
}
// String fields
else
{
destInfo[userInfoLength++] = '\\';
Q_strcpy(&destInfo[userInfoLength], key);
userInfoLength += Q_strlen(key);
destInfo[userInfoLength++] = '\\';
Q_strcpy(&destInfo[userInfoLength], value);
userInfoLength += Q_strlen(value);
} }
// don't write truncated keys/values
size_t len = Q_sprintf(add, "\\%s\\%s", field->name, value);
if (userInfoLength + len < maxsize) {
Q_strcpy(destInfo + userInfoLength, add);
userInfoLength += len;
} }
}
destInfo[userInfoLength] = '\0'; destInfo[userInfoLength] = '\0';
} }
#endif // REHLDS_FIXES #endif // REHLDS_FIXES

View File

@ -48,11 +48,16 @@ const char *Info_ValueForKey(const char *s, const char *key);
void Info_RemoveKey(char *s, const char *key); void Info_RemoveKey(char *s, const char *key);
void Info_RemovePrefixedKeys(char *s, const char prefix); void Info_RemovePrefixedKeys(char *s, const char prefix);
qboolean Info_IsKeyImportant(const char *key); qboolean Info_IsKeyImportant(const char *key);
char *Info_FindLargestKey(char *s, int maxsize); const char *Info_FindLargestKey(const char *s, int maxsize);
#ifdef REHLDS_FIXES
qboolean Info_SetValueForStarKey(char *s, const char *key, const char *value, size_t maxsize);
#else
void Info_SetValueForStarKey(char *s, const char *key, const char *value, int maxsize); void Info_SetValueForStarKey(char *s, const char *key, const char *value, int maxsize);
#endif
void Info_SetValueForKey(char *s, const char *key, const char *value, int maxsize); void Info_SetValueForKey(char *s, const char *key, const char *value, int maxsize);
void Info_Print(const char *s); void Info_Print(const char *s);
qboolean Info_IsValid(const char *s); qboolean Info_IsValid(const char *s);
#ifdef REHLDS_FIXES #ifdef REHLDS_FIXES
void Info_CollectFields(char *destInfo, const char *srcInfo, const char *collectedKeysOfFields); void Info_SetFieldsToTransmit();
void Info_CollectFields(char *destInfo, const char *srcInfo, size_t maxsize);
#endif #endif

View File

@ -356,7 +356,6 @@ void EXT_FUNC PF_traceline_DLL(const float *v1, const float *v2, int fNoMonsters
void EXT_FUNC TraceHull(const float *v1, const float *v2, int fNoMonsters, int hullNumber, edict_t *pentToSkip, TraceResult *ptr) void EXT_FUNC TraceHull(const float *v1, const float *v2, int fNoMonsters, int hullNumber, edict_t *pentToSkip, TraceResult *ptr)
{ {
hullNumber = hullNumber;
if (hullNumber < 0 || hullNumber > 3) if (hullNumber < 0 || hullNumber > 3)
hullNumber = 0; hullNumber = 0;

View File

@ -265,6 +265,35 @@ char *ED_ParseEdict(char *data, edict_t *ent)
Q_strcpy(keyname, "angles"); Q_strcpy(keyname, "angles");
} }
#ifdef REHLDS_FIXES
else if (!Q_strcmp(keyname, "model"))
{
// local model?
if (com_token[0] == '*')
{
// find empty slot
int i;
for (i = 0; i < MAX_MODELS; i++)
{
if (!g_psv.model_precache[i])
break;
}
int index = Q_atoi(com_token + 1);
g_psv.model_precache[i] = localmodels[index];
g_psv.models[i] = Mod_ForName(localmodels[index], FALSE, FALSE);
g_psv.model_precache_flags[i] |= RES_FATALIFMISSING;
#ifdef REHLDS_OPT_PEDANTIC
{
int __itmp = i;
g_rehlds_sv.modelsMap.put(g_psv.model_precache[i], __itmp);
}
#endif
}
}
#endif
kvd.szClassName = className; kvd.szClassName = className;
kvd.szKeyName = keyname; kvd.szKeyName = keyname;

View File

@ -1937,7 +1937,7 @@ int EXT_FUNC SV_CheckKeyInfo_internal(netadr_t *adr, char *protinfo, unsigned sh
s = Info_ValueForKey(protinfo, "raw"); s = Info_ValueForKey(protinfo, "raw");
if (s[0] == 0 || (nAuthProtocol == 2 && Q_strlen(s) != 32)) if (s[0] == '\0' || (nAuthProtocol == 2 && Q_strlen(s) != 32))
{ {
SV_RejectConnection(adr, "Invalid authentication certificate length.\n"); SV_RejectConnection(adr, "Invalid authentication certificate length.\n");
return 0; return 0;
@ -2061,7 +2061,6 @@ int SV_CheckUserInfo(netadr_t *adr, char *userinfo, qboolean bIsReconnecting, in
const char *s; const char *s;
char newname[MAX_NAME]; char newname[MAX_NAME];
int proxies; int proxies;
int i;
if (!NET_IsLocalAddress(*adr)) if (!NET_IsLocalAddress(*adr))
{ {
@ -2086,13 +2085,15 @@ int SV_CheckUserInfo(netadr_t *adr, char *userinfo, qboolean bIsReconnecting, in
} }
} }
i = Q_strlen(userinfo); #ifndef REHLDS_FIXES
int i = Q_strlen(userinfo);
if (i <= 4 || Q_strstr(userinfo, "\\\\") || userinfo[i - 1] == '\\') if (i <= 4 || Q_strstr(userinfo, "\\\\") || userinfo[i - 1] == '\\')
{ {
SV_RejectConnection(adr, "Unknown HLTV client type.\n"); SV_RejectConnection(adr, "Unknown HLTV client type.\n");
return 0; return 0;
} }
#endif
Info_RemoveKey(userinfo, "password"); Info_RemoveKey(userinfo, "password");
@ -2132,9 +2133,9 @@ int SV_CheckUserInfo(netadr_t *adr, char *userinfo, qboolean bIsReconnecting, in
#endif #endif
#ifdef REHLDS_FIXES #ifdef REHLDS_FIXES
if (name[0] == 0 || !Q_stricmp(name, "console") || Q_strstr(name, "..") || Q_strstr(name, "\"") || Q_strstr(name, "\\")) if (name[0] == '\0' || !Q_stricmp(name, "console"))
#else // REHLDS_FIXES #else // REHLDS_FIXES
if (name[0] == 0 || !Q_stricmp(name, "console") || Q_strstr(name, "..") != NULL) if (name[0] == '\0' || !Q_stricmp(name, "console") || Q_strstr(name, "..") != NULL)
#endif // REHLDS_FIXES #endif // REHLDS_FIXES
{ {
Info_SetValueForKey(userinfo, "name", "unnamed", MAX_INFO_STRING); Info_SetValueForKey(userinfo, "name", "unnamed", MAX_INFO_STRING);
@ -2147,11 +2148,10 @@ int SV_CheckUserInfo(netadr_t *adr, char *userinfo, qboolean bIsReconnecting, in
if (SV_CheckForDuplicateNames(userinfo, bIsReconnecting, nReconnectSlot)) if (SV_CheckForDuplicateNames(userinfo, bIsReconnecting, nReconnectSlot))
{ {
Q_strncpy(name, Info_ValueForKey(userinfo, "name"), MAX_NAME - 1); Q_strncpy(name, Info_ValueForKey(userinfo, "name"), MAX_NAME - 1);
name[MAX_NAME - 1] = 0; name[MAX_NAME - 1] = '\0';
} }
s = Info_ValueForKey(userinfo, "*hltv"); s = Info_ValueForKey(userinfo, "*hltv");
if (!s[0]) if (!s[0])
return 1; return 1;
@ -3760,17 +3760,12 @@ void SV_FullClientUpdate(client_t *cl, sizebuf_t *sb)
char info[MAX_INFO_STRING]; char info[MAX_INFO_STRING];
#ifdef REHLDS_FIXES #ifdef REHLDS_FIXES
if (sv_rehlds_userinfo_transmitted_fields.string[0] != '\0') Info_CollectFields(info, cl->userinfo, MAX_INFO_STRING);
{ #else // REHLDS_FIXES
Info_CollectFields(info, cl->userinfo, sv_rehlds_userinfo_transmitted_fields.string);
}
else
#endif // REHLDS_FIXES
{
Q_strncpy(info, cl->userinfo, sizeof(info) - 1); Q_strncpy(info, cl->userinfo, sizeof(info) - 1);
info[sizeof(info) - 1] = 0; info[sizeof(info) - 1] = '\0';
Info_RemovePrefixedKeys(info, '_'); Info_RemovePrefixedKeys(info, '_');
} #endif // REHLDS_FIXES
g_RehldsHookchains.m_SV_WriteFullClientUpdate.callChain(SV_WriteFullClientUpdate_internal, GetRehldsApiClient(cl), info, MAX_INFO_STRING, sb, GetRehldsApiClient((sb == &g_psv.reliable_datagram) ? nullptr : host_client)); g_RehldsHookchains.m_SV_WriteFullClientUpdate.callChain(SV_WriteFullClientUpdate_internal, GetRehldsApiClient(cl), info, MAX_INFO_STRING, sb, GetRehldsApiClient((sb == &g_psv.reliable_datagram) ? nullptr : host_client));
} }
@ -4914,12 +4909,7 @@ void SV_ExtractFromUserinfo(client_t *cl)
Q_UnicodeRepair(newname); Q_UnicodeRepair(newname);
} }
if (newname[0] == '\0' || !Q_stricmp(newname, "console") if (newname[0] == '\0' || !Q_stricmp(newname, "console"))
#ifdef REHLDS_FIXES
|| Q_strstr(newname, "..") || Q_strstr(newname, "\"") || Q_strstr(newname, "\\"))
#else // REHLDS_FIXES
)
#endif // REHLDS_FIXES
{ {
Info_SetValueForKey(userinfo, "name", "unnamed", MAX_INFO_STRING); Info_SetValueForKey(userinfo, "name", "unnamed", MAX_INFO_STRING);
} }
@ -4940,26 +4930,26 @@ void SV_ExtractFromUserinfo(client_t *cl)
ISteamGameServer_BUpdateUserData(cl->network_userid.m_SteamID, cl->name, 0); ISteamGameServer_BUpdateUserData(cl->network_userid.m_SteamID, cl->name, 0);
val = Info_ValueForKey(userinfo, "rate"); val = Info_ValueForKey(userinfo, "rate");
if (val[0] != 0) if (val[0] != '\0')
{ {
i = Q_atoi(val); i = Q_atoi(val);
cl->netchan.rate = Q_clamp(float(i), MIN_RATE, MAX_RATE); cl->netchan.rate = Q_clamp(float(i), MIN_RATE, MAX_RATE);
} }
val = Info_ValueForKey(userinfo, "topcolor"); val = Info_ValueForKey(userinfo, "topcolor");
if (val[0] != 0) if (val[0] != '\0')
cl->topcolor = Q_atoi(val); cl->topcolor = Q_atoi(val);
else else
Con_DPrintf("topcolor unchanged for %s\n", cl->name); Con_DPrintf("topcolor unchanged for %s\n", cl->name);
val = Info_ValueForKey(userinfo, "bottomcolor"); val = Info_ValueForKey(userinfo, "bottomcolor");
if (val[0] != 0) if (val[0] != '\0')
cl->bottomcolor = Q_atoi(val); cl->bottomcolor = Q_atoi(val);
else else
Con_DPrintf("bottomcolor unchanged for %s\n", cl->name); Con_DPrintf("bottomcolor unchanged for %s\n", cl->name);
val = Info_ValueForKey(userinfo, "cl_updaterate"); val = Info_ValueForKey(userinfo, "cl_updaterate");
if (val[0] != 0) if (val[0] != '\0')
{ {
i = Q_atoi(val); i = Q_atoi(val);
if (i >= 10) if (i >= 10)
@ -4969,13 +4959,13 @@ void SV_ExtractFromUserinfo(client_t *cl)
} }
val = Info_ValueForKey(userinfo, "cl_lw"); val = Info_ValueForKey(userinfo, "cl_lw");
cl->lw = val[0] != 0 ? Q_atoi(val) != 0 : 0; cl->lw = val[0] != '\0' ? Q_atoi(val) != 0 : 0;
val = Info_ValueForKey(userinfo, "cl_lc"); val = Info_ValueForKey(userinfo, "cl_lc");
cl->lc = val[0] != 0 ? Q_atoi(val) != 0 : 0; cl->lc = val[0] != '\0' ? Q_atoi(val) != 0 : 0;
val = Info_ValueForKey(userinfo, "*hltv"); val = Info_ValueForKey(userinfo, "*hltv");
cl->proxy = val[0] != 0 ? Q_atoi(val) == TYPE_PROXY : 0; cl->proxy = val[0] != '\0' ? Q_atoi(val) == TYPE_PROXY : 0;
SV_CheckUpdateRate(&cl->next_messageinterval); SV_CheckUpdateRate(&cl->next_messageinterval);
SV_CheckRate(cl); SV_CheckRate(cl);
@ -5813,6 +5803,10 @@ void EXT_FUNC SV_ActivateServer_internal(int runPhysics)
Q_sprintf(szCommand, "exec %s\n", mapchangecfgfile.string); Q_sprintf(szCommand, "exec %s\n", mapchangecfgfile.string);
Cbuf_AddText(szCommand); Cbuf_AddText(szCommand);
} }
#ifdef REHLDS_FIXES
Info_SetFieldsToTransmit();
#endif
} }
void SV_ServerShutdown(void) void SV_ServerShutdown(void)
@ -6059,7 +6053,6 @@ int SV_SpawnServer(qboolean bIsDemo, char *server, char *startspot)
g_psv.model_precache[0] = pr_strings; g_psv.model_precache[0] = pr_strings;
#ifndef REHLDS_FIXES #ifndef REHLDS_FIXES
g_psv.generic_precache[0] = pr_strings; g_psv.generic_precache[0] = pr_strings;
#endif // REHLDS_FIXES
for (i = 1; i < g_psv.worldmodel->numsubmodels; i++) for (i = 1; i < g_psv.worldmodel->numsubmodels; i++)
{ {
@ -6074,6 +6067,7 @@ int SV_SpawnServer(qboolean bIsDemo, char *server, char *startspot)
} }
#endif #endif
} }
#endif // REHLDS_FIXES
Q_memset(&g_psv.edicts->v, 0, sizeof(entvars_t)); Q_memset(&g_psv.edicts->v, 0, sizeof(entvars_t));

View File

@ -80,6 +80,7 @@ inline char *_strlwr(char *start)
#define Q_strstr A_strstr #define Q_strstr A_strstr
#define Q_strchr strchr #define Q_strchr strchr
#define Q_strrchr strrchr #define Q_strrchr strrchr
#define Q_strtok strtok
#define Q_strlwr A_strtolower #define Q_strlwr A_strtolower
#define Q_strupr A_strtoupper #define Q_strupr A_strtoupper
#define Q_sprintf sprintf #define Q_sprintf sprintf
@ -120,6 +121,7 @@ inline char *_strlwr(char *start)
#define Q_strstr strstr #define Q_strstr strstr
#define Q_strchr strchr #define Q_strchr strchr
#define Q_strrchr strrchr #define Q_strrchr strrchr
#define Q_strtok strtok
#define Q_strlwr _strlwr #define Q_strlwr _strlwr
#define Q_strupr _strupr #define Q_strupr _strupr
#define Q_sprintf sprintf #define Q_sprintf sprintf
@ -144,18 +146,17 @@ inline char *_strlwr(char *start)
#define Q_fmod fmod #define Q_fmod fmod
#endif // #if defined(ASMLIB_H) && defined(HAVE_OPT_STRTOOLS) #endif // #if defined(ASMLIB_H) && defined(HAVE_OPT_STRTOOLS)
// a safe variant of strcpy that truncates the result to fit in the destination buffer // size - sizeof(buffer)
template <size_t size> inline char *Q_strlcpy(char *dest, const char *src, size_t size) {
char *Q_strlcpy(char (&dest)[size], const char *src) {
Q_strncpy(dest, src, size - 1); Q_strncpy(dest, src, size - 1);
dest[size - 1] = '\0'; dest[size - 1] = '\0';
return dest; return dest;
} }
inline char *Q_strnlcpy(char *dest, const char *src, size_t n) { // a safe variant of strcpy that truncates the result to fit in the destination buffer
Q_strncpy(dest, src, n - 1); template <size_t size>
dest[n - 1] = '\0'; char *Q_strlcpy(char (&dest)[size], const char *src) {
return dest; return Q_strlcpy(dest, src, size);
} }
// safely concatenate two strings. // safely concatenate two strings.

View File

@ -3,8 +3,6 @@
#include "cppunitlite/TestHarness.h" #include "cppunitlite/TestHarness.h"
TEST(PrefixedKeysRemove, Info, 1000) { TEST(PrefixedKeysRemove, Info, 1000) {
EngineInitializer engInitGuard;
struct testdata_t { struct testdata_t {
const char* inData; const char* inData;
const char* outData; const char* outData;
@ -12,9 +10,7 @@ TEST(PrefixedKeysRemove, Info, 1000) {
testdata_t testdata[] = { testdata_t testdata[] = {
{ "", "" }, { "", "" },
{ "key\\value", "key\\value" },
{ "\\key\\value", "\\key\\value" }, { "\\key\\value", "\\key\\value" },
{ "_key\\value", "" },
{ "\\_key\\value", "" }, { "\\_key\\value", "" },
{ "\\k\\v\\_u\\t\\y\\z", "\\k\\v\\y\\z" }, { "\\k\\v\\_u\\t\\y\\z", "\\k\\v\\y\\z" },
{ "\\_k\\v\\u\\t\\y\\z", "\\u\\t\\y\\z" }, { "\\_k\\v\\u\\t\\y\\z", "\\u\\t\\y\\z" },
@ -34,8 +30,6 @@ TEST(PrefixedKeysRemove, Info, 1000) {
} }
TEST(SetValueForStarKey, Info, 1000) { TEST(SetValueForStarKey, Info, 1000) {
EngineInitializer engInitGuard;
struct testdata_t { struct testdata_t {
const char* initialInfo; const char* initialInfo;
const char* key; const char* key;
@ -45,19 +39,25 @@ TEST(SetValueForStarKey, Info, 1000) {
testdata_t testdata[] = { testdata_t testdata[] = {
// Behavior // Behavior
{ "", "a", "b", "\\a\\b" }, { "\\a\\b", "c", "d", "\\a\\b\\c\\d" },
{ "\\a\\b\\c\\d", "a", "b", "\\c\\d\\a\\b" }, { "\\a\\b\\c\\d", "b", "f", "\\a\\b\\c\\d\\b\\f" },
{ "a\\b\\c\\d", "a", "b", "\\c\\d\\a\\b" }, { "\\a\\b\\c\\d", "c", "", "\\a\\b" },
{ "\\a\\b\\c\\d\\e\\f", "c", "", "\\a\\b\\e\\f" },
{ "\\a\\b\\c\\d\\e\\f", "z", "", "\\a\\b\\c\\d\\e\\f" },
{ "\\a\\b\\c\\d", "a", "e", "\\c\\d\\a\\e" },
{ "\\a\\b\\c\\d", "e", "f", "\\a\\b\\c\\d\\e\\f" }, { "\\a\\b\\c\\d", "e", "f", "\\a\\b\\c\\d\\e\\f" },
{ "a\\b\\c\\d", "e", "f", "a\\b\\c\\d\\e\\f" },
{ "\\a\\b\\c\\d", "b", "c", "\\a\\b\\c\\d\\b\\c" }, { "\\a\\b\\c\\d", "b", "c", "\\a\\b\\c\\d\\b\\c" },
{ "\\a\\b\\c\\d\\e\\f", "c", "q", "\\a\\b\\e\\f\\c\\q" }, { "\\a\\b\\c\\d", "team", "aBcD", "\\a\\b\\c\\d\\team\\abcd" },
{ "\\a\\b\\c", "e", "f", "\\a\\b\\c\\e\\f" },
{ "\\a\\b\\c\\", "e", "f", "\\a\\b\\c\\\\e\\f" },
{ "\\a\\b\\\\c", "e", "f", "\\a\\b\\\\c\\e\\f" },
// Invalid keys/values
{ "\\a\\b", "c", nullptr, "\\a\\b" },
{ "\\a\\b", nullptr, "c", "\\a\\b" },
{ "\\a\\b", "c", "d..", "\\a\\b" },
{ "\\a\\b", "..c", "d", "\\a\\b" },
{ "\\a\\b", "c\"", "d", "\\a\\b" },
{ "\\a\\b", "c", "d\"", "\\a\\b" },
{ "\\a\\b", "c\\", "d", "\\a\\b" },
{ "\\a\\b", "c", "d\\", "\\a\\b" },
//limits //limits
{ //do nothing since 'team' is not important key { //do nothing since 'team' is not important key
@ -95,9 +95,44 @@ TEST(SetValueForStarKey, Info, 1000) {
} }
} }
TEST(RemoveKeyValue, Info, 1000) { #ifdef REHLDS_FIXES
EngineInitializer engInitGuard; TEST(SetValueForStarKeyResult, Info, 1000) {
struct testdata_t {
const char* initialInfo;
const char* key;
const char* value;
bool success;
};
testdata_t testdata[] = {
// Behavior
{ "\\a\\b", "c", "d", true },
{ "\\a\\b\\c\\d", "b", "f", true },
{ "\\a\\b\\c\\d", "b", "c", true },
// Invalid keys/values
{ "\\a\\b", "c", nullptr, false },
{ "\\a\\b", nullptr, "c", false },
{ "\\a\\b", "c", "d..", false },
{ "\\a\\b", "..c", "d", false },
{ "\\a\\b", "c\"", "d", false },
{ "\\a\\b", "c", "d\"", false },
{ "\\a\\b", "c\\", "d", false },
{ "\\a\\b", "c", "d\\", false },
};
for (int i = 0; i < ARRAYSIZE(testdata); i++) {
testdata_t* d = &testdata[i];
char localInfo[256];
strcpy(localInfo, d->initialInfo);
localInfo[255] = 0;
bool result = Info_SetValueForStarKey(localInfo, d->key, d->value, 256);
CHECK("Invalid info string", d->success == result);
}
}
#endif
TEST(RemoveKeyValue, Info, 1000) {
struct testdata_t { struct testdata_t {
const char* initialInfo; const char* initialInfo;
const char* key; const char* key;
@ -107,18 +142,14 @@ TEST(RemoveKeyValue, Info, 1000) {
testdata_t testdata[] = { testdata_t testdata[] = {
{ "", "a", "" }, { "", "a", "" },
{ "\\a\\b", "a", "" }, { "\\a\\b", "a", "" },
{ "\\a\\", "a", "" },
{ "\\a\\\\", "a", "\\" },
{ "\\a", "a", "" },
{ "a", "a", "" },
{ "a\\", "a", "" },
{ "a\\b", "a", "" },
{ "a\\b\\", "a", "\\" },
{ "\\a\\b\\c\\d\\e\\f", "d", "\\a\\b\\c\\d\\e\\f" }, { "\\a\\b\\c\\d\\e\\f", "d", "\\a\\b\\c\\d\\e\\f" },
{ "\\a\\b\\c\\d\\e\\f", "c", "\\a\\b\\e\\f" }, { "\\a\\b\\c\\d\\e\\f", "c", "\\a\\b\\e\\f" },
{ "a\\b\\c\\d\\e\\f", "d", "a\\b\\c\\d\\e\\f" }, #ifdef REHLDS_FIXES
{ "a\\b\\c\\d\\e\\f", "c", "a\\b\\e\\f" }, { "\\abc\\def\\x\\y\\ab\\cd", "ab", "\\abc\\def\\x\\y" },
{ "a\\b\\c\\d\\e\\f", "a", "\\c\\d\\e\\f" }, #else
{ "\\abc\\def\\x\\y\\ab\\cd", "ab", "\\x\\y" },
#endif
{ "\\ab\\cd", "abc", "\\ab\\cd" }
}; };
for (int i = 0; i < ARRAYSIZE(testdata); i++) { for (int i = 0; i < ARRAYSIZE(testdata); i++) {
@ -131,8 +162,6 @@ TEST(RemoveKeyValue, Info, 1000) {
} }
TEST(GetKeyValue, Info, 1000) { TEST(GetKeyValue, Info, 1000) {
EngineInitializer engInitGuard;
struct testdata_t { struct testdata_t {
const char* info; const char* info;
const char* key; const char* key;
@ -145,16 +174,8 @@ TEST(GetKeyValue, Info, 1000) {
{ "\\a\\", "a", "" }, { "\\a\\", "a", "" },
{ "\\a\\\\", "a", "" }, { "\\a\\\\", "a", "" },
{ "\\a", "a", "" }, { "\\a", "a", "" },
{ "a", "a", "" },
{ "a\\", "a", "" },
{ "a\\b", "a", "b" },
{ "a\\b\\", "a", "b" },
{ "\\a\\b\\c\\d\\e\\f", "d", "" }, { "\\a\\b\\c\\d\\e\\f", "d", "" },
{ "\\a\\b\\c\\d\\e\\f", "c", "d" }, { "\\a\\b\\c\\d\\e\\f", "c", "d" },
{ "a\\b\\c\\d\\e\\f", "d", "" },
{ "a\\b\\c\\d\\e\\f", "c", "d" },
{ "a\\b\\c\\d\\e\\f", "e", "f" },
{ "\\a\\b\\c\\d\\e\\f", "e", "f" }, { "\\a\\b\\c\\d\\e\\f", "e", "f" },
}; };
@ -166,3 +187,122 @@ TEST(GetKeyValue, Info, 1000) {
ZSTR_EQUAL("Invalid info value", d->result, res); ZSTR_EQUAL("Invalid info value", d->result, res);
} }
} }
TEST(FindLargestKey, Info, 1000) {
struct testdata_t {
const char* info;
const char* result;
};
testdata_t testdata[] = {
{ "", "" },
{ "\\name\\a\\model\\b", "" },
{ "\\name\\a\\model\\b\\c\\d", "c" },
{ "\\name\\a\\1234567890abcdef\\b\\model\\c\\1234567890abcdefghi\\d\\rate\\1000", "1234567890abcdefghi" },
{ "\\name\\a\\1234567890abcdefghi\\b\\model\\c\\1234567890abcdef\\d\\rate\\1000", "1234567890abcdefghi" }
};
for (int i = 0; i < ARRAYSIZE(testdata); i++) {
testdata_t* d = &testdata[i];
const char* res = Info_FindLargestKey(d->info, MAX_INFO_STRING);
ZSTR_EQUAL("Invalid info value", d->result, res);
}
}
TEST(InfoIsValid, Info, 1000) {
struct testdata_t {
const char* info;
qboolean result;
};
testdata_t testdata[] = {
{ "", false }, // by original design
#ifdef REHLDS_FIXES
{ "a\\b", false },
#endif
{ "\\a\\b", true },
{ "\\a\\b\\c", false },
{ "\\a\\b\\c\\", false },
{ "\\a\\b\\c\\\\", false },
#ifdef REHLDS_FIXES
{ "\\a\\b\\\\d", false },
{ "\\a\\b\\\\d\\", false },
{ "\\a\\b..c", false },
{ "\\a\\b\"c", false },
{ "\\a..b\\c", false },
{ "\\a\"b\\c", false },
#endif
{ "\\a\\bbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbb", false },
{ "\\bbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbb\\c", false },
#ifdef REHLDS_FIXES
{ "\\a\\b\\c\\d\\a\\e", false },
#endif
{ "\\aaab\\b\\c\\d\\aaa\\e", true },
{ "\\aaa\\b\\c\\d\\aaab\\e", true }
};
for (int i = 0; i < ARRAYSIZE(testdata); i++) {
testdata_t* d = &testdata[i];
char error[256];
snprintf(error, sizeof error, "Invalid result for string '%s'", d->info);
qboolean res = Info_IsValid(d->info);
CHECK(error, d->result == res);
}
}
#ifdef REHLDS_FIXES
TEST(InfoCollectFields, Info, 1000)
{
struct testdata_t {
const char* info;
const char* cvar;
const char* result;
};
testdata_t testdata[] = {
{ "\\cl_updaterate\\100\\topcolor\\60\\name\\abcdefghijklmnop\\*sid\\12332432525345\\_vgui_menus\\1\\model\\urban\\anykey\\1", "", "\\cl_updaterate\\100\\topcolor\\60\\name\\abcdefghijklmnop\\*sid\\12332432525345\\model\\urban\\anykey\\1" },
{ "\\cl_updaterate\\100\\topcolor\\60\\name\\abcdefghijklmnop\\*sid\\12332432525345\\_vgui_menus\\1\\model\\urban", "rate", "" },
{ "\\cl_updaterate\\100\\topcolor\\60\\name\\abcdefghijklmnop\\*sid\\12332432525345\\_vgui_menus\\1\\model\\urban", "topcolor\\*sid\\_vgui_menus", "\\topcolor\\60\\*sid\\12332432525345" },
{ "\\*hltv\\1dsgs", "*hltv", "\\*hltv\\1" },
{ "\\name\\player", "bottomcolor\\name", "\\name\\player" },
};
for (int i = 0; i < ARRAYSIZE(testdata); i++) {
testdata_t* d = &testdata[i];
char destinfo[MAX_INFO_STRING];
sv_rehlds_userinfo_transmitted_fields.string = (char *)d->cvar;
Info_SetFieldsToTransmit();
Info_CollectFields(destinfo, d->info, MAX_INFO_STRING);
ZSTR_EQUAL("Invalid info string", d->result, destinfo);
}
}
#endif
TEST(Info_IsKeyImportant, Info, 1000)
{
struct testdata_t {
const char* key;
bool result;
};
testdata_t testdata[] = {
{ "bottomcolor", true },
{ "bot", false },
{ "*any", true },
{ "_any", false },
{ "model2", false }
};
for (int i = 0; i < ARRAYSIZE(testdata); i++) {
testdata_t* d = &testdata[i];
bool result = Info_IsKeyImportant(d->key);
CHECK("wrong result", d->result == result);
}
}