Skip to content

Commit e794709

Browse files
authored
Merge pull request #2 from spumer/feature/2200-last-stand
feat(*): The Last Stand-update support
2 parents 5b9dd87 + 8002ddb commit e794709

6 files changed

+84
-101
lines changed

gamedata/left4fix.sig.txt

+8-8
Original file line numberDiff line numberDiff line change
@@ -6,8 +6,8 @@
66
{
77
"VersusCompletionScore"
88
{
9-
"linux" "976"
10-
"windows" "984"
9+
"linux" "996"
10+
"windows" "1000"
1111
}
1212

1313
"MarkerDivisor"
@@ -190,22 +190,22 @@
190190
"DIV_CODE_UpdateMarkersReached"
191191
{
192192
"library" "server"
193-
"linux" "\xE8\x2A\x2A\x2A\x2A\xF3\x0F\x2A\x2A\x2A\xC1\xF8\x02"
194-
"windows" "\xF3\x0F\x10\x45\xFC\xC1\xF8\x02"
193+
"linux" "\x89\x04\x24\xE8\x2A\x2A\x2A\x2A\xF3\x0F\x2A\x45\xE4\xC1\xF8\x02"
194+
//"windows" "\xF3\x0F\x10\x45\xFC\xC1\xF8\x02"
195195
}
196196

197197
"DIV_CODE_AddSurvivorStats"
198198
{
199199
"library" "server"
200-
"linux" "\xE8\x2A\x2A\x2A\x2A\xC1\xF8\x02"
201-
"windows" "\x0F\x57\xC0\xC1\xF8\x02"
200+
"linux" "\x89\x04\x24\xE8\x2A\x2A\x2A\x2A\xC1\xF8\x02"
201+
//"windows" "\x0F\x57\xC0\xC1\xF8\x02"
202202
}
203203

204204
"DIV_CODE_GetVersusCompletion"
205205
{
206206
"library" "server"
207-
"linux" "\x8B\x55\x2A\xA1\x2A\x2A\x2A\x2A\x8B\xBA\xE8\x0D\x00\x00\x89\x2A\x2A\xC1\xFF\x02"
208-
"windows" "\x33\xC9\xC1\xF8\x02"
207+
"linux" "\xB8\x00\x00\x00\x00\xC1\xFB\x02\x85\xDB\x0F\x48\xD8"
208+
//"windows" "\x33\xC9\xC1\xF8\x02"
209209
}
210210
}
211211
}

src/Makefile

+10-6
Original file line numberDiff line numberDiff line change
@@ -5,10 +5,12 @@
55
### EDIT THESE PATHS FOR YOUR OWN SETUP ###
66
###########################################
77

8-
SMSDK ?= ../../_SDK_PLATFORM/sourcemod
9-
SRCDS_BASE ?= ../../_SDK_PLATFORM/srcds
10-
HL2SDK_L4D2 ?= ../../_SDK_PLATFORM/hl2sdk-l4d2
11-
MMSOURCE ?= ../../_SDK_PLATFORM/mmsource
8+
SDK_PLATFORM ?= ../../../_SDK_PLATFORM
9+
10+
SMSDK = $(SDK_PLATFORM)/sourcemod
11+
SRCDS_BASE ?= $(SDK_PLATFORM)/srcds
12+
HL2SDK_L4D2 ?= $(SDK_PLATFORM)/hl2sdk-l4d2
13+
MMSOURCE ?= $(SDK_PLATFORM)/mmsource
1214

1315
#####################################
1416
### EDIT BELOW FOR OTHER PROJECTS ###
@@ -40,7 +42,7 @@ C_OPT_FLAGS = -DNDEBUG -Ofast -flto -funroll-loops -pipe -fno-strict-aliasing -f
4042
C_DEBUG_FLAGS = -D_DEBUG -DDEBUG -g -ggdb3
4143
C_GCC4_FLAGS = -fvisibility=hidden
4244
CPP_GCC4_FLAGS = -fvisibility-inlines-hidden -std=c++11
43-
CPP = gcc
45+
CPP = gcc-4.8
4446

4547
##########################
4648
### SDK CONFIGURATIONS ###
@@ -96,7 +98,9 @@ endif
9698
BINARY = $(PROJECT).ext.$(LIB_EXT)
9799

