Differences

This shows you the differences between two versions of the page.

Link to this comparison view

Both sides previous revisionPrevious revision
Next revision
Previous revision
trs:room_geometry [2021/03/01 14:31] – [Common Fields of a Room Structure] add TR1 room flag 0x100 stohrendorftrs:room_geometry [2024/12/13 18:40] (current) – [Room Vertex Structure] stohrendorf
Line 162: Line 162:
 ==== Room Light Structure ==== ==== Room Light Structure ====
  
-<note> TR engines always used static room lights only for processing lighting on entities (such as Lara, enemies, doors, and others). This is called //external lighting//. For room meshes, they used so-called internal, or //pre-baked//  lighting, which is done on level building stage: lights are calculated and applied to room faces via vertex colours. There is no way to change room lighting when the level is compiled — meaning, any changes in light positions, intensities and colour won’t affect room faces. </note>+<note> TR engines always used static room lights only for processing lighting on entities (such as Lara, enemies, doors, and others). This is called //external lighting//. For room meshes, they used so-called internal, or //pre-baked// lighting, which is done on level building stage: lights are calculated and applied to room faces via vertex colours. There is no way to change room lighting when the level is compiled — meaning, any changes in light positions, intensities and colour won’t affect room faces. </note>
  
 There are four different types of room light structures. First one is used in TR1-2, second is used in TR3, third is used in TR4, and fourth is used in TR5. Here is the description of each: There are four different types of room light structures. First one is used in TR1-2, second is used in TR3, third is used in TR4, and fourth is used in TR5. Here is the description of each:
Line 177: Line 177:
     uint32_t Fade1;         // Falloff value     uint32_t Fade1;         // Falloff value
 }; };
-</code> 
  
-''<nowiki>X/Y/Z</nowiki>''  are in world coordinates. ''<nowiki>Intensity1/Intensity2</nowiki>''  are almost always equal. This lighting only affects //externally-lit//  objects. Tomb Raider 1 has only the first of the paired ''<nowiki>Intensity</nowiki>''  and ''<nowiki>Fade</nowiki>''  values.+</code>
  
-''<nowiki>Intensity1</nowiki>''  ranges from ''0'' (dark) to ''0x1FFF'' (bright)However, some rooms occasionally have some lights with intensity greater than ''0x1FFF'' (for example, look at room #9, 2nd light in ''<nowiki>level1.phd</nowiki>''). ''<nowiki>Fade1</nowiki>''  is the maximum distance the light shines on, and ranges from ''0'' to ''0x7FFF''.+''<nowiki>X/Y/Z</nowiki>'' are in world coordinates. ''<nowiki>Intensity1/Intensity2</nowiki>'' are almost always equalThis lighting only affects //externally-lit// objects. Tomb Raider 1 has only the first of the paired ''<nowiki>Intensity</nowiki>'' and ''<nowiki>Fade</nowiki>'' values.
  
 +''<nowiki>Intensity1</nowiki>'' ranges from ''0'' (dark) to ''0x1FFF'' (bright). However, some rooms occasionally have some lights with intensity greater than ''0x1FFF'' (for example, look at room #9, 2nd light in ''<nowiki>level1.phd</nowiki>''). ''<nowiki>Fade1</nowiki>'' is the maximum distance the light shines on, and ranges from ''0'' to ''0x7FFF''.
 === TR2 Room Lighting === === TR2 Room Lighting ===
  
Line 198: Line 198:
     uint32_t Fade2;         // Only in TR2     uint32_t Fade2;         // Only in TR2
 }; };
-</code> 
  
-''<nowiki>Intensity2</nowiki>''  and ''<nowiki>Fade2</nowiki>''  values are seemingly not used. ''<nowiki>Intensity1</nowiki>''  can go very well beyond ''0x1FFF'', right to ''0x7FFF'' (ultra bright light). Above ''0x7FFF'', it is always black, so the number is pseudo-signed (negative values are always treated as zero).+</code>
  
 +''<nowiki>Intensity2</nowiki>'' and ''<nowiki>Fade2</nowiki>'' values are seemingly not used. ''<nowiki>Intensity1</nowiki>'' can go very well beyond ''0x1FFF'', right to ''0x7FFF'' (ultra bright light). Above ''0x7FFF'', it is always black, so the number is pseudo-signed (negative values are always treated as zero).
 === TR3 Room Lighting === === TR3 Room Lighting ===
  
Line 226: Line 226:
   }   }
 }; };
-</code> 
  
