Edict t: Difference between revisions
		
		
		
		
		
		Jump to navigation
		Jump to search
		
				
		
 
 Note:An edict is not reliable between frames, as it can be freed from one entity and and reallocated to another at any time. Use
Note:An edict is not reliable between frames, as it can be freed from one entity and and reallocated to another at any time. Use 
 Warning:Never modify
Warning:Never modify 
 Note:Entities which are found in the entity lump but are not recognized by the server do not create edicts. This will commonly be encountered with info_ladder in modern versions of
Note:Entities which are found in the entity lump but are not recognized by the server do not create edicts. This will commonly be encountered with info_ladder in modern versions of  Counter-Strike: Source and info_texlights in ZHLT.
 Counter-Strike: Source and info_texlights in ZHLT.
 Warning:In most games,
Warning:In most games,  Source can have up to 2048 edicts allocated at any one time. Going over will cause the engine to crash with an error message.
 Source can have up to 2048 edicts allocated at any one time. Going over will cause the engine to crash with an error message.
 Tip:
Tip: GoldSrc has a maximum of 1200 edicts by default (900 prior to 25th-anniversary update), but this can be raised to 2048 in a mod by editing its liblist.gam to include the line
 GoldSrc has a maximum of 1200 edicts by default (900 prior to 25th-anniversary update), but this can be raised to 2048 in a mod by editing its liblist.gam to include the line  Confirm:What about
 Confirm:What about  Sven Co-op?
 Sven Co-op? 
 Tip:
Tip:
		
	
| Dave Gomboc (talk | contribs) mNo edit summary | SirYodaJedi (talk | contribs)  | ||
| (31 intermediate revisions by 11 users not shown) | |||
| Line 1: | Line 1: | ||
| {{ | {{toc-right}} {{DISPLAYTITLE:edict_t}} | ||
| {{ | |||
| '''<code>edict_t</code>''' ("entity dictionary") is an [[interface]] [[struct]] that allows [[entity|entities]] to cross the client/server divide: with one attached, an entity has the same [[entity index|index]] at both ends. The edict also manages the state of the entity's [[DataTable]] and provides a common representation across all DLLs. It cannot be used on the client. | |||
| {{note|An edict is not reliable between frames, as it can be freed from one entity and and reallocated to another at any time. Use <code>[[CHandle]]</code> for long-term storage.}} | |||
| {{warning|Never modify <code>edict_t</code> or <code>CBaseEdict</code>. If the memory footprint of either changes, engine DLLs will start accessing invalid addresses.}} | |||
| == | {{note|Entities which are found in the entity lump but are not recognized by the server do not create edicts. This will commonly be encountered with {{ent|info_ladder}} in modern versions of {{css|2}} and {{ent|info_texlights|engine=goldsrc}} in [[ZHLT]].}} | ||
| [[ | |||
| [[Category: | ==Limitations== | ||
| {{warning|In most games, {{src|4.1}} can have [[entity limit|up to 2048]] edicts allocated at any one time. Going over will cause the engine to crash with an error message.}} | |||
| {{tip|{{gldsrc|4.1}} has a maximum of 1200 edicts by default (900 prior to 25th-anniversary update), but this can be raised to 2048 in a mod by editing its [[liblist.gam]] to include the line {{code|edicts{{nbsp}}"2048"}}.{{modernConfirm|What about {{sven|2}}?}} }} | |||
| ==Avoiding== | |||
| ===Server only entities=== | |||
| {{src|only}}<br> | |||
| Edicts are assigned by <code>[[CBaseEntity]]::PostConstructor()</code>. If your entity won't ever need to be transmitted to the client, [[Constructor|construct]] it like so: | |||
| <source lang=cpp>CMyEntity() : CBaseEntity(true) { /* your constructor code */ }</source> | |||
| {{tip|<code>CServerOnlyEntity</code> already does this. <code>[[CLogicalEntity]]</code> inherits, and as such none of Valve's [[Special:PrefixIndex/logic|logic entities]] or [[:Category:Constraints|VPhysics constraints]] have edicts.}} | |||
| Edict-less entities count towards the [[Entity_limit|non-networked entity limit]], which is the same or higher than the edict limit. | |||
| Entities compiled into the map which are not recognized by the game code do not count towards any entity limit in-game as they are simply not spawned. | |||
| ===Client-only entities=== | |||
| {{todo|If the game is not singleplayer, and the entity is non-solid and is not part of {{cmd|s_PreserveEnts}}, it can be removed server-side. See [[env_sprite_clientside]] source code in Alien Swarm for example implementation in Source. In GoldSrc, use [[MAKE_STATIC]].}} | |||
| ==Getting== | |||
| ; <code>edict_t* CBaseEntity::edict()</code> | |||
| : Simple accessor. Returns null if there is no edict. | |||
| ; <code>edict_t* INDEXENT( int iEdictNum )</code> | |||
| ; <code>int ENTINDEX( edict_t* pEdict )</code> | |||
| : Converts between an [[entity index]] and its attached edict. | |||
| ; <code>int GetEntityCount()</code> | |||
| : Returns the current number of assigned edicts (''not'' entities). | |||
| ==Setting== | |||
| ; <code>void [[CBaseEntity]]::NetworkProp()->AttachEdict(edict_t* pRequiredEdict);</code> | |||
| : Assigns an edict via the entity's <code>[[CServerNetworkProperty]]</code> object. Pass NULL unless a specific edict is required. | |||
| ; <code>void CBaseEntity::NetworkProp()->DetachEdict()</code> | |||
| : Detaches the entity's assigned edict. It will go back into the edict pool for re-use. | |||
| ==Notable members== | |||
| ; <code>[[CBaseEntity]]* GetUnknown()->GetBaseEntity()</code> | |||
| : Pointer to the entity's local representation. | |||
| ; <code>char* GetClassName()</code> | |||
| : The [[classname]] of the assigned entity. | |||
| ; <code>[[bool]] IsFree()</code> | |||
| : Whether there is currently an entity assigned to this edict. | |||
| ; <code>[[float]] freetime</code> | |||
| : The server timestamp at which the edict was last freed. Can be used to avoid reallocating the edict immediately. | |||
| ; <code>bool HasStateChanged()</code> | |||
| : Whether anything in the [[DataTable]] needs transmission. | |||
| ; <code>void StateChanged()</code> | |||
| : Tell the edict of a change to the entity's DataTable that needs transmission. | |||
| ; <code>[[IServerEntity]]* GetIServerEntity()</code> | |||
| ; <code>[[IServerNetworkable]]* GetNetworkable()</code> | |||
| ; <code>[[ICollideable]]* GetCollideable()</code> | |||
| : Accessors to various interfaces. | |||
| ==See also== | |||
| * [[Entity index]] | |||
| * [[Entity limit]] | |||
| * <code>[[CHandle]]</code> | |||
| * [http://www.mail-archive.com/hlcoders@list.valvesoftware.com/msg01215.html Discussion on GoldSrc edicts] | |||
| * [http://www.quake2.com/dll/doc/ent_atr.html Quake 2's edict_t] | |||
| [[Category:Interfaces]] | |||
Latest revision as of 11:18, 23 April 2025
edict_t ("entity dictionary") is an interface struct that allows entities to cross the client/server divide: with one attached, an entity has the same index at both ends. The edict also manages the state of the entity's DataTable and provides a common representation across all DLLs. It cannot be used on the client.
 Note:An edict is not reliable between frames, as it can be freed from one entity and and reallocated to another at any time. Use
Note:An edict is not reliable between frames, as it can be freed from one entity and and reallocated to another at any time. Use CHandle for long-term storage. Warning:Never modify
Warning:Never modify edict_t or CBaseEdict. If the memory footprint of either changes, engine DLLs will start accessing invalid addresses. Note:Entities which are found in the entity lump but are not recognized by the server do not create edicts. This will commonly be encountered with info_ladder in modern versions of
Note:Entities which are found in the entity lump but are not recognized by the server do not create edicts. This will commonly be encountered with info_ladder in modern versions of  Counter-Strike: Source and info_texlights in ZHLT.
 Counter-Strike: Source and info_texlights in ZHLT.Limitations
 Warning:In most games,
Warning:In most games,  Source can have up to 2048 edicts allocated at any one time. Going over will cause the engine to crash with an error message.
 Source can have up to 2048 edicts allocated at any one time. Going over will cause the engine to crash with an error message. Tip:
Tip: GoldSrc has a maximum of 1200 edicts by default (900 prior to 25th-anniversary update), but this can be raised to 2048 in a mod by editing its liblist.gam to include the line
 GoldSrc has a maximum of 1200 edicts by default (900 prior to 25th-anniversary update), but this can be raised to 2048 in a mod by editing its liblist.gam to include the line edicts "2048". Confirm:What about
 Confirm:What about  Sven Co-op?
 Sven Co-op?Avoiding
Server only entities
(only in  )
)
Edicts are assigned by CBaseEntity::PostConstructor(). If your entity won't ever need to be transmitted to the client, construct it like so:
CMyEntity() : CBaseEntity(true) { /* your constructor code */ }
 Tip:
Tip:CServerOnlyEntity already does this. CLogicalEntity inherits, and as such none of Valve's logic entities or VPhysics constraints have edicts.Edict-less entities count towards the non-networked entity limit, which is the same or higher than the edict limit.
Entities compiled into the map which are not recognized by the game code do not count towards any entity limit in-game as they are simply not spawned.
Client-only entities
Todo: If the game is not singleplayer, and the entity is non-solid and is not part of 
s_PreserveEnts, it can be removed server-side. See env_sprite_clientside source code in Alien Swarm for example implementation in Source. In GoldSrc, use MAKE_STATIC.Getting
- edict_t* CBaseEntity::edict()
- Simple accessor. Returns null if there is no edict.
- edict_t* INDEXENT( int iEdictNum )
- int ENTINDEX( edict_t* pEdict )
- Converts between an entity index and its attached edict.
- int GetEntityCount()
- Returns the current number of assigned edicts (not entities).
Setting
- void CBaseEntity::NetworkProp()->AttachEdict(edict_t* pRequiredEdict);
- Assigns an edict via the entity's CServerNetworkPropertyobject. Pass NULL unless a specific edict is required.
- void CBaseEntity::NetworkProp()->DetachEdict()
- Detaches the entity's assigned edict. It will go back into the edict pool for re-use.
Notable members
- CBaseEntity* GetUnknown()->GetBaseEntity()
- Pointer to the entity's local representation.
- char* GetClassName()
- The classname of the assigned entity.
- bool IsFree()
- Whether there is currently an entity assigned to this edict.
- float freetime
- The server timestamp at which the edict was last freed. Can be used to avoid reallocating the edict immediately.
- bool HasStateChanged()
- Whether anything in the DataTable needs transmission.
- void StateChanged()
- Tell the edict of a change to the entity's DataTable that needs transmission.
- IServerEntity* GetIServerEntity()
- IServerNetworkable* GetNetworkable()
- ICollideable* GetCollideable()
- Accessors to various interfaces.