98100
GCC_VERSION := $(shell $(CPP) -dumpversion >&1 | cut -b1)
99-
ifeq "$(GCC_VERSION)" "4"
101+
102+
# apply flags for GCC 4, 5 and 6 versions
103+
ifneq (,$(filter $(GCC_VERSION),4 5 6))
100104
CFLAGS += $(C_GCC4_FLAGS)
101105
CPPFLAGS += $(CPP_GCC4_FLAGS)
102106
endif

src/codepatch/patch_description.txt

+23-23
Original file line numberDiff line numberDiff line change
@@ -1,44 +1,44 @@
11

2-
UpdateMarkersReached: E8 ? ? ? ? F3 0F 2A ? ? C1 F8 02 (EAX, EBX, EDX)
3-
AddSurvivorStats: E8 ? ? ? ? C1 F8 02 (EAX, ECX, EDX)
4-
GetVersusCompletion: 8B 55 ? A1 ? ? ? ? 8B BA E8 0D 00 00 89 ? ? C1 FF 02 (EAX, EDI, EDX)
2+
UpdateMarkersReached: 89 04 24 E8 ? ? ? ? F3 0F 2A 45 E4 C1 F8 02 (EAX, EDX)
3+
AddSurvivorStats: 89 04 24 E8 ? ? ? ? C1 F8 02(EAX, ECX, EDX)
4+
GetVersusCompletion: B8 00 00 00 00 C1 FB 02 85 DB 0F 48 D8 (EAX, EDX, EBX)
55

66

77
UpdateMarkersReached_patch:
88

99
E9 ? ? ? ? /* JMP [address] */
10-
8B 80 E8 0D 00 00 /* mov eax, [eax+3560] */
10+
8B 80 FC 0D 00 00 /* mov eax, [eax+3580] */
1111
31 D2 /* xor edx, edx */
1212
BB 0A 00 00 00 /* mov ebx, 10 */
1313
F7 F3 /* div ebx */
14+
F3 0F 2A 45 E4 /* Preserve xmm0 value for original funcion: cvtsi2ss xmm0, [ebp-0x1c] */
1415
E9 ? ? ? ? /* JMP [address] ;jump to next instruction */
1516
/* NOP 3 bytes (sar eax, 2) */
1617

1718
Result:
18-
{ 0x8B, 0x80, 0xE8, 0x0D, 0x00, 0x00, 0x31, 0xD2, 0xBB, 0x0A, 0x00, 0x00, 0x00, 0xF7, 0xF3 }
19+
{ 0x8B, 0x80, 0xFC, 0x0D, 0x00, 0x00, 0xBA, 0x0A, 0x00, 0x00, 0x00, 0xF7, 0xF2, 0xF3, 0x0F, 0x2A, 0x45, 0xE4 }
1920

2021
AddSurvivorStats_patch:
21-
E9 ? ? ? ? /* JMP [address] */
22-
8B 80 E8 0D 00 00 /* mov eax, [eax+3560] */
23-
31 D2 /* xor edx, edx */
24-
B9 0A 00 00 00 /* mov ecx, 10 */
25-
F7 F1 /* div ecx */
26-
E9 ? ? ? ? /* JMP [address] ;jump to next instruction */
22+
8b 80 fc 0d 00 00 /* mov eax, [eax+3580] */
23+
31 D2 /* xor edx, edx */
24+
b9 0a 00 00 00 /* mov ecx, 10 */
25+
F7 F1 /* div ecx */
26+
85 C0 /* test eax, eax */
27+
89 45 AC /* mov [ebp+var_54], eax
28+
/* our patch smaller than original, nop other */
29+
90 90 90 90 90 90 90 /* nop(s) */
30+
2731
Result:
28-
{ 0x8B, 0x80, 0xE8, 0x0D, 0x00, 0x00, 0x31, 0xD2, 0xB9, 0x0A, 0x00, 0x00, 0x00, 0xF7, 0xF1 }
32+
{ 0x8B, 0x80, 0xFC, 0x0D, 0x00, 0x00, 0xB9, 0x0A, 0x00, 0x00, 0x00, 0xF7, 0xF1, 0x85, 0xC0, 0x89, 0x45, 0xAC, 0x90, 0x90, 0x90, 0x90, 0x90, 0x90, 0x90, 0x90, 0x90 }
2933

3034
GetVersusCompletion_patch:
3135

