This article's documentation is for anything that uses the Source engine. Click here for more information.

Source Multiplayer Networking: Difference between revisions

From Valve Developer Community
Jump to navigation Jump to search
m (cleanup (need an article on tickrate))
 
(138 intermediate revisions by 28 users not shown)
Line 1: Line 1:
== Overview ==
{{LanguageBar}}
Multiplayer games based on the [[Source Engine]] use a Client/Server networking architecture. Usually a server is a dedicated host that runs the game and is authoritative about world simulation, game rules, and player input processing. A client is a player's computer connected to a game server. The client and server communicate with each other by sending small data packets at a high frequency (usually 20 to 30 packets per second). A client receives the current world state from the server and generates video and audio output based on these updates. The client also samples data from input devices (keyboard, mouse, microphone, etc.) and sends these input samples back to the server for further processing. Clients only communicate with the game server and not between each other (like in a peer-to-peer application). In contrast with a single player game, a multiplayer game has to deal with a variety of new problems caused by packet-based communication.
<noinclude></noinclude>
{{toc-right}}


Network bandwidth is limited, so the server can't send a new update packet to all clients for every single world change. Instead, the server takes snapshots of the current world state at a constant rate and broadcasts these snapshots to the clients. Network packets take a certain amount of time to travel between the client and the server (i.e. the ping time). This means that the client time is always a little bit behind the server time. Furthermore, client input packets are also delayed on their way back, so the server is processing temporally delayed user commands. In addition, each client has a different network delay which varies over time due to other background traffic and the client's framerate. These time differences between server and client causes logical problems, becoming worse with increasing network latencies. In fast-paced action games, even a delay of a few milliseconds can cause a laggy gameplay feeling and make it hard to hit other players or interact with moving objects. Besides bandwidth limitations and network latencies, information can get lost due to network packet loss.
Multiplayer games based on the {{src|4}} use a [[Wikipedia:Client-server|Client-Server]] networking architecture. Usually a server is a dedicated host that runs the game and is authoritative about world simulation, game rules, and player input processing. A client is a player's computer connected to a game server. The client and server communicate with each other by sending small data packets at a high frequency (usually 20 to 30 packets per second). A client receives the current world state from the server and generates video and audio output based on these updates. The client also samples data from input devices (keyboard, mouse, microphone, etc.) and sends these input samples back to the server for further processing. Clients only communicate with the game server and not between each other (like in a peer-to-peer application). In contrast with a single player game, a multiplayer game has to deal with a variety of new problems caused by packet-based communication.


[[Image:Networking1.gif|center|Time differences between client and server]]
Network bandwidth is limited, so the server can't send a new update packet to all clients for every single world change. Instead, the server takes snapshots of the current world state at a constant rate and broadcasts these snapshots to the clients. Network packets take a certain amount of time to travel between the client and the server (i.e. half the ping time). This means that the client time is always a little bit behind the server time. Furthermore, client input packets are also delayed on their way back, so the server is processing temporally delayed user commands. In addition, each client has a different network delay which varies over time due to other background traffic and the client's framerate. These time differences between server and client causes logical problems, becoming worse with increasing network latencies. In fast-paced action games, even a delay of a few milliseconds can cause a laggy gameplay feeling and make it hard to hit other players or interact with moving objects. Besides bandwidth limitations and network latencies, information can get lost due to network packet loss.


To cope with all these issues introduced by network communication, the Source engine uses multiple techniques to solve these problems, or at least make them less visible to the player. These techniques include data compression, interpolation, prediction, and lag compensation. These techniques are tightly coupled, and changes made within one system may affect other systems. This document describes the general functionality of these systems and how they work together.
[[File:Source_Networking_Highres.png|750px|center|Time differences between client and server|link=]]
 
To cope with the issues introduced by network communication, the Source engine server employs techniques such as data compression and lag compensation which are invisible to the client. The client then performs prediction and interpolation to further improve the experience.
 