-''<nowiki>Intensity</nowiki>'' is the power of the light and ranges mainly from ''0'' (low power) to ''0x1FFF'' (high power). +</code>
-''<nowiki>Fade</nowiki>'' is the distance max the light can shine on. Range is mainly from ''0'' to ''0x7FFF''.+
  
 +''<nowiki>Intensity</nowiki>'' is the power of the light and ranges mainly from ''0'' (low power) to ''0x1FFF'' (high power). ''<nowiki>Fade</nowiki>'' is the distance max the light can shine on. Range is mainly from ''0'' to ''0x7FFF''.
 === TR4 Room Lighting === === TR4 Room Lighting ===
  
Line 252: Line 251:
           float dx, dy, dz;    // Direction - used only by sun and spot lights           float dx, dy, dz;    // Direction - used only by sun and spot lights
 }; };
 +
 </code> </code>
  
 ''<nowiki>LightType</nowiki>'' was extended and is now somewhat similar to D3D light type, but there are some differences. ''<nowiki>LightType</nowiki>'' was extended and is now somewhat similar to D3D light type, but there are some differences.
  
-  * **0** — Sun+   * **0** — Sun
   * **1** — Light   * **1** — Light
   * **2** — Spot   * **2** — Spot
Line 287: Line 287:
         float r, g, b;       // Colour of the light         float r, g, b;       // Colour of the light
  
-       uint32_t Separator    // Dummy value = 0xCDCDCDCD+       uint32_t shadow;    // Dummy value = 0xCDCDCDCD
  
         float In;            // Cosine of the IN value for light / size of IN value         float In;            // Cosine of the IN value for light / size of IN value
Line 303: Line 303:
       uint8_t Filler[3];     // Dummy values = 3 x 0xCD       uint8_t Filler[3];     // Dummy values = 3 x 0xCD
 }; };
 +
 </code> </code>
  
Line 309: Line 310:
 { {
         float x, y, z;       // Position of light, in world coordinates         float x, y, z;       // Position of light, in world coordinates
-        float r, g, b;       // Colour of the light+        float rad, sqrad; 
 +        float den; 
 +        float r, g, b; 
 +};
  
-       uint32_t Separator    // Dummy value = 0xCDCDCDCD 
- 
-        float In;            // Cosine of the IN value for light / size of IN value 
-        float Out;           // Cosine of the OUT value for light / size of OUT value 
-}; 
 </code> </code>
- 
-**TODO**: investigate exact meaning of ''tr5_fog_bulb'' fields. 
  
 ''<nowiki>x,y,z</nowiki>''  values shouldn’t be used by sun type light, but sun seems to have a large ''<nowiki>x</nowiki>''  value (9 million, give or take), a zero ''<nowiki>y</nowiki>''  value, and a small ''<nowiki>z</nowiki>''  value (4..20) in the original TR5 levels. ''<nowiki>x,y,z</nowiki>''  values shouldn’t be used by sun type light, but sun seems to have a large ''<nowiki>x</nowiki>''  value (9 million, give or take), a zero ''<nowiki>y</nowiki>''  value, and a small ''<nowiki>z</nowiki>''  value (4..20) in the original TR5 levels.
Line 333: Line 330:
  
 ''<nowiki>x2</nowiki>'', ''<nowiki>y2</nowiki>'', ''<nowiki>z2</nowiki>'', ''<nowiki>dx2</nowiki>'', ''<nowiki>dy2</nowiki>''  and ''<nowiki>dz2</nowiki>''  values repeat previous corresponding information in long data types instead of floats. ''<nowiki>x2</nowiki>'', ''<nowiki>y2</nowiki>'', ''<nowiki>z2</nowiki>'', ''<nowiki>dx2</nowiki>'', ''<nowiki>dy2</nowiki>''  and ''<nowiki>dz2</nowiki>''  values repeat previous corresponding information in long data types instead of floats.
 +
  
 ==== Room Vertex Structure ==== ==== Room Vertex Structure ====
Line 366: Line 364:
        int16_t Lighting;        int16_t Lighting;
       uint16_t Attributes; // A set of flags for special rendering effects       uint16_t Attributes; // A set of flags for special rendering effects
-       int16_t Lighting2 // Almost always equal to Lighting1+       int16_t RenderedLighting; // Internal field to store the actually rendered lighting
 }; };
 </code> </code>
Line 543: Line 541:
     uint16_t NumLayerVertices;   // Number of vertices in this layer (2 bytes)     uint16_t NumLayerVertices;   // Number of vertices in this layer (2 bytes)
     uint16_t NumLayerVerticesWater; // Number of underwater vertices in this layer (2 bytes)     uint16_t NumLayerVerticesWater; // Number of underwater vertices in this layer (2 bytes)