32-
E9 ? ? ? ? /* JMP [address] */
33-
8B 45 08 /* mov eax, [ebp+arg_0] */
34-
/* copy A1 ? ? ? ? to trampoline end */
35-
8B 80 E8 0D 00 00 /* mov eax, [eax+3560] */
36-
/* copy 89 ? ? to trampoline end */
37-
31 D2 /* xor edx, edx */
38-
BF 0A 00 00 00 /* mov edi, 10 */
39-
F7 F7 /* div edi */
40-
89 C7 /* mov edi, eax */
41-
E9 ? ? ? ? /* JMP [address] ;jump to next instruction */
36+
89 d8 /* mov eax,ebx */
37+
31 D2 /* xor edx, edx */
38+
bb 0a 00 00 00 /* mov ebx, 10 */
39+
f7 f3 /* div ebx */
40+
89 c3 /* mov ebx, eax */
41+
4242
Result:
43-
{ 0x8B, 0x45, 0x08, 0x8B, 0x80, 0xE8, 0x0D, 0x00, 0x00, 0x31, 0xD2, 0xBF, 0x0A, 0x00, 0x00, 0x00, 0xF7, 0xF7, 0x89, 0xC7 }
43+
{ 0x89, 0xD8, 0xBB, 0x0A, 0x00, 0x00, 0x00, 0xF7, 0xF3, 0x89, 0xC3, 0x90, 0x90 }
4444

src/codepatch/score_code_linux.cpp

+16-31
Original file line numberDiff line numberDiff line change
@@ -33,20 +33,21 @@
3333
#include "extension.h"
3434
#include "asm/asm.h"
3535
#include "CDetour/detourhelpers.h"
36+
#include "codepatch/patch_utils.hpp"
3637

3738
#define OP_MOV 0xA1
3839
#define OP_MOV_SIZE 5
3940

4041
// TODO: Create CUT/PASTE masks functions for wrap instructions inside my patch
4142

42-
unsigned char UpdateMarkersReached_orig[] = { 0xE8, 0x2A, 0x2A, 0x2A, 0x2A, 0xF3, 0x0F, 0x2A, 0x2A, 0x2A, 0xC1, 0xF8, 0x02 };
43-
unsigned char UpdateMarkersReached_patch[] = { 0x8B, 0x80, 0xE8, 0x0D, 0x00, 0x00, 0x31, 0xD2, 0xBB, TEAM_SIZE, 0x00, 0x00, 0x00, 0xF7, 0xF3 };
43+
unsigned char UpdateMarkersReached_orig[] = { 0x89, 0x04, 0x24, 0xE8, 0x07, 0x9E, 0xC6, 0xFF, 0xF3, 0x0F, 0x2A, 0x45, 0xE4, 0xC1, 0xF8, 0x02 };
44+
unsigned char UpdateMarkersReached_patch[] = { 0x8B, 0x80, 0xFC, 0x0D, 0x00, 0x00, 0x31, 0xD2, 0xBB, TEAM_SIZE, 0x00, 0x00, 0x00, 0xF7, 0xF3, 0xF3, 0x0F, 0x2A, 0x45, 0xE4 };
4445

45-
unsigned char AddSurvivorStats_orig[] = { 0xE8, 0x2A, 0x2A, 0x2A, 0x2A, 0xC1, 0xF8, 0x02 };
46-
unsigned char AddSurvivorStats_patch[] = { 0x8B, 0x80, 0xE8, 0x0D, 0x00, 0x00, 0x31, 0xD2, 0xB9, TEAM_SIZE, 0x00, 0x00, 0x00, 0xF7, 0xF1 };
46+
unsigned char AddSurvivorStats_orig[] = { 0x89, 0x04, 0x24, 0xE8, 0x89, 0xCB, 0xC4, 0xFF, 0xC1, 0xF8, 0x02, 0xF3, 0x0F, 0x2A, 0xC0, 0x0F, 0x2F, 0x05, 0x90, 0x95, 0xBA, 0x00, 0xF3, 0x0F, 0x11, 0x45, 0xAC };
47+
unsigned char AddSurvivorStats_patch[] = { 0x8B, 0x80, 0xFC, 0x0D, 0x00, 0x00, 0x31, 0xD2, 0xB9, TEAM_SIZE, 0x00, 0x00, 0x00, 0xF7, 0xF1, 0x85, 0xC0, 0x89, 0x45, 0xAC, 0x90, 0x90, 0x90, 0x90, 0x90, 0x90, 0x90};
4748

