Item Save Glitch, or ISG for short, is a developer variable meant as a fix for the Half-Life 2 demo at E3 2003.
Whenever its active, every VPhys object in the game loses its collision.
This includes the player's VPhys Bounding Box, also known as the Havok Bounding Box, allowing the player to clip through anything, and put props inside of themself in a way very similar to quickclip.
Clipping this way is however significantly harder than quickclipping, due to the fact that while ISG is active, props simply phase through walls and floors.
For an unknwon reason, getting stuck in props with ISG only works when the player is crouched. The game will not recognize the player being stuck if they're standing.
Despite ISG being possible in every version of most Half-Life 2 derivative games, consistent setups are only known for Half-Life 2, Episode One, Episode Two, and Portal. Any instance of it happening outside of these games was accidental, and it is not known how it can happen.
Once active, the only way to get rid of ISG is to restart the game.
ISG as a glitch (not as a term) been known to exist for a long time, as it has happened to numerous people on their casual playthroughs of the game. There are documented examples of it happening as early as 2005 in Half-Life 2: Deathmatch, thanks old steam forums.
The first actual semi-consistent way to get ISG was found in Portal in 2014, and that is where the name "Item Save Glitch" comes from.
After the setup has been found, people assumed that ISG is a variation of Save Glitch happening to props, instead of the player, as the setup utilizes rubbing a prop against a portal. This later turned out to be false, but the name remained.
Just 2 years later, in 2016 margen64 discovered BBT. During his experiments with the glitch, he has managed to achieve ISG in Half-Life 2. By this point, some setups involving cheat commands were also known. Unfortunately, similarly to BBT, this would quickly be forgotten and neither of the glitches would start seeing any use.
In August of 2023, a Portal runner, nick, has managed to achieve ISG on accident while performing BBT, and posted a video of it to the community. This time the discovery wasn't forgotten, and in the same month PegasDoggeAM has managed to recreate it while utilizing BBT, although no one knew about it (except Rama) until January.
If you just want to play around with ISG, you can use the
spt_set_isg
command from SPT.
ISG in Portal is performed by first placing a portal on the very edge of a wall, then moving an object halfway trough the portal, but without directly moving it through, and lastly moving a different object through the first one.
A video demonstration can be seen here:
https://www.youtube.com/watch?v=ZB1CrtS75vI
The method for executing ISG in Half-Life 2 utilizes sending a physics object to a location the game considers "Infinite" by utilizing BBT.
First, start up BBT as usual, and hit the floating prop. Make sure that it flies away, really quickly. It is not required to continue floating after hitting the prop.
Shortly after it flies away, (not too early, not too late) execute a saveload. The saveload can't be done immidietely, as the physics object will take a few seconds (usually only 2 or 3) to reach infinity.
After the saveload, the game will be on a crash timer. ISG needs to happen before that timer is up.
In order to make it happen, simply make a physics object (like a prop) touch another, sleeping physics object. This works better with some than with others, and there are instances where this method seems can seem pretty random.
If when the 2 objects touch, one collides with the other like normal, or if it simply goes inside of the other, try saveloading again.
If the collision continues behaving like normal, and a crash doesn't occur, the prop didn't fly away far enough for ISG to happen.
The prop which was hit, should start falling through the floor, and the moment that happens load back to an old save, from before BBT was started. If this is not done, or done too late, the game will crash. ISG should be active now.
It is not well known why ISG happens, but the way it works once it is active, is known.
The only publicly available code which contains the ISG variable is from the Half-Life 2 source code leak from 2003. Because of that, code snippets will not be shown.
The physics engine uses IVP_Mindist
objects to keep track of pairs of nearby collidable objects. These objects also estimate when the pair is about to collide based on factors such as position and velocity, and will schedule an event which will call IVP_Mindist::do_impact
(line 1221 of ivp_impact.cxx
). One of the first things this function does is recalculate the set of nearby collidable objects (and thus mindists) ultimately through a call to IVP_Mindist::recheck_ov_element
(line 652 of ivp_mindist.cxx
). This latter function will create new mindists for new objects in the set, and delete mindists for objects no longer in the set.
One issue that can occur is if during the recalculation that occurs during impact processing, the two objects involved are no longer nearby or collidable, this could result in deletion of the currently processed mindist, resulting in a use-after-free. To work around this, a mechanism was set up to defer the deletion of such mindists. Before recalculating nearby objects in do_impact
, the variable g_pCurrentMindist
is set to the mindist, and then set to NULL after recalculation. Inside recheck_ov_element
the variable is checked when about to delete the mindist. If the mindists are equal, the function instead sets a flag called g_fDeferDeleteMindist
.
Afterwards in do_impact
, the flag is checked. If it is, the mindist of the specific physics object is deleted, and the rest of the function is skipped (this includes reporting of collision events). The flag is not reset afterwards. Since the variable is always true after being set, this affects every physics object in the game (except the player, which uses a different physics system), essentially resulting in collisions being disabled.
A comment from ivp_mindist.cxx
implies that this was a fix for E3 2003:
HACKHACK: E3 2003, get around delete of collision object that is calling us up the stack
This condition can occur if the game forgets to call CBaseEntity::CollisionRulesChanged
or IPhysicsObject::RecheckCollisionFilter
(called by the former) when changing some property used by its collision filter (IPhysicsCollisionEvent::ShouldCollide
). RecheckCollisionFilter
calls recheck_ov_element
to recalculate nearby collidable objects, taking the updated collision filter into account. A comment in CollisionRulesChanged
(part of the SDK source code in game/shared/baseentity_shared.cpp
) warns about this:
ivp maintains state based on recent return values from the collision filter, so anything
that can change the state that a collision filter will return (like m_Solid) needs to call RecheckCollisionFilter.
Failure to do so will result in stale mindists that can trigger ISG by continuing to schedule impact events. One instance where Portal forgets to call CollisionRulesChanged
is when objects move in and out of the Portal Clone Area. This is a large region that extends in front but not behind the portal (this can be seen with the sv_portal_debug_touch 1
and developer 1
commands). Objects in this region are special as Portal's collision filter prevents objects in portals from colliding with any object not in this region. In the video above a mindist is created as the radio exits the cube on the visible half (in the Portal Clone Area), made stale by moving it outside the area over the invisible half, then ISG triggered as the radio impacts the invisible half.
(TODO: Likely not entirely accurate. Needs verification)