-    uint16_t UnknownL1;+    uint16_t NumLayerVerticesShore;
     uint16_t NumLayerRectangles; // Number of rectangles in this layer (2 bytes)     uint16_t NumLayerRectangles; // Number of rectangles in this layer (2 bytes)
     uint16_t NumLayerTriangles;  // Number of triangles in this layer (2 bytes)     uint16_t NumLayerTriangles;  // Number of triangles in this layer (2 bytes)
-    uint16_t UnknownL2;+    uint16_t NumLayerRectanglesWater// Number of rectangles containing water vertices 
 +    uint16_t NumLayerTrianglesWater; // Number of trianglescontaining water vertices
  
     uint16_t Filler;             // Always 0     uint16_t Filler;             // Always 0
-    uint16_t Filler2;            // Always 0 
  
     // The following 6 floats define the bounding box for the layer     // The following 6 floats define the bounding box for the layer
Line 560: Line 558:
     float    LayerBoundingBoxZ2;     float    LayerBoundingBoxZ2;
  
-    uint32_t Filler3;     // Always 0 (4 bytes)+    uint32_t Filler3;     // Always 0 (4 bytes), internal pointer
     uint32_t VerticesOffset;   // Those fields are overwritten at level loading      uint32_t VerticesOffset;   // Those fields are overwritten at level loading 
     uint32_t PolyOffset;       // by the ones present in the tr5_room struct + an offset     uint32_t PolyOffset;       // by the ones present in the tr5_room struct + an offset
-    uint32_t PolyOffset2;      // i.e. the values are not read, the fields are there for storage purposes+    uint32_t PrelightOffset;      // i.e. the values are not read, the fields are there for storage purposes
 } }
 </code> </code>
  
-''<nowiki>UnknownL2</nowiki>''  appears to be the number of double sided textures in this layer, however is sometimes 1 off (2 bytes).+''<nowiki>NumLayerVerticesShore</nowiki>''  appears to be the number of double sided textures in this layer, however is sometimes 1 off (2 bytes).
  
 ===== The Whole Room Structure ===== ===== The Whole Room Structure =====
Line 747: Line 745:
 ==== TR5 Room Structure ==== ==== TR5 Room Structure ====
  
