OBD:Instance file format: Difference between revisions

brain dump chapter 1
(the basics)
(brain dump chapter 1)
Line 1: Line 1:
All instance files begin with a 64 byte header followed by 3 "descriptor" arrays, a data table and a name table. Among other things the header contains the number of descriptors in each of the 3 arrays and the offset of the data and name tables (relative to the start of the file).
{{Table}}
{{Table}}
{{OBD_Table_Header}}
{{OBD_Table_Header}}
{{OBDtrBK|Instance file header }}
{{OBDtr| 0x00 | int64  | | 1F 27 DC 33 DF BC 03 00 | 0x0003BCDF33DC271F | PC template checksum; PC Demo and Mac files use 0x0003BCDF23C13061 instead }}
{{OBDtr| 0x00 | int64  | | 1F 27 DC 33 DF BC 03 00 | 0x0003BCDF33DC271F | template checksum }}
{{OBDtr| 0x08 | int32  | | 31 33 52 56            | 'VR31'            | .dat version; .oni files use 'VR32' instead }}
{{OBDtr| 0x08 | int32  | | 31 33 52 56            | 'VR31'            | version }}
{{OBDtr| 0x0C | int64  | | 40 00 14 00 10 00 08 00 | 0x0008001000140040 | signature }}
{{OBDtr| 0x0C | int64  | | 40 00 14 00 10 00 08 00 | 0x0008001000140040 | signature }}
{{OBDtr| 0x14 | int32  | | F9 05 00 00 | 1529      | instance count  }}
{{OBDtr| 0x14 | int32  | | F9 05 00 00 | 1529      | instance descriptor count  }}
{{OBDtr| 0x18 | int32  | | D5 01 00 00 | 469      | name count }}
{{OBDtr| 0x18 | int32  | | D5 01 00 00 | 469      | name descriptor count }}
{{OBDtr| 0x1C | int32  | | 32 00 00 00 | 50        | template count }}
{{OBDtr| 0x1C | int32  | | 32 00 00 00 | 50        | template descriptor count }}
{{OBDtr| 0x20 | int32  | | 80 89 00 00 | 0x008980  | data table offset }}
{{OBDtr| 0x20 | int32  | | 80 89 00 00 | 0x008980  | data table offset }}
{{OBDtr| 0x24 | int32  | | 20 6E 55 00 | 0x556E20  | data table size }}
{{OBDtr| 0x24 | int32  | | 20 6E 55 00 | 0x556E20  | data table size }}
{{OBDtr| 0x28 | int32  | | A0 F7 55 00 | 0x55F7A0  | name table offset }}
{{OBDtr| 0x28 | int32  | | A0 F7 55 00 | 0x55F7A0  | name table offset }}
{{OBDtr| 0x2C | int32  | | 35 1D 00 00 | 0x001D35  | name table size }}
{{OBDtr| 0x2C | int32  | | 35 1D 00 00 | 0x001D35  | name table size }}
{{OBDtr| 0x30 | int32  | | 00 00 00 00 |          | unused }}
{{OBDtr| 0x30 | int32  | | 00 00 00 00 |          | OniSplit only: raw table offset }}
{{OBDtr| 0x34 | int32  | | 00 00 00 00 |          | unused }}
{{OBDtr| 0x34 | int32  | | 00 00 00 00 |          | OniSplit only: raw table size }}
{{OBDtr| 0x38 | int32  | | 00 00 00 00 |          | unused }}
{{OBDtr| 0x38 | int32  | | 00 00 00 00 |          | unused }}
{{OBDtr| 0x3C | int32  | | 00 00 00 00 |          | unused }}
{{OBDtr| 0x3C | int32  | | 00 00 00 00 |          | unused }}
{{OBDtrBK|First instance descriptor }}
|}
 
 
 
The "instance descriptors" array stores information about every instance contained in the file.
 
{{Table}}
{{OBD_Table_Header}}
{{OBDtr| 0x00 | tag    | | 41 53 49 41 | 'AISA'    | template tag }}
{{OBDtr| 0x00 | tag    | | 41 53 49 41 | 'AISA'    | template tag }}
{{OBDtr| 0x04 | int32  | | 08 00 00 00 | 0x0008    | data offset (relative to data table) }}
{{OBDtr| 0x04 | int32  | | 08 00 00 00 | 0x0008    | data offset (relative to data table) }}
Line 23: Line 31:
{{OBDtr| 0x10 | int32  | | 00 00 00 00 | 0        | flags; possible values:
{{OBDtr| 0x10 | int32  | | 00 00 00 00 | 0        | flags; possible values:
:0x'''01''' 00 00 00 - unnamed
:0x'''01''' 00 00 00 - unnamed
:0x'''02''' 00 00 00 - empty }}
:0x'''02''' 00 00 00 - empty  
{{OBDtrBK|First name index entry }}
:0x'''04''' 00 00 00 - never used; appears to mean "big endian" data }}
|}
 
 
 
The "name descriptors" array stores the numbers of all named instances in alphabetically order. This allows the engine to do a binary search to faster find instances by name. It is also used when finding instances by type.
 
