Line 3:
Line 3:
'''/dev/di''' is the [[IOS]] driver used to control the disc drive. This documentation is mostly based on the most recent version (dated Jun 3 2009 07:49:09 and included in [[IOS58]] and [[IOS80]]). Names are based on function names found in Nintendo titles (which print an error message including the name if the Ioctl or Ioctlv fails). DVDLowRequestAudioStatus and DVDLowAudioStream are not found in Wii titles, but the names can be found in [https://wiki.dolphin-emu.org/index.php?title=Ships_with_Debugging_Symbols debug symbols included in various Gamecube games].
'''/dev/di''' is the [[IOS]] driver used to control the disc drive. This documentation is mostly based on the most recent version (dated Jun 3 2009 07:49:09 and included in [[IOS58]] and [[IOS80]]). Names are based on function names found in Nintendo titles (which print an error message including the name if the Ioctl or Ioctlv fails). DVDLowRequestAudioStatus and DVDLowAudioStream are not found in Wii titles, but the names can be found in [https://wiki.dolphin-emu.org/index.php?title=Ships_with_Debugging_Symbols debug symbols included in various Gamecube games].
+
== Input structure ==
The input to all /dev/di commands (other than enable DVD video) is the following struct, which must be sized 0x20 and aligned 4:
The input to all /dev/di commands (other than enable DVD video) is the following struct, which must be sized 0x20 and aligned 4:
Line 23:
Line 24:
(DiIoctl) Note: This is normal for DVD software before 6-24
(DiIoctl) Note: This is normal for DVD software before 6-24
</pre></blockquote>
</pre></blockquote>
+
+
This probably means IOCTL numbers were created on June 24th of some year.
== Return values ==
== Return values ==
Line 76:
Line 79:
== Version history ==
== Version history ==
−
There are '''13''' known normal versions (along with '''9''' [[#vWii note|matching vWii versions]]) of the DI driver found in various [[IOS History|IOS versions]], based on the IOS versions present on NUS and those found on various game discs. These are generalized into 5 version families, based on observable behavior (this is not strictly chronological, presumably as Nintendo was working on multiple versions with the same features at the same time). It is quite likely that there are additional changes not noted here.
+
There are '''14''' known normal versions (along with '''9''' [[#vWii note|matching vWii versions]]) of the DI driver found in various [[IOS History|IOS versions]], based on the IOS versions present on NUS and those found on various game discs. These are generalized into 5 version families, based on observable behavior (this is not strictly chronological, presumably as Nintendo was working on multiple versions with the same features at the same time). It is quite likely that there are additional changes not noted here.
The DI driver includes a full set of [[:/dev/es|ES]] IoctlV wrappers, although it only uses ES_DiVerify and ES_DiVerifyWithTicketView. It also includes instructions for all [[syscalls]], even though most are not used. Both of those change across versions, even though those differences do not actually show up in practice.
The DI driver includes a full set of [[:/dev/es|ES]] IoctlV wrappers, although it only uses ES_DiVerify and ES_DiVerifyWithTicketView. It also includes instructions for all [[syscalls]], even though most are not used. Both of those change across versions, even though those differences do not actually show up in practice.
Line 210:
Line 213:
|data-sort-value="z" {{Partial|vWii}}
|data-sort-value="z" {{Partial|vWii}}
|-
|-
−
|rowspan="6" | [[#Group E|E]]
+
|rowspan="7" | [[#Group E|E]]
|data-sort-value="1227541149"| [[#Nov 24 2008 15:39:09|Nov 24 2008 15:39:09]]
|data-sort-value="1227541149"| [[#Nov 24 2008 15:39:09|Nov 24 2008 15:39:09]]
−
|rowspan="6" {{No}}
+
|rowspan="7" {{No}}
−
|rowspan="6" {{Yes}}
+
|rowspan="7" {{Yes}}
−
|rowspan="6" {{Yes}}
+
|rowspan="7" {{Yes}}
−
|rowspan="6" {{Partial}}
+
|rowspan="7" {{Partial}}
−
|rowspan="6" {{Partial}}
+
|rowspan="7" {{Partial}}
−
|rowspan="6" {{Partial}}
+
|rowspan="7" {{Partial}}
−
|rowspan="6" {{Yes}}
+
|rowspan="7" {{Yes}}
−
|rowspan="6" {{Yes}}
+
|rowspan="7" {{Yes}}
−
|rowspan="6" {{Yes}}
+
|rowspan="7" {{Yes}}
−
|rowspan="6" {{Yes}}
+
|rowspan="7" {{Yes}}
−
|rowspan="6" {{Yes}}
+
|rowspan="7" {{Yes}}
−
|rowspan="6" data-sort-value="121"| 0x79
+
|rowspan="7" data-sort-value="121"| 0x79
|data-sort-value="66"| 0x42
|data-sort-value="66"| 0x42
+
| {{No}}
+
|-
+
|data-sort-value="1239037807"| [[#Apr 6 2009 17:10:07|Apr 6 2009 17:10:07]]
+
|data-sort-value="68"| 0x44
| {{No}}
| {{No}}
|-
|-
Line 266:
Line 273:
{| class="wikitable"
{| class="wikitable"
−
! MD5
+
! SHA-1
−
| colspan="3"| 43575ada3e27b20543fc13be1395800e
+
| colspan="3"| 5032764e723e0db7e6d7f434219c9d50289a1cab
|-
|-
! Thing
! Thing
Line 320:
Line 327:
{| class="wikitable"
{| class="wikitable"
−
! MD5
+
! SHA-1
−
| colspan="3"| 43d861243ca8ae5370b08e810566bc06
+
| colspan="3"| 9dce75d14e01f6efc8d56821c139490792b8b3f9
|-
|-
! Thing
! Thing
Line 368:
Line 375:
{| class="wikitable"
{| class="wikitable"
−
! MD5
+
! SHA-1
| colspan="3" {{Not tested|Varies}}
| colspan="3" {{Not tested|Varies}}
|-
|-
Line 412:
Line 419:
{| class="wikitable"
{| class="wikitable"
−
! MD5
+
! SHA-1
| colspan="3" {{Not tested|Varies}}
| colspan="3" {{Not tested|Varies}}
|-
|-
Line 501:
Line 508:
{| class="wikitable"
{| class="wikitable"
−
! MD5
+
! SHA-1
−
| colspan="3"| c808d8b90a74a4ee808b199a1b1e8d53
+
| colspan="3"| 260be947a08f57f6ef51086427fe222fd4040399
|-
|-
! Thing
! Thing
Line 545:
Line 552:
{| class="wikitable"
{| class="wikitable"
−
! MD5
+
! SHA-1
−
| colspan="3"| 0f6a5281e0cec477147baabf0c69fbdc
+
| colspan="3"| fb308a9a1d9341df9517db155f4383162325dcc0
|-
|-
! Thing
! Thing
Line 601:
Line 608:
{| class="wikitable"
{| class="wikitable"
−
! MD5
+
! SHA-1
−
| colspan="3"| 366021c440e6377044f8ca8c94e2e6bc
+
| colspan="3"| 57667279972205462da427535a75a913574f2798
|-
|-
! Thing
! Thing
Line 645:
Line 652:
{| class="wikitable"
{| class="wikitable"
−
! MD5
+
! SHA-1
−
| colspan="3"| 49f714dd1a0985fbd4c44ee9fe4f945a
+
| colspan="3"| 92b9a637383729b25fbcb663f2895f66c6d9c987
|-
|-
! Thing
! Thing
Line 686:
Line 693:
==== Jul 24 2008 20:08:45 ====
==== Jul 24 2008 20:08:45 ====
−
Only found in [[IOS38]].
+
Only found in [[IOS38]]. Note that this also has a version string of <code>$IOSVersion: DIP: 07/24/08 20:08:<mark>44</mark> 64M $</code>, probably just due to the two timestamps being determined at separate instants.
Identical to the Jul 14 2008 19:25:32 build apart from the priority of the main thread being set to 0x1b instead of 0x54 (all versions other than this and Jul 24 2008 00:30:13 use 0x54). This results in byte differences at address 20207c2c (file offset 7d54), as well as in some ELF header area (file offset 114), and the timestamps.
Identical to the Jul 14 2008 19:25:32 build apart from the priority of the main thread being set to 0x1b instead of 0x54 (all versions other than this and Jul 24 2008 00:30:13 use 0x54). This results in byte differences at address 20207c2c (file offset 7d54), as well as in some ELF header area (file offset 114), and the timestamps.
{| class="wikitable"
{| class="wikitable"
−
! MD5
+
! SHA-1
−
| colspan="3"| ef1a8c1270f82e0993f504f1e17a5152
+
| colspan="3"| b4cdc54a5912d64f9ef1e516931ab32d64677a9c
|-
|-
! Thing
! Thing
Line 735:
Line 742:
{| class="wikitable"
{| class="wikitable"
−
! MD5
+
! SHA-1
−
| colspan="3"| c85216d854989109ec95912e8993d4c6
+
| colspan="3"| e04f3abe93ca9b9a2518c2ddc3d273e43caed1f8
|-
|-
! Thing
! Thing
Line 779:
Line 786:
{| class="wikitable"
{| class="wikitable"
−
! MD5
+
! SHA-1
−
| colspan="3"| 777436b7131e08e59672a927d6ea3559
+
| colspan="3"| b62ad5ea5a2e03d2fb73e93dad1d34c102ec357a
|-
|-
! Thing
! Thing
Line 824:
Line 831:
{| class="wikitable"
{| class="wikitable"
! MD5
! MD5
−
| colspan="3"| 0878e12c454539450e9e75c19d5a75e9
+
| colspan="3"| 1d1723825d53b5389ec80c89c8a3aa06701ae07d
|-
|-
! Thing
! Thing
Line 892:
Line 899:
* [[IOS53]] (all versions)
* [[IOS53]] (all versions)
* [[IOS55]] (all versions)
* [[IOS55]] (all versions)
+
+
Note that this also has a version string of <code>$IOSVersion: DIP: 07/11/08 14:34:<mark>26</mark> 64M $</code>, probably just due to the two timestamps being determined at separate instants.
{| class="wikitable"
{| class="wikitable"
−
! MD5
+
! SHA-1
−
| colspan="3"| 382d4a5cafdb1e28ba039d25db7c4c1f
+
| colspan="3"| bff35f53a0ed9f69b15a224552fcc73372308099
|-
|-
! Thing
! Thing
Line 940:
Line 949:
{| class="wikitable"
{| class="wikitable"
−
! MD5
+
! SHA-1
−
| colspan="3"| 108011e89e557d4e8adf1a02f87cb8ea
+
| colspan="3"| 0fc5a88a327ee2b5c4ce3dc05faf8c7ef3bbcc1b
|-
|-
! Thing
! Thing
Line 986:
Line 995:
{| class="wikitable"
{| class="wikitable"
−
! MD5
+
! SHA-1
−
| colspan="3"| 72122c88cdcd4279cc09e197d3079624
+
| colspan="3"| d254a265d25d96723a566e6f877f5df05a645699
|-
|-
! Thing
! Thing
Line 1,030:
Line 1,039:
{| class="wikitable"
{| class="wikitable"
−
! MD5
+
! SHA-1
−
| colspan="3"| 4ec5199c40a08746ebd6931e181d6737
+
| colspan="3"| e7a0824785268df455d56c1803620eff180d6556
|-
|-
! Thing
! Thing
Line 1,074:
Line 1,083:
{| class="wikitable"
{| class="wikitable"
−
! MD5
+
! SHA-1
−
| colspan="3"| ecbcd75c788ffab75f291dd510e440c0
+
| colspan="3"| 8835422c143de4b359fea4a0a56aef9386caa53d
|-
|-
! Thing
! Thing
Line 1,131:
Line 1,140:
{| class="wikitable"
{| class="wikitable"
−
! MD5
+
! SHA-1
−
| colspan="3"| 48e1be8f767feb59cbc51aa4329d735a
+
| colspan="3"| 96b035dafcfaf826d1772abd07b8014aed15035f
|-
|-
! Thing
! Thing
Line 1,143:
Line 1,152:
| 139B0000
| 139B0000
| 0x7F00
| 0x7F00
+
|-
+
| Data (ES vars)
+
| 20208000
+
| 139B8000
+
| 0x140
+
|-
+
| BSS (zero'd)
+
| 20209000
+
| 139B9000
+
| 0x2BDC4
+
|-
+
| Stack
+
| 2022cdc4
+
| ?
+
| 0x8000
+
|-
+
| Protected heap
+
| 20209020
+
| ?
+
| 0x4000
+
|-
+
| Open heap
+
| 13600000
+
| ?
+
| 0x18000
+
|}
+
+
==== Apr 6 2009 17:10:07 ====
+
+
Used exclusively by [[IOS56]] v5146, which is not found on NUS (but can be found on e.g. Guitar Hero 5).
+
+
No changes to the actual driver code from the Nov 24 2008 version, but some of the ES wrapper code changed. These also cause string constants to shift, which makes byte comparisons slightly annoying. The changes:
+
+
* ES_AddTicket (20204514, Ioctlv 0x01) no longer always uses a size of 0x2a4, but will instead use 0x2a4 plus a 32-bit size at offset 0x2a8 if the byte at offset 0x1bc is nonzero.
+
* ES_GetTicketFromView (20204fc0, Ioctlvs 0x43 and 0x44) was added
+
+
{| class="wikitable"
+
! SHA-1
+
| colspan="3"| 9a915fd77389a79c7fa516e4aac4e30e4e1174ad
+
|-
+
! Thing
+
! Virtual address
+
! Physical address
+
! Size
+
|-
+
| Code (and entry point)
+
| 20200000
+
| 139B0000
+
| 0x7FB4
|-
|-
| Data (ES vars)
| Data (ES vars)
Line 1,184:
Line 1,242:
No changes to the actual driver code from the Nov 24 2008 version, but some of the ES wrapper code changed. These also cause string constants to shift, which makes byte comparisons slightly annoying. The changes:
No changes to the actual driver code from the Nov 24 2008 version, but some of the ES wrapper code changed. These also cause string constants to shift, which makes byte comparisons slightly annoying. The changes:
−
* ES_AddTicket (20204514, Ioctlv 0x01) no longer always uses a size of 0x2a4, but will instead use 0x2a4 plus a 32-bit size at offset 0x2a8 if the byte at offset 0x1bc is nonzero.
−
* ES_GetTicketFromView (20204fc0, Ioctlvs 0x43 and 0x44) was added
* Ioctlv 0x45 (20205cb8) was added
* Ioctlv 0x45 (20205cb8) was added
{| class="wikitable"
{| class="wikitable"
−
! MD5
+
! SHA-1
−
| colspan="3"| 89f7dc21f07e2cae97c3a571b23d8abd
+
| colspan="3"| 4e04e88ec7250de84a1e788ae69fdad9351330a8
|-
|-
! Thing
! Thing
Line 1,233:
Line 1,289:
{| class="wikitable"
{| class="wikitable"
−
! MD5
+
! SHA-1
−
| colspan="3"| a92e0407a61fe6812724cf1ed4ccab68
+
| colspan="3"| d5dfeb42909a20453c0f574e9a8c41f50792bf8f
|-
|-
! Thing
! Thing
Line 1,277:
Line 1,333:
{| class="wikitable"
{| class="wikitable"
−
! MD5
+
! SHA-1
−
| colspan="3"| c4d353400390fc748813076464b33996
+
| colspan="3"| 43cbc9d451df6296214347f8a33349b2dda843f0
|-
|-
! Thing
! Thing
Line 1,321:
Line 1,377:
{| class="wikitable"
{| class="wikitable"
−
! MD5
+
! SHA-1
−
| colspan="3"| 6912011de2cf16c358865f2cc8316055
+
| colspan="3"| e961f817c53ad3d87af96635df99bc6ee70ed056
|-
|-
! Thing
! Thing
Line 1,365:
Line 1,421:
{| class="wikitable"
{| class="wikitable"
−
! MD5
+
! SHA-1
−
| colspan="3"| 6b87425010ac8256ee5d331a8073838b
+
| colspan="3"| 86bec5ad6815c2bf1f154690f2abd0e6141f0f8b
|-
|-
! Thing
! Thing
Line 1,907:
Line 1,963:
<li>0x83 DVDLowGetLength</li>
<li>0x83 DVDLowGetLength</li>
<li>0x84 Get DIIMMBUF</li>
<li>0x84 Get DIIMMBUF</li>
−
<li>0x85 DVDLowUnmaskCoverInterrupt</li>
+
<li>0x85 DVDLowMaskCoverInterrupt</li>
<li>0x86 DVDLowClearCoverInterrupt</li>
<li>0x86 DVDLowClearCoverInterrupt</li>
−
<li>0x87</li>
+
<li>0x87 DVDLowUnmaskStatusInterrupts</li>
<li>0x88 DVDLowGetCoverStatus</li>
<li>0x88 DVDLowGetCoverStatus</li>
−
<li>0x89 Enable Cover Interrupt</li>
+
<li>0x89 DVDLowUnmaskCoverInterrupt</li>
<li>0x8B DVDLowOpenPartition ioctl</li>
<li>0x8B DVDLowOpenPartition ioctl</li>
<li>0x8E DVDLowEnableDvdVideo</li>
<li>0x8E DVDLowEnableDvdVideo</li>
Line 2,185:
Line 2,241:
|}
|}
−
=== 0x85 DVDLowUnmaskCoverInterrupt ===
+
=== 0x85 DVDLowMaskCoverInterrupt ===
+
+
Disables the cover interrupt by clearing bit 1 of DICVR (leaving bit zero unchanged). Does not clear the cover interrupt if it is currently asserted (does not write bit 2). Actual code is <code>DICVR = (DICVR & ~4 & ~2)</code>.
−
Disables the cover interrupt by clearing bit 1 of DICVR (leaving bit zero unchanged). Does not clear the cover interrupt if it is currently asserted (does not write bit 2).
+
Titles have a DVDLowMaskCoverInterrupt function that is dummied out to always return 1; this function is used by DVDInit in the exact same place that gamecube titles write <code>DICVR = 0</code> (which should be equivalent, as writes to bit 0 which indicates the cover status presumably do nothing{{check}}). However, since it is stubbed out, there is no way of being sure that 0x85 was actually used by that function.
The output buffer is not used, and it may be null. Its size is not checked.
The output buffer is not used, and it may be null. Its size is not checked.
Line 2,219:
Line 2,277:
|}
|}
−
=== <s>0x87</s> ===
+
=== <s>0x87 DVDLowUnmaskStatusInterrupts</s> ===
−
Dummied out; does nothing (and always returns 1). Possibly an ID reserved for a PPC-only command (DVDLowBreak?), as is also done with DVDLowSetSpinupFlag?
+
Dummied out; does nothing (and always returns 1).
+
+
Titles have a DVDLowUnmaskStatusInterrupts function that is dummied out to always return 1; this function is used by DVDInit in the exact same place that gamecube titles write <code>DISR = 0x2a</code> (which enables DEINTMASK, TCINTMASK, and BRKINTMASK, and does not clear any asserted interrupts). However, since it is stubbed out, there is no way of being sure that 0x87 was actually used by that function.
The output buffer is not used, and it may be null. Its size is not checked.
The output buffer is not used, and it may be null. Its size is not checked.
Line 2,253:
Line 2,313:
|}
|}
−
=== 0x89 Enable Cover Interrupt ===
+
=== 0x89 DVDLowUnmaskCoverInterrupt ===
+
+
Enables the cover interrupt by setting bit 1 of DICVR (leaving bit zero unchanged). Does not clear the cover interrupt if it is currently asserted (does not write bit 2). Actual code is <code>DICVR = ((DICVR & ~4) | 2)</code>.
−
Enables the cover interrupt by setting bit 1 of DICVR (leaving bit zero unchanged). Does not clear the cover interrupt if it is currently asserted (does not write bit 2).
+
Debug symbols list a function called DVDLowUnmaskCoverInterrupt, but no actual function remains as it was removed as unused (and even if it did still exist, it presumably would be dummied out to just return 1 as it is only 8 bytes). Therefore, there is no way to be certain that 0x89 actually was called DVDLowUnmaskCoverInterrupt, but it seems very likely based on DVDLowMaskCoverInterrupt.
The output buffer is not used, and it may be null. Its size is not checked.
The output buffer is not used, and it may be null. Its size is not checked.
Line 2,272:
Line 2,334:
=== 0x8A DVDLowReset ===
=== 0x8A DVDLowReset ===
−
Resets the drive, using [[IOS/Syscalls|syscalls]] 0x44, 0x45, and 0x46. If a reset is already in progress (syscall_check_di_reset returns true), then it immediately calls syscall_deassert_di_reset; otherwise, it calls syscall_assert_di_reset, waits 12µs, and then calls syscall_deassert_di_reset. Afterwards, registers are reset in the same way as DVDLowNotifyReset other than the cover interrupt. The cover interrupt is temporarilly disabled during this process, but is reenabled afterwards if it was enabled before.
+
Resets the drive, using [[IOS/Syscalls|syscalls]] 0x44, 0x45, and 0x46. If a reset is already in progress (check_di_reset returns true), then it immediately calls deassert_di_reset; otherwise, it calls assert_di_reset, waits 12µs, and then calls deassert_di_reset. Afterwards, registers are reset in the same way as DVDLowNotifyReset other than the cover interrupt. The cover interrupt is temporarilly disabled during this process, but is reenabled afterwards if it was enabled before.
Enable spinup is passed to syscall 0x4e, which activates the DI_SPIN [[Hardware/Hollywood_GPIOs|GPIO]] if it is 0 and disables it otherwise.
Enable spinup is passed to syscall 0x4e, which activates the DI_SPIN [[Hardware/Hollywood_GPIOs|GPIO]] if it is 0 and disables it otherwise.