-As it was mentioned before, TR5 room structure was almost completely changed, when compared to previous versions. For example, TR5 shuffles numerous values and structures in almost chaotic manner, and introduces a bunch of completely new parameters (mostly to deal with //layers//). Also, there is vast amount of //fillers//  and //separators//, which contain no specific data.+As it was mentioned before, TR5 room structure was almost completely changed, when compared to previous versions. For example, TR5 shuffles numerous values and structures in almost chaotic manner, and introduces a bunch of completely new parameters (mostly to deal with //layers//). Also, there is vast amount of //fillers// and //separators//, which contain no specific data.
  
-<note> The one possible reason for such ridiculous structure change is an attempt to //crypt file format//, so it won’t be accessed by unofficial level editing tools, which received major development by that time. Another possible reason is whole TR5 development process was rushed, as the team developed //Tomb Raider: Angel of Darkness//  at the very same time. </note>+<note> The one possible reason for such ridiculous structure change is an attempt to //crypt file format//, so it won’t be accessed by unofficial level editing tools, which received major development by that time. Another possible reason is whole TR5 development process was rushed, as the team developed //Tomb Raider: Angel of Darkness// at the very same time. </note>
  
 <note> There are multiple pointer fields in the structure, they are relative to the **end** of the ''tr5_room'' structure, i.e. to the beginning of the ''tr5_room_data'' structure.</note> <note> There are multiple pointer fields in the structure, they are relative to the **end** of the ''tr5_room'' structure, i.e. to the beginning of the ''tr5_room_data'' structure.</note>
  
-{{anchor:tr5_room}}+ {{anchor:tr5_room}}
  
 <code cpp> <code cpp>
Line 762: Line 760:
     uint32_t RoomDataSize; // size of following fields (208 bytes) + tr5_room_data (dynamic)     uint32_t RoomDataSize; // size of following fields (208 bytes) + tr5_room_data (dynamic)
  
-    char* RoomData; // is filled by the game at run time. Always 0xCDCDCDCD in level files (4 bytes), +    char* RoomData; // is filled by the game at run time. Always 0xCDCDCDCD in level files (4 bytes),
  
     uint16_t* NumPortals; // points to tr5_room_data.NumPortals     uint16_t* NumPortals; // points to tr5_room_data.NumPortals
Line 783: Line 781:
     uint8_t  ReverbInfo;     uint8_t  ReverbInfo;
     uint8_t  AlternateGroup;     uint8_t  AlternateGroup;
-    +
     int8_t MeshEffect;     int8_t MeshEffect;
     int8_t bound_active;     int8_t bound_active;
-     + 
-    int16_t left;         // always 0x7FFF  +    int16_t left;         // always 0x7FFF
     int16_t right;        // always 0     int16_t right;        // always 0
-    int16_t top;          // always 0x7FFF  +    int16_t top;          // always 0x7FFF 
-    int16_t bottom;       // always 0     +    int16_t bottom;       // always 0 
-    +
     int16_t test_left;    // always 0xCDCD     int16_t test_left;    // always 0xCDCD
     int16_t test_right;   // always 0xCDCD     int16_t test_right;   // always 0xCDCD
     int16_t test_top;     // always 0xCDCD     int16_t test_top;     // always 0xCDCD
-    int16_t test_bottom;  // always 0xCDCD  +    int16_t test_bottom;  // always 0xCDCD 
-    +
     int16_t item_number;  // always 0xFFFF (-1)     int16_t item_number;  // always 0xFFFF (-1)
     int16_t fx_number;    // always 0xFFFF (-1)     int16_t fx_number;    // always 0xFFFF (-1)
Line 803: Line 801:
     uint16_t Flags;     uint16_t Flags;
  
-    uint32_t Unknown1+    uint32_t nVerts
-    uint32_t Unknown2;     // Always 0 +    uint32_t nWaterVerts;     // Always 0 
-    uint32_t Unknown3;     // Always 0+    uint32_t nShoreVerts;     // Always 0
  
-    uint32_t Separator;    // 0xCDCDCDCD+    uint32_t Separator;    // 0xCDCDCDCD, internally a pointer
  
-    uint16_t Unknown4; +    uint32_t Separator// internally a pointer to the face data
-    uint16_t Unknown5;+
  
     float RoomX;     float RoomX;
Line 816: Line 813:
     float RoomZ;     float RoomZ;
  
-    uint32_t Separator[4]; // Always 0xCDCDCDCD +    uint32_t vnormals; // internal pointer 
-    uint32_t Separator;    // 0 for normal rooms and 0xCDCDCDCD for null rooms +    uint32_t fnormals// internal pointer 
-    uint32_t Separator   // Always 0xCDCDCDCD+    uint32_t prelight; // internal pointer 
 +    uint32_t prelightwater; // internal pointer 
 +    uint32_t watercalc; // internal pointer, 0 for normal rooms and 0xCDCDCDCD for null rooms 
 +    uint32_t verts; // internal pointer
  
     uint32_t NumRoomTriangles;     uint32_t NumRoomTriangles;
Line 838: Line 838:
     tr5_room_vertex* Vertices; // points to tr5_room_data.Vertices     tr5_room_vertex* Vertices; // points to tr5_room_data.Vertices
     void* PolyOffset; // points to tr5_room_data.Faces     void* PolyOffset; // points to tr5_room_data.Faces
-    void* PolyOffset2; // points to tr5_room_data.Faces+    void* PrelightOffset; // points to tr5_room_data.Faces, internal?
  
     uint32_t NumVertices;     uint32_t NumVertices;
  
-    uint32_t Separator[4];  // Always 0xCDCDCDCD+    // Always 0xCDCDCDCD 
 +    float fLeft; 
 +    float fRight; 
 +    float fTop; 
 +    float fBottom;
 } }
 // immediately after // immediately after
Line 864: Line 868:
     tr5_room_vertex Vertices[NumVertices];     tr5_room_vertex Vertices[NumVertices];
 } }
 +
 </code> </code>
  
-''<nowiki>XELA</nowiki>''  landmark seemingly serves as a header for room structure. It is clear that //XELA//  is a reversed //ALEX//, which is most likely the name of TR5 programmer, //Alex Davis//. It probably indicates that Alex Davis is responsible for changes in room structures.+''<nowiki>XELA</nowiki>'' landmark seemingly serves as a header for room structure. It is clear that //XELA// is a reversed //ALEX//, which is most likely the name of TR5 programmer, //Alex Davis//. It probably indicates that Alex Davis is responsible for changes in room structures
 + 
 +''<nowiki>RoomDataSize</nowiki>'' is a handy value determining the size of the following data. You can use this value to quickly //parse thru// to the next room.
  
