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

BSP (Source): Difference between revisions

From Valve Developer Community
Jump to navigation Jump to search
(→‎BSP Versions: Ok, it uses 21. But I'm pretty sure the header is wrong...)
mNo edit summary
 
(367 intermediate revisions by 66 users not shown)
Line 1: Line 1:
''This article is based on [[User:Rof|Rof's]] [http://www.geocities.com/cofrdrbob/bspformat.html "The Source Engine BSP File Format" from October 2005], retrieved from Geocities before the service shut down. A mirror to the original version can be found [http://www.bagthorpe.org/bob/cofrdrbob/bspformat.html here].''
{{LanguageBar}}
{{Source topicon}}{{Subpage|[[BSP]]}}
{{TabsBar|main=BSP}}


{{toc-right}}
{{Inline note|''This article is based on [[User:Rof|Rof's]] {{Wayback|0/http://www.geocities.com/cofrdrbob/bspformat.html|The Source Engine BSP File Format}} (from October 2005), retrieved from Geocities before the service shut down. A mirror to the original version can be found [http://www.bagthorpe.org/bob/cofrdrbob/bspformat.html here].''}}{{Toc-right}}
{{Todo|Some versions of [[BSPZIP]] {{In|{{tf2}} {{Counter-Strike: Source}} {{Day of Defeat: Source}} {{hl2dm}} {{hldms}} {{strata}} {{jbep3}}|text=included in}} support LZMA compression of the lumps and packed files {{in|{{src13mp}} {{tf2branch}} {{strata}} {{jbep3}} and possibly more|text=supported by}}; update this page to reflect how this affects the format.}}


== Introduction ==
== Introduction ==
This document describes the structure of the [[BSP]] file format used by the Source engine. The format is more closely derived from [[BSP38|the BSP format used in Quake II]] than [[BSP30|that of GoldSrc]] (both of which are derived from [[BSP29|Quake I's BSP format]]). Because of this, Max McGuire's article, [http://www.flipcode.com/archives/Quake_2_BSP_File_Format.shtml Quake 2 BSP File Format] is also of invaluable help in understanding the overall structure of the format and the parts of it that have remained the same or similar to its predecessors.


This document describes the structure of the [[BSP]] file format used by the Source engine. The format is similar but not identical to the BSP file formats of the Half-Life 1 engine, which is in turn based on the Quake 1 and Quake 2 file formats, plus that of the later Quake 3:Arena. Because of this, Max McGuire's article, [http://flipcode.com/articles/article_q2bsp.shtml Quake 2 BSP File Format] has been of invaluable help in understanding the overall structure of the format and the parts of it that have remained the same or similar to its predecessors.
This document is an extension of notes made by Rof during the writing of their {{hl2|4}} BSP file decompiler, [[VMEX]]. It therefore focuses on those parts of the format necessary to perform [[Decompiling Maps|map decompilation]] (conversion of the BSP file back into a [[Valve Map Format|VMF]] file which can be loaded by the Hammer map editor).


This document is an extension of notes made during the writing of my Half-Life 2 bsp file decompiler, [[VMEX]]. It therefore focusses on those parts of the format necessary to perform map decompilation (conversion of the bsp file back into a [[VMF]] file which can be loaded by the Hammer map editor). Some parts of the format are not needed for this process, but what information I have about these sections will be mentioned.
Most of the information in this document comes from the Max McGuire article referenced above, from the source code included in the Source SDK (particularly the C header file ''public/bspfile.h''), and from Rof's own experimentation during the writing of VMEX.


Most of the information in this document comes from the Max McGuire article referenced above, from the source code included in the Source SDK (particularly the C header file ''public/bspfile.h''), and from my own experimentation during the writing of VMEX. This document is completely unofficial and should not be considered any kind of official specification from Valve, nor am I affiliated with Valve in any way. Any corrections or information on the unknown parts of the format will be gratefully received.
A certain familiarity with C/C++, geometry, and Source mapping terms is assumed on the part of the reader. Code (mostly C structures) is given in a <code>fixed width font</code>. Sometimes the structures as shown are modified from their actual definitions in the SDK header files, for reasons of clarity and consistency.


This document describes version 19 of the BSP file format as used by the Source engine, which is used by [[Half-Life 2]] single player (HL2SP), [[Half-Life 2 Deathmatch]] (HL2DM), and [[Counter-Strike:Source]] (CS:S). The game [[Vampire The Masquerade - Bloodlines]] (VTMB) uses a modified form of an earlier format, version 17; known differences will be mentioned in their respective sections, however because no SDK is available for VTMB, this information is mostly guesswork.
== Overview ==
The BSP file contains the vast majority of the information needed by the Source engine to render and play a map. This includes the geometry of all the polygons in the level; references to the names and orientation of the textures to be drawn on those polygons; the data used to simulate the physical behaviour of the player and other items during the game; the location and properties of all brush-based, model (prop) based, and non-visible (logical) entities in the map; and the BSP tree and visibility table used to locate the player location in the map geometry and to render the visible map as efficiently as possible. Optionally, the map file can also contain any custom textures and models used on the level, embedded inside the map's Pakfile lump (see below).
 
Information not stored in the BSP file includes the map description text displayed by multiplayer games (such as {{Counter-Strike: Source|4}} or {{hl2dm|4}}) after loading the map (stored in the file ''mapname''.txt) and the AI navigation file used by non-player characters (NPCs) which need to navigate the map (stored in the file ''mapname''.nav). Because of the way the Source engine file system works, these external files may also be embedded in the BSP file's [[Source BSP File Format#Pakfile|Pakfile]] lump, though usually they are not.


Preliminary information on version 20 of the format, which supports high dynamic range ([[HDR]]) lighting as used in [[Day of Defeat: Source]] (DoD:S), is currently tentatively covered.
Historically, map files were stored in the game's corresponding [[GCF|Steam Game Cache File (GCF)]], but these are no longer used since 2013. In current versions of all of Valve's games, maps are stored directly in the OS file system. Rarely, such as in some third-party games or mods (especially {{Black Mesa|4}} and most {{src04|3}}/{{src06|3}} games that have been updated with [[SteamPipe]]), maps may be stored in [[VPK]] files; these can be extracted using {{gcfscape|4}} or the newer {{vpkedit|4}}.


A certain familiarity with C/C , geometry, and HL2 mapping terms is assumed on the part of the reader. Code (mostly C structures) is given in a <code>fixed width font</code>. Sometimes the structures as shown are modified from their actual definitions in the SDK header files, for reasons of clarity and consistency.
The data in the BSP file is stored according to the default endianness of the CPU it is intended to be loaded on: PC (x86) and {{switch|1}} (ARM) use little-endian, while {{ps3|1}} and {{Xbox360|1}} (both PowerPC) use big-endian. Byte-swapping is required when loading a little-endian file on a big-endian format platform such as Java and vice versa.


== Overview ==
{{Warning|There are many conflicting versions of [[BSP]]! Their compatibility isn't clear as even BSPs with the same Version Number can have underlying differences that will render them incompatible with games other than their source. Therefore it is a good rule of thumb to always try and stick with the intended game and its compile tools when messing around with BSPs.}}
 
{{Clr}}
 
== BSP file header ==
The BSP file starts with a header. This structure identifies the file as a Valve Source Engine BSP file, identifies the version of the format, and is then followed by a directory listing of the location, length, and version of up to 64 subsections of the file, known as ''lumps'', that store different parts of the map data. Finally, the map revision is given.
 
The structure of the header is given in the SDK's ''public/bspfile.h'' header file, a file which will be referencing extensively throughout this document. The header is 1036 bytes long in total:


The BSP file contains the vast majority of the information needed by the Source engine to render and play a map. This includes the geometry of all the polygons in the level; references to the names and orientation of the textures to be drawn on those polygons; the data used to simulate the physical behaviour of the player and other items during the game; the location and properties of all brush-based, model (prop) based, and non-visible (logical) entities in the map; and the BSP tree and visibility table used to locate the player location in the map geometry and to render the visible map as efficiently as possible. Optionally, the map file can also contain any custom textures and models used on the level, embedded inside the map's Pakfile lump (see below).
{{Note|1=This struct has been renamed to <code style=padding:0>BSPHeader_t</code> in {{Alien Swarm|4}}.}}
<syntaxhighlight lang="cpp">struct dheader_t
{
int    ident;                 // BSP file identifier
int    version;               // BSP file version
lump_t  lumps[HEADER_LUMPS];   // lump directory array
int    mapRevision;           // the map's revision (iteration, version) number
};</syntaxhighlight>


Information not stored in the BSP file includes the map description text displayed by HL2DM and CS:S after loading the map (stored in the file ''mapname''.txt) and the AI navigation file used by non-player characters (NPCs) which need to navigate the map (stored in the file ''mapname''.nav). Because of the way the Source engine file system works, these external files may also be embedded in the bsp file's Pakfile lump, though usually they are not.
Where <code>ident</code> is a 4-byte magic number defined as:


Official map files are stored in the [[Game Cache File|Steam Game Cache File (GCF)]] format, and are accessed through the Steam filesystem by the game engine. They can be extracted from the GCF files using Nemesis' [[GCFScape]] for perusal outside of Steam. Newer games using the [[VPK]] file format usually have their maps directly on the file system.
<syntaxhighlight lang="cpp">// little-endian "VBSP"  0x50534256
#define IDBSPHEADER (('P'<<24)+('S'<<16)+('B'<<8)+'V')</syntaxhighlight>


The data in the BSP file is stored throughout in little-endian byte format, in common with the preceding BSP formats as used by HL1, Quake, etc. Byte-swabbing is required if loading the file on a big-endian format platform such as Java.
The first four bytes of the file are thus always <code>VBSP</code> (in ASCII). These bytes identify the file as a Valve BSP file; other BSP file formats use a different magic number (such as for id Software's Quake engine games, which start with <code>IBSP</code>). The {{gldsrc|4}} BSP format does not use any magic number at all. The order of the magic number can also be used to determine the endianness of the file: <code>VBSP</code> is used for little-endian and <code>PSBV</code> for big-endian.


== BSP Versions ==
The second integer is the version of the BSP file format (BSPVERSION); for Source games, it ranges from 19 to 20 (21 since {{l4d2|4}}) with the exception of VTMB, which uses an earlier version of the format, 17 (see table below). Note that BSP file formats for other engines (HL1, Quake series, etc.) use entirely different version number ranges.


This table lists the different BSP version being used in some games based on the Source Engine.
=== Versions ===
This table gives an overview over the different BSP versions being used in some games/software based on the {{source|4|nt=1}}.<br />


{| class="standard-table"
{{Todo|add {{Day of Infamy|4}}, {{Apex|4}} and possibly others, as long they have modified lumps. See [[:Template:GameOnSource]]. }}
!Version
{| class=standard-table
!Game(s)
! Version !! Games/Software !! Notes
|-
| <span id="BSP 17">17</span>
| {{vtmb|4}}
| modified: dface_t
|-
| 17-18
| {{hl2|4}} (Beta)
| leaked beta
|-
| rowspan=2 | <span id="BSP 19">19</span>
| {{sineps|4}}
|
|-
| {{hl2xbox|4}}
| Custom lightmap format
|-
| 18-19
| {{hls|4}}
| Only c4a1y.bsp is 18. The rest are 19.<br>Tools create v20 BSPs.
|-
| rowspan=5 | <span id="BSP 19-20">19-20</span>
| {{hl2|4}}
| 19 when released, partially 20 with engine update introduced in Lost Coast ({{code|d1_trainstation_01}} only), and later more maps are v20 since Source 2007/2009 (Orange Box) up to Source 2013.</br>Console port ({{Xbox360|2}} and {{ps3|2}}) uses BSP 20 with LZMA compression.<br>All maps are now 20 after 20th anniversary update.<br>Tools create v20 BSPs.</br>
|-
| {{hl2dm|4}}
| rowspan=4 | 19 when released, partially 20 since Source 2009 (Orange Box) update in 2010 (for {{Counter-Strike: Source}}, {{Day of Defeat: Source}}), and Source 2013 (since 2013) for {{hldms}}. Supports LZMA compressed BSP Files.<br>{{hl2dm}} All maps are now 20 after 20th anniversary update.<br>Tools create same format as {{tf2|4}}.
|-
| {{hldms|4}}
|-
| {{Counter-Strike: Source|4}}
|-
| {{Day of Defeat: Source|4}}
|-
| rowspan=24 | <span id="BSP 20">20</span>
|-
| {{srcsdk06|4}}</br>{{srcsdk07|4}}</br>{{srcsdk13sp|4}}
| HDR lightmaps support. Tool creates v20 BSPs.
|-
| {{hl2lc|4}}
| HDR lightmaps support
|-
| {{hl2ep1|4}}
|
|-
| {{hl2ep2|4}}
|
|-
| {{gmod|4}}
| modified supports BSP Version 21, supports LZMA compressed BSP Files 
|-
| {{tf2|4}}</br>{{srcsdk13mp|4}}
| modified ( newly compiled maps ): StaticPropLump_t (version = 10, stores prop lightmap resolution info and expanded bitflags); LZMA compressed game lumps, entity info, and pakfiles; ambient lighting lump (additional ambient cubes per visleaf)</br>Console port of {{tf2|1}} ({{Xbox360|2}} and {{ps3|2}}) support LZMA compression, but maps are not compressed.
|-
| {{portal|4}}
| Same as {{hl2|1}} (PC). Console port ({{Xbox360|2}} and {{ps3|2}}) support LZMA compression, but maps are not compressed.
|-
| {{l4d|4}}
| modified: StaticPropLump_t ( version = 8 ) and dworldlight_t
|-
| {{zc|4}}
| modified: StaticPropLump_t ( version = 7 )
|-
| {{Dark Messiah of Might and Magic|4}}
| modified: dheader_t, StaticPropLump_t, texinfo_t, dgamelump_t, dmodel_t
|-
| {{Vindictus|4}}
| many modified structs
|-
| {{ship|4}}
| modified: StaticPropLump_t
|-
| {{bgt|4}}
| modified: StaticPropLump_t
|-
| {{Black Mesa|4}}
| modified: StaticPropLump_t ( version = 11 ) (commercial/Steam release only)
|-
| {{ftb|4}} (open beta version)
| modified: StaticPropLump_t ( version = 7 ), with a StaticPropLump_t ( version = 10 ) alike fields order
|-
| {{Eternal Silence|4}}
|
|-
| {{NT|4}}
|
|-
| {{P3|4}}
|
|-
| {{hl2u|4}}
|
|-
| {{InsurgencyMIC|4}}
|
|-
| {{portalrtx|4}}
|
|-
| {{tacint|4}} (Retail/FIX Launcher Release)
| 256-bit XOR Encryption (not just limited to BSPs)
|-
| {{eyedc|4}}
|
|-
| 19-22
| {{jbep3|4}}
| Map Compile Tools produce maps of the same format as {{Counter-Strike: Global Offensive|4|nt=1}}. Also supports LZMA-compressed game lumps, entity info, and pakfiles
{{Note|BSP verison 22 support is seemingly only for {{tacint|4}}, any other game with BSP version 22 is untested and most likely unsupported.}}
|-
| 20-21
| {{sfm|4}}
|
|-
| rowspan=14 | <span id="BSP 21">21</span>
| {{l4d2|4}}
| modified: lump_t, old <code>dbrushside_t</code>
|-
| {{Alien Swarm|4}}
|
|-
| {{Portal 2|4}}
| modified: StaticPropLump_t ( version = 9 ), <code>dbrushside_t</code> and other structs
|-
| {{Portal2Sixense|4}}
|
|-
| {{Counter-Strike: Global Offensive|4}}
| modified: StaticPropLump_t ( version = 11 )
|-
| {{Dear Esther|4}} (commercial release)
| modified: StaticPropLump_t
|-
| {{insurgency|4}} (commercial release)
|
|-
| {{tsp|4}} (commercial release)
|
|-
| {{tbg|4}}
|
|-
| {{Dino D-Day|4}}
|
|-
| {{nd|4}}
|
|-
|-
|17
| {{Rl|4}}
|[http://en.wikipedia.org/wiki/Vampire:_the_masquerade_bloodlines Vampire: The Masquerade Bloodlines]
|
|-
|-
|18
| {{Alien Swarm: Reactive Drop|4}}
|[[Half-Life 2|Half-Life 2 (Beta)]]
|
|-
|-
|19
| {{lw|4}}
|
|
[[Half-Life 2]]<br />
[[Half-Life 2: Deathmatch]]<br />
[[Counter-Strike: Source]]<br />
[[Day of Defeat: Source]]<br />
|-
|-
|20
| rowspan=3 | <span id="BSP 22">22</span>
| {{infra|4}}
|-
| {{tacint|4}} (Steam Release)
| Derived from {{p2|4}} version 21, with lump #24 being used to store client side entities ( similar to lump #0 )
|-
| rowspan=2 | {{Dota 2|4}} (pre-Reborn/original release)
| early betas, modified: <code>dbrushside_t</code>, ddispinfo_t
|-
| <span id="BSP 23">23</span>
| modified: <code>dbrushside_t</code>, ddispinfo_t, doverlay_t
|-
| <span id="BSP 25">25</span>
| {{Stratabranch|4}}
| [https://wiki.stratasource.org/shared/reference/general/bsp-v25 In development]
|-
| <span id="BSP 27">27</span>
| {{Contagion|4}}
|
|
[[Half-Life 2]] (partially, Source 2009 update)<br />
[[Half-Life 2: Episode One]]<br />
[[Half-Life 2: Episode Two]]<br />
[[Half-Life 2: Lost Coast]]<br />
[[Counter-Strike: Source]] (Source 2007 update)<br />
[[Day of Defeat: Source]] (Source 2007 update)<br />
[[Team Fortress 2]]<br />
[[Portal]]<br />
[[Left 4 Dead]]<br />
|-
|-
|21
| <span id="BSP 29">29</span>{{Cite|1}}
| {{Titanfall|4}}
| Heavily modified from Portal 2 - see [[BSP_(Source)/Game-Specific#Titanfall|Game-Specific page - Titanfall section]].
|-
| <span id="BSP 36">36</span>{{Cite|2}}</br><span id="BSP 37">37</span>{{Cite|1}}
| {{Titanfall2|4}}
| Heavily modified from Titanfall</br>BSP 36 is only used in PS4 Beta (Tech Test), release version use BSP 37.{{Todo|List changes}}
|-
| <span id="BSP 47">47</span>{{Cite|1}}
| {{Apex|4}}
| Heavily modified from Titanfall 2, with expanded map sizes.{{Todo|List changes}}
|-
| <span id="BSP 100">100</span>
| {{cso2|4}}
|
|
[[Left 4 Dead 2]]<br />
|-
[[Alien Swarm]] {{confirm|actually uses 20 with wrong header}}<br />
|}
|}


== BSP File Header ==
For more details on game-specific BSP formats, see [[Source BSP File Format/Game-Specific]].
 
The BSP file starts with a header. This structure identifies the file as a Valve Source Engine BSP file, identifies the version of the format, and is then followed by a directory listing of the location, length, and version of up to 64 subsections of the file, known as ''lumps'', that store different parts of the map data. Finally, the map revision is given.


The structure of the header is given in the SDK's ''public/bspfile.h'' header file, a file which I will be referencing extensively throughout this document. The header is 1036 bytes long in total:
=== Lump structure ===
 
Then follows an array of 16-byte <code>lump_t</code> structures. HEADER_LUMPS is defined as 64, so there are 64 entries. However, depending on the game and version, some lumps can be undefined or empty.
<source lang="cpp">struct dheader_t
{
int ident; // BSP file identifier
int version; // BSP file version
lump_t lumps[HEADER_LUMPS]; // lump directory array
int mapRevision; // the map's revision (iteration, version) number
};</source>
 
Here <code>ident</code> is a 4-byte magic number defined as:
 
<source lang="cpp">IDBSPHEADER (('P'<<24) ('S'<<16) ('B'<<8) 'V') //little-endian 'VBSP';</source>
 
The first four bytes of the file are thus always 'V' 'B' 'S' 'P' (in ASCII). These bytes identify the file as a Valve BSP file; other BSP file formats use a different magic number (such as for iD Software's Quake engine games, which start with 'IBSP'). The HL1 BSP format does not use any magic number at all.
 
The second integer is the version of the BSP file format (BSPVERSION); for HL2 games (until the release of HDR lighting) this was 19 (decimal); VTMB uses an earlier version of the format, 17. Note that BSP file formats for other engines (HL1, Quake series, etc.) use entirely different version number ranges.
 
The newest Valve BSP version is 20, for maps supporting HDR lighting. This is currently only used in maps for DoD:S and The Lost Coast, but presumably all forthcoming maps for the Source engine will use this version number.
 
Then follows an array of 16-byte <code>lump_t</code> structures. HEADER_LUMPS is defined as 64, so there are 64 entries, however only 52 of these lumps are currently used, the rest being undefined.


Each <code>lump_t</code> is defined in ''bspfile.h''<nowiki>: </nowiki>
Each <code>lump_t</code> is defined in ''bspfile.h''<nowiki>: </nowiki>


<source lang="cpp">struct lump_t
<syntaxhighlight lang="cpp">struct lump_t
{
{
int fileofs; // offset into file (bytes)
int   fileofs;     // offset into file (bytes)
int filelen; // length of lump(bytes)
int   filelen;     // length of lump (bytes)
int version; // lump format version
int   version;     // lump format version
char fourCC[4]; // lump ident code
char   fourCC[4];   // lump ident code
};</source>
};</syntaxhighlight>


In BSP version 21, the ''version'' field moved to the top:
The first two integers contain the byte offset (from the beginning of the bsp file) and byte length of that lump's data block; an integer defining the version number of the format of that lump (usually zero), and then a four byte identifier that is usually 0, 0, 0, 0. For compressed lumps, the fourCC contains the uncompressed lump data size in integer form (see section [[Source BSP File Format#Lump_compression|Lump compression]] for details). Unused members of the lump_t array (those that have no data to point to) have all elements set to zero.
{{todo|Some BSPs (tested in {{p2}}) have lumps without data (zero length) but still with offsets/versions set. Expand on this.}}


<source lang="cpp">struct lump_t
Lump offsets (and their corresponding data lumps) are always rounded up to the nearest 4-byte boundary, though the lump length may not be.
{
int version; // lump format version
int fileofs; // offset into file (bytes)
int filelen; // length of lump(bytes)
char fourCC[4]; // lump ident code
};</source>


The first two integers contain the byte offset (from the beginning of the bsp file) and byte length of that lump's data block; an integer defining the version number of the format of that lump (usually zero), and then a four byte identifier that is in practice always 0, 0, 0, 0. Unused members of the lump_t array (those that have no data to point to) have all elements set to zero.
=== Lump types ===
The type of data pointed to by the <code>lump_t</code> array is defined by its position in the array; for example, the first lump in the array '''(Lump 0)''' is always the BSP file's entity data (see below). The actual location of the data in the BSP file is defined by the offset and length entries for that lump, and does not need to be in any particular order in the file; for example, the entity data is usually stored towards the end of the BSP file despite being first in the lump array. The array of lump_t headers is therefore a directory of the actual lump data, which may be located anywhere else in the file.


Lump offsets (and their corresponding data lumps) are always rounded up to the nearest 4-byte boundary, though the lump length may not be.
{{Warning|Although the format does not require the lumps in a specific order, game lumps within <code>LUMP_GAME_LUMP</code> are denoted by their offset relative to the beginning of the BSP (except on console versions of {{portal2|2}}). When moving this lump these offsets must be recalculated.}}


=== Lump Types ===
The order of the lumps in the array is defined as:


The type of data pointed to by the <code>lump_t</code> array is defined by its position in the array; for example, the first lump in the array '''(Lump 0)''' is always the BSP file's entity data (see below). The actual location of the data in the BSP file is defined by the offset and length entries for that lump, and does not need to be in any particular order in the file; for example, the entity data is usually stored towards the end of the BSP file despite being first in the lump array. The array of lump_t headers is therefore a directory of the actual lump data, which may be located anywhere else in the file.
{{Todo|{{P3|4}}'s Compile Tools produce BSPs with LUMP_FACES with 346 surfaces, figure out where to put that in this table. }}
 
The order of the lumps in the array is defined as (lumps with unknown or uncertain purpose are marked with (?), ''italic text'' marks developer comments from the bspfile.h):


{| class="standard-table"
{| class="standard-table"
!Index
! Index
!Engine
! width=200 | Engine
!Name
! Name
!Name in bspfile.h
! Purpose
!Purpose
|-
| 0 || {{SourceBranch|2004}} || LUMP_ENTITIES || Map entities
|-
|-
|0
| 1 || {{SourceBranch|2004}} || LUMP_PLANES || Plane array
|[[File:Icon hl2.png|link=Source]]
|Entities
|LUMP_ENTITIES
|Map entities
|-
|-
|1
| 2 || {{SourceBranch|2004}} || LUMP_TEXDATA || Index to texture names
|[[File:Icon hl2.png|link=Source]]
|Planes
|LUMP_PLANES
|Plane array
|-
|-
|2
| 3 || {{SourceBranch|2004}} || LUMP_VERTEXES || Vertex array
|[[File:Icon hl2.png|link=Source]]
|Texdata
|LUMP_TEXDATA
|Index to texture names
|-
|-
|3
| 4 || {{SourceBranch|2004}} || LUMP_VISIBILITY || Compressed visibility bit arrays
|[[File:Icon hl2.png|link=Source]]
|Vertexes
|LUMP_VERTEXES
|Vertex array
|-
|-
|4
| 5 || {{SourceBranch|2004}} || LUMP_NODES || BSP tree nodes
|[[File:Icon hl2.png|link=Source]]
|Visibility
|LUMP_VISIBILITY
|Compressed visibility bit arrays
|-
|-
|5
| 6 || {{SourceBranch|2004}} || LUMP_TEXINFO || Face texture array
|[[File:Icon hl2.png|link=Source]]
|Nodes
|LUMP_NODES
|BSP tree nodes
|-
|-
|6
| 7 || {{SourceBranch|2004}} || LUMP_FACES || Face array
|[[File:Icon hl2.png|link=Source]]
|Texinfo
|LUMP_TEXINFO
|Face texture array
|-
|-
|7
| 8 || {{SourceBranch|2004}} || LUMP_LIGHTING || Lightmap samples
|[[File:Icon hl2.png|link=Source]]
|Faces
|LUMP_FACES
|Face array
|-
|-
|8
| 9 || {{SourceBranch|2004}} || LUMP_OCCLUSION || Occlusion polygons and vertices
|[[File:Icon hl2.png|link=Source]]
|Lighting
|LUMP_LIGHTING
|Lightmap samples
|-
|-
|9
| 10 || {{SourceBranch|2004}} || LUMP_LEAFS || BSP tree leaf nodes
|[[File:Icon hl2.png|link=Source]]
|Occlusion
|LUMP_OCCLUSION
|Occlusion data(?)
|-
|-
|10
| 11 || {{SourceBranch|2007}} || LUMP_FACEIDS || Correlates between dfaces and Hammer face IDs. Also used as random seed for [[Detail props|detail prop]] placement.
|[[File:Icon hl2.png|link=Source]]
|Leaves
|LUMP_LEAFS
|BSP tree leaf nodes
|-
|-
|11
| 12 || {{SourceBranch|2004}} || LUMP_EDGES || Edge array
|[[File:Tf2-16px.png|link=Source 2009]]
|FaceIDs
|LUMP_FACEIDS
|(?)
|-
|-
|12
| 13 || {{SourceBranch|2004}} || LUMP_SURFEDGES || Index of edges
|[[File:Icon hl2.png|link=Source]]
|Edges
|LUMP_EDGES
|Edge array
|-
|-
|13
| 14 || {{SourceBranch|2004}} || LUMP_MODELS || Brush models (geometry of brush entities)
|[[File:Icon hl2.png|link=Source]]
|Surfedges
|LUMP_SURFEDGES
|Index of edges
|-
|-
|14
| 15 || {{SourceBranch|2004}} || LUMP_WORLDLIGHTS || Internal world lights converted from the entity lump
|[[File:Icon hl2.png|link=Source]]
|Models
|LUMP_MODELS
|Brush models (geometry of brush entities)
|-
|-
|15
| 16 || {{SourceBranch|2004}} || LUMP_LEAFFACES || Index to faces in each leaf
|[[File:Icon hl2.png|link=Source]]
|Worldlights
|LUMP_WORLDLIGHTS
|Light entities
|-
|-
|16
| 17 || {{SourceBranch|2004}} || LUMP_LEAFBRUSHES || Index to brushes in each leaf
|[[File:Icon hl2.png|link=Source]]
|LeafFaces
|LUMP_LEAFFACES
|Index to faces in each leaf
|-
|-
|17
| 18 || {{SourceBranch|2004}} || LUMP_BRUSHES || Brush array
|[[File:Icon hl2.png|link=Source]]
|LeafBrushes
|LUMP_LEAFBRUSHES
|Index to brushes in each leaf
|-
|-
|18
| 19 || {{SourceBranch|2004}} || LUMP_BRUSHSIDES || Brushside array
|[[File:Icon hl2.png|link=Source]]
|Brushes
|LUMP_BRUSHES
|Brush array
|-
|-
|19
| 20 || {{SourceBranch|2004}} || LUMP_AREAS || Area array
|[[File:Icon hl2.png|link=Source]]
|Brushsides
|LUMP_BRUSHSIDES
|Brushside array
|-
|-
|20
| 21 || {{SourceBranch|2004}} || LUMP_AREAPORTALS || Portals between areas
|[[File:Icon hl2.png|link=Source]]
|Areas
|LUMP_AREAS
|Area array
|-
|-
|21
| rowspan=3 | 22 || {{SourceBranch|2004}} || LUMP_PORTALS || {{Confirm|Polygons defining the boundary between adjacent leaves?}}
|[[File:Icon hl2.png|link=Source]]
|AreaPortals
|LUMP_AREAPORTALS
|Portals between areas
|-
|-
| rowspan=3 |22
| {{SourceBranch|2007}} || LUMP_UNUSED0 || Unused
|[[File:Icon hl2.png|link=Source]]
|Portals
|LUMP_PORTALS
|Polygons defining the boundary between adjacent leaves(?)
|-
|-
|[[File:Tf2-16px.png|link=Source 2009]]
| {{SourceBranch|l4d2}} || LUMP_PROPCOLLISION || Static props convex hull lists
|Unused
|LUMP_UNUSED0
|
|-
|-
|[[File:L4D2-16px.png|link=Left 4 Dead (engine branch)]]
| rowspan=3 | 23 || {{SourceBranch|2004}} || LUMP_CLUSTERS || Leaves that are enterable by the player
|PropCollision
|LUMP_PROPCOLLISION
|''static props convex hull lists''
|-
|-
| rowspan=3 |23
| {{SourceBranch|2007}} || LUMP_UNUSED1 || Unused
|[[File:Icon hl2.png|link=Source]]
|Clusters
|LUMP_CLUSTERS
|Leaves that are enterable by the player
|-
|-
|[[File:Tf2-16px.png|link=Source 2009]]
| {{SourceBranch|l4d2}} || LUMP_PROPHULLS || Static prop convex hulls
|Unused
|LUMP_UNUSED1
|
|-
|-
|[[File:L4D2-16px.png|link=Left 4 Dead 2 (engine branch)]]
| rowspan=4 | 24 || {{SourceBranch|2004}} || LUMP_PORTALVERTS || Vertices of portal polygons
|PropHulls
|LUMP_PROPHULLS
|''static prop convex hulls''
|-
|-
| rowspan=3 |24
| {{SourceBranch|2007}} || LUMP_UNUSED2 || Unused
|[[File:Icon hl2.png|link=Source]]
|PortalVerts
|LUMP_PORTALVERTS
|Vertices of portal polygons
|-
|-
|[[File:Tf2-16px.png|link=Source 2009]]
| [[File:Icon-TacInt256.png|16px|link=Tactical Intervention]] [[Tactical Intervention|Source (TacInt branch)]]|| LUMP_FAKEENTITIES || Used to store client side entities ( Similar to Lump #0 )
|Unused
|LUMP_UNUSED2
|
|-
|-
|[[File:L4D2-16px.png|link=Left 4 Dead 2 (engine branch)]]
| {{SourceBranch|l4d2}} || LUMP_PROPHULLVERTS || Static prop collision vertices
|PropHullVerts
|LUMP_PROPHULLVERTS
|''static prop collision verts''
|-
|-
| rowspan=3 |25
| rowspan=3 | 25 || {{SourceBranch|2004}} || LUMP_CLUSTERPORTALS || {{Confirm|Polygons defining the boundary between adjacent clusters?}}
|[[File:Icon hl2.png|link=Source]]
|Clusterportals
|LUMP_CLUSTERPORTALS
|Polygons defining the boundary between adjacent clusters(?)
|-
|-
|[[File:Tf2-16px.png|link=Source 2009]]
| {{SourceBranch|2007}} || LUMP_UNUSED3 || Unused
|Unused
|LUMP_UNUSED3
|
|-
|-
|[[File:L4D2-16px.png|link=Left 4 Dead 2 (engine branch)]]
| {{SourceBranch|l4d2}} || LUMP_PROPTRIS || Static prop per hull triangle index start/count
|PropHullVerts
|LUMP_PROPTRIS
|''static prop per hull triangle index start/count''
|-
|-
|26
| 26 || {{SourceBranch|2004}} || LUMP_DISPINFO || Displacement surface array
|[[File:Icon hl2.png|link=Source]]
|Dispinfo
|LUMP_DISPINFO
|Displacement surface array
|-
|-
|27
| 27 || {{SourceBranch|2004}} || LUMP_ORIGINALFACES || Brush faces array before splitting
|[[File:Icon hl2.png|link=Source]]
|OriginalFaces
|LUMP_ORIGINALFACES
|Brush faces array before BSP splitting
|-
|-
|28
| 28 || {{SourceBranch|2007}} || LUMP_PHYSDISP || Displacement physics collision data
|[[File:Tf2-16px.png|link=Source 2009]]
|PhysDisp
|LUMP_PHYSDISP
|(?)
|-
|-
|29
| 29 || {{SourceBranch|2004}} || LUMP_PHYSCOLLIDE || Physics collision data
|[[File:Icon hl2.png|link=Source]]
|PhysCollide
|LUMP_PHYSCOLLIDE
|Physics collision data(?)
|-
|-
|30
| 30 || {{SourceBranch|2004}} || LUMP_VERTNORMALS || Face plane normals
|[[File:Icon hl2.png|link=Source]]
|VertNormals
|LUMP_VERTNORMALS
|Vertex normals(?)
|-
|-
|31
| 31 || {{SourceBranch|2004}} || LUMP_VERTNORMALINDICES || Face plane normal index array
|[[File:Icon hl2.png|link=Source]]
|VertNormalIndices
|LUMP_VERTNORMALINDICES
|Vertex normal index array(?)
|-
|-
|32
| 32 || {{SourceBranch|2004}} || LUMP_DISP_LIGHTMAP_ALPHAS || Displacement lightmap alphas (unused/empty since Source 2006)
|[[File:Icon hl2.png|link=Source]]
|DispLightmapAlphas
|LUMP_DISP_LIGHTMAP_ALPHAS
|Displacement lightmap data(?)
|-
|-
|33
| 33 || {{SourceBranch|2004}} || LUMP_DISP_VERTS || Vertices of displacement surface meshes
|[[File:Icon hl2.png|link=Source]]
|DispVerts
|LUMP_DISP_VERTS
|Vertices of displacement surface meshes
|-
|-
|34
| 34 || {{SourceBranch|2004}} || LUMP_DISP_LIGHTMAP_SAMPLE_POSITIONS || Displacement lightmap sample positions
|[[File:Icon hl2.png|link=Source]]
|DispLightmapSamplePos
|LUMP_DISP_LIGHTMAP_SAMPLE_POSITIONS
|Displacement lightmap data(?)
|-
|-
|35
| 35 || {{SourceBranch|2004}} || LUMP_GAME_LUMP || Game-specific data lump
|[[File:Icon hl2.png|link=Source]]
|GameLump
|LUMP_GAME_LUMP
|Game-specific data lump
|-
|-
|36
| 36 || {{SourceBranch|2004}} || LUMP_LEAFWATERDATA || Data for leaf nodes that are inside water
|[[File:Icon hl2.png|link=Source]]
|LeafWaterData
|LUMP_LEAFWATERDATA
|(?)
|-
|-
|37
| 37 || {{SourceBranch|2004}} || LUMP_PRIMITIVES || Water polygon data
|[[File:Icon hl2.png|link=Source]]
|Primitives
|LUMP_PRIMITIVES
|Non-polygonal primatives(?)
|-
|-
|38
| 38 || {{SourceBranch|2004}} || LUMP_PRIMVERTS || Water polygon vertices
|[[File:Icon hl2.png|link=Source]]
|PrimVerts
|LUMP_PRIMVERTS
|(?)
|-
|-
|39
| 39 || {{SourceBranch|2004}} || LUMP_PRIMINDICES || Water polygon vertex index array
|[[File:Icon hl2.png|link=Source]]
|PrimIndices
|LUMP_PRIMINDICES
|(?)
|-
|-
|40
| 40 || {{SourceBranch|2004}} || LUMP_PAKFILE || Embedded uncompressed or LZMA-compressed Zip-format file
|[[File:Icon hl2.png|link=Source]]
|Pakfile
|LUMP_PAKFILE
|Embedded uncompressed-Zip format file
|-
|-
|41
| 41 || {{SourceBranch|2004}} || LUMP_CLIPPORTALVERTS || Clipped portal polygon vertices
|[[File:Icon hl2.png|link=Source]]
|ClipPortalVerts
|LUMP_CLIPPORTALVERTS
|(?)
|-
|-
|42
| 42 || {{SourceBranch|2004}} || LUMP_CUBEMAPS || env_cubemap location array
|[[File:Icon hl2.png|link=Source]]
|Cubemaps
|LUMP_CUBEMAPS
|Env_cubemap location array
|-
|-
|43
| 43 || {{SourceBranch|2004}} || LUMP_TEXDATA_STRING_DATA || Texture name data
|[[File:Icon hl2.png|link=Source]]
|TexdataStringData
|LUMP_TEXDATA_STRING_DATA
|Texture name data
|-
|-
|44
| 44 || {{SourceBranch|2004}} || LUMP_TEXDATA_STRING_TABLE || Index array into texdata string data
|[[File:Icon hl2.png|link=Source]]
|TexdataStringTable
|LUMP_TEXDATA_STRING_TABLE
|Index array into texdata string data
|-
|-
|45
| 45 || {{SourceBranch|2004}} || LUMP_OVERLAYS || info_overlay data array
|[[File:Icon hl2.png|link=Source]]
|Overlays
|LUMP_OVERLAYS
|info_overlay array
|-
|-
|46
| 46 || {{SourceBranch|2004}} || LUMP_LEAFMINDISTTOWATER || Distance from leaves to water
|[[File:Icon hl2.png|link=Source]]
|LeafMinDistToWater
|LUMP_LEAFMINDISTTOWATER
|(?)
|-
|-
|47
| 47 || {{SourceBranch|2004}} || LUMP_FACE_MACRO_TEXTURE_INFO || Macro texture info for faces
|[[File:Icon hl2.png|link=Source]]
|FaceMacroTextureInfo
|LUMP_FACE_MACRO_TEXTURE_INFO
|(?)
|-
|-
|48
| 48 || {{SourceBranch|2004}} || LUMP_DISP_TRIS || Displacement surface triangles
|[[File:Icon hl2.png|link=Source]]
|DispTris
|LUMP_DISP_TRIS
|Displacement surface triangles
|-
|-
| rowspan=2 |49
| rowspan=2 | 49 || {{SourceBranch|2004}} || LUMP_PHYSCOLLIDESURFACE || Compressed win32-specific Havok terrain surface collision data. Deprecated and no longer used.
|[[File:Icon hl2.png|link=Source]]
|PhysCollideSurface
|LUMP_PHYSCOLLIDESURFACE
|''deprecated. We no longer use win32-speicifc havok compression on terrain''
|-
|-
|[[File:L4D2-16px.png|link=Left 4 Dead 2 (engine branch)]]
| {{SourceBranch|l4d2}} || LUMP_PROP_BLOB || Static prop triangle and string data
|PropBlob
|LUMP_PROP_BLOB
|''static prop triangle & string data''
|-
|-
|50
| 50 || {{SourceBranch|2006}} || LUMP_WATEROVERLAYS || Tied to any entity that uses the [[info_overlay_transition]] helper in FGD
|[[File:Icon hl2.png|link=Source 2006]]
|WaterOverlays
|LUMP_WATEROVERLAYS
|info_overlay's on water faces(?)
|-
|-
| rowspan=2 |51
| rowspan=2 | 51 || {{SourceBranch|2006}} || LUMP_LIGHTMAPPAGES || Alternate lightdata implementation for Xbox
|[[File:Icon hl2.png|link=Source 2006]]
|LightmapPages
|LUMP_LIGHTMAPPAGES
|''xbox: alternate lightdata implementation''
|-
|-
|[[File:Tf2-16px.png|link=Source 2009]]
| {{SourceBranch|2007}} || LUMP_LEAF_AMBIENT_INDEX_HDR || Index of LUMP_LEAF_AMBIENT_LIGHTING_HDR
|LeafAmbientIndexHDR
|LUMP_LEAF_AMBIENT_INDEX_HDR
|''index of LUMP_LEAF_AMBIENT_LIGHTING_HDR''
|-
|-
| rowspan=2 |52
| rowspan=2 | 52 || {{SourceBranch|2006}} || LUMP_LIGHTMAPPAGEINFOS || Alternate lightdata indices for Xbox
|[[File:Icon hl2.png|link=Source 2006]]
|LightmapPageInfos
|LUMP_LIGHTMAPPAGEINFOS
|''xbox: indexed by faces for alternate lightdata''
|-
|-
|[[File:Tf2-16px.png|link=Source 2009]]
| {{SourceBranch|2007}} || LUMP_LEAF_AMBIENT_INDEX || Index of LUMP_LEAF_AMBIENT_LIGHTING
|LeafAmbientIndex
|LUMP_LEAF_AMBIENT_INDEX
|''index of LUMP_LEAF_AMBIENT_LIGHTING''
|-
|-
|53
| 53 || {{SourceBranch|2004-v2}}</br>{{SourceBranch|2006}} || LUMP_LIGHTING_HDR || HDR lightmap samples
|[[File:Icon hl2.png|link=Source 2006]]
|LightingHDR
|LUMP_LIGHTING_HDR
|HDR related lighting data(?)
|-
|-
|54
| 54 || {{SourceBranch|2004-v2}}</br>{{SourceBranch|2006}} || LUMP_WORLDLIGHTS_HDR || Internal HDR world lights converted from the entity lump
|[[File:Icon hl2.png|link=Source 2006]]
|WorldlightsHDR
|LUMP_WORLDLIGHTS_HDR
|HDR related worldlight data(?)
|-
|-
|55
| 55 || {{SourceBranch|2004-v2}}</br>{{SourceBranch|2006}} || LUMP_LEAF_AMBIENT_LIGHTING_HDR || Per-leaf ambient light samples (HDR)
|[[File:Icon hl2.png|link=Source 2006]]
|LeaflightHDR1
|LUMP_LEAF_AMBIENT_LIGHTING_HDR
|HDR related leaf lighting data(?)
|-
|-
|56
| 56 || {{SourceBranch|2004-v2}}</br>{{SourceBranch|2006}} || LUMP_LEAF_AMBIENT_LIGHTING || Per-leaf ambient light samples (LDR)
|[[File:Icon hl2.png|link=Source 2006]]
|LeaflightHDR2
|LUMP_LEAF_AMBIENT_LIGHTING
|HDR related leaf lighting data(?)
|-
|-
|57
| 57 || {{SourceBranch|2006}} || LUMP_XZIPPAKFILE || XZip version of pak file for Xbox. Deprecated.
|[[File:Icon hl2.png|link=Source 2006]]
|XZipPakFile
|LUMP_XZIPPAKFILE
|XZip version of pak file for Xbox (deprecated)
|-
|-
|58
| 58 || {{SourceBranch|2006}} || LUMP_FACES_HDR || ''HDR maps may have different face data''
|[[File:Icon hl2.png|link=Source 2006]]
|FacesHDR
|LUMP_FACES_HDR
|''HDR maps may have different face data''
|-
|-
|59
| 59 || {{SourceBranch|2006}} || LUMP_MAP_FLAGS || Extended level-wide flags. Not present in all levels.
|[[File:Icon hl2.png|link=Source 2006]]
|MapFlags
|LUMP_MAP_FLAGS
|''extended level-wide flags. not present in all levels''
|-
|-
|60
| 60 || {{SourceBranch|2007}} || LUMP_OVERLAY_FADES || Fade distances for overlays
|[[File:Tf2-16px.png|link=Source 2009]]
|OverlayFades
|LUMP_OVERLAY_FADES
|Fade distances for overlays
|-
|-
|61
| 61 || {{SourceBranch|l4d}} || LUMP_OVERLAY_SYSTEM_LEVELS || System level settings (min/max CPU & GPU to render this overlay)
|[[File:L4D-16px.png|link=Left 4 Dead (engine branch)]]
|OverlaySystemLevels
|LUMP_OVERLAY_SYSTEM_LEVELS
|''System level settings (min/max CPU & GPU to render this overlay)''
|-
|-
|62
| 62 || {{SourceBranch|l4d2}} || LUMP_PHYSLEVEL || {{W|PhysX}} model of the World Brush. {{Confirm|The code that generates this only present within {{l4d2|4}} (only {{Code|c5m1_waterfront_sndscape}} appears to contains data on this lump), unknown if any Engine Licensees with access to the L4D2 Branch also have access to it.}}
|[[File:L4D2-16px.png|link=Left 4 Dead (engine branch)]]
|PhysLevel
|LUMP_PHYSLEVEL
|(?)
|-
|-
|63
| 63 || {{SourceBranch|2010}} || LUMP_DISP_MULTIBLEND || Displacement multiblend info
|[[File:Portal-16px.png|link=Source 2010]]
|MultiBlend
|LUMP_DISP_MULTIBLEND
|''Displacement multiblend info''
|}
|}


Lumps 53-56 are only used in version 20 BSP files. Lumps 22-25 are unused in version 20.
Lumps 53-56 are only used in version 20+ BSP files. Lumps 22-25 are unused in version 20.


The structure of the data lumps for the known entries is described below. Many of the lumps are simple arrays of structures; however some are of variable length depending on their content. The maximum size or number of entries in each lump is also defined in the ''bspfile.h'' file, as MAX_MAP_*.
The structure of the data lumps for the known entries is described below. Many of the lumps are simple arrays of structures; however some are of variable length depending on their content. The maximum size or number of entries in each lump is also defined in the ''bspfile.h'' file, as MAX_MAP_*.


Finally, the header ends with an integer containing the map revision number. This number is based on the revision number of the map's vmf file, which seems to increase each time the map is saved in the Hammer editor.
Finally, the header ends with an integer containing the map revision number. This number is based on the revision number of the map's vmf file (<code>mapversion</code>), which is increased each time the map is saved in the Hammer editor.


Immediately following the header is the first data lump. This can be any lump in the preceding list (pointed to using the offset field of that lump), though in practice the first data lump is Lump 1, the plane data array.
Immediately following the header is the first data lump. This can be any lump in the preceding list (pointed to using the offset field of that lump), though in practice the first data lump is Lump 1, the plane data array.
=== Lump compression ===
BSP files for console platforms such as PlayStation 3 and Xbox 360 usually have their lumps compressed with LZMA. In this case, the lump data starts with the following header (from ''public/tier1/lzmaDecoder.h''), which is used in place of the standard 13-byte LZMA header:
<syntaxhighlight lang="cpp">struct lzma_header_t
{
unsigned int    id;
unsigned int    actualSize;        // always little endian
unsigned int    lzmaSize;          // always little endian
unsigned char  properties[5];
};
</syntaxhighlight>
Where <code>id</code> is defined as:
<syntaxhighlight lang="cpp">// little-endian "LZMA"
#define LZMA_ID (('A'<<24)|('M'<<16)|('Z'<<8)|('L'))</syntaxhighlight>
'''lzmaSize''' denotes the size (in bytes) of compressed data, it is equal to the size of a lump minus 17 bytes (lzma header). '''actualSize''' denotes the size of decompressed data.
'''properties[5]''' field are used solely for LZMA decoding.
There are two special cases for compression: <code>LUMP_PAKFILE</code> is never compressed as a lump (the contents of the zip are compressed instead) and each of the game lumps in <code>LUMP_GAME_LUMP</code> are compressed individually. The compressed size of a game lump can be determined by subtracting the current game lump's offset with that of the next entry. For this reason, when game lumps are compressed the last game lump is always an empty dummy which only contains the offset.
If you wish to manipulate a compressed lump in an external tool, you need to convert the header mentioned above to the standard LZMA header format. It is equivalent to:
<syntaxhighlight lang="cpp">struct standard_lzma_header
{
unsigned char properties[5];
unsigned long long actualSize;      // 64-bit little endian
};
</syntaxhighlight>


== Lumps ==
== Lumps ==
=== Plane ===
=== Plane ===
The basis of the BSP geometry is defined by planes, which are used as splitting surfaces across the BSP tree structure.
The basis of the BSP geometry is defined by planes, which are used as splitting surfaces across the BSP tree structure.


The plane lump '''(Lump 1)''' is an array of <code>dplane_t</code> structures:
The plane lump '''(Lump 1)''' is an array of <code>dplane_t</code> structures:


<source lang="cpp">struct dplane_t
<syntaxhighlight lang="cpp">struct dplane_t
{
{
Vector normal; // normal vector
Vector normal;   // normal vector
float dist; // distance from origin
float   dist;     // distance from origin
int type; // plane axis identifier
int     type;     // plane axis identifier
};</source>
};</syntaxhighlight>


where the Vector type is a 3-vector defined as:
where the [[Vector]] type is a 3-vector defined as:


<source lang="cpp">struct Vector
<syntaxhighlight lang="cpp">struct Vector
{
{
float x;
float x;
float y;
float y;
float z;
float z;
};</source>
};</syntaxhighlight>


Floats are 4 bytes long; there are thus 20 bytes per plane, and the plane lump should be a multiple of 20 bytes long.
Floats are 4 bytes long; there are thus 20 bytes per plane, and the plane lump should be a multiple of 20 bytes long.
Line 605: Line 504:
The plane is represented by the element <code>normal</code>, a normal vector, which is a unit vector (length 1.0) perpendicular to the plane's surface. The position of the plane is given by <code>dist</code>, which is the distance from the map origin (0,0,0) to the nearest point on the plane.
The plane is represented by the element <code>normal</code>, a normal vector, which is a unit vector (length 1.0) perpendicular to the plane's surface. The position of the plane is given by <code>dist</code>, which is the distance from the map origin (0,0,0) to the nearest point on the plane.


Mathematically, the plane is described by the set of points (''x, y, z'') in the equation:
Mathematically, the plane is described by the set of points (''x, y, z'') which satisfy the equation:


''F''(''x,y,z'') = A''x'' B''y'' C''z'' D
A''x'' + B''y'' + C''z'' = D


where A, B, and C are given by the components <code>normal.x</code>, <code>normal.y</code> and <code>normal.z</code>, and D is <code>dist</code>. Each plane is infinite in extent, and divides the whole of the map coordinate volume into three pieces, on the plane (F=0), in front of the plane (F&gt;0), and behind the plane (F&lt;0).
where A, B, and C are given by the components <code>normal.x</code>, <code>normal.y</code> and <code>normal.z</code>, and D is <code>dist</code>. Each plane is infinite in extent, and divides the whole of the map coordinate volume into three pieces, on the plane (F=0), in front of the plane (F&gt;0), and behind the plane (F&lt;0).
Line 613: Line 512:
Note that planes have a particular orientation, corresponding to which side is considered "in front" of the plane, and which is "behind". The orientation of a plane can be flipped by negating the A, B, C, and D components.
Note that planes have a particular orientation, corresponding to which side is considered "in front" of the plane, and which is "behind". The orientation of a plane can be flipped by negating the A, B, C, and D components.


The <code>type</code> member of the structure seems to contain flags that indicate planes that are perpendicular to coordinates axes, but is usually not used.
The <code>type</code> member of the structure contains the axis that the plane is facing. It can be 0-5, with 0, 1, and 2 corresponding with X, Y, and Z respectively. Values of 3, 4 and 5 are used when planes are not along an axis, with each number corresponding to the axis it is closest to.


The can be up to 65536 planes in a map (MAX_MAP_PLANES).
There can be up to 65536 planes in a map (MAX_MAP_PLANES).


=== Vertex ===
=== Vertex ===
The vertex lump '''(Lump 3)''' is an array of coordinates of all the vertices (corners) of brushes in the map geometry. Each vertex is a <code>Vector</code> of 3 floats (x, y, and z), giving 12 bytes per vertex.
The vertex lump '''(Lump 3)''' is an array of coordinates of all the vertices (corners) of brushes in the map geometry. Each vertex is a <code>Vector</code> of 3 floats (x, y, and z), giving 12 bytes per vertex.


Line 626: Line 524:


=== Edge ===
=== Edge ===
The edge lump '''(Lump 12)''' is an array of <code>dedge_t</code> structures:
The edge lump '''(Lump 12)''' is an array of <code>dedge_t</code> structures:


<source lang="cpp">struct dedge_t
<syntaxhighlight lang="cpp">struct dedge_t
{
{
unsigned short v[2]; // vertex indices
unsigned short v[2]; // vertex indices
};</source>
};</syntaxhighlight>


Each edge is simply a pair of vertex indices (which index into the vertex lump array). The edge is defined as the straight line between the two vertices. Usually, the edge array is referenced through the Surfedge array (see below).
Each edge is simply a pair of vertex indices (which index into the vertex lump array). The edge is defined as the straight line between the two vertices. Usually, the edge array is referenced through the Surfedge array (see below).
Line 639: Line 536:


=== Surfedge ===
=== Surfedge ===
The Surfedge lump '''(Lump 13)''', presumable short for surface edge, is an array of (signed) integers. Surfedges are used to reference the edge array, in a somewhat complex way. The value in the surfedge array can be positive or negative. The absolute value of this number is an index into the edge array: if positive, it means the edge is defined from the first to the second vertex; if negative, from the second to the first vertex.
The Surfedge lump '''(Lump 13)''', presumable short for surface edge, is an array of (signed) integers. Surfedges are used to reference the edge array, in a somewhat complex way. The value in the surfedge array can be positive or negative. The absolute value of this number is an index into the edge array: if positive, it means the edge is defined from the first to the second vertex; if negative, from the second to the first vertex.


Line 646: Line 542:
There is a limit of 512000 (MAX_MAP_SURFEDGES) surfedges per map. Note that the number of surfedges is not necessarily the same as the number of edges in the map.
There is a limit of 512000 (MAX_MAP_SURFEDGES) surfedges per map. Note that the number of surfedges is not necessarily the same as the number of edges in the map.


=== Face and Original Face ===
=== Face and original face ===
 
The face lump '''(Lump 7)''' contains the major geometry of the map, used by the game engine to render the viewpoint of the player. The face lump contains faces after they have undergone the BSP splitting process; they therefore do not directly correspond to the faces of brushes created in Hammer. Faces are always flat, convex polygons, though they can contain edges that are co-linear.
The face lump '''(Lump 7)''' contains the major geometry of the map, used by the game engine to render the viewpoint of the player. The face lump contains faces after they have undergone the BSP splitting process; they therefore do not directly correspond to the faces of brushes created in Hammer. Faces are always flat, convex polygons, though they can contain edges that are co-linear.


The face lump is one of the more complex structures of the map file. It is an array of <code>dface_t</code> entries, each 56 bytes long:
The face lump is one of the more complex structures of the map file. It is an array of <code>dface_t</code> entries, each 56 bytes long:


<source lang="cpp">struct dface_t
<syntaxhighlight lang="cpp">struct dface_t
{
{
unsigned short planenum; // the plane number
unsigned short planenum;               // the plane number
byte side; // faces opposite to the node's plane direction
byte           side;                   // faces opposite to the node's plane direction
byte onNode; // 1 of on node, 0 if in leaf
byte           onNode;                 // 1 of on node, 0 if in leaf
int firstedge; // index into surfedges
int             firstedge;             // index into surfedges
short numedges; // number of surfedges
short           numedges;               // number of surfedges
short texinfo; // texture info
short           texinfo;               // texture info
short dispinfo; // displacement info
short           dispinfo;               // displacement info
short surfaceFogVolumeID; // ?
short           surfaceFogVolumeID;     // ?
byte styles[4]; // switchable lighting info
byte           styles[4];             // switchable lighting info
int lightofs; // offset into lightmap lump
int             lightofs;               // offset into lightmap lump
float area; // face area in units^2
float           area;                   // face area in units^2
int LightmapTextureMinsInLuxels[2]; // texture lighting info
int             LightmapTextureMinsInLuxels[2]; // texture lighting info
int LightmapTextureSizeInLuxels[2]; // texture lighting info
int             LightmapTextureSizeInLuxels[2]; // texture lighting info
int origFace; // original face this was split from
int             origFace;               // original face this was split from
unsigned short numPrims; // primitives
unsigned short numPrims;               // primitives
unsigned short firstPrimID;
unsigned short firstPrimID;
unsigned int smoothingGroups; // lightmap smoothing group
unsigned int   smoothingGroups;       // lightmap smoothing group
};</source>
};</syntaxhighlight>


The first member <code>planenum </code> is the plane number, i.e., the index into the plane array that corresponds to the plane that is aligned with this face in the world. <code>Side</code> is zero if this plane faces in the same direction as the face (i.e. "out" of the face) or non-zero otherwise.
The first member <code>planenum </code> is the plane number, i.e., the index into the plane array that corresponds to the plane that is aligned with this face in the world. <code>Side</code> is zero if this plane faces in the same direction as the face (i.e. "out" of the face) or non-zero otherwise.
[[File:bsp_geometry_s.gif|right|frame|Animated x-ray view of {{ent|Half-Life 2 map reference#d1_trainstation_01|alt=d1_trainstation_01}} with the three types of geometry data. [[Media:bsp_geometry.gif|View in full size here.]]]]


<code>Firstedge</code> is an index into the Surfedge array; this and the following <code>numedges</code> entries in the surfedge array define the edges of the face. As mentioned above, whether the value in the surfedge array is positive or negative indicates whether the corresponding pair of vertices listed in the Edge array should be traced from the first vertex to the second, or vice versa. The vertices which make up the face are thus referenced in clockwise order; when looking towards the face, each edge is traced in a clockwise direction. This makes rendering the faces easier, and allows quick culling of faces that face away from the viewpoint.
<code>Firstedge</code> is an index into the Surfedge array; this and the following <code>numedges</code> entries in the surfedge array define the edges of the face. As mentioned above, whether the value in the surfedge array is positive or negative indicates whether the corresponding pair of vertices listed in the Edge array should be traced from the first vertex to the second, or vice versa. The vertices which make up the face are thus referenced in clockwise order; when looking towards the face, each edge is traced in a clockwise direction. This makes rendering the faces easier, and allows quick culling of faces that face away from the viewpoint.
Line 685: Line 582:
The original face lump '''(Lump 27)''' has the same structure as the face lump, but contains the array of faces before the BSP splitting process is done. These faces are therefore closer to the original brush faces present in the precompile map than the face array, and there are less of them. The <code>origFace</code> entry for all original faces is zero. The maximum size of the original face array is also 65536 entries.
The original face lump '''(Lump 27)''' has the same structure as the face lump, but contains the array of faces before the BSP splitting process is done. These faces are therefore closer to the original brush faces present in the precompile map than the face array, and there are less of them. The <code>origFace</code> entry for all original faces is zero. The maximum size of the original face array is also 65536 entries.


Both the face and original face arrays are culled; that is, many faces present before compilation of the map (primarily those that face towards the "void" outside the map) are remove from the array.
Both the face and original face arrays are culled; that is, many faces present before compilation of the map (primarily those that face towards the "void" outside the map) are removed from the array.
 
Version 17 BSP files contain a substantially modified <code>dface_t</code> structure. The known elements are:


MAXLIGHTMAPS changed from 4 to 8  // 4 lightstyle for day + 4 for night = 8 ( nightime lightmapping system )
=== Brush and brushside ===
The brush lump '''(Lump 18)''' contains all [[brush]]es that were present in the original VMF file before compiling. Unlike faces, brushes are [[w:Constructive solid geometry|constructive solid geometry (CSG)]] defined by planes instead of edges and vertices. It is the presence of the brush and brushside lumps in Source BSP files that makes decompiling them a much easier job than for GoldSrc files, which lacked this info. The lump is an array of 12-byte <code>dbrush_t</code> structures:


<source lang="cpp">struct dface_bsp17_t
<syntaxhighlight lang="cpp">struct dbrush_t
{
{
colorRGBExp32 m_AvgLightColor[MAXLIGHTMAPS]; // For computing lighting information
int   firstside;     // first brushside
unsigned short planenum;
int   numsides;     // number of brushsides
byte side; // faces opposite to the node's plane direction
int   contents;     // contents flags
byte onNode; // 1 of on node, 0 if in leaf
};</syntaxhighlight>
int firstedge; // we must support > 64k edges
short numedges;
short texinfo;
short dispinfo;
short surfaceFogVolumeID;
byte styles[MAXLIGHTMAPS]; // lighting info
byte day[MAXLIGHTMAPS]; // Nightime lightmapping system
byte night[MAXLIGHTMAPS]; // Nightime lightmapping system
int lightofs; // start of [numstyles*surfsize] samples
float area;
int m_LightmapTextureMinsInLuxels[2];
int m_LightmapTextureSizeInLuxels[2];
int origFace;   // reference the original face this face was derived from
unsigned int smoothingGroups;
};
</source>
 
The extra data seems to be related to lighting of the face, and makes the length of the structure 104 bytes per face. Both the face lump and the original face lump in version 17 files use this structure.


=== Brush and Brushside ===
The first integer <code>firstside </code>is an index into the brushside array lump, this and the following <code>numsides</code> brushsides make up all the sides in this brush. The <code>contents</code> entry contains the [[BSP flags (Source)|contents flags, as defined in ''bspflags.h'']].
 
The brush lump '''(Lump 18)''' contains all brushes that were present in the original vmf file before compiling. (It is the presence of the brush and brushside lumps in HL2 bsp files that makes decompiling them a much easier job than for HL1 files, which lacked this info). The lump is an array of 12-byte <code>dbrush_t</code> structures:
 
<source lang="cpp">struct dbrush_t
{
int firstside; // first brushside
int numsides; // number of brushsides
int contents; // contents flags
};</source>
 
The first integer <code>firstside </code>is an index into the brushside array lump, this and the following <code>numsides</code> brushsides make up all the sides in this brush. The <code>contents</code> entry contains bitflags which determine the contents of this brush. The values are binary-ORed together, and are defined in the ''public/bspflags.h'' file:
 
<source lang="cpp">
CONTENTS_EMPTY 0 // No contents
CONTENTS_SOLID 0x1 // an eye is never valid in a solid
CONTENTS_WINDOW 0x2 // translucent, but not watery (glass)
CONTENTS_AUX 0x4
CONTENTS_GRATE 0x8 // alpha-tested "grate" textures. Bullets/VIS pass through, but solids don't
CONTENTS_SLIME 0x10
CONTENTS_WATER 0x20
CONTENTS_MIST 0x40
CONTENTS_OPAQUE 0x80 // things that cannot be seen through (may be non-solid though)
CONTENTS_TESTFOGVOLUME 0x100 // can see into a fogvolume (water)
CONTENTS_MOVEABLE 0x4000
CONTENTS_AREAPORTAL 0x8000
CONTENTS_PLAYERCLIP 0x10000
CONTENTS_MONSTERCLIP 0x20000
CONTENTS_CURRENT_0 0x40000
CONTENTS_CURRENT_90 0x80000
CONTENTS_CURRENT_180 0x100000
CONTENTS_CURRENT_270 0x200000
CONTENTS_CURRENT_UP 0x400000
CONTENTS_CURRENT_DOWN 0x800000
CONTENTS_ORIGIN 0x1000000 // removed before bsping an entity
CONTENTS_MONSTER 0x2000000 // should never be on a brush, only in game
CONTENTS_DEBRIS 0x4000000
CONTENTS_DETAIL 0x8000000 // brushes to be added after vis leafs
CONTENTS_TRANSLUCENT 0x10000000 // auto set if any surface has trans
CONTENTS_LADDER 0x20000000
CONTENTS_HITBOX 0x40000000 // use accurate hitboxes on trace</source>
 
Some of these flags seem to be inherited from previous game engines and are not used in Source maps. They are also used to describe to contents of the map's leaves (see below). The CONTENTS_DETAIL flag is used to mark brushes that were in [[func_detail]] entities before compiling.


The brush array is limited to 8192 entries (MAX_MAP_BRUSHES).
The brush array is limited to 8192 entries (MAX_MAP_BRUSHES).
Line 764: Line 600:
The brushside lump '''(Lump 19)''' is an array of 8-byte structures:
The brushside lump '''(Lump 19)''' is an array of 8-byte structures:


<source lang="cpp">struct dbrushside_t
<syntaxhighlight lang="cpp">struct dbrushside_t
{
{
unsigned short planenum; // facing out of the leaf
unsigned short planenum;     // facing out of the leaf
short texinfo; // texture info
short           texinfo;     // texture info
short dispinfo; // displacement info
short           dispinfo;     // displacement info
short bevel; // is the side a bevel plane?
#if BSPVERSION >= 21
};</source>
unsigned char  bevel;       // is the side a bevel plane?
 
unsigned char  thin;         // is any edge less than step size?
<code>Planenum</code> is an index info the plane array, giving the plane corresponding to this brushside. <code>Texinfo</code> and <code>dispinfo</code> are references into the texture and displacement info lumps. <code>Bevel</code> is zero for normal brush sides, but 1 if the side is a bevel plane (which seem to be used for collision detection).
#else
short          bevel;        // is the side a bevel plane?
#endif
};</syntaxhighlight>


Unlike the face array, brushsides are not culled (removed) where they touch the void. Void-facing sides do however have their texinfo entry changed to that of a [[NODRAW]] texture during the compile process. Note there is no direct way of linking brushes and brushsides and the corresponding face array entries which are used to render that brush.
<code>Planenum</code> is an index info the plane array, giving the plane corresponding to this brushside. <code>Texinfo</code> and <code>dispinfo</code> are references into the texture and displacement info lumps. <code>Bevel</code> is zero for normal brush sides, but 1 if the side is a bevel plane (which seem to be used for collision detection). BSP version 21 added a "thin" marker for brushsides where any edge length is less than step size (16 units).


The maximum number of brushsides is 65536 (MAX_MAP_BRUSHSIDES). The maximum number of brushsides on a single brush is 128 (MAX_BRUSH_SIDES).
Unlike the face array, brushsides are not culled (removed) where they touch the void. Void-facing sides do however have their texinfo entry changed to the <code>tools/[[Tool textures#nodraw|toolsnodraw]]</code> texture during the compile process. Note there is no direct way of linking brushes and brushsides and the corresponding face array entries which are used to render that brush. Brushsides are used by the engine to calculate all player physics collision with world brushes. (Vphysics objects use lump 29 instead.)


=== Node and Leaf ===
The maximum number of brushsides is 65536 ([[MAX_MAP_BRUSHSIDES]]). The maximum number of brushsides on a single brush is 128 (MAX_BRUSH_SIDES).


The node array '''(Lump 5)''' and leaf array '''(Lump 10)''' define the Binary Space Partition (BSP) tree structure of the map. The BSP tree is used by the engine to quickly determine the location of the player's viewpoint with respect to the map geometry, and along with the visibility information (see below), to decide which parts of the map are to be drawn.
=== Node and leaf ===
The node array '''(Lump 5)''' and leaf array '''(Lump 10)''' define the [[BSP tree|Binary Space Partition (BSP) tree]] structure of the map. The BSP tree is used by the engine to quickly determine the location of the player's viewpoint with respect to the map geometry, and along with the visibility information (see below), to decide which parts of the map are to be drawn.


The nodes and leaves form a tree structure. Each leaf represents a defined volume of the map, and each node represents the volume which is the sum of all its child nodes and leaves further down the tree.
The nodes and leaves form a tree structure. Each leaf represents a defined volume of the map, and each node represents the volume which is the sum of all its child nodes and leaves further down the tree.
Line 786: Line 626:
Each node has exactly two children, which can be either another node or a leaf. A child node has two further children, and so on until all branches of the tree are terminated with leaves, which have no children. Each node also references a plane in the plane array. When determining the player's viewpoint, the engine is trying to find which leaf the viewpoint falls inside. It first compares the coordinates of the point with the plane referenced in the headnode (Node 0). If the point is in front of the plane, it then moves to the first child of the node; otherwise, it moves to the second child. If the child is a leaf, then it has completed its task. If it is another node, it then performs the same check against the plane referenced in this node, and follows the children as before. It therefore traverses the BSP tree until it finds which leaf the viewpoint lies in. The leaves, then, completely divide up the map volume into a set of non-overlapping, convex volumes defined by the planes of their parent nodes.
Each node has exactly two children, which can be either another node or a leaf. A child node has two further children, and so on until all branches of the tree are terminated with leaves, which have no children. Each node also references a plane in the plane array. When determining the player's viewpoint, the engine is trying to find which leaf the viewpoint falls inside. It first compares the coordinates of the point with the plane referenced in the headnode (Node 0). If the point is in front of the plane, it then moves to the first child of the node; otherwise, it moves to the second child. If the child is a leaf, then it has completed its task. If it is another node, it then performs the same check against the plane referenced in this node, and follows the children as before. It therefore traverses the BSP tree until it finds which leaf the viewpoint lies in. The leaves, then, completely divide up the map volume into a set of non-overlapping, convex volumes defined by the planes of their parent nodes.


For more information on how the BSP tree is constructed, see the article "BSP for dummies" (http://www.planetquake.com/qxx/bsp/).
For more information on how the BSP tree is constructed, see the article "[http://web.archive.org/web/20050426034532/http://www.planetquake.com/qxx/bsp/ BSP for dummies]".


The node array consists of 32-byte structures:
The node array consists of 32-byte structures:


<source lang="cpp">struct dnode_t
<syntaxhighlight lang="cpp">struct dnode_t
{
{
int planenum; // index into plane array
int             planenum;       // index into plane array
int children[2]; // negative numbers are -(leafs 1), not nodes
int             children[2];   // negative numbers are -(leafs + 1), not nodes
short mins[3]; // for frustom culling
short           mins[3];       // for frustum culling
short maxs[3];
short           maxs[3];
unsigned short firstface; // index into face array
unsigned short firstface;     // index into face array
unsigned short numfaces; // counting both sides
unsigned short numfaces;       // counting both sides
short area; // If all leaves below this node are in the same area, then
short           area;           // If all leaves below this node are in the same area, then
// this is the area index. If not, this is -1.
                                // this is the area index. If not, this is -1.
short paddding; // pad to 32 bytes length
short           paddding;       // pad to 32 bytes length
};</source>
};</syntaxhighlight>


<code>Planenum</code> is the entry in the plane array. The <code>children[]</code> members are the two children of this node; if positive, they are node indices; if negative, the value (-1-''child'') is the index into the leaf array (e.g., the value -100 would reference leaf 99).
<code>Planenum</code> is the entry in the plane array. The <code>children[]</code> members are the two children of this node; if positive, they are node indices; if negative, the value (-1-''child'') is the index into the leaf array (e.g., the value -100 would reference leaf 99).
Line 807: Line 647:
The members <code>mins[]</code> and <code>maxs[]</code> are coordinates of a rough bounding box surrounding the contents of this node. The <code>firstface</code> and <code>numfaces</code> are indices into the face array that show which map faces are contained in this node, or zero if none are. The <code>area</code> value is the map area of this node (see below). There can be a maximum of 65536 nodes in a map (MAX_MAP_NODES).
The members <code>mins[]</code> and <code>maxs[]</code> are coordinates of a rough bounding box surrounding the contents of this node. The <code>firstface</code> and <code>numfaces</code> are indices into the face array that show which map faces are contained in this node, or zero if none are. The <code>area</code> value is the map area of this node (see below). There can be a maximum of 65536 nodes in a map (MAX_MAP_NODES).


The leaf array is an array with 56 bytes per element:
The leaf array is an array with 56 or 32 bytes per element, depending on the lump version:


<source lang="cpp">struct dleaf_t
<syntaxhighlight lang="cpp">struct dleaf_t
{
{
int contents; // OR of all brushes (not needed?)
int             contents;             // OR of all brushes (not needed?)
short cluster; // cluster this leaf is in
short           cluster;             // cluster this leaf is in
short area:9; // area this leaf is in
short           area:9;               // area this leaf is in
short flags:7; // flags
short           flags:7;             // flags
short mins[3]; // for frustum culling
short           mins[3];             // for frustum culling
short maxs[3];
short           maxs[3];
unsigned short firstleafface; // index into leaffaces
unsigned short firstleafface;       // index into leaffaces
unsigned short numleaffaces;
unsigned short numleaffaces;
unsigned short firstleafbrush; // index into leafbrushes
unsigned short firstleafbrush;       // index into leafbrushes
unsigned short numleafbrushes;
unsigned short numleafbrushes;
short leafWaterDataID; // -1 for not in water
short           leafWaterDataID;     // -1 for not in water
CompressedLightCube ambientLighting; // Precaculated light info for entities.
 
short padding; // padding to 4-byte boundary
//!!! NOTE: for lump version 0 (usually in maps of version 19 or lower) uncomment the next line
};</source>
//CompressedLightCube   ambientLighting;     // Precaculated light info for entities.
short                 padding;             // padding to 4-byte boundary
};</syntaxhighlight>


The leaf structure has similar contents to the node structure, except it has no children and no reference plane. Additional entries are the <code>contents</code> flags (see the brush lump, above), which shows the contents of any brushes in the leaf, and the <code>cluster</code> number of the leaf (see below). The <code>area</code> and <code>flags</code> members share a 16-bit bitfield and contain the area number and flags relating to the leaf. <code>Firstleafface</code> and <code>numleaffaces</code> index into the leafface array and show which faces are inside this leaf, if any. <code>Firstleafbrush</code> and <code>numleafbrushes</code> likewise index brushes inside this leaf through the leafbrush array.
The leaf structure has similar contents to the node structure, except it has no children and no reference plane. Additional entries are the <code>contents</code> flags (see the brush lump, above), which shows the contents of any brushes in the leaf, and the <code>cluster</code> number of the leaf (see below). The <code>area</code> and <code>flags</code> members share a 16-bit bitfield and contain the area number and flags relating to the leaf. <code>Firstleafface</code> and <code>numleaffaces</code> index into the leafface array and show which faces are inside this leaf, if any. <code>Firstleafbrush</code> and <code>numleafbrushes</code> likewise index brushes inside this leaf through the leafbrush array.
Line 837: Line 679:


=== LeafFace and LeafBrush ===
=== LeafFace and LeafBrush ===
The leafface lump '''(Lump 16)''' is an array of unsigned shorts which are used to map from faces referenced in the leaf structure to indices in the face array. The leafbrush lump (also an array of unsigned shorts)'''(Lump 17)''' does the same thing for brushes referenced in leaves. Their maximum sizes are both 65536 entries (MAX_MAP_LEAFFACES, MAX_MAP_LEAFBRUSHES).


The leafface lump '''(Lump 16)''' is an array of shorts which are used to map from faces referenced in the leaf structure to indices in the face array. The leafbrush lump '''(Lump 17)''' does the same thing for brushes referenced in leaves. Their maximum sizes are both 65536 entries (MAX_MAP_LEAFFACES, MAX_MAP_LEAFBRUSHES).
=== Textures ===
 
=== Texinfo, Texdata, TexdataStringData and TexdataStringTable ===
 
The texture information in a map is split across a number of different lumps. The Texinfo lump is the most fundamental, referenced by the face and brushside arrays, and it in turn references the other texture lumps.
The texture information in a map is split across a number of different lumps. The Texinfo lump is the most fundamental, referenced by the face and brushside arrays, and it in turn references the other texture lumps.


==== Texinfo ====
The texinfo lump '''(Lump 6)''' contains an array of <code>texinfo_t</code> structures:
The texinfo lump '''(Lump 6)''' contains an array of <code>texinfo_t</code> structures:


<source lang="cpp">struct texinfo_t
<syntaxhighlight lang="cpp">struct texinfo_t
{
{
float textureVecs[2][4]; // [s/t][xyz offset]
float   textureVecs[2][4];   // [s/t][xyz offset]
float lightmapVecs[2][4]; // [s/t][xyz offset] - length is in units of texels/area
float   lightmapVecs[2][4];   // [s/t][xyz offset] - length is in units of texels/area
int flags; // miptex flags overrides
int     flags;               // miptex flags overrides
int texdata; // Pointer to texture name, size, etc.
int     texdata;             // Pointer to texture name, size, etc.
}</source>
}</syntaxhighlight>


Each texinfo is 72 bytes long.
Each texinfo is 72 bytes long.
Line 858: Line 699:
The first array of floats is in essence two vectors that represent how the texture is orientated and scaled when rendered on the world geometry. The two vectors, '''s''' and '''t''', are the mapping of the left-to-right and down-to-up directions in the texture pixel coordinate space, onto the world. Each vector has an x, y, and z component, plus an offset which is the "shift" of the texture in that direction relative to the world. The length of the vectors represent the scaling of the texture in each direction.
The first array of floats is in essence two vectors that represent how the texture is orientated and scaled when rendered on the world geometry. The two vectors, '''s''' and '''t''', are the mapping of the left-to-right and down-to-up directions in the texture pixel coordinate space, onto the world. Each vector has an x, y, and z component, plus an offset which is the "shift" of the texture in that direction relative to the world. The length of the vectors represent the scaling of the texture in each direction.


The 2D coordinates (''u, v'') of a texture pixel (or ''texel'') are mapped to the world coordinates (''x, y, z'') of a point on a face by:
The 2D coordinates ([[UV map|''u, v'']]) of a texture pixel (or ''[[texel]]'') are mapped to the world coordinates (''x, y, z'') of a point on a face by:


''u'' = ''tv''<sub>0,0</sub> . ''x'' ''tv''<sub>0,1</sub> . ''y'' ''tv''<sub>0,2</sub> . ''z'' ''tv''<sub>0,3</sub>
''u'' = ''tv''<sub>0,0</sub> * ''x'' + ''tv''<sub>0,1</sub> * ''y'' ''tv''<sub>0,2</sub> * ''z'' + ''tv''<sub>0,3</sub>


''v'' = ''tv''<sub>1,0</sub> . ''x'' ''tv''<sub>1,1</sub> . ''y'' ''tv''<sub>1,2</sub> . ''z'' ''tv''<sub>1,3</sub>
''v'' = ''tv''<sub>1,0</sub> * ''x'' + ''tv''<sub>1,1</sub> * ''y'' + ''tv''<sub>1,2</sub> * ''z'' + ''tv''<sub>1,3</sub>


where ''tv''<sub>A,B</sub> is <code>textureVecs[A][B]</code>.
(ie. The dot product of the vectors with the vertex plus the offset in that direction. Where ''tv''<sub>A,B</sub> is <code>textureVecs[A][B]</code>.


The <code>lightmapVecs</code> float array performs a similar mapping of the lightmap samples of the texture onto the world.
Furthermore, after calculating (u, v), to convert them to texture coordinates which you would send to your graphics card, divide u and v by the width and height of the texture respectively.


The <code>flags</code> entry contains bitflags which are defined in ''bspflags.h'':
The <code>lightmapVecs</code> float array performs a similar mapping of the lightmap samples of the texture onto the world. It is the same formula but with lightmapVecs instead of textureVecs, and then subtracting the [0] and [1] values of LightmapTextureMinsInLuxels for u and v respectively. LightmapTextureMinsInLuxels is referenced in dface_t;


<source lang="cpp">SURF_LIGHT 0x0001 // value will hold the light strength
The <code>flags</code> entry contains [[BSP flags (Source)|surface flags, as defined in ''bspflags.h'']].
SURF_SLICK 0x0002 // effects game physics
SURF_SKY 0x0004 // don't draw, but add to skybox
SURF_WARP 0x0008 // turbulent water warp
SURF_TRANS 0x0010 // surface is transparent
SURF_WET 0x0020 // the surface is wet
SURF_FLOWING 0x0040 // scroll towards angle
SURF_NODRAW 0x0080 // don't bother referencing the texture
SURF_HINT 0x0100 // make a primary bsp splitter
SURF_SKIP 0x0200 // completely ignore, allowing non-closed brushes
SURF_NOLIGHT 0x0400 // Don't calculate light on this surface
SURF_BUMPLIGHT 0x0800 // calculate three lightmaps for the surface for bumpmapping
SURF_NOSHADOWS 0x1000 // Don't receive shadows
SURF_NODECALS 0x2000 // Don't receive decals
SURF_NOCHOP 0x4000 // Don't subdivide patches on this surface
SURF_HITBOX 0x8000 // surface is part of a hitbox</source>
 
The flags seem to be derived from the texture's .vmt file contents, and specify special properties of that texture.


==== Texdata ====
Finally the <code>texdata</code> entry is an index into the Texdata array, and specifies the actual texture.
Finally the <code>texdata</code> entry is an index into the Texdata array, and specifies the actual texture.


Line 895: Line 720:
The texdata array '''(Lump 2)''' consists of the structures:
The texdata array '''(Lump 2)''' consists of the structures:


<source lang="cpp">struct dtexdata_t
<syntaxhighlight lang="cpp">struct dtexdata_t
{
{
Vector reflectivity; // RGB reflectivity
Vector reflectivity;           // RGB reflectivity
int nameStringTableID; // index into TexdataStringTable
int     nameStringTableID;       // index into TexdataStringTable
int width, height; // source image
int     width, height;           // source image
int view_width, view_height;
int     view_width, view_height;
};</source>
};</syntaxhighlight>


The <code>reflectivity</code> vector corresponds to the RGB components of the reflectivity of the texture, as derived from the material's .vtf file. This is probably used in radiosity (lighting) calculations of what light bounces from the texture's surface. The <code>nameStringTableID</code> is an index into the TexdataStringTable array (below). The other members relate to the texture's source image.
The <code>reflectivity</code> vector corresponds to the RGB components of the reflectivity of the texture, as derived from the material's .vtf file. This is probably used in radiosity (lighting) calculations of what light bounces from the texture's surface. The <code>nameStringTableID</code> is an index into the TexdataStringTable array (below). The other members relate to the texture's source image.


==== TexdataStringData and TexdataStringTable ====
The TexdataStringTable '''(Lump 44)''' is an array of integers which are offsets into the TexdataStringData (lump 43). The TexdataStringData lump consists of concatenated null-terminated strings giving the texture name.
The TexdataStringTable '''(Lump 44)''' is an array of integers which are offsets into the TexdataStringData (lump 43). The TexdataStringData lump consists of concatenated null-terminated strings giving the texture name.


Line 910: Line 736:


=== Model ===
=== Model ===
A Model, in the terminology of the BSP file format, is a collection of brushes and faces, often called a "[[bmodel]]". It should not be confused with the prop models used in Hammer, which are usually called "studiomodels" in the SDK source.


A Model, in the terminology of the BSP file format, is a collection of brushes and faces, often called a "bmodel". It should not be confused with the prop models used in Hammer, which are usually called "studiomodels" in the SDK source.
The model lump '''(Lump 14)''' consists of an array of 48-byte <code>dmodel_t</code> structures:


The model lump '''(Lump 14)''' consists of an array of 24-byte <code>dmodel_t</code> structures:
<syntaxhighlight lang="cpp">struct dmodel_t
 
<source lang="cpp">struct dmodel_t
{
{
Vector mins, maxs; // bounding box
Vector mins, maxs;           // bounding box
Vector origin; // for sounds or lights
Vector origin;               // for sounds or lights
int headnode; // index into node array
int     headnode;             // index into node array
int firstface, numfaces; // index into face array
int     firstface, numfaces;   // index into face array
};</source>
};</syntaxhighlight>


<code>Mins</code> and <code>maxs</code> are the bounding points of the model. <code>Origin</code> is the coordinates of the model in the world, if set. <code>Headnode</code> is the index of the top node in the node array of the BSP tree which describes this model. <code>Firstface</code> and <code>numfaces</code> index into the face array and give the faces which make up this model.
<code>Mins</code> and <code>maxs</code> are the bounding points of the model. <code>Origin</code> is the coordinates of the model in the world, if set. <code>Headnode</code> is the index of the top node in the node array of the BSP tree which describes this model. <code>Firstface</code> and <code>numfaces</code> index into the face array and give the faces which make up this model.


The first model in the array (Model 0) is always "worldspawn", the overall geometry of the whole map excluding entities (but including func_detail brushes). The subsequent models in the array are associated with brush entities, and referenced from the entity lump.
The first model in the array (Model 0) is always "worldspawn", the overall geometry of the whole map excluding entities (but including func_detail and func_ladder brushes). The subsequent models in the array are associated with brush entities, and referenced from the entity lump.
 
There is a limit of 1024 models in a map (MAX_MAP_MODELS), including the worldspawn model zero.


=== Visibity ===
There is a limit of 1024 models in a map (MAX_MAP_MODELS), including the worldspawn model zero. This is increased to 4069 in {{jbep3|2}}.


=== Visibility ===
The visibility lump '''(Lump 4)''' is in a somewhat different format to the previously mentioned lumps. To understand it, some discussion of how the Source engine's visibility system works in necessary.
The visibility lump '''(Lump 4)''' is in a somewhat different format to the previously mentioned lumps. To understand it, some discussion of how the Source engine's visibility system works in necessary.


As mentioned in the "Node and Leaf Lumps" section above, every point in the map falls into exactly one convex volume called a leaf. All leaves that are on the inside of the map (not touching the void), and that are not covered by a solid brush can potentially have the player's viewpoint inside it during normal gameplay. Each of these enterable leaves (also called ''visleaves'') gets assigned a cluster number. In HL2 BSP files, each enterable leaf corresponds to just one cluster.
As mentioned in the [[Source BSP File Format#Node_and_leaf|Node and leaf lumps]] section above, every point in the map falls into exactly one convex volume called a leaf. All leaves that are on the inside of the map (not touching the void), and that are not covered by a solid brush can potentially have the player's viewpoint inside it during normal gameplay. Each of these enterable leaves (also called ''[[Visleaf|visleaves]]'') gets assigned a cluster number. In Source BSP files, each enterable leaf corresponds to just one cluster.


(The terminology is slightly confusing here. According to the "Quake 2 BSP File Format" article, in the Q2 engine there could be multiple adjacent leaves in each cluster - thus the cluster is so called because it is a cluster of leaves. As I understand it, it seems from the HL2 SDK source that this situation may also occur during the compilation of HL2 maps; however, after the VVIS compile process is finished these adjacent leaves (and their parent nodes) are merged into a single leaf. In all finished HL2 maps I have examined, it seems there is only ever one leaf per cluster. Therefore, in HL2 BSP files the distinction between clusters and enterable leaves (visleaves) is not meaningful.)
(The terminology is slightly confusing here. According to the "Quake 2 BSP File Format" article, in the Q2 engine there could be multiple adjacent leaves in each cluster - thus the cluster is so called because it is a cluster of leaves. It seems from the SDK source that this situation may also occur during the compilation of Source maps; however, after the [[VVIS]] compile process is finished these adjacent leaves (and their parent nodes) are generally merged into a single leaf. In older Source maps (prior to {{Counter-Strike: Global Offensive|4}}) it seems there is only ever one leaf per cluster. However, in some CS:GO maps multiple leaves can belong to the same cluster in the final compiled BSP. This seems to occur particularly in the 3D skybox of most CS:GO maps, and can be seen in the main playable area of more recently refurbished maps such as [[de_cbble]] and [[de_nuke]].)


Each cluster, then, is a volume in the map that the player can potentially be in. To render the map quickly, the game engine draws the geometry of only those clusters which are visible from the current cluster. Clusters which are completely occluded from view from the player's cluster need not be drawn. Calculating cluster-to-cluster visibility is the responsibility of the VVIS compile tool, and the resulting data is stored in the Visibility lump.
Each cluster, then, is a volume in the map that the player can potentially be in. To render the map quickly, the game engine draws the geometry of only those clusters which are visible from the current cluster. Clusters which are completely occluded from view from the player's cluster need not be drawn. Calculating cluster-to-cluster visibility is the responsibility of the VVIS compile tool, and the resulting data is stored in the Visibility lump.
Line 941: Line 765:
Once the engine knows a cluster is visible, the leaf data references all faces present in that cluster, allowing the contents of the cluster to be rendered.
Once the engine knows a cluster is visible, the leaf data references all faces present in that cluster, allowing the contents of the cluster to be rendered.


The data is stored as an array of bit-vectors; for each cluster, a list of which other clusters are visible from it are stored as individual bits (1 if visible, 0 if occluded) in an array, with the ''n''th bit position corresponding to the ''n''th cluster. This is known as the cluster's Potentially Visible Set (PVS). Because of the large size of this data, the bit vectors are compressed by run-length encoding groups of zero bits in each vector.
The data is stored as an array of bit-vectors; for each cluster, a list of which other clusters are visible from it are stored as individual bits (1 if visible, 0 if occluded) in an array, with the ''n''th bit position corresponding to the ''n''th cluster. This is known as the cluster's [[PVS|Potentially Visible Set (PVS)]]. Because of the large size of this data, the bit vectors are compressed by run-length encoding groups of zero bits in each vector.


There is also a Potentially Audible Set (PAS) array created for each cluster; this marks which clusters can hear sounds occurring in other clusters. The PAS seems to be created by merging the PVS bits of all clusters in current cluster's PVS.
There is also a [[PAS|Potentially Audible Set (PAS)]] array created for each cluster; this marks which clusters can hear sounds occurring in other clusters. The PAS seems to be created by merging the PVS bits of all clusters in current cluster's PVS.


The Visibilty lump is defined as:
The Visibilty lump is defined as:


<source lang="cpp">struct dvis_t
<syntaxhighlight lang="cpp">struct dvis_t
{
{
int numclusters;
int numclusters;
int byteofs[numclusters][2]
int byteofs[numclusters][2]
};</source>
};</syntaxhighlight>


The first integer is the number of clusters in the map. It is followed by an array of integers giving the byte offset from the start of the lump to the start of the PVS bit array for each cluster, followed by the offset to the PAS array. Immediately following the array are the compressed bit vectors.
The first integer is the number of clusters in the map. It is followed by an array of integers giving the byte offset from the start of the lump to the start of the PVS bit array for each cluster, followed by the offset to the PAS array. Immediately following the array are the compressed bit vectors.
Line 962: Line 786:


=== Entity ===
=== Entity ===
{{inline note|''See also: [[Patching levels with lump files]]''}}


:''See also: [[Patching levels with lump files]]''
The entity lump '''(Lump 0)''' is a null-terminated ASCII text buffer which stores the entity data in a format very similar to [[MAP]] files, but without any brushes. Its general form is:
 
The entity lump '''(Lump 0)''' is an ASCII text buffer which stores the entity data in a format very similar to uncompiled [[VMF]] files. Its general form is:


<pre>{
<pre>{
Line 995: Line 818:
}</pre>
}</pre>


Entities are defined between opening and closing braces (<code>{</code> and <code>}</code>) and list on each line a pair of key/value properties inside quotation marks. The first entity is always [[worldspawn]]. The <code>classname</code> property gives the entity type, and the <code>[[targetname]]</code> property gives the entity's name as defined in Hammer (if it has one). The <code>model</code> property is slightly special if it starts with an asterisk (*), the following number is an index into the model array (see above) which corresponds to a [[brush entity]] 'model'. Otherwise, the value contains the name of a compiled [[model]]. Other key/value pairs correspond to the properties of the entity as set in Hammer.
Entities are defined between opening and closing braces (<code>{</code> and <code>}</code>) and list on each line a pair of key/value properties inside quotation marks. The first entity is always [[worldspawn]]. The <code>[[classname]]</code> property gives the entity type, and the <code>[[targetname]]</code> property gives the entity's name as defined in Hammer (if it has one). The <code>model</code> property is slightly special if it starts with an asterisk (*), the following number is an index into the model array (see above) which corresponds to a [[bmodel]]. Otherwise, the value contains the name of a compiled [[model]], or sometimes a [[sprite]]. Other key/value pairs correspond to the properties of the entity as set in Hammer. [[Outputs]] are stored as regular keyvalues.


Note that some entities (including func_detail, env_cubemap, info_overlay and prop_static) are  [[internal entity|internal]], and are stripped from the entity lump during the compile process, normally because they are absorbed by [[the world]].
{{note|Some entities (including func_detail, env_cubemap, info_overlay and prop_static) are  [[:Category:Internal entities|internal]], and are stripped from the entity lump during the compile process, normally because they are absorbed by [[Worldspawn|the world]].}}


The entity lump can contain up to 8192 entities (<code>MAX_MAP_ENTITIES</code>; this is twice the engine's [[entity limit]]). Each key can be a maximum of 32 characters (<code>MAX_KEY</code>) and the value up to 1024 characters (<code>MAX_VALUE</code>).
There is no limit on the size of the entity lump inside the engine. VBSP has <code>MAX_MAP_ENTITIES</code>, which differs from the engine's actual [[entity limit]], and as such can be circumnavigated with a custom VBSP.
Each key can be a maximum of 32 characters (<code>MAX_KEY</code>) and the value up to 1024 characters (<code>MAX_VALUE</code>).


=== Game ===
In {{tf2|4}} the Entity lump may be compressed. In such case the entity data is encoded using LZMA algorithm (see [[Source BSP File Format#Lump_compression|Lump compression]]).


=== Game lump ===
The Game lump '''(Lump 35)''' seems to be intended to be used for map data that is specific to a particular game using the Source engine, so that the file format can be extended without altering the previously defined format. It starts with a game lump header:
The Game lump '''(Lump 35)''' seems to be intended to be used for map data that is specific to a particular game using the Source engine, so that the file format can be extended without altering the previously defined format. It starts with a game lump header:


<source lang="cpp">struct dgamelumpheader_t
<syntaxhighlight lang="cpp">struct dgamelumpheader_t
{
{
int lumpCount; // number of game lumps
int lumpCount; // number of game lumps
dgamelump_t gamelump[lumpCount];
dgamelump_t gamelump[lumpCount];
};</source>
};</syntaxhighlight>


where the gamelump directory array is defined by:
where the gamelump directory array is defined by:


<source lang="cpp">struct dgamelump_t
<syntaxhighlight lang="cpp">struct dgamelump_t
{
{
int id; // gamelump ID
int             id;       // gamelump ID
unsigned short flags; // flags
unsigned short flags;     // flags
unsigned short version; // gamelump version
unsigned short version;   // gamelump version
int fileofs; // offset to this gamelump
int             fileofs;   // offset to this gamelump
int filelen; // length
int             filelen;   // length
};</source>
};</syntaxhighlight>


The gamelump is identified by the 4-byte <code>id</code> member, which defines what data is stored in it, and the byte position of the data (from the start of the file) and its length is given in <code>fileofs</code> and <code>filelen</code>.
The gamelump is identified by the 4-byte <code>id</code> member, which defines what data is stored in it, and the byte position of the data and its length is given in <code>fileofs</code> and <code>filelen</code>. Note that <code>fileofs</code> is relative to the beginning of the BSP file, not to the game lump offset. One exception is the console version of {{portal2|4}}, where <code>fileofs</code> is relative to the game lump offset, as one would expect.


Of interest is the gamelump which is used to store prop_static entities, which uses the gamelump ID of 'sprp' ASCII (1936749168 decimal). Unlike most other entities, prop_statics are not stored in the entity lump. The gamelump formats used in HL2 are defined in the ''public/gamebspfile.h'' header file.
==== Static props ====
Of interest is the gamelump which is used to store [[prop_static]] entities, which uses the gamelump ID of 'sprp' ASCII (1936749168 decimal). Unlike most other entities, static props are not stored in the entity lump. The gamelump formats used in Source are defined in the ''public/gamebspfile.h'' header file.


The first element of the prop_static game lump is the dictionary; this is an integer count followed by the list of model (prop) names used in the map:
The first element of the static prop game lump is the dictionary; this is an integer count followed by the list of model (prop) names used in the map:


<source lang="cpp">struct StaticPropDictLump_t
<syntaxhighlight lang="cpp">struct StaticPropDictLump_t
{
{
int dictEntries;
int dictEntries;
char name[dictEntries]; // model name
char name[128][dictEntries]; // model name
};</source>
};</syntaxhighlight>


Each <code>name</code> entry is 128 characters long, null-padded to this length.
Each <code>name</code> entry is 128 characters long, null-padded to this length.
Line 1,038: Line 864:
Following the dictionary is the leaf array:
Following the dictionary is the leaf array:


<source lang="cpp">struct StaticPropLeafLump_t
<syntaxhighlight lang="cpp">struct StaticPropLeafLump_t
{
{
int leafEntries;
int leafEntries;
unsigned short leaf[leafEntries];
unsigned short leaf[leafEntries];
};</source>
};</syntaxhighlight>


Presumably, this array is used to index into the leaf lump to locate the leaves that each prop static is located in. Note that a prop static may span several leaves.
Presumably, this array is used to index into the leaf lump to locate the leaves that each prop static is located in. Note that a prop static may span several leaves.
Line 1,048: Line 874:
Next, an integer giving the number of <code>StaticPropLump_t</code> entries, followed by that many structures themselves:
Next, an integer giving the number of <code>StaticPropLump_t</code> entries, followed by that many structures themselves:


<source lang="cpp">struct StaticPropLump_t
<syntaxhighlight lang="cpp">struct StaticPropLump_t
{
{
// v4
// v4
Vector Origin; // origin
Vector         Origin;           // origin
QAngle Angles; // orientation (pitch roll yaw)
QAngle         Angles;           // orientation (pitch yaw roll)
unsigned short PropType; // index into model name dictionary
unsigned short FirstLeaf; // index into leaf array
// v4
unsigned short LeafCount;
unsigned short PropType;         // index into model name dictionary
unsigned char Solid; // solidity type
unsigned short FirstLeaf;         // index into leaf array
unsigned char Flags;
unsigned short LeafCount;
int Skin; // model skin numbers
unsigned char   Solid;             // solidity type
float FadeMinDist;
// every version except v7*
float FadeMaxDist;
unsigned char   Flags;
Vector LightingOrigin; // for lighting
// v4 still
int             Skin;             // model skin numbers
float           FadeMinDist;
float           FadeMaxDist;
Vector         LightingOrigin;   // for lighting
// since v5
// since v5
float ForcedFadeScale; // fade distance scale
float           ForcedFadeScale;   // fade distance scale
// v6 and v7 only
// v6, v7, and v7* only
unsigned short  MinDXLevel;     // minimum DirectX version to be visible
unsigned short  MinDXLevel;       // minimum DirectX version to be visible
unsigned short  MaxDXLevel;     // maximum DirectX version to be visible
unsigned short  MaxDXLevel;       // maximum DirectX version to be visible
        // since v8
// v7* only
unsigned int    Flags;
unsigned short  LightmapResX;      // lightmap image width
unsigned short LightmapResY;      // lightmap image height
// since v8
unsigned char  MinCPULevel;
unsigned char  MinCPULevel;
unsigned char  MaxCPULevel;
unsigned char  MaxCPULevel;
unsigned char  MinGPULevel;
unsigned char  MinGPULevel;
unsigned char  MaxGPULevel;
unsigned char  MaxGPULevel;
        // since v7
// since v7
        color32        DiffuseModulation; // per instance color and alpha modulation
color32        DiffuseModulation; // per instance color and alpha modulation
        // since v9
// v9 and v10 only
        bool            DisableX360;     // if true, don't show on XBox 360
bool            DisableX360;       // if true, don't show on XBox 360 (4-bytes long)
};</source>
// since v10
unsigned int    FlagsEx;          // Further bitflags.
// since v11
float          UniformScale;      // Prop scale
};</syntaxhighlight>
 
The coordinates of the prop are given by the <code>Origin</code> member; its orientation (pitch, yaw, roll) is given by the <code>Angles</code> entry, which is a 3-float vector. The <code>PropType</code> element is an index into the dictionary of prop model names, given above. The other elements correspond to the location of the prop in the BSP structure of the map, its lighting, and other entity properties as set in {{hammer|4}}. The other elements (<code>ForcedFadeScale</code>, etc.) are only present in the static prop structure if the gamelump's specified version is high enough (see <code>dgamelump_t.version</code>); both version 4 and version 5 static prop gamelumps are used in official {{hl2|4}} maps. Version 6 has been encountered in {{tf2|4}}; Version 7 is used in some {{l4d|4}} maps, and a modified version 7 is present in [[Zeno Clash]] maps. Version 8 is used predominantly in {{l4d|4}}, and version 9 in {{l4d2|4}}. The new version 10 appears in Tactical Intervention. Version 11 is used in {{Counter-Strike: Global Offensive|4}} since the addition of uniform prop scaling (before this it was version 10). After version 7, DX level options were removed. In version 11 XBox 360 flags were removed.
 
Version 7* is used by games built on {{src13mp|4}} ({{tf2|4}}, {{Counter-Strike: Source|4}}, etc.) and may come across as either version 7 or 10. Specifically, {{tf2|4}} has referred to it as version 7 in the past but now refers to it as version 10 even though they are identical. This version's structure is based on version 6 but rearranged such that <code>Flags</code> is an <code>int</code> and at the bottom, above two new entries. These new entries (<code>LightmapResX</code> and <code>LightmapResY</code>) control the width and height of the prop's lightmap image and are specific to this version.


The coordinates of the prop are given by the <code>Origin</code> member; its orientation (pitch, roll, yaw) is given by the <code>Angles</code> entry, which is a 3-float vector. The <code>PropType</code> element is an index into the dictionary of prop model names, given above. The other elements correspond to the location of the prop in the BSP structure of the map, its lighting, and other entity properties as set in Hammer. The last elements (<code>ForcedFadeScale</code> and and unknown value) are only present in the prop_static structure if the gamelump's specified is high eneough (see <code>dgamelump_t.version</code>); both version 4 and version 5 static prop gamelumps are used in official HL2 maps. Version 6 has been encountered in TF2. Version 7 is used in some Left 4 Dead maps, and a modified version 7 is present in [[Zeno Clash]] maps. Version 8 is used predominantly in [[Left 4 Dead]], and version 9 in [[Left 4 Dead 2]].
For a list of flags used by Flags and FlagsEX, see [[BSP (Source)/Static prop flags|static prop flags]].


Other gamelumps used in HL2 BSP files are the detail prop gamelump (ID is 'dprp'), and the detail prop lighting lump (ID: 'dplt'). These are used for the prop_detail entities (grass tufts, etc.) automatically emitted by certain textures when placed on displacement surfaces. In version 20 BSP files there is also another gamelump (ID: 'dplh') which is probably related to HDR lighting of detail props. Note that these gamelump IDs will appear reversed in a hex editor, and make a lot more sense seen that way. For example, the prop gamelump would be "prps".
==== Other ====
Other gamelumps used in Source BSP files are the detail prop gamelump (<code>dprp</code>), and the detail prop lighting lump (<code>dplt</code> for LDR and <code>dplh</code> for HDR). These are used for [[prop_detail]] entities (grass tufts, etc.) automatically emitted by certain textures when placed on displacement surfaces.


There does not seem to be a specified limit on the size of the game lump.
There does not seem to be a specified limit on the size of the game lump.


=== Dispinfo, DispVerts and DispTris ===
=== Displacements ===
Displacement surfaces are the most complex parts of a BSP file, and only part of their format is covered here. Their data is split over a number of different data lumps in the file, but the fundamental reference to them is through the dispinfo lump '''(Lump 26)'''. Dispinfos are referenced from the face, original face, and brushside arrays.


Displacement surfaces are the most complex parts of a BSP file, and I will cover only part of their format here. Their data is split over a number of different data lumps in the file, but the fundamental reference to them is through the dispinfo lump '''(Lump 26)'''. Dispinfos are referenced from the face, original face, and brushside arrays.
==== DispInfo ====
 
<syntaxhighlight lang="cpp">struct ddispinfo_t
<source lang="cpp">struct ddispinfo_t
{
{
Vector startPosition; // start position used for orientation
Vector               startPosition;               // start position used for orientation
int DispVertStart; // Index into LUMP_DISP_VERTS.
int                   DispVertStart;               // Index into LUMP_DISP_VERTS.
int DispTriStart; // Index into LUMP_DISP_TRIS.
int                   DispTriStart;                 // Index into LUMP_DISP_TRIS.
int power; // power - indicates size of surface (2^power 1)
int                   power;                       // power - indicates size of surface (2^power 1)
int minTess; // minimum tesselation allowed
int                   minTess;                     // minimum tesselation allowed
float smoothingAngle; // lighting smoothing angle
float                 smoothingAngle;               // lighting smoothing angle
int contents; // surface contents
int                   contents;                     // surface contents
unsigned short MapFace; // Which map face this displacement comes from.
unsigned short       MapFace;                     // Which map face this displacement comes from.
int LightmapAlphaStart; // Index into ddisplightmapalpha.
int                   LightmapAlphaStart;           // Index into ddisplightmapalpha.
int LightmapSamplePositionStart; // Index into LUMP_DISP_LIGHTMAP_SAMPLE_POSITIONS.
int                   LightmapSamplePositionStart; // Index into LUMP_DISP_LIGHTMAP_SAMPLE_POSITIONS.
CDispNeighbor EdgeNeighbors[4]; // Indexed by NEIGHBOREDGE_ defines.
CDispNeighbor         EdgeNeighbors[4];             // Indexed by NEIGHBOREDGE_ defines.
CDispCornerNeighbors CornerNeighbors[4]; // Indexed by CORNER_ defines.
CDispCornerNeighbors CornerNeighbors[4];           // Indexed by CORNER_ defines.
unsigned long AllowedVerts[ALLOWEDVERTS_SIZE]; // active verticies
unsigned int          AllowedVerts[10];             // active verticies
};</source>
};</syntaxhighlight>


The structure is 176 bytes long. The <code>startPosition</code> element is the coordinates of the first corner of the displacement. <code>DispVertStart</code> and <code>DispTriStart</code> are indices into the DispVerts and DispTris lumps. The <code>power</code> entry gives the number of subdivisions in the displacement surface - allowed values are 2, 3 and 4, and these correspond to 4, 8 and 16 subdivisions on each side of the displacement surface. The structure also references any neighbouring displacements on the sides or the corners of this displacement through the <code>EdgeNeighbors</code> and <code>CornerNeighbors</code> members. There are complex rules governing the order that these neighbour displacements are given; see the comments in ''bspfile.h'' for more. The <code>MapFace</code> value is an index into the face array and is face that was turned into a displacement surface. This face is used to set the texture and overall physical location and boundaries of the displacement.
The structure is 176 bytes long. The <code>startPosition</code> element is the coordinates of the first corner of the displacement. <code>DispVertStart</code> and <code>DispTriStart</code> are indices into the DispVerts and DispTris lumps. The <code>power</code> entry gives the number of subdivisions in the displacement surface - allowed values are 2, 3 and 4, and these correspond to 4, 8 and 16 subdivisions on each side of the displacement surface. The structure also references any neighbouring displacements on the sides or the corners of this displacement through the <code>EdgeNeighbors</code> and <code>CornerNeighbors</code> members. There are complex rules governing the order that these neighbour displacements are given; see the comments in ''bspfile.h'' for more. The <code>MapFace</code> value is an index into the face array and is face that was turned into a displacement surface. This face is used to set the texture and overall physical location and boundaries of the displacement.


==== DispVerts ====
The DispVerts lump '''(Lump 33)''' contains the vertex data of the displacements. It is given by:
The DispVerts lump '''(Lump 33)''' contains the vertex data of the displacements. It is given by:


<source lang="cpp">struct dDispVert
<syntaxhighlight lang="cpp">struct dDispVert
{
{
Vector vec; // Vector field defining displacement volume.
Vector vec;   // Vector field defining displacement volume.
float dist; // Displacement distances.
float   dist;   // Displacement distances.
float alpha; // "per vertex" alpha values.
float   alpha; // "per vertex" alpha values.
};</source>
};</syntaxhighlight>


where <code>vec</code> is the normalized vector of the offset of each displacement vertex from its original (flat) position; <code>dist</code> is the distance the offset has taken place; and <code>alpha</code> is the alpha-blending of the texture at that vertex.
where <code>vec</code> is the normalized vector of the offset of each displacement vertex from its original (flat) position; <code>dist</code> is the distance the offset has taken place; and <code>alpha</code> is the alpha-blending of the texture at that vertex.


A displacement of power ''p'' references (2^''p'' 1)^2 dispverts in the array, starting from the <code>DispVertStart</code> index.
A displacement of power ''p'' references (2^''p'' + 1)^2 dispverts in the array, starting from the <code>DispVertStart</code> index.


==== DispTris ====
The DispTris lump '''(Lump 48)''' contains "triangle tags" or flags related to the properties of a particular triangle in the displacement mesh:
The DispTris lump '''(Lump 48)''' contains "triangle tags" or flags related to the properties of a particular triangle in the displacement mesh:


<source lang="cpp">struct dDispTri
<syntaxhighlight lang="cpp">struct dDispTri
{
{
unsigned short Tags; // Displacement triangle tags.
unsigned short Tags;   // Displacement triangle tags.
};</source>
};</syntaxhighlight>


where the flags are:
where the flags are:


<source lang="cpp">DISPTRI_TAG_SURFACE 1
{| class=standard-table
DISPTRI_TAG_WALKABLE 2
! Name !! Value
DISPTRI_TAG_BUILDABLE 4
|-
DISPTRI_FLAG_SURFPROP1 8
| <code>DISPTRI_TAG_SURFACE</code> || 0x1
DISPTRI_FLAG_SURFPROP2 16</source>
|-
| <code>DISPTRI_TAG_WALKABLE</code> || 0x2
|-
| <code>DISPTRI_TAG_BUILDABLE</code> || 0x4
|-
| <code>DISPTRI_FLAG_SURFPROP1</code> || 0x8
|-
| <code>DISPTRI_FLAG_SURFPROP2</code> || 0x10
|}


There are 2x(2^''p'')^2 DispTri entries for a displacement of power ''p''. They are presumably used to indicate properties for each triangle of the displacement such as whether the surface is walkable at that point (not too steep to climb).
There are 2x(2^''p'')^2 DispTri entries for a displacement of power ''p''. They are presumably used to indicate properties for each triangle of the displacement such as whether the surface is walkable at that point (not too steep to climb).
Line 1,142: Line 995:


=== Pakfile ===
=== Pakfile ===
The Pakfile lump '''(Lump 40)''' is a special lump that can contains multiple files which are embedded into the bsp file. Usually, they contain special texture (.vtf) and material (.vmt) files which are used to store the reflection maps from env_cubemap entities in the map; these files are built and placed in the Pakfile lump when the [[Cubemaps#Building|<code>buildcubemaps</code>]] console command is executed. The Pakfile can optionally contain such things as custom textures and prop models used in the map, and are placed into the bsp file by using the [[BSPZIP]] program (or alternate programs such as [[Pakrat]]). These files are integrated into the game engine's file system and will be loaded preferentially before externally located files are used.


The Pakfile lump '''(Lump 40)''' is a special lump that can contains multiple files which are embedded into the bsp file. Usually, they contain special texture (.vtf) and material (.vmt) files which are used to store the reflection maps from env_cubemap entities in the map; these files are built and placed in the Pakfile lump when the "buildcubemaps" console command is executed. The Pakfile can optionally contain such things as custom textures and prop models used in the map, and are placed into the bsp file by using the BSPZIP program (or alternate programs such as [pakrat.html Pakrat]). These files are integrated into the game engine's filesystem and will be loaded preferentially before externally located files are used.
The format of the Pakfile lump is identical to that used by the Zip compression utility when no compression is specified (i.e., the individual files are stored in uncompressed format). In some branches, such as {{src13mp|2}}, {{gmod|2}}, {{jbep3|2}}, and {{strata|2}}, LZMA compression can be used as well. If the Pakfile lump is extracted and written to a file, it can therefore be opened with WinZip and similar programs.
 
The format of the Pakfile lump is identical to that used by the Zip compression utility when no compression is specified (i.e., the individual files are stored in uncompressed format). If the Pakfile lump is extracted and written to a file, it can therefore be opened with WinZip and similar programs.


The header ''public/zip_uncompressed.h'' defines the structures present in the Pakfile lump. The last element in the lump is a <code>ZIP_EndOfCentralDirRecord</code> structure. This points to an array of <code>ZIP_FileHeader</code> structures immediately preceeding it, one for each file present in the Pak. Each of these headers then point to <code>ZIP_LocalFileHeader</code> structures that are followed by that file's data.
The header ''public/zip_uncompressed.h'' defines the structures present in the Pakfile lump. The last element in the lump is a <code>ZIP_EndOfCentralDirRecord</code> structure. This points to an array of <code>ZIP_FileHeader</code> structures immediately preceeding it, one for each file present in the Pak. Each of these headers then point to <code>ZIP_LocalFileHeader</code> structures that are followed by that file's data.
Line 1,152: Line 1,004:


=== Cubemap ===
=== Cubemap ===
The Cubemap lump '''(Lump 42)''' is an array of 16-byte <code>dcubemapsample_t</code> structures:


The Cubemap lump '''(Lump 42)''' is an array of 13-byte <code>dcubemapsample_t</code> structures:
<syntaxhighlight lang="cpp">struct dcubemapsample_t
 
<source lang="cpp">struct dcubemapsample_t
{
{
int origin[3]; // position of light snapped to the nearest integer
int   origin[3];   // position of light snapped to the nearest integer
unsigned char size; // resolution of cubemap, 0 - default
int    size;         // resolution of cubemap, 0 - default
};</source>
};</syntaxhighlight>


The <code>dcubemapsample_t</code> structure defines the location of a env_cubemap entity in the map. The <code>origin</code> member contains integer x,y,z coordinates of the cubemap, and the <code>size</code> member is resolution of the cubemap, specified as 2^(''size''-1) pixels square. If set as 0, the default size of 6 (32x32 pixels) is used. There can be a maximum of 1024 (MAX_MAP_CUBEMAPSAMPLES) cubemaps in a file.
The <code>dcubemapsample_t</code> structure defines the location of a env_cubemap entity in the map. The <code>origin</code> member contains integer x,y,z coordinates of the cubemap, and the <code>size</code> member is resolution of the cubemap, specified as 2^(''size''-1) pixels square. If set as 0, the default size of 6 (32x32 pixels) is used. There can be a maximum of 1024 (MAX_MAP_CUBEMAPSAMPLES) cubemaps in a file.
Line 1,165: Line 1,016:
When the "buildcubemaps" console command is performed, six snapshots of the map (one for each direction) are taken at the location of each env_cubemap entity. These snapshots are stored in a multi-frame texture (vtf) file, which is added to the Pakfile lump (see above). The textures are named <code>cX_Y_Z.vtf</code>, where (X,Y,Z) are the (integer) coordinates of the corresponding cubemap.
When the "buildcubemaps" console command is performed, six snapshots of the map (one for each direction) are taken at the location of each env_cubemap entity. These snapshots are stored in a multi-frame texture (vtf) file, which is added to the Pakfile lump (see above). The textures are named <code>cX_Y_Z.vtf</code>, where (X,Y,Z) are the (integer) coordinates of the corresponding cubemap.


Faces containing materials that are environment mapped (e.g. shiny textures) reference their assigned cubemap through their material name. A face with a material named (e.g.) <code>walls/shiny.vmt</code> is altered (new Texinfo &amp; Texdata entries are created) to refer to a renamed material <code>maps/''mapname''/walls/shiny_X_Y_Z.vmt</code>, where (X,Y,Z) are the cubemap coordinates as before. This .vmt file is also stored in the Pakfile, and references the cubemap .vtf file through its <code>$envmap</code> property.
Faces containing materials that are environment mapped (e.g. shiny textures) reference their assigned cubemap through their material name. A face with a material named (e.g.) <code>walls/shiny.vmt</code> is altered (new Texinfo & Texdata entries are created) to refer to a renamed material <code>maps/''mapname''/walls/shiny_X_Y_Z.vmt</code>, where (X,Y,Z) are the cubemap coordinates as before. This .vmt file is also stored in the Pakfile, and references the cubemap .vtf file through its <code>$envmap</code> property.


Version 20 files contain extra <code>cX_Y_Z.hdr.vtf</code> files in the Pakfile lump, containing HDR texture files in RGBA16161616F (16-bit per channel) format.
Version 20 files contain extra <code>cX_Y_Z.hdr.vtf</code> files in the Pakfile lump, containing HDR texture files in RGBA16161616F (16-bit per channel) format.


=== Overlay ===
=== Overlay ===
Unlike the simpler decals (infodecal entities), info_overlays are removed from the entity lump and stored separately in the Overlay lump '''(Lump 45)'''. The structure is reflects the properties of the entity in {{hammer|4}} almost exactly:


Unlike the simpler decals (infodecal entities), info_overlays are removed from the entity lump and stored separately in the Overlay lump '''(Lump 45)'''. The structure is reflects the properties of the entity in Hammer almost exactly:
<syntaxhighlight lang="cpp">struct doverlay_t
 
<source lang="cpp">struct doverlay_t
{
{
int Id;
int             Id;
short TexInfo;
short           TexInfo;
unsigned short FaceCountAndRenderOrder;
unsigned short FaceCountAndRenderOrder;
int Ofaces[OVERLAY_BSP_FACE_COUNT];
int             Ofaces[OVERLAY_BSP_FACE_COUNT];
float U[2];
float           U[2];
float V[2];
float           V[2];
Vector UVPoints[4];
Vector         UVPoints[4];
Vector Origin;
Vector         Origin;
Vector BasisNormal;
Vector         BasisNormal;
};</source>
};</syntaxhighlight>


The <code>FaceCountAndRenderOrder</code> member is split into two parts; the lower 14 bits are the number of faces that the overlay appears on, with the top 2 bits being the render order of the overlay (for overlapping decals). The <code>Ofaces</code> array, which is 64 elements in size (OVERLAY_BSP_FACE_COUNT) are the indices into the face array indicating which map faces the overlay should be displayed on. The other elements set the texture, scale, and orientation of the overlay decal. There can be a maximum of 512 overlays per file (MAX_MAP_OVERLAYS).
The <code>FaceCountAndRenderOrder</code> member is split into two parts; the lower 14 bits are the number of faces that the overlay appears on, with the top 2 bits being the render order of the overlay (for overlapping decals). The <code>Ofaces</code> array, which is 64 elements in size (OVERLAY_BSP_FACE_COUNT) are the indices into the face array indicating which map faces the overlay should be displayed on. The other elements set the texture, scale, and orientation of the overlay decal. There is no enforced limit on overlays inside the engine. VBSP enforces a limit of 512 (<code>MAX_MAP_OVERLAYS</code>, 1024 in {{Counter-Strike: Global Offensive|4}}), but custom compilers can circumvent this.


=== Lighting ===
=== Lighting ===
The lighting lump '''(Lump 8)''' is used to store the static lightmap samples of map faces. Each lightmap sample is a colour tint that multiplies the colours of the underlying texture pixels, to produce lighting of varying intensity. These lightmaps are created during the VRAD phase of map compilation and are referenced from the <code>dface_t</code> structure. The current lighting lump version is 1.


The lighting lump '''(Lump 8)''' is used to store the static lightmap samples of map faces. Each lightmap sample is a colour tint that multiplies the colours of the underlying texture pixels, to produce lighting of varying intensity. These lightmaps are created during the VRAD phase of map compilation and are referenced from the <code>dface_t</code> structure. The current lighting lump version is 1.
Each <code>dface_t</code> may have a up to four lightstyles (<code>MAXLIGHTMAPS</code>) defined in its <code>styles[]</code> array (which contains 255 to represent no lightstyle). The number of luxels in each direction of the face is given by the two <code>LightmapTextureSizeInLuxels[]</code> members (plus 1), and the total number of luxels per face is thus:


Each <code>dface_t</code> may have a up to four lightstyles defined in its <code>styles[]</code> array (which contains 255 to represent no lightstyle). The number of luxels in each direction of the face is given by the two <code>LightmapTextureSizeInLuxels[]</code> members (plus 1), and the total number of luxels per face is thus (<code>LightmapTextureSizeInLuxels[0]</code> 1) * (<code>LightmapTextureSizeInLuxels[1]</code> 1).
<code>(LightmapTextureSizeInLuxels[0] + 1) * (LightmapTextureSizeInLuxels[1] + 1)</code>


Each face gives a byte offset into the lighting lump in its <code>lightofs</code> member (if no lighting information is used for this face e.g. faces with skybox, nodraw and invisible textures, <code>lightofs</code> is -1.) There are (''number of lightstyles'')*(''number of luxels'') lightmap samples for each face, where each sample is a 4-byte ColorRGBExp32 structure:
Each face gives a byte offset into the lighting lump in its <code>lightofs</code> member (if no lighting information is used for this face e.g. faces with skybox, nodraw and invisible textures, <code>lightofs</code> is -1.) There are (''number of lightstyles'')*(''number of luxels'') lightmap samples for each face, where each sample is a 4-byte ColorRGBExp32 structure:


<source lang="cpp">struct ColorRGBExp32
<syntaxhighlight lang="cpp">struct ColorRGBExp32
{
{
byte r, g, b;
byte r, g, b;
signed char exponent;
signed char exponent;
};</source>
};</syntaxhighlight>


Standard RGB format can be obtained from this by multiplying each colour component by 2^(''exponent''). For faces with bumpmapped textures, there are four times the usual number of lightmap samples, presumably containing samples used to compute the bumpmapping.
Standard RGB format can be obtained from this by multiplying each colour component by 2^(''exponent''). For faces with bumpmapped textures, there are four times the usual number of lightmap samples, with one lightmap lit from each [[normal]] axis, and a fourth flat one used when bumpmapping is disabled.


Immediately preceeding the <code>lightofs</code>-referenced sample group, there are single samples containing the average lighting on the face, one for each lightstyle, in reverse order from that given in the <code>styles[]</code> array.
Immediately preceeding the <code>lightofs</code>-referenced sample group, there are single samples containing the average lighting on the face, one for each lightstyle, in reverse order from that given in the <code>styles[]</code> array.
Line 1,208: Line 1,059:
Version 20 BSP files contain a second, identically sized lighting lump '''(Lump 53)'''. This is presumed to store more accurate (higher-precision) HDR data for each lightmap sample. The format is currently unknown, but is also 32 bits per sample.
Version 20 BSP files contain a second, identically sized lighting lump '''(Lump 53)'''. This is presumed to store more accurate (higher-precision) HDR data for each lightmap sample. The format is currently unknown, but is also 32 bits per sample.


The maximum size of the lighting lump is 0x1000000 bytes, i.e. 16 Mb (MAX_MAP_LIGHTING).
The maximum size of the lighting lump is 0x1000000 bytes (MAX_MAP_LIGHTING), i.e. 16 MB, and can be mapped onto a 1024x512x32 atlas. This limitation is not enforced in the engine and compilers can freely exceed it.
{{Confirm|What size atlas does Source actually use? IdTech 2 and GoldSrc used 128x128 atlases (see [[AllocBlock]]).}}
 
=== Ambient Lighting ===
The ambient lighting lumps '''(Lumps 55 and 56)''' are present in BSP version 20 and later. Lump 55 is used for HDR lighting, and Lump 56 is used for LDR lighting. These lumps are used to store the volumetric [[ambient light]]ing information in each leaf (i.e. lighting information for entities such as NPCs, the viewmodel, and non-static props). Prior to version 20, this data was stored in the leaf lump (Lump 10), in the <code>dleaf_t</code> structure, with far less precision than this newer lump allows.
 
Both ambient lighting lumps are arrays of <code>dleafambientlighting_t</code> structures:
 
<syntaxhighlight lang="cpp">struct dleafambientlighting_t
{
CompressedLightCube cube;
byte x; // fixed point fraction of leaf bounds
byte y; // fixed point fraction of leaf bounds
byte z; // fixed point fraction of leaf bounds
byte pad; // unused
};</syntaxhighlight>
 
Each leaf is associated with a number of these <code>dleafambientlighting_t</code> structures, each of which contains a cube of ambient light data at the position specified by the <code>x</code>, <code>y</code>, and <code>z</code> members. These coordinates are stored as fractions of the leaf's bounding box, i.e. a value of 0 for <code>x</code> represents the westmost position in the leaf, a value of 255 represents the eastmost position in the leaf, and a value of 128 represents the center of the leaf.
 
The lighting data for each sample is represented by a <code>CompressedLightCube</code> structure, which is simply an array of 6 <code>ColorRGBExp32</code> structures as described in the previous section:
 
<syntaxhighlight lang="cpp">struct CompressedLightCube
{
ColorRGBExp32 m_Color[6];
};</syntaxhighlight>
 
Each lighting sample in the array corresponds to the amount of lighting being received from each cardinal direction in 3D space.
 
At compile time, [[VRAD]] will randomly generate a number of ambient light sample positions for each leaf, and store the lighting information at each sample point in a <code>dleafambientlighting_t</code> structure. To associate each leaf with its collection of ambient samples, the ambient lighting index lumps '''(Lumps 51 and 52)''' are used. Lump 51 stores ambient light index information for HDR, and lump 52 stores the same information for LDR.
 
The ambient light index lumps are arrays of <code>dleafambientindex_t</code> structures:
 
<syntaxhighlight lang="cpp">struct dleafambientindex_t
{
unsigned short ambientSampleCount;
unsigned short firstAmbientSample;
};</syntaxhighlight>
 
The N<sup>th</sup> <code>dleafambientindex_t</code> structure in this array always corresponds to the N<sup>th</sup> leaf in the <code>dleaf_t</code> array. The <code>ambientSampleCount</code> field is the number of ambient samples associated with the corresponding leaf, and <code>firstAmbientSample</code> is an index into the ambient lighting array, referring to the first sample in the array that is associated with this leaf.
 
=== Occlusion ===
 
The occlusion lump '''(Lump 9)''' contains the polygon geometry and some flags used by [[func_occluder]] entities. Unlike other brush entities, func_occluders don't use the 'model' key in the [[#Entity|entity lump]]. Instead, the brushes are split from the entities during the compile process and numeric occluder keys are assigned as 'occludernum'. Brush sides textured with <code>tools/[[tool textures#occluder|toolsoccluder]]</code> or <code>tools/[[tool textures#trigger|toolstrigger]]</code> are then stored together with the occluder keys and some additional info in this lump.
 
The lump is divided into three parts and begins with a integer value with the total number of occluders, followed by an array of <code>doccluderdata_t</code> fields of the same size.
The next part begins with another integer value, this time for the total number of occluder polygons, as well as an array of <code>doccluderpolydata_t</code> fields of equal size. Part three begins with another integer value for the amount of occluder polygon vertices, followed by an array of integer values for the vertex indices, again of the same size.
 
<syntaxhighlight lang="cpp">struct doccluder_t
{
int count;
doccluderdata_t data[count];
int polyDataCount;
doccluderpolydata_t polyData[polyDataCount];
int vertexIndexCount;
int vertexIndices[vertexIndexCount];
};</syntaxhighlight>
 
The <code>doccluderdata_t</code> structure contains flags and dimensions of the occluder, as well as the area where it remains.
<code>firstpoly</code> is the first index into the <code>doccluderpolydata_t</code> with a total of <code>polycount</code> entries.
 
<syntaxhighlight lang="cpp">struct doccluderdata_t
{
int flags;
int firstpoly; // index into doccluderpolys
int polycount; // amount of polygons
Vector mins;         // minima of all vertices
Vector maxs;         // maxima of all vertices
// since v1
int area;
};</syntaxhighlight>
 
Occluder polygons are stored in the <code>doccluderpolydata_t</code> structure and contain the <code>firstvertexindex</code> field, which is the first index into the vertex array of the occluder, which are again indices for the vertex array of the [[#Vertex|vertex lump]] '''(Lump 3)'''. The total number of vertex indices is stored in <code>vertexcount</code>.
 
<syntaxhighlight lang="cpp">struct doccluderpolydata_t
{
int firstvertexindex; // index into doccludervertindices
int vertexcount; // amount of vertex indices
int planenum;
};</syntaxhighlight>
 
=== Physics ===
The physcollide lump '''(Lump 29)''' contains physics data for the world.
 
The lump consists of a sequence of ''models'', where each model consists of:
 
* a dphysmodel_t header
 
    struct dphysmodel_t
    {
        int modelIndex;  // Perhaps the index of the model to which this physics model applies?
        int dataSize;    // Total size of the collision data sections
        int keydataSize; // Size of the text section
        int solidCount;  // Number of collision data sections
    };
 
* a series of collision data sections (including compactsurfaceheader_t header)
* a text section
 
The lump is terminated by a dphysmodel_t structure with the modelIndex set to -1.
 
The last two parts appear to be identical to the [[PHY]] file format, which means their exact contents are unknown.
Note that the compactsurfaceheader_t structure contains the data size of each collision data section (including the rest of the header), so the lump can be parsed as follows:
 
    while(true) {
        header = readHeader();
        if(header.modelIndex == -1)
            break;
       
        for(int k = 0; k < header.solidCount; k++) {
            size = read4ByteInt();
            collisionData = readBytes(size);
        }
       
        textData = readBytes(header.keydataSize);
    }
 
=== Worldlights ===
The '''worldlights''' lumps ('''Lump 15''' for LDR and '''Lump 54''' for HDR) contain information on each static light entity in the world, and are used to provide direct lighting for dynamically lit entities. The data is generated by VRAD from the entity lump. VRAD uses information from these lumps instead of referring to light entities from '''Entity lump'''.
Both lumps have the same data structure containing contiguous '''dworldlight_t''' structs:
<syntaxhighlight lang="cpp">// lights that were used to illuminate the world
enum emittype_t
{
emit_surface, // 90 degree spotlight
emit_point, // simple point light source
emit_spotlight, // spotlight with penumbra
emit_skylight, // directional light with no falloff (surface must trace to SKY texture)
emit_quakelight, // linear falloff, non-lambertian
emit_skyambient, // spherical light source with no falloff (surface must trace to SKY texture)
};
 
// Flags for dworldlight_t::flags
#define DWL_FLAGS_INAMBIENTCUBE 0x0001 // This says that the light was put into the per-leaf ambient cubes.
 
struct dworldlight_t
{
Vector origin;
Vector intensity;
Vector normal; // for surfaces and spotlights
int cluster;
emittype_t type; //int, 0 - surface, 1 - point, 2 - spot, 3 - sky, 4 - quakelight, 5 - sky ambient
int style;
float stopdot; // start of penumbra for emit_spotlight
float stopdot2; // end of penumbra for emit_spotlight
float exponent; //
float radius; // cutoff distance
// falloff for emit_spotlight + emit_point:
// 1 / (constant_attn + linear_attn * dist + quadratic_attn * dist^2)
float constant_attn;
float linear_attn;
float quadratic_attn;
int flags; // Uses a combination of the DWL_FLAGS_ defines.
int texinfo; //
int owner; // entity that this light it relative to
};</syntaxhighlight>
The size of this struct is 88 bytes, therefore a number of entries in the lump can be determined by dividing the '''lump length''' by 88.
 
Up to two lights from this lump may influence dynamic lighting.
 
{{Confirm|Does the worldlights lump's dynamic lighting work by creating [[elight]]s on the MDL?}}
{{note|If VRAD tries to exceed MAX_MAP_WORLDLIGHTS, it will spit out a [[too many lights]] error. This is a compiler limit, not an engine one, and can be bypassed with a custom RAD compiler.}}


=== Other ===
=== Other ===
{{todo|Some of these information are likely guesses and need further research.}}


There are nineteen other lumps defined in the HL2 BSP file format that have not yet been covered. These lumps were not needed for the creation of a decompiler, and so I have not researched them or their formats. There are also four lumps only present in version 20 BSP files. I will give general information and likely guesses to the content of these lumps.
*The Areas lump '''(Lump 20)''' references the Areaportals lump '''(Lump 21)''' and is used with func_areaportal and func_areaportalwindow entities to define sections of the map that can be switched to render or not render.


The Occlusion lump '''(Lump 9)''' contains data on func_occluder entities which are switchable entities that block the drawing of visible entities behind them.
*The Portals '''(Lump 22)''', Clusters '''(Lump 23)''', PortalVerts '''(Lump 24)''', ClusterPortals '''(Lump 25)''', and ClipPortalVerts '''(Lump 41)''' lumps are used by the VVIS phase of the compile to ascertain which clusters can see which other clusters. A cluster is a player-enterable leaf volume in the map (see above). A "portal" is a polygon boundary between a cluster or leaf and an adjacent cluster or leaf. Most of this information is also used by the VRAD program to calculate static lighting, and then is removed from the bsp file.


The Worldlights lump '''(Lump 15)''' contains information on each static light entity in the world, and seems to be used to provide semi-dynamic lighting for moving entities.
*PhysCollide Lumps '''(Lump 29)''' and PhysCollideSurface '''(Lump 49)''' lumps seem to be related to the physical simulation of entity collisions in the game engine.


The Areas lump '''(Lump 20)''' references the Areaportals lump '''(Lump 21)''' and is used with func_areaportal and func_areaportalwindow entities to define sections of the map that can be switched to render or not render.
*The VertNormal '''(Lump 30)''' and VertNormalIndices '''(Lump 31)''' lumps may be related to smoothing of lightmaps on faces.


The Portals '''(Lump 22)''', Clusters '''(Lump 23)''', PortalVerts '''(Lump 24)''', ClusterPortals '''(Lump 25)''', and ClipPortalVerts '''(Lump 41)''' lumps are used by the VVIS phase of the compile to ascertain which clusters can see which other clusters. A cluster is a player-enterable leaf volume in the map (see above). A "portal" is a polygon boundary between a cluster or leaf and an adjacent cluster or leaf. Most of this information is also used by the VRAD program to calculate static lighting, and then is removed from the bsp file.
*The FaceMacroTextureInfo lump '''(Lump 47)''' is a short array containing the same number of members as the number of faces in the map. If the entry for a face contains anything other than -1 (0xFFFF), it is an index of a texture name in the TexDataStringTable. In VRAD, the corresponding texture is mapped onto the world extents, and used to modulate the lightmaps of that face. There is also a base macro texture (located at <code>materials/macro/''mapname''/base.vtf</code>) that is applied to all faces if found. Only maps in VTMB seem to make any use of macro textures.


PhysCollide Lumps '''(Lump 29)''' and PhysCollideSurface '''(Lump 49)''' lumps seem to be related to the physical simulation of entity collisions in the game engine.
*LeafWaterData '''(Lump 36)''' and LeafMinDistToWater '''(Lump 46)''' lumps may be used to determine player position with respect to water volumes.


The VertNormal '''(Lump 30)''' and VertNormalIndices '''(Lump 31)''' lumps may be related to smoothing of lightmaps on faces.
*The Primitives '''(Lump 37)''', PrimVerts '''(Lump 38)''' and PrimIndices '''(Lump 39)''' lumps are used in reference to "non-polygonal primitives". They are also sometimes called "waterstrips", "waterverts" and "waterindices" in the SDK Source, since they were originally only used to subdivide water meshes. They are now used to prevent the appearance of cracks between adjacent faces, if the face edges contain a "T-junction" (a vertex collinearly between two other vertices). The PrimIndices lump defines a set of triangles between face vertices, that tessellate the face. They are referenced from the Primatives lump, which is in turn referenced by the face lump data. Current maps do not seem to use the PrimVerts lump at all. ([http://web.archive.org/web/20071110230828/http://www.chatbear.com/board.plm?a=viewthread&b=4991&t=137,1118051039,3296&s=0&id=862840 Ref.])


The FaceMacroTextureInfo lump '''(Lump 47)''' is a short array containing the same number of members as the number of faces in the map. If the entry for a face contains anything other than -1 (0xFFFF), it is an index of a texture name in the TexDataStringTable. In VRAD, the corresponding texture is mapped onto the world extents, and used to modulate the lightmaps of that face. There is also a base macro texture (located at <code>materials/macro/''mapname''/base.vtf</code>) that is applied to all faces if found. Only maps in VTMB seem to make any use of macro textures.
*Version 20 files containing HDR lighting information have four extra lumps, the contents of which are currently uncertain. Lump 53 is always the same size as the standard lighting lump '''(Lump 8)''' and probably contains higher-precision data for each lightmap sample. Lump 54 is the same size as the worldlight lump '''(Lump 15)''' and presumably contains HDR-related data for each light entity.


LeafWaterData '''(Lump 36)''' and LeafMinDistToWater '''(Lump 46)''' lumps may be used to determine player position with respect to water volumes.
== CRC32 checksum ==
Despite not being a part of BSP format, a CRC32 checksum of a BSP map plays an important role. It is used to compare clientside and serverside version of a map file. In case of the checksum being different an error is thrown ("Your map ''map_name.bsp'' differs from the server").


The Primitives '''(Lump 37)''', PrimVerts '''(Lump 38)''' and PrimIndices '''(Lump 39)''' lumps are used in reference to "non-polygonal primitives". They are also sometimes called "waterstrips", "waterverts" and "waterindices" in the SDK Source, since they were originally only used to subdivide water meshes. They are now used to prevent the appearance of cracks between adjacent faces, if the face edges contain a "T-junction" (a vertex collinearly between two other vertices). The PrimIndices lump defines a set of triangles between face vertices, that tessellate the face. They are referenced from the Primatives lump, which is in turn referenced by the face lump data. Current maps do not seem to use the PrimVerts lump at all. ([http://www.chatbear.com/board.plm?a=viewthread&b=4991&t=137,1118051039,3296&s=0&id=862840#3 Ref.])
BSP map checksum is not merely a checksum of an entire file - instead, it is calculated based only on BSP lumps, excluding the Entity lump '''(Lump 0)'''. BSP header does not contribute to the result, additionaly, because the lump bytes are fed into CRC algorithm in an order of their indices (Entity lump, index 0 - first, Planes lump, index 1 - second etc.) the order they are stored in a file does not affect the final result (except the Game lump '''(Lump 35)''' as the <code>fileofs</code> of Game lumps is relative to the start of the file).


Version 20 files containing HDR lighting information have four extra lumps, the contents of which are currently uncertain. Lump 53 is always the same size as the standard lighting lump '''(Lump 8)''' and probably contains higher-precision data for each lightmap sample. Lump 54 is the same size as the worldlight lump '''(Lump 15)''' and presumably contains HDR-related data for each light entity. Lumps 55 and 56 both seem to be 24-byte records (possibly CompressedLightCube structures) with the same count as the number of leaves in the map. They are probably thus HDR-related per-leaf lighting information.
Java code for calculating the checksum ([https://github.com/jakgor471/BSPEntSpy/blob/main/src/bspentspy/SourceBSPFile.java#L1097 BSPEntSpy, SourceBSPFile.java]):
<syntaxhighlight lang="java">public long computeChecksum() throws IOException {
crcAlgo.reset();
byte[] block = new byte[10240];
for(GenericLump lump : lumps) {
if(lump.index == ENTLUMP)
continue;
bspfile.seek(lump.offset);
int toRead = (int)lump.length;
while(toRead > 0) {
int read = bspfile.read(block, 0, (int)Math.min(toRead, block.length));
crcAlgo.update(block, 0, read);
toRead -= read;
}
}
return crcAlgo.getValue();
}
</syntaxhighlight>
{{references|1=
{{ref2|cite id=1|[https://github.com/MRVN-Radiant/MRVN-Radiant/blob/main/tools/remap/source/games.h MRVN-Radient source code ({{code|games.h}})]}}
{{ref2|cite id=2|[https://github.com/snake-biscuits/bsp_regen/blob/main/src/bsp.hpp bsp_regen source code ({{code|bsp.hpp}})]
<syntaxhighlight lang="cpp">bool is_valid() {
        if (header_->magic == MAGIC_rBSP) {
            switch (header_->version) {
                case 29:  // Titanfall
                case 36:  // Titanfall 2 [PS4] (Tech Test)
                case 37:  // Titanfall 2
                    return (header_->_127 == 127);
                default:
                    return false;
            }
</syntaxhighlight>
}}
}}


[[Category:Technical]]
[[Category:Source]]
[[Category:File formats]]

Latest revision as of 03:41, 13 October 2025

English (en)Esperanto (eo)Русский (ru)Translate (Translate)
<BSP
edit
This article is based on Rof's Icon-Wayback.png The Source Engine BSP File Format (from October 2005), retrieved from Geocities before the service shut down. A mirror to the original version can be found here.
Todo: Some versions of BSPZIP (included in Team Fortress 2 Counter-Strike: Source Day of Defeat: Source Half-Life 2: Deathmatch Half-Life Deathmatch: Source Strata Source Jabroni Brawl: Episode 3) support LZMA compression of the lumps and packed files (supported by Source 2013 Multiplayer Team Fortress 2 branch Strata Source Jabroni Brawl: Episode 3 and possibly more); update this page to reflect how this affects the format.

Introduction

This document describes the structure of the BSP file format used by the Source engine. The format is more closely derived from the BSP format used in Quake II than that of GoldSrc (both of which are derived from Quake I's BSP format). Because of this, Max McGuire's article, Quake 2 BSP File Format is also of invaluable help in understanding the overall structure of the format and the parts of it that have remained the same or similar to its predecessors.

This document is an extension of notes made by Rof during the writing of their Half-Life 2 Half-Life 2 BSP file decompiler, VMEX. It therefore focuses on those parts of the format necessary to perform map decompilation (conversion of the BSP file back into a VMF file which can be loaded by the Hammer map editor).

Most of the information in this document comes from the Max McGuire article referenced above, from the source code included in the Source SDK (particularly the C header file public/bspfile.h), and from Rof's own experimentation during the writing of VMEX.

A certain familiarity with C/C++, geometry, and Source mapping terms is assumed on the part of the reader. Code (mostly C structures) is given in a fixed width font. Sometimes the structures as shown are modified from their actual definitions in the SDK header files, for reasons of clarity and consistency.

Overview

The BSP file contains the vast majority of the information needed by the Source engine to render and play a map. This includes the geometry of all the polygons in the level; references to the names and orientation of the textures to be drawn on those polygons; the data used to simulate the physical behaviour of the player and other items during the game; the location and properties of all brush-based, model (prop) based, and non-visible (logical) entities in the map; and the BSP tree and visibility table used to locate the player location in the map geometry and to render the visible map as efficiently as possible. Optionally, the map file can also contain any custom textures and models used on the level, embedded inside the map's Pakfile lump (see below).

Information not stored in the BSP file includes the map description text displayed by multiplayer games (such as Counter-Strike: Source Counter-Strike: Source or Half-Life 2: Deathmatch Half-Life 2: Deathmatch) after loading the map (stored in the file mapname.txt) and the AI navigation file used by non-player characters (NPCs) which need to navigate the map (stored in the file mapname.nav). Because of the way the Source engine file system works, these external files may also be embedded in the BSP file's Pakfile lump, though usually they are not.

Historically, map files were stored in the game's corresponding Steam Game Cache File (GCF), but these are no longer used since 2013. In current versions of all of Valve's games, maps are stored directly in the OS file system. Rarely, such as in some third-party games or mods (especially Black Mesa Black Mesa and most Source 2004/Source 2006 games that have been updated with SteamPipe), maps may be stored in VPK files; these can be extracted using GCFScape GCFScape or the newer VPKEdit VPKEdit.

The data in the BSP file is stored according to the default endianness of the CPU it is intended to be loaded on: PC (x86) and Nintendo Switch (ARM) use little-endian, while PlayStation 3 and Xbox 360 (both PowerPC) use big-endian. Byte-swapping is required when loading a little-endian file on a big-endian format platform such as Java and vice versa.

Warning.pngWarning:There are many conflicting versions of BSP! Their compatibility isn't clear as even BSPs with the same Version Number can have underlying differences that will render them incompatible with games other than their source. Therefore it is a good rule of thumb to always try and stick with the intended game and its compile tools when messing around with BSPs.

BSP file header

The BSP file starts with a header. This structure identifies the file as a Valve Source Engine BSP file, identifies the version of the format, and is then followed by a directory listing of the location, length, and version of up to 64 subsections of the file, known as lumps, that store different parts of the map data. Finally, the map revision is given.

The structure of the header is given in the SDK's public/bspfile.h header file, a file which will be referencing extensively throughout this document. The header is 1036 bytes long in total:

Note.pngNote:This struct has been renamed to BSPHeader_t in Alien Swarm Alien Swarm.
struct dheader_t
{
	int     ident;                  // BSP file identifier
	int     version;                // BSP file version
	lump_t  lumps[HEADER_LUMPS];    // lump directory array
	int     mapRevision;            // the map's revision (iteration, version) number
};

Where ident is a 4-byte magic number defined as:

// little-endian "VBSP"   0x50534256
#define IDBSPHEADER	(('P'<<24)+('S'<<16)+('B'<<8)+'V')

The first four bytes of the file are thus always VBSP (in ASCII). These bytes identify the file as a Valve BSP file; other BSP file formats use a different magic number (such as for id Software's Quake engine games, which start with IBSP). The GoldSrc GoldSrc BSP format does not use any magic number at all. The order of the magic number can also be used to determine the endianness of the file: VBSP is used for little-endian and PSBV for big-endian.

The second integer is the version of the BSP file format (BSPVERSION); for Source games, it ranges from 19 to 20 (21 since Left 4 Dead 2 Left 4 Dead 2) with the exception of VTMB, which uses an earlier version of the format, 17 (see table below). Note that BSP file formats for other engines (HL1, Quake series, etc.) use entirely different version number ranges.

Versions

This table gives an overview over the different BSP versions being used in some games/software based on the Source Source Engine.

Todo: add Day of Infamy Day of Infamy, Apex Legends Apex Legends and possibly others, as long they have modified lumps. See Template:GameOnSource.
Version Games/Software Notes
17 Vampire: The Masquerade – Bloodlines Vampire: The Masquerade – Bloodlines modified: dface_t
17-18 Half-Life 2 Half-Life 2 (Beta) leaked beta
19 SiN Episodes SiN Episodes
Half-Life 2 (Xbox) Half-Life 2 (Xbox) Custom lightmap format
18-19 Half-Life: Source Half-Life: Source Only c4a1y.bsp is 18. The rest are 19.
Tools create v20 BSPs.
19-20 Half-Life 2 Half-Life 2 19 when released, partially 20 with engine update introduced in Lost Coast (d1_trainstation_01 only), and later more maps are v20 since Source 2007/2009 (Orange Box) up to Source 2013.
Console port (Xbox 360 Xbox 360 and PlayStation 3 PlayStation 3) uses BSP 20 with LZMA compression.
All maps are now 20 after 20th anniversary update.
Tools create v20 BSPs.
Half-Life 2: Deathmatch Half-Life 2: Deathmatch 19 when released, partially 20 since Source 2009 (Orange Box) update in 2010 (for Counter-Strike: Source, Day of Defeat: Source), and Source 2013 (since 2013) for Half-Life Deathmatch: Source. Supports LZMA compressed BSP Files.
Half-Life 2: Deathmatch All maps are now 20 after 20th anniversary update.
Tools create same format as Team Fortress 2 Team Fortress 2.
Half-Life Deathmatch: Source Half-Life Deathmatch: Source
Counter-Strike: Source Counter-Strike: Source
Day of Defeat: Source Day of Defeat: Source
20
Source SDK Base 2006 Source SDK Base 2006
Source SDK Base 2007 Source SDK Base 2007
Source SDK Base 2013 - Singleplayer Source SDK Base 2013 - Singleplayer
HDR lightmaps support. Tool creates v20 BSPs.
Half-Life 2: Lost Coast Half-Life 2: Lost Coast HDR lightmaps support
Half-Life 2: Episode One Half-Life 2: Episode One
Half-Life 2: Episode Two Half-Life 2: Episode Two
Garry's Mod Garry's Mod modified supports BSP Version 21, supports LZMA compressed BSP Files
Team Fortress 2 Team Fortress 2
Source SDK Base 2013 - Multiplayer Source SDK Base 2013 - Multiplayer
modified ( newly compiled maps ): StaticPropLump_t (version = 10, stores prop lightmap resolution info and expanded bitflags); LZMA compressed game lumps, entity info, and pakfiles; ambient lighting lump (additional ambient cubes per visleaf)
Console port of Team Fortress 2 (Xbox 360 Xbox 360 and PlayStation 3 PlayStation 3) support LZMA compression, but maps are not compressed.
Portal Portal Same as Half-Life 2 (PC). Console port (Xbox 360 Xbox 360 and PlayStation 3 PlayStation 3) support LZMA compression, but maps are not compressed.
Left 4 Dead Left 4 Dead modified: StaticPropLump_t ( version = 8 ) and dworldlight_t
Zeno Clash Zeno Clash modified: StaticPropLump_t ( version = 7 )
Dark Messiah of Might and Magic Dark Messiah of Might and Magic modified: dheader_t, StaticPropLump_t, texinfo_t, dgamelump_t, dmodel_t
Vindictus Vindictus many modified structs
The Ship: Murder Party The Ship: Murder Party modified: StaticPropLump_t
Bloody Good Time Bloody Good Time modified: StaticPropLump_t
Black Mesa Black Mesa modified: StaticPropLump_t ( version = 11 ) (commercial/Steam release only)
Fairy Tale Busters Fairy Tale Busters (open beta version) modified: StaticPropLump_t ( version = 7 ), with a StaticPropLump_t ( version = 10 ) alike fields order
Eternal Silence Eternal Silence
NEOTOKYO° NEOTOKYO°
Postal III Postal III
Half-Life 2: Update Half-Life 2: Update
INSURGENCY: Modern Infantry Combat INSURGENCY: Modern Infantry Combat
Portal with RTX Portal with RTX
Tactical Intervention Tactical Intervention (Retail/FIX Launcher Release) 256-bit XOR Encryption (not just limited to BSPs)
E.Y.E: Divine Cybermancy E.Y.E: Divine Cybermancy
19-22 Jabroni Brawl: Episode 3 Jabroni Brawl: Episode 3 Map Compile Tools produce maps of the same format as Counter-Strike: Global Offensive CS:GO. Also supports LZMA-compressed game lumps, entity info, and pakfiles
Note.pngNote:BSP verison 22 support is seemingly only for Tactical Intervention Tactical Intervention, any other game with BSP version 22 is untested and most likely unsupported.
20-21 Source Filmmaker Source Filmmaker
21 Left 4 Dead 2 Left 4 Dead 2 modified: lump_t, old dbrushside_t
Alien Swarm Alien Swarm
Portal 2 Portal 2 modified: StaticPropLump_t ( version = 9 ), dbrushside_t and other structs
Portal 2 Sixense Perceptual Pack Portal 2 Sixense Perceptual Pack
Counter-Strike: Global Offensive Counter-Strike: Global Offensive modified: StaticPropLump_t ( version = 11 )
Dear Esther Dear Esther (commercial release) modified: StaticPropLump_t
Insurgency Insurgency (commercial release)
The Stanley Parable The Stanley Parable (commercial release)
The Beginner's Guide The Beginner's Guide
Dino D-Day Dino D-Day
Nuclear Dawn Nuclear Dawn
Revelations 2012 Revelations 2012
Alien Swarm: Reactive Drop Alien Swarm: Reactive Drop
Lambda Wars Lambda Wars
22 INFRA INFRA
Tactical Intervention Tactical Intervention (Steam Release) Derived from Portal 2 Portal 2 version 21, with lump #24 being used to store client side entities ( similar to lump #0 )
Dota 2 Dota 2 (pre-Reborn/original release) early betas, modified: dbrushside_t, ddispinfo_t
23 modified: dbrushside_t, ddispinfo_t, doverlay_t
25 Strata Source Strata Source In development
27 Contagion Contagion
29[1] Titanfall Titanfall Heavily modified from Portal 2 - see Game-Specific page - Titanfall section.
36[2]
37[1]
Titanfall 2 Titanfall 2 Heavily modified from Titanfall
BSP 36 is only used in PS4 Beta (Tech Test), release version use BSP 37.
Todo: List changes
47[1] Apex Legends Apex Legends Heavily modified from Titanfall 2, with expanded map sizes.
Todo: List changes
100 Counter-Strike Online 2 Counter-Strike Online 2

For more details on game-specific BSP formats, see Source BSP File Format/Game-Specific.

Lump structure

Then follows an array of 16-byte lump_t structures. HEADER_LUMPS is defined as 64, so there are 64 entries. However, depending on the game and version, some lumps can be undefined or empty.

Each lump_t is defined in bspfile.h:

struct lump_t
{
	int    fileofs;      // offset into file (bytes)
	int    filelen;      // length of lump (bytes)
	int    version;      // lump format version
	char   fourCC[4];    // lump ident code
};

The first two integers contain the byte offset (from the beginning of the bsp file) and byte length of that lump's data block; an integer defining the version number of the format of that lump (usually zero), and then a four byte identifier that is usually 0, 0, 0, 0. For compressed lumps, the fourCC contains the uncompressed lump data size in integer form (see section Lump compression for details). Unused members of the lump_t array (those that have no data to point to) have all elements set to zero.

Todo: Some BSPs (tested in Portal 2) have lumps without data (zero length) but still with offsets/versions set. Expand on this.

Lump offsets (and their corresponding data lumps) are always rounded up to the nearest 4-byte boundary, though the lump length may not be.

Lump types

The type of data pointed to by the lump_t array is defined by its position in the array; for example, the first lump in the array (Lump 0) is always the BSP file's entity data (see below). The actual location of the data in the BSP file is defined by the offset and length entries for that lump, and does not need to be in any particular order in the file; for example, the entity data is usually stored towards the end of the BSP file despite being first in the lump array. The array of lump_t headers is therefore a directory of the actual lump data, which may be located anywhere else in the file.

Warning.pngWarning:Although the format does not require the lumps in a specific order, game lumps within LUMP_GAME_LUMP are denoted by their offset relative to the beginning of the BSP (except on console versions of Portal 2 Portal 2). When moving this lump these offsets must be recalculated.

The order of the lumps in the array is defined as:

Todo: Postal III Postal III's Compile Tools produce BSPs with LUMP_FACES with 346 surfaces, figure out where to put that in this table.
Index Engine Name Purpose
0 Source 2004 Source 2004 LUMP_ENTITIES Map entities
1 Source 2004 Source 2004 LUMP_PLANES Plane array
2 Source 2004 Source 2004 LUMP_TEXDATA Index to texture names
3 Source 2004 Source 2004 LUMP_VERTEXES Vertex array
4 Source 2004 Source 2004 LUMP_VISIBILITY Compressed visibility bit arrays
5 Source 2004 Source 2004 LUMP_NODES BSP tree nodes
6 Source 2004 Source 2004 LUMP_TEXINFO Face texture array
7 Source 2004 Source 2004 LUMP_FACES Face array
8 Source 2004 Source 2004 LUMP_LIGHTING Lightmap samples
9 Source 2004 Source 2004 LUMP_OCCLUSION Occlusion polygons and vertices
10 Source 2004 Source 2004 LUMP_LEAFS BSP tree leaf nodes
11 Source 2007 Source 2007/2009 LUMP_FACEIDS Correlates between dfaces and Hammer face IDs. Also used as random seed for detail prop placement.
12 Source 2004 Source 2004 LUMP_EDGES Edge array
13 Source 2004 Source 2004 LUMP_SURFEDGES Index of edges
14 Source 2004 Source 2004 LUMP_MODELS Brush models (geometry of brush entities)
15 Source 2004 Source 2004 LUMP_WORLDLIGHTS Internal world lights converted from the entity lump
16 Source 2004 Source 2004 LUMP_LEAFFACES Index to faces in each leaf
17 Source 2004 Source 2004 LUMP_LEAFBRUSHES Index to brushes in each leaf
18 Source 2004 Source 2004 LUMP_BRUSHES Brush array
19 Source 2004 Source 2004 LUMP_BRUSHSIDES Brushside array
20 Source 2004 Source 2004 LUMP_AREAS Area array
21 Source 2004 Source 2004 LUMP_AREAPORTALS Portals between areas
22 Source 2004 Source 2004 LUMP_PORTALS
Confirm:Polygons defining the boundary between adjacent leaves?
Source 2007 Source 2007/2009 LUMP_UNUSED0 Unused
Left 4 Dead engine branch Source (L4D2 Branch) LUMP_PROPCOLLISION Static props convex hull lists
23 Source 2004 Source 2004 LUMP_CLUSTERS Leaves that are enterable by the player
Source 2007 Source 2007/2009 LUMP_UNUSED1 Unused
Left 4 Dead engine branch Source (L4D2 Branch) LUMP_PROPHULLS Static prop convex hulls
24 Source 2004 Source 2004 LUMP_PORTALVERTS Vertices of portal polygons
Source 2007 Source 2007/2009 LUMP_UNUSED2 Unused
Icon-TacInt256.png Source (TacInt branch) LUMP_FAKEENTITIES Used to store client side entities ( Similar to Lump #0 )
Left 4 Dead engine branch Source (L4D2 Branch) LUMP_PROPHULLVERTS Static prop collision vertices
25 Source 2004 Source 2004 LUMP_CLUSTERPORTALS
Confirm:Polygons defining the boundary between adjacent clusters?
Source 2007 Source 2007/2009 LUMP_UNUSED3 Unused
Left 4 Dead engine branch Source (L4D2 Branch) LUMP_PROPTRIS Static prop per hull triangle index start/count
26 Source 2004 Source 2004 LUMP_DISPINFO Displacement surface array
27 Source 2004 Source 2004 LUMP_ORIGINALFACES Brush faces array before splitting
28 Source 2007 Source 2007/2009 LUMP_PHYSDISP Displacement physics collision data
29 Source 2004 Source 2004 LUMP_PHYSCOLLIDE Physics collision data
30 Source 2004 Source 2004 LUMP_VERTNORMALS Face plane normals
31 Source 2004 Source 2004 LUMP_VERTNORMALINDICES Face plane normal index array
32 Source 2004 Source 2004 LUMP_DISP_LIGHTMAP_ALPHAS Displacement lightmap alphas (unused/empty since Source 2006)
33 Source 2004 Source 2004 LUMP_DISP_VERTS Vertices of displacement surface meshes
34 Source 2004 Source 2004 LUMP_DISP_LIGHTMAP_SAMPLE_POSITIONS Displacement lightmap sample positions
35 Source 2004 Source 2004 LUMP_GAME_LUMP Game-specific data lump
36 Source 2004 Source 2004 LUMP_LEAFWATERDATA Data for leaf nodes that are inside water
37 Source 2004 Source 2004 LUMP_PRIMITIVES Water polygon data
38 Source 2004 Source 2004 LUMP_PRIMVERTS Water polygon vertices
39 Source 2004 Source 2004 LUMP_PRIMINDICES Water polygon vertex index array
40 Source 2004 Source 2004 LUMP_PAKFILE Embedded uncompressed or LZMA-compressed Zip-format file
41 Source 2004 Source 2004 LUMP_CLIPPORTALVERTS Clipped portal polygon vertices
42 Source 2004 Source 2004 LUMP_CUBEMAPS env_cubemap location array
43 Source 2004 Source 2004 LUMP_TEXDATA_STRING_DATA Texture name data
44 Source 2004 Source 2004 LUMP_TEXDATA_STRING_TABLE Index array into texdata string data
45 Source 2004 Source 2004 LUMP_OVERLAYS info_overlay data array
46 Source 2004 Source 2004 LUMP_LEAFMINDISTTOWATER Distance from leaves to water
47 Source 2004 Source 2004 LUMP_FACE_MACRO_TEXTURE_INFO Macro texture info for faces
48 Source 2004 Source 2004 LUMP_DISP_TRIS Displacement surface triangles
49 Source 2004 Source 2004 LUMP_PHYSCOLLIDESURFACE Compressed win32-specific Havok terrain surface collision data. Deprecated and no longer used.
Left 4 Dead engine branch Source (L4D2 Branch) LUMP_PROP_BLOB Static prop triangle and string data
50 Source 2006 Source 2006 LUMP_WATEROVERLAYS Tied to any entity that uses the info_overlay_transition helper in FGD
51 Source 2006 Source 2006 LUMP_LIGHTMAPPAGES Alternate lightdata implementation for Xbox
Source 2007 Source 2007/2009 LUMP_LEAF_AMBIENT_INDEX_HDR Index of LUMP_LEAF_AMBIENT_LIGHTING_HDR
52 Source 2006 Source 2006 LUMP_LIGHTMAPPAGEINFOS Alternate lightdata indices for Xbox
Source 2007 Source 2007/2009 LUMP_LEAF_AMBIENT_INDEX Index of LUMP_LEAF_AMBIENT_LIGHTING
53 Source 2004 Source 2004 (since Half-Life 2: Lost Coast)
Source 2006 Source 2006
LUMP_LIGHTING_HDR HDR lightmap samples
54 Source 2004 Source 2004 (since Half-Life 2: Lost Coast)
Source 2006 Source 2006
LUMP_WORLDLIGHTS_HDR Internal HDR world lights converted from the entity lump
55 Source 2004 Source 2004 (since Half-Life 2: Lost Coast)
Source 2006 Source 2006
LUMP_LEAF_AMBIENT_LIGHTING_HDR Per-leaf ambient light samples (HDR)
56 Source 2004 Source 2004 (since Half-Life 2: Lost Coast)
Source 2006 Source 2006
LUMP_LEAF_AMBIENT_LIGHTING Per-leaf ambient light samples (LDR)
57 Source 2006 Source 2006 LUMP_XZIPPAKFILE XZip version of pak file for Xbox. Deprecated.
58 Source 2006 Source 2006 LUMP_FACES_HDR HDR maps may have different face data
59 Source 2006 Source 2006 LUMP_MAP_FLAGS Extended level-wide flags. Not present in all levels.
60 Source 2007 Source 2007/2009 LUMP_OVERLAY_FADES Fade distances for overlays
61 Left 4 Dead engine branch Source (L4D Branch) LUMP_OVERLAY_SYSTEM_LEVELS System level settings (min/max CPU & GPU to render this overlay)
62 Left 4 Dead engine branch Source (L4D2 Branch) LUMP_PHYSLEVEL Wikipedia icon PhysX model of the World Brush.
Confirm:The code that generates this only present within Left 4 Dead 2 Left 4 Dead 2 (only c5m1_waterfront_sndscape appears to contains data on this lump), unknown if any Engine Licensees with access to the L4D2 Branch also have access to it.
63 Alien Swarm engine branch Source (Alien Swarm branch) LUMP_DISP_MULTIBLEND Displacement multiblend info

Lumps 53-56 are only used in version 20+ BSP files. Lumps 22-25 are unused in version 20.

The structure of the data lumps for the known entries is described below. Many of the lumps are simple arrays of structures; however some are of variable length depending on their content. The maximum size or number of entries in each lump is also defined in the bspfile.h file, as MAX_MAP_*.

Finally, the header ends with an integer containing the map revision number. This number is based on the revision number of the map's vmf file (mapversion), which is increased each time the map is saved in the Hammer editor.

Immediately following the header is the first data lump. This can be any lump in the preceding list (pointed to using the offset field of that lump), though in practice the first data lump is Lump 1, the plane data array.

Lump compression

BSP files for console platforms such as PlayStation 3 and Xbox 360 usually have their lumps compressed with LZMA. In this case, the lump data starts with the following header (from public/tier1/lzmaDecoder.h), which is used in place of the standard 13-byte LZMA header:

struct lzma_header_t
{
	unsigned int    id;
	unsigned int    actualSize;         // always little endian
	unsigned int    lzmaSize;           // always little endian
	unsigned char   properties[5];
};

Where id is defined as:

// little-endian "LZMA"
#define LZMA_ID	(('A'<<24)|('M'<<16)|('Z'<<8)|('L'))

lzmaSize denotes the size (in bytes) of compressed data, it is equal to the size of a lump minus 17 bytes (lzma header). actualSize denotes the size of decompressed data. properties[5] field are used solely for LZMA decoding.

There are two special cases for compression: LUMP_PAKFILE is never compressed as a lump (the contents of the zip are compressed instead) and each of the game lumps in LUMP_GAME_LUMP are compressed individually. The compressed size of a game lump can be determined by subtracting the current game lump's offset with that of the next entry. For this reason, when game lumps are compressed the last game lump is always an empty dummy which only contains the offset.

If you wish to manipulate a compressed lump in an external tool, you need to convert the header mentioned above to the standard LZMA header format. It is equivalent to:

struct standard_lzma_header
{
	unsigned char		properties[5];
	unsigned long long	actualSize;      // 64-bit little endian
};

Lumps

Plane

The basis of the BSP geometry is defined by planes, which are used as splitting surfaces across the BSP tree structure.

The plane lump (Lump 1) is an array of dplane_t structures:

struct dplane_t
{
	Vector  normal;   // normal vector
	float   dist;     // distance from origin
	int     type;     // plane axis identifier
};

where the Vector type is a 3-vector defined as:

struct Vector
{
	float x;
	float y;
	float z;
};

Floats are 4 bytes long; there are thus 20 bytes per plane, and the plane lump should be a multiple of 20 bytes long.

The plane is represented by the element normal, a normal vector, which is a unit vector (length 1.0) perpendicular to the plane's surface. The position of the plane is given by dist, which is the distance from the map origin (0,0,0) to the nearest point on the plane.

Mathematically, the plane is described by the set of points (x, y, z) which satisfy the equation:

Ax + By + Cz = D

where A, B, and C are given by the components normal.x, normal.y and normal.z, and D is dist. Each plane is infinite in extent, and divides the whole of the map coordinate volume into three pieces, on the plane (F=0), in front of the plane (F>0), and behind the plane (F<0).

Note that planes have a particular orientation, corresponding to which side is considered "in front" of the plane, and which is "behind". The orientation of a plane can be flipped by negating the A, B, C, and D components.

The type member of the structure contains the axis that the plane is facing. It can be 0-5, with 0, 1, and 2 corresponding with X, Y, and Z respectively. Values of 3, 4 and 5 are used when planes are not along an axis, with each number corresponding to the axis it is closest to.

There can be up to 65536 planes in a map (MAX_MAP_PLANES).

Vertex

The vertex lump (Lump 3) is an array of coordinates of all the vertices (corners) of brushes in the map geometry. Each vertex is a Vector of 3 floats (x, y, and z), giving 12 bytes per vertex.

Note that vertices can be shared between faces, if the vertices coincide exactly.

There are a maximum of 65536 vertices in a map (MAX_MAP_VERTS).

Edge

The edge lump (Lump 12) is an array of dedge_t structures:

struct dedge_t
{
	unsigned short  v[2];  // vertex indices
};

Each edge is simply a pair of vertex indices (which index into the vertex lump array). The edge is defined as the straight line between the two vertices. Usually, the edge array is referenced through the Surfedge array (see below).

As for vertices, edges can be shared between adjacent faces. There is a limit of 256000 edges in a map (MAX_MAP_EDGES).

Surfedge

The Surfedge lump (Lump 13), presumable short for surface edge, is an array of (signed) integers. Surfedges are used to reference the edge array, in a somewhat complex way. The value in the surfedge array can be positive or negative. The absolute value of this number is an index into the edge array: if positive, it means the edge is defined from the first to the second vertex; if negative, from the second to the first vertex.

By this method, the Surfedge array allows edges to be referenced for a particular direction. (See the face lump entry below for more on why this is done).

There is a limit of 512000 (MAX_MAP_SURFEDGES) surfedges per map. Note that the number of surfedges is not necessarily the same as the number of edges in the map.

Face and original face

The face lump (Lump 7) contains the major geometry of the map, used by the game engine to render the viewpoint of the player. The face lump contains faces after they have undergone the BSP splitting process; they therefore do not directly correspond to the faces of brushes created in Hammer. Faces are always flat, convex polygons, though they can contain edges that are co-linear.

The face lump is one of the more complex structures of the map file. It is an array of dface_t entries, each 56 bytes long:

struct dface_t
{
	unsigned short  planenum;               // the plane number
	byte            side;                   // faces opposite to the node's plane direction
	byte            onNode;                 // 1 of on node, 0 if in leaf
	int             firstedge;              // index into surfedges
	short           numedges;               // number of surfedges
	short           texinfo;                // texture info
	short           dispinfo;               // displacement info
	short           surfaceFogVolumeID;     // ?
	byte            styles[4];              // switchable lighting info
	int             lightofs;               // offset into lightmap lump
	float           area;                   // face area in units^2
	int             LightmapTextureMinsInLuxels[2]; // texture lighting info
	int             LightmapTextureSizeInLuxels[2]; // texture lighting info
	int             origFace;               // original face this was split from
	unsigned short  numPrims;               // primitives
	unsigned short  firstPrimID;
	unsigned int    smoothingGroups;        // lightmap smoothing group
};

The first member planenum is the plane number, i.e., the index into the plane array that corresponds to the plane that is aligned with this face in the world. Side is zero if this plane faces in the same direction as the face (i.e. "out" of the face) or non-zero otherwise.

Animated x-ray view of d1_trainstation_01 with the three types of geometry data. View in full size here.

Firstedge is an index into the Surfedge array; this and the following numedges entries in the surfedge array define the edges of the face. As mentioned above, whether the value in the surfedge array is positive or negative indicates whether the corresponding pair of vertices listed in the Edge array should be traced from the first vertex to the second, or vice versa. The vertices which make up the face are thus referenced in clockwise order; when looking towards the face, each edge is traced in a clockwise direction. This makes rendering the faces easier, and allows quick culling of faces that face away from the viewpoint.

Texinfo is an index into the Texinfo array (see below), and represents the texture to be drawn on the face. Dispinfo is an index into the Dispinfo array is the face is a displacement surface (in which case, the face defines the boundaries of the surface); otherwise, it is -1. SurfaceFogVolumeID appears to be related to drawing fogging when the player's viewpoint is underwater or looking through water.

OrigFace is the index of the original face which was split to produce this face. NumPrims and firstPrimID are related to the drawing of "Non-polygonal primitives" (see below). The other members of the structure are used to reference face-lighting info (see the Lighting lump, below).

The face array is limited to 65536 (MAX_MAP_FACES) entries.

The original face lump (Lump 27) has the same structure as the face lump, but contains the array of faces before the BSP splitting process is done. These faces are therefore closer to the original brush faces present in the precompile map than the face array, and there are less of them. The origFace entry for all original faces is zero. The maximum size of the original face array is also 65536 entries.

Both the face and original face arrays are culled; that is, many faces present before compilation of the map (primarily those that face towards the "void" outside the map) are removed from the array.

Brush and brushside

The brush lump (Lump 18) contains all brushes that were present in the original VMF file before compiling. Unlike faces, brushes are constructive solid geometry (CSG) defined by planes instead of edges and vertices. It is the presence of the brush and brushside lumps in Source BSP files that makes decompiling them a much easier job than for GoldSrc files, which lacked this info. The lump is an array of 12-byte dbrush_t structures:

struct dbrush_t
{
	int    firstside;     // first brushside
	int    numsides;      // number of brushsides
	int    contents;      // contents flags
};

The first integer firstside is an index into the brushside array lump, this and the following numsides brushsides make up all the sides in this brush. The contents entry contains the contents flags, as defined in bspflags.h.

The brush array is limited to 8192 entries (MAX_MAP_BRUSHES).

The brushside lump (Lump 19) is an array of 8-byte structures:

struct dbrushside_t
{
	unsigned short  planenum;     // facing out of the leaf
	short           texinfo;      // texture info
	short           dispinfo;     // displacement info
#if BSPVERSION >= 21
	unsigned char   bevel;        // is the side a bevel plane?
	unsigned char   thin;         // is any edge less than step size?
#else
	short           bevel;        // is the side a bevel plane?
#endif
};

Planenum is an index info the plane array, giving the plane corresponding to this brushside. Texinfo and dispinfo are references into the texture and displacement info lumps. Bevel is zero for normal brush sides, but 1 if the side is a bevel plane (which seem to be used for collision detection). BSP version 21 added a "thin" marker for brushsides where any edge length is less than step size (16 units).

Unlike the face array, brushsides are not culled (removed) where they touch the void. Void-facing sides do however have their texinfo entry changed to the tools/toolsnodraw texture during the compile process. Note there is no direct way of linking brushes and brushsides and the corresponding face array entries which are used to render that brush. Brushsides are used by the engine to calculate all player physics collision with world brushes. (Vphysics objects use lump 29 instead.)

The maximum number of brushsides is 65536 (MAX_MAP_BRUSHSIDES). The maximum number of brushsides on a single brush is 128 (MAX_BRUSH_SIDES).

Node and leaf

The node array (Lump 5) and leaf array (Lump 10) define the Binary Space Partition (BSP) tree structure of the map. The BSP tree is used by the engine to quickly determine the location of the player's viewpoint with respect to the map geometry, and along with the visibility information (see below), to decide which parts of the map are to be drawn.

The nodes and leaves form a tree structure. Each leaf represents a defined volume of the map, and each node represents the volume which is the sum of all its child nodes and leaves further down the tree.

Each node has exactly two children, which can be either another node or a leaf. A child node has two further children, and so on until all branches of the tree are terminated with leaves, which have no children. Each node also references a plane in the plane array. When determining the player's viewpoint, the engine is trying to find which leaf the viewpoint falls inside. It first compares the coordinates of the point with the plane referenced in the headnode (Node 0). If the point is in front of the plane, it then moves to the first child of the node; otherwise, it moves to the second child. If the child is a leaf, then it has completed its task. If it is another node, it then performs the same check against the plane referenced in this node, and follows the children as before. It therefore traverses the BSP tree until it finds which leaf the viewpoint lies in. The leaves, then, completely divide up the map volume into a set of non-overlapping, convex volumes defined by the planes of their parent nodes.

For more information on how the BSP tree is constructed, see the article "BSP for dummies".

The node array consists of 32-byte structures:

struct dnode_t
{
	int             planenum;       // index into plane array
	int             children[2];    // negative numbers are -(leafs + 1), not nodes
	short           mins[3];        // for frustum culling
	short           maxs[3];
	unsigned short  firstface;      // index into face array
	unsigned short  numfaces;       // counting both sides
	short           area;           // If all leaves below this node are in the same area, then
	                                // this is the area index. If not, this is -1.
	short           paddding;       // pad to 32 bytes length
};

Planenum is the entry in the plane array. The children[] members are the two children of this node; if positive, they are node indices; if negative, the value (-1-child) is the index into the leaf array (e.g., the value -100 would reference leaf 99).

The members mins[] and maxs[] are coordinates of a rough bounding box surrounding the contents of this node. The firstface and numfaces are indices into the face array that show which map faces are contained in this node, or zero if none are. The area value is the map area of this node (see below). There can be a maximum of 65536 nodes in a map (MAX_MAP_NODES).

The leaf array is an array with 56 or 32 bytes per element, depending on the lump version:

struct dleaf_t
{
	int             contents;             // OR of all brushes (not needed?)
	short           cluster;              // cluster this leaf is in
	short           area:9;               // area this leaf is in
	short           flags:7;              // flags
	short           mins[3];              // for frustum culling
	short           maxs[3];
	unsigned short  firstleafface;        // index into leaffaces
	unsigned short  numleaffaces;
	unsigned short  firstleafbrush;       // index into leafbrushes
	unsigned short  numleafbrushes;
	short           leafWaterDataID;      // -1 for not in water

	//!!! NOTE: for lump version 0 (usually in maps of version 19 or lower) uncomment the next line
	//CompressedLightCube   ambientLighting;      // Precaculated light info for entities.
	short                 padding;              // padding to 4-byte boundary
};

The leaf structure has similar contents to the node structure, except it has no children and no reference plane. Additional entries are the contents flags (see the brush lump, above), which shows the contents of any brushes in the leaf, and the cluster number of the leaf (see below). The area and flags members share a 16-bit bitfield and contain the area number and flags relating to the leaf. Firstleafface and numleaffaces index into the leafface array and show which faces are inside this leaf, if any. Firstleafbrush and numleafbrushes likewise index brushes inside this leaf through the leafbrush array.

The ambientLighting element is related to lighting of objects in the leaf, and consists of a CompressedLightCube structure, which is 24 bytes in length. Version 17 BSP files have a modified dleaf_t structure that omits the ambient lighting data, making the entry for each leaf only 32 bytes in length. The same shortened structure is also used for version 20 BSP files, with the ambient lighting information for LDR and HDR probably contained in the new lumps 55 and 56.

All leaves are convex polyhedra, and are defined by the planes of their parent nodes. They do not overlap. Any point in the coordinate space is in one and only one leaf of the map. A leaf which is not filled with a solid brush and can be entered by the player in the usual course of the game has a cluster number set; this is used in conjunction with the visibility information (below).

There are usually multiple, unconnected BSP trees in a map. Each one corresponds to an entry in model array (see below) and the headnode of each tree is referenced there. The first tree is the worldspawn model, the overall geometry of the level. Successive trees are the models of each brush entity in the map.

The creation of the BSP tree is done by the VBSP program, during the first phase of map compilation. Exactly how the tree is created, and how the map is divided into leaves, can be influenced by the map author by the use of HINT brushes, func_details, and the careful layout of all brushes in the map.

LeafFace and LeafBrush

The leafface lump (Lump 16) is an array of unsigned shorts which are used to map from faces referenced in the leaf structure to indices in the face array. The leafbrush lump (also an array of unsigned shorts)(Lump 17) does the same thing for brushes referenced in leaves. Their maximum sizes are both 65536 entries (MAX_MAP_LEAFFACES, MAX_MAP_LEAFBRUSHES).

Textures

The texture information in a map is split across a number of different lumps. The Texinfo lump is the most fundamental, referenced by the face and brushside arrays, and it in turn references the other texture lumps.

Texinfo

The texinfo lump (Lump 6) contains an array of texinfo_t structures:

struct texinfo_t
{
	float   textureVecs[2][4];    // [s/t][xyz offset]
	float   lightmapVecs[2][4];   // [s/t][xyz offset] - length is in units of texels/area
	int     flags;                // miptex flags overrides
	int     texdata;              // Pointer to texture name, size, etc.
}

Each texinfo is 72 bytes long.

The first array of floats is in essence two vectors that represent how the texture is orientated and scaled when rendered on the world geometry. The two vectors, s and t, are the mapping of the left-to-right and down-to-up directions in the texture pixel coordinate space, onto the world. Each vector has an x, y, and z component, plus an offset which is the "shift" of the texture in that direction relative to the world. The length of the vectors represent the scaling of the texture in each direction.

The 2D coordinates (u, v) of a texture pixel (or texel) are mapped to the world coordinates (x, y, z) of a point on a face by:

u = tv0,0 * x + tv0,1 * y + tv0,2 * z + tv0,3

v = tv1,0 * x + tv1,1 * y + tv1,2 * z + tv1,3

(ie. The dot product of the vectors with the vertex plus the offset in that direction. Where tvA,B is textureVecs[A][B].

Furthermore, after calculating (u, v), to convert them to texture coordinates which you would send to your graphics card, divide u and v by the width and height of the texture respectively.

The lightmapVecs float array performs a similar mapping of the lightmap samples of the texture onto the world. It is the same formula but with lightmapVecs instead of textureVecs, and then subtracting the [0] and [1] values of LightmapTextureMinsInLuxels for u and v respectively. LightmapTextureMinsInLuxels is referenced in dface_t;

The flags entry contains surface flags, as defined in bspflags.h.

Texdata

Finally the texdata entry is an index into the Texdata array, and specifies the actual texture.

The index of a Texinfo (referenced from a face or brushside) may be given as -1; this indicates that no texture information is associated with this face. This occurs on compiling brush faces given the SKIP, CLIP, or INVISIBLE type textures in the editor.

The texdata array (Lump 2) consists of the structures:

struct dtexdata_t
{
	Vector  reflectivity;            // RGB reflectivity
	int     nameStringTableID;       // index into TexdataStringTable
	int     width, height;           // source image
	int     view_width, view_height;
};

The reflectivity vector corresponds to the RGB components of the reflectivity of the texture, as derived from the material's .vtf file. This is probably used in radiosity (lighting) calculations of what light bounces from the texture's surface. The nameStringTableID is an index into the TexdataStringTable array (below). The other members relate to the texture's source image.

TexdataStringData and TexdataStringTable

The TexdataStringTable (Lump 44) is an array of integers which are offsets into the TexdataStringData (lump 43). The TexdataStringData lump consists of concatenated null-terminated strings giving the texture name.

There can be a maximum of 12288 texinfos in a map (MAX_MAP_TEXINFO). There is a limit of 2048 texdatas in the array (MAX_MAP_TEXDATA) and up to 256000 bytes in the TexdataStringData data block (MAX_MAP_TEXDATA_STRING_DATA). Texture name strings are limited to 128 characters (TEXTURE_NAME_LENGTH).

Model

A Model, in the terminology of the BSP file format, is a collection of brushes and faces, often called a "bmodel". It should not be confused with the prop models used in Hammer, which are usually called "studiomodels" in the SDK source.

The model lump (Lump 14) consists of an array of 48-byte dmodel_t structures:

struct dmodel_t
{
	Vector  mins, maxs;            // bounding box
	Vector  origin;                // for sounds or lights
	int     headnode;              // index into node array
	int     firstface, numfaces;   // index into face array
};

Mins and maxs are the bounding points of the model. Origin is the coordinates of the model in the world, if set. Headnode is the index of the top node in the node array of the BSP tree which describes this model. Firstface and numfaces index into the face array and give the faces which make up this model.

The first model in the array (Model 0) is always "worldspawn", the overall geometry of the whole map excluding entities (but including func_detail and func_ladder brushes). The subsequent models in the array are associated with brush entities, and referenced from the entity lump.

There is a limit of 1024 models in a map (MAX_MAP_MODELS), including the worldspawn model zero. This is increased to 4069 in Jabroni Brawl: Episode 3 Jabroni Brawl: Episode 3.

Visibility

The visibility lump (Lump 4) is in a somewhat different format to the previously mentioned lumps. To understand it, some discussion of how the Source engine's visibility system works in necessary.

As mentioned in the Node and leaf lumps section above, every point in the map falls into exactly one convex volume called a leaf. All leaves that are on the inside of the map (not touching the void), and that are not covered by a solid brush can potentially have the player's viewpoint inside it during normal gameplay. Each of these enterable leaves (also called visleaves) gets assigned a cluster number. In Source BSP files, each enterable leaf corresponds to just one cluster.

(The terminology is slightly confusing here. According to the "Quake 2 BSP File Format" article, in the Q2 engine there could be multiple adjacent leaves in each cluster - thus the cluster is so called because it is a cluster of leaves. It seems from the SDK source that this situation may also occur during the compilation of Source maps; however, after the VVIS compile process is finished these adjacent leaves (and their parent nodes) are generally merged into a single leaf. In older Source maps (prior to Counter-Strike: Global Offensive Counter-Strike: Global Offensive) it seems there is only ever one leaf per cluster. However, in some CS:GO maps multiple leaves can belong to the same cluster in the final compiled BSP. This seems to occur particularly in the 3D skybox of most CS:GO maps, and can be seen in the main playable area of more recently refurbished maps such as de_cbble and de_nuke.)

Each cluster, then, is a volume in the map that the player can potentially be in. To render the map quickly, the game engine draws the geometry of only those clusters which are visible from the current cluster. Clusters which are completely occluded from view from the player's cluster need not be drawn. Calculating cluster-to-cluster visibility is the responsibility of the VVIS compile tool, and the resulting data is stored in the Visibility lump.

Once the engine knows a cluster is visible, the leaf data references all faces present in that cluster, allowing the contents of the cluster to be rendered.

The data is stored as an array of bit-vectors; for each cluster, a list of which other clusters are visible from it are stored as individual bits (1 if visible, 0 if occluded) in an array, with the nth bit position corresponding to the nth cluster. This is known as the cluster's Potentially Visible Set (PVS). Because of the large size of this data, the bit vectors are compressed by run-length encoding groups of zero bits in each vector.

There is also a Potentially Audible Set (PAS) array created for each cluster; this marks which clusters can hear sounds occurring in other clusters. The PAS seems to be created by merging the PVS bits of all clusters in current cluster's PVS.

The Visibilty lump is defined as:

struct dvis_t
{
	int	numclusters;
	int	byteofs[numclusters][2]
};

The first integer is the number of clusters in the map. It is followed by an array of integers giving the byte offset from the start of the lump to the start of the PVS bit array for each cluster, followed by the offset to the PAS array. Immediately following the array are the compressed bit vectors.

The decoding of the run-length compression works as follows: To find the PVS of a given cluster, start at the byte given by the offset in the byteofs[] array. If the current byte in the PVS buffer is zero, the following byte multiplied by 8 is the number of clusters to skip that are not visible. If the current byte is non-zero, the bits that are set correspond to clusters that are visible from this cluster. Continue until the number of clusters in the map is reached.

Example C code to decompress the bit vectors can be found in the "Quake 2 BSP File Format" document.

The maximum size of the Visibility lump is 0x1000000 bytes (MAX_MAP_VISIBILITY); that is, 16 Mb.

Entity

See also: Patching levels with lump files

The entity lump (Lump 0) is a null-terminated ASCII text buffer which stores the entity data in a format very similar to MAP files, but without any brushes. Its general form is:

{
	"world_maxs" "480 480 480"
	"world_mins" "-480 -480 -224"
	"maxpropscreenwidth" "-1"
	"skyname" "sky_wasteland02"
	"classname" "worldspawn"
}
{
	"origin" "-413.793 -384 -192"
	"angles" "0 0 0"
	"classname" "info_player_start"
}
{
	"model" "*1"
	"targetname" "secret_1"
	"origin" "424 -1536 1800"
	"Solidity" "1"
	"StartDisabled" "0"
	"InputFilter" "0"
	"disablereceiveshadows" "0"
	"disableshadows" "0"
	"rendermode" "0"
	"renderfx" "0"
	"rendercolor" "255 255 255"
	"renderamt" "255"
	"classname" "func_brush"
}

Entities are defined between opening and closing braces ({ and }) and list on each line a pair of key/value properties inside quotation marks. The first entity is always worldspawn. The classname property gives the entity type, and the targetname property gives the entity's name as defined in Hammer (if it has one). The model property is slightly special if it starts with an asterisk (*), the following number is an index into the model array (see above) which corresponds to a bmodel. Otherwise, the value contains the name of a compiled model, or sometimes a sprite. Other key/value pairs correspond to the properties of the entity as set in Hammer. Outputs are stored as regular keyvalues.

Note.pngNote:Some entities (including func_detail, env_cubemap, info_overlay and prop_static) are internal, and are stripped from the entity lump during the compile process, normally because they are absorbed by the world.

There is no limit on the size of the entity lump inside the engine. VBSP has MAX_MAP_ENTITIES, which differs from the engine's actual entity limit, and as such can be circumnavigated with a custom VBSP. Each key can be a maximum of 32 characters (MAX_KEY) and the value up to 1024 characters (MAX_VALUE).

In Team Fortress 2 Team Fortress 2 the Entity lump may be compressed. In such case the entity data is encoded using LZMA algorithm (see Lump compression).

Game lump

The Game lump (Lump 35) seems to be intended to be used for map data that is specific to a particular game using the Source engine, so that the file format can be extended without altering the previously defined format. It starts with a game lump header:

struct dgamelumpheader_t
{
	int lumpCount;  // number of game lumps
	dgamelump_t gamelump[lumpCount];
};

where the gamelump directory array is defined by:

struct dgamelump_t
{
	int             id;        // gamelump ID
	unsigned short  flags;     // flags
	unsigned short  version;   // gamelump version
	int             fileofs;   // offset to this gamelump
	int             filelen;   // length
};

The gamelump is identified by the 4-byte id member, which defines what data is stored in it, and the byte position of the data and its length is given in fileofs and filelen. Note that fileofs is relative to the beginning of the BSP file, not to the game lump offset. One exception is the console version of Portal 2 Portal 2, where fileofs is relative to the game lump offset, as one would expect.

Static props

Of interest is the gamelump which is used to store prop_static entities, which uses the gamelump ID of 'sprp' ASCII (1936749168 decimal). Unlike most other entities, static props are not stored in the entity lump. The gamelump formats used in Source are defined in the public/gamebspfile.h header file.

The first element of the static prop game lump is the dictionary; this is an integer count followed by the list of model (prop) names used in the map:

struct StaticPropDictLump_t
{
	int	dictEntries;
	char	name[128][dictEntries];	// model name
};

Each name entry is 128 characters long, null-padded to this length.

Following the dictionary is the leaf array:

struct StaticPropLeafLump_t
{
	int leafEntries;
	unsigned short	leaf[leafEntries];
};

Presumably, this array is used to index into the leaf lump to locate the leaves that each prop static is located in. Note that a prop static may span several leaves.

Next, an integer giving the number of StaticPropLump_t entries, followed by that many structures themselves:

struct StaticPropLump_t
{
	// v4
	Vector          Origin;            // origin
	QAngle          Angles;            // orientation (pitch yaw roll)
	
	// v4
	unsigned short  PropType;          // index into model name dictionary
	unsigned short  FirstLeaf;         // index into leaf array
	unsigned short  LeafCount;
	unsigned char   Solid;             // solidity type
	// every version except v7*
	unsigned char   Flags;
	// v4 still
	int             Skin;              // model skin numbers
	float           FadeMinDist;
	float           FadeMaxDist;
	Vector          LightingOrigin;    // for lighting
	// since v5
	float           ForcedFadeScale;   // fade distance scale
	// v6, v7, and v7* only
	unsigned short  MinDXLevel;        // minimum DirectX version to be visible
	unsigned short  MaxDXLevel;        // maximum DirectX version to be visible
	// v7* only
	unsigned int    Flags;
	unsigned short  LightmapResX;      // lightmap image width
	unsigned short	LightmapResY;      // lightmap image height
	// since v8
	unsigned char   MinCPULevel;
	unsigned char   MaxCPULevel;
	unsigned char   MinGPULevel;
	unsigned char   MaxGPULevel;
	// since v7
	color32         DiffuseModulation; // per instance color and alpha modulation
	// v9 and v10 only
	bool            DisableX360;       // if true, don't show on XBox 360 (4-bytes long)
	// since v10
	unsigned int    FlagsEx;           // Further bitflags.
	// since v11
	float           UniformScale;      // Prop scale
};

The coordinates of the prop are given by the Origin member; its orientation (pitch, yaw, roll) is given by the Angles entry, which is a 3-float vector. The PropType element is an index into the dictionary of prop model names, given above. The other elements correspond to the location of the prop in the BSP structure of the map, its lighting, and other entity properties as set in Hammer Hammer. The other elements (ForcedFadeScale, etc.) are only present in the static prop structure if the gamelump's specified version is high enough (see dgamelump_t.version); both version 4 and version 5 static prop gamelumps are used in official Half-Life 2 Half-Life 2 maps. Version 6 has been encountered in Team Fortress 2 Team Fortress 2; Version 7 is used in some Left 4 Dead Left 4 Dead maps, and a modified version 7 is present in Zeno Clash maps. Version 8 is used predominantly in Left 4 Dead Left 4 Dead, and version 9 in Left 4 Dead 2 Left 4 Dead 2. The new version 10 appears in Tactical Intervention. Version 11 is used in Counter-Strike: Global Offensive Counter-Strike: Global Offensive since the addition of uniform prop scaling (before this it was version 10). After version 7, DX level options were removed. In version 11 XBox 360 flags were removed.

Version 7* is used by games built on Source 2013 Multiplayer Source 2013 Multiplayer (Team Fortress 2 Team Fortress 2, Counter-Strike: Source Counter-Strike: Source, etc.) and may come across as either version 7 or 10. Specifically, Team Fortress 2 Team Fortress 2 has referred to it as version 7 in the past but now refers to it as version 10 even though they are identical. This version's structure is based on version 6 but rearranged such that Flags is an int and at the bottom, above two new entries. These new entries (LightmapResX and LightmapResY) control the width and height of the prop's lightmap image and are specific to this version.

For a list of flags used by Flags and FlagsEX, see static prop flags.

Other

Other gamelumps used in Source BSP files are the detail prop gamelump (dprp), and the detail prop lighting lump (dplt for LDR and dplh for HDR). These are used for prop_detail entities (grass tufts, etc.) automatically emitted by certain textures when placed on displacement surfaces.

There does not seem to be a specified limit on the size of the game lump.

Displacements

Displacement surfaces are the most complex parts of a BSP file, and only part of their format is covered here. Their data is split over a number of different data lumps in the file, but the fundamental reference to them is through the dispinfo lump (Lump 26). Dispinfos are referenced from the face, original face, and brushside arrays.

DispInfo

struct ddispinfo_t
{
	Vector                startPosition;                // start position used for orientation
	int                   DispVertStart;                // Index into LUMP_DISP_VERTS.
	int                   DispTriStart;                 // Index into LUMP_DISP_TRIS.
	int                   power;                        // power - indicates size of surface (2^power 1)
	int                   minTess;                      // minimum tesselation allowed
	float                 smoothingAngle;               // lighting smoothing angle
	int                   contents;                     // surface contents
	unsigned short        MapFace;                      // Which map face this displacement comes from.
	int                   LightmapAlphaStart;           // Index into ddisplightmapalpha.
	int                   LightmapSamplePositionStart;  // Index into LUMP_DISP_LIGHTMAP_SAMPLE_POSITIONS.
	CDispNeighbor         EdgeNeighbors[4];             // Indexed by NEIGHBOREDGE_ defines.
	CDispCornerNeighbors  CornerNeighbors[4];           // Indexed by CORNER_ defines.
	unsigned int          AllowedVerts[10];             // active verticies
};

The structure is 176 bytes long. The startPosition element is the coordinates of the first corner of the displacement. DispVertStart and DispTriStart are indices into the DispVerts and DispTris lumps. The power entry gives the number of subdivisions in the displacement surface - allowed values are 2, 3 and 4, and these correspond to 4, 8 and 16 subdivisions on each side of the displacement surface. The structure also references any neighbouring displacements on the sides or the corners of this displacement through the EdgeNeighbors and CornerNeighbors members. There are complex rules governing the order that these neighbour displacements are given; see the comments in bspfile.h for more. The MapFace value is an index into the face array and is face that was turned into a displacement surface. This face is used to set the texture and overall physical location and boundaries of the displacement.

DispVerts

The DispVerts lump (Lump 33) contains the vertex data of the displacements. It is given by:

struct dDispVert
{
	Vector  vec;    // Vector field defining displacement volume.
	float   dist;   // Displacement distances.
	float   alpha;  // "per vertex" alpha values.
};

where vec is the normalized vector of the offset of each displacement vertex from its original (flat) position; dist is the distance the offset has taken place; and alpha is the alpha-blending of the texture at that vertex.

A displacement of power p references (2^p + 1)^2 dispverts in the array, starting from the DispVertStart index.

DispTris

The DispTris lump (Lump 48) contains "triangle tags" or flags related to the properties of a particular triangle in the displacement mesh:

struct dDispTri
{
	unsigned short Tags;   // Displacement triangle tags.
};

where the flags are:

Name Value
DISPTRI_TAG_SURFACE 0x1
DISPTRI_TAG_WALKABLE 0x2
DISPTRI_TAG_BUILDABLE 0x4
DISPTRI_FLAG_SURFPROP1 0x8
DISPTRI_FLAG_SURFPROP2 0x10

There are 2x(2^p)^2 DispTri entries for a displacement of power p. They are presumably used to indicate properties for each triangle of the displacement such as whether the surface is walkable at that point (not too steep to climb).

There are a limit of 2048 Dispinfos per map, and the limits of DispVerts and DispTris are such that all 2048 displacements could be of power 4 (maximally subdivided).

Other displacement-related data are the DispLightmapAlphas (Lump 32) and DispLightmapSamplePos (Lump 34) lumps, which seem to relate to lighting of each displacement surface.

Pakfile

The Pakfile lump (Lump 40) is a special lump that can contains multiple files which are embedded into the bsp file. Usually, they contain special texture (.vtf) and material (.vmt) files which are used to store the reflection maps from env_cubemap entities in the map; these files are built and placed in the Pakfile lump when the buildcubemaps console command is executed. The Pakfile can optionally contain such things as custom textures and prop models used in the map, and are placed into the bsp file by using the BSPZIP program (or alternate programs such as Pakrat). These files are integrated into the game engine's file system and will be loaded preferentially before externally located files are used.

The format of the Pakfile lump is identical to that used by the Zip compression utility when no compression is specified (i.e., the individual files are stored in uncompressed format). In some branches, such as Source 2013 Multiplayer Source 2013 Multiplayer, Garry's Mod Garry's Mod, Jabroni Brawl: Episode 3 Jabroni Brawl: Episode 3, and Strata Source Strata Source, LZMA compression can be used as well. If the Pakfile lump is extracted and written to a file, it can therefore be opened with WinZip and similar programs.

The header public/zip_uncompressed.h defines the structures present in the Pakfile lump. The last element in the lump is a ZIP_EndOfCentralDirRecord structure. This points to an array of ZIP_FileHeader structures immediately preceeding it, one for each file present in the Pak. Each of these headers then point to ZIP_LocalFileHeader structures that are followed by that file's data.

The Pakfile lump is usually the last element of the bsp file.

Cubemap

The Cubemap lump (Lump 42) is an array of 16-byte dcubemapsample_t structures:

struct dcubemapsample_t
{
	int    origin[3];    // position of light snapped to the nearest integer
	int    size;         // resolution of cubemap, 0 - default
};

The dcubemapsample_t structure defines the location of a env_cubemap entity in the map. The origin member contains integer x,y,z coordinates of the cubemap, and the size member is resolution of the cubemap, specified as 2^(size-1) pixels square. If set as 0, the default size of 6 (32x32 pixels) is used. There can be a maximum of 1024 (MAX_MAP_CUBEMAPSAMPLES) cubemaps in a file.

When the "buildcubemaps" console command is performed, six snapshots of the map (one for each direction) are taken at the location of each env_cubemap entity. These snapshots are stored in a multi-frame texture (vtf) file, which is added to the Pakfile lump (see above). The textures are named cX_Y_Z.vtf, where (X,Y,Z) are the (integer) coordinates of the corresponding cubemap.

Faces containing materials that are environment mapped (e.g. shiny textures) reference their assigned cubemap through their material name. A face with a material named (e.g.) walls/shiny.vmt is altered (new Texinfo & Texdata entries are created) to refer to a renamed material maps/mapname/walls/shiny_X_Y_Z.vmt, where (X,Y,Z) are the cubemap coordinates as before. This .vmt file is also stored in the Pakfile, and references the cubemap .vtf file through its $envmap property.

Version 20 files contain extra cX_Y_Z.hdr.vtf files in the Pakfile lump, containing HDR texture files in RGBA16161616F (16-bit per channel) format.

Overlay

Unlike the simpler decals (infodecal entities), info_overlays are removed from the entity lump and stored separately in the Overlay lump (Lump 45). The structure is reflects the properties of the entity in Hammer Hammer almost exactly:

struct doverlay_t
{
	int             Id;
	short           TexInfo;
	unsigned short  FaceCountAndRenderOrder;
	int             Ofaces[OVERLAY_BSP_FACE_COUNT];
	float           U[2];
	float           V[2];
	Vector          UVPoints[4];
	Vector          Origin;
	Vector          BasisNormal;
};

The FaceCountAndRenderOrder member is split into two parts; the lower 14 bits are the number of faces that the overlay appears on, with the top 2 bits being the render order of the overlay (for overlapping decals). The Ofaces array, which is 64 elements in size (OVERLAY_BSP_FACE_COUNT) are the indices into the face array indicating which map faces the overlay should be displayed on. The other elements set the texture, scale, and orientation of the overlay decal. There is no enforced limit on overlays inside the engine. VBSP enforces a limit of 512 (MAX_MAP_OVERLAYS, 1024 in Counter-Strike: Global Offensive Counter-Strike: Global Offensive), but custom compilers can circumvent this.

Lighting

The lighting lump (Lump 8) is used to store the static lightmap samples of map faces. Each lightmap sample is a colour tint that multiplies the colours of the underlying texture pixels, to produce lighting of varying intensity. These lightmaps are created during the VRAD phase of map compilation and are referenced from the dface_t structure. The current lighting lump version is 1.

Each dface_t may have a up to four lightstyles (MAXLIGHTMAPS) defined in its styles[] array (which contains 255 to represent no lightstyle). The number of luxels in each direction of the face is given by the two LightmapTextureSizeInLuxels[] members (plus 1), and the total number of luxels per face is thus:

(LightmapTextureSizeInLuxels[0] + 1) * (LightmapTextureSizeInLuxels[1] + 1)

Each face gives a byte offset into the lighting lump in its lightofs member (if no lighting information is used for this face e.g. faces with skybox, nodraw and invisible textures, lightofs is -1.) There are (number of lightstyles)*(number of luxels) lightmap samples for each face, where each sample is a 4-byte ColorRGBExp32 structure:

struct ColorRGBExp32
{
	byte r, g, b;
	signed char exponent;
};

Standard RGB format can be obtained from this by multiplying each colour component by 2^(exponent). For faces with bumpmapped textures, there are four times the usual number of lightmap samples, with one lightmap lit from each normal axis, and a fourth flat one used when bumpmapping is disabled.

Immediately preceeding the lightofs-referenced sample group, there are single samples containing the average lighting on the face, one for each lightstyle, in reverse order from that given in the styles[] array.

Version 20 BSP files contain a second, identically sized lighting lump (Lump 53). This is presumed to store more accurate (higher-precision) HDR data for each lightmap sample. The format is currently unknown, but is also 32 bits per sample.

The maximum size of the lighting lump is 0x1000000 bytes (MAX_MAP_LIGHTING), i.e. 16 MB, and can be mapped onto a 1024x512x32 atlas. This limitation is not enforced in the engine and compilers can freely exceed it.

Confirm:What size atlas does Source actually use? IdTech 2 and GoldSrc used 128x128 atlases (see AllocBlock).

Ambient Lighting

The ambient lighting lumps (Lumps 55 and 56) are present in BSP version 20 and later. Lump 55 is used for HDR lighting, and Lump 56 is used for LDR lighting. These lumps are used to store the volumetric ambient lighting information in each leaf (i.e. lighting information for entities such as NPCs, the viewmodel, and non-static props). Prior to version 20, this data was stored in the leaf lump (Lump 10), in the dleaf_t structure, with far less precision than this newer lump allows.

Both ambient lighting lumps are arrays of dleafambientlighting_t structures:

struct dleafambientlighting_t
{
	CompressedLightCube	cube;
	byte x;		// fixed point fraction of leaf bounds
	byte y;		// fixed point fraction of leaf bounds
	byte z;		// fixed point fraction of leaf bounds
	byte pad;	// unused
};

Each leaf is associated with a number of these dleafambientlighting_t structures, each of which contains a cube of ambient light data at the position specified by the x, y, and z members. These coordinates are stored as fractions of the leaf's bounding box, i.e. a value of 0 for x represents the westmost position in the leaf, a value of 255 represents the eastmost position in the leaf, and a value of 128 represents the center of the leaf.

The lighting data for each sample is represented by a CompressedLightCube structure, which is simply an array of 6 ColorRGBExp32 structures as described in the previous section:

struct CompressedLightCube
{
	ColorRGBExp32 m_Color[6];
};

Each lighting sample in the array corresponds to the amount of lighting being received from each cardinal direction in 3D space.

At compile time, VRAD will randomly generate a number of ambient light sample positions for each leaf, and store the lighting information at each sample point in a dleafambientlighting_t structure. To associate each leaf with its collection of ambient samples, the ambient lighting index lumps (Lumps 51 and 52) are used. Lump 51 stores ambient light index information for HDR, and lump 52 stores the same information for LDR.

The ambient light index lumps are arrays of dleafambientindex_t structures:

struct dleafambientindex_t
{
	unsigned short ambientSampleCount;
	unsigned short firstAmbientSample;
};

The Nth dleafambientindex_t structure in this array always corresponds to the Nth leaf in the dleaf_t array. The ambientSampleCount field is the number of ambient samples associated with the corresponding leaf, and firstAmbientSample is an index into the ambient lighting array, referring to the first sample in the array that is associated with this leaf.

Occlusion

The occlusion lump (Lump 9) contains the polygon geometry and some flags used by func_occluder entities. Unlike other brush entities, func_occluders don't use the 'model' key in the entity lump. Instead, the brushes are split from the entities during the compile process and numeric occluder keys are assigned as 'occludernum'. Brush sides textured with tools/toolsoccluder or tools/toolstrigger are then stored together with the occluder keys and some additional info in this lump.

The lump is divided into three parts and begins with a integer value with the total number of occluders, followed by an array of doccluderdata_t fields of the same size. The next part begins with another integer value, this time for the total number of occluder polygons, as well as an array of doccluderpolydata_t fields of equal size. Part three begins with another integer value for the amount of occluder polygon vertices, followed by an array of integer values for the vertex indices, again of the same size.

struct doccluder_t
{
	int			count;
	doccluderdata_t		data[count];
	int			polyDataCount;
	doccluderpolydata_t	polyData[polyDataCount];
	int			vertexIndexCount;
	int			vertexIndices[vertexIndexCount];
};

The doccluderdata_t structure contains flags and dimensions of the occluder, as well as the area where it remains. firstpoly is the first index into the doccluderpolydata_t with a total of polycount entries.

struct doccluderdata_t
{
	int	flags;
	int	firstpoly;	// index into doccluderpolys
	int	polycount;	// amount of polygons
	Vector	mins;	        // minima of all vertices
	Vector	maxs;	        // maxima of all vertices
	// since v1
	int	area;
};

Occluder polygons are stored in the doccluderpolydata_t structure and contain the firstvertexindex field, which is the first index into the vertex array of the occluder, which are again indices for the vertex array of the vertex lump (Lump 3). The total number of vertex indices is stored in vertexcount.

struct doccluderpolydata_t
{
	int	firstvertexindex;	// index into doccludervertindices
	int	vertexcount;		// amount of vertex indices
	int	planenum;
};

Physics

The physcollide lump (Lump 29) contains physics data for the world.

The lump consists of a sequence of models, where each model consists of:

  • a dphysmodel_t header
   struct dphysmodel_t
   {
       int modelIndex;  // Perhaps the index of the model to which this physics model applies?
       int dataSize;    // Total size of the collision data sections
       int keydataSize; // Size of the text section
       int solidCount;  // Number of collision data sections
   };
  • a series of collision data sections (including compactsurfaceheader_t header)
  • a text section

The lump is terminated by a dphysmodel_t structure with the modelIndex set to -1.

The last two parts appear to be identical to the PHY file format, which means their exact contents are unknown. Note that the compactsurfaceheader_t structure contains the data size of each collision data section (including the rest of the header), so the lump can be parsed as follows:

   while(true) {
       header = readHeader();
       if(header.modelIndex == -1)
           break;
       
       for(int k = 0; k < header.solidCount; k++) {
           size = read4ByteInt();
           collisionData = readBytes(size);
       }
       
       textData = readBytes(header.keydataSize);
   }

Worldlights

The worldlights lumps (Lump 15 for LDR and Lump 54 for HDR) contain information on each static light entity in the world, and are used to provide direct lighting for dynamically lit entities. The data is generated by VRAD from the entity lump. VRAD uses information from these lumps instead of referring to light entities from Entity lump. Both lumps have the same data structure containing contiguous dworldlight_t structs:

// lights that were used to illuminate the world
enum emittype_t
{
	emit_surface,		// 90 degree spotlight
	emit_point,			// simple point light source
	emit_spotlight,		// spotlight with penumbra
	emit_skylight,		// directional light with no falloff (surface must trace to SKY texture)
	emit_quakelight,	// linear falloff, non-lambertian
	emit_skyambient,	// spherical light source with no falloff (surface must trace to SKY texture)
};

// Flags for dworldlight_t::flags
#define DWL_FLAGS_INAMBIENTCUBE		0x0001	// This says that the light was put into the per-leaf ambient cubes.

struct dworldlight_t
{
	Vector		origin;					
	Vector		intensity;
	Vector		normal;		// for surfaces and spotlights
	int		cluster;
	emittype_t	type; //int, 0 - surface, 1 - point, 2 - spot, 3 - sky, 4 - quakelight, 5 - sky ambient 
	int		style;
	float		stopdot;		// start of penumbra for emit_spotlight
	float		stopdot2;		// end of penumbra for emit_spotlight
	float		exponent;		// 
	float		radius;			// cutoff distance
	// falloff for emit_spotlight + emit_point: 
	// 1 / (constant_attn + linear_attn * dist + quadratic_attn * dist^2)
	float		constant_attn;	
	float		linear_attn;
	float		quadratic_attn;
	int		flags;			// Uses a combination of the DWL_FLAGS_ defines.
	int		texinfo;		// 
	int		owner;			// entity that this light it relative to
};

The size of this struct is 88 bytes, therefore a number of entries in the lump can be determined by dividing the lump length by 88.

Up to two lights from this lump may influence dynamic lighting.

Confirm:Does the worldlights lump's dynamic lighting work by creating elights on the MDL?
Note.pngNote:If VRAD tries to exceed MAX_MAP_WORLDLIGHTS, it will spit out a too many lights error. This is a compiler limit, not an engine one, and can be bypassed with a custom RAD compiler.

Other

Todo: Some of these information are likely guesses and need further research.
  • The Areas lump (Lump 20) references the Areaportals lump (Lump 21) and is used with func_areaportal and func_areaportalwindow entities to define sections of the map that can be switched to render or not render.
  • The Portals (Lump 22), Clusters (Lump 23), PortalVerts (Lump 24), ClusterPortals (Lump 25), and ClipPortalVerts (Lump 41) lumps are used by the VVIS phase of the compile to ascertain which clusters can see which other clusters. A cluster is a player-enterable leaf volume in the map (see above). A "portal" is a polygon boundary between a cluster or leaf and an adjacent cluster or leaf. Most of this information is also used by the VRAD program to calculate static lighting, and then is removed from the bsp file.
  • PhysCollide Lumps (Lump 29) and PhysCollideSurface (Lump 49) lumps seem to be related to the physical simulation of entity collisions in the game engine.
  • The VertNormal (Lump 30) and VertNormalIndices (Lump 31) lumps may be related to smoothing of lightmaps on faces.
  • The FaceMacroTextureInfo lump (Lump 47) is a short array containing the same number of members as the number of faces in the map. If the entry for a face contains anything other than -1 (0xFFFF), it is an index of a texture name in the TexDataStringTable. In VRAD, the corresponding texture is mapped onto the world extents, and used to modulate the lightmaps of that face. There is also a base macro texture (located at materials/macro/mapname/base.vtf) that is applied to all faces if found. Only maps in VTMB seem to make any use of macro textures.
  • LeafWaterData (Lump 36) and LeafMinDistToWater (Lump 46) lumps may be used to determine player position with respect to water volumes.
  • The Primitives (Lump 37), PrimVerts (Lump 38) and PrimIndices (Lump 39) lumps are used in reference to "non-polygonal primitives". They are also sometimes called "waterstrips", "waterverts" and "waterindices" in the SDK Source, since they were originally only used to subdivide water meshes. They are now used to prevent the appearance of cracks between adjacent faces, if the face edges contain a "T-junction" (a vertex collinearly between two other vertices). The PrimIndices lump defines a set of triangles between face vertices, that tessellate the face. They are referenced from the Primatives lump, which is in turn referenced by the face lump data. Current maps do not seem to use the PrimVerts lump at all. (Ref.)
  • Version 20 files containing HDR lighting information have four extra lumps, the contents of which are currently uncertain. Lump 53 is always the same size as the standard lighting lump (Lump 8) and probably contains higher-precision data for each lightmap sample. Lump 54 is the same size as the worldlight lump (Lump 15) and presumably contains HDR-related data for each light entity.

CRC32 checksum

Despite not being a part of BSP format, a CRC32 checksum of a BSP map plays an important role. It is used to compare clientside and serverside version of a map file. In case of the checksum being different an error is thrown ("Your map map_name.bsp differs from the server").

BSP map checksum is not merely a checksum of an entire file - instead, it is calculated based only on BSP lumps, excluding the Entity lump (Lump 0). BSP header does not contribute to the result, additionaly, because the lump bytes are fed into CRC algorithm in an order of their indices (Entity lump, index 0 - first, Planes lump, index 1 - second etc.) the order they are stored in a file does not affect the final result (except the Game lump (Lump 35) as the fileofs of Game lumps is relative to the start of the file).

Java code for calculating the checksum (BSPEntSpy, SourceBSPFile.java):

public long computeChecksum() throws IOException {
	crcAlgo.reset();
		
	byte[] block = new byte[10240];
	for(GenericLump lump : lumps) {
		if(lump.index == ENTLUMP)
			continue;
			
		bspfile.seek(lump.offset);
			
		int toRead = (int)lump.length;
			
		while(toRead > 0) {
			int read = bspfile.read(block, 0, (int)Math.min(toRead, block.length));
				
			crcAlgo.update(block, 0, read);
			toRead -= read;
		}
	}
		
	return crcAlgo.getValue();
}

References

References
2. bsp_regen source code (bsp.hpp)
bool is_valid() {
        if (header_->magic == MAGIC_rBSP) {
            switch (header_->version) {
                case 29:  // Titanfall
                case 36:  // Titanfall 2 [PS4] (Tech Test)
                case 37:  // Titanfall 2
                    return (header_->_127 == 127);
                default:
                    return false;
            }
.