In memory of Ben “bushing” Byer, who passed away on Monday, February 8th, 2016.

Difference between revisions of "Wiimote/Extension Controllers"

From WiiBrew
Jump to navigation Jump to search
m (The word protocol was accidentally spelled "Ppotocol" in the 3rd paragraph)
 
(142 intermediate revisions by 49 users not shown)
Line 1: Line 1:
The Wiimote includes a 6-pin expansion port that allows external peripherals to be connected to it. Communications are bidirectional synchronous serial (the protocol is unknown), and the devices provide a virtual register block of length 0x100 that is mapped at 0xa40000 in the Wiimote's address space. '''Communications are encrypted''', as detailed in [[Wiimote#Extension Controllers]]. Nintendo has currently released two extensions, the [[#Nunchuk|Nunchuk]] and the [[#Classic Controller|Classic Controller]]. In addition, Guitar Hero guitars also act as extension controllers. The Balance board acts like a dummy Wiimote with its own type of extension controller.
+
[[File:Wii_Connector.jpg|200px|thumb|right|Pins on the Wii expansion controller cable]]The Wii Remote includes a 6-pin expansion port that allows external peripherals to be connected to it. Communications are bidirectional synchronous serial I2C, and the devices provide a virtual register block of length 0x100 that is mapped at 0xa40000 in the Wii Remote's address space (the 7-bit I2C address is 0x52). '''Communications are encrypted''', as detailed below. Nintendo released two extensions, the [[Wiimote/Extension Controllers/Nunchuck|Nunchuk]] and the [[Wiimote/Extension Controllers/Classic Controller|Classic Controller]]. In addition, Guitar Hero guitars and drums also act as extension controllers. The [[Wii Balance Board|balance board]] acts like a dummy Wiimote with its own type of built-in extension controller. There are also many third party versions of classic controllers and nunchuks.
  
Peripheral Protocol is 400kHz "fast" I2C, with slave address 0x52. It should be possible to make homebrew peripherals; see the [http://www.kako.com/neta/2007-003/2007-003.html Tenkey project] (japanese) or the [http://www.twiidler.org Twiidler project]
+
The Wii Motion Plus is a small expansion which itself has an expansion port, allowing people to plug the Nunchuk into the Wii Motion Plus and the Wii Motion Plus into the Wii Remote.
  
= Nunchuk =
+
The peripheral Protocol is 400kHz "fast" I2C, with slave address 0x52. It is easy to make homebrew peripherals; see the [http://www.kako.com/neta/2007-003/2007-003.html Tenkey numberpad project](Japanese), [http://www.kako.com/neta/2007-004/2007-004.html gyro project] (Japanese), the [http://www.twiidler.org Twiidler project], [http://code.google.com/p/circle-of-current/wiki/WiiDrumController the DIY Wii drum controller], the [http://www.openchord.org OpenChord Guitar], or the [http://youtube.com/watch?v=eBN5kq_VuiU DIY Wii guitar proof of concept demo video].
The Nunchuk is identified by the 16-bit constant 0x0000 (0xFEFE encrypted) at register address 0xa400fe. It provides three-axis acceleration data, two digital buttons, and an X-Y analog stick.
 
== Data Format ==
 
The Nunchuk reports its information as 6 bytes of data, readable at 0xa40008 and streamable using Data Reporting Modes that include Extension bytes (unused bytes are filled with 0x00). The data is packed into the six bytes as follows ('''after decryption'''):
 
  
{| style="border-collapse: collapse; padding: 0.2em 0.2em 0.2em 0.2em; text-align: center;"
+
== Registers / Initialization ==
|- style="background-color: #ddd;"
 
| style="background-color: #fff;" |  
 
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center;" colspan="8"| '''Bit'''
 
|- style="background-color: #cdc;"
 
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddd;" | '''Byte'''
 
| style="border: 1px solid #ccc; padding: 0.2em; width:3.3em;" | '''7'''
 
| style="border: 1px solid #ccc; padding: 0.2em; width:3.3em;" | '''6'''
 
| style="border: 1px solid #ccc; padding: 0.2em; width:3.3em;" | '''5'''
 
| style="border: 1px solid #ccc; padding: 0.2em; width:3.3em;" | '''4'''
 
| style="border: 1px solid #ccc; padding: 0.2em; width:3.3em;" | '''3'''
 
| style="border: 1px solid #ccc; padding: 0.2em; width:3.3em;" | '''2'''
 
| style="border: 1px solid #ccc; padding: 0.2em; width:3.3em;" | '''1'''
 
| style="border: 1px solid #ccc; padding: 0.2em; width:3.3em;" | '''0'''
 
|- style="background-color: #ded;"
 
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eee;" | 0
 
| style="border: 1px solid #ccc; padding: 0.2em;" colspan="8" | '''SX'''<span style="color: #777;">&lt;<span style="color: #c00;">7:0</span>&gt;</span>
 
|- style="background-color: #ded;"
 
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eee;" | 1
 
| style="border: 1px solid #ccc; padding: 0.2em;" colspan="8" | '''SY'''<span style="color: #777;">&lt;<span style="color: #c00;">7:0</span>&gt;</span>
 
|- style="background-color: #ded;"
 
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eee;" | 2
 
| style="border: 1px solid #ccc; padding: 0.2em;" colspan="8" | '''AX'''<span style="color: #777;">&lt;<span style="color: #c00;">7:0</span>&gt;</span>
 
|- style="background-color: #ded;"
 
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eee;" | 3
 
| style="border: 1px solid #ccc; padding: 0.2em;" colspan="8" | '''AY'''<span style="color: #777;">&lt;<span style="color: #c00;">7:0</span>&gt;</span>
 
|- style="background-color: #ded;"
 
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eee;" | 4
 
| style="border: 1px solid #ccc; padding: 0.2em;" colspan="8" | '''AZ'''<span style="color: #777;">&lt;<span style="color: #c00;">7:0</span>&gt;</span>
 
|- style="background-color: #ded;"
 
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eee;" | 5
 
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddd; color:#888;" colspan="6" | Unknown
 
| style="border: 1px solid #ccc; padding: 0.2em;" | '''B'''<span style="color: #c00;">C</span>
 
| style="border: 1px solid #ccc; padding: 0.2em;" | '''B'''<span style="color: #c00;">Z</span>
 
|}
 
SX,SY are the Analog Stick X and Y positions, while AX, AY, and AZ are the accelerometer data (in the same format as described in [[Wiimote#Accelerometer]]). BC and BZ are the state of the C and Z buttons (0=pressed).
 
  
Nintendo games calibrate the center position of the Analog Stick upon power-up or insertion of the Nunchuk.
+
{{BoxCenter|'''Reminder'''<br />Remember to set bit 2 (0x04) on the first byte of the Output Reports to write to registers!}}
  
== Wireless Nunchuk ==
+
Wii Motion Plus is mapped at register 0xa60000.
It looks like Wireless Nunchuks operate differently than the original Nunchuk-Extension. Anyone have a clue?
 
  
=== Nyko Kama ===
+
Other extension Controllers are mapped at register address 0xa40000. The data is 0x100 bytes long, and it is mirrored across the entire 16-bit address space up to 0xa4FFFF. These registers are readable and writable. Communications are optionally encrypted, and explicit initialization is required to disable encryption (see below).
  
The Nyko Kama is just a simple wireless Nunchuk, with a Sync and Power button. It uses two AAA batteries.
+
In encrypted mode, bytes can be decrypted using the following transformation:
  
The Nyko Kama does not work with current homebrew or PC Wiimote software. It is unknown how to make it work.
+
decrypted_byte = (encrypted_byte XOR table1[address%8]) + table2[address%8]
  
=== Datel Blade-FX ===
+
Where table1 and table2 are 8-byte tables calculated based on the 16-byte key sent by the host, and address is the address of the byte being read (only the bottom 3 bits matter). If the host key is 16 zero bytes, table1[x] and table2[x] are all 0x97. The value 0x17 was used previously, which is equivalent. Proof of why any two pairs of table entries a,b and x,y are equivalent if (a^b^x^y)&0x80 == 0 is left as an exercise to the reader.
  
This wireless Nunchuk can emulate either a Nunchuk or a Classic Controller. At startup it behaves like a Nunchuk, but pressing the "Classic" button puts it into Classic Controller mode. While in Classic Controller mode, the FUNCTION LED will be solidly lit (red). Classic controller emulation allows it to be used in the Wii system menu, with the joystick controlling the cursor, and the Z button selecting, and the C button having no visible effect. It also allows it to work (according to the instructions) with some Virtual Console games.
+
The calculations should be performed '''mod 256''', that is, truncated to 8 bits (in languages such as C and Python, use <span style="font-family: monospace;">&0xFF</span> or work directly with 8-bit datatypes).
  
This wireless Nunchuk also has a cheat feature that remembers a motion that you do and repeats it continuously without you having to keep doing the motion. That is done with the "FX" button.
+
If the Wii Remote is initialized using the new method listed below, then the encryption of the extension bytes is disabled and they need not be decrypted using the transform listed above.
  
There is also a "SYNC" button where the cord would be. Pressing it once connects the wireless nunchuk. While connected, the STATUS LED will slowly flash.  
+
The Wii Remote will stream data bytes from the Extension Controller starting at offset 0x08, and continuing for however many bytes the current [[Wiimote#Data_Reporting| Data Reporting Mode]] requires. Data streamed through the Input Reports must also be decrypted using the above transformation if encryption is enabled.
  
When the wireless adapter is plugged in, but the wireless nunchuk has not been synced or turned on, then it behaves as though nothing was plugged in. No status report is generated when the wireless adapter is plugged in or unplugged, unless the nunchuk is synced. Querying the status report returns 0 for the extension flag (instead of 2).
+
== Identification ==
 +
Once initialized, the last '''six''' bytes of the register block identify the connected Extension Controller. A six-byte read of register 0xa400fa will return these bytes. The Extension Controller must have been initialized prior to this. There are two ways of initializing the extension.
  
But when the wireless nunchuk is synced, then it sends a status report indicating that an extension is connected. Trying to read the 16 bytes from 0xF0 of the expansion (to identify the expansion) successfully returns "FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF", which isn't particularly helpful. Trying to read any 16 byte block has the same effect, it returns success, but it is all FFs.
+
=== The Old Way ===
+
The old way to initialize the extension was by writing the single encryption byte 0x00 to 0x(4)A40040, but that only works on Nintendo's own brand extensions and the GH3 Guitar, and will fail on 3rd party wireless nunchuks and GHWT extensions. If it fails, the 6 bytes will be FFFF FFFF FFFF. With this method you must decrypt the extension bytes to read them.
BUT... trying to read the calibration data from 0x20 or 0x30 (although it still returns success as above) causes the Nunchuk to disconnect, sending a status report to indicate no extension is plugged in. It will automatically reconnect 0.5 seconds later. While disconnected, attempting to read from the Nunchuk returns error 7.
 
  
Expansion data (in the input reports that include expansion data) is all 0xFF, just like trying to read the Nunchuk memory.
+
=== The New Way ===
 +
The new way to initialize the extension is by writing 0x55 to 0x(4)A400F0, then writing 0x00 to 0x(4)A400FB. It works on all extensions, and makes the extension type bytes '''unencrypted'''. This means that you no longer have to decrypt the extension bytes using the [[#Registers / Initialization | transform]] listed above.  
  
As soon as the "Classic" button is pressed, a status message is sent saying that the extension has been unplugged. It will stay "unplugged" until the status button is released again, however long they hold down the button for. Once the "Classic" button has been released, it will wait about half a second and then send a status message saying the extension has been plugged in again. If the Classic button is pressed twice before it sent the released message, then it may send another disconnect and connect message after the connect message. It is not currently possible to tell whether it is in Classic Controller or Nunchuk mode. But we can use this method to read when the "Classic" button is held down (although it will always look like it was held down longer than it really was).
 
  
Actually unplugging the adapter will give the same disconnect message, so after a few seconds one should assume that they did a real disconnect instead of holding down the "Classic" button. In one case, after unplugging and replugging the adapter, the BladeFX started returning all 00 instead of FF when reading the expansion type from 0xF0. The data returned from report 0x34 also had 00 00 00 as the last 3 extension bytes, even though the first 16 (starting from 0x08 in extension memory) were all FF. Trying to read the calibration from 0x20 still returned all 0xFF and then disconnected as usual. Pressing buttons on the BladeFX had no effect on the data returned in report 0x34. I have not done any additional experiments with this state yet.
+
{| style="border-collapse: collapse; padding: 0.2em 0.2em 0.2em 0.2em;"
 
+
|- style="text-align: center;"
Clearly the current method of turning on an expansion doesn't work for the Datel Blade-FX, and we need some way to turn it on. Perhaps then the calibration can be read without an error, and we can get real data.
+
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #fdd;" | '''Encrypted'''
 
+
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #dfd;" | '''Decrypted'''
It is not yet known how to make the Datel Blade-FX work, although it can be detected when it is synced, due to its strange behaviour, and the "Classic" button can be read. Nothing else is readable yet.
+
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddf;" | '''Meaning'''
 
+
|-
[[User:CarlKenner|CarlKenner]] 03:50, 28 November 2008 (UTC)
+
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center; background-color: #efe;" | 0000 0000 0000*
 +
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center; background-color: #efe;" | 0x2E2E
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eef;" | BladeFX adapter that has been initialised old way, unplugged then replugged
 +
|-
 +
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center; background-color: #efe;" | FFFF FFFF FFFF*
 +
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center; background-color: #efe;" | 0xFFFF
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eef;" | GHWT Guitar or Drums or synced BladeFX or other wireless nunchuk initialised the old way
 +
|-
 +
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center; background-color: #fee;" | 0xFEFE
 +
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center; background-color: #efe;" | 0000 A420 0000
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eef;" | [[Wiimote/Extension_Controllers/Nunchuck|Nunchuk]]
 +
|-
 +
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center; background-color: #fee;" | 0xFDFD
 +
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center; background-color: #efe;" | 0000 A420 0101
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eef;" | [[Wiimote/Extension_Controllers/Classic_Controller|Classic Controller]]
 +
|-
 +
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center; background-color: #fee;" | 0xFDFD
 +
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center; background-color: #efe;" | 0100 A420 0101
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eef;" | [[Wiimote/Extension_Controllers/Classic_Controller_Pro|Classic Controller Pro]] & NES/SNES controller from the NES/SNES mini
 +
|-
 +
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center; background-color: #fee;" | 0xFEEB
 +
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center; background-color: #efe;" | FF00 A420 0013
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eef;" | [[Wiimote/Extension_Controllers/Drawsome_Tablet|Drawsome Graphics Tablet]]
 +
|-
 +
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center; background-color: #fee;" | 0xFDFB
 +
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center; background-color: #efe;" | 0000 A420 0103
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eef;" | [[Wiimote/Extension_Controllers/Guitar Hero (Wii) Guitars|GH3 or GHWT Guitar]]
 +
|-
 +
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center; background-color: #fee;" | 0xFDFB
 +
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center; background-color: #efe;" | 0100 A420 0103
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eef;" | [[Wiimote/Extension_Controllers/Guitar Hero World Tour (Wii) Drums|Guitar Hero World Tour Drums]]
 +
|-
 +
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center; background-color: #efe;" | 0300 A420 0103*
 +
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center; background-color: #efe;" |
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eef;" | [[Wiimote/Extension Controllers/DJ Hero (Wii) Turntable|DJ Hero Turntable]]
 +
|-
 +
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center; background-color: #efe;" | 0000 A420 0111*
 +
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center; background-color: #efe;" |
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eef;" | [[Wiimote/Extension Controllers/TaTaCon|Taiko no Tatsujin TaTaCon (Drum controller)]]
 +
|-
 +
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center; background-color: #efe;" | FF00 A420 0112*
 +
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center; background-color: #efe;" |
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eef;" | [[Wiimote/Extension_Controllers/uDraw_GameTablet|uDraw GameTablet (Graphics tablet)]]
 +
|-
 +
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center; background-color: #efe;" | 0000 A420 0310*
 +
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center; background-color: #efe;" |
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eef;" | [[Wiimote/Extension_Controllers/Densha_de_GO!_(Wii)_Shinkansen_Controller|Densha de GO! Shinkansen Controller]]
 +
|-
 +
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center; background-color: #efe;" | 0000 A420 0402*
 +
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center; background-color: #efe;" | 0x2A2C
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eef;" | [[Wii Balance Board]]
 +
|-
 +
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center; background-color: #efe;" | 0000 A420 0005*
 +
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center; background-color: #efe;" |
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eef;" | Inactive [[Wiimote/Extension Controllers/Wii Motion Plus|Wii Motion Plus]] (Built-in)
 +
|-
 +
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center; background-color: #efe;" | 0000 A420 0405*
 +
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center; background-color: #efe;" |
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eef;" | Activated [[Wiimote/Extension Controllers/Wii Motion Plus|Wii Motion Plus]]
 +
|-
 +
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center; background-color: #efe;" | 0000 A420 0505*
 +
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center; background-color: #efe;" |
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eef;" | Activated [[Wiimote/Extension Controllers/Wii Motion Plus|Wii Motion Plus]] in Nunchuck passthrough mode
 +
|-
 +
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center; background-color: #efe;" | 0000 A420 0705*
 +
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center; background-color: #efe;" |
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eef;" | Activated [[Wiimote/Extension Controllers/Wii Motion Plus|Wii Motion Plus]] in Classic Controller passthrough mode
 +
|}
  
= Classic Controller =
+
(4)A60000:
The Classic Controller is identified by the 16-bit constant 0x0101 (0xFDFD encrypted) at register address 0xa400fe. It provides 15 buttons, two of which are analog triggers and provide both "clicked" status and a distance pressed measurement, and two X-Y analog sticks.
 
== Data Format ==
 
The Classic Controller reports its information as 6 bytes of data, readable at 0xa40008 and streamable using Data Reporting Modes that include Extension bytes (unused bytes are filled with 0x00). The data is packed into the six bytes as follows ('''after decryption'''):
 
  
{| style="border-collapse: collapse; padding: 0.2em 0.2em 0.2em 0.2em; text-align: center;"
+
{| style="border-collapse: collapse; padding: 0.2em 0.2em 0.2em 0.2em;"
|- style="background-color: #ddd;"
+
|- style="text-align: center;"
| style="background-color: #fff;" | &nbsp;
+
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #fdd;" | '''Encrypted'''
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center;" colspan="8"| '''Bit'''
+
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #dfd;" | '''Decrypted'''
|- style="background-color: #cdc;"
+
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddf;" | '''Meaning'''
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddd;" | '''Byte'''
+
|-
| style="border: 1px solid #ccc; padding: 0.2em; width:3.3em;" | '''7'''
+
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center; background-color: #efe;" | 0000 A620 0005*
| style="border: 1px solid #ccc; padding: 0.2em; width:3.3em;" | '''6'''
+
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center; background-color: #efe;" |
| style="border: 1px solid #ccc; padding: 0.2em; width:3.3em;" | '''5'''
+
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eef;" | Inactive [[Wiimote/Extension Controllers/Wii Motion Plus|Wii Motion Plus]]
| style="border: 1px solid #ccc; padding: 0.2em; width:3.3em;" | '''4'''
+
|-
| style="border: 1px solid #ccc; padding: 0.2em; width:3.3em;" | '''3'''
+
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center; background-color: #efe;" | 0000 A620 0405*
| style="border: 1px solid #ccc; padding: 0.2em; width:3.3em;" | '''2'''
+
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center; background-color: #efe;" |
| style="border: 1px solid #ccc; padding: 0.2em; width:3.3em;" | '''1'''
+
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eef;" | No-longer active [[Wiimote/Extension Controllers/Wii Motion Plus|Wii Motion Plus]]
| style="border: 1px solid #ccc; padding: 0.2em; width:3.3em;" | '''0'''
+
|-
|- style="background-color: #ded;"
+
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center; background-color: #efe;" | 0000 A620 0505*
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eee;" | 0
+
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center; background-color: #efe;" |
| style="border: 1px solid #ccc; padding: 0.2em;" colspan="2" | '''RX'''<span style="color: #777;">&lt;<span style="color: #c00;">4:3</span>&gt;</span>
+
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eef;" | No-longer nunchuk-passthrough [[Wiimote/Extension Controllers/Wii Motion Plus|Wii Motion Plus]]
| style="border: 1px solid #ccc; padding: 0.2em;" colspan="6" | '''LX'''<span style="color: #777;">&lt;<span style="color: #c00;">5:0</span>&gt;</span>
+
|-
|- style="background-color: #ded;"
+
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center; background-color: #efe;" | 0000 A620 0705*
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eee;" | 1
+
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center; background-color: #efe;" |
| style="border: 1px solid #ccc; padding: 0.2em;" colspan="2" | '''RX'''<span style="color: #777;">&lt;<span style="color: #c00;">2:1</span>&gt;</span>
+
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eef;" | No-longer classic-passthrough [[Wiimote/Extension Controllers/Wii Motion Plus|Wii Motion Plus]]
| style="border: 1px solid #ccc; padding: 0.2em;" colspan="6" | '''LY'''<span style="color: #777;">&lt;<span style="color: #c00;">5:0</span>&gt;</span>
 
|- style="background-color: #ded;"
 
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eee;" | 2
 
| style="border: 1px solid #ccc; padding: 0.2em;" | '''RX'''<span style="color: #777;">&lt;<span style="color: #c00;">0</span>&gt;</span>
 
| style="border: 1px solid #ccc; padding: 0.2em;" colspan="2" | '''LT'''<span style="color: #777;">&lt;<span style="color: #c00;">4:3</span>&gt;</span>
 
| style="border: 1px solid #ccc; padding: 0.2em;" colspan="5" | '''RY'''<span style="color: #777;">&lt;<span style="color: #c00;">4:0</span>&gt;</span>
 
|- style="background-color: #ded;"
 
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eee;" | 3
 
| style="border: 1px solid #ccc; padding: 0.2em;" colspan="3" | '''LT'''<span style="color: #777;">&lt;<span style="color: #c00;">2:0</span>&gt;</span>
 
| style="border: 1px solid #ccc; padding: 0.2em;" colspan="5" | '''RT'''<span style="color: #777;">&lt;<span style="color: #c00;">4:0</span>&gt;</span>
 
|- style="background-color: #ded;"
 
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eee;" | 4
 
| style="border: 1px solid #ccc; padding: 0.2em;" | '''B'''<span style="color: #c00;">DR</span>
 
| style="border: 1px solid #ccc; padding: 0.2em;" | '''B'''<span style="color: #c00;">DD</span>
 
| style="border: 1px solid #ccc; padding: 0.2em;" | '''B'''<span style="color: #c00;">LT</span>
 
| style="border: 1px solid #ccc; padding: 0.2em;" | '''B'''<span style="color: #c00;">-</span>
 
| style="border: 1px solid #ccc; padding: 0.2em;" | '''B'''<span style="color: #c00;">H</span>
 
| style="border: 1px solid #ccc; padding: 0.2em;" | '''B'''<span style="color: #c00;">+</span>
 
| style="border: 1px solid #ccc; padding: 0.2em;" | '''B'''<span style="color: #c00;">RT</span>
 
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddd; color: #888;" | 1
 
|- style="background-color: #ded;"
 
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eee;" | 5
 
| style="border: 1px solid #ccc; padding: 0.2em;" | '''B'''<span style="color: #c00;">ZL</span>
 
| style="border: 1px solid #ccc; padding: 0.2em;" | '''B'''<span style="color: #c00;">B</span>
 
| style="border: 1px solid #ccc; padding: 0.2em;" | '''B'''<span style="color: #c00;">Y</span>
 
| style="border: 1px solid #ccc; padding: 0.2em;" | '''B'''<span style="color: #c00;">A</span>
 
| style="border: 1px solid #ccc; padding: 0.2em;" | '''B'''<span style="color: #c00;">X</span>
 
| style="border: 1px solid #ccc; padding: 0.2em;" | '''B'''<span style="color: #c00;">ZR</span>
 
| style="border: 1px solid #ccc; padding: 0.2em;" | '''B'''<span style="color: #c00;">DL</span>
 
| style="border: 1px solid #ccc; padding: 0.2em;" | '''B'''<span style="color: #c00;">DU</span>
 
 
|}
 
|}
LX,LY are the left Analog Stick X and Y (0-61), RX and RY are the right Analog Stick X and Y (0-31), and LT and RT are the Left and Right Triggers (0-31). The left Analog Stick has twice the precision of the other analog values.
 
 
BD{L,R,U,D} are the D-Pad direction buttons. B{ZR,ZL,A,B,X,Y,+,H,-} are the discrete buttons. B{LT,RT} are the digital button click of LT and RT. All buttons are 0 when pressed.
 
 
Nintendo games calibrate the center position of the Analog Sticks upon power-up or insertion of the Classic Controller.
 
  
= Guitar Hero Guitar =
 
The Guitar is identified by the 16-bit constant 0x0103 (0xFDFB encrypted) at register address 0xa400fe. It a two-axis analog thumbstick, a one-axis analog "whammy bar" and 9 buttons.
 
== Data Format ==
 
The Guitar reports its information as 6 bytes of data, readable at 0xa40008 and streamable using Data Reporting Modes that include Extension bytes (unused bytes are filled with 0x00). The data is packed into the six bytes as follows ('''after decryption'''):
 
  
{| style="border-collapse: collapse; padding: 0.2em 0.2em 0.2em 0.2em; text-align: center;"
+
The values marked with a * are actually already unencrypted, and should not be decrypted. They are given here along with unencrypted version for those wanting to use simplified detection code. You can easily tell whether a value should be decrypted or not by looking at the other values in the 0xa400fa to 0xa400fd region.
|- style="background-color: #ddd;"
 
| style="background-color: #fff;" | &nbsp;
 
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center;" colspan="8"| '''Bit'''
 
|- style="background-color: #cdc;"
 
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddd;" | '''Byte'''
 
| style="border: 1px solid #ccc; padding: 0.2em; width:3.3em;" | '''7'''
 
| style="border: 1px solid #ccc; padding: 0.2em; width:3.3em;" | '''6'''
 
| style="border: 1px solid #ccc; padding: 0.2em; width:3.3em;" | '''5'''
 
| style="border: 1px solid #ccc; padding: 0.2em; width:3.3em;" | '''4'''
 
| style="border: 1px solid #ccc; padding: 0.2em; width:3.3em;" | '''3'''
 
| style="border: 1px solid #ccc; padding: 0.2em; width:3.3em;" | '''2'''
 
| style="border: 1px solid #ccc; padding: 0.2em; width:3.3em;" | '''1'''
 
| style="border: 1px solid #ccc; padding: 0.2em; width:3.3em;" | '''0'''
 
|- style="background-color: #ded;"
 
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eee;" | 0
 
| style="border: 1px solid #ccc; padding: 0.2em;" colspan="8" | '''SX'''<span style="color: #777;">&lt;<span style="color: #c00;">7:0</span>&gt;</span>
 
|- style="background-color: #ded;"
 
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eee;" | 1
 
| style="border: 1px solid #ccc; padding: 0.2em;" colspan="8" | '''SY'''<span style="color: #777;">&lt;<span style="color: #c00;">7:0</span>&gt;</span>
 
|- style="background-color: #ded;"
 
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eee;" | 2
 
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddd; color:#888;" colspan="8" | Unknown
 
|- style="background-color: #ded;"
 
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eee;" | 3
 
| style="border: 1px solid #ccc; padding: 0.2em;" colspan="8" | '''WB'''<span style="color: #777;">&lt;<span style="color: #c00;">7:0</span>&gt;</span>
 
|- style="background-color: #ded;"
 
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eee;" | 4
 
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddd; color:#888;" | ?
 
| style="border: 1px solid #ccc; padding: 0.2em;" | '''B'''<span style="color: #c00;">D</span>
 
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddd; color:#888;" | ?
 
| style="border: 1px solid #ccc; padding: 0.2em;" | '''B'''<span style="color: #c00;">-</span>
 
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddd; color:#888;" | ?
 
| style="border: 1px solid #ccc; padding: 0.2em;" | '''B'''<span style="color: #c00;">+</span>
 
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddd; color:#888;" | ?
 
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddd; color:#888;" | ?
 
|- style="background-color: #ded;"
 
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eee;" | 5
 
| style="border: 1px solid #ccc; padding: 0.2em;" | '''B'''<span style="color: #c00;">O</span>
 
| style="border: 1px solid #ccc; padding: 0.2em;" | '''B'''<span style="color: #c00;">R</span>
 
| style="border: 1px solid #ccc; padding: 0.2em;" | '''B'''<span style="color: #c00;">B</span>
 
| style="border: 1px solid #ccc; padding: 0.2em;" | '''B'''<span style="color: #c00;">G</span>
 
| style="border: 1px solid #ccc; padding: 0.2em;" | '''B'''<span style="color: #c00;">Y</span>
 
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddd; color:#888;" | ?
 
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddd; color:#888;" | ?
 
| style="border: 1px solid #ccc; padding: 0.2em;" | '''B'''<span style="color: #c00;">U</span>
 
|}
 
SX,SY are the Analog Stick X and Y positions. BG,BR,BY,BB,BO are the fret buttons by color (Green, Red, Yellow, Blue, Orange).  B-,B+ are the + and - buttons at the base of the controller.  BU and BD are up and down on the strum bar, respectively.  WB is the analog whammy bar.
 
  
= Balance Board =
+
The "Partially inserted" condition allegedly occurs when the connector is loose or has a bad connection. Usually, this will correct itself upon full insertion. However, it has been known to "stick" in the partially inserted state rarely. This is a hardware glitch, the fix is to simply disconnect and reconnect the Extension Controller. Third party controllers such as wireless nunchuks also return this value if initialized with the old method, in which case the whole last 16 bytes will be FF.  
The Balance Board extension is identified by the 16-bit constant 0x2A2C (0x0402 encrypted) at register address 0xa400fe. This extension is only found connected to the [[Wii Balance Board]] device, and is documented on that page.
 
  
= Wii Motion Plus =
+
Contrary to previous documentation, the 0000 0000 0000 value does NOT occur when nothing is inserted. Instead you get error 7 when you try to read the expansion type. A successful 0000 0000 0000 only occurs when a BladeFX wireless nunchuk is synced, initialized with the old method, then the adapter is unplugged, then the adapter is replugged.
The Wii Motion Plus extension is unusual in that it has its own extension port on the back allowing a Nunchuk for example to be plugged in along with the Wii Motion Plus. That might radically change the way it is detected and used compared to other extensions.
 
  
The Wii Motion Plus consists of two rate-gyros. These gyros measure rotation SPEED, but can't actually measure absolute orientation, which causes drift if not corrected by some other data source. Speed is one step better than the acceleration reported by the accelerometers. One of the two gyros will obviously have to measure yaw, since that can't be measured by accelerometers. The other axis is unknown, but I would guess pitch. Unlike the rotation measured by accelerometers, whose axes are defined by gravity, the gyro axes are relative to the angle of the Wiimote. So even if it can measure yaw when the Wiimote is in the normal orientation, it won't be able to detect yaw when the Wiimote is vertical.
+
A [[Wii Balance Board]] extension is only found in a Wii Balance Board. Although it exposes functionality as an Extension Controller, they are not separable and this controller is documented separately.
  
In addition to providing yaw, the gyros also allow the software to tell the difference between orientation and acceleration, which are difficult to tell apart from the accelerometers which measure a mixture of both.
+
=== Data format code ===
 +
After some research of [https://www.nintendo.co.jp/support/oss/index.html NES/SNES Classic Mini's Classic Controller driver source code], it turned out that the penultimate byte of the register block identify (e.g. the byte at offset 0x(4)A400FE) is data format code. So 0x00 is the [[Wiimote/Extension Controllers/Nunchuck|Nunchuck]] data format and 0x01 is the default [[Wiimote/Extension Controllers/Classic Controller|Classic Controller]] data format. Actually, this byte is writable for some controllers. So, official Classic Controller supports data formats 0x01, 0x02 and 0x03. Most old third party Classic Controllers supports only 0x01, NES/SNES Classic Mini tries to set the data format byte to 0x03, that's why most of them don't work. Guitar Hero controller uses mode 0x03 by default, so it shares the data format with Classic Controller, that's why guitar can be used in Wii's main menu.
  
Wii Motion Plus will be available in early 2009 packaged along with Wii Sports Resort.
+
=== Encryption setup ===
 +
After the identification is read, the encryption can be set up if required. This is done by enabling encryption by writing 0xAA to extension register 0xF0, and then writing the 16-byte key to register 0x40. The key is written in 3 blocks of 6, 6, and 4 bytes. (Writing a single encryption byte to register 0x40 will work on genuine Nintendo controllers, but not on others). After this the extension can be operated in full encryption mode.
  
[http://www.ailive.net/lm2.html Wii Motion Plus Documentation at AiLive]
+
== List of extension controllers ==
 +
*[[Wiimote/Extension Controllers/Nunchuck|Nunchuck]]
 +
*[[Wiimote/Extension Controllers/Classic Controller|Classic Controller]] ([[Wiimote/Extension Controllers/Classic Controller Pro|Pro]])
 +
*[[Wiimote/Extension_Controllers/Drawsome_Tablet|Drawsome Graphics Tablet]]
 +
*[[Wiimote/Extension Controllers/Guitar Hero (Wii) Guitars|Guitar Hero (Wii) Guitars]]
 +
*[[Wiimote/Extension Controllers/Guitar Hero World Tour (Wii) Drums|Guitar Hero World Tour (Wii) Drums]]
 +
*[[Wiimote/Extension Controllers/DJ Hero (Wii) Turntable|DJ Hero (Wii) Turntable]]
 +
*[[Wiimote/Extension Controllers/TaTaCon|Taiko no Tatsujin (Wii) TaTaCon (Drum controller)]]
 +
*[[Wiimote/Extension Controllers/Wii Motion Plus|Wii Motion Plus]]
  
[[Category:Hardware]]
+
[[Category:Peripherals]]

Latest revision as of 19:46, 27 February 2024

Pins on the Wii expansion controller cable

The Wii Remote includes a 6-pin expansion port that allows external peripherals to be connected to it. Communications are bidirectional synchronous serial I2C, and the devices provide a virtual register block of length 0x100 that is mapped at 0xa40000 in the Wii Remote's address space (the 7-bit I2C address is 0x52). Communications are encrypted, as detailed below. Nintendo released two extensions, the Nunchuk and the Classic Controller. In addition, Guitar Hero guitars and drums also act as extension controllers. The balance board acts like a dummy Wiimote with its own type of built-in extension controller. There are also many third party versions of classic controllers and nunchuks.

The Wii Motion Plus is a small expansion which itself has an expansion port, allowing people to plug the Nunchuk into the Wii Motion Plus and the Wii Motion Plus into the Wii Remote.

The peripheral Protocol is 400kHz "fast" I2C, with slave address 0x52. It is easy to make homebrew peripherals; see the Tenkey numberpad project(Japanese), gyro project (Japanese), the Twiidler project, the DIY Wii drum controller, the OpenChord Guitar, or the DIY Wii guitar proof of concept demo video.

Registers / Initialization

Reminder
Remember to set bit 2 (0x04) on the first byte of the Output Reports to write to registers!

Wii Motion Plus is mapped at register 0xa60000.

Other extension Controllers are mapped at register address 0xa40000. The data is 0x100 bytes long, and it is mirrored across the entire 16-bit address space up to 0xa4FFFF. These registers are readable and writable. Communications are optionally encrypted, and explicit initialization is required to disable encryption (see below).

In encrypted mode, bytes can be decrypted using the following transformation:

decrypted_byte = (encrypted_byte XOR table1[address%8]) + table2[address%8]

Where table1 and table2 are 8-byte tables calculated based on the 16-byte key sent by the host, and address is the address of the byte being read (only the bottom 3 bits matter). If the host key is 16 zero bytes, table1[x] and table2[x] are all 0x97. The value 0x17 was used previously, which is equivalent. Proof of why any two pairs of table entries a,b and x,y are equivalent if (a^b^x^y)&0x80 == 0 is left as an exercise to the reader.

The calculations should be performed mod 256, that is, truncated to 8 bits (in languages such as C and Python, use &0xFF or work directly with 8-bit datatypes).

If the Wii Remote is initialized using the new method listed below, then the encryption of the extension bytes is disabled and they need not be decrypted using the transform listed above.

The Wii Remote will stream data bytes from the Extension Controller starting at offset 0x08, and continuing for however many bytes the current Data Reporting Mode requires. Data streamed through the Input Reports must also be decrypted using the above transformation if encryption is enabled.

Identification

Once initialized, the last six bytes of the register block identify the connected Extension Controller. A six-byte read of register 0xa400fa will return these bytes. The Extension Controller must have been initialized prior to this. There are two ways of initializing the extension.

The Old Way

The old way to initialize the extension was by writing the single encryption byte 0x00 to 0x(4)A40040, but that only works on Nintendo's own brand extensions and the GH3 Guitar, and will fail on 3rd party wireless nunchuks and GHWT extensions. If it fails, the 6 bytes will be FFFF FFFF FFFF. With this method you must decrypt the extension bytes to read them.

The New Way

The new way to initialize the extension is by writing 0x55 to 0x(4)A400F0, then writing 0x00 to 0x(4)A400FB. It works on all extensions, and makes the extension type bytes unencrypted. This means that you no longer have to decrypt the extension bytes using the transform listed above.


Encrypted Decrypted Meaning
0000 0000 0000* 0x2E2E BladeFX adapter that has been initialised old way, unplugged then replugged
FFFF FFFF FFFF* 0xFFFF GHWT Guitar or Drums or synced BladeFX or other wireless nunchuk initialised the old way
0xFEFE 0000 A420 0000 Nunchuk
0xFDFD 0000 A420 0101 Classic Controller
0xFDFD 0100 A420 0101 Classic Controller Pro & NES/SNES controller from the NES/SNES mini
0xFEEB FF00 A420 0013 Drawsome Graphics Tablet
0xFDFB 0000 A420 0103 GH3 or GHWT Guitar
0xFDFB 0100 A420 0103 Guitar Hero World Tour Drums
0300 A420 0103* DJ Hero Turntable
0000 A420 0111* Taiko no Tatsujin TaTaCon (Drum controller)
FF00 A420 0112* uDraw GameTablet (Graphics tablet)
0000 A420 0310* Densha de GO! Shinkansen Controller
0000 A420 0402* 0x2A2C Wii Balance Board
0000 A420 0005* Inactive Wii Motion Plus (Built-in)
0000 A420 0405* Activated Wii Motion Plus
0000 A420 0505* Activated Wii Motion Plus in Nunchuck passthrough mode
0000 A420 0705* Activated Wii Motion Plus in Classic Controller passthrough mode

(4)A60000:

Encrypted Decrypted Meaning
0000 A620 0005* Inactive Wii Motion Plus
0000 A620 0405* No-longer active Wii Motion Plus
0000 A620 0505* No-longer nunchuk-passthrough Wii Motion Plus
0000 A620 0705* No-longer classic-passthrough Wii Motion Plus


The values marked with a * are actually already unencrypted, and should not be decrypted. They are given here along with unencrypted version for those wanting to use simplified detection code. You can easily tell whether a value should be decrypted or not by looking at the other values in the 0xa400fa to 0xa400fd region.

The "Partially inserted" condition allegedly occurs when the connector is loose or has a bad connection. Usually, this will correct itself upon full insertion. However, it has been known to "stick" in the partially inserted state rarely. This is a hardware glitch, the fix is to simply disconnect and reconnect the Extension Controller. Third party controllers such as wireless nunchuks also return this value if initialized with the old method, in which case the whole last 16 bytes will be FF.

Contrary to previous documentation, the 0000 0000 0000 value does NOT occur when nothing is inserted. Instead you get error 7 when you try to read the expansion type. A successful 0000 0000 0000 only occurs when a BladeFX wireless nunchuk is synced, initialized with the old method, then the adapter is unplugged, then the adapter is replugged.

A Wii Balance Board extension is only found in a Wii Balance Board. Although it exposes functionality as an Extension Controller, they are not separable and this controller is documented separately.

Data format code

After some research of NES/SNES Classic Mini's Classic Controller driver source code, it turned out that the penultimate byte of the register block identify (e.g. the byte at offset 0x(4)A400FE) is data format code. So 0x00 is the Nunchuck data format and 0x01 is the default Classic Controller data format. Actually, this byte is writable for some controllers. So, official Classic Controller supports data formats 0x01, 0x02 and 0x03. Most old third party Classic Controllers supports only 0x01, NES/SNES Classic Mini tries to set the data format byte to 0x03, that's why most of them don't work. Guitar Hero controller uses mode 0x03 by default, so it shares the data format with Classic Controller, that's why guitar can be used in Wii's main menu.

Encryption setup

After the identification is read, the encryption can be set up if required. This is done by enabling encryption by writing 0xAA to extension register 0xF0, and then writing the 16-byte key to register 0x40. The key is written in 3 blocks of 6, 6, and 4 bytes. (Writing a single encryption byte to register 0x40 will work on genuine Nintendo controllers, but not on others). After this the extension can be operated in full encryption mode.

List of extension controllers