-''<nowiki>RoomDataSize</nowiki>''  is a handy value determining the size of the following dataYou can use this value to quickly //parse thru//  to the next room.+''<nowiki>RoomX</nowiki>'', ''<nowiki>RoomY</nowiki>'' and ''<nowiki>RoomZ</nowiki>'' values are positions of room in world coordinates**NOTE:** If room is //null room//, then each of these values will be ''<nowiki>0xCDCDCDCD</nowiki>''.
  
-''<nowiki>RoomX</nowiki>'', ''<nowiki>RoomY</nowiki>''  and ''<nowiki>RoomZ</nowiki>''  values are positions of room in world coordinates. **NOTE:**  If room is //null room//, then each of these values will be ''<nowiki>0xCDCDCDCD</nowiki>''.+''<nowiki>NumRoomTriangles</nowiki>'' and ''<nowiki>NumRoomRectangles</nowiki>'' are respectively the numbers of triangular and rectangular faces in a given room. **NOTE:** If room is //null room//, each of these values will be ''<nowiki>0xCDCDCDCD</nowiki>''.
  
-''<nowiki>NumRoomTriangles</nowiki>''  and ''<nowiki>NumRoomRectangles</nowiki>''  are respectively the numbers of triangular and rectangular faces in a given room. **NOTE:**  If room is //null room//, each of these values will be ''<nowiki>0xCDCDCDCD</nowiki>''.+''<nowiki>LightDataSize</nowiki>'' is the size of the light data in bytes (//not// in [[:trs:room_geometry#tr5_room_light|[tr5_room_light]]] units).
  
-''<nowiki>LightDataSize</nowiki>''  is the size of the light data in bytes (//not//  in [[trs:room_geometry#tr5_room_light|[tr5_room_light]]] units).+''<nowiki>RoomYTop</nowiki>'' and ''<nowiki>RoomYBottom</nowiki>'' are equal to ''<nowiki>yTop</nowiki>'' and ''<nowiki>yBottom</nowiki>'' values in [[:trs:room_geometry#tr_room_info|[tr_room_info]]] structure. If room is a //null room//, both of these values are ''<nowiki>0xCDCDCDCD</nowiki>''.
  
-''<nowiki>RoomYTop</nowiki>''  and ''<nowiki>RoomYBottom</nowiki>''  are equal to ''<nowiki>yTop</nowiki>''  and ''<nowiki>yBottom</nowiki>''  values in [[trs:room_geometry#tr_room_info|[tr_room_info]]] structure. If room is a //null room//, both of these values are ''<nowiki>0xCDCDCDCD</nowiki>''.+''<nowiki>NumLayers</nowiki>'' is a number of layers (volumes) in this room.
  
-''<nowiki>NumLayers</nowiki>''  is a number of layers (volumes) in this room.+''<nowiki>VerticesSize</nowiki>'' is the size of vertex data block in bytes. Therefore, it //must// be a multiple of [[:trs:room_geometry#tr5_room_vertex|[tr5_room_vertex]]] size, else it means the block size is wrong.
  
-''<nowiki>VerticesSize</nowiki>''  is the size of vertex data block in bytes. Therefore, it //must//  be multiple of [[trs:room_geometry#tr5_room_vertex|[tr5_room_vertex]]] sizeelse it means the block size is wrong.+''<nowiki>Faces</nowiki>'' is a sequential data array for the room polygons (both [[:trs:fundamentals#tr_face4|[tr_face4]]] and [[:trs:fundamentals#tr_face3|[tr_face3]]]),
  
-''<nowiki>Faces</nowiki>''  is a sequential data array for the room polygons (both [[trs:fundamentals#tr_face4|[tr_face4]]] and [[trs:fundamentals#tr_face3|[tr_face3]]]),+<note> ''<nowiki>Faces</nowiki>'' array is strictly linked with ''<nowiki>NumLayers</nowiki>'' value. The data is sequentially structured for each layer — at first it lists first layer’s rectangles then triangles, followed by the second layer’s rectangles and trianglesand so on, until all layers are done. </note>
  
-<note> ''<nowiki>Faces</nowiki>''  array is strictly linked with ''<nowiki>NumLayers</nowiki>''  value. The data is sequentially structured for each layer — at first it lists first layer’s rectangles then triangles, followed by the second layer’s rectangles and triangles, and so on, until all layers are done. </note> 
  
 ==== Common Fields of a Room Structure ==== ==== Common Fields of a Room Structure ====
trs/room_geometry.1614609094.txt.gz · Last modified: 2021/03/01 14:31 by stohrendorf
Back to top
CC Attribution-Share Alike 4.0 International
Driven by DokuWiki Recent changes RSS feed Valid CSS Valid XHTML 1.0