48-
unsigned char GetVersusCompletion_orig[] = { 0x8B, 0x55, 0x2A, 0xA1, 0x2A, 0x2A, 0x2A, 0x2A, 0x8B, 0xBA, 0xE8, 0x0D, 0x00, 0x00, 0x89, 0x2A, 0x2A, 0xC1, 0xFF, 0x02 };
49-
unsigned char GetVersusCompletion_patch[] = { 0x8B, 0x45, 0x08, 0x8B, 0x80, 0xE8, 0x0D, 0x00, 0x00, 0x31, 0xD2, 0xBF, TEAM_SIZE, 0x00, 0x00, 0x00, 0xF7, 0xF7, 0x89, 0xC7 };
49+
unsigned char GetVersusCompletion_orig[] = { 0xB8, 0x00, 0x00, 0x00, 0x00, 0xC1, 0xFB, 0x02, 0x85, 0xDB, 0x0F, 0x48, 0xD8 };
50+
unsigned char GetVersusCompletion_patch[] = { 0x89, 0xD8, 0x31, 0xD2, 0xBB, TEAM_SIZE, 0x00, 0x00, 0x00, 0xF7, 0xF3, 0x89, 0xC3 };
5051

5152

5253
void ScoreCode::Patch() {
@@ -65,40 +66,24 @@ void ScoreCode::Patch() {
6566
// prepare the trampoline
6667
m_injectMarker = (unsigned char *)sengine->AllocatePageMemory(sizeof(UpdateMarkersReached_patch) + OP_JMP_SIZE);
6768
copy_bytes(UpdateMarkersReached_patch, m_injectMarker, sizeof(UpdateMarkersReached_patch));
68-
inject_jmp(m_injectMarker + sizeof(UpdateMarkersReached_patch), m_pMarkers + OP_JMP_SIZE);
69+
inject_jmp(m_injectMarker + sizeof(UpdateMarkersReached_patch), m_pMarkers + OP_JMP_SIZE); // inject jump to position afterward JMP which brought us here
70+
6971
// copy original code to our buffer
7072
SetMemPatchable(m_pMarkers, sizeof(UpdateMarkersReached_orig));
7173
copy_bytes(m_pMarkers, UpdateMarkersReached_orig, sizeof(UpdateMarkersReached_orig));
7274
// inject jmp to trampoline and nop some bytes after target instruction
7375
inject_jmp(m_pMarkers, m_injectMarker);
74-
fill_nop(m_pMarkers + sizeof(UpdateMarkersReached_orig) - 3, 3);
76+
fill_nop(m_pMarkers + OP_JMP_SIZE, sizeof(UpdateMarkersReached_orig) - OP_JMP_SIZE);
7577

76-
// prepare the trampoline
77-
m_injectStats = (unsigned char *)sengine->AllocatePageMemory(sizeof(AddSurvivorStats_patch) + OP_JMP_SIZE);
78-
copy_bytes(AddSurvivorStats_patch, m_injectStats, sizeof(AddSurvivorStats_patch));
79-
inject_jmp(m_injectStats + sizeof(AddSurvivorStats_patch), m_pL4DStats + sizeof(AddSurvivorStats_orig));
80-
// copy original code to our buffer
78+
// // before patch copy original code to our buffer
8179
SetMemPatchable(m_pL4DStats, sizeof(AddSurvivorStats_orig));
8280
copy_bytes(m_pL4DStats, AddSurvivorStats_orig, sizeof(AddSurvivorStats_orig));
83-
// inject jmp to trampoline
84-
inject_jmp(m_pL4DStats, m_injectStats);
85-
fill_nop(m_pL4DStats + OP_JMP_SIZE, sizeof(AddSurvivorStats_orig) - OP_JMP_SIZE);
81+
copy_bytes(AddSurvivorStats_patch, m_pL4DStats, sizeof(AddSurvivorStats_patch));
8682

87-
// prepare the trampoline
88-
// patch size: (division code size) + (original MOV (A1) to EAX) + (original short MOV (89)) + (JMP back)
89-
size_t compl_patch_size = sizeof(GetVersusCompletion_patch) + OP_MOV_SIZE + 3 + OP_JMP_SIZE;
90-
m_injectCompl = (unsigned char *)sengine->AllocatePageMemory(compl_patch_size);
91-
unsigned char *pInjectEnd = m_injectCompl;
92-
copy_bytes(GetVersusCompletion_patch, m_injectCompl, sizeof(GetVersusCompletion_patch)); pInjectEnd += sizeof(GetVersusCompletion_patch);
93-
copy_bytes(m_pCompletion + 3, pInjectEnd, OP_MOV_SIZE); pInjectEnd += OP_MOV_SIZE;
94-
copy_bytes(m_pCompletion + sizeof(GetVersusCompletion_orig) - 6, pInjectEnd, 3); pInjectEnd += 3;
95-
inject_jmp(pInjectEnd, m_pCompletion + sizeof(GetVersusCompletion_orig));
96-
// copy original code to our buffer
83+
// // before patch copy original code to our buffer
9784
SetMemPatchable(m_pCompletion, sizeof(GetVersusCompletion_orig));
9885
copy_bytes(m_pCompletion, GetVersusCompletion_orig, sizeof(GetVersusCompletion_orig));
99-
// inject jmp to trampoline
100-
inject_jmp(m_pCompletion, m_injectCompl);
101-
fill_nop(m_pCompletion + OP_JMP_SIZE, sizeof(GetVersusCompletion_orig) - OP_JMP_SIZE);
86+
copy_bytes(GetVersusCompletion_patch, m_pCompletion, sizeof(GetVersusCompletion_patch));
10287

10388
m_isPatched = true;
10489
}
@@ -108,8 +93,8 @@ void ScoreCode::Unpatch() {
10893

10994
ISourcePawnEngine *sengine = g_pSM->GetScriptingEngine();
11095
if(m_injectMarker) { copy_bytes(m_pMarkers, UpdateMarkersReached_orig, sizeof(UpdateMarkersReached_orig)); sengine->FreePageMemory(m_injectMarker); }
111-
if(m_injectStats) { copy_bytes(m_pL4DStats, AddSurvivorStats_orig, sizeof(AddSurvivorStats_orig)); sengine->FreePageMemory(m_injectStats); }
112-
if(m_injectCompl) { copy_bytes(m_pCompletion, GetVersusCompletion_orig, sizeof(GetVersusCompletion_orig)); sengine->FreePageMemory(m_injectCompl); }
96+
copy_bytes(m_pL4DStats, AddSurvivorStats_orig, sizeof(AddSurvivorStats_orig));
97+
copy_bytes(m_pCompletion, GetVersusCompletion_orig, sizeof(GetVersusCompletion_orig));
11398

11499
m_isPatched = false;
115100
}

src/detours/on_recompute_versus_completion.cpp

+26-32
Original file line numberDiff line numberDiff line change
@@ -41,46 +41,46 @@
4141

4242
/*
4343
typedef struct {
44-
uint8_t unknownData[888]; // 0 bytes
45-
uint32_t versus_score_bonus[2]; // 888 bytes // temp buffer to store chapter score for second team
46-
uint32_t versus_campaign_score[2]; // 896 bytes
47-
uint32_t chapter_score[2]; // 904 bytes // full filled only after EndVersusRoundEnd
48-
uint8_t unknownData1[72]; // 912 bytes
49-
uint32_t survivors_completion_score[2][4]; // 976 bytes
50-
uint32_t survivors_death_score[2][4]; // 1008 bytes
51-
uint8_t survivors_escaped[4]; // 1040 bytes // in-game buffer
52-
uint32_t versus_completion_score; // 1044 bytes // in-game buffer
53-
uint32_t versus_survival_multiplier[2]; // 1048 bytes // user count who survived
54-
uint8_t unknownData2[92]; // 1056 bytes
55-
uint8_t areTeamsFlipped // 1148 bytes
56-
uint8_t roundFirstHalf // 1149 bytes
57-
uint8_t isTransitionToNextMap; // 1150 bytes
58-
uint8_t unknownData3[5]; // 1151 bytes
59-
uint32_t sacrafice_escaped_mask; // 1156 bytes // e.g. 1000 0110 0000 0001 1000 0110 0000 0001 where bit position is a client index
44+
uint8_t unknownData[908]; // 0 bytes
45+
uint32_t versus_score_bonus[2]; // 908 bytes // temp buffer to store chapter score for second team
46+
uint32_t versus_campaign_score[2]; // 916 bytes
47+
uint32_t chapter_score[2]; // 924 bytes // full filled only after EndVersusRoundEnd
48+
uint8_t unknownData1[72]; // 932 bytes
49+
uint32_t survivors_completion_score[2][4]; // 996 bytes
50+
uint32_t survivors_death_score[2][4]; // 1028 bytes
51+
uint8_t survivors_escaped[4]; // 1060 bytes // in-game buffer
52+
uint32_t versus_completion_score; // 1064 bytes // in-game buffer
53+
uint32_t versus_survival_multiplier[2]; // 1068 bytes // user count who survived
54+
uint8_t unknownData2[92]; // 1076 bytes
55+
uint8_t areTeamsFlipped // 1168 bytes
56+
uint8_t roundFirstHalf // 1169 bytes
57+
uint8_t isTransitionToNextMap; // 1170 bytes
58+
uint8_t unknownData3[5]; // 1171 bytes
59+
uint32_t sacrafice_escaped_mask; // 1176 bytes // e.g. 1000 0110 0000 0001 1000 0110 0000 0001 where bit position is a client index
6060
} CTerrorGameRules_t;
6161
*/
6262

6363
namespace Detours
6464
{
65-
#ifdef WIN32
6665
int cmp_uint_desc(const void* p1, const void* p2) {
6766
return static_cast<int>(*(uint32_t*)p2 - *(uint32_t*)p1);
6867
}
69-
#endif
7068

7169
int __thiscall RecomputeVersusCompletion::OnRecompute(bool arg) {
72-
int i;
73-
bool team = AreTeamsFlipped(*g_pDirector);
70+
int newTotalResult = myRecompute(this);
71+
if(newTotalResult <= g_totalResult) {
72+
return g_totalResult;
73+
}
7474

75-
i = myRecompute(this);
76-
if(i > g_totalResult) g_totalResult = i;
75+
g_totalResult = newTotalResult;
7776

7877
uint32_t (*piVersusSurvivorCompletion)[4] = reinterpret_cast<uint32_t(*)[4]>((unsigned char *)(this) + g_versusSurvivorCompletionOffset);
7978

8079
/**
8180
* Fix score shows by pressing TAB
8281
*/
83-
NotifyNetworkStateChanged();
82+
83+
bool team = AreTeamsFlipped(*g_pDirector);
8484

8585
piVersusSurvivorCompletion[team][0] =
8686
piVersusSurvivorCompletion[team][1] =
@@ -89,6 +89,8 @@ namespace Detours
8989

9090
piVersusSurvivorCompletion[team][0] += g_totalResult % 4;
9191

92+
NotifyNetworkStateChanged();
93+
9294
L4D_DEBUG_LOG("Recompute return: %d.", g_totalResult);
9395

9496
return g_totalResult;
@@ -149,15 +151,7 @@ namespace Detours
149151

150152
result += r_appendScores(pNextScore, TEAM_SIZE - (pNextScore - g_iHighestVersusSurvivorCompletion), g_dead_players, arraysize(g_dead_players));
151153

152-
qsort(g_iHighestVersusSurvivorCompletion, TEAM_SIZE, sizeof(uint32_t),
153-
#ifdef WIN32
154-
cmp_uint_desc
155-
#else
156-
[] (const void* p1, const void* p2) {
157-
return static_cast<int>(*(uint32_t*)p2 - *(uint32_t*)p1);
158-
}
159-
#endif
160-
);
154+
qsort(g_iHighestVersusSurvivorCompletion, TEAM_SIZE, sizeof(uint32_t), cmp_uint_desc);
161155
return result;
162156
}
163157
}

src/sdk/smsdk_config.h

+1-1
Original file line numberDiff line numberDiff line change
@@ -43,7 +43,7 @@
4343
/* Basic information exposed publicly */
4444
#define SMEXT_CONF_NAME "Left4Fix"
4545
#define SMEXT_CONF_DESCRIPTION "Fix score completion on servers have 8+ players"
46-
#define SMEXT_CONF_VERSION QUOTE(2.0.2 (TEAM_SIZE players))
46+
#define SMEXT_CONF_VERSION QUOTE(2.0.3 (TEAM_SIZE players))
4747
#define SMEXT_CONF_AUTHOR "spumer"
4848
#define SMEXT_CONF_URL "https://forums.alliedmods.net/member.php?u=151387"
4949
#define SMEXT_CONF_LOGTAG "L4FIX"

0 commit comments

Comments
 (0)