{{Table}}
{{OBD_Table_Header}}
{{OBDtr| 0x00 | int32  | | 00 00 00 00 | 0    | instance number }}
{{OBDtr| 0x00 | int32  | | 00 00 00 00 | 0    | instance number }}
{{OBDtr| 0x04 | int32  | | 00 00 00 00 | 0    | unused }}
{{OBDtr| 0x04 | int32  | | 00 00 00 00 | 0    | runtime: pointer to instance name }}
{{OBDtrBK|First template table entry }}
|}
 
 
 
The "template descriptor" array contains information about all templates used in the file. The template checksum is used to prevent loading of instance files that are not compatible with the current engine version.
 
{{Table}}
{{OBD_Table_Header}}
{{OBDtr| 0x00 | int64  | | A0 6D 12 00 00 00 00 00 | 0x126DA0    | template checksum }}
{{OBDtr| 0x00 | int64  | | A0 6D 12 00 00 00 00 00 | 0x126DA0    | template checksum }}
{{OBDtr| 0x0C | tag    | | 41 4E 42 41            | 'ABNA'      | template tag }}
{{OBDtr| 0x0C | tag    | | 41 4E 42 41            | 'ABNA'      | template tag }}
{{OBDtr| 0x08 | int32  | | 01 00 00 00            | 1          | number of instances that use this template }}
{{OBDtr| 0x08 | int32  | | 01 00 00 00            | 1          | number of instances that use this template }}
{{OBDtrBK|First instance }}
|}
{{OBDtr| 0x00 | int32  | | 01 00 00 00 | 1          | instance id <nowiki>=</nowiki> (instance number << 8) | 1}}
 
 
 
The data table stores all the instance data.
 
{{Table}}
{{OBD_Table_Header}}
{{OBDtr| 0x00 | int32  | | 01 00 00 00 | 1          | instance id }}
{{OBDtr| 0x04 | int32  | | 01 00 00 02 | 0x02000001 | file id <nowiki>=</nowiki> hash of the filename }}
{{OBDtr| 0x04 | int32  | | 01 00 00 02 | 0x02000001 | file id <nowiki>=</nowiki> hash of the filename }}
{{OBDtr| 0x08 |        | |            |            | [[OBD:File_types|instance specific data]] }}
{{OBDtr| 0x08 |        | |            |            | [[OBD:File_types|instance specific data]] }}
{{OBDtrBK|First name }}
|}
 
 
 
The name table stores all the instance names as C style strings (terminated by 0).
 
{{Table}}
{{OBD_Table_Header}}
{{OBDtr| 0x00 | string  | | 41 49 ... 00 | "AISAlevel1_scripts" | name string (0 terminated) }}
{{OBDtr| 0x00 | string  | | 41 49 ... 00 | "AISAlevel1_scripts" | name string (0 terminated) }}
|}
|}
;Instance (.dat) file
An instance file is a dump of engine's in memory data structures. It is acompanied by a .raw file and a .sep file (the .sep file is only used and present in the PC Demo and Mac versions of the game) which stores additional data (usually large and unstructured like texture or sound data) needed by some instance types. [[OniSplit]] generated .oni files are PC .dat files with all the data contained by the .raw/.sep files appended at the end.
;Binary (.raw, .sep) file
Binary files do not have any file header. The only rule about binary files is that all data parts are stored 32 byte aligned and the first 32 byte of the file are always 0 (reserved to represent NULL pointers). Instances store file offsets into binary files and at loaded time the offsets are converted to pointers.
;Instance file name
An instance file name has the following structure:
levelN_T.dat
where N is the level number (from 0 to 127) and T is the type of file. Known types are "Final" and "Tools". The original exe only loads "Final" files.
;Level 0 file
Level 0 file does not acually contain a level but instances shared across all levels. It is loaded first when the game starts and never unloaded. All other level files are only loaded when the coresponding level starts and unloaded when it ends.
;Instance descriptors
There are 3 types of instance descriptors:
*unnamed - they are referenced by other instances in the same file and the engine never reaches them directly
*named and not empty - they can be referenced by other instances in any file and the engine can use their name or template tag to find them
*named and empty - the instance data is stored in a different file and they exist only to associate an instance id with a name; when an instance references such an instance id the engine searches all the loaded files for a non empty instance with the same name
;Instance ID
The ID of an instance is computed as:
(instance_descriptor_index << 8) <nowiki>|</nowiki> 1.
The 1 allows the engine to know which IDs have already been converted to pointers (a instance pointer will always be 8 byte aligned so it can never have the bit 0 set).
;File ID
The file ID is computed from the name of the instance file. For "_Final" files the file ID is computed as:
(level_number << 25) <nowiki>|</nowiki> 1
Again the 1 allows the engine to know which file IDs have already been converted to pointers.
;Templates
An instance can have pointers to other instances but since pointers are only valid in memory they are converted to instance identifiers when the file is saved and converted back to pointers when the file is loaded into memory. To be able to do this the engine must know where pointers are and this is done using "templates". A template contains:
*a checksum of the data contained by the template (the checksum algorithm is unknown)
*a 4-letter tag used to identify the template (ABNA, ONCC, WMDD etc.)
*a short description of the data structure like "BSP Tree Node Array"
*a list of all data structure's fields and their types
*other data that appears to be unused like size of the fixed part and size of an array element for data structures that contain variable length arrays
708

edits