diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..3e8d0a9 --- /dev/null +++ b/.gitignore @@ -0,0 +1,2 @@ +*.psd +releases/ \ No newline at end of file diff --git a/README.md b/README.md index 8085d2f..b329b70 100644 --- a/README.md +++ b/README.md @@ -1,2 +1,85 @@ -# nojump-boost-fix - Changes StepMove behavior that generates free Z velocity when moving up an incline +# NoJump Boost Fix + +This SourceMod Plugin removes the StepMove effect that generates free Z velocity when moving up an incline. + +- Supports CS:GO and CS:S (the free Z velocity effect does not exist in Team Fortress 2). +- Requires DHooks, which is included with SourceMod 1.11+. If you use an earlier version of SourceMod, you can get DHooks directly [here](https://github.com/peace-maker/DHooks2/releases/latest). Earlier versions of DHooks are probably fine (Detours support is not required). +- Works with other movement fix plugins. + +The plugin's functionality can be toggled with the ConVar `nojump_boost_fix`. + +--- + +## "No-Jump" Slope Boosts + +There's a weird effect in some Source games where it can be better to not hold the jump key to autohop when landing on an uphill incline. Not only can this give you more speed up the incline than if you had jumped, but you actually end up with *more speed than should be physically possible*. + +This is all due to the StepMove function -- the game code that handles walking up slopes and stairs. This is just part of the game, and has actually been around since Quake 1 which Source is ultimately derived from. I'm not really sure why this effect was implemented in either case. + +## How StepMove Works + +Each tick, the game performs a sequence of steps to move the player, updating their position, velocity, and other factors. If the player starts the tick on the ground, the game does "ground movement" for the whole tick (other types of movement include air movement and water movement). + +When doing ground movement, if the game detects something in front of the player that would keep them from moving in a straight line such as a slope or staircase, then the StepMove function is called to navigate it. + + StepMove has this basic process: + +>1. From the starting point, try moving the player in a straight line, deflecting off of any objects in the way. +>2. Return to the starting point and try "stepping" up to 18 units directly upward, and *then* try moving forward in a straight line, deflecting off of any objects in the way. Finally, "step" up to 18 units directly back down. +>3. Compare the results of Step 1 and Step 2, and use the version that went farther horizontally. If Step 2 did not end up on shallow ground after stepping down, the results of Step 1 are used. +>4. If the results of Step 2 are chosen, use the X and Y velocities from that step, but use the Z velocity from Step 1. **(!!!)** + + +![Steps 1 and 2 Diagram](steps.gif) + +In almost all cases, the up-over-down movement from Step 2 goes farthest because it deflects off of the slope later than in Step 1 (if at all). + +The problematic part is Step 4. This uses the player's X-Y velocity from Step 2 -- which is usually their entire original velocity because no deflection occurred -- and combines it with the Z velocity from Step 1 -- which is whatever value resulted from deflecting off the slope in that step. + +The Z velocity gained in Step 1 came at the cost of X-Y velocity decreasing, which you could describe as the velocity being rotated to point in a different direction (and in fact, the resulting speed is actually lower). Step 4 breaks this continuity by using the player's original X-Y velocity, giving the player some new Z velocity "for free". + +--- + +#### StepMove example on a 30° incline with 1000 u/s velocity + +| | Initial | Step 1 | Step 2 | Final | +| -----------------: | ------: | -----: | -----: | -----------: | +| **X-Y Speed** | 1000 | 750 | 1000 | 1000 | +| **Vertical Speed** | 0 | 433 | 0 | 433 | +| **Overall Speed** | 1000 | 866 | 1000 | **1089 (!)** | + +#### The same scenario but where the player autohops when landing + +| | Initial | After Jumping | Final (AirMove) | +| -----------------: | ------: | -------------: | --------------: | +| **X-Y Speed** | 1000 | 1000 | 880 | +| **Vertical Speed** | 0 | 302 | 508 | +| **Overall Speed** | 1000 | 1044 | **1017** | + +The main takeaway from this example is that **the player gains more "free" vertical speed from StepMove (433 u/s) than they would from actually jumping (302 u/s)**. This effect is more extreme the steeper the incline is. StepMove also lets the player move one extra tick before actually colliding with the incline. + +For the curious, the source code for StepMove can be found [here](https://github.com/alliedmodders/hl2sdk/blob/sdk2013/game/shared/gamemovement.cpp#L1517-L1607). It is effectively identical across most Source games. The source code for the same process in Quake 1 -- with the same boost behavior -- can be found [here](https://github.com/id-Software/Quake/blob/master/QW/client/pmove.c#L259-L309). + +## The Fix + +This SourceMod plugin eliminates the boost effect by making StepMove use the results from either Step 1 *or* Step 2 without ever trying to combine them. + +Whichever result goes farther horizontally is used in its entirety, with one caveat: if Step 1 results in a velocity that would cause the player to start sliding (Z speed must be greater than 140 u/s), then that result is always used. This is important because sliding causes the player to move differently, and the original StepMove would have caused the player to start sliding in this case as well. + +This fix is the same used in Momentum Mod, though this plugin uses a different implementation than modifying StepMove directly. + +This fix is not necessary for Team Fortress 2 as that game's version of StepMove never combines the results of Steps 1 and 2. + +## Why Fix It? + +I made this plugin to give the movement community the ability to remove this behavior if they collectively decide to do so. It is definitely a fun effect to take advantage of, but the existence of it might be a little over-the-top to some people, especially since it wasn't *really* a thing before RNGFix. It also doesn't feel right for jumping to ever be *slower* than not jumping, even if it is "natural" for the game to be doing this. + +## How RNGFix Plays Into This + +Some people attribute this boost effect to [RNGFix](https://github.com/jason-e/rngfix), which is understandable even if RNGFix does not actually cause it directly. + +Triggering this effect is really only possible if the player gets "good RNG" when landing on an uphill incline, and the player also needs to be moving fast enough that the free boost is greater than would be gained by jumping. This is entirely possible without any movement plugins, though is fairly unlikely without RNGFix. Players also had a big disadvantage if they avoided autohopping in this case before RNGFix, so this effect was not really known. + +There are actually a small number of maps with boosters on slopes which were unknowningly tuned based on this StepMove boost effect. For example, the booster at the beginning of Stage 10 on surf_lt_omnific (a map much older than RNGFix) works perfectly if you slide down onto it and do not hold jump. If you hold jump, you end up with less speed and will not make it to the first ramp! This also means that the booster will not work quite right with this plugin installed, so the fix should be disabled on that map, or the booster's strength should be adjusted to compensate. + +This is a separate plugin from RNGFix for better visibility and because it is not "RNG" related. \ No newline at end of file diff --git a/plugin/gamedata/nojumpboostfix.games.txt b/plugin/gamedata/nojumpboostfix.games.txt new file mode 100644 index 0000000..e48dc1a --- /dev/null +++ b/plugin/gamedata/nojumpboostfix.games.txt @@ -0,0 +1,107 @@ +"Games" +{ + "#default" + { + "Signatures" + { + "CreateInterface" + { + "library" "server" + "windows" "@CreateInterface" + "linux" "@CreateInterface" + } + } + } + + "csgo" + { + "Keys" + { + "NON_JUMP_VELOCITY" "140.0" + } + + "Offsets" + { + // Virtual function offsets + + "CGameMovement::TryPlayerMove" + { + "windows" "42" + "linux" "43" + } + + "CGameMovement::StepMove" + { + "windows" "72" + "linux" "73" + } + + + // Class member offsets + + "CMoveData::m_vecVelocity" + { + "windows" "64" + "linux" "64" + } + + "CMoveData::m_outStepHeight" + { + "windows" "116" + "linux" "116" + } + + "CMoveData::m_vecAbsOrigin" + { + "windows" "172" + "linux" "172" + } + } + } + + "cstrike" + { + "Keys" + { + "NON_JUMP_VELOCITY" "140.0" + } + + "Offsets" + { + // Virtual function offsets + + "CGameMovement::TryPlayerMove" + { + "windows" "32" + "linux" "33" + } + + "CGameMovement::StepMove" + { + "windows" "53" + "linux" "54" + } + + + // Class member offsets + + "CMoveData::m_vecVelocity" + { + "windows" "64" + "linux" "64" + } + + "CMoveData::m_outStepHeight" + { + "windows" "100" + "linux" "100" + } + + "CMoveData::m_vecAbsOrigin" + { + "windows" "152" + "linux" "152" + } + } + } +} diff --git a/plugin/scripting/nojumpboostfix.sp b/plugin/scripting/nojumpboostfix.sp new file mode 100644 index 0000000..9fcdff4 --- /dev/null +++ b/plugin/scripting/nojumpboostfix.sp @@ -0,0 +1,232 @@ +#include +#include + +#pragma semicolon 1 +#pragma newdecls required + +public Plugin myinfo = +{ + name = "No-Jump Boost Fix", + author = "rio", + description = "Changes StepMove behavior that generates free Z velocity when moving up an incline", + version = "1.0.0", + url = "https://github.com/jason-e/no-jump-boost-fix" +}; + +float NON_JUMP_VELOCITY; + +int MOVEDATA_VELOCITY; +int MOVEDATA_OUTSTEPHEIGHT; +int MOVEDATA_ORIGIN; + +ConVar g_cvEnabled; + +Handle g_hTryPlayerMoveHookPost; + +Handle g_hStepMoveHookPre; +Handle g_hStepMoveHookPost; + +Address g_pGameMovement; + +// StepMove call state +Address g_mv; + +bool g_bInStepMove = false; +int g_iTPMCalls = 0; + +float g_vecStartPos[3]; +float g_flStartStepHeight; + +float g_vecDownPos[3]; +float g_vecDownVel[3]; + +float g_vecUpVel[3]; + +public void OnPluginStart() +{ + g_cvEnabled = CreateConVar("nojump_boost_fix", "1", "Enable NoJump Boost Fix.", FCVAR_NOTIFY, true, 0.0, true, 1.0); + + AutoExecConfig(); + + Handle gc = LoadGameConfigFile("nojumpboostfix.games"); + if (gc == null) + { + SetFailState("Failed to load nojumpboostfix gamedata"); + } + + char njv[16]; + if (!GameConfGetKeyValue(gc, "NON_JUMP_VELOCITY", njv, sizeof(njv))) + { + SetFailState("Failed to get NON_JUMP_VELOCITY"); + } + NON_JUMP_VELOCITY = StringToFloat(njv); + + MOVEDATA_VELOCITY = GetRequiredOffset(gc, "CMoveData::m_vecVelocity"); + MOVEDATA_OUTSTEPHEIGHT = GetRequiredOffset(gc, "CMoveData::m_outStepHeight"); + MOVEDATA_ORIGIN = GetRequiredOffset(gc, "CMoveData::m_vecAbsOrigin"); + + StartPrepSDKCall(SDKCall_Static); + if (!PrepSDKCall_SetFromConf(gc, SDKConf_Signature, "CreateInterface")) + { + SetFailState("Failed to get CreateInterface signature"); + } + PrepSDKCall_AddParameter(SDKType_String, SDKPass_Pointer); + PrepSDKCall_AddParameter(SDKType_PlainOldData, SDKPass_Pointer, VDECODE_FLAG_ALLOWNULL); + PrepSDKCall_SetReturnInfo(SDKType_PlainOldData, SDKPass_Plain); + Handle CreateInterface = EndPrepSDKCall(); + if (CreateInterface == null) + { + SetFailState("Unable to prepare SDKCall for CreateInterface"); + } + + g_pGameMovement = SDKCall(CreateInterface, "GameMovement001", 0); + if (!g_pGameMovement) + { + SetFailState("Failed to get IGameMovement singleton pointer"); + } + + int offset = GetRequiredOffset(gc, "CGameMovement::TryPlayerMove"); + g_hTryPlayerMoveHookPost = DHookCreate(offset, HookType_Raw, ReturnType_Int, ThisPointer_Ignore, DHook_TryPlayerMovePost); + DHookAddParam(g_hTryPlayerMoveHookPost, HookParamType_VectorPtr); + DHookAddParam(g_hTryPlayerMoveHookPost, HookParamType_ObjectPtr); + DHookRaw(g_hTryPlayerMoveHookPost, true, g_pGameMovement); + + offset = GetRequiredOffset(gc, "CGameMovement::StepMove"); + g_hStepMoveHookPre = DHookCreate(offset, HookType_Raw, ReturnType_Void, ThisPointer_Ignore, DHook_StepMovePre); + DHookAddParam(g_hStepMoveHookPre, HookParamType_VectorPtr); + DHookAddParam(g_hStepMoveHookPre, HookParamType_ObjectPtr); + DHookRaw(g_hStepMoveHookPre, false, g_pGameMovement); + g_hStepMoveHookPost = DHookCreate(offset, HookType_Raw, ReturnType_Void, ThisPointer_Ignore, DHook_StepMovePost); + DHookAddParam(g_hStepMoveHookPost, HookParamType_VectorPtr); + DHookAddParam(g_hStepMoveHookPost, HookParamType_ObjectPtr); + DHookRaw(g_hStepMoveHookPost, true, g_pGameMovement); + + delete CreateInterface; + delete gc; +} + +int GetRequiredOffset(Handle gc, const char[] key) +{ + int offset = GameConfGetOffset(gc, key); + if (offset == -1) SetFailState("Failed to get %s offset", key); + + return offset; +} + +any GetMoveData(int offset) +{ + return LoadFromAddress(g_mv + view_as
(offset), NumberType_Int32); +} + +void GetMoveDataVector(int offset, float vector[3]) +{ + for (int i = 0; i < 3; i++) + { + vector[i] = GetMoveData(offset + i*4); + } +} + +void SetMoveData(int offset, any value) +{ + StoreToAddress(g_mv + view_as
(offset), value, NumberType_Int32); +} + +void SetMoveDataVector(int offset, const float vector[3]) +{ + for (int i = 0; i < 3; i++) + { + SetMoveData(offset + i*4, vector[i]); + } +} + +public MRESReturn DHook_StepMovePre(Handle hParams) +{ + if (!g_cvEnabled.BoolValue) + return MRES_Ignored; + + g_bInStepMove = true; + g_iTPMCalls = 0; + g_mv = view_as
(LoadFromAddress(g_pGameMovement+view_as
(0x8), NumberType_Int32)); + + GetMoveDataVector(MOVEDATA_ORIGIN, g_vecStartPos); + g_flStartStepHeight = view_as(GetMoveData(MOVEDATA_OUTSTEPHEIGHT)); + + return MRES_Handled; +} + +public MRESReturn DHook_TryPlayerMovePost(Handle hReturn, Handle hParams) +{ + if (!g_bInStepMove) + return MRES_Ignored; + + if (!g_cvEnabled.BoolValue) + return MRES_Ignored; + + g_iTPMCalls++; + + switch (g_iTPMCalls) + { + case 1: + { + // This was the call for the "down" move. + GetMoveDataVector(MOVEDATA_ORIGIN, g_vecDownPos); + GetMoveDataVector(MOVEDATA_VELOCITY, g_vecDownVel); + } + case 2: + { + // This was the call for the "up" move. + // At this time, the origin doesn't include the step down, but we don't need it anyway. + GetMoveDataVector(MOVEDATA_VELOCITY, g_vecUpVel); + } + default: + { + SetFailState("TryPlayerMove ran more than two times in one StepMove call?"); + } + } + + return MRES_Handled; +} + +public MRESReturn DHook_StepMovePost(Handle hParams) +{ + g_bInStepMove = false; + + if (!g_cvEnabled.BoolValue) + return MRES_Ignored; + + float vecFinalPos[3]; + GetMoveDataVector(MOVEDATA_ORIGIN, vecFinalPos); + + if (g_iTPMCalls == 2 && GetVectorDistance(vecFinalPos, g_vecDownPos, true) != 0.0) + { + // StepMove chose the "up" result, which means it also used just the Z-velocity + // from the "down" result. We don't want to do that because it can lead to the + // player getting to keep all of their horizontal velocity, but also getting some + // Z-velocity for free. Instead, we want to use one entire result or the other. + + if (g_vecDownVel[2] > NON_JUMP_VELOCITY) + { + // In this case, the "down" result gave the player enough Z-velocity to start sliding up. + // The "up" result went farther, but we actually really want to keep the "down" result's + // Z-velocity because sliding is the more important outcome -- so use the "down" result. + SetMoveDataVector(MOVEDATA_ORIGIN, g_vecDownPos); + SetMoveDataVector(MOVEDATA_VELOCITY, g_vecDownVel); + + float flStepDist = g_vecDownPos[2] - g_vecStartPos[2]; + if (flStepDist > 0.0) + { + SetMoveData(MOVEDATA_OUTSTEPHEIGHT, g_flStartStepHeight + flStepDist); + } + } + else + { + // The "up" result is fine, but use the "up" result's actual velocity without combining it. + // Doing this probably doesn't matter because we know the "down" Z-velocity is not more than + // NON_JUMP_VELOCITY, which means the player will still be on the ground after CategorizePostion + // and their Z-velocity will be reset to zero -- but let's do this anyway to be totally sure. + SetMoveDataVector(MOVEDATA_VELOCITY, g_vecUpVel); + } + } + + return MRES_Handled; +} diff --git a/steps.gif b/steps.gif new file mode 100644 index 0000000..54f5957 Binary files /dev/null and b/steps.gif differ