{{Note|If you follow a player in "First-Person" as a spectator in a game or [[SourceTV]], you don't exactly see what the player sees. Spectators see the game world without [[lag compensation]].}}


== Basic networking ==
== Basic networking ==
The server simulates the game in discrete time steps called ticks. By default, 66 ticks per second are simulated, but mods can specify their own [[tickrate]]. For example [[Counter-Strike: Source]] uses a lower tickrate of 33 ticks/second to reduce the server CPU load. During each tick, the server processes incoming user commands, runs a physical simulation step, checks the game rules, and updates all object states. After simulating a tick, the server decides if any client needs a world update and takes a snapshot of the current world state if necessary. A higher tickrate increases the simulation precision, but also requires more CPU power and available bandwidth on both server and client. The server admin may override the default tickrate with the <code>-tickrate</code> command line parameter, though tickrate changes done this way are not recommended because the mod may not work as designed if its tickrate is changed.


Clients usually have only a limited amount of available bandwidth. In the worst case, players with a modem connection can't receive more than 5 to 7 KB/sec. If the server tried to send them updates with a higher data rate, packet loss would be unavoidable. Therefore, the client has to tell the server its incoming bandwidth capacity by setting the console variable <code>rate</code> (in bytes/second). This is the most important network variable for clients and it has to be set correctly for an optimal gameplay experience. The client can request a certain snapshot rate by changing <code>cl_updaterate</code> (default 20), but the server will never send more updates than simulated ticks or exceed the requested client <code>rate</code> limit. Server admins can limit data rate values requested by clients with <code>sv_minrate</code> and <code>sv_maxrate</code> (both in bytes/second). Also the snapshot rate can be restricted with <code>sv_minupdaterate</code> and <code>sv_maxupdaterate</code> (both in snapshots/second).  
The server simulates the game in discrete time steps called ticks. By default, the timestep is 15ms, so 66.666... ticks per second are simulated, but mods can specify their own tickrate. During each tick, the server processes incoming user commands, runs a physical simulation step, checks the game rules, and updates all object states. After simulating a tick, the server decides if any client needs a world update and takes a snapshot of the current world state if necessary. A higher tickrate increases the simulation precision, but also requires more CPU power and available bandwidth on both server and client. The server admin may override the default tickrate with the {{code|<nowiki>-tickrate</nowiki>}} command line parameter, though tickrate changes done this way are not recommended because the mod may not work as designed if its tickrate is changed.


The client creates ''user commands'' from sampling input devices with the same tick rate that the server is running with. A user command is basically a snapshot of the current keyboard and mouse state. But instead of sending a new packet to the server for each user command, the client sends command packets at a certain rate of packets per second (usually 30). This means two or more user commands are transmitted within the same packet. Clients can increase the command rate with <code>cl_cmdrate</code>. This will increase responsiveness but requires more outgoing bandwidth, too.
{{note|The {{code|<nowiki>-tickrate</nowiki>}} command line parameter is not available newer version of CS:S, DoD:S, TF2, L4D and L4D2, and was disabled in code likely because changing tickrate may cause causes server timing issues for certain games. <br>The tickrate is set to 66 in CSS, DoD S and TF2. 60 in CS 1.6 and HL1, and 30 in L4D, L4D2 and TFC.<br>Some older version of those games can have it's tickrate changed, however. Users can also install this [[#Server plugins|server plugins ↓]] to restore the disabled {{code|<nowiki>-tickrate</nowiki>}} command line parameter.}}


Game data is compressed using ''delta compression'' to reduce network load. That means the server doesn't send a full world snapshot each time, but rather only changes (a delta snapshot) that happened since the last acknowledged update. With each packet sent between the client and server, acknowledge numbers are attached to keep track of their data flow. Usually full (non-delta) snapshots are only sent when a game starts or a client suffers from heavy packet loss for a couple of seconds. Clients can request a full snapshot manually with the <code>cl_fullupdate</code> command.
Clients usually have only a limited amount of available bandwidth. In the worst case, players with a modem connection can't receive more than 5 to 7 KB/sec. If the server tried to send them updates with a higher data rate, packet loss would be unavoidable. Therefore, the client has to tell the server its incoming bandwidth capacity by setting the console variable {{code|<nowiki>rate</nowiki>}} (in bytes/second). This is the most important network variable for clients and it has to be set correctly for an optimal gameplay experience. The client can request a certain snapshot rate by changing {{code|<nowiki>cl_updaterate</nowiki>}} (default 20), but the server will never send more updates than simulated ticks or exceed the requested client {{code|<nowiki>rate</nowiki>}} limit. Server admins can limit data rate values requested by clients with {{code|<nowiki>sv_minrate</nowiki>}} and {{code|<nowiki>sv_maxrate</nowiki>}} (both in bytes/second). Also the snapshot rate can be restricted with {{code|<nowiki>sv_minupdaterate</nowiki>}} and {{code|<nowiki>sv_maxupdaterate</nowiki>}} (both in snapshots/second).
 
The client creates ''user commands'' from sampling input devices with the same tick rate that the server is running with. A user command is basically a snapshot of the current keyboard and mouse state. But instead of sending a new packet to the server for each user command, the client sends command packets at a certain rate of packets per second (usually 30). This means two or more user commands are transmitted within the same packet. Clients can increase the command rate with {{code|<nowiki>cl_cmdrate</nowiki>}}. This will increase responsiveness but requires more outgoing bandwidth, too.
 
Game data is compressed using ''delta compression'' to reduce network load. That means the server doesn't send a full world snapshot each time, but rather only changes (a delta snapshot) that happened since the last acknowledged update. With each packet sent between the client and server, acknowledge numbers are attached to keep track of their data flow. Usually full (non-delta) snapshots are only sent when a game starts or a client suffers from heavy packet loss for a couple of seconds. Clients can request a full snapshot manually with the {{code|<nowiki>cl_fullupdate</nowiki>}} command.


Responsiveness, or the time between user input and its visible feedback in the game world, are determined by lots of factors, including the server/client CPU load, simulation tickrate, data rate and snapshot update settings, but mostly by the network packet traveling time. The time between the client sending a user command, the server responding to it, and the client receiving the server's response is called the ''latency'' or ''ping'' (or round trip time). Low latency is a significant advantage when playing a multiplayer online game. Techniques like prediction and lag compensation try to minimize that advantage and allow a fair game for players with slower connections. Tweaking networking setting can help to gain a better experience if the necessary bandwidth and CPU power is available. We recommend keeping the default settings, since improper changes may cause more negative side effects than actual benefits.
Responsiveness, or the time between user input and its visible feedback in the game world, are determined by lots of factors, including the server/client CPU load, simulation tickrate, data rate and snapshot update settings, but mostly by the network packet traveling time. The time between the client sending a user command, the server responding to it, and the client receiving the server's response is called the ''latency'' or ''ping'' (or round trip time). Low latency is a significant advantage when playing a multiplayer online game. Techniques like prediction and lag compensation try to minimize that advantage and allow a fair game for players with slower connections. Tweaking networking setting can help to gain a better experience if the necessary bandwidth and CPU power is available. We recommend keeping the default settings, since improper changes may cause more negative side effects than actual benefits.
== Servers for these games that support altering Tickrate ==
{{Note|Most singleplayer games such as {{hl2|4}} can have it's tickrate changed.}}
{{clarify|what about {{cs|4}}?, {{hl1|4}} seems to use 60 or something according to "{{code|+graph}}", it is unknown if this can be altered.}}
{{Clarify|How about other {{src2|4}}.}}
'''The tickrate can be altered by using the {{code|<nowiki>-tickrate</nowiki>}} parameter for these games:'''
=== {{src|3.1}} ===
*{{csgo|4}} (Default: 64)
** Most servers commonly runs on either 64 or 128 tickrate.
*{{Dota2|4}} (original release, Default: 30)
*{{gmod|4}} (Default: 66)
*{{hl2dm|4}} (Default: 66)
*{{portal2|4}} (Default: 60)
'''By default, without using [[#Server plugins|server plugins ↓]], some servers tickrate cannot be altered for these games due to various reasons such as {{Gldsrc|4}} lacking the {{code|-tickrate}} [[Command_line_options|command line parameter]], or intentionally disabled in-code by Valve due to "server ops are abusing it" and/or "using {{code|-tickrate}} may cause server timing-related issues":'''
=== {{gldsrc|3.1}} ===
==== 60 Tickrate ====
*{{cs|4}}
*{{hl|4}}
==== 30 Tickrate ====
*{{tfc|4}}
=== {{src|3.1}} ===
==== 66 Tickrate ====
*{{css|4}}
** Formerly 33 prior to Orange Box (Source 2009) update.
*{{dods|4}}
** {{code|-tickrate}} removed after [http://store.steampowered.com/news/6510/ October 13, 2011].
*{{tf2|4}}
==== 30 Tickrate ====
*{{l4d|4}}
*{{l4d2|4}}
=== {{src2|3.1}} ===
==== 64 Tickrate, with Sub-tick ====
*{{cs2|4}}
**{{note|Originally can be changed during Limited Test but later was hardcoded to 64-tick.}}
{{MConfirm|Does the {{code|-tickrate}} command (or similar) exists or works with GoldSrc games?}}
{{Note|In previous versions on some of those games, it can be changed. Alternatively user can also use [[#Server plugins|server plugins ↓]].}}
{{Note|The lowest or highest values depends on the game. Minimum tickrate is 11.}}
=== <span id="Server plugins">Using server plugins to restore {{code|-tickrate}} for games that have it disabled</span> ===
Users can install this server plugins to make the {{code|-tickrate}} command works again on games that have it disabled. Note that loading plugins would require {{code|-insecure}} (disabling [[Valve Anti-Cheat|VAC]]) for it to work.<br>[https://github.com/daemon32/tickrate_enabler All Source games] / [https://github.com/accelerator74/Tickrate-Enabler/ L4D & L4D2]
== Steam Networking ==
'''[https://partner.steamgames.com/doc/features/multiplayer/networking Steam Networking]''' is a new feature which allows self-hosting using [https://partner.steamgames.com/doc/features/multiplayer/steamdatagramrelay Steam Datagram Relay], meaning users can easily create joinable servers without port forwarding. User can enable/disable Steam Networking through Create Server menu, or through console command: {{ent|sv_use_steam_networking}}.
'''The following games supports Steam Networking:'''
=== {{gldsrc|3.1}} ===
Added since Half-Life 20th Anniversary Update.
*{{cs|4}}
*{{hl|4}} (and all non-standalone games based directly on Half-Life)
=== {{src|3.1}} ===
==== {{tf2branch|3.1}} games ====
Added since February 18, 2025 update for existing games running on TF2 branch, while other games running on {{src13mp|1}} have been upgraded to TF2 branch on the same day, adding Steam Networking support.
*{{css|4}}
*{{dods|4}}
*{{hl2dm|4}}
*{{hldms|4}}
*{{tf2|4}}
*{{srcsdk13mp|4}} (2025 version)
==== {{src13mp|3.1}} games ====
*{{synergy|4}}
=== {{src2|3.1}} ===
{{Todo}}


== Entity interpolation ==
== Entity interpolation ==
By default, the client receives about 20 snapshot per second. If the objects (entities) in the world were only rendered at the positions received by the server, moving objects and animation would look choppy and jittery. Dropped packets would also cause noticeable glitches. The trick to solve this problem is to go back in time for rendering, so positions and animations can be continuously interpolated between two recently received snapshot. This technique is called client side entity interpolation and is enabled by default with <code>cl_interpolate 1</code>. With 20 snapshots per second, a new update arrives about every 50 milliseconds. If the client render time is shifted back by 50 milliseconds, entities can be always interpolated between the last received snapshot and the snapshot before that. The Source engine does the entity interpolation with a 100-millisecond delay (<code>cl_interp 0.1</code>). This way, even if one snapshot is lost, there are always two valid snapshots to interpolate between. Take a look at the following figure showing the arrival times of incoming world snapshots:


[[Image:Interpolation.gif|center|Interpolation timeline]]
{{Main|Interpolation}}
 
By default, the client receives about 20 snapshot per second. If the objects (entities) in the world were only rendered at the positions received by the server, moving objects and animation would look choppy and jittery. Dropped packets would also cause noticeable glitches. The trick to solve this problem is to go back in time for rendering, so positions and animations can be continuously interpolated between two recently received snapshots. With 20 snapshots per second, a new update arrives about every 50 milliseconds. If the client render time is shifted back by 50 milliseconds, entities can be always interpolated between the last received snapshot and the snapshot before that.
 
Source defaults to an interpolation period ('lerp') of 100-milliseconds ({{code|<nowiki>cl_interp 0.1</nowiki>}}); this way, even if one snapshot is lost, there are always two valid snapshots to interpolate between. Take a look at the following figure showing the arrival times of incoming world snapshots:
 
[[File:Source_Multiplayer_Networking_-_Interpolation.png|750px|center|Interpolation timeline|link=]]


The last snapshot received on the client was at tick 344 or 10.30 seconds. The client time continues to increase based on this snapshot and the client frame rate. If a new video frame is rendered, the rendering time is the current client time 10.32 minus the view interpolation delay of 0.1 seconds. This would be 10.22 in our example and all entities and their animations are interpolated using the correct fraction between snapshot 340 and 342.
The last snapshot received on the client was at tick 344 or 10.30 seconds. The client time continues to increase based on this snapshot and the client frame rate. If a new video frame is rendered, the rendering time is the current client time 10.32 minus the view interpolation delay of 0.1 seconds. This would be 10.22 in our example and all entities and their animations are interpolated using the correct fraction between snapshot 340 and 342.


Since we have an interpolation delay of 100 milliseconds, the interpolation would even work if snapshot 342 were missing due to packet loss. Then the interpolation could use snapshots 340 and 344. If more than one snapshot in a row is dropped, interpolation can't work perfectly because it runs out of snapshots in the history buffer. In that case the renderer uses extrapolation (<code>cl_extrapolate 1</code>) and tries a simple linear extrapolation of entities based on their known history so far. The extrapolation is done only for 0.25 seconds of packet loss (<code>cl_extrapolate_amount</code>), since the prediction errors would become too big after that.  
Since we have an interpolation delay of 100 milliseconds, the interpolation would even work if snapshot 342 were missing due to packet loss. Then the interpolation could use snapshots 340 and 344. If more than one snapshot in a row is dropped, interpolation can't work perfectly because it runs out of snapshots in the history buffer. In that case the renderer uses extrapolation ({{code|<nowiki>cl_extrapolate 1</nowiki>}}) and tries a simple linear extrapolation of entities based on their known history so far. The extrapolation is done only for 0.25 seconds of packet loss ({{code|<nowiki>cl_extrapolate_amount</nowiki>}}), since the prediction errors would become too big after that.  


The entity interpolation is causing a constant view "lag" of 100 milliseconds, even if you're playing on a listen server (server and client on the same machine). So if you turn on <code>sv_showhitboxes</code> the player hitboxes are drawn in server time, meaning they are ahead of the rendered player model by 100 milliseconds. This doesn't mean you have to lead your aiming when shooting at other players since the server-side lag compensation knows about client entity interpolation and corrects this error. If you turn off interpolation on a listen server (<code>cl_interpolate 0</code> and <code>cl_lagcompensation 0</code>), the drawn hitboxes will match the rendered player model again, but the animations and moving objects will become very jittery. Failing to turn off lag compensation (<code>cl_lagcompensation 0</code>) with interpolation on can result in undesired results.
'''Entity interpolation causes a constant view "lag" of 100 milliseconds by default ({{code|<nowiki>cl_interp 0.1</nowiki>}}), even if you're playing on a listenserver''' (server and client on the same machine). This doesn't mean you have to lead your aiming when shooting at other players since the server-side lag compensation knows about client entity interpolation and corrects this error.
 
{{tip|More recent Source games have the {{code|<nowiki>cl_interp_ratio</nowiki>}} cvar. With this you can easily and safely decrease the interpolation period by setting {{code|<nowiki>cl_interp</nowiki>}} to 0, then increasing the value of {{code|<nowiki>cl_updaterate</nowiki>}} (the useful limit of which depends on server tickrate). You can check your final lerp with {{code|<nowiki>net_graph 1</nowiki>}}.}}
 
{{note|If you turn on {{code|<nowiki>sv_showhitboxes</nowiki>}} (not available in Source 2009) you will see player hitboxes drawn in server time, meaning they are ahead of the rendered player model by the lerp period. This is perfectly normal!}}


== Input prediction ==
== Input prediction ==
Lets assume a player has a network latency of 100 milliseconds and starts to move forward. The information that the <code>+FORWARD</code> key is pressed is stored in a user command and send to the server. There the user command is processed by the movement code and the player's character is moved forward in the game world. This world state change is transmitted to all clients with the next snapshot update. So the player would see his own change of movement with a 100 milliseconds delay after he started walking. This delay applies to all players actions like movement, shooting weapons, etc. and becomes worse with higher latencies.


A delay between player input and corresponding visual feedback creates a strange, unnatural feeling and makes it hard to move or aim precisely. Client-side input prediction (<code>cl_predict 1</code>) is a way to remove this delay and let the player's actions feel more instant. Instead of waiting for the server to update your own position, the local client just predicts the results of its own user commands. Therefore, the client runs exactly the same code and rules the server will use to process the user commands. After the prediction is finished, the local player will move instantly to the new location while the server still sees him at the old place.
{{Main|Prediction}}
 
Lets assume a player has a network latency of 150 milliseconds and starts to move forward. The information that the {{code|<nowiki>+FORWARD</nowiki>}} key is pressed is stored in a user command and send to the server. There the user command is processed by the movement code and the player's character is moved forward in the game world. This world state change is transmitted to all clients with the next snapshot update. So the player would see his own change of movement with a 150 milliseconds delay after he started walking. This delay applies to all players actions like movement, shooting weapons, etc. and becomes worse with higher latencies.
 
A delay between player input and corresponding visual feedback creates a strange, unnatural feeling and makes it hard to move or aim precisely. Client-side input prediction ({{code|<nowiki>cl_predict 1</nowiki>}}) is a way to remove this delay and let the player's actions feel more instant. Instead of waiting for the server to update your own position, the local client just predicts the results of its own user commands. Therefore, the client runs exactly the same code and rules the server will use to process the user commands. After the prediction is finished, the local player will move instantly to the new location while the server still sees him at the old place.


After 100 milliseconds, the client will receive the server snapshot that contains the changes based on the user command he predicted earlier. Then the client compares the server position with his predicted position. If they are different, a prediction error has occurred. This indicates that the client didn't have the correct information about other entities and the environment when it processed the user command. Then the client has to correct its own position, since the server has final authority over client-side prediction. If <code>cl_showerror 1</code> is turned on, clients can see when prediction errors happen. Prediction error correction can be quite noticeable and may cause the client's view to jump erratically. By gradually correcting this error over a short amount of time (<code>cl_smoothtime</code>), errors can be smoothly corrected. Prediction error smoothing can be turned off with <code>cl_smooth 0</code>.
After 150 milliseconds, the client will receive the server snapshot that contains the changes based on the user command he predicted earlier. Then the client compares the server position with his predicted position. If they are different, a prediction error has occurred. This indicates that the client didn't have the correct information about other entities and the environment when it processed the user command. Then the client has to correct its own position, since the server has final authority over client-side prediction. If {{code|<nowiki>cl_showerror 1</nowiki>}} is turned on, clients can see when prediction errors happen. Prediction error correction can be quite noticeable and may cause the client's view to jump erratically. By gradually correcting this error over a short amount of time ({{code|<nowiki>cl_smoothtime</nowiki>}}), errors can be smoothly corrected. Prediction error smoothing can be turned off with {{code|<nowiki>cl_smooth 0</nowiki>}}.


Predicting an object's behavior only works if the clients knows the same rules and state of the object as the server. That's usually not the case since the server knows more internal information about objects than the clients do. Clients see only a small part of the world and just get enough information to render objects. Therefore, prediction works only for your own player, and the weapons controlled by you. Proper prediction of other players or interactive objects is not possible on the client at this point.
Prediction is only possible for the local player and entities affected only by him, since prediction works by using the client's keypresses to make a "best guess" of where the player will end up. Predicting other players would require literally predicting the future with no data, since there's no way to instantaneously get keypresses from them.


== [[Lag Compensation|Lag compensation]] ==
== Lag compensation <!-- linked to from [[Lag Compensation]] --> ==
Let's say a player shoots at a target at client time 10.5. The firing information is packed into a user command and sent to the server. While the packet is on its way through the network, the server continues to simulate the world, and the target might have moved to a different position. The user command arrives at server time 10.6 and the server wouldn't detect the hit, even though the player has aimed exactly at the target. This error is corrected by the server-side lag compensation (<code>sv_unlag 1</code>)


The lag compensation system keeps a history of all recent player positions for a time span of about one second (can be changed with <code>sv_maxunlag</code>). If a user command is executed, the server estimates at what time the command was created. This command execution time is calculated as followed:
:''All source code for lag compensation and view interpolation is available in the Source SDK. See [[Lag compensation]] for implementation details.''


Command Execution Time = Current Server Time - Packet Round-Trip-Time - Client View Interpolation
Let's say a player shoots at a target at client time 10.5. The firing information is packed into a user command and sent to the server. While the packet is on its way through the network, the server continues to simulate the world, and the target might have moved to a different position. The user command arrives at server time 10.6 and the server wouldn't detect the hit, even though the player has aimed exactly at the target. This error is corrected by the server-side lag compensation.


Then the server moves all other players back to where they were at the command execution time. The user command is executed and the hit is detected correctly.  However, since entity interpolation is compensated here, failing to have it on can cause undesired results. After the user command has been processed, the players are moved back to their original position. On a listen server you can enable <code>sv_showimpacts 1</code> to see the different server and client hitboxes:
The lag compensation system keeps a history of all recent player positions for one second. If a user command is executed, the server estimates at what time the command was created as follows:


[[Image:Lag compensation.jpg|center|Visualisation of lag compensation: server and client hitboxes]]
<span style="text-align:center;font-family:monospace;display:block;font-size:1.2em;"><strong>Command Execution Time</strong> = <strong>Current Server Time</strong> - <strong>Packet Latency</strong> - <strong>Client View [[Interpolation]]</strong></span>


This screenshot was taken on a listen server with 200 milliseconds of lag (using <code>net_fakelag</code>), right after the server confirmed the hit. The red hitbox shows the target position on the client where it was 100 milliseconds ago. Since then, the target continued to move to the left while the user command was traveling to the server. After the user command arrived, the server restored the target position (blue hitbox) based on the estimated command execution time. The server traces the shot and confirms the hit (the client sees blood effects). Client and server hitboxes don't exactly match because of small precision errors in time measurement. Even a small difference of a few milliseconds can cause an error of several inches for fast-moving objects. Multiplayer hit detection is not pixel perfect and has known precision limitations based on the tickrate and the speed of moving objects. Increasing the tickrate does improve the precision of hit detection, but also requires more CPU, memory, and bandwidth capacity for server and clients.
Then the server moves all other players - ''only'' players - back to where they were at the command execution time. The user command is executed and the hit is detected correctly. After the user command has been processed, the players revert to their original positions.


The question arises, why is hit detection so complicated on the server? Doing the back tracking of player positions and dealing with precision errors while hit detection could be done client-side way easier and with pixel precision. The client would just tell the server with a "hit" message what player has been hit and where. We can't allow that simply because a game server can't trust the clients on such important decisions. Even if the client is "clean" and protected by [[Wikipedia:Valve_Anti-Cheat|Valve-Anti-Cheat]], the packets could be still modified on a 3rd machine while routed to the game server. These "cheat proxies" could inject "hit" messages into the network packet without being detected by VAC (a "man-in-the-middle" attack).
{{note|Since entity interpolation is included in the equation, failing to have it on can cause undesired results.}}


Network latencies and lag compensation can create paradoxes that seem illogical compared to the real world. For example, you can be hit by an attacker you can't even see anymore because you already took cover. What happened is that the server moved your player hitboxes back in time, where you were still exposed to your attacker. This inconsistency problem can't be solved in general because of the relative slow packet speeds. In the real world, you don't notice this problem because light (the packets) travels so fast and you and everybody around you see the same world as it is right now.
On a listen server you can enable {{code|<nowiki>sv_showimpacts 1</nowiki>}} to see the different server and client hitboxes:
 
[[File:Lag compensation.jpg|center|Visualisation of lag compensation: server and client hitboxes]]
 
This screenshot was taken on a listen server with 200 milliseconds of lag (using {{code|<nowiki>net_fakelag</nowiki>}}), right after the server confirmed the hit. The red hitbox shows the target position on the client where it was 100ms + interp period ago. Since then, the target continued to move to the left while the user command was travelling to the server. After the user command arrived, the server restored the target position (blue hitbox) based on the estimated command execution time. The server traces the shot and confirms the hit (the client sees blood effects).
 
Client and server hitboxes don't exactly match because of small precision errors in time measurement. Even a small difference of a few milliseconds can cause an error of several inches for fast-moving objects. Multiplayer hit detection is not pixel perfect and has known precision limitations based on the tickrate and the speed of moving objects.
 
The question arises, why is hit detection so complicated on the server? Doing the back tracking of player positions and dealing with precision errors while hit detection could be done client-side way easier and with pixel precision. The client would just tell the server with a "hit" message what player has been hit and where. We can't allow that simply because a game server can't trust the clients on such important decisions. Even if the client is "clean" and protected by {{vac|2|nt=0}}, the packets could be still modified on a 3rd machine while routed to the game server. These "cheat proxies" could inject "hit" messages into the network packet without being detected by VAC (a "man-in-the-middle" attack).
 
Network latencies and lag compensation can create paradoxes that seem illogical compared to the real world. For example, you can be hit by an attacker you can't even see anymore because you already took cover. What happened is that the server moved your player hitboxes back in time, where you were still exposed to your attacker. This inconsistency problem can't be solved in general because of the relatively slow packet speeds. In the real world, you don't notice this problem because light (the packets) travels so fast and you and everybody around you sees the same world as it is right now.


== Net graph ==
== Net graph ==
The Source engine offers a couple of tools to check your client connection speed and quality. The most popular one is the net graph, which can be enabled with <code>net_graph 2</code>. Incoming packets are represented by small lines moving from right to left. The height of each line reflects size of a packet. If a gap appears between lines, a packet was lost or arrived out of order. The lines are color-coded depending on what kind of data they contain.


Under the net graph, the first line shows your current rendered frames per second, your average latency, and the current value of <code>cl_updaterate</code>. The second line shows the size in bytes of the last incoming packet (snapshots), the average incoming bandwidth, and received packets per second. The third line shows the same data just for outgoing packets (user commands).
The Source engine offers a couple of tools to check your client connection speed and quality. The most popular one is the net graph, which can be enabled with {{code|<nowiki>net_graph 2</nowiki>}} (or {{code|<nowiki>+graph</nowiki>}}). Incoming packets are represented by small lines moving from right to left. The height of each line reflects size of a packet. If a gap appears between lines, a packet was lost or arrived out of order. The lines are color-coded depending on what kind of data they contain.
 
Under the net graph, the first line shows your current rendered frames per second, your average latency, and the current value of {{code|<nowiki>cl_updaterate</nowiki>}}. The second line shows the size in bytes of the last incoming packet (snapshots), the average incoming bandwidth, and received packets per second. The third line shows the same data just for outgoing packets (user commands).


[[Image:Net graph2.jpg|center|net_graph output]]
[[File:Net graph.jpg|center|net_graph output]]


== Optimizations ==
== Optimizations ==
The default networking settings are designed for playing on dedicated server on the Internet. The settings are balanced to work well for most client/server hardware and network configurations. For Internet games the only console variable that should be adjusted on the client is "rate", which defines your available bytes/second bandwidth of your network connection. Good values for "rate" is 4500 for modems, 6000 for ISDN, 10000 DSL and above.  
The default networking settings are designed for playing on dedicated server on the Internet. The settings are balanced to work well for most client/server hardware and network configurations. For Internet games the only console variable that should be adjusted on the client is "rate", which defines your available bytes/second bandwidth of your network connection. Good values for "rate" is 4500 for modems, 6000 for ISDN, 10000 DSL and above.  


In an high-performance network environment, where the server and all clients have the necessary hardware resources available, it's possible to tweak bandwidth and tickrate settings to gain more gameplay precision. Increasing the server tickrate gernally improves movement and shooting precision but comes with a higher CPU cost. A CS:S server running with tickrate 100 generates about 3x more CPU load then a default tickrate 33 server. That can cause serious calculation lags, especially when lots of people are shooting at the same time. It's not suggested to run a game server with a higher tickrate then 66 to reserve necessary CPU resources for critical situations.
In an high-performance network environment, where the server and all clients have the necessary hardware resources available, it's possible to tweak bandwidth and tickrate settings to gain more gameplay precision. Increasing the server tickrate generally improves movement and shooting precision but comes with a higher CPU cost. A Source server running with tickrate 100 generates about 1.5x more CPU load than a default tickrate 66. That can cause serious calculation lags, especially when lots of people are shooting at the same time. It's not suggested to run a game server with a higher tickrate than 66 to reserve necessary CPU resources for critical situations.
 
{{note|It is not possible to change tickrate on CSS, DoD S TF2, L4D and L4D2 because changing tickrate causes server timing issues. The tickrate is set to 66 in CSS, DoD S and TF2, 60 in HL, CS 1.6 and most GoldSrc games, and 30 in L4D, L4D2 and TFC.<br> Changing tickrate is however, possible with older version of those games.}}


If the game server is running with a higher tickrate, clients can increase their snapshot update rate (cl_updaterate) and user command rate (cl_cmdrate), if the necessary bandwidth (rate) is available. The snapshot udapte rate is limited by the server tickrate, a server can't send more then one update per tick. So for a tickrate 66 server, the highest client value for cl_updaterate would be 66. If you increase the snapshot rate and encounter packet loss or choke, you have to turn it down again. With an increased cl_snapshotrate you can also lower the view interpolation delay (cl_interp). The default interpolation delay is 0.1 seconds, which derives from the default cl_updaterate 20. View interpolation delay gives a moving player a small advantage over a stationary player since the moving player can see his target a split second earlier. This effect is unavoidable, but it can be reduced by decreasing the view interpolation delay. If both players are moving, the view lag delay is affecting both players and nobody has an advantage.  
If the game server is running with a higher tickrate, clients can increase their snapshot update rate (cl_updaterate) and user command rate (cl_cmdrate), if the necessary bandwidth (rate) is available. The snapshot update rate is limited by the server tickrate, a server can't send more then one update per tick. So for a tickrate 66 server, the highest client value for cl_updaterate would be 66. If you increase the snapshot rate and encounter packet loss or choke, you have to turn it down again. With an increased cl_updaterate you can also lower the view interpolation delay (cl_interp). The default interpolation delay is 0.1 seconds, which derives from the default cl_updaterate 20. View interpolation delay gives a moving player a small advantage over a stationary player since the moving player can see his target a split second earlier. This effect is unavoidable, but it can be reduced by decreasing the view interpolation delay. If both players are moving, the view lag delay is affecting both players and nobody has an advantage.  


The relation between snapshot rate and view interpolation delay is the following:
This is the relation between snapshot rate and view interpolation delay is the following:


cl_interp = 2 / cl_updaterate
<span style="text-align:center; font-weight:bold; font-family:monospace;display:block;font-size:1.2em;">interpolation period = max( cl_interp, cl_interp_ratio / cl_updaterate )</span>


For example, if your client receives 66 update per second, you can set cl_interp to 0.03. This way your view interpolation lag is lowered from 100ms to 30ms. Removing the view interpolation lag completely by turning it off (cl_interpolate 0) will not work and cause jittery animations and worse hit detection.
"Max(x,y)" means "whichever of these is higher". You can set {{code|<nowiki>cl_interp</nowiki>}} to 0 and still have a safe amount of interp. You can then increase cl_updaterate to decrease your interp period further, but don't exceed tickrate (66) or flood your connection with more data than it can handle.


Here some more tips:
===Tips===


* Don't change console settings unless you are 100% sure what you are doing. Most "high-performance" setting cause exactly the opposite effect, if the server or network can't handle the load.
;Don't change console settings unless you are 100% sure what you are doing:Most "high-performance" setting cause exactly the opposite effect, if the server or network can't handle the load.
* Don't turn off view interpolation and/or lag compensation, it will not improve movement or shooting precision.
;Don't turn off view interpolation and/or lag compensation:It will not improve movement or shooting precision.
* Optimized setting for one client may not work for other clients. Do not just use settings from other clients without verifing them for your system.
;Optimized setting for one client may not work for other clients:Do not just use settings from other clients without verifing them for your system.
* If you follow a player in "First-Person" as a spectator in a game or [[SourceTV]], you don't exactly see what the player sees. Spectators see the game world without lag compensation.
* All implementation details for lag compensation and view interpolation are part of the public Source SDK.


== See also ==
== See also ==
* [[Networking Entities]]
 
* [[Lag Compensation]]
* [[Prediction]], [[Interpolation]], [[Lag compensation]]
* [[Latency Compensating Methods in Client/Server In-game Protocol Design and Optimization]]
* [[TF2 Network Graph]]
* [[TF2 Network Graph]]
* [[Networking Entities]] (for programmers)
* [[Wikipedia:Lag|Lag on Wikipedia]]
{{source topicon}}<!-- Source category are automatically added by this template.-->


{{otherlang:en}} {{otherlang:en:jp|Source Multiplayer Networking:jp}}, {{otherlang:en:ru|Source Multiplayer Networking:ru}}
[[Category:Networking]]
[[Category:Dedicated Server]]


[[Category:Technical]]
[[Category:Counter-Strike: Global Offensive]]
[[Category:Half-Life 2: Deathmatch]]

Latest revision as of 04:32, 9 June 2025

English (en)Español (es)日本語 (ja)Русский (ru)中文 (zh)Translate (Translate)

Multiplayer games based on the Source Source use a Client-Server networking architecture. Usually a server is a dedicated host that runs the game and is authoritative about world simulation, game rules, and player input processing. A client is a player's computer connected to a game server. The client and server communicate with each other by sending small data packets at a high frequency (usually 20 to 30 packets per second). A client receives the current world state from the server and generates video and audio output based on these updates. The client also samples data from input devices (keyboard, mouse, microphone, etc.) and sends these input samples back to the server for further processing. Clients only communicate with the game server and not between each other (like in a peer-to-peer application). In contrast with a single player game, a multiplayer game has to deal with a variety of new problems caused by packet-based communication.

Network bandwidth is limited, so the server can't send a new update packet to all clients for every single world change. Instead, the server takes snapshots of the current world state at a constant rate and broadcasts these snapshots to the clients. Network packets take a certain amount of time to travel between the client and the server (i.e. half the ping time). This means that the client time is always a little bit behind the server time. Furthermore, client input packets are also delayed on their way back, so the server is processing temporally delayed user commands. In addition, each client has a different network delay which varies over time due to other background traffic and the client's framerate. These time differences between server and client causes logical problems, becoming worse with increasing network latencies. In fast-paced action games, even a delay of a few milliseconds can cause a laggy gameplay feeling and make it hard to hit other players or interact with moving objects. Besides bandwidth limitations and network latencies, information can get lost due to network packet loss.

Time differences between client and server

To cope with the issues introduced by network communication, the Source engine server employs techniques such as data compression and lag compensation which are invisible to the client. The client then performs prediction and interpolation to further improve the experience.

Note.pngNote:If you follow a player in "First-Person" as a spectator in a game or SourceTV, you don't exactly see what the player sees. Spectators see the game world without lag compensation.

Basic networking

The server simulates the game in discrete time steps called ticks. By default, the timestep is 15ms, so 66.666... ticks per second are simulated, but mods can specify their own tickrate. During each tick, the server processes incoming user commands, runs a physical simulation step, checks the game rules, and updates all object states. After simulating a tick, the server decides if any client needs a world update and takes a snapshot of the current world state if necessary. A higher tickrate increases the simulation precision, but also requires more CPU power and available bandwidth on both server and client. The server admin may override the default tickrate with the -tickrate command line parameter, though tickrate changes done this way are not recommended because the mod may not work as designed if its tickrate is changed.

Note.pngNote:The -tickrate command line parameter is not available newer version of CS:S, DoD:S, TF2, L4D and L4D2, and was disabled in code likely because changing tickrate may cause causes server timing issues for certain games.
The tickrate is set to 66 in CSS, DoD S and TF2. 60 in CS 1.6 and HL1, and 30 in L4D, L4D2 and TFC.
Some older version of those games can have it's tickrate changed, however. Users can also install this server plugins ↓ to restore the disabled -tickrate command line parameter.

Clients usually have only a limited amount of available bandwidth. In the worst case, players with a modem connection can't receive more than 5 to 7 KB/sec. If the server tried to send them updates with a higher data rate, packet loss would be unavoidable. Therefore, the client has to tell the server its incoming bandwidth capacity by setting the console variable rate (in bytes/second). This is the most important network variable for clients and it has to be set correctly for an optimal gameplay experience. The client can request a certain snapshot rate by changing cl_updaterate (default 20), but the server will never send more updates than simulated ticks or exceed the requested client rate limit. Server admins can limit data rate values requested by clients with sv_minrate and sv_maxrate (both in bytes/second). Also the snapshot rate can be restricted with sv_minupdaterate and sv_maxupdaterate (both in snapshots/second).

The client creates user commands from sampling input devices with the same tick rate that the server is running with. A user command is basically a snapshot of the current keyboard and mouse state. But instead of sending a new packet to the server for each user command, the client sends command packets at a certain rate of packets per second (usually 30). This means two or more user commands are transmitted within the same packet. Clients can increase the command rate with cl_cmdrate. This will increase responsiveness but requires more outgoing bandwidth, too.

Game data is compressed using delta compression to reduce network load. That means the server doesn't send a full world snapshot each time, but rather only changes (a delta snapshot) that happened since the last acknowledged update. With each packet sent between the client and server, acknowledge numbers are attached to keep track of their data flow. Usually full (non-delta) snapshots are only sent when a game starts or a client suffers from heavy packet loss for a couple of seconds. Clients can request a full snapshot manually with the cl_fullupdate command.

Responsiveness, or the time between user input and its visible feedback in the game world, are determined by lots of factors, including the server/client CPU load, simulation tickrate, data rate and snapshot update settings, but mostly by the network packet traveling time. The time between the client sending a user command, the server responding to it, and the client receiving the server's response is called the latency or ping (or round trip time). Low latency is a significant advantage when playing a multiplayer online game. Techniques like prediction and lag compensation try to minimize that advantage and allow a fair game for players with slower connections. Tweaking networking setting can help to gain a better experience if the necessary bandwidth and CPU power is available. We recommend keeping the default settings, since improper changes may cause more negative side effects than actual benefits.

Servers for these games that support altering Tickrate

Note.pngNote:Most singleplayer games such as Half-Life 2 Half-Life 2 can have it's tickrate changed.
Clarify: what about Counter-Strike Counter-Strike?, Half-Life Half-Life seems to use 60 or something according to "+graph", it is unknown if this can be altered.
Clarify: How about other Source 2 Source 2.

The tickrate can be altered by using the -tickrate parameter for these games:

Source

By default, without using server plugins ↓, some servers tickrate cannot be altered for these games due to various reasons such as GoldSrc GoldSrc lacking the -tickrate command line parameter, or intentionally disabled in-code by Valve due to "server ops are abusing it" and/or "using -tickrate may cause server timing-related issues":

GoldSrc

60 Tickrate

30 Tickrate

Source

66 Tickrate

30 Tickrate

Source 2

64 Tickrate, with Sub-tick

  • Counter-Strike 2 Counter-Strike 2
    • Note.pngNote:Originally can be changed during Limited Test but later was hardcoded to 64-tick.
Confirm:Does the -tickrate command (or similar) exists or works with GoldSrc games?
Note.pngNote:In previous versions on some of those games, it can be changed. Alternatively user can also use server plugins ↓.
Note.pngNote:The lowest or highest values depends on the game. Minimum tickrate is 11.

Using server plugins to restore -tickrate for games that have it disabled

Users can install this server plugins to make the -tickrate command works again on games that have it disabled. Note that loading plugins would require -insecure (disabling VAC) for it to work.
All Source games / L4D & L4D2

Steam Networking

Steam Networking is a new feature which allows self-hosting using Steam Datagram Relay, meaning users can easily create joinable servers without port forwarding. User can enable/disable Steam Networking through Create Server menu, or through console command: sv_use_steam_networking.

The following games supports Steam Networking:

GoldSrc

Added since Half-Life 20th Anniversary Update.

Source

Team Fortress 2 branch games

Added since February 18, 2025 update for existing games running on TF2 branch, while other games running on Source 2013 Multiplayer have been upgraded to TF2 branch on the same day, adding Steam Networking support.

Source 2013 Multiplayer games

Source 2

[Todo]

Entity interpolation

Main article:  Interpolation

By default, the client receives about 20 snapshot per second. If the objects (entities) in the world were only rendered at the positions received by the server, moving objects and animation would look choppy and jittery. Dropped packets would also cause noticeable glitches. The trick to solve this problem is to go back in time for rendering, so positions and animations can be continuously interpolated between two recently received snapshots. With 20 snapshots per second, a new update arrives about every 50 milliseconds. If the client render time is shifted back by 50 milliseconds, entities can be always interpolated between the last received snapshot and the snapshot before that.

Source defaults to an interpolation period ('lerp') of 100-milliseconds (cl_interp 0.1); this way, even if one snapshot is lost, there are always two valid snapshots to interpolate between. Take a look at the following figure showing the arrival times of incoming world snapshots:

Interpolation timeline

The last snapshot received on the client was at tick 344 or 10.30 seconds. The client time continues to increase based on this snapshot and the client frame rate. If a new video frame is rendered, the rendering time is the current client time 10.32 minus the view interpolation delay of 0.1 seconds. This would be 10.22 in our example and all entities and their animations are interpolated using the correct fraction between snapshot 340 and 342.

Since we have an interpolation delay of 100 milliseconds, the interpolation would even work if snapshot 342 were missing due to packet loss. Then the interpolation could use snapshots 340 and 344. If more than one snapshot in a row is dropped, interpolation can't work perfectly because it runs out of snapshots in the history buffer. In that case the renderer uses extrapolation (cl_extrapolate 1) and tries a simple linear extrapolation of entities based on their known history so far. The extrapolation is done only for 0.25 seconds of packet loss (cl_extrapolate_amount), since the prediction errors would become too big after that.

Entity interpolation causes a constant view "lag" of 100 milliseconds by default (cl_interp 0.1), even if you're playing on a listenserver (server and client on the same machine). This doesn't mean you have to lead your aiming when shooting at other players since the server-side lag compensation knows about client entity interpolation and corrects this error.

Tip.pngTip:More recent Source games have the cl_interp_ratio cvar. With this you can easily and safely decrease the interpolation period by setting cl_interp to 0, then increasing the value of cl_updaterate (the useful limit of which depends on server tickrate). You can check your final lerp with net_graph 1.
Note.pngNote:If you turn on sv_showhitboxes (not available in Source 2009) you will see player hitboxes drawn in server time, meaning they are ahead of the rendered player model by the lerp period. This is perfectly normal!

Input prediction

Main article:  Prediction

Lets assume a player has a network latency of 150 milliseconds and starts to move forward. The information that the +FORWARD key is pressed is stored in a user command and send to the server. There the user command is processed by the movement code and the player's character is moved forward in the game world. This world state change is transmitted to all clients with the next snapshot update. So the player would see his own change of movement with a 150 milliseconds delay after he started walking. This delay applies to all players actions like movement, shooting weapons, etc. and becomes worse with higher latencies.

A delay between player input and corresponding visual feedback creates a strange, unnatural feeling and makes it hard to move or aim precisely. Client-side input prediction (cl_predict 1) is a way to remove this delay and let the player's actions feel more instant. Instead of waiting for the server to update your own position, the local client just predicts the results of its own user commands. Therefore, the client runs exactly the same code and rules the server will use to process the user commands. After the prediction is finished, the local player will move instantly to the new location while the server still sees him at the old place.

After 150 milliseconds, the client will receive the server snapshot that contains the changes based on the user command he predicted earlier. Then the client compares the server position with his predicted position. If they are different, a prediction error has occurred. This indicates that the client didn't have the correct information about other entities and the environment when it processed the user command. Then the client has to correct its own position, since the server has final authority over client-side prediction. If cl_showerror 1 is turned on, clients can see when prediction errors happen. Prediction error correction can be quite noticeable and may cause the client's view to jump erratically. By gradually correcting this error over a short amount of time (cl_smoothtime), errors can be smoothly corrected. Prediction error smoothing can be turned off with cl_smooth 0.

Prediction is only possible for the local player and entities affected only by him, since prediction works by using the client's keypresses to make a "best guess" of where the player will end up. Predicting other players would require literally predicting the future with no data, since there's no way to instantaneously get keypresses from them.

Lag compensation

All source code for lag compensation and view interpolation is available in the Source SDK. See Lag compensation for implementation details.

Let's say a player shoots at a target at client time 10.5. The firing information is packed into a user command and sent to the server. While the packet is on its way through the network, the server continues to simulate the world, and the target might have moved to a different position. The user command arrives at server time 10.6 and the server wouldn't detect the hit, even though the player has aimed exactly at the target. This error is corrected by the server-side lag compensation.

The lag compensation system keeps a history of all recent player positions for one second. If a user command is executed, the server estimates at what time the command was created as follows:

Command Execution Time = Current Server Time - Packet Latency - Client View Interpolation

Then the server moves all other players - only players - back to where they were at the command execution time. The user command is executed and the hit is detected correctly. After the user command has been processed, the players revert to their original positions.

Note.pngNote:Since entity interpolation is included in the equation, failing to have it on can cause undesired results.

On a listen server you can enable sv_showimpacts 1 to see the different server and client hitboxes:

Visualisation of lag compensation: server and client hitboxes

This screenshot was taken on a listen server with 200 milliseconds of lag (using net_fakelag), right after the server confirmed the hit. The red hitbox shows the target position on the client where it was 100ms + interp period ago. Since then, the target continued to move to the left while the user command was travelling to the server. After the user command arrived, the server restored the target position (blue hitbox) based on the estimated command execution time. The server traces the shot and confirms the hit (the client sees blood effects).

Client and server hitboxes don't exactly match because of small precision errors in time measurement. Even a small difference of a few milliseconds can cause an error of several inches for fast-moving objects. Multiplayer hit detection is not pixel perfect and has known precision limitations based on the tickrate and the speed of moving objects.

The question arises, why is hit detection so complicated on the server? Doing the back tracking of player positions and dealing with precision errors while hit detection could be done client-side way easier and with pixel precision. The client would just tell the server with a "hit" message what player has been hit and where. We can't allow that simply because a game server can't trust the clients on such important decisions. Even if the client is "clean" and protected by VAC Valve Anti-Cheat, the packets could be still modified on a 3rd machine while routed to the game server. These "cheat proxies" could inject "hit" messages into the network packet without being detected by VAC (a "man-in-the-middle" attack).

Network latencies and lag compensation can create paradoxes that seem illogical compared to the real world. For example, you can be hit by an attacker you can't even see anymore because you already took cover. What happened is that the server moved your player hitboxes back in time, where you were still exposed to your attacker. This inconsistency problem can't be solved in general because of the relatively slow packet speeds. In the real world, you don't notice this problem because light (the packets) travels so fast and you and everybody around you sees the same world as it is right now.

Net graph

The Source engine offers a couple of tools to check your client connection speed and quality. The most popular one is the net graph, which can be enabled with net_graph 2 (or +graph). Incoming packets are represented by small lines moving from right to left. The height of each line reflects size of a packet. If a gap appears between lines, a packet was lost or arrived out of order. The lines are color-coded depending on what kind of data they contain.

Under the net graph, the first line shows your current rendered frames per second, your average latency, and the current value of cl_updaterate. The second line shows the size in bytes of the last incoming packet (snapshots), the average incoming bandwidth, and received packets per second. The third line shows the same data just for outgoing packets (user commands).

net_graph output

Optimizations

The default networking settings are designed for playing on dedicated server on the Internet. The settings are balanced to work well for most client/server hardware and network configurations. For Internet games the only console variable that should be adjusted on the client is "rate", which defines your available bytes/second bandwidth of your network connection. Good values for "rate" is 4500 for modems, 6000 for ISDN, 10000 DSL and above.

In an high-performance network environment, where the server and all clients have the necessary hardware resources available, it's possible to tweak bandwidth and tickrate settings to gain more gameplay precision. Increasing the server tickrate generally improves movement and shooting precision but comes with a higher CPU cost. A Source server running with tickrate 100 generates about 1.5x more CPU load than a default tickrate 66. That can cause serious calculation lags, especially when lots of people are shooting at the same time. It's not suggested to run a game server with a higher tickrate than 66 to reserve necessary CPU resources for critical situations.

Note.pngNote:It is not possible to change tickrate on CSS, DoD S TF2, L4D and L4D2 because changing tickrate causes server timing issues. The tickrate is set to 66 in CSS, DoD S and TF2, 60 in HL, CS 1.6 and most GoldSrc games, and 30 in L4D, L4D2 and TFC.
Changing tickrate is however, possible with older version of those games.

If the game server is running with a higher tickrate, clients can increase their snapshot update rate (cl_updaterate) and user command rate (cl_cmdrate), if the necessary bandwidth (rate) is available. The snapshot update rate is limited by the server tickrate, a server can't send more then one update per tick. So for a tickrate 66 server, the highest client value for cl_updaterate would be 66. If you increase the snapshot rate and encounter packet loss or choke, you have to turn it down again. With an increased cl_updaterate you can also lower the view interpolation delay (cl_interp). The default interpolation delay is 0.1 seconds, which derives from the default cl_updaterate 20. View interpolation delay gives a moving player a small advantage over a stationary player since the moving player can see his target a split second earlier. This effect is unavoidable, but it can be reduced by decreasing the view interpolation delay. If both players are moving, the view lag delay is affecting both players and nobody has an advantage.

This is the relation between snapshot rate and view interpolation delay is the following:

interpolation period = max( cl_interp, cl_interp_ratio / cl_updaterate )

"Max(x,y)" means "whichever of these is higher". You can set cl_interp to 0 and still have a safe amount of interp. You can then increase cl_updaterate to decrease your interp period further, but don't exceed tickrate (66) or flood your connection with more data than it can handle.

Tips

Don't change console settings unless you are 100% sure what you are doing
Most "high-performance" setting cause exactly the opposite effect, if the server or network can't handle the load.
Don't turn off view interpolation and/or lag compensation
It will not improve movement or shooting precision.
Optimized setting for one client may not work for other clients
Do not just use settings from other clients without verifing them for your system.

See also