Line 1:
Line 1:
β
[[Category:Software]]
+
'''Title metadata''' (aka '''TMD''') is a format used to store information about a [[title]] and all its installed contents, including which contents they consist of and their SHA1 hashes.
β
== Nintendo Wii Title-Metadata (tmd) file structure ==
+
TMDs seem to have been originally intended to all be stored in [[:/title/00000001/00000002/data/tmds.sys]], much like ticket.sys on the iQue Player; on release consoles, most TMDs are still stored there, but the TMD files actually used are stored in separate title.tmd files.
β
A "title" is a standalone entity -- a game, a channel, etc. Titles can be made up of multiple "contents". (Don't ask me. I just work here.)
+
== Structure ==
β
+
=== Signed blob header ===
β
Many operations are done in terms of 64-byte blocks, which means you will often see padding out to the nearest 64-byte boundary at the end of a field.
+
{| class="wikitable"
+
|- style="background-color: #ddd;"
+
! Absolute offset
+
! Length
+
! Description
+
|-
+
| 0x000
+
| 4
+
| Signature type (always 0x10001 for RSA-2048 w/ SHA-1)
+
|-
+
| 0x004
+
| 256
+
| Signature covering the main header as well as all CMDs
+
|-
+
| 0x104
+
| 60
+
| Padding for 64-byte alignment
+
|}
β
=== TMD file structure ===
+
=== Main header ===
β
==== Header ====
+
{| class="wikitable"
β
{| style="border-collapse: collapse; padding: 0.2em 0.2em 0.2em 0.2em;"
β
|- style="background-color: #ddd;"
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #cdc;" | '''Start'''
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ccc;" | '''Length'''
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #dcc;" | '''Description'''
β
|- style="background-color: #ddd;"
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ded;" | 0x000
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddd;" | 4
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #edd;" | Signature type
β
|- style="background-color: #ddd;"
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ded;" | 0x004
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddd;" | 256
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #edd;" | Signature
β
|- style="background-color: #ddd;"
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ded;" | 0x104
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddd;" | 60
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #edd;" | Padding modulo 64
β
|- style="background-color: #ddd;"
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ded;" | 0x140
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddd;" | 64
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #edd;" | Issuer
β
|- style="background-color: #ddd;"
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ded;" | 0x180
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddd;" | 1
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #edd;" | Version
β
|- style="background-color: #ddd;"
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ded;" | 0x181
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddd;" | 1
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #edd;" | ca_crl_version
β
|- style="background-color: #ddd;"
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ded;" | 0x182
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddd;" | 1
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #edd;" | signer_crl_version
β
|- style="background-color: #ddd;"
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ded;" | 0x183
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddd;" | 1
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #edd;" | Padding modulo 64
β
|- style="background-color: #ddd;"
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ded;" | 0x184
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddd;" | 8
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #edd;" | System Version
β
|- style="background-color: #ddd;"
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ded;" | 0x18C
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddd;" | 8
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #edd;" | Title ID
β
|- style="background-color: #ddd;"
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ded;" | 0x194
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddd;" | 4
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #edd;" | Title type
β
|- style="background-color: #ddd;"
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ded;" | 0x198
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddd;" | 2
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #edd;" | Group ID
|- style="background-color: #ddd;"
|- style="background-color: #ddd;"
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ded;" | 0x19A
+
! Absolute offset
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddd;" | 62
+
! Length
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #edd;" | reserved
+
! Description
β
|- style="background-color: #ddd;"
+
|-
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ded;" | 0x1D8
+
| 0x140
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddd;" | 4
+
| 64
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #edd;" | Access rights
+
| Certificate issuer
β
|- style="background-color: #ddd;"
+
|-
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ded;" | 0x1DC
+
| 0x180
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddd;" | 2
+
| 1
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #edd;" | Title version
+
| Version
β
|- style="background-color: #ddd;"
+
|-
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ded;" | 0x1DE
+
| 0x181
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddd;" | 2
+
| 1
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #edd;" | Number of contents (nbr_cont)
+
| ca_crl_version
β
|- style="background-color: #ddd;"
+
|-
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ded;" | 0x1E0
+
| 0x182
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddd;" | 2
+
| 1
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #edd;" | boot index
+
| signer_crl_version
β
|- style="background-color: #ddd;"
+
|-
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ded;" | 0x1E2
+
| 0x183
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddd;" | 2
+
| 1
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #edd;" | Padding modulo 64
+
| Is vWii (1 for vWii titles, 0 for normal titles)
β
|- style="background-color: #ddd;"
+
|-
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ded;" | 0x1E4
+
| 0x184
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddd;" | 36*nbr_cont
+
| 8
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #edd;" | Contents
+
| System Version (the IOS that the title needs, Set to 0 for IOS itself. For the boot2, this is identical to the boot2 version.)
+
|-
+
| 0x18C
+
| 8
+
| Title ID
+
|-
+
| 0x194
+
| 4
+
| Title type
+
|-
+
| 0x198
+
| 2
+
| Group ID
+
|-
+
| 0x19A
+
| 2
+
| Zero
+
|-
+
| 0x19C
+
| 2
+
| Region (0: Japan, 1: USA, 2: Europe, 3: Region Free, 4: Korea)
+
|-
+
| 0x19E
+
| 16
+
| Ratings
+
|-
+
| 0x1AE
+
| 12
+
| Reserved
+
|-
+
| 0x1BA
+
| 12
+
| IPC Mask
+
|-
+
| 0x1C6
+
| 18
+
| Reserved
+
|-
+
| 0x1D8
+
| 4
+
| Access rights (flags for [[DVDX|DVD-video access]] and [http://hackmii.com/2009/08/of-tmds-and-hardware/ full PPC hardware access])
+
|-
+
| 0x1DC
+
| 2
+
| Title version
+
|-
+
| 0x1DE
+
| 2
+
| Number of contents (nbr_cont)
+
|-
+
| 0x1E0
+
| 2
+
| boot index (content index for boot file. For [[Broadway]] titles, this is typically the [[NAND Boot Program]]. For [[Starlet]] titles, this is the main binary with the kernel/ES/FS, or the single binary for monolithic IOSes)
+
|-
+
| 0x1E2
+
| 2
+
| Minor version (unused - the term typically refers to the lower half of the main version instead)
|}
|}
β
==== Content ====
+
β
{| style="border-collapse: collapse; padding: 0.2em 0.2em 0.2em 0.2em;"
+
=== Content metadata (CMD) ===
β
|- style="background-color: #ddd;"
+
Following the main header (starting at offset 0x1E4) is a list of CMDs (one per content).
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #cdc;" | '''Start'''
+
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ccc;" | '''Length'''
+
{| class="wikitable"
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #dcc;" | '''Description'''
+
|-
β
|- style="background-color: #ddd;"
+
! Offset
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ded;" | 0x00
+
! Length
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddd;" | 4
+
! Description
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #edd;" | Content ID
+
|-
β
|- style="background-color: #ddd;"
+
| 0x00
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ded;" | 0x04
+
| 4
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddd;" | 2
+
| Content ID
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #edd;" | Index
+
|-
β
|- style="background-color: #ddd;"
+
| 0x04
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ded;" | 0x06
+
| 2
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddd;" | 2
+
| Index
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #edd;" | Type
+
|-
β
|- style="background-color: #ddd;"
+
| 0x06
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ded;" | 0x08
+
| 2
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddd;" | 8
+
| Type (0x0001: Normal, 0x4001: DLC, 0x8001: [[:/shared1|Shared]])
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #edd;" | Size
+
|-
β
|- style="background-color: #ddd;"
+
| 0x08
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ded;" | 0x10
+
| 8
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddd;" | 20
+
| Size
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #edd;" | SHA1 hash
+
|-
+
| 0x10
+
| 20
+
| SHA1 hash
|}
|}
β
==== Certificates ====
+
β
{| style="border-collapse: collapse; padding: 0.2em 0.2em 0.2em 0.2em;"
+
=== Certificates ===
β
|- style="background-color: #ddd;"
+
{| class="wikitable"
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #cdc;" | '''Start'''
+
|-
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ccc;" | '''Length'''
+
! Offset
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #dcc;" | '''Description'''
+
! Length
β
|- style="background-color: #ddd;"
+
! Description
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ded;" | 0x000
+
|-
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddd;" | 4
+
| 0x000
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #edd;" | Signature type
+
| 4
β
|- style="background-color: #ddd;"
+
| Signature type
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ded;" | 0x004
+
|-
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddd;" | 256
+
| 0x004
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #edd;" | Signature
+
| 256
β
|- style="background-color: #ddd;"
+
| Signature
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ded;" | 0x104
+
|-
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddd;" | 32
+
| 0x104
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #edd;" | Issuer
+
| 64
β
|- style="background-color: #ddd;"
+
| Issuer
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ded;" | 0x124
+
|-
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddd;" | 4
+
| 0x124
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #edd;" | Tag
+
| 4
β
|- style="background-color: #ddd;"
+
| Public Key Type
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ded;" | 0x128
+
|-
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddd;" | 64
+
| 0x128
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #edd;" | Name
+
| 64
β
|- style="background-color: #ddd;"
+
| Name
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ded;" | 0x168
+
|-
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddd;" |
+
| 0x12C
β
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #edd;" | Key
+
| 4
+
| Date
+
|-
+
| 0x16C
+
|
+
| Public Key
|}
|}
β
=== C code application ===
+
== Example code application ==
<source lang="c">
<source lang="c">
typedef unsigned char u8;
typedef unsigned char u8;
Line 162:
Line 190:
u32 cid; // content id
u32 cid; // content id
u16 index; // # number of the file
u16 index; // # number of the file
β
u16 type;
+
u16 type; // normal: 0x0001; dlc: 0x4001; shared: 0x8001
u64 size;
u64 size;
u8 hash [20]; // SHA1 hash content
u8 hash [20]; // SHA1 hash content
Line 168:
Line 196:
</source>
</source>
<source lang="c">
<source lang="c">
β
enum sig_type {
+
enum sig_type {
β
RSA_2048 = 0x00010001,
+
RSA_2048 = 0x00010001,
β
RSA_4096 = 0x00010000
+
RSA_4096 = 0x00010000
β
};
+
};
+
+
// High 32 bits of the title ID
+
enum title_type : u32 {
+
System = 0x00000001,
+
Game = 0x00010000,
+
Channel = 0x00010001,
+
SystemChannel = 0x00010002,
+
GameWithChannel = 0x00010004,
+
DLC = 0x00010005,
+
HiddenChannel = 0x00010008,
+
};
+
+
// title_type (offset 0x194)
+
enum title_flags {
+
// All official titles have this flag set.
+
Default = 0x1,
+
Unknown_0x4 = 0x4,
+
// Used for DLC titles.
+
Data = 0x8,
+
Unknown_0x10 = 0x10,
+
// Seems to be used for WFS titles.
+
Maybe_WFS = 0x20,
+
Unknown_CT = 0x40,
+
};
</source>
</source>
<source lang="c">
<source lang="c">
Line 182:
Line 234:
u8 ca_crl_version;
u8 ca_crl_version;
u8 signer_crl_version;
u8 signer_crl_version;
β
u8 fill2;
+
u8 vwii;
u64 sys_version;
u64 sys_version;
u64 title_id;
u64 title_id;
Line 192:
Line 244:
u16 num_contents;
u16 num_contents;
u16 boot_index;
u16 boot_index;
β
u16 fill3;
+
u16 fill2;
content_record contents[num_contents];
content_record contents[num_contents];
} tmd;
} tmd;
Line 201:
Line 253:
u8 sig[256]; // 256 for RSA_2048, 512 for RSA_4096
u8 sig[256]; // 256 for RSA_2048, 512 for RSA_4096
u8 issuer[32];
u8 issuer[32];
β
u32 tag; // identifies what is being signed
+
u32 key_type; // the type of public key
u8 name[64]; // name of thing being signed
u8 name[64]; // name of thing being signed
u8 key[...];
u8 key[...];
</source>
</source>
β
[[Category:Wii_Software]]
+
+
There is also a structure called a TmdView which is select sections of the Tmd. It has a length of 0x60+0x10*number_of_contents. The structure is somewhat like this [as returned by ES_GetTmdView] :
+
<source lang="c">
+
struct tmd_view_content_t
+
{
+
uint32_t id;
+
uint16_t index;
+
uint16_t type;
+
uint64_t size;
+
};
+
+
struct tmd_view_t
+
{
+
uint8_t version; // 0x0000;
+
uint8_t filler[3];
+
uint64_t ios_title_id; //0x0004
+
uint64_t title_id; // 0x00c
+
uint32_t title_type; //0x0014
+
uint16_t group_id; //0x0018
+
uint8_t reserved[0x3e]; //0x001a this is the same reserved 0x3e bytes from the tmd
+
uint16_t title_version; //0x0058
+
uint16_t number_contents; //0x005a
+
tmd_view_content_t contents[]; //0x005c
+
};
+
</source>
+
+
[[Category:File formats]]