From e618d2809e086d2df7c41b4c3d11b78512ffe4b7 Mon Sep 17 00:00:00 2001
From: GAMMACASE <31375974+GAMMACASE@users.noreply.github.com>
Date: Sat, 18 Jan 2020 11:29:38 +0300
Subject: [PATCH] Improved performance of the plugin
Added StoreToAddressFast() function, which is significantly faster then native StoreToAddress().
---
.../sourcemod/gamedata/momsurffix2.games.txt | 16 ++
.../scripting/momsurffix/gamemovement.sp | 2 +-
.../scripting/momsurffix/gametrace.sp | 8 +-
.../sourcemod/scripting/momsurffix/utils.sp | 6 +-
addons/sourcemod/scripting/momsurffix2.sp | 181 +++++++++++++-----
5 files changed, 157 insertions(+), 56 deletions(-)
diff --git a/addons/sourcemod/gamedata/momsurffix2.games.txt b/addons/sourcemod/gamedata/momsurffix2.games.txt
index 60d606d..fffdf1c 100644
--- a/addons/sourcemod/gamedata/momsurffix2.games.txt
+++ b/addons/sourcemod/gamedata/momsurffix2.games.txt
@@ -47,6 +47,22 @@
"CTraceFilterSimple::size" "16"
}
+ "Addresses"
+ {
+ "CGameMovement::TryPlayerMove_Start"
+ {
+ "windows"
+ {
+ "signature" "CGameMovement::TryPlayerMove"
+ }
+ "linux"
+ {
+ "signature" "CGameMovement::TryPlayerMove"
+ }
+ "offset" "0"
+ }
+ }
+
"Offsets"
{
"OSType"
diff --git a/addons/sourcemod/scripting/momsurffix/gamemovement.sp b/addons/sourcemod/scripting/momsurffix/gamemovement.sp
index c331b9d..34ac84f 100644
--- a/addons/sourcemod/scripting/momsurffix/gamemovement.sp
+++ b/addons/sourcemod/scripting/momsurffix/gamemovement.sp
@@ -67,7 +67,7 @@ methodmap CGameMovement < AddressBase
property int m_nTraceCount
{
public get() { return LoadFromAddress(this.Address + offsets.cgmoffsets.m_nTraceCount, NumberType_Int32); }
- public set(int _tracecount) { StoreToAddress(this.Address + offsets.cgmoffsets.m_nTraceCount, _tracecount, NumberType_Int32); }
+ public set(int _tracecount) { StoreToAddressCustom(this.Address + offsets.cgmoffsets.m_nTraceCount, _tracecount, NumberType_Int32); }
}
}
diff --git a/addons/sourcemod/scripting/momsurffix/gametrace.sp b/addons/sourcemod/scripting/momsurffix/gametrace.sp
index c839566..6c7a795 100644
--- a/addons/sourcemod/scripting/momsurffix/gametrace.sp
+++ b/addons/sourcemod/scripting/momsurffix/gametrace.sp
@@ -279,25 +279,25 @@ methodmap CTraceFilterSimple < AllocatableBase
property Address vptr
{
public get() { return view_as
(LoadFromAddress(this.Address + offsets.ctfsoffsets.vptr, NumberType_Int32)); }
- public set(Address _vtbladdr) { StoreToAddress(this.Address + offsets.ctfsoffsets.vptr, view_as(_vtbladdr), NumberType_Int32); }
+ public set(Address _vtbladdr) { StoreToAddressCustom(this.Address + offsets.ctfsoffsets.vptr, view_as(_vtbladdr), NumberType_Int32); }
}
property CBaseHandle m_pPassEnt
{
public get() { return view_as(LoadFromAddress(this.Address + offsets.ctfsoffsets.m_pPassEnt, NumberType_Int32)); }
- public set(CBaseHandle _passent) { StoreToAddress(this.Address + offsets.ctfsoffsets.m_pPassEnt, view_as(_passent), NumberType_Int32); }
+ public set(CBaseHandle _passent) { StoreToAddressCustom(this.Address + offsets.ctfsoffsets.m_pPassEnt, view_as(_passent), NumberType_Int32); }
}
property int m_collisionGroup
{
public get() { return LoadFromAddress(this.Address + offsets.ctfsoffsets.m_collisionGroup, NumberType_Int32); }
- public set(int _collisiongroup) { StoreToAddress(this.Address + offsets.ctfsoffsets.m_collisionGroup, _collisiongroup, NumberType_Int32); }
+ public set(int _collisiongroup) { StoreToAddressCustom(this.Address + offsets.ctfsoffsets.m_collisionGroup, _collisiongroup, NumberType_Int32); }
}
property Address m_pExtraShouldHitCheckFunction
{
public get() { return view_as(LoadFromAddress(this.Address + offsets.ctfsoffsets.m_pExtraShouldHitCheckFunction, NumberType_Int32)); }
- public set(Address _checkfnc) { StoreToAddress(this.Address + offsets.ctfsoffsets.m_pExtraShouldHitCheckFunction, view_as(_checkfnc), NumberType_Int32); }
+ public set(Address _checkfnc) { StoreToAddressCustom(this.Address + offsets.ctfsoffsets.m_pExtraShouldHitCheckFunction, view_as(_checkfnc), NumberType_Int32); }
}
public CTraceFilterSimple()
diff --git a/addons/sourcemod/scripting/momsurffix/utils.sp b/addons/sourcemod/scripting/momsurffix/utils.sp
index 7aa081f..6c2e240 100644
--- a/addons/sourcemod/scripting/momsurffix/utils.sp
+++ b/addons/sourcemod/scripting/momsurffix/utils.sp
@@ -71,19 +71,19 @@ methodmap Vector < AllocatableBase
property float x
{
- public set(float _x) { StoreToAddress(this.Address, view_as(_x), NumberType_Int32); }
+ public set(float _x) { StoreToAddressCustom(this.Address, view_as(_x), NumberType_Int32); }
public get() { return view_as(LoadFromAddress(this.Address, NumberType_Int32)); }
}
property float y
{
- public set(float _y) { StoreToAddress(this.Address + 4, view_as(_y), NumberType_Int32); }
+ public set(float _y) { StoreToAddressCustom(this.Address + 4, view_as(_y), NumberType_Int32); }
public get() { return view_as(LoadFromAddress(this.Address + 4, NumberType_Int32)); }
}
property float z
{
- public set(float _z) { StoreToAddress(this.Address + 8, view_as(_z), NumberType_Int32); }
+ public set(float _z) { StoreToAddressCustom(this.Address + 8, view_as(_z), NumberType_Int32); }
public get() { return view_as(LoadFromAddress(this.Address + 8, NumberType_Int32)); }
}
diff --git a/addons/sourcemod/scripting/momsurffix2.sp b/addons/sourcemod/scripting/momsurffix2.sp
index 37805d3..1b884a3 100644
--- a/addons/sourcemod/scripting/momsurffix2.sp
+++ b/addons/sourcemod/scripting/momsurffix2.sp
@@ -11,13 +11,16 @@ public Plugin myinfo = {
name = "Momentum surf fix \'2",
author = "GAMMA CASE",
description = "Ported surf fix from momentum mod.",
- version = "1.0.5",
+ version = "1.0.6",
url = "http://steamcommunity.com/id/_GAMMACASE_/"
};
#define FLT_EPSILON 1.192092896e-07
#define MAX_CLIP_PLANES 5
+#define ASM_PATCH_LEN 24
+#define ASM_START_OFFSET 100
+
enum OSType
{
OSUnknown = -1,
@@ -36,23 +39,31 @@ EngineVersion gEngineVersion;
#include "momsurffix/gametrace.sp"
#include "momsurffix/gamemovement.sp"
-ConVar gRampBumpCount, gBounce, gRampInitialRetraceLength;
+ConVar gRampBumpCount,
+ gBounce,
+ gRampInitialRetraceLength,
+ gASMOptimizations,
+ gNoclipWorkAround;
float vec3_origin[3] = {0.0, 0.0, 0.0};
bool gBasePlayerLoadedTooEarly;
+PatchHandler gASMPatch;
+Handle gStoreToAddressFast;
+Address gTryPlayerMoveStart;
+
#if defined DEBUG_PROFILE
#include "profiler"
-#define PROF_START if(gProf) gProf.Start()
-#define PROF_STOP%1; if(gProf)\
+#define PROF_START() if(gProf) gProf.Start()
+#define PROF_STOP(%1)%2; if(gProf)\
{\
gProf.Stop();\
- Prof_Check();\
+ Prof_Check(%1);\
}
Profiler gProf;
-float gTotal;
-int gTotalCount, gTotalCountNeeded;
+ArrayList gProfData;
+float gProfTime;
#else
#define PROF_START%1;
#define PROF_STOP%1;
@@ -68,8 +79,10 @@ public void OnPluginStart()
gRampBumpCount = CreateConVar("momsuffix_ramp_bumpcount", "8", "Helps with fixing surf/ramp bugs", .hasMin = true, .min = 4.0, .hasMax = true, .max = 16.0);
gRampInitialRetraceLength = CreateConVar("momsuffix_ramp_initial_retrace_length", "0.2", "Amount of units used in offset for retraces", .hasMin = true, .min = 0.2, .hasMax = true, .max = 5.0);
+ gASMOptimizations = CreateConVar("momsuffix_enable_asm_optimizations", "1", "Enables ASM optimizations, that may improve performance of the plugin", .hasMin = true, .min = 0.0, .hasMax = true, .max = 1.0);
+ gNoclipWorkAround = CreateConVar("momsuffix_enable_noclip_workaround", "1", "Enables workaround to prevent issue #1, can actually help if momsuffix_enable_asm_optimizations is 0", .hasMin = true, .min = 0.0, .hasMax = true, .max = 1.0);
gBounce = FindConVar("sv_bounce");
- ASSERT(gBounce);
+ ASSERT_MSG(gBounce, "\"sv_bounce\" convar wasn't found!");
AutoExecConfig();
@@ -84,6 +97,7 @@ public void OnPluginStart()
InitGameMovement(gd);
SetupDhooks(gd);
+ SetupASMOptimizations(gd);
delete gd;
}
@@ -102,6 +116,8 @@ public void OnMapStart()
public void OnPluginEnd()
{
CleanUpUtils();
+ if(gASMPatch.Address != Address_Null)
+ gASMPatch.Restore();
}
public Action SM_Dumpmempool(int client, int args)
@@ -116,39 +132,59 @@ public Action SM_Prof(int client, int args)
{
if(args < 1)
{
- ReplyToCommand(client, SNAME..."Usage: sm_prof ");
+ ReplyToCommand(client, SNAME..."Usage: sm_prof ");
return Plugin_Handled;
}
char buff[32];
GetCmdArg(1, buff, sizeof(buff));
- gTotalCountNeeded = StringToInt(buff);
+ gProfTime = StringToFloat(buff);
- if(gTotalCountNeeded <= 0)
+ if(gProfTime <= 0.1)
{
- ReplyToCommand(client, SNAME..."Only positive values allowed.")
+ ReplyToCommand(client, SNAME..."Time should be higher then 0.1 seconds.")
return Plugin_Handled;
}
+ gProfData = new ArrayList(3);
gProf = new Profiler();
+ CreateTimer(gProfTime, Prof_Check_Timer, client);
- ReplyToCommand(client, SNAME..."Profiler started, awaiting %i iterations.", gTotalCountNeeded);
+ ReplyToCommand(client, SNAME..."Profiler started, awaiting %.2f seconds.", gProfTime);
return Plugin_Handled;
}
-stock void Prof_Check()
+stock void Prof_Check(int idx)
{
- gTotal += gProf.Time;
- gTotalCount++;
-
- if(gTotalCount >= gTotalCountNeeded)
+ int idx2;
+ if(gProfData.Length - 1 < idx)
+ {
+ idx2 = gProfData.Push(gProf.Time);
+ gProfData.Set(idx2, 1, 1);
+ gProfData.Set(idx2, idx, 2);
+ }
+ else
{
- PrintToServer(SNAME..."Prifiling completed, avg time: %f", gTotal / float(gTotalCount));
+ idx2 = gProfData.FindValue(idx, 2);
- delete gProf;
+ gProfData.Set(idx2, view_as(gProfData.Get(idx2)) + gProf.Time);
+ gProfData.Set(idx2, gProfData.Get(idx2, 1) + 1, 1);
}
}
+
+public Action Prof_Check_Timer(Handle timer, int client)
+{
+ ReplyToCommand(client, SNAME..."Profiler finished:");
+ if(gProfData.Length == 0)
+ ReplyToCommand(client, SNAME..."There was no profiling data...");
+
+ for(int i = 0; i < gProfData.Length; i++)
+ ReplyToCommand(client, SNAME..."[%i] Avg time: %f | Calls: %i", i, view_as(gProfData.Get(i)) / float(gProfData.Get(i, 1)), gProfData.Get(i, 1));
+
+ delete gProf;
+ delete gProfData;
+}
#endif
/*CGameMovement dummy;
@@ -158,6 +194,48 @@ public Action SM_TestMem(int client, int args)
return Plugin_Handled;
}*/
+void SetupASMOptimizations(GameData gd)
+{
+ //CGameMovement::TryPlayerMove_Start
+ gTryPlayerMoveStart = gd.GetAddress("CGameMovement::TryPlayerMove_Start");
+ ASSERT_MSG(gTryPlayerMoveStart, "Can't find start of the \"CGameMovement::TryPlayerMove\" function.");
+
+ gASMPatch = PatchHandler(gTryPlayerMoveStart + ASM_START_OFFSET);
+ gASMPatch.Save(ASM_PATCH_LEN);
+
+ Address start = gASMPatch.Address;
+
+ /*StoreToAddressFast asm:
+ * push ebp
+ * mov ebp, esp
+ *
+ * mov eax, [ebp + 12]
+ * mov ecx, [ebp + 8]
+ * mov [ecx], eax
+ *
+ * mov esp, ebp
+ * pop ebp
+ * ret 8
+ */
+
+ StoreToAddress(start, 0x8B_EC_8B_55, NumberType_Int32);
+ StoreToAddress(start + 4, 0x4D_8B_0C_45, NumberType_Int32);
+ StoreToAddress(start + 8, 0x8B_01_89_08, NumberType_Int32);
+ StoreToAddress(start + 12, 0x08_C2_5D_E5, NumberType_Int32);
+ StoreToAddress(start + 16, 0x00, NumberType_Int8);
+
+ //StoreToAddressFast
+ StartPrepSDKCall(SDKCall_Static);
+
+ PrepSDKCall_SetAddress(start);
+ PrepSDKCall_AddParameter(SDKType_PlainOldData, SDKPass_Plain);
+ PrepSDKCall_AddParameter(SDKType_PlainOldData, SDKPass_Plain);
+ PrepSDKCall_SetReturnInfo(SDKType_PlainOldData, SDKPass_Plain);
+
+ gStoreToAddressFast = EndPrepSDKCall();
+ ASSERT(gStoreToAddressFast);
+}
+
void ValidateGameAndOS(GameData gd)
{
gOSType = view_as(gd.GetOffset("OSType"));
@@ -202,7 +280,8 @@ int TryPlayerMove(CGameMovement pThis, Vector pFirstDest, CGameTrace pFirstTrace
Vector vecVelocity = pThis.mv.m_vecVelocity;
vecVelocity.ToArray(original_velocity);
vecVelocity.ToArray(primal_velocity);
- pThis.mv.m_vecAbsOrigin.ToArray(fixed_origin);
+ Vector vecAbsOrigin = pThis.mv.m_vecAbsOrigin;
+ vecAbsOrigin.ToArray(fixed_origin);
Vector plane_normal;
static Vector alloced_vector, alloced_vector2;
@@ -263,7 +342,7 @@ int TryPlayerMove(CGameMovement pThis, Vector pFirstDest, CGameTrace pFirstTrace
alloced_vector.ToArray(valid_plane);
}
//TODO: should be replaced with normal solution!! Currently hack to fix issue #1.
- else if(vecVelocity.z < -6.25 && vecVelocity.z > 0.0)
+ else if(!gNoclipWorkAround.BoolValue || (vecVelocity.z < -6.25 && vecVelocity.z > 0.0))
{
//Quite heavy part of the code, should not be triggered much or else it'll impact performance by a lot!!!
float offsets[3];
@@ -275,13 +354,13 @@ int TryPlayerMove(CGameMovement pThis, Vector pFirstDest, CGameTrace pFirstTrace
float offset[3], offset_mins[3], offset_maxs[3], buff[3];
Ray_t ray = Ray_t();
- PROF_START;
for(i = 0; i < 3; i++)
{
for(j = 0; j < 3; j++)
{
for(h = 0; h < 3; h++)
{
+ PROF_START();
offset[0] = offsets[i];
offset[1] = offsets[j];
offset[2] = offsets[h];
@@ -304,7 +383,9 @@ int TryPlayerMove(CGameMovement pThis, Vector pFirstDest, CGameTrace pFirstTrace
offset_maxs[1] /= 2.0;
if(offset[2] < 0.0)
offset_maxs[2] /= 2.0;
+ PROF_STOP(0);
+ PROF_START();
if(gEngineVersion == Engine_CSGO)
ray.Init((AddVectors(fixed_origin, offset, buff), buff), (SubtractVectors(end, offset, offset), offset),
(SubtractVectors(VectorToArray(GetPlayerMins(pThis)), offset_mins, offset_mins), offset_mins),
@@ -313,11 +394,17 @@ int TryPlayerMove(CGameMovement pThis, Vector pFirstDest, CGameTrace pFirstTrace
ray.Init((AddVectors(fixed_origin, offset, buff), buff), (SubtractVectors(end, offset, offset), offset),
(SubtractVectors(VectorToArray(GetPlayerMinsCSS(pThis, alloced_vector)), offset_mins, offset_mins), offset_mins),
(AddVectors(VectorToArray(GetPlayerMaxsCSS(pThis, alloced_vector2)), offset_maxs, offset_maxs), offset_maxs));
+ PROF_STOP(1);
+ PROF_START();
UTIL_TraceRay(ray, MASK_PLAYERSOLID, pThis, COLLISION_GROUP_PLAYER_MOVEMENT, pm);
+ PROF_STOP(2);
+ PROF_START();
pm.plane.normal.FromArray(vec3_origin);
+ PROF_STOP(3);
+ PROF_START();
plane_normal = pm.plane.normal;
//PrintToServer(SNAME..."DEBUG: TraceFired! plane_normal: [%f | %f | %f]", plane_normal.x, plane_normal.y, plane_normal.z);
@@ -327,10 +414,10 @@ int TryPlayerMove(CGameMovement pThis, Vector pFirstDest, CGameTrace pFirstTrace
valid_planes++;
plane_normal.ToArray(valid_plane);
}
+ PROF_STOP(4);
}
}
}
- PROF_STOP;
ray.Free();
if(valid_planes > 0 && !CloseEnough(valid_plane, view_as({0.0, 0.0, 0.0})))
@@ -366,13 +453,11 @@ int TryPlayerMove(CGameMovement pThis, Vector pFirstDest, CGameTrace pFirstTrace
if(stuck_on_ramp && has_valid_plane)
{
alloced_vector.FromArray(fixed_origin);
- //TracePlayerBBoxCustom(pThis, fixed_origin, end, MASK_PLAYERSOLID, COLLISION_GROUP_PLAYER_MOVEMENT, pm);
TracePlayerBBox(pThis, alloced_vector, alloced_vector2, MASK_PLAYERSOLID, COLLISION_GROUP_PLAYER_MOVEMENT, pm);
pm.plane.normal.FromArray(valid_plane);
}
else
- //TracePlayerBBoxCustom(pThis, VectorToArray(pThis.mv.m_vecAbsOrigin), end, MASK_PLAYERSOLID, COLLISION_GROUP_PLAYER_MOVEMENT, pm);
- TracePlayerBBox(pThis, pThis.mv.m_vecAbsOrigin, alloced_vector2, MASK_PLAYERSOLID, COLLISION_GROUP_PLAYER_MOVEMENT, pm);
+ TracePlayerBBox(pThis, vecAbsOrigin, alloced_vector2, MASK_PLAYERSOLID, COLLISION_GROUP_PLAYER_MOVEMENT, pm);
}
if(bumpcount > 0 && pThis.player.m_hGroundEntity == view_as(-1) && !IsValidMovementTrace(pThis, pm))
@@ -382,20 +467,11 @@ int TryPlayerMove(CGameMovement pThis, Vector pFirstDest, CGameTrace pFirstTrace
continue;
}
- /*if(pm.allsolid)
- {
- vecVelocity.FromArray(vec3_origin);
-
- pm.Free();
- return 4;
- }*/
-
if(pm.fraction > 0.0)
{
if((bumpcount == 0 || pThis.player.m_hGroundEntity != view_as(-1)) && numbumps > 0 && pm.fraction == 1.0)
{
CGameTrace stuck = CGameTrace();
- //TracePlayerBBoxCustom(pThis, VectorToArray(pm.endpos), VectorToArray(pm.endpos), MASK_PLAYERSOLID, COLLISION_GROUP_PLAYER_MOVEMENT, stuck);
TracePlayerBBox(pThis, pm.endpos, pm.endpos, MASK_PLAYERSOLID, COLLISION_GROUP_PLAYER_MOVEMENT, stuck);
if((stuck.startsolid || stuck.fraction != 1.0) && bumpcount == 0)
@@ -421,8 +497,8 @@ int TryPlayerMove(CGameMovement pThis, Vector pFirstDest, CGameTrace pFirstTrace
stuck_on_ramp = false;
vecVelocity.ToArray(original_velocity);
- pThis.mv.m_vecAbsOrigin.FromArray(VectorToArray(pm.endpos));
- pThis.mv.m_vecAbsOrigin.ToArray(fixed_origin);
+ vecAbsOrigin.FromArray(VectorToArray(pm.endpos));
+ vecAbsOrigin.ToArray(fixed_origin);
allFraction += pm.fraction;
numplanes = 0;
}
@@ -452,6 +528,7 @@ int TryPlayerMove(CGameMovement pThis, Vector pFirstDest, CGameTrace pFirstTrace
if(numplanes == 1 && pThis.player.m_MoveType == MOVETYPE_WALK && pThis.player.m_hGroundEntity != view_as(-1))
{
Vector vec1 = Vector();
+ PROF_START();
if(planes[0][2] >= 0.7)
{
vec1.FromArray(original_velocity);
@@ -469,6 +546,7 @@ int TryPlayerMove(CGameMovement pThis, Vector pFirstDest, CGameTrace pFirstTrace
ClipVelocity(pThis, vec1, alloced_vector2, alloced_vector, 1.0 + gBounce.FloatValue * (1.0 - pThis.player.m_surfaceFriction));
alloced_vector.ToArray(new_velocity);
}
+ PROF_STOP(5);
vecVelocity.FromArray(new_velocity);
VectorCopy(new_velocity, original_velocity);
@@ -623,16 +701,6 @@ stock bool IsValidMovementTrace(CGameMovement pThis, CGameTrace tr)
return true;
}
-/*stock void TracePlayerBBoxCustom(CGameMovement pThis, float start[3], float end[3], int mask, int collisionGroup, CGameTrace trace)
-{
- Ray_t ray = Ray_t();
-
- ray.Init(start, end, VectorToArray(GetPlayerMins(pThis)), VectorToArray(GetPlayerMaxs(pThis)));
- UTIL_TraceRay(ray, mask, pThis, collisionGroup, trace);
-
- ray.Free();
-}*/
-
stock void UTIL_TraceRay(Ray_t ray, int mask, CGameMovement gm, int collisionGroup, CGameTrace trace)
{
if(gEngineVersion == Engine_CSGO)
@@ -658,4 +726,21 @@ stock void UTIL_TraceRay(Ray_t ray, int mask, CGameMovement gm, int collisionGro
filter.Free();
}
+}
+
+//Faster then native StoreToAddress by ~45 times.
+stock void StoreToAddressFast(Address addr, any data)
+{
+ ASSERT(gStoreToAddressFast);
+
+ int ret = SDKCall(gStoreToAddressFast, addr, data);
+ ASSERT(ret == data);
+}
+
+stock void StoreToAddressCustom(Address addr, any data, NumberType type)
+{
+ if(gASMOptimizations.BoolValue && gStoreToAddressFast)
+ StoreToAddressFast(addr, data);
+ else
+ StoreToAddress(addr, view_as(data), type);
}
\ No newline at end of file