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

Difference between revisions of "Wiimote"

From WiiBrew
Jump to navigation Jump to search
((Re-)fix broken SDP link)
 
(229 intermediate revisions by 50 users not shown)
Line 1: Line 1:
This page is intended to be a technical guide to the Wii Remote. For an excellent high-level overview of the Wii Remote (aka Wiimote), see the [http://en.wikipedia.org/wiki/Wii_Remote Wikipedia entry].
+
{{Cleanup||Make sections collapsable. Add a See Also section}}
 +
{{Wikify}}
  
== Communication ==
+
[[File:Wii-Remote-Top-surface.jpg|200px|thumb|right|Wii remote circuit board, top surface]][[File:Wii Remote uncovered.jpg|200px|thumb|right|Wii remote circuit board, bottom surface]]{{selfref|This article is a technical guide to the Wii Remote. For a high-level overview of the Wii Remote, see the [http://en.wikipedia.org/wiki/Wii_Remote Wikipedia entry].}}
  
The Wiimote communicates with the Wii via a Bluetooth wireless link. The Bluetooth controller is a [http://www.broadcom.com/products/Bluetooth/Bluetooth-RF-Silicon-and-Software-Solutions/BCM2042 Broadcom 2042] chip, which is designed to be used with devices which follow the Bluetooth Human Interface Device (HID) standard, such as keyboards and mice. The Bluetooth HID is directly based upon the [http://en.wikipedia.org/wiki/USB_human_interface_device_class USB HID] standard, and much of the same documentation applies.
+
[[File:Wii Remote Broadcom.jpg|200px|thumb|right|Broadcom BCM2042 in a Wii remote]]The '''Wii Remote''' (informally known as the '''Wiimote''') is the Wii's main input device. It is a wireless device, using standard Bluetooth technology to communicate with the Wii. It is built around a [http://www.broadcom.com/products/Bluetooth/Bluetooth-RF-Silicon-and-Software-Solutions/BCM2042 Broadcom BCM2042] bluetooth System-on-a-chip, and contains multiple peripherals that provide data to it, as well as an expansion port for external add-ons. The Wii Remote uses (and, at times, abuses) the standard Bluetooth HID protocol to communicate with the host, which is directly based upon the [http://en.wikipedia.org/wiki/USB_human_interface_device_class USB HID] standard. As such, it will appear as a standard input device to any Bluetooth host. However, the Wii Remote does not make use of the standard data types and HID descriptor, and only describes its report format length, leaving the actual contents undefined, which makes it useless with standard HID drivers (but some [[Wiimote Driver]]s exist). The Wii Remote actually uses a fairly complex set of operations, transmitted through HID Output reports, and returns a number of different data packets through its Input reports, which contain the data from its peripherals.
  
When queried with the Bluetooth Service Discovery Protocol ([http://www.palowireless.com/infotooth/tutorial/sdp.asp SDP]), the Wiimote reports back a great deal of information, listed at Wii Bluetooth Specs. In particular it reports:
+
= Summary =
{| border="1"
 
|-
 
|Name || Nintendo RVL-CNT-01
 
|-
 
|Vendor ID || 0x057e
 
|-
 
|Product ID || 0x0306
 
|}
 
  
 +
Reverse engineering and documenting all of the Wii Remote's features is a work in progress. Here are the known features and their status:
  
The Wiimote does not appear to require any of the authentication or encryption features of the Bluetooth standard. In order to interface with it, one must first put the controller into discoverable mode by either pressing the 1 and 2 buttons at the same time, or by pressing the red sync button under the battery cover. Once in this mode, the Wiimote can be queried by the Bluetooth HID driver on the host. If the HID driver on the host does not connect to the Wiimote within 20 seconds, the Wiimote will turn itself off. Holding down the 1 and 2 buttons continuously will force the Wiimote to stay in discoverable mode without turning off. This does not work with the sync button, however. When in discoverable mode, all 4 of the Player LEDs will blink.
+
{| style="background-color: #eef; border-collapse: collapse; padding: 0.2em 0.2em 0.2em 0.2em;"
 
 
=== HID Interface ===
 
 
 
The HID standard allows devices to be self-describing, using a HID descriptor block. This block includes an enumeration of reports that the device understands. A report can be thought of similar to a network port assigned to a particular service. Reports are unidirectional however, and the HID descriptor lists for each port the direction (Input or Output) and the payload size for each port. Like all Bluetooth HID devices, the Wiimote reports its HID descriptor block when queried using the SDP protocol. A human-readable version of the block is shown at Wii_bluetooth_specs, and is summarized in the following table:
 
 
 
{|
 
|- valign="top"
 
|
 
'''Output'''
 
{| border="1"
 
! Report ID
 
! Payload Size
 
! Known Functions
 
 
|-
 
|-
| 0x11
+
| style="border: 1px solid #ccc; padding: 0.2em; width: 14em;" | '''[[#Bluetooth Communication | Bluetooth Communication]]'''
| 1
+
| style="border: 1px solid #ccc;" | [[File:ButtonGreen.png|16px]]
|
+
| style="border: 1px solid #ccc; padding: 0.2em;" | Connecting to the Wii Remote and listening for connections works.
[[Wiimote#Player LEDs | Player LEDs]], [[Wiimote#Force Feedback | Force Feedback]]
 
 
|-
 
|-
| 0x12
+
| style="border: 1px solid #ccc; padding: 0.2em;" | '''[[#Core Buttons | Core Buttons]]'''
| 2
+
| style="border: 1px solid #ccc;" | [[File:ButtonGreen.png|16px]]
|
+
| style="border: 1px solid #ccc; padding: 0.2em;" | All working.
[[Report type / ID]]
 
 
|-
 
|-
| 0x13
+
| style="border: 1px solid #ccc; padding: 0.2em;" | '''[[#Accelerometer | Accelerometer]]'''
| 1
+
| style="border: 1px solid #ccc;" | [[File:ButtonGreen.png|16px]]
|
+
| style="border: 1px solid #ccc; padding: 0.2em;" | All working.
[[Wiimote#IR_Sensor | IR Sensor Enable]]
 
 
|-
 
|-
| 0x14
+
| style="border: 1px solid #ccc; padding: 0.2em;" | '''[[#IR Camera | IR Camera]]'''
| 1
+
| style="border: 1px solid #ccc;" | [[File:ButtonGreen.png|16px]]
|
+
| style="border: 1px solid #ccc; padding: 0.2em;" | All working.
[[Wiimote#Speaker | Enable speaker]]
 
 
|-
 
|-
| 0x15
+
| style="border: 1px solid #ccc; padding: 0.2em;" | '''[[#Power Button | Power Button]]'''
| 1
+
| style="border: 1px solid #ccc;" | [[File:ButtonGreen.png|16px]]
|
+
| style="border: 1px solid #ccc; padding: 0.2em;" | All working.
[[Controller status]]
 
 
|-
 
|-
| 0x16
+
| style="border: 1px solid #ccc; padding: 0.2em;" | '''[[#Speaker | Speaker]]'''
| 21
+
| style="border: 1px solid #ccc;" | [[File:ButtonGreen.png|16px]]
|
+
| style="border: 1px solid #ccc; padding: 0.2em;" | All working.
[[Wiimote#On-board_Memory | Write data]]
 
 
|-
 
|-
| 0x17
+
| style="border: 1px solid #ccc; padding: 0.2em;" | '''[[#Player LEDs | Player LEDs]]'''
| 6
+
| style="border: 1px solid #ccc;" | [[File:ButtonGreen.png|16px]]
|
+
| style="border: 1px solid #ccc; padding: 0.2em;" | All working.
[[Wiimote#On-board_Memory | Read data]]
 
 
|-
 
|-
| 0x18
+
| style="border: 1px solid #ccc; padding: 0.2em;" | '''[[#Status Information | Status Information]]'''
| 21
+
| style="border: 1px solid #ccc;" | [[File:ButtonGreen.png|16px]]
|
+
| style="border: 1px solid #ccc; padding: 0.2em;" | Battery and extension info in Status Report
[[Wiimote#Speaker | Speaker data]]
 
 
|-
 
|-
| 0x19
+
| style="border: 1px solid #ccc; padding: 0.2em;" | '''[[#Extension Controllers | Extension Controllers]]'''
| 1
+
| style="border: 1px solid #ccc;" | [[File:ButtonYellow.png|16px]]
|
+
| style="border: 1px solid #ccc; padding: 0.2em;" | Official extensions are supported, however many 3rd party extensions are not understood. See [[Wiimote/Extension_Controllers | Extension Controllers page]]  
[[Wiimote#Speaker | Mute speaker]]
 
 
|-
 
|-
| 0x1a
+
| colspan="3" style="border: 0px; background-color: #ffffff; " |
| 1
+
{|align="center" style="margin-top: 2px; font-size: 8pt; background-color:#e4e4f4;border:1px solid #ccc; border-collapse: collapse;"
|
+
| style="padding: 0.2em; border: 1px solid #ccc;" | '''Legend'''
[[Wiimote#IR_Sensor | IR Sensor Enable 2]]
+
| style="padding: 0.2em;"| [[File:ButtonGreen.png|16px]] Perfect or near-perfect
 +
| style="padding: 0.2em;"| [[File:ButtonYellow.png|16px]] Usable but not complete
 +
| style="padding: 0.2em;"| [[File:ButtonRed.png|16px]] Unusable
 +
|}
 
|}
 
|}
|
+
 
'''Input'''
+
= Bluetooth Communication =
{| border="1"
+
The wiimote communicates with the host via standard bluetooth protocol. The wiimote can be placed into discoverable mode for 20s by pressing the '''''sync''''' button on its back under the battery cover. Holding down the 1 and 2 button continuously will force the wiimote to stay in discoverable mode without turning off. This does not work with the ''sync'' button, though. When in discoverable mode, a number of the player LEDs based on the battery level will blink. With full battery all four LEDs will blink, the lower the battery the less LEDs will blink. During device inquiry the host will find all discoverable nearby wiimotes. Now the host can establish a bluetooth baseband connection to the wiimote, '''no bluetooth pairing is needed''', however, if bluetooth pairing is performed, the wiimote is able to reconnect to the host if disconnected.
! Report ID
+
After a bluetooth baseband connection is established (with or without pairing) the [[#HID Interface | HID channels can be opened]] and used for reading and writing reports from/to the wiimote.
! Payload Size
+
 
! Known Functions
+
The newer Wiimote RVL-CNT-01-TR shuts down immediately upon receiving any HID output report if it has been turned on using the 1 + 2 method, although it works using the sync button. It is possible that authentication is now mandatory for the 1 + 2 temporary sync{{check}}.
|-
+
 
| 0x20
+
== Bluetooth Pairing ==
| 6
+
The wiimote supports the legacy bluetooth pairing methods. This involves sending a PIN to the wiimote. Bluetooth pairing is not required to use a wiimote and you can proceed by establishing a HID connection without pairing at all. However, if the wiimote is paired, it will actively seek out for its last connected host on disconnection and reestablish the connection. The following section explains the bluetooth device pairing, if no pairing is required, [[#HID Interface | skip this section]].
|
+
 
[[Wiimote#Expansion_Port | Expansion Port]]
+
Bluetooth pairing must be initiated by the host by sending a "Require Authentication" HCI command to its bluetooth device. The bluetooth device will ask the host for a link key, which must be rejected so it will ask for a PIN-Code. The PIN-Code is the binary bluetooth address of the wiimote backwards. Following a short piece of C code to calculate the PIN:
|-
+
 
| 0x21
+
<pre>
| 21
+
Lets assume the Wiimote has the bluetooth address "00:1E:35:3B:7E:6D". If you want the PIN for bluetooth pairing in a simple string, do the following:
|
+
 
[[Wiimote#On-board_Memory | Read data]]
+
char pin[6];
|-
+
pin[0] = 0x6D;
| 0x22
+
pin[1] = 0x7E;
| 4
+
pin[2] = 0x3B;
|
+
pin[3] = 0x35;
[[Wiimote#On-board_Memory | Write data]]
+
pin[4] = 0x1E;
|-
+
pin[5] = 0x00;
| 0x30
+
 
| 2
+
Now "pin" contains your bluetooth pin that should be used for pairing your devices.
|
+
</pre>
[[Wiimote#Buttons | Buttons only]]
+
 
|-
+
If connecting by holding down the 1+2 buttons, the PIN is the bluetooth address of the wiimote backwards, if connecting by pressing the "sync" button on the back of the wiimote, then the PIN is the bluetooth address of the host backwards.
| 0x31
+
 
| 5
+
After sending the PIN to the bluetooth device via HCI commands, the wiimote will return a "Authentication Accepted" command and the pairing is established (both devices are '''bonded''' now). After pairing you continue with [[#HID Interface | establishing the HID connection]] the ''same way as without pairing''.
|
+
 
[[Wiimote#Buttons | Buttons]] | [[Wiimote#Motion_Sensor | Motion Sensing Report]]
+
If the host successfully bonded with the wiimote '''and established an HID connection''' the wiimote will save the bluetooth address of the host and enable ''single press reconnection''. That means if the wiimote is now disconnected from the host, it will actively seek out for the host if '''any''' button is pressed and establish a baseband and HID connection. The wiimote will never actively send pairing requests since this is not needed. Also remember that this works with '''any''' button not only the power-button. However, after establishing the connection, the wiimote sends a [[#Data Reporting | button-input-report]] and this allows the host to see what button was pressed. So the host may reject the new connection if any button except the power-button was pressed.
|-
+
 
| 0x32
+
The new bluetooth pairing method SSP (Secure Simple Pairing) is not supported. Also it is not yet investigated whether a link key has to be created (by sending a PIN) on every connection or whether the link key can be saved and reused on new connections. Though, creating a new link key on every connection works fine.
| 16
+
 
|
+
The wiimote has space for several host addresses (at least 3 are known to work) so it can be paired with more than one host (like PC or Wii) and it will try in reverse order to reconnect to the hosts. That is, the last paired host is tried first and so one. If button 1 and 2 or the sync button on its back are pressed, the wiimote will not actively seek out for its host but instead place itself in discoverable mode and wait for incoming connections so bluetooth pairing does not conflict with normal host-side connections.
[[Wiimote#Buttons | Buttons]] | IR??
+
 
 +
It is not known how to remove the hosts addresses from the wiimote, however, with some investigation it should be possible to locate them in the EEPROM and manipulate them. If this is considered a security issue, then don't pair your devices.
 +
 
 +
== SDP information ==
 +
When queried with the Bluetooth Service Discovery Protocol ([https://web.archive.org/web/20131006043725/http://www.palowireless.com/infotooth/tutorial/sdp.asp SDP]), the Wii Remote reports back a great deal of information. In particular, it reports:
 +
{| style="border: 1px solid #ccc; background-color: #eef; border-collapse: collapse; padding: 0.2em 0.2em 0.2em 0.2em;"
 
|-
 
|-
| 0x33
+
| style="border: 1px solid #ccc; padding: 0.2em;" |  
| 17
+
| style="border: 1px solid #ccc; padding: 0.2em;" | Wii Remote/old Wii Remote Plus
|
+
| style="border: 1px solid #ccc; padding: 0.2em;" | new Wii Remote Plus
[[Wiimote#Buttons | Buttons]] | [[Wiimote#Motion_Sensor | Motion Sensing Report]]
 
 
|-
 
|-
| 0x34
+
| style="border: 1px solid #ccc; padding: 0.2em;" | '''Name'''
| 21
+
| style="border: 1px solid #ccc; padding: 0.2em;" | Nintendo RVL-CNT-01
|
+
| style="border: 1px solid #ccc; padding: 0.2em;" | Nintendo RVL-CNT-01-TR
[[Wiimote#Buttons | Buttons]] | IR??
 
 
|-
 
|-
| 0x35
+
| style="border: 1px solid #ccc; padding: 0.2em;" | '''Vendor ID'''
| 21
+
| style="border: 1px solid #ccc; padding: 0.2em;" | 0x057e
|
+
| style="border: 1px solid #ccc; padding: 0.2em;" | 0x057e
[[Wiimote#Buttons | Buttons]] | [[Wiimote#Motion_Sensor | Motion Sensing Report]]
 
 
|-
 
|-
| 0x36
+
| style="border: 1px solid #ccc; padding: 0.2em;" | '''Product ID'''
| 21
+
| style="border: 1px solid #ccc; padding: 0.2em;" | 0x0306
|
+
| style="border: 1px solid #ccc; padding: 0.2em;" | 0x0330
[[Wiimote#Buttons | Buttons]] | IR??
 
 
|-
 
|-
| 0x37
+
| style="border: 1px solid #ccc; padding: 0.2em;" | '''Major Device Class'''
| 21
+
| style="border: 1px solid #ccc; padding: 0.2em;" | 1280
|
+
| style="border: 1px solid #ccc; padding: 0.2em;" | ?
[[Wiimote#Buttons | Buttons]] | [[Wiimote#Motion_Sensor | Motion Sensing Report]]
 
 
|-
 
|-
| 0x3d
+
| style="border: 1px solid #ccc; padding: 0.2em;" | '''Minor Device Class'''
| 21
+
| style="border: 1px solid #ccc; padding: 0.2em;" | 4
|
+
| style="border: 1px solid #ccc; padding: 0.2em;" | ?
[[Wiimote#Buttons | Buttons]] | IR??
 
 
|-
 
|-
| 0x3e
+
| style="border: 1px solid #ccc; padding: 0.2em;" | '''Service Class'''
| 21
+
| style="border: 1px solid #ccc; padding: 0.2em;" | 0
|
+
| style="border: 1px solid #ccc; padding: 0.2em;" | ?
[[Wiimote#Buttons | Buttons]] | [[Wiimote#Motion_Sensor | Motion Sensing Report]] | IR??
 
 
|-
 
|-
| 0x3f
+
| style="border: 1px solid #ccc; padding: 0.2em;" | '''(Summary of all Class Values)'''
| 21
+
| style="border: 1px solid #ccc; padding: 0.2em;" | 0x002504
|
+
| style="border: 1px solid #ccc; padding: 0.2em;" | 0x000508
[[Wiimote#Buttons | Buttons]] | [[Wiimote#Motion_Sensor | Motion Sensing Report]] | IR??
 
|}
 
 
|}
 
|}
  
Note that "Ouput" refers to packets are sent from the host to the Wiimote, and "Input" refers to packets that are sent from the Wiimote to the host. For clarity, the convention in this document is to show packets including the Bluetooth header (in parentheses), report ID (also called channel ID in some places), and payload, as described in sections 7.3 and 7.4 of the [http://www.bluetooth.com/NR/rdonlyres/0BE438ED-DC1B-41D1-AAC0-1AAA956097A2/980/HID_SPEC_V10.pdf Bluetooth HID specification]. Each byte is written out in hexadecimal, without the 0x prefix, separated by spaces. For example
+
== HID Interface ==
 +
Establishing a HID connection with the Wii Remote can be done on PSM 0x11 for the control pipe and PSM 0x13 for the data pipe using the Bluetooth L2CAP protocol. On Windows you don't need to deal with L2CAP yourself, and can use high-level windows HID functions.
  
(a1) 30 00 00
+
The HID standard allows devices to be self-describing, using a HID descriptor block. This block includes an enumeration of reports that the device understands. A report can be thought of similar to a network port assigned to a particular service. Reports are unidirectional however, and the HID descriptor lists for each port the direction (Input or Output) and the payload size for each port. Like all Bluetooth HID devices, the Wii Remote reports its HID descriptor block when queried using the SDP protocol. However, no information regarding the actual data units within each report is returned, only the length in bytes.
  
is a DATA input packet (0xa1), on channel 0x30, with the two byte payload 0x00, 0x00.
+
Note: An "Input" report is sent by the Wii Remote to the host. An "Output" report is sent by the host to the Wii Remote. When using a Wii Remote, all input reports are prepended with 0xa1 and all output reports are prepended with 0xa2; this is the "(a2)" in many example reports below. Output reports are sent over the data pipe, which is also used to read input reports (thus, the control pipe is essentially unused).
  
Force Feedback is accessible through ALL output channels the same way. This is not included above to avoid clutter.
+
The original Wiimotes (RVL-CNT-01) allowed sending commands using SET REPORT (0x52) over the control pipe, instead of using the data pipe; however, this form does not work on the newer RVL-CNT-01-TR, and as such is not recommended.
  
=== Linux Notes ===
+
These are the reports the Wii Remote uses, and their use:
 +
{| style="border: 1px solid #bbb; border-collapse: collapse; padding: 0.2em 0.2em 0.2em 0.2em;"
 +
|- style="background-color: #ddd;"
 +
| style="border: 1px solid #ccc; padding: 0.2em;" | '''I'''/'''O'''
 +
| style="border: 1px solid #ccc; padding: 0.2em;" | '''ID(s)'''
 +
| style="border: 1px solid #ccc; padding: 0.2em;" | '''Size'''
 +
| style="border: 1px solid #ccc; padding: 0.2em;" | '''Function'''
 +
|- style="background-color: #eef;"
 +
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center;" | O
 +
| style="border: 1px solid #ccc; padding: 0.2em;" | 0x10
 +
| style="border: 1px solid #ccc; padding: 0.2em;" | 1
 +
| style="border: 1px solid #ccc; padding: 0.2em;" | [[#Rumble|Rumble]]
 +
|- style="background-color: #eef;"
 +
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center;" | O
 +
| style="border: 1px solid #ccc; padding: 0.2em;" | 0x11
 +
| style="border: 1px solid #ccc; padding: 0.2em;" | 1
 +
| style="border: 1px solid #ccc; padding: 0.2em;" | [[#Player LEDs | Player LEDs]]
 +
|- style="background-color: #eef;"
 +
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center;" | O
 +
| style="border: 1px solid #ccc; padding: 0.2em;" | 0x12
 +
| style="border: 1px solid #ccc; padding: 0.2em;" | 2
 +
| style="border: 1px solid #ccc; padding: 0.2em;" | [[#Data Reporting | Data Reporting]] mode
 +
|- style="background-color: #eef;"
 +
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center;" | O
 +
| style="border: 1px solid #ccc; padding: 0.2em;" | 0x13
 +
| style="border: 1px solid #ccc; padding: 0.2em;" | 1
 +
| style="border: 1px solid #ccc; padding: 0.2em;" | [[#IR Camera | IR Camera]] Enable
 +
|- style="background-color: #eef;"
 +
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center;" | O
 +
| style="border: 1px solid #ccc; padding: 0.2em;" | 0x14
 +
| style="border: 1px solid #ccc; padding: 0.2em;" | 1
 +
| style="border: 1px solid #ccc; padding: 0.2em;" | [[#Speaker | Speaker]] Enable
 +
|- style="background-color: #eef;"
 +
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center;" | O
 +
| style="border: 1px solid #ccc; padding: 0.2em;" | 0x15
 +
| style="border: 1px solid #ccc; padding: 0.2em;" | 1
 +
| style="border: 1px solid #ccc; padding: 0.2em;" | [[#Status Information| Status Information]] Request
 +
|- style="background-color: #eef;"
 +
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center;" | O
 +
| style="border: 1px solid #ccc; padding: 0.2em;" | 0x16
 +
| style="border: 1px solid #ccc; padding: 0.2em;" | 21
 +
| style="border: 1px solid #ccc; padding: 0.2em;" | Write [[#Memory and Registers | Memory and Registers]]
 +
|- style="background-color: #eef;"
 +
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center;" | O
 +
| style="border: 1px solid #ccc; padding: 0.2em;" | 0x17
 +
| style="border: 1px solid #ccc; padding: 0.2em;" | 6
 +
| style="border: 1px solid #ccc; padding: 0.2em;" | Read [[#Memory and Registers | Memory and Registers]]
 +
|- style="background-color: #eef;"
 +
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center;" | O
 +
| style="border: 1px solid #ccc; padding: 0.2em;" | 0x18
 +
| style="border: 1px solid #ccc; padding: 0.2em;" | 21
 +
| style="border: 1px solid #ccc; padding: 0.2em;" | [[#Speaker | Speaker]] Data
 +
|- style="background-color: #eef;"
 +
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center;" | O
 +
| style="border: 1px solid #ccc; padding: 0.2em;" | 0x19
 +
| style="border: 1px solid #ccc; padding: 0.2em;" | 1
 +
| style="border: 1px solid #ccc; padding: 0.2em;" | [[#Speaker | Speaker]] Mute
 +
|- style="background-color: #eef;"
 +
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center;" | O
 +
| style="border: 1px solid #ccc; padding: 0.2em;" | 0x1a
 +
| style="border: 1px solid #ccc; padding: 0.2em;" | 1
 +
| style="border: 1px solid #ccc; padding: 0.2em;" | [[#IR Camera | IR Camera]] Enable 2
 +
|- style="background-color: #efe;"
 +
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center;" | I
 +
| style="border: 1px solid #ccc; padding: 0.2em;" | 0x20
 +
| style="border: 1px solid #ccc; padding: 0.2em;" | 6
 +
| style="border: 1px solid #ccc; padding: 0.2em;" | [[Wiimote#0x20:_Status | Status Information]]
 +
|- style="background-color: #efe;"
 +
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center;" | I
 +
| style="border: 1px solid #ccc; padding: 0.2em;" | 0x21
 +
| style="border: 1px solid #ccc; padding: 0.2em;" | 21
 +
| style="border: 1px solid #ccc; padding: 0.2em;" | Read [[#Memory and Registers | Memory and Registers]] Data
 +
|- style="background-color: #efe;"
 +
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center;" | I
 +
| style="border: 1px solid #ccc; padding: 0.2em;" | 0x22
 +
| style="border: 1px solid #ccc; padding: 0.2em;" | 4
 +
| style="border: 1px solid #ccc; padding: 0.2em;" | [[#0x22: Acknowledge output report, return function result|Acknowledge output report, return function result]]
 +
|- style="background-color: #efe;"
 +
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center;" | I
 +
| style="border: 1px solid #ccc; padding: 0.2em;" | 0x30-0x3f
 +
| style="border: 1px solid #ccc; padding: 0.2em;" | 2-21
 +
| style="border: 1px solid #ccc; padding: 0.2em;" | [[#Data Reporting | Data reports]]
 +
|}
  
On Linux hosts, the [http://www.bluez.org/ BlueZ] bluetooth drivers, included in recent 2.4 and 2.6 kernels, allow communication with Bluetooth devices. The BlueZ stack includes a kernel module, hidp, which connects to Bluetooth HID-capable devices. To instruct the kernel to connect to the Wiimote, you will need the bluez-utils package, which includes the hidd daemon. First put the Wiimote into discoverable mode, then run
+
For clarity, the convention in this document is to show packets including the Bluetooth-HID command (in parentheses), report ID, and payload, as described in sections 7.3 and 7.4 of the [http://www.bluetooth.com/SiteCollectionDocuments/HID_SPEC_V10.pdf Bluetooth HID specification]. Each byte is written out in hexadecimal, without the 0x prefix, separated by spaces. For example,
  
  hidd --search
+
  (a1) 30 00 00
  
This will scan for all HID-capable devices and connect to them. After receiving the connection, the Player LEDs will continue to blink, but the Wiimote will not power down until the connection is broken. By default, hidd will maintain the connection for 30 minutes beyond the last packet exchange. Received packets can be observed using the hcidump command:
+
is a DATA input packet (0xa1), on channel 0x30, with the two byte payload 0x00, 0x00. When using higher level HID functions rather than Bluetooth functions, the bytes in parentheses will never be present.
  
hcidump -X
+
Force Feedback is accessible through the first byte of ALL output reports in the same way. This is not included above to avoid clutter.
  
Sending packets to the Wiimote requires access to the control and interrupt Bluetooth sockets established by hidd. A modified version of hidd is needed to retain these sockets, and a customized version for the Wiimote will be posted soon.
+
=== Output Report common information ===
  
It should be noted that the HIDP driver creates a /dev/input/eventX when the Wiimote connection is established (see dmesg for a message reporting which event device it is). Preliminary tests suggest this interface is unsuitable for accessing the Wiimote because of the unusual HID information it reports. This is why a lower-level userspace daemon is thought to be required to communicate with the Wiimote directly via Bluetooth sockets.
+
The first byte in many Output reports has a similar meaning. In every single Output Report, bit 0 (0x01) of the first byte controls the Rumble feature. Additionally, bit 2 (0x04) is used in several Output Reports as the ON/OFF flag for the specific feature controlled by it. For example, sending 0x04 to Report 0x19 (Speaker Mute) will mute the speaker:
 +
(a2) 19 04
  
 +
Sending 0x00 will unmute it:
 +
(a2) 19 00
  
=== MacOS X Notes ===
+
These Output Reports share the above behavior: Data Reporting Mode (0x12), IR Camera Enable (0x13), Speaker Enable (0x14), Speaker Mute (0x19), IR Enable 2 (0x1a).
  
*
+
=== Input Report common information ===
  
Now you can control your mouse using the IR sensor and a pair of IR sources. See: http://homepage.mac.com/ianrickard/wiimote/
+
The first two bytes of ALL input reports, except 0x3d, contain the Core Buttons (BB BB). This includes all the 0x2~ status reports, not just the 0x3~ data reports. 0x3d is an exception, since it only returns expansion information.
  
Geniuses at blog.hiroaki.jp developed a program called "DarWiin Remote". This program pairs the Wiimote with a bluetooth compatible Mac. Like the other programs emerging for Windows; this program graphs the accelerometer in the Wiimote. There is an "Enable Mouse" button that allows the Wiimote to be used as a Mouse [it is still buggy, as it is an early release]. You can fool with the LEDs on the Wiimote, as well. The most striking and worthy features is that DarWiin Remote makes the Wiimote fully compatible with OSX's intuitive media center, Front Row. The controls are as follows:
+
= Status Reporting =
  
    * Up - Up
+
== 0x20: Status ==
    * Down - Down
+
To request the status report, send anything to Output Report 0x15. The Status Report will also be automatically sent when an [[#Extension Controllers | Extension Controller]] is connected or disconnected.
    * Left - Left
 
    * Right - Right
 
    * A - Left-Click
 
    * B - Return (Play/Pause in FrontRow)
 
    * Minus - Command + Left (Volume Down in FrontRow)
 
    * Home - Command + ESC (Enter FrontRow and "Menu" button)
 
    * Plus - Command + Right (Volume Up in FrontRow)
 
    * 1 - Page up
 
    * 2 - Page down
 
  
More information at the developers blog: http://blog.hiroaki.jp/2006/12/000433.html
+
This will request the status report (and turn off [[#Rumble | rumble]]):
 +
(a2) 15 00
  
 +
This report is sent either on request (in response to report 0x15), or in response to an expansion being plugged in or unplugged (or synced if wireless). If this report is received when not requested, the application 'MUST' send report 0x12 to change the data reporting mode, otherwise no further data reports will be received.
  
 +
(a1) 20 BB BB LF 00 00 VV
  
 +
BBBB is the core [[#Buttons | Buttons]] data. VV is the current battery level, L is the LED state, and F is a bitmask of flags indicating, whether the battery is flat, whether an expansion is currently connected, etc.
  
The Bluetooth Setup Assistant can be used to setup the connection, by selecting the "Any Device" option, and then selecting "Do not use a passkey with this device" in the Passkey options. However, due to the unusual HID descriptor for the Wiimote, OS X does not activate the Bluetooth HID driver, and the Wiimote powers off at the end of the 20 second timeout. An OS X application that connects to the Wiimote and manually reimplements the HID protocol is in progress, as described in [http://www.wiili.org/forum/osx-wiimote-enabler-t229.html this forum topic].
+
The Wii Remote can report its status, which includes the state of a few basic settings, the status of the [[#Extension Controllers | Extension Controller]] (connected or disconnected), and the battery level.
  
It is possible to access the Wiimote from Linux running directly on Mac hardware, or a Linux virtual machine running inside a virtualizer like [http://www.parallels.com/ Parallels]. See [http://www.wiili.org/forum/accessing-the-wiimote-from-parallels-vm-on-macos-x-t194.html this post] for more information about how to use Bluetooth devices from a Linux VM.
+
VV is the current battery level, and LF is a bitmask of flags:
 
+
{| style="border-collapse: collapse; padding: 0.2em 0.2em 0.2em 0.2em;"
=== Windows XP Notes ===
+
|- style="background-color: #ddd; text-align: center;"
 
+
| style="border: 1px solid #ccc; padding: 0.2em;" | '''Bit'''
There is a Windows "driver" called [[GlovePIE]][http://carl.kenner.googlepages.com/glovepie_download]. It can read the buttons and accelerations, and it can set the leds and the rumble.
+
| style="border: 1px solid #ccc; padding: 0.2em;" | '''Mask'''
 
+
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddf;" | '''Meaning'''
The wiimote is accessible exacly like a USB HID device: using Delphi and the [http://www.delphi-gems.com/HID.php TJvHidDeviceController] component. You can access all the info with the demo program (ReadWriteDemo\SimpleHIDWrite.dpr). I am using W2K.
+
|- style="background-color: #eee;"
 
+
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center;" | 0
The wiimote can be paired with Windows XP and a generic driver is installed. It appears in the game controller's panel but cannot be configured in any way.
+
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center;" | 0x01
 
+
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eef;" | Battery is nearly empty
Some people have had trouble sending reports to the Wiimote. Some people have got around this by using a different bluetooth stack. Being unable to send reports to the wiimote makes it impossible to read the acceleration data. The Bluesoleil bluetooth stack is most recommended to fix these problems. The problem of sending reports occurs with the TJvHidDeviceController component.
+
|- style="background-color: #eee;"
 
+
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center;" | 1
== Inputs ==
+
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center;" | 0x02
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eef;" | An [[#Extension Controllers | Extension Controller]] is connected
 +
|- style="background-color: #eee;"
 +
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center;" | 2
 +
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center;" | 0x04
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eef;" | [[#Speaker | Speaker]] enabled
 +
|- style="background-color: #eee;"
 +
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center;" | 3
 +
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center;" | 0x08
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eef;" | IR camera enabled
 +
|- style="background-color: #eee;"
 +
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center;" | 4
 +
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center;" | 0x10
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eef;" | [[#Player LEDs | LED]] 1
 +
|- style="background-color: #eee;"
 +
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center;" | 5
 +
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center;" | 0x20
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eef;" | [[#Player LEDs | LED]] 2
 +
|- style="background-color: #eee;"
 +
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center;" | 6
 +
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center;" | 0x40
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eef;" | [[#Player LEDs | LED]] 3
 +
|- style="background-color: #eee;"
 +
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center;" | 7
 +
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center;" | 0x80
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eef;" | [[#Player LEDs | LED]] 4
 +
|}
  
=== Buttons ===
+
== 0x21: Read Memory Data ==
 +
This report is sent when a read memory request is made. It returns 1 to 16 bytes of data at a time.
  
There are 12 buttons on the Wiimote. Four of them are arranged into a directional pad, and the rest are spread over the controller.
+
(a1) 21 BB BB SE AA AA DD DD DD DD DD DD DD DD DD DD DD DD DD DD DD DD
  
By default, whenever a button is pressed or released, a packet is sent to the host via HID input report 30H, with a payload containing a 2-byte bitmask with the current state of all the buttons.
+
BBBB is the core [[#Buttons | Buttons]] data.  
  
The button state also seems to be included in the first two bytes of all other input reports.
+
S (high nybble of SE) is the size in bytes, minus one, for the current data packet. This is 0xf (16 bytes) for all but the last packet, where it might be less if the requested number of bytes is not a multiple of 16.
  
Some of the bits in the first two bytes don't seem to be directly related to button presses, and are somewhat unknown. Their activity seems to point towards them being the LSBs of the motion bytes, but it is impossible to determine which since their data is too random.
+
E (low nybble of SE) is the error flag. Known error values are 0 for no error, 7 when attempting to read from a write-only register or an expansion that is not connected, and 8 when attempting to read from nonexistant memory addresses.  
  
When the A button is pressed, this HID DATA input packet is received:
+
AA AA are the 2 least significant bytes of the absolute memory address of the first byte of data returned (the high byte of the offset is not returned, and neither is whether it is a register or memory that is being used. Thus, this must be known from the read request).
  
(a1) 30 00 08
+
The DD bytes are the data, padded with zeroes to 16 bytes. If more than 16 bytes are requested, multiple packets will be received, with AA AA addresses increasing by 16 each time.
  
and when it is released, this is packet received:
+
== 0x22: Acknowledge output report, return function result ==
  
(a1) 30 00 00
+
This input report is sent to the host to report an error related to an output report, or the function result from that output report. It is sent when bit 1 of the first byte of any output report is set.
  
The bit assignments (in big endian order) for the buttons are:
+
(a1) 22 BB BB RR EE
{| border="1"
 
! Button
 
! Number (dec)
 
! Value (hex)
 
|-
 
| Two
 
| 1
 
| 0x0001
 
|-
 
| One
 
| 2
 
| 0x0002
 
|-
 
| B
 
| 3
 
| 0x0004
 
|-
 
| A
 
| 4
 
| 0x0008
 
|-
 
| Minus
 
| 5
 
| 0x0010
 
|-
 
| ? motion ?
 
| 6
 
| 0x0020
 
|-
 
| ? motion ?
 
| 7
 
| 0x0040
 
|-
 
| Home
 
| 8
 
| 0x0080
 
|-
 
| Left
 
| 9
 
| 0x0100
 
|-
 
| Right
 
| 10
 
| 0x0200
 
|-
 
| Down
 
| 11
 
| 0x0400
 
|-
 
| Up
 
| 12
 
| 0x0800
 
|-
 
| Plus
 
| 13
 
| 0x1000
 
|-
 
| ? motion ?
 
| 14
 
| 0x2000
 
|-
 
| ? motion ?
 
| 15
 
| 0x4000
 
|-
 
| ? Reading Mii ?
 
| 16
 
| 0x8000
 
|}
 
  
The power button is unusual in that it sends a disconnect request when held down instead of emitting a normal button code.
+
BBBB is the core [[#Buttons | Buttons]] data.  
  
=== Motion Sensor ===
+
RR is the output report number that the Wii remote is acknowledging it received.
  
[[Image:Wiimote_axis2.png|thumb|200px|Wiimote coordinate system]]
+
EE is the error code or function result.
 +
00 = success.
 +
03 = error.
 +
04 = unknown (possibly returned by report 16H, 17H or 18H)
 +
05 = unknown (possibly returned by report 12H).
 +
08 = unknown (possibly returned bt report 16H).
  
The motion of the remote is sensed by a 3-axis linear accelerometer located slightly left of the large A button. The integrated circuit is the [http://www.analog.com/en/prod/0%2C2877%2CADXL330%2C00.html ADXL330 ]([http://www.analog.com/UploadedFiles/Data_Sheets/ADXL330.pdf data sheet]), manufactured by Analog Devices. This device is physically rated to measure accelerations over a range of at least +/- 3g with 10% sensitivity.
+
= Data Reporting =
  
Inside the chip is a small micromechanical structure which is supported by springs built out of silicon. Differential capacitance measurements allow the net displacement of the tiny mass to be converted to a voltage, which is then digitized. It is important to note that the sensor does not measure the acceleration of the Wiimote, but rather the force exerted by the test mass on its supporting springs. Due to the sign convention used, this quantity is proportional to the net force exerted by the player's hand on the Wiimote when holding it. Thus, at rest on a flat table, the accelerometer reports vertical force of +g (the mass can be normalized away into the aribitrary units), and when dropped reports a force of nearly zero.
+
The Wii Remote has a number of different data reporting modes. Each of these modes combines certain Core data features with data from external peripherals, and sends it to the host through one of the report IDs, determined by the mode. The data format from the peripherals is determined by the peripherals themselves, all the Wii Remote controller does is pull bytes from them and send them out to the host. Due to this, certain feature combinations are not available, as there are not enough bytes for them in any of the output modes.
  
The sensor uses a right-handed coordinate system with the positive X-axis to the left, and the positive Z-axis pointing upward, when the remote is held horizontally, as shown in the diagram at the right. Forces on each axis are digitized to 8 bit unsigned integers, with the zero scale set to 0x80. Manufacturing and calibration defects of course cause some intrinsic zero offsets. However, the Earth's gravitational field and a flat, level surface upon which to rest the remote, in principle, allow any offsets or chip skew to be measured and calibrated away in software. Decomposing the force measured by the sensor into rotation and linear components is tricky, and discussed further on the [Motion analysis] page.
+
The Data Reporting Mode is set by sending a two-byte command to Report 0x12:
  
The Wiimote does not normally report motion sensor readings to the host, but can be requested by sending SET_REPORT request to channel 0x12:
+
(a2) 12 TT MM
  
(52) 12 00 31
+
Bit 2 of TT specifies whether continuous reporting is desired. If bit 2 (0x04) is set, the Wii Remote will send reports whether there has been any change to the data or not. Otherwise, the Wii Remote will only send an output report when the data has changed.
  
The 3rd byte is a bitmask. 0x01 turns on the rumble, and 0x04 turns on continuous output. If 0x04 is not set, packets are only output when the values change (almost always when the motion sensor is enabled). If 0x04 is set, values are output continuously (it's obvious when channel/mode 0x30 is chosen, which disables motion readback. With byte3=0x04, the button values are output multiple times a second. With byte3=0x00, they are output only when you press or release a button). The 4th byte specifies which HID channel to which log sensor output should be sent. After receiving this command once, the Wiimote will send back stream of DATA INPUT packets on the requested channel/mode (0x31 in the above example), where the 5th, 6th and 7th bytes contain the X, Y, and Z readings of the accelerometer. A sample packet when the Wiimote is at rest, face up, on a table is:
+
MM specifies the Reporting Mode. Each Mode is specified by the Output Report ID that the data will be sent to. For example, this will set mode to 0x33:
 +
(a2) 12 00 33
  
(a1) 31 40 20 86 8a a5
+
Data will then arrive through Input Report 0x33.
  
where 0x86 is the X-axis measurement, 0x8a is the Y-axis measurement, and 0xa5 is the Z-axis measurement. The first two bytes of the payload, 0x40 and 0x20, are the button values and some unknown bits. It is not known what these bits are, as they change apparently randomly with no obvious pattern. One possible explanation is that they are the LSBs of the motion sensor values. The button values are still in the same place and have the same meanings as in the Buttons section.
+
Upon powerup, the Data Reporting Mode defaults to 0x30. Following a connection or disconnection event on the Extension Port, data reporting is disabled and the Data Reporting Mode must be reset before new data can arrive.
  
Other channels can be selected for motion sensor reports, including 0x31, 0x33, 0x35, 0x37, 0x3e and 0x3f. If either 0x3e or 0x3f are selected then sensor readings will alternate between the two channels. The length of the report payload will depend upon the channel, as show in the table in HID Interface section. Channels which have payloads longer than needed for the motion sensor are used to report information from other peripherals, such as the IR camera. If these are disabled, 0xff data is returned in the extra space:
+
Modes which include Accelerometer data also embed part of it in the unused Buttons bits. In all modes except for 0x3e/0x3f, the Buttons data includes the LSBs of the Accelerometer data. In mode 0x3e/0x3f, the interleaved Buttons data includes the Z-axis Accelerometer data.
  
(a1) 33 40 00 86 8a a5 ff ff ff ff ff ff ff ff ff ff ff ff ff
+
== 0x30: Core Buttons ==
 +
This mode returns data from the buttons in the Wii Remote:
  
Motion sensor reports can be stopped by setting the output channel to 0x30:
+
(a1) 30 BB BB
  
(52) 12 00 30
+
BBBB is the core [[#Buttons | Buttons]] data.
  
It seems that the channel mode is actually a mode selection / bitmask. Button output is always enabled. Mode 0x30 is just buttons, 0x31 is motion sensor, 0x32 is IR camera (???), and 0x33 is both IR camera and motion sensor. The two sets of data are tacked onto each other, first the button values, then the three motion sensor bytes, then the camera bytes. Other modes seem to include motion data and / or IR data in different ways. What looks like it might be IR data is always 0xff currently, as we don't know how to enable the camera yet.
+
== 0x31: Core Buttons and Accelerometer ==
 +
This mode returns data from the buttons and the accelerometer in the Wii Remote:
  
=== IR Sensor ===
+
(a1) 31 BB BB AA AA AA
  
During R&D, Nintendo discovered the motion sensors were not accurate enough to use the remote to control an on-screen cursor. To correct this, they augmented the remote with an infrared image sensor on the front designed to locate two IR beacons within the controller's field of view. The beacons are housed within a device misleadingly called the sensor bar. The sensor bar is powered by the Wii base unit, and contains 2 groups of IR LEDs, spaced 7.5 inches apart. Each group is composed of 5 LEDs, but homemade sensor bars have been effective with fewer LEDs, so long as the intensity is sufficient. The cable from the Wii to the sensor bar only carries power. No information is passed either to or from the sensor bar, and the intensity is not modulated in any way.
+
BBBB is the core [[#Buttons | Buttons]] data. AA AA AA is the [[#Accelerometer | Accelerometer]] data.
  
These two sources of IR light are tracked by a [http://www.pixart.com.tw/ PixArt] sensor in the front of the Wiimote housing. By tracking the locations of these two points in the sensors 2D field of view, the system can derive more accurate pointing information. Not much is known about this feature yet, but circumstantial evidence from the [http://www.wiiregamers.com/news/2492 Nintendo/PixArt] press release suggests that Nintendo is using a PixArt System-on-a-Chip to process the images on-board the Wiimote and sends the minimum information needed for tracking back to the base unit. Transmitting full 2D images constantly would require a prohibitive amount of bandwidth, especially when multiple remotes are in use.
+
== 0x32: Core Buttons with 8 Extension bytes ==
 +
This mode returns data from the buttons in the Wii Remote, and data from an extension controller connected to it:
  
The Wiimote detects and transfers up to four IR hotspots back to the host. Various amounts of data can be requested, from position values only, position and size, to position, size and pixel value. The amount of different configurations are quite numerous, and also ties in with the connected peripheral device.
+
(a1) 32 BB BB EE EE EE EE EE EE EE EE
  
Work is being done on how to enable and use the IR sensor. Currently, we can enable it and set the correct sensitivity, and receive position and brightness data successfully.
+
BBBB is the core [[#Buttons | Buttons]] data. The 8 EE bytes are from the [[#Extension Controllers | Extension Controller]] currently connected to the Wii Remote.
  
==== marcan's info ====
+
== 0x33: Core Buttons and Accelerometer with 12 IR bytes ==
 +
This mode returns data from the buttons, accelerometer, and IR Camera in the Wii Remote:
  
Here are the reports you need to send to activate IR transmission:
+
  (a1) 33 BB BB AA AA AA II II II II II II II II II II II II
  (52) 12 00 33
 
(52) 13 04
 
(52) 1A 04
 
(52) 16 04 B0 00 30 01 08 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00
 
(52) 16 04 B0 00 06 01 90 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00
 
(52) 16 04 B0 00 08 01 C0 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00
 
(52) 16 04 B0 00 1A 01 40 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00
 
(52) 16 04 B0 00 33 01 33 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00
 
  
Python / pseudocode follows.
+
BBBB is the core [[#Buttons | Buttons]] data. AA AA AA is the [[#Accelerometer | Accelerometer]] data. The 12 II bytes are from the built-in [[#IR Camera | IR Camera]].
  
FEATURE_DISABLE = 0x00
+
== 0x34: Core Buttons with 19 Extension bytes ==
FEATURE_ENABLE = 0x04
+
This mode returns data from the buttons in the Wii Remote, and data from an extension controller connected to it:
 
IR_MODE_OFF = 0
 
IR_MODE_STD = 1
 
IR_MODE_EXP = 3
 
IR_MODE_FULL = 5
 
 
CMD_SET_REPORT = 0x52
 
 
RID_LEDS = 0x11
 
RID_MODE = 0x12
 
RID_IR_EN = 0x13
 
RID_SPK_EN = 0x14
 
RID_STATUS = 0x15
 
RID_WMEM = 0x16
 
RID_RMEM = 0x17
 
RID_SPK = 0x18
 
RID_SPK_MUTE = 0x19
 
RID_IR_EN2 = 0x1A
 
 
MODE_BASIC = 0x30
 
MODE_ACC = 0x31
 
MODE_ACC_IR = 0x33
 
MODE_FULL = 0x3e
 
 
def send(cmd,report,data):
 
    Send cmd, report, data to Wiimote. If cmd is CMD_SET_REPORT, this amounts to sending data to the specified report.
 
    Interface code to Wiimote goes here.
 
 
def setmode(mode,cont):
 
        aux = 0
 
        if rmbl:
 
                aux |= 0x01
 
        if cont:
 
                aux |= 0x04
 
        send(CMD_SET_REPORT,RID_MODE,[aux,mode])
 
 
# size here is redundant, since we can just use len(data) if we want.
 
def senddata(data,offset,size): # see writing to data under On-board memory.
 
        of1 = offset >>; 24 & 0xFF #extract offset bytes
 
        of2 = offset >> 16 & 0xFF
 
        of3 = offset >> 8 & 0xFF
 
        of4 = offset & 0xFF
 
        data2 = data + [0]*(16-len(data)) # append zeros to pad data if less than 16 bytes
 
        if len(data2) > 16:
 
                data2 = data2[:16] # crop data if we have too much
 
        # format is [OFFSET (BIGENDIAN),SIZE,DATA (16bytes)]
 
        send(CMD_SET_REPORT,RID_WMEM,[of1,of2,of3,of4,size]+data2)
 
 
# this seems to be the minimal code to get it to work
 
setmode(MODE_ACC_IR,0)
 
send(CMD_SET_REPORT,RID_IR_EN,[FEATURE_ENABLE])
 
send(CMD_SET_REPORT,RID_IR_EN2,[FEATURE_ENABLE])
 
senddata([8],0x04B00030,1) # enable IR data out
 
senddata([0x90],0x04B00006,1) # sensitivity constants (guessed, Cliff seems to have more data, but this works for me)
 
senddata([0xC0],0x04B00008,1)
 
senddata([0x40],0x04B0001A,1)
 
senddata([IR_MODE_EXP],0x04B00033,1) # enable IR output with specified data format
 
  
  # this is Cliff's version pythonified, probably more accurate as far as sensitivity. Works pretty much the same for me.
+
  (a1) 34 BB BB EE EE EE EE EE EE EE EE EE EE EE EE EE EE EE EE EE EE EE
setmode(MODE_ACC_IR,0)
 
send(CMD_SET_REPORT,RID_IR_EN,[FEATURE_ENABLE])
 
send(CMD_SET_REPORT,RID_IR_EN2,[FEATURE_ENABLE])
 
senddata([1],0x04B00030,1) # seems to enable the IR peripheral
 
senddata([0x02, 0x00, 0x00, 0x71, 0x01, 0x00, 0xaa, 0x00, 0x64],0x04B00000,9)
 
senddata([0x63, 0x03],0x04B0001A,2)
 
# this seems incorrect - for FULL IR mode, we must use FULL wiimote mode (0x3e).
 
# otherwise the data is probably garbled.
 
senddata([IR_MODE_FULL],0x04B00033,1)
 
senddata([8],0x04B00030,1) # Enable data output. Can be specified first it seems, we don't really need to be in mode 1.
 
  
Output format EXP is three bytes per dot recognized. Bytes 0 and 1 are X and Y, byte 2 is the LSBs of X and Y and a size value. In binary:
+
BBBB is the core [[#Buttons | Buttons]] data. The 19 EE bytes are from the [[#Extension Controllers | Extension Controller]] currently connected to the Wii Remote.
xxxxxxxx yyyyyyyy yyxxssss
 
  
No dot is indicated by 0xff data. This means you will not see any change in the output data even if you did do all the initialization correctly, unless you actually point the Wiimote at a (powered) sensor bar!
+
== 0x35: Core Buttons and Accelerometer with 16 Extension Bytes ==
 +
This mode returns data from the buttons and accelerometer in the Wii Remote, and data from an extension controller connected to it:
  
Output format for FULL is unknown, but we know the data is interleaved. Data sent to report 0x3e contains the data for the first two dots recognized, data sent to 0x3f contains the data for the next two dots recognized, if any.
+
(a1) 35 BB BB AA AA AA EE EE EE EE EE EE EE EE EE EE EE EE EE EE EE EE
  
Note that the original hidd patch has trouble with 0x00 bytes in commands. Make sure you're sending the right data.
+
BBBB is the core [[#Buttons | Buttons]] data. AA AA AA is the [[#Accelerometer | Accelerometer]] data. The 16 EE bytes are from the [[#Extension Controllers | Extension Controller]] currently connected to the Wii Remote.
  
 +
== 0x36: Core Buttons with 10 IR bytes and 9 Extension Bytes ==
 +
This mode returns data from the buttons and IR camera in the Wii Remote, and data from an extension controller connected to it:
  
 +
(a1) 36 BB BB II II II II II II II II II II EE EE EE EE EE EE EE EE EE
  
==== Cliff's info ====
+
BBBB is the core [[#Buttons | Buttons]] data. The 10 II bytes are from the built-in [[#IR Camera | IR Camera]], and the 9 EE bytes are from the [[#Extension Controllers | Extension Controller]] currently connected to the Wii Remote.
  
Set the report 0x12 to a report number such as 0x33.
+
== 0x37: Core Buttons and Accelerometer with 10 IR bytes and 6 Extension Bytes ==
 +
This mode returns data from the buttons, accelerometer, and IR camera in the Wii Remote, and data from an extension controller connected to it:
  
You need to start by setting reports 0x13 and 0x1a to 0x04 to enable transmission.
+
(a1) 37 BB BB AA AA AA II II II II II II II II II II EE EE EE EE EE EE
  
Then to activate the IR Sensor you need to write to the wiimote's memory. Set 0x04b00030 to 1, then set the sensitivity, then set 0x04b00033 to 5, then set 0x04b00030 to 8.
+
BBBB is the core [[#Buttons | Buttons]] data. AA AA AA is the [[#Accelerometer | Accelerometer]] data. The 10 II bytes are from the built-in [[#IR Camera | IR Camera]], and the 6 EE bytes are from the [[#Extension Controllers | Extension Controller]] currently connected to the Wii Remote.
  
The sensitivity data is 9 bytes at 0x04b00000 and 2 bytes at 0x04b0001a. For midrange sensitivity, it's {0x02, 0x00, 0x00, 0x71, 0x01, 0x00, 0xaa, 0x00, 0x64}, and {0x63, 0x03}.
+
== 0x3d: 21 Extension Bytes ==
 +
This mode returns data from an extension controller connected to the Wii Remote. It is the only input report that does not include core buttons.
  
== Outputs ==
+
(a1) 3d EE EE EE EE EE EE EE EE EE EE EE EE EE EE EE EE EE EE EE EE EE
  
=== Player LEDs ===
+
The 21 EE bytes are from the [[#Extension Controllers | Extension Controller]] currently connected to the Wii Remote.
  
The bottom edge of the remote body contains 4 blue LEDs. These LEDs are used during normal play to indicate that the remote is in Bluetooth discoverable mode (all blinking), or to indicate the player number of the controller (one light illuminated). The LEDs are independently controllable, however, using SET_REPORT ouput packet to channel 11, which has a payload size of 1. The most-significant 4 bits control each LED, with bit 4 corresponding to the player 1 LED. Channel 11 also can control the rumble feature, so it is best to keep the 4 least-significant bits zero in order to avoid vibrating the controller. For example, this packet turns on the player 1 LED only:
+
== 0x3e / 0x3f: Interleaved Core Buttons and Accelerometer with 36 IR bytes ==
 +
Both 0x3e and 0x3f are equivalent, and return data alternately through report IDs 0x3e and 0x3f. The data is interleaved, and is returned at half the speed of other modes (as two reports are needed for a single data unit). This mode returns data from the buttons, accelerometer, and IR camera in the Wii Remote:
  
  (52) 11 10
+
  (a1) 3e BB BB AA II II II II II II II II II II II II II II II II II II
 +
(a1) 3f BB BB AA II II II II II II II II II II II II II II II II II II
  
Since each LED has its own bit, any combination of LEDs can be illuminated.
+
BBBB is the core button data, as specified in the [[#Buttons | Buttons]] section. AA AA is the [[#Accelerometer | Accelerometer]] data, in a format specific to this mode described in the [[#Interleaved Accelerometer Reporting| Interleaved Accelerometer Reporting]] section. The 36 II bytes are from the built-in [[#IR Camera | IR Camera]].
  
=== Force Feedback ===
+
= Memory and Registers =
 +
The Wii Remote includes a built-in [[#EEPROM Memory | EEPROM memory]], part of which is accessible to the user to store that. This user part is used to store calibration constants, as well as the Mii Data. Additionally, many peripherals on the Wii Remote have [[#Control Registers | registers]] which are accessible through a portion of the address space.
  
Force feedback is provided via a "rumble pack" style device inside the Wiimote body. It is roughly an off-center weight connected to a motor which can be activated to cause the controller to vibrate. The motor can be activated by sending a SET_REPORT output packet to channels 0x11, 0x13, 0x14, 0x15, 0x19 or 0x1a with the least significant bit set:
+
Both built-in memory and peripheral registers are accessed using the same reports, where a flag is used to select between the two.
  
(52) 13 01
+
== Reading and Writing ==
  
And the vibration can be turned off by clearing the bit:
+
To read data, commands are sent to Output Report 0x17:
  
  (52) 13 00
+
  (a2) 17 MM FF FF FF SS SS
  
So far, it appears all channels are equivalent, though using channel 0x11 is not advised because it also controls the player LEDs.
+
FF FF FF is the offset, and SS SS is the size to read in bytes (both in big-endian format). Bit 2 (0x04) of MM selects the address space. Clearing this bit results in reading from [[#EEPROM Memory | EEPROM Memory]], while setting it results in reading from the [[#Control Registers | control registers]]. Setting bit 3 (0x08) also works to access registers, but setting both results in errors. As with all other reports, it also includes the Rumble flag, which must be set to the current rumble state to avoid affecting it.
  
=== Speaker ===
+
Data read is returned through Input Report 0x21:
  
A small speaker is embedded in the top face of the controller to provide audio feedback. The control method of this speaker is still somewhat unknown.
+
(a1) 21 BB BB SE FF FF DD DD DD DD DD DD DD DD DD DD DD DD DD DD DD DD
  
== Onboard Memory ==
+
BB BB is the state of the buttons on the Wii Remote. During data reads, regular input reporting is temporarily suspended. Button data is available through the data input reports, but no other input data can be collected while the transfer lasts. FF FF is the offset expressed in absolute memory address of the Wii remote memory for the first byte of data returned (the high byte of the offset is not returned, and neither is which data memory is being used. Thus, this must be known from the read request). E (low nybble of SE) is the error flag. Known error values are 0 for no error, 7 when attempting to read from a write-only register, and 8 when attempting to read from nonexistant memory. S (high nybble of SE) is the size in bytes, minus one, for the current data packet. This is 0xf (16 bytes) for all but the last packet, where it might be less if the requested number of bytes is not a multiple of 16. The DD bytes are the data, padded with zeroes to 16 bytes. If more than 16 bytes are requested, multiple packets will be received, with FF FF offsets increasing by 16 each time.
  
=== Flash Memory ===
+
To write data, commands are sent to Output Report 0x16:
  
On a blank wiimote, which was bought seperately and has never been connected to a Wii, and has never received a report other than 0x17 (read memory) with the first byte as 00; the memory is structured like this:
+
(a2) 16 MM FF FF FF SS DD DD DD DD DD DD DD DD DD DD DD DD DD DD DD DD
  
Addresses 0x0000 to 0x003F:
+
The meaning of the bytes is the same as during reads, except that size can be a maximum of 16 bytes (as there is only space for that much data), and the actual data to write follows (the DD bytes), padded out to 16 bytes.
  
  A1 AA 8B 99 AE 9E 78 30 A7 74 D3 A1 AA 8B 99 AE
+
Some kind of acknowledgement is received on Input Report 0x22. This has not been investigated yet.
  9E 78 30 A7 74 D3 82 82 82 15 9C 9C 9E 38 40 3E
 
  82 82 82 15 9C 9C 9E 38 40 3E 00 00 00 00 00 00
 
  00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00
 
  
 +
== EEPROM Memory ==
 +
[[File:Wii Remote Flash.jpg|thumb|right|Wii remote Flash memory chip|]]There is a 128kbit (= 16kB) EEPROM chip ([http://www.st.com/stonline/products/literature/ds/4578/m24128-bw.pdf Data Sheet] / [http://www.sparkfun.com/tutorial/WiiRemote/NintendoWii-I2C-Data.zip Full EEPROM dump from a sample Wii Remote]) in the Wii Remote. Part of its contents include code for the built-in microcontroller, and a generic section which can be freely read and written by the host. This section is 0x1700 bytes long, and part of this memory is used to store the  [[Wiimote/Mii Data|Mii Data]]. It can be accessed by reading from/writing to addresses 0x0000-0x16FF in the Wii Remote's virtual memory space; in the actual EEPROM chip, the data is located at 0x0070-0x176F.
  
Addresses 0x0040 to 0x15FF: all zeros
+
The [[Wiimote/Firmware|firmware stored in the Wiimote has been disassembled]].
  
Addresses 0x16FF to 0xFFFF: Don't exist. They return an error if you try to read from them. You won't get the error if you start reading from at or before the 0x16FF boundary, but you will just get zeroes for the invalid bytes in that case.
+
The [http://www.broadcom.com/products/Bluetooth/Bluetooth-RF-Silicon-and-Software-Solutions/BCM2042 BCM2042] microcontroller built into the Wii Remote includes a large 108kb on-chip ROM section for storing firmware. If the EEPROM chip really contains code for the BCM2042 then this was probably done to make firmware updates possible, so there might be a way of accessing the other parts of the EEPROM via Bluetooth as well.
 +
From the [http://www.broadcom.com/collateral/pb/2042-PB03-R.pdf BCM2042 Product Brief]: "ROM-based design eliminates external flash memories; Flash option offered to support feature development".
  
For the Flash memory the address is only 2 bytes. So address 0x010000 is the treated the same as address 0x0000. This is true all the way up to 0xFF0000. That byte (0x00FF0000) is always ignored unless the most significant byte (0xFF000000) has bit 2 (0x04) set.
+
On a virgin Wii Remote, acquired separately (not bundled with a Wii), that has never communicated with any device (except the PC used to dump the memory contents), most of the memory is blank (0x00). However, the first few bytes contain some information:
  
===Control Registers===
+
0000:  A1 AA 8B 99 AE 9E 78 30 A7 74 D3 A1 AA 8B 99 AE
 +
0010:  9E 78 30 A7 74 D3 82 82 82 15 9C 9C 9E 38 40 3E
 +
0020:  82 82 82 15 9C 9C 9E 38 40 3E 00 00 00 00 00 00
 +
0030:  00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00
  
The control registers have bit 2 (0x04) set in the first byte. Bit 0 (0x01) in the first byte is the rumble flag, and is not considered part of the address, so 0x05A20000 is the same address as 0x04A20000. Only registers 0x04A20000 to 0x04A30000 are readable. But they don't seem to provide any useful information on a fresh new Wiimote.
+
This can be better visualized as two sequences, each one repeated twice:
 +
0000:  A1 AA 8B 99 AE 9E 78 30 A7 74 D3
 +
000B:  A1 AA 8B 99 AE 9E 78 30 A7 74 D3
 +
0016:  82 82 82 15 9C 9C 9E 38 40 3E
 +
0020:  82 82 82 15 9C 9C 9E 38 40 3E
  
Registers 0x04000000 - 0x49FFFFF return error 7 (instead of error 8 like normal nonexistant memory). Perhaps it is write only, or perhaps it doesn't exist.
+
These are two blocks of calibration data (for redundancy), where the last byte of each is a checksum (all prior bytes plus 0x55). Nintendo's libraries will only use a calibration block if its checksum is correct, falling back to the second one if the first one is incorrect, and falling back to default values otherwise.
  
Registers 0x04A00000 - 0x4A1FFFF don't exist because they return error 8 (like normal non-existant memory)
+
The first block contains calibration data related to the IR camera: the position of 4 reference values.  Nintendo's code uses this to compute a center offset, scale, and possibly an angle offset.  It also looks like the values could appear in any order; code exists to check for values in each quadrant (below and above x=512 and y=384).
  
Registers 0x04A20000 - 0x4A30000 are readable!! But are all the byte 0xFF. Except registers whose address ends in 0xFF, which is the byte 0x00.
+
{| style="border-collapse: collapse; padding: 0.2em 0.2em 0.2em 0.2em; text-align: center;"
 +
|- style="background-color: #ddd;"
 +
| style="background-color: #fff;" colspan="2" | &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;" colspan="2" | '''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;" | 0x00
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eee;" | 0x0B
 +
| style="border: 1px solid #ccc; padding: 0.2em;" colspan="8" | '''X 1'''<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;" | 0x01
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eee;" | 0x0C
 +
| style="border: 1px solid #ccc; padding: 0.2em;" colspan="8" | '''Y 1'''<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;" | 0x02
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eee;" | 0x0D
 +
| style="border: 1px solid #ccc; padding: 0.2em;" colspan="2" | '''Y 1'''<span style="color: #777;">&lt;<span style="color: #c00;">9:8</span>&gt;</span>
 +
| style="border: 1px solid #ccc; padding: 0.2em;" colspan="2" | '''X 1'''<span style="color: #777;">&lt;<span style="color: #c00;">9:8</span>&gt;</span>
 +
| style="border: 1px solid #ccc; padding: 0.2em;" colspan="2" | '''Y 2'''<span style="color: #777;">&lt;<span style="color: #c00;">9:8</span>&gt;</span>
 +
| style="border: 1px solid #ccc; padding: 0.2em;" colspan="2" | '''X 2'''<span style="color: #777;">&lt;<span style="color: #c00;">9:8</span>&gt;</span>
 +
|- style="background-color: #ded;"
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eee;" | 0x03
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eee;" | 0x0E
 +
| style="border: 1px solid #ccc; padding: 0.2em;" colspan="8" | '''X 2'''<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;" | 0x04
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eee;" | 0x0F
 +
| style="border: 1px solid #ccc; padding: 0.2em;" colspan="8" | '''Y 2'''<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;" | 0x05
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eee;" | 0x10
 +
| style="border: 1px solid #ccc; padding: 0.2em;" colspan="8" | '''X 3'''<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;" | 0x06
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eee;" | 0x11
 +
| style="border: 1px solid #ccc; padding: 0.2em;" colspan="8" | '''Y 3'''<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;" | 0x07
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eee;" | 0x12
 +
| style="border: 1px solid #ccc; padding: 0.2em;" colspan="2" | '''Y 3'''<span style="color: #777;">&lt;<span style="color: #c00;">9:8</span>&gt;</span>
 +
| style="border: 1px solid #ccc; padding: 0.2em;" colspan="2" | '''X 3'''<span style="color: #777;">&lt;<span style="color: #c00;">9:8</span>&gt;</span>
 +
| style="border: 1px solid #ccc; padding: 0.2em;" colspan="2" | '''Y 4'''<span style="color: #777;">&lt;<span style="color: #c00;">9:8</span>&gt;</span>
 +
| style="border: 1px solid #ccc; padding: 0.2em;" colspan="2" | '''X 4'''<span style="color: #777;">&lt;<span style="color: #c00;">9:8</span>&gt;</span>
 +
|- style="background-color: #ded;"
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eee;" | 0x08
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eee;" | 0x13
 +
| style="border: 1px solid #ccc; padding: 0.2em;" colspan="8" | '''X 4'''<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;" | 0x09
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eee;" | 0x14
 +
| style="border: 1px solid #ccc; padding: 0.2em;" colspan="8" | '''Y 4'''<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;" | 0x0A
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eee;" | 0x15
 +
| style="border: 1px solid #ccc; padding: 0.2em;" colspan="8" | '''Checksum'''
 +
|}
  
Registers 0x04B00000 - 0x4BFFFFF return error 7 when you read them, but some of them are known to be settable, and to control the IR camera in the Wiimote. The first 9 bytes are part of the IR sensitivity settings. Bytes 0x04B0001A and 0x04B0001B are also part of the IR sensitivity settings. Byte 0x04B00030 turns IR on and off (8 is on, 1 is off in order to set sensitivity). Byte 0x04B00033 sets the IR mode (0x33 tells it to return IR data as 4 lots of 3 bytes).
+
The data given above decodes to (0A1, 2AA), (399, 28B), (39E, 078), (0A7, 030).
  
Registers 0x04C00000 - 0x4FFFFFF return error 7 when you read them.
+
{{Collapse|title=Additional examples|text=<ul><li><code>b4 b8 b8 a0 ad b6 66 30 a5 61 18</code>: (3B4, 2B8), (0A0, 2B8), (3B6, 066), (0A5, 030)</li><li><code>70 5e 03 7b 50 83 9b b8 7a a5 e6</code>: (070, 05E), (37B, 003), (383, 29B), (07A, 2B8)</li><li><code>83 9b 80 7e 50 8a 4c 3b 8d 98 f7</code>: (083, 29B), (07E, 080), (38A, 04C), (38D, 23B)</li><li><code>6f a9 8b 7d ae 78 5d 03 83 67 e5</code>: (06F, 2A9), (37D, 28B), (078, 05D), (383, 003)</li></ul>}}
  
 +
The four bytes starting at 0x0016 and 0x0020 store the calibrated zero offsets for the accelerometer (high 8 bits of X,Y,Z in the first three bytes, low 2 bits packed in the fourth byte as --XXYYZZ). The four bytes at 0x001A and 0x24 store the force of gravity on those axes. The byte at 0x1e and 0x28 seems to be unused, but apparently used to contain flags related to the speaker volume and the rumble motor (Nintendo games will print a debug message related to it but do not seem to use the value elsewhere).  On all checked remotes, that byte has been 0x40.
  
=== Reading and Writing ===
+
{| style="border-collapse: collapse; padding: 0.2em 0.2em 0.2em 0.2em; text-align: center;"
 +
|- style="background-color: #ddd;"
 +
| style="background-color: #fff;" colspan="2" | &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;" colspan="2" | '''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;" | 0x16
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eee;" | 0x20
 +
| style="border: 1px solid #ccc; padding: 0.2em;" colspan="8" | '''0G X Acceleration'''<span style="color: #777;">&lt;<span style="color: #c00;">9:2</span>&gt;</span>
 +
|- style="background-color: #ded;"
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eee;" | 0x17
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eee;" | 0x21
 +
| style="border: 1px solid #ccc; padding: 0.2em;" colspan="8" | '''0G Y Acceleration'''<span style="color: #777;">&lt;<span style="color: #c00;">9:2</span>&gt;</span>
 +
|- style="background-color: #ded;"
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eee;" | 0x18
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eee;" | 0x22
 +
| style="border: 1px solid #ccc; padding: 0.2em;" colspan="8" | '''0G Z Acceleration'''<span style="color: #777;">&lt;<span style="color: #c00;">9:2</span>&gt;</span>
 +
|- style="background-color: #ded;"
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eee;" | 0x19
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eee;" | 0x23
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddd; color: #888;" colspan="2" | 0
 +
| style="border: 1px solid #ccc; padding: 0.2em;" colspan="2" | '''0G X Acc'''<span style="color: #777;">&lt;<span style="color: #c00;">1:0</span>&gt;</span>
 +
| style="border: 1px solid #ccc; padding: 0.2em;" colspan="2" | '''0G Y Acc'''<span style="color: #777;">&lt;<span style="color: #c00;">1:0</span>&gt;</span>
 +
| style="border: 1px solid #ccc; padding: 0.2em;" colspan="2" | '''0G Z Acc'''<span style="color: #777;">&lt;<span style="color: #c00;">1:0</span>&gt;</span>
 +
|- style="background-color: #ded;"
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eee;" | 0x1A
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eee;" | 0x24
 +
| style="border: 1px solid #ccc; padding: 0.2em;" colspan="8" | '''1G X Acceleration'''<span style="color: #777;">&lt;<span style="color: #c00;">9:2</span>&gt;</span>
 +
|- style="background-color: #ded;"
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eee;" | 0x1B
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eee;" | 0x25
 +
| style="border: 1px solid #ccc; padding: 0.2em;" colspan="8" | '''1G Y Acceleration'''<span style="color: #777;">&lt;<span style="color: #c00;">9:2</span>&gt;</span>
 +
|- style="background-color: #ded;"
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eee;" | 0x1C
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eee;" | 0x26
 +
| style="border: 1px solid #ccc; padding: 0.2em;" colspan="8" | '''1G Z Acceleration'''<span style="color: #777;">&lt;<span style="color: #c00;">9:2</span>&gt;</span>
 +
|- style="background-color: #ded;"
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eee;" | 0x1D
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eee;" | 0x27
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddd; color: #888;" colspan="2" | 0
 +
| style="border: 1px solid #ccc; padding: 0.2em;" colspan="2" | '''1G X Acc'''<span style="color: #777;">&lt;<span style="color: #c00;">1:0</span>&gt;</span>
 +
| style="border: 1px solid #ccc; padding: 0.2em;" colspan="2" | '''1G Y Acc'''<span style="color: #777;">&lt;<span style="color: #c00;">1:0</span>&gt;</span>
 +
| style="border: 1px solid #ccc; padding: 0.2em;" colspan="2" | '''1G Z Acc'''<span style="color: #777;">&lt;<span style="color: #c00;">1:0</span>&gt;</span>
 +
|- style="background-color: #ded;"
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eee;" | 0x1E
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eee;" | 0x28
 +
| style="border: 1px solid #ccc; padding: 0.2em;" | '''Motor'''
 +
| style="border: 1px solid #ccc; padding: 0.2em;" colspan="7" | '''Volume'''
 +
|- style="background-color: #ded;"
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eee;" | 0x1F
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eee;" | 0x29
 +
| style="border: 1px solid #ccc; padding: 0.2em;" colspan="8" | '''Checksum'''
 +
|}
  
We can now read data. Here's the command:
+
At 0x16D0, there is some more unknown data:
  
  (52) 17 FF FF FF FF SS SS
+
  16D0:  00 00 00 FF 11 EE 00 00 33 CC 44 BB 00 00 66 99
 +
16E0:  77 88 00 00 2B 01 E8 13 00 00 00 00 00 00 00 00
 +
16F0:  00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00
  
FF FF FF FF is the offset (big-endian format). SS SS (big-endian format) is the size in bytes.
+
In contrast to the data at 0x0000, this data seems to differ in only a few bytes between different Wii Remotes.
  
It was once thought that this was the correct format:
+
The "User Data" section appears to start with a 0x38-byte header, repeated twice (so 0x70 bytes total).  The header starts with an 8-byte timestamp (from TBL and TBU; 1/60750000ths of a second since January 1 2000).  That is followed by a 34-byte (17 characters, or 16 excluding the null terminator) Unicode name (which does not necessarily match the game's name), followed by the 4-byte game ID, then the application type (the value at 0x80003184). Then there is a checksum (at 0x2f), which is the sum of bytes 0 through 0x2e plus 0x55.  Bytes 0x30 through 0x37 do not seem to be used and were all 0.
  (52) 17 0r 00 FF FF SS SS
 
  
The low bit of the first byte (of the address) is the usual rumble flag. The rumble flag is not part of the address. You should always set this bit to whatever the current rumble state should be. It does not affect the address. If you don't set this bit to the current rumble state you will accidentally change the rumble state just by reading (or writing) the memory.
+
Known memory ranges are listed below.
  
There is only 5.5K of Flash RAM on the Wiimote, which is addressed between 0x0000 and 0x15FF. But there are also internal control registers which can be set, but only some of them can be read. The control registers begin with 0x04. The returned packets store addresses as only two bytes. And addresses wrap around after 0xFFFF.
+
{| 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: #ded;" | '''Start'''
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #dde;" | '''End'''
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddd;" | '''Length'''
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #edd;" | '''Initial Value'''
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddd;" | '''Use'''
 +
|- style="background-color: #ddd;"
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ded;" | 0x0000
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #dde;" | 0x0029
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddd;" | 0x002A
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #edd;" | See above
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddd;" | Calibration values / pre-set data
 +
|- style="background-color: #ddd;"
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ded;" | 0x002A
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #dde;" | 0x0FC9
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddd;" | 0x0FA0
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #edd;" | Zeroed
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddd; color: #888;" | User data / Unknown uses
 +
|- style="background-color: #ddd;"
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ded;" | 0x0FCA
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #dde;" | 0x12B9
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddd;" | 0x02f0
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #edd;" | Zeroed
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddd;" | [[Wiimote/Mii Data|Mii Data]] block 1
 +
|- style="background-color: #ddd;"
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ded;" | 0x12BA
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #dde;" | 0x15A9
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddd;" | 0x02f0
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #edd;" | Zeroed
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddd;" | [[Wiimote/Mii Data|Mii Data]] block 2
 +
|- style="background-color: #ddd;"
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ded;" | 0x15AA
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #dde;" | 0x16CF
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddd;" | 0x0126
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #edd;" | Zeroed
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddd; color: #888;" | Unknown / Unused
 +
|- style="background-color: #ddd;"
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ded;" | 0x16D0
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #dde;" | 0x16FF
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddd;" | 0x0030
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #edd;" | See above
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddd; color: #888;" | Unknown data
 +
|}
  
The responses look like this:
+
The top byte of the address is unused, which means memory is mirrored every 0x10000 bytes. Reading from unused addresses where the low 16 bits are >= 0x1700 will result in error returns.
  
        btns? SE FF FF data
+
== Control Registers ==
        vvvvv vv vv vv v------>
 
(a1) 21 80 00 f0 11 f0 80 6c 8c c7 c2 5d 2e bd 40 00 4e 00 99 80 08 b1                             
 
  
 +
The Wii Remote has several memory mapped register spaces corresponding to different peripherals in it. These include the [[#Speaker | Speaker]], [[#Extension Controllers | Extension Controllers]], and the [[#IR Camera | IR Camera]].
  
E is the error flag. E is 8 if you try to read from bytes that don't exist (are greater than 5.5K, or 0x15FF), and 7 if you try to read from write-only registers, or 0 if no error. S (the high nibble, needs to be shifted right 4 bits) is the size in bytes, minus one, for the current packet. FF is the offset of the current packet (big-endian). Everything else is the data (16 bytes max, if there is an incomplete trailer S is set to something other than 0xF and the data is padded out with zeros.) If you requested more than 16 bytes, then you will receive multiple packets, unless you get an error where E is 8.
+
{{BoxCenter|'''Reminder'''<br />Remember to set bit 2 (0x04) on the first byte of the Output Report, otherwise you'll overwrite EEPROM memory!}}
  
Writing data is as follows:
+
The peripheral to access is selected by the first byte of the address, and the lower 16 bits specify the register to access within that peripheral. The lowest bit of the high byte is ignored, which means every peripheral is mirrored at its address + 0x10000. Known peripherals are listed below:
  
        FF FF FF FF SS data
+
{| style="border-collapse: collapse; padding: 0.2em 0.2em 0.2em 0.2em;"
        vv vv vv vv vv v------>
+
|- style="background-color: #ddd;"
(52) 16 00 00 00 00 10 57 69 69 57 69 6c 6c 52 6f 63 6b 59 6f 75 21 21
+
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ded;" | '''Start'''
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #dde;" | '''End'''
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddd;" | '''Use'''
 +
|- style="background-color: #ddd;"
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ded;" | 0xA20000
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #dde;" | 0xA20009
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddd;" | [[#Speaker | Speaker]] settings
 +
|- style="background-color: #ddd;"
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ded;" | 0xA40000
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #dde;" | 0xA400FF
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddd;" | [[Wiimote/Extension Controllers|Extension Controller]] settings and data
 +
|- style="background-color: #ddd;"
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ded;" | 0xA60000
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #dde;" | 0xA600FF
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddd;" | [[Wiimote/Extension Controllers/Wii Motion Plus|Wii Motion Plus]] settings and data
 +
|- style="background-color: #ddd;"
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ded;" | 0xB00000
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #dde;" | 0xB00033
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddd;" | [[#IR Camera | IR Camera]] settings
 +
|}
  
FF,SS same meaning as reading (you can only write 16 bytes at a time here, so SS is only one byte). 16 bytes of data follow. It seems we get some kind of acknowledge on Input 0x22.
+
Most of these are also mirrored across the high bits of the individual peripheral. For example, the second byte of the address is ignored in the Extension controller address, which means any address of the form 0xA4xx00 will work (as will 0xA5xx00).
  
Notes of unknown validity: The offset is 32 bits at least on the Write Data command. Offsets with the "rumble bit" set will trigger rumble. The high byte of the offset seems to specify a different memory type other than internal Flash. Offsets beginning with 0x04B00000 are used to configure the IR sensor.
+
= Input Features =
 +
The Wii Remote has two input features that are controlled directly by the Broadcom chip: a [[#Accelerometer | Three-Axis Accelerometer]] and 11 [[#Buttons | Buttons]]. Additionally, it has an [[#IR Camera | IR Camera]] with an object tracking processor, and an expansion port that allows for external input features such as those contained in the Nunchuk and the Classic Controller (see [[#Extension Controllers | Extension Controllers]]).
  
 +
== Buttons ==
 +
The Wii Remote has 11 buttons on its front face, and one trigger-style button on the back. Of these, the Power button is special and is treated differently by the Wii Remote. All the other buttons are independantly accessible through a two-byte bitmask which is transmitted first in most Input Reports. A button will report a 1-bit if pressed, or a 0-bit otherwise. By default, these are sent only when the state of any button changes, in [[#0x30: Core Buttons | Data Reporting Mode 0x30]]. However, the Wii Remote may be configured to report the state of the buttons continuously; see [[#Data Reporting | Data Reporting]].
  
 +
=== Core Buttons ===
 +
The Wii Remote has 11 buttons that are used as regular input devices: A, B (trigger), a 4-directional D-Pad, +, -, Home, 1, and 2. These are reported as bits in a two-byte bitmask. These are the assignments, in big-endian order:
 +
{| style="border: 1px solid #bbb; border-collapse: collapse; padding: 0.2em 0.2em 0.2em 0.2em;"
 +
|- style="background-color: #ddd;"
 +
| style="border: 1px solid #ccc; padding: 0.2em;" | '''Bit'''
 +
| style="border: 1px solid #ccc; padding: 0.2em;" | '''Mask'''
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ded;" | '''First Byte'''
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #dde;" | '''Second Byte'''
 +
|- style="background-color: #eee;"
 +
| style="border: 1px solid #ccc; padding: 0.2em;" | 0
 +
| style="border: 1px solid #ccc; padding: 0.2em;" | 0x01
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ded;" | D-Pad Left
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #dde;" | Two
 +
|- style="background-color: #eee;"
 +
| style="border: 1px solid #ccc; padding: 0.2em;" | 1
 +
| style="border: 1px solid #ccc; padding: 0.2em;" | 0x02
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ded;" | D-Pad Right
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #dde;" | One
 +
|- style="background-color: #eee;"
 +
| style="border: 1px solid #ccc; padding: 0.2em;" | 2
 +
| style="border: 1px solid #ccc; padding: 0.2em;" | 0x04
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ded;" | D-Pad Down
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #dde;" | B
 +
|- style="background-color: #eee;"
 +
| style="border: 1px solid #ccc; padding: 0.2em;" | 3
 +
| style="border: 1px solid #ccc; padding: 0.2em;" | 0x08
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ded;" | D-Pad Up
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #dde;" | A
 +
|- style="background-color: #eee;"
 +
| style="border: 1px solid #ccc; padding: 0.2em;" | 4
 +
| style="border: 1px solid #ccc; padding: 0.2em;" | 0x10
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ded;" | Plus
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #dde;" | Minus
 +
|- style="background-color: #eee;"
 +
| style="border: 1px solid #ccc; padding: 0.2em;" | 5
 +
| style="border: 1px solid #ccc; padding: 0.2em;" | 0x20
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ded; color: #888" | Other uses
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #dde; color: #888" | Other uses
 +
|- style="background-color: #eee;"
 +
| style="border: 1px solid #ccc; padding: 0.2em;" | 6
 +
| style="border: 1px solid #ccc; padding: 0.2em;" | 0x40
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ded; color: #888" | Other uses
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #dde; color: #888" | Other uses
 +
|- style="background-color: #eee;"
 +
| style="border: 1px solid #ccc; padding: 0.2em;" | 7
 +
| style="border: 1px solid #ccc; padding: 0.2em;" | 0x80
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ded; color: #888" | Unknown
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #dde;" | Home
 +
|}
  
== Expansion Port ==
+
=== Power Button ===
 +
When the Wii Remote is turned off, pressing the Power button will attempt to wake up the Wii that is synchronized to it. The mechanism for this is unknown, and it is handled entirely within the Wii's bluetooth module. When the Wii Remote is turned on and connected to a host, pressing and holding the Power button for a few seconds will turn the Wii Remote off and request disconnection from the host. The disconnection reason included with the Baseband (ACL) disconnection request indicates that the power button was pressed: REMOTE DEVICE TERMINATED CONNECTION DUE TO POWER OFF (0x15). Another possible value is REMOTE DEVICE TERMINATED CONNECTION DUE TO LOW RESOURCES (0x14), which indicates that the Wii Remote performed a controlled shut down due to a low battery condition.
  
The expansion port on the bottom of the unit is used to connect the remote to auxiliary controllers which augment the input options of the Wiimote. Auxiliary controllers use the Bluetooth interface of the remote to communicate with the Wii base unit, allowing them to be much simpler and cheaper to build. Currently available controllers are the Nunchuk and the Classic controller.
+
=== Sync Button ===
 +
The sync button is hidden under the battery cover. When the Sync button is pressed, the Wii remote will disconnect from whatever it is currently connected to, make itself discoverable, and accept pairing or connection requests for exactly 20 seconds (regardless of how long the button is held down for).
  
The expansion port itself is a ???-style connector with 6 contacts. Two of the contacts are slightly longer, which allows them to make contact first when a plug is inserted. Presumably these contacts carry Vcc and ground, much like standard USB connectors. The remaining 4 contacts most likely carry data, and the small number suggests that a serial interface of some sort is used. The connector is not suitable for charging the remote. More information is needed on the electrical specifications of this connector.
+
The "syncing" of a Wii Remote involves standard Bluetooth pairing. When the Sync button is pressed on the remote, it will accept pairing requests. The required PIN is the hosts's Bluetooth address, backwards (last byte first), in binary (6 bytes). Most current Bluetooth implementations don't deal with this correctly, as they usually consider the PIN to be a regular null-terminated ASCII string (no 00 bytes, etc) and most Bluetooth addresses will contain null bytes. Any further steps that need to be taken after the Wii Remote is paired have not been reverse engineered yet.
  
The status of the expansion port is indicated with report 20H. This report is sent whenever the status of the expansion port changes, or when the computer sends output report 15H to request the status.
+
Once the Wii Remote is synced, when a button is pressed, it will actively seek out its paired host and try to connect to it, instead of the other way around. Establishing a connection can be done on PSM 0x11 for writing and PSM 0x13 for reading using the Bluetooth L2CAP protocol.
  
After an attachment is plugged in or unplugged, no other reports are sent except 20H. The computer needs to send an output request (report 12H) to retrieve new input reports.
+
=== Button Hardware ===
  
The format of input report 20H consists of the report number, then two bytes which are presumably the button state like in other reports, then a status flags byte, then two unknown 00 bytes, then the battery level.
+
The physical hardware of the buttons varies: there are membrane switches and microswitch click buttons. There has been some success soldering wires to the membrane switch contacts and actuating the switch through an external switch. The following table describes the physical hardware for each input.
  
Status byte flags
+
{| class="wikitable" border="1"
{| border="1"
+
|-
! bit
+
! Function
! Value (hex)
+
! Switch type
! Flag
+
! Circuit board surface
 
|-
 
|-
| 0
+
| A
| 0x01
+
| [http://en.wikipedia.org/wiki/Membrane_switch membrane]
| ? 0 ?
+
| top, SW9
 +
|-
 +
| B
 +
| membrane
 +
| bottom, SW8
 +
|-
 +
| -
 +
| [http://en.wikipedia.org/wiki/Microswitch microswitch]
 +
| top, SW10
 +
|-
 +
| Home
 +
| microswitch
 +
| top, SW11
 +
|-
 +
| +
 +
| microswitch
 +
| top, SW5
 
|-
 
|-
 
| 1
 
| 1
| 0x02
+
| membrane
| any attachment plugged in
+
| top, SW7
 
|-
 
|-
 
| 2
 
| 2
| 0x04
+
| membrane
| ? 0 ?
+
| top, SW6
 +
|-
 +
| Up
 +
| membrane
 +
| top, SW4
 
|-
 
|-
| 3
+
| Down
| 0x08
+
| membrane
| continuous output mode
+
| top, SW3
 
|-
 
|-
| 4
+
| Left
| 0x10
+
| membrane
| LED 1
+
| top, SW1
 
|-
 
|-
| 5
+
| Right
| 0x20
+
| membrane
| LED 2
+
| top, SW2
 
|-
 
|-
| 6
+
| Sync
| 0x40
+
|  
| LED 3
+
| bottom, SW12
 
|-
 
|-
| 7
+
| Power
| 0x80
+
|
| LED 4
+
| top, SW13
 +
|}
 +
 
 +
== Accelerometer ==
 +
[[File:Wii-Remote-Accel.jpg|right|thumb|200px|ADXL330 in a Wii remote]][[File:Wiimote_axis2.png|right|thumb|200px|Coordinate system used by Wii Remote]]
 +
The Wii Remote includes a three-axis linear accelerometer located on the top suface of the circuit board, slightly left of the large A button.  The integrated circuit is the [http://www.seekic.com/icdata/%20ADXL330.html ADXL330] ([http://www.analog.com/UploadedFiles/Data_Sheets/ADXL330.pdf data sheet]), manufactured by Analog Devices.  This device is physically rated to measure accelerations over a range of at least +/- 3g with 10% sensitivity.
 +
 
 +
Since the accelerometer actually measures the force exerted by a set of small proof masses inside of it with respect to its enclosure, the accelerometer measures linear acceleration in a free fall frame of reference. If the Wii Remote is in free fall, it will report zero acceleration. At rest, it will report an upward acceleration (+Z, when horizontal) equal to the acceleration due to gravity, g (approximately 9.8 m/s²) but in the opposite direction. This fact can be used to derive tilt from the acceleration outputs when the Wii Remote is reasonably still.
 +
 
 +
=== Normal Accelerometer Reporting ===
 +
In all [[#Data Reporting | Data Reporting Modes]] which include Accelerometer data except for [[#0x3e / 0x3f: Interleaved Core Buttons and Accelerometer with 36 IR bytes | mode 0x3e/0x3f]], the accelerometer data is reported as three consecutive bytes:
 +
 
 +
(a1) RR BB BB XX YY ZZ [...]
 +
 
 +
XX, YY, and ZZ are unsigned bytes representing the acceleration in each of the three axis, where zero acceleration is approximately 0x80. The coordinate system is shown in the diagram above (note that this is different from the coordinate system used by GlovePIE). Additionally, the BB BB [[#Buttons | Buttons]] bytes also include the LSBs of the acceleration values in the unused bits, according to the following table:
 +
 
 +
{| style="border-collapse: collapse; padding: 0.2em 0.2em 0.2em 0.2em; text-align: center;"
 +
|- style="background-color: #ddd;"
 +
| style="background-color: #fff;" colspan="2" | &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:2em;" | '''7'''
 +
| style="border: 1px solid #ccc; padding: 0.2em; width:2em;" | '''6'''
 +
| style="border: 1px solid #ccc; padding: 0.2em; width:2em;" | '''5'''
 +
| style="border: 1px solid #ccc; padding: 0.2em; width:2em;" | '''4'''
 +
| style="border: 1px solid #ccc; padding: 0.2em; width:2em;" | '''3'''
 +
| style="border: 1px solid #ccc; padding: 0.2em; width:2em;" | '''2'''
 +
| style="border: 1px solid #ccc; padding: 0.2em; width:2em;" | '''1'''
 +
| style="border: 1px solid #ccc; padding: 0.2em; width:2em;" | '''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; background-color: #ddd;" | &nbsp;
 +
| style="border: 1px solid #ccc; padding: 0.2em;" colspan="2" | '''X'''<span style="color: #777;">&lt;<span style="color: #c00;">1:0</span>&gt;</span>
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddd;" colspan="5" | &nbsp;
 +
|- style="background-color: #ded;"
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eee;" | 1
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddd;" | &nbsp;
 +
| style="border: 1px solid #ccc; padding: 0.2em;" | '''Z'''<span style="color: #777;">&lt;<span style="color: #c00;">1</span>&gt;</span>
 +
| style="border: 1px solid #ccc; padding: 0.2em;" | '''Y'''<span style="color: #777;">&lt;<span style="color: #c00;">1</span>&gt;</span>
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddd;" colspan="5" | &nbsp;
 +
|}
 +
 
 +
Note that X has 10 bits of precision, while Y and Z only have 9. For consistency, they are assumed all to have a 10-bit range and the LSB is always set to zero for Y and Z.
 +
 
 +
=== Interleaved Accelerometer Reporting ===
 +
In [[#0x3e / 0x3f: Interleaved Core Buttons and Accelerometer with 36 IR bytes | Data Reporting Mode 0x3e/0x3f]], the accelerometer data is spread over two reports:
 +
 
 +
(a1) 3e BB BB XX [...]
 +
(a1) 3f BB BB YY [...]
 +
 
 +
In this mode, the LSBs are not available. Instead, X and Y acceleration is reported as a single byte, and the Z value is encoded in the unused bits of the BB BB [[#Buttons | Buttons]] data as follows:
 +
 
 +
{| style="border-collapse: collapse; padding: 0.2em 0.2em 0.2em 0.2em; text-align: center;"
 +
|- style="background-color: #ddd;"
 +
| style="background-color: #fff;" colspan="2" | &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;" | '''Report ID'''
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddd;" | '''Byte'''
 +
| style="border: 1px solid #ccc; padding: 0.2em; width:2em;" | '''7'''
 +
| style="border: 1px solid #ccc; padding: 0.2em; width:2em;" | '''6'''
 +
| style="border: 1px solid #ccc; padding: 0.2em; width:2em;" | '''5'''
 +
| style="border: 1px solid #ccc; padding: 0.2em; width:2em;" | '''4'''
 +
| style="border: 1px solid #ccc; padding: 0.2em; width:2em;" | '''3'''
 +
| style="border: 1px solid #ccc; padding: 0.2em; width:2em;" | '''2'''
 +
| style="border: 1px solid #ccc; padding: 0.2em; width:2em;" | '''1'''
 +
| style="border: 1px solid #ccc; padding: 0.2em; width:2em;" | '''0'''
 +
|- style="background-color: #ded;"
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eee;" | 0x3e
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eee;" | 0
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddd;" | &nbsp;
 +
| style="border: 1px solid #ccc; padding: 0.2em;" colspan="2" | '''Z'''<span style="color: #777;">&lt;<span style="color: #c00;">5:4</span>&gt;</span>
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddd;" colspan="5" | &nbsp;
 +
|- style="background-color: #ded;"
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eee;" | 0x3e
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eee;" | 1
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddd;" | &nbsp;
 +
| style="border: 1px solid #ccc; padding: 0.2em;" colspan="2" | '''Z'''<span style="color: #777;">&lt;<span style="color: #c00;">7:6</span>&gt;</span>
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddd;" colspan="5" | &nbsp;
 +
|- style="background-color: #ded;"
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eee;" | 0x3f
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eee;" | 0
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddd;" | &nbsp;
 +
| style="border: 1px solid #ccc; padding: 0.2em;" colspan="2" | '''Z'''<span style="color: #777;">&lt;<span style="color: #c00;">1:0</span>&gt;</span>
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddd;" colspan="5" | &nbsp;
 +
|- style="background-color: #ded;"
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eee;" | 0x3f
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eee;" | 1
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddd;" | &nbsp;
 +
| style="border: 1px solid #ccc; padding: 0.2em;" colspan="2" | '''Z'''<span style="color: #777;">&lt;<span style="color: #c00;">3:2</span>&gt;</span>
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddd;" colspan="5" | &nbsp;
 +
|}
 +
 
 +
== IR Camera ==
 +
[[File:Wii-Remote-Camera.jpg|200px|thumb|right|Wii remote camera]] The Wii Remote includes a 128x96 monochrome camera with built-in image processing. The camera looks through an infrared pass filter in the remote's plastic casing. The camera's built-in image processing is capable of tracking up to 4 moving objects, and these data are the only data available to the host. Raw pixel data is not available to the host, so the camera cannot be used to take a conventional picture. The built-in processor uses 8x subpixel analysis to provide 1024x768 resolution for the tracked points. The Sensor Bar that comes with the Wii includes two IR LED clusters at each end, which are tracked by the Wii Remote to provide pointing information. The distance between the centers of the LED clusters is 20 cm (as measured on one unit).
 +
 
 +
The IR Camera is enabled by setting bit 2 on output reports 0x13 and 0x1a:
 +
(a2) 13 04
 +
(a2) 1a 04
 +
 
 +
The first enables a 24MHz pixel clock on pin 7 of the camera.  The second pulls pin 4 low - probably an active-low enable.
 +
 
 +
=== Mechanical Characteristics ===
 +
 
 +
The camera component is mounted on the bottom surface of the circuit board.  The camera module itself is mounted in a socket perpendicular to the circuit board; to remove just the camera module, no desoldering is required.  The process is as follows:
 +
 
 +
First, orient the camera so that you are looking into the lens with the PCB horizontal and below the lens..  There are four metal clips, two on each of the
 +
vertical sides of the socket.  Use something tiny to slide between each metal clip and the camera module: maybe wire wrap wire?  Then look at the back of the camera module, opposite the lens.  There is a small rectangular hole in the middle of each vertical side of the socket.  Use a pin or something to pry/press the camera module out.
 +
 
 +
Once the camera module is free of its socket, it may be further disassembled by gently prising up the tiny PCB with gold contacts; this is gently glued to the module's structure, but will come loose without damage.  At this point you have three pieces: the camera socket, still attached to the Wiimote PCB, the camera module housing, complete with lens and dichroic filter (of unknown optical properties), and a tiny PCB with the camera chip and eight gold contacts on the bottom.
 +
 
 +
=== Optical Characteristics ===
 +
 
 +
The IR camera has an effective field of view is about 33 degrees horizontally and 23 degrees vertically (as measured on one unit). With the IR-pass filter intact, 940nm sources are detected with approximately twice the intensity of equivalent 850nm sources, but are not resolved as well at close distances. If the filter is removed, it can track any bright object.  However, the IR filter referred to here is not only the dark plastic window of the wiimote but also a teensy slab of dichroic-coated glass inside the camera module.  One may operate the wiimote having installed neither, one or the other, or both filters.
 +
 
 +
=== Initialization ===
 +
 
 +
{{BoxCenter|'''Reminder'''<br />Remember to set bit 2 (0x04) on the first byte of the Output Reports to write to registers!}}
 +
 
 +
The following procedure should be followed to turn on the IR Camera:
 +
 
 +
#Enable IR Camera (Send 0x04 to Output Report 0x13)
 +
#Enable IR Camera 2 (Send 0x04 to Output Report 0x1a)
 +
#Write 0x08 to register 0xb00030
 +
#Write Sensitivity Block 1 to registers at 0xb00000
 +
#Write Sensitivity Block 2 to registers at 0xb0001a
 +
#Write Mode Number to register 0xb00033
 +
#Write 0x08 to register 0xb00030 (again)
 +
 
 +
After these steps, the Wii Remote will be in one of 3 states: IR camera on but not taking data, IR camera on and taking data and half sensitivity, IR camera on and taking data at full sensitivity.  Which state you end up in appears to be pretty much random.  Repeat the steps until you're in the desired state. To avoid the random state put a delay of at least 50ms between every single byte transmission.
 +
 
 +
The Wii preforms these steps slightly different, differences in bold:
 +
#Enable IR Pixel Clock (send '''0x06''' to Output Report 0x13)
 +
#Enable IR Logic (send '''0x06''' to Output Report 0x1A)
 +
#Write '''0x01''' to register 0xb00030
 +
#Write Sensitivity Block 1 to registers at 0xb00000
 +
#Write Sensitivity Block 2 to registers at 0xb0001a
 +
#Write Mode Number to register 0xb00033
 +
#Write 0x08 to register 0xb00030 (again)
 +
 
 +
Adding bit 0x02 to reports 0x13 and 0x1a is a request for acknowledgement (if set, wiimote will respond with report 0x22).
 +
 
 +
=== Sensitivity Settings ===
 +
Sensitivity is controlled by two configuration blocks, 9 bytes and 2 bytes long. The following settings are known to work:
 +
 
 +
{| style="border-collapse: collapse; padding: 0.2em 0.2em 0.2em 0.2em;"
 +
|- style="text-align: center;"
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ded;" | '''Block 1'''
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #dde;" | '''Block 2'''
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddd;" | '''Notes'''
 +
|- style="text-align: center;"
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #efe; font-family: monospace;" | 00 00 00 00 00 00 90 00 C0
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eef; font-family: monospace;" | 40 00
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eee; text-align: left;" | Suggested by [[User:Marcan | Marcan]]
 +
|- style="text-align: center;"
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #efe; font-family: monospace;" | 00 00 00 00 00 00 FF 00 0C
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eef; font-family: monospace;" | 00 00
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eee; text-align: left;" | Suggested by Kestrel (max sensitivity)
 +
|- style="text-align: center;"
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #efe; font-family: monospace;" | 00 00 00 00 00 00 90 00 41
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eef; font-family: monospace;" | 40 00
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eee; text-align: left;" | Suggested by inio (high sensitivity)
 +
|- style="text-align: center;"
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #efe; font-family: monospace;" | 02 00 00 71 01 00 64 00 fe
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eef; font-family: monospace;" | fd 05
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eee; text-align: left;" | Wii level 1
 +
|- style="text-align: center;"
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #efe; font-family: monospace;" | 02 00 00 71 01 00 96 00 b4
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eef; font-family: monospace;" | b3 04
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eee; text-align: left;" | Wii level 2
 +
|- style="text-align: center;"
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #efe; font-family: monospace;" | 02 00 00 71 01 00 aa 00 64
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eef; font-family: monospace;" | 63 03
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eee; text-align: left;" | Wii level 3 (Suggested by Cliff)
 +
|- style="text-align: center;"
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #efe; font-family: monospace;" | 02 00 00 71 01 00 c8 00 36
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eef; font-family: monospace;" | 35 03
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eee; text-align: left;" | Wii level 4
 +
|- style="text-align: center;"
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #efe; font-family: monospace;" | 07 00 00 71 01 00 72 00 20
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eef; font-family: monospace;" | 1f 03
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eee; text-align: left;" | Wii level 5
 +
|}
 +
 
 +
The last byte of Block 1 determines the intensity sensitivity, with increasing values reducing the sensitivity. Both bytes of Block 2 must be zero for the full sensitivity range to be available. Setting the sensitivity as high as possible, without unwanted light being tracked, is recommended to achieve the highest subpixel resolution. As the sensitivity is reduced, the subpixel resolution also reduces, approaching the true sensor resolution of 128x96.
 +
 
 +
=== Data Formats ===
 +
The IR Camera can return different sets of data describing the objects it is tracking. When the IR camera identifies an object, it assigns it to the first available object slot. If an object moves out of view, its slot is marked as empty (returns 0xFF data), but other objects retain their slots. For example, if the camera is tracking two objects and the first moves out of view, the data returned will be [empty, second object, empty, empty]. With more than four objects visible, the camera is prone to rapidly switching between some of them. This could allow perception of more than four objects, at a reduced response speed and reliability.
 +
 
 +
{| style="border-collapse: collapse; padding: 0.2em 0.2em 0.2em 0.2em;"
 +
|- style="background-color: #dde; text-align: center;"
 +
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center;" | '''Mode'''
 +
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center; background-color: #ded;" | '''Mode Number'''
 +
|- style="background-color: #eef; text-align: center;"
 +
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center;" | [[#Basic Mode | Basic]]
 +
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center; background-color: #efe;" | 1
 +
|- style="background-color: #eef; text-align: center;"
 +
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center;" | [[#Extended Mode | Extended]]
 +
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center; background-color: #efe;" | 3
 +
|- style="background-color: #eef; text-align: center;"
 +
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center;" | [[#Full Mode | Full]]
 +
| style="border: 1px solid #ccc; padding: 0.2em; text-align: center; background-color: #efe;" | 5
 +
|}
 +
 
 +
The data format MUST match the number of bytes available in the [[#Data Reporting | Reporting Mode]] selected. Even choosing a mode with space for more bytes than necessary will not work, it has to be an exact match.
 +
 
 +
==== Basic Mode ====
 +
In Basic Mode, the IR Camera returns 10 bytes of data corresponding to the X and Y locations of each of the four dots. Each location is encoded in 10 bits and has a range of 0-1023 for the X dimension, and 0-767 for the Y dimension. Each pair of dots is packed into 5 bytes, and two of these are transmitted for a total of 4 dots and 10 bytes.
 +
 
 +
This is the data format for a pair of objects:
 +
{| style="border-collapse: collapse; padding: 0.2em 0.2em 0.2em 0.2em; text-align: center;"
 +
|- 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:2em;" | '''7'''
 +
| style="border: 1px solid #ccc; padding: 0.2em; width:2em;" | '''6'''
 +
| style="border: 1px solid #ccc; padding: 0.2em; width:2em;" | '''5'''
 +
| style="border: 1px solid #ccc; padding: 0.2em; width:2em;" | '''4'''
 +
| style="border: 1px solid #ccc; padding: 0.2em; width:2em;" | '''3'''
 +
| style="border: 1px solid #ccc; padding: 0.2em; width:2em;" | '''2'''
 +
| style="border: 1px solid #ccc; padding: 0.2em; width:2em;" | '''1'''
 +
| style="border: 1px solid #ccc; padding: 0.2em; width:2em;" | '''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" | '''X1'''<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" | '''Y1'''<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="2" | '''Y1'''<span style="color: #777;">&lt;<span style="color: #c00;">9:8</span>&gt;</span>
 +
| style="border: 1px solid #ccc; padding: 0.2em;" colspan="2" | '''X1'''<span style="color: #777;">&lt;<span style="color: #c00;">9:8</span>&gt;</span>
 +
| style="border: 1px solid #ccc; padding: 0.2em;" colspan="2" | '''Y2'''<span style="color: #777;">&lt;<span style="color: #c00;">9:8</span>&gt;</span>
 +
| style="border: 1px solid #ccc; padding: 0.2em;" colspan="2" | '''X2'''<span style="color: #777;">&lt;<span style="color: #c00;">9:8</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" | '''X2'''<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" | '''Y2'''<span style="color: #777;">&lt;<span style="color: #c00;">7:0</span>&gt;</span>
 +
|}
 +
 
 +
==== Extended Mode ====
 +
In Extended Mode, the IR Camera returns the same data as it does in Basic Mode, plus a rough size value for each object. The data is returned as 12 bytes, three bytes per object. Size has a range of 0-15.
 +
 
 +
This is the data format for each object:
 +
{| style="border-collapse: collapse; padding: 0.2em 0.2em 0.2em 0.2em; text-align: center;"
 +
|- 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:2em;" | '''7'''
 +
| style="border: 1px solid #ccc; padding: 0.2em; width:2em;" | '''6'''
 +
| style="border: 1px solid #ccc; padding: 0.2em; width:2em;" | '''5'''
 +
| style="border: 1px solid #ccc; padding: 0.2em; width:2em;" | '''4'''
 +
| style="border: 1px solid #ccc; padding: 0.2em; width:2em;" | '''3'''
 +
| style="border: 1px solid #ccc; padding: 0.2em; width:2em;" | '''2'''
 +
| style="border: 1px solid #ccc; padding: 0.2em; width:2em;" | '''1'''
 +
| style="border: 1px solid #ccc; padding: 0.2em; width:2em;" | '''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" | '''X'''<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" | '''Y'''<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="2" | '''Y'''<span style="color: #777;">&lt;<span style="color: #c00;">9:8</span>&gt;</span>
 +
| style="border: 1px solid #ccc; padding: 0.2em;" colspan="2" | '''X'''<span style="color: #777;">&lt;<span style="color: #c00;">9:8</span>&gt;</span>
 +
| style="border: 1px solid #ccc; padding: 0.2em;" colspan="4" | '''S'''<span style="color: #777;">&lt;<span style="color: #c00;">3:0</span>&gt;</span>
 +
|}
 +
 
 +
==== Full Mode ====
 +
In Full Mode, the IR Camera returns even more data, 9 bytes per object for a total of 36 bytes for all four. The data is split up between two input reports of 18 bytes each (see [[#0x3e / 0x3f: Interleaved Core Buttons and Accelerometer with 36 IR bytes | Data Reporting Mode 0x3e/0x3f]]). The first three bytes of each object are the same as the extended mode, and are followed by the bounding box of the pixels included in the blob along with a deeper intensity value.  The data format of each object is:
 +
 
 +
{| style="border-collapse: collapse; padding: 0.2em 0.2em 0.2em 0.2em; text-align: center;"
 +
|- 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:2em;" | '''7'''
 +
| style="border: 1px solid #ccc; padding: 0.2em; width:2em;" | '''6'''
 +
| style="border: 1px solid #ccc; padding: 0.2em; width:2em;" | '''5'''
 +
| style="border: 1px solid #ccc; padding: 0.2em; width:2em;" | '''4'''
 +
| style="border: 1px solid #ccc; padding: 0.2em; width:2em;" | '''3'''
 +
| style="border: 1px solid #ccc; padding: 0.2em; width:2em;" | '''2'''
 +
| style="border: 1px solid #ccc; padding: 0.2em; width:2em;" | '''1'''
 +
| style="border: 1px solid #ccc; padding: 0.2em; width:2em;" | '''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" | '''X'''<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" | '''Y'''<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="2" | '''Y'''<span style="color: #777;">&lt;<span style="color: #c00;">9:8</span>&gt;</span>
 +
| style="border: 1px solid #ccc; padding: 0.2em;" colspan="2" | '''X'''<span style="color: #777;">&lt;<span style="color: #c00;">9:8</span>&gt;</span>
 +
| style="border: 1px solid #ccc; padding: 0.2em;" colspan="4" | '''S'''<span style="color: #777;">&lt;<span style="color: #c00;">3: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; background-color: #ddd;" | 0
 +
| style="border: 1px solid #ccc; padding: 0.2em;" colspan="7" | '''X min'''<span style="color: #777;">&lt;<span style="color: #c00;">6: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;" | 0
 +
| style="border: 1px solid #ccc; padding: 0.2em;" colspan="7" | '''Y min'''<span style="color: #777;">&lt;<span style="color: #c00;">6: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;" | 0
 +
| style="border: 1px solid #ccc; padding: 0.2em;" colspan="7" | '''X max'''<span style="color: #777;">&lt;<span style="color: #c00;">6:0</span>&gt;</span>
 +
|- style="background-color: #ded;"
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eee;" | 6
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddd;" | 0
 +
| style="border: 1px solid #ccc; padding: 0.2em;" colspan="7" | '''Y max'''<span style="color: #777;">&lt;<span style="color: #c00;">6:0</span>&gt;</span>
 +
|- style="background-color: #ded;"
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eee;" | 7
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddd;"  colspan="8" | 0
 +
|- style="background-color: #ded;"
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #eee;" | 8
 +
| style="border: 1px solid #ccc; padding: 0.2em;" colspan="8" | '''Intensity'''<span style="color: #777;">&lt;<span style="color: #c00;">7:0</span>&gt;</span>
 +
|}
 +
 
 +
= Feedback Features =
 +
The Wii Remote sports three feedback features: [[#Player LEDs | Player LEDs]], [[#Rumble | Rumble]], and the [[#Speaker | Speaker]].
 +
 
 +
== Player LEDs ==
 +
[[File:Wii-Remote-LEDs.jpg|200px|thumb|right|Wii remote player LEDs]]There are four blue LEDs on the front face of the Wii Remote. During discovery and before initialization, these LEDs blink at a fixed rate. The number of blinking LEDs is proportional to the battery voltage, indicating battery charge (all four are lit for newly charged batteries, and only the first is lit if the batteries are low and should be replaced).
 +
 
 +
During gameplay with the Wii, one LED is lit to indicate the player number assigned to the Wii Remote. However, the LEDs are independently controllable by the host, and can be set to display any pattern. They can also be modulated at a moderately high speed, enabling some brightness control at the cost of a lot of Bluetooth bandwidth. Sigma-delta modulation works reasonably well for this.
 +
 
 +
The LEDs can be controlled by sending a report with ID 0x11:
 +
 
 +
(a2) 11 LL
 +
 
 +
The high nybble of LL controls the four LEDs. Bit 4 of LL controls the first LED, and bit 7 controls the last:
 +
 
 +
{| style="border-collapse: collapse; padding: 0.2em 0.2em 0.2em 0.2em;"
 +
|- style="background-color: #ddd; text-align: center;"
 +
| style="border: 1px solid #ccc; padding: 0.2em;" | '''Bit'''
 +
| style="border: 1px solid #ccc; padding: 0.2em;" | '''Mask'''
 +
| style="border: 1px solid #ccc; padding: 0.2em; background-color: #ddf;" | '''LEDs'''
 +
|- style="background-color: #eee; text-align: center;"
 +
| style="border: 1px solid #ccc; padding: 0.2em;" | 4
 +
| style="border: 1px solid #ccc; padding: 0.2em;" | 0x10
 +
| style="border: 1px solid #ccc; padding: 0.2em;" |
 +
{| style="border-collapse: collapse; background-color: #fff; border: 1px solid #999; text-align: center; line-height: 0.5; margin-top: +1px; margin-left: +1px;"
 +
|- style="color: #999;"
 +
| style="width: 1.6em; padding-bottom: 0.2em;" | ·<br /><span style="color: #37f">■</span>
 +
| style="width: 1.6em; padding-bottom: 0.2em;" | ··<br /><span style="color: #444">■</span>
 +
| style="width: 1.6em; padding-bottom: 0.2em;" | ···<br /><span style="color: #444">■</span>
 +
| style="width: 1.6em; padding-bottom: 0.2em;" | ····<br /><span style="color: #444">■</span>
 +
|}
 +
|- style="background-color: #eee; text-align: center;"
 +
| style="border: 1px solid #ccc; padding: 0.2em;" | 5
 +
| style="border: 1px solid #ccc; padding: 0.2em;" | 0x20
 +
| style="border: 1px solid #ccc; padding: 0.2em;" |
 +
{| style="border-collapse: collapse; background-color: #fff; border: 1px solid #999; text-align: center; line-height: 0.5; margin-top: +1px; margin-left: +1px;"
 +
|- style="color: #999;"
 +
| style="width: 1.6em; padding-bottom: 0.2em;" | ·<br /><span style="color: #444">■</span>
 +
| style="width: 1.6em; padding-bottom: 0.2em;" | ··<br /><span style="color: #37f">■</span>
 +
| style="width: 1.6em; padding-bottom: 0.2em;" | ···<br /><span style="color: #444">■</span>
 +
| style="width: 1.6em; padding-bottom: 0.2em;" | ····<br /><span style="color: #444">■</span>
 
|}
 
|}
 +
|- style="background-color: #eee; text-align: center;"
 +
| style="border: 1px solid #ccc; padding: 0.2em;" | 6
 +
| style="border: 1px solid #ccc; padding: 0.2em;" | 0x40
 +
| style="border: 1px solid #ccc; padding: 0.2em;" |
 +
{| style="border-collapse: collapse; background-color: #fff; border: 1px solid #999; text-align: center; line-height: 0.5; margin-top: +1px; margin-left: +1px;"
 +
|- style="color: #999;"
 +
| style="width: 1.6em; padding-bottom: 0.2em;" | ·<br /><span style="color: #444">■</span>
 +
| style="width: 1.6em; padding-bottom: 0.2em;" | ··<br /><span style="color: #444">■</span>
 +
| style="width: 1.6em; padding-bottom: 0.2em;" | ···<br /><span style="color: #37f">■</span>
 +
| style="width: 1.6em; padding-bottom: 0.2em;" | ····<br /><span style="color: #444">■</span>
 +
|}
 +
|- style="background-color: #eee; text-align: center;"
 +
| style="border: 1px solid #ccc; padding: 0.2em;" | 7
 +
| style="border: 1px solid #ccc; padding: 0.2em;" | 0x80
 +
| style="border: 1px solid #ccc; padding: 0.2em;" |
 +
{| style="border-collapse: collapse; background-color: #fff; border: 1px solid #999; text-align: center; line-height: 0.5; margin-top: +1px; margin-left: +1px;"
 +
|- style="color: #999;"
 +
| style="width: 1.6em; padding-bottom: 0.2em;" | ·<br /><span style="color: #444">■</span>
 +
| style="width: 1.6em; padding-bottom: 0.2em;" | ··<br /><span style="color: #444">■</span>
 +
| style="width: 1.6em; padding-bottom: 0.2em;" | ···<br /><span style="color: #444">■</span>
 +
| style="width: 1.6em; padding-bottom: 0.2em;" | ····<br /><span style="color: #37f">■</span>
 +
|}
 +
|}
 +
 +
Turning off all LEDs for an extended period of time is discouraged, as it might lead the user to believe the Wii Remote is turned off and disconnected, when in fact it is active.
 +
 +
The LEDs are surface mount parts, driven at 2.66 VDC.
 +
 +
== Rumble ==
 +
[[File:Wii_Remote_Rumble.jpg|200px|thumb|right|Wii remote rumble motor]]The Wii Remote includes a rumble feature, which is implemented as a small motor attached to an off-center weight. It will cause the controller to vibrate when activated.
 +
 +
The rumble motor is primarilly controlled via the report with ID 0x10:
 +
 +
(a2) 10 RR
 +
 +
Setting RR to 1 enables rumble, and setting it to 0 disables rumble.  However, note that the rumble motor can be turned on or off through any of the Output Reports, not just 0x10. Setting the LSB (bit 0) of the first byte of any output report will activate the rumble motor, and unsetting it will deactivate it. For example, the following report will turn the rumble motor on:
 +
 +
(a2) 11 01
 +
 +
However, this will also have the side-effect of turning off all LEDs. The rumble bit needs to be set properly with every single report sent, to avoid inadvertently turning the rumble motor off.
 +
 +
Different photos of the rumble motor hardware show different markings. One example is ''SEM 8728DA''. The Wii Remote drives it at 3.3 VDC and it draws 35 mA. It would be reasonable to think that the rumble motor could be removed and the motor replaced with another device with equal voltage and equal or less current draw.
 +
 +
== Speaker ==
 +
 +
The Wii Remote has a small low-quality 21mm piezo-electric speaker, used for short sound effects during gameplay. The sound is streamed directly from the host, and the speaker has some adjustable parameters.
 +
 +
The speaker is controlled by using three output reports, together with a section of the register address space of the Wii Remote.
 +
 +
Report 0x14 is used to enable or disable the speaker. Setting bit 2 will enable the speaker, and clearing it will disable it. For example, to enable the speaker, send:
 +
(a2) 14 04
 +
 +
Report 0x19 is used to mute or unmute the speaker, and works identically to report 0x14. 0x04 will mute the speaker, and 0x00 will unmute it.
 +
 +
Report 0x18 is used to send speaker data. 1-20 bytes may be sent at once:
 +
(a2) 18 LL DD DD DD DD DD DD DD DD DD DD DD DD DD DD DD DD DD DD DD DD
 +
 +
LL specifies the data length, shifted left by three bits. The DD bytes are the speaker data. To fullfill the report length requirements, the data must be padded if it is less than 20 bytes long. Sound data must be sent at the proper rate.
 +
 +
=== Initialization Sequence ===
 +
 +
{{BoxCenter|'''Reminder'''<br />Remember to set bit 2 (0x04) on the first byte of the Output Reports to write to registers!}}
 +
 +
The following sequence will initialize the speaker:
 +
 +
#Enable speaker (Send 0x04 to Output Report 0x14)
 +
#Mute speaker (Send 0x04 to Output Report 0x19)
 +
#Write 0x01 to register 0xa20009
 +
#Write 0x08 to register 0xa20001
 +
#Write [[#Speaker Configuration | 7-byte configuration]] to registers 0xa20001-0xa20008
 +
#Write 0x01 to register 0xa20008
 +
#Unmute speaker (Send 0x00 to Output Report 0x19)
 +
 +
=== Speaker Configuration ===
 +
 +
7 bytes control the speaker settings, including volume. The full purpose of these bytes is not known, but the following values seem to produce some sound:
 +
 +
00 FF RR RR VV 00 00
 +
 +
RR RR specify the sample rate (little-endian format), using the following formulae:
 +
 +
pcm_sample_rate = 12000000 / rate_value
 +
adpcm_sample_rate = 6000000 / rate_value
 +
 +
The standard value is 0x7d0, for 3000Hz 4-bit PCM
  
For example, when the classic controller or nunchuk is plugged in, input report 20H is sent:
+
FF configures the data format. Setting it to 0x40 configures the speaker to use signed 8-bit PCM, while setting it to 0x00 configures it to use 4-bit Yamaha ADPCM. VV specifies the volume, which has a range of 0x00-0xFF for 8-bit mode, and 0x00-0x40 for 4-bit mode.
  
  (a1) 20 00 00 02 00 00 C0
+
This configuration can be used to play 4-bit ADPCM sound at 3000Hz:
 +
  00 00 D0 07 40 00 00
  
The 02H indicates the classic controller or nunchuk is plugged in. The C0H is the battery level, and has nothing to do with the nunchuk or classic controller. It will slowly decrease the more you use up the batteries, but sometimes it increases.
+
This configuration can be used to play 8-bit PCM sound at 1500Hz sample rate:
 +
00 40 40 1f 40 00 00
  
When the classic controller or nunchuk is unplugged, input report 20H is sent again, but with 00 instead of the 02:
+
It looks like the best compromise between sample rate and slow bluetooth chipsets / drivers is playing 8-bit PCM at 2000Hz, so a new 20-byte chunk of audio data is sent every 10 milliseconds.
 +
00 40 70 17 60 00 00
  
(a1) 20 00 00 00 00 00 C0
+
=== Sound Data Format ===
  
== Batteries ==
+
The Wii Remote can use multiple sound formats at multiple sampling rates. PC drivers currently seem unable to keep up with the higher rates.
  
You can read the battery level with report 0x20. It is received when something is plugged in, or unplugged from, the expansion port. Or you can request it by sending report 0x15 with the payload set to anything without bit 1 (rumble) set.
+
The 4-bit ADPCM is Yamaha ADPCM (for example, as implemented in ffmpeg).
  
(a1) 20 00 00 02 00 00 C0
+
8-bit signed PCM mode works, but when in 8-bit mode the sampling frequency must be made so low that the audio quality is pretty bad.
  
The 0xC0 at the end is the battery level (in this case, for the brand new alkaline batteries that come with it).
+
= Extension Controllers =
 +
''[[Wiimote/Extension Controllers]]''
  
If you receive this message unsolicited, you need to send report 0x12 with the report number again before you will receive more data.
+
= Notes =
  
== Unknown behavior / Random findings ==
+
== See Also ==
 +
* [[Wiimote/Firmware|Disassembled Firmware]] shows raw firmware dumps from several wiimotes.
 +
* [[Wiimote/Extension Controllers|Extension Controllers]] explains the protocol used by the wiimote extensions.
 +
* [[Wiimote/Motion analysis|Motion analysis]] gives hints how to implement accelerometer parsers.
 +
* [[Wiimote/Pointing|Pointing]] shows how to create a pointing device with IR data.
 +
* [[Wiimote/Library|Library]] lists several wiimote library implementations.
 +
* [[Wiimote/Mii Data|Mii Data]] shows raw wiimote data dumps.
  
Sending a byte to ID 0x15 returns the Extension status report (at 0x20, see above), besides setting rumble. Seems everything sets rumble if bit 0 of the first byte is set.
+
== Acknowledgements ==
 +
''Some of the information here is based on the documentation at Wiili''
  
The third byte in the extension status report (00/02 above) can take other values. It seems only that bit is the status, other bits can take on other values depending on the bytes sent to 0x13 and 0x14. For example, sending 0x04 to ID 0x13 makes the values in byte 3 be 08/0a instead of 00/02.
+
[[Category:Peripherals]]

Latest revision as of 12:16, 16 October 2022

Wii remote circuit board, top surface
Wii remote circuit board, bottom surface
This article is a technical guide to the Wii Remote. For a high-level overview of the Wii Remote, see the Wikipedia entry.
Broadcom BCM2042 in a Wii remote

The Wii Remote (informally known as the Wiimote) is the Wii's main input device. It is a wireless device, using standard Bluetooth technology to communicate with the Wii. It is built around a Broadcom BCM2042 bluetooth System-on-a-chip, and contains multiple peripherals that provide data to it, as well as an expansion port for external add-ons. The Wii Remote uses (and, at times, abuses) the standard Bluetooth HID protocol to communicate with the host, which is directly based upon the USB HID standard. As such, it will appear as a standard input device to any Bluetooth host. However, the Wii Remote does not make use of the standard data types and HID descriptor, and only describes its report format length, leaving the actual contents undefined, which makes it useless with standard HID drivers (but some Wiimote Drivers exist). The Wii Remote actually uses a fairly complex set of operations, transmitted through HID Output reports, and returns a number of different data packets through its Input reports, which contain the data from its peripherals.

Summary

Reverse engineering and documenting all of the Wii Remote's features is a work in progress. Here are the known features and their status:

Bluetooth Communication ButtonGreen.png Connecting to the Wii Remote and listening for connections works.
Core Buttons ButtonGreen.png All working.
Accelerometer ButtonGreen.png All working.
IR Camera ButtonGreen.png All working.
Power Button ButtonGreen.png All working.
Speaker ButtonGreen.png All working.
Player LEDs ButtonGreen.png All working.
Status Information ButtonGreen.png Battery and extension info in Status Report
Extension Controllers ButtonYellow.png Official extensions are supported, however many 3rd party extensions are not understood. See Extension Controllers page
Legend ButtonGreen.png Perfect or near-perfect ButtonYellow.png Usable but not complete ButtonRed.png Unusable

Bluetooth Communication

The wiimote communicates with the host via standard bluetooth protocol. The wiimote can be placed into discoverable mode for 20s by pressing the sync button on its back under the battery cover. Holding down the 1 and 2 button continuously will force the wiimote to stay in discoverable mode without turning off. This does not work with the sync button, though. When in discoverable mode, a number of the player LEDs based on the battery level will blink. With full battery all four LEDs will blink, the lower the battery the less LEDs will blink. During device inquiry the host will find all discoverable nearby wiimotes. Now the host can establish a bluetooth baseband connection to the wiimote, no bluetooth pairing is needed, however, if bluetooth pairing is performed, the wiimote is able to reconnect to the host if disconnected. After a bluetooth baseband connection is established (with or without pairing) the HID channels can be opened and used for reading and writing reports from/to the wiimote.

The newer Wiimote RVL-CNT-01-TR shuts down immediately upon receiving any HID output report if it has been turned on using the 1 + 2 method, although it works using the sync button. It is possible that authentication is now mandatory for the 1 + 2 temporary sync[check].

Bluetooth Pairing

The wiimote supports the legacy bluetooth pairing methods. This involves sending a PIN to the wiimote. Bluetooth pairing is not required to use a wiimote and you can proceed by establishing a HID connection without pairing at all. However, if the wiimote is paired, it will actively seek out for its last connected host on disconnection and reestablish the connection. The following section explains the bluetooth device pairing, if no pairing is required, skip this section.

Bluetooth pairing must be initiated by the host by sending a "Require Authentication" HCI command to its bluetooth device. The bluetooth device will ask the host for a link key, which must be rejected so it will ask for a PIN-Code. The PIN-Code is the binary bluetooth address of the wiimote backwards. Following a short piece of C code to calculate the PIN:

Lets assume the Wiimote has the bluetooth address "00:1E:35:3B:7E:6D". If you want the PIN for bluetooth pairing in a simple string, do the following:

char pin[6];
pin[0] = 0x6D;
pin[1] = 0x7E;
pin[2] = 0x3B;
pin[3] = 0x35;
pin[4] = 0x1E;
pin[5] = 0x00;

Now "pin" contains your bluetooth pin that should be used for pairing your devices.

If connecting by holding down the 1+2 buttons, the PIN is the bluetooth address of the wiimote backwards, if connecting by pressing the "sync" button on the back of the wiimote, then the PIN is the bluetooth address of the host backwards.

After sending the PIN to the bluetooth device via HCI commands, the wiimote will return a "Authentication Accepted" command and the pairing is established (both devices are bonded now). After pairing you continue with establishing the HID connection the same way as without pairing.

If the host successfully bonded with the wiimote and established an HID connection the wiimote will save the bluetooth address of the host and enable single press reconnection. That means if the wiimote is now disconnected from the host, it will actively seek out for the host if any button is pressed and establish a baseband and HID connection. The wiimote will never actively send pairing requests since this is not needed. Also remember that this works with any button not only the power-button. However, after establishing the connection, the wiimote sends a button-input-report and this allows the host to see what button was pressed. So the host may reject the new connection if any button except the power-button was pressed.

The new bluetooth pairing method SSP (Secure Simple Pairing) is not supported. Also it is not yet investigated whether a link key has to be created (by sending a PIN) on every connection or whether the link key can be saved and reused on new connections. Though, creating a new link key on every connection works fine.

The wiimote has space for several host addresses (at least 3 are known to work) so it can be paired with more than one host (like PC or Wii) and it will try in reverse order to reconnect to the hosts. That is, the last paired host is tried first and so one. If button 1 and 2 or the sync button on its back are pressed, the wiimote will not actively seek out for its host but instead place itself in discoverable mode and wait for incoming connections so bluetooth pairing does not conflict with normal host-side connections.

It is not known how to remove the hosts addresses from the wiimote, however, with some investigation it should be possible to locate them in the EEPROM and manipulate them. If this is considered a security issue, then don't pair your devices.

SDP information

When queried with the Bluetooth Service Discovery Protocol (SDP), the Wii Remote reports back a great deal of information. In particular, it reports:

Wii Remote/old Wii Remote Plus new Wii Remote Plus
Name Nintendo RVL-CNT-01 Nintendo RVL-CNT-01-TR
Vendor ID 0x057e 0x057e
Product ID 0x0306 0x0330
Major Device Class 1280 ?
Minor Device Class 4 ?
Service Class 0 ?
(Summary of all Class Values) 0x002504 0x000508

HID Interface

Establishing a HID connection with the Wii Remote can be done on PSM 0x11 for the control pipe and PSM 0x13 for the data pipe using the Bluetooth L2CAP protocol. On Windows you don't need to deal with L2CAP yourself, and can use high-level windows HID functions.

The HID standard allows devices to be self-describing, using a HID descriptor block. This block includes an enumeration of reports that the device understands. A report can be thought of similar to a network port assigned to a particular service. Reports are unidirectional however, and the HID descriptor lists for each port the direction (Input or Output) and the payload size for each port. Like all Bluetooth HID devices, the Wii Remote reports its HID descriptor block when queried using the SDP protocol. However, no information regarding the actual data units within each report is returned, only the length in bytes.

Note: An "Input" report is sent by the Wii Remote to the host. An "Output" report is sent by the host to the Wii Remote. When using a Wii Remote, all input reports are prepended with 0xa1 and all output reports are prepended with 0xa2; this is the "(a2)" in many example reports below. Output reports are sent over the data pipe, which is also used to read input reports (thus, the control pipe is essentially unused).

The original Wiimotes (RVL-CNT-01) allowed sending commands using SET REPORT (0x52) over the control pipe, instead of using the data pipe; however, this form does not work on the newer RVL-CNT-01-TR, and as such is not recommended.

These are the reports the Wii Remote uses, and their use:

I/O ID(s) Size Function
O 0x10 1 Rumble
O 0x11 1 Player LEDs
O 0x12 2 Data Reporting mode
O 0x13 1 IR Camera Enable
O 0x14 1 Speaker Enable
O 0x15 1 Status Information Request
O 0x16 21 Write Memory and Registers
O 0x17 6 Read Memory and Registers
O 0x18 21 Speaker Data
O 0x19 1 Speaker Mute
O 0x1a 1 IR Camera Enable 2
I 0x20 6 Status Information
I 0x21 21 Read Memory and Registers Data
I 0x22 4 Acknowledge output report, return function result
I 0x30-0x3f 2-21 Data reports

For clarity, the convention in this document is to show packets including the Bluetooth-HID command (in parentheses), report ID, and payload, as described in sections 7.3 and 7.4 of the Bluetooth HID specification. Each byte is written out in hexadecimal, without the 0x prefix, separated by spaces. For example,

(a1) 30 00 00

is a DATA input packet (0xa1), on channel 0x30, with the two byte payload 0x00, 0x00. When using higher level HID functions rather than Bluetooth functions, the bytes in parentheses will never be present.

Force Feedback is accessible through the first byte of ALL output reports in the same way. This is not included above to avoid clutter.

Output Report common information

The first byte in many Output reports has a similar meaning. In every single Output Report, bit 0 (0x01) of the first byte controls the Rumble feature. Additionally, bit 2 (0x04) is used in several Output Reports as the ON/OFF flag for the specific feature controlled by it. For example, sending 0x04 to Report 0x19 (Speaker Mute) will mute the speaker:

(a2) 19 04

Sending 0x00 will unmute it:

(a2) 19 00

These Output Reports share the above behavior: Data Reporting Mode (0x12), IR Camera Enable (0x13), Speaker Enable (0x14), Speaker Mute (0x19), IR Enable 2 (0x1a).

Input Report common information

The first two bytes of ALL input reports, except 0x3d, contain the Core Buttons (BB BB). This includes all the 0x2~ status reports, not just the 0x3~ data reports. 0x3d is an exception, since it only returns expansion information.

Status Reporting

0x20: Status

To request the status report, send anything to Output Report 0x15. The Status Report will also be automatically sent when an Extension Controller is connected or disconnected.

This will request the status report (and turn off rumble):

(a2) 15 00

This report is sent either on request (in response to report 0x15), or in response to an expansion being plugged in or unplugged (or synced if wireless). If this report is received when not requested, the application 'MUST' send report 0x12 to change the data reporting mode, otherwise no further data reports will be received.

(a1) 20 BB BB LF 00 00 VV

BBBB is the core Buttons data. VV is the current battery level, L is the LED state, and F is a bitmask of flags indicating, whether the battery is flat, whether an expansion is currently connected, etc.

The Wii Remote can report its status, which includes the state of a few basic settings, the status of the Extension Controller (connected or disconnected), and the battery level.

VV is the current battery level, and LF is a bitmask of flags:

Bit Mask Meaning
0 0x01 Battery is nearly empty
1 0x02 An Extension Controller is connected
2 0x04 Speaker enabled
3 0x08 IR camera enabled
4 0x10 LED 1
5 0x20 LED 2
6 0x40 LED 3
7 0x80 LED 4

0x21: Read Memory Data

This report is sent when a read memory request is made. It returns 1 to 16 bytes of data at a time.

(a1) 21 BB BB SE AA AA DD DD DD DD DD DD DD DD DD DD DD DD DD DD DD DD

BBBB is the core Buttons data.

S (high nybble of SE) is the size in bytes, minus one, for the current data packet. This is 0xf (16 bytes) for all but the last packet, where it might be less if the requested number of bytes is not a multiple of 16.

E (low nybble of SE) is the error flag. Known error values are 0 for no error, 7 when attempting to read from a write-only register or an expansion that is not connected, and 8 when attempting to read from nonexistant memory addresses.

AA AA are the 2 least significant bytes of the absolute memory address of the first byte of data returned (the high byte of the offset is not returned, and neither is whether it is a register or memory that is being used. Thus, this must be known from the read request).

The DD bytes are the data, padded with zeroes to 16 bytes. If more than 16 bytes are requested, multiple packets will be received, with AA AA addresses increasing by 16 each time.

0x22: Acknowledge output report, return function result

This input report is sent to the host to report an error related to an output report, or the function result from that output report. It is sent when bit 1 of the first byte of any output report is set.

(a1) 22 BB BB RR EE

BBBB is the core Buttons data.

RR is the output report number that the Wii remote is acknowledging it received.

EE is the error code or function result. 00 = success. 03 = error. 04 = unknown (possibly returned by report 16H, 17H or 18H) 05 = unknown (possibly returned by report 12H). 08 = unknown (possibly returned bt report 16H).

Data Reporting

The Wii Remote has a number of different data reporting modes. Each of these modes combines certain Core data features with data from external peripherals, and sends it to the host through one of the report IDs, determined by the mode. The data format from the peripherals is determined by the peripherals themselves, all the Wii Remote controller does is pull bytes from them and send them out to the host. Due to this, certain feature combinations are not available, as there are not enough bytes for them in any of the output modes.

The Data Reporting Mode is set by sending a two-byte command to Report 0x12:

(a2) 12 TT MM

Bit 2 of TT specifies whether continuous reporting is desired. If bit 2 (0x04) is set, the Wii Remote will send reports whether there has been any change to the data or not. Otherwise, the Wii Remote will only send an output report when the data has changed.

MM specifies the Reporting Mode. Each Mode is specified by the Output Report ID that the data will be sent to. For example, this will set mode to 0x33:

(a2) 12 00 33

Data will then arrive through Input Report 0x33.

Upon powerup, the Data Reporting Mode defaults to 0x30. Following a connection or disconnection event on the Extension Port, data reporting is disabled and the Data Reporting Mode must be reset before new data can arrive.

Modes which include Accelerometer data also embed part of it in the unused Buttons bits. In all modes except for 0x3e/0x3f, the Buttons data includes the LSBs of the Accelerometer data. In mode 0x3e/0x3f, the interleaved Buttons data includes the Z-axis Accelerometer data.

0x30: Core Buttons

This mode returns data from the buttons in the Wii Remote:

(a1) 30 BB BB

BBBB is the core Buttons data.

0x31: Core Buttons and Accelerometer

This mode returns data from the buttons and the accelerometer in the Wii Remote:

(a1) 31 BB BB AA AA AA

BBBB is the core Buttons data. AA AA AA is the Accelerometer data.

0x32: Core Buttons with 8 Extension bytes

This mode returns data from the buttons in the Wii Remote, and data from an extension controller connected to it:

(a1) 32 BB BB EE EE EE EE EE EE EE EE

BBBB is the core Buttons data. The 8 EE bytes are from the Extension Controller currently connected to the Wii Remote.

0x33: Core Buttons and Accelerometer with 12 IR bytes

This mode returns data from the buttons, accelerometer, and IR Camera in the Wii Remote:

(a1) 33 BB BB AA AA AA II II II II II II II II II II II II 

BBBB is the core Buttons data. AA AA AA is the Accelerometer data. The 12 II bytes are from the built-in IR Camera.

0x34: Core Buttons with 19 Extension bytes

This mode returns data from the buttons in the Wii Remote, and data from an extension controller connected to it:

(a1) 34 BB BB EE EE EE EE EE EE EE EE EE EE EE EE EE EE EE EE EE EE EE 

BBBB is the core Buttons data. The 19 EE bytes are from the Extension Controller currently connected to the Wii Remote.

0x35: Core Buttons and Accelerometer with 16 Extension Bytes

This mode returns data from the buttons and accelerometer in the Wii Remote, and data from an extension controller connected to it:

(a1) 35 BB BB AA AA AA EE EE EE EE EE EE EE EE EE EE EE EE EE EE EE EE

BBBB is the core Buttons data. AA AA AA is the Accelerometer data. The 16 EE bytes are from the Extension Controller currently connected to the Wii Remote.

0x36: Core Buttons with 10 IR bytes and 9 Extension Bytes

This mode returns data from the buttons and IR camera in the Wii Remote, and data from an extension controller connected to it:

(a1) 36 BB BB II II II II II II II II II II EE EE EE EE EE EE EE EE EE

BBBB is the core Buttons data. The 10 II bytes are from the built-in IR Camera, and the 9 EE bytes are from the Extension Controller currently connected to the Wii Remote.

0x37: Core Buttons and Accelerometer with 10 IR bytes and 6 Extension Bytes

This mode returns data from the buttons, accelerometer, and IR camera in the Wii Remote, and data from an extension controller connected to it:

(a1) 37 BB BB AA AA AA II II II II II II II II II II EE EE EE EE EE EE

BBBB is the core Buttons data. AA AA AA is the Accelerometer data. The 10 II bytes are from the built-in IR Camera, and the 6 EE bytes are from the Extension Controller currently connected to the Wii Remote.

0x3d: 21 Extension Bytes

This mode returns data from an extension controller connected to the Wii Remote. It is the only input report that does not include core buttons.

(a1) 3d EE EE EE EE EE EE EE EE EE EE EE EE EE EE EE EE EE EE EE EE EE

The 21 EE bytes are from the Extension Controller currently connected to the Wii Remote.

0x3e / 0x3f: Interleaved Core Buttons and Accelerometer with 36 IR bytes

Both 0x3e and 0x3f are equivalent, and return data alternately through report IDs 0x3e and 0x3f. The data is interleaved, and is returned at half the speed of other modes (as two reports are needed for a single data unit). This mode returns data from the buttons, accelerometer, and IR camera in the Wii Remote:

(a1) 3e BB BB AA II II II II II II II II II II II II II II II II II II
(a1) 3f BB BB AA II II II II II II II II II II II II II II II II II II

BBBB is the core button data, as specified in the Buttons section. AA AA is the Accelerometer data, in a format specific to this mode described in the Interleaved Accelerometer Reporting section. The 36 II bytes are from the built-in IR Camera.

Memory and Registers

The Wii Remote includes a built-in EEPROM memory, part of which is accessible to the user to store that. This user part is used to store calibration constants, as well as the Mii Data. Additionally, many peripherals on the Wii Remote have registers which are accessible through a portion of the address space.

Both built-in memory and peripheral registers are accessed using the same reports, where a flag is used to select between the two.

Reading and Writing

To read data, commands are sent to Output Report 0x17:

(a2) 17 MM FF FF FF SS SS

FF FF FF is the offset, and SS SS is the size to read in bytes (both in big-endian format). Bit 2 (0x04) of MM selects the address space. Clearing this bit results in reading from EEPROM Memory, while setting it results in reading from the control registers. Setting bit 3 (0x08) also works to access registers, but setting both results in errors. As with all other reports, it also includes the Rumble flag, which must be set to the current rumble state to avoid affecting it.

Data read is returned through Input Report 0x21:

(a1) 21 BB BB SE FF FF DD DD DD DD DD DD DD DD DD DD DD DD DD DD DD DD

BB BB is the state of the buttons on the Wii Remote. During data reads, regular input reporting is temporarily suspended. Button data is available through the data input reports, but no other input data can be collected while the transfer lasts. FF FF is the offset expressed in absolute memory address of the Wii remote memory for the first byte of data returned (the high byte of the offset is not returned, and neither is which data memory is being used. Thus, this must be known from the read request). E (low nybble of SE) is the error flag. Known error values are 0 for no error, 7 when attempting to read from a write-only register, and 8 when attempting to read from nonexistant memory. S (high nybble of SE) is the size in bytes, minus one, for the current data packet. This is 0xf (16 bytes) for all but the last packet, where it might be less if the requested number of bytes is not a multiple of 16. The DD bytes are the data, padded with zeroes to 16 bytes. If more than 16 bytes are requested, multiple packets will be received, with FF FF offsets increasing by 16 each time.

To write data, commands are sent to Output Report 0x16:

(a2) 16 MM FF FF FF SS DD DD DD DD DD DD DD DD DD DD DD DD DD DD DD DD

The meaning of the bytes is the same as during reads, except that size can be a maximum of 16 bytes (as there is only space for that much data), and the actual data to write follows (the DD bytes), padded out to 16 bytes.

Some kind of acknowledgement is received on Input Report 0x22. This has not been investigated yet.

EEPROM Memory

Wii Remote Flash.jpg

There is a 128kbit (= 16kB) EEPROM chip (Data Sheet / Full EEPROM dump from a sample Wii Remote) in the Wii Remote. Part of its contents include code for the built-in microcontroller, and a generic section which can be freely read and written by the host. This section is 0x1700 bytes long, and part of this memory is used to store the Mii Data. It can be accessed by reading from/writing to addresses 0x0000-0x16FF in the Wii Remote's virtual memory space; in the actual EEPROM chip, the data is located at 0x0070-0x176F.

The firmware stored in the Wiimote has been disassembled.

The BCM2042 microcontroller built into the Wii Remote includes a large 108kb on-chip ROM section for storing firmware. If the EEPROM chip really contains code for the BCM2042 then this was probably done to make firmware updates possible, so there might be a way of accessing the other parts of the EEPROM via Bluetooth as well. From the BCM2042 Product Brief: "ROM-based design eliminates external flash memories; Flash option offered to support feature development".

On a virgin Wii Remote, acquired separately (not bundled with a Wii), that has never communicated with any device (except the PC used to dump the memory contents), most of the memory is blank (0x00). However, the first few bytes contain some information:

0000:  A1 AA 8B 99 AE 9E 78 30 A7 74 D3 A1 AA 8B 99 AE
0010:  9E 78 30 A7 74 D3 82 82 82 15 9C 9C 9E 38 40 3E
0020:  82 82 82 15 9C 9C 9E 38 40 3E 00 00 00 00 00 00
0030:  00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00

This can be better visualized as two sequences, each one repeated twice:

0000:  A1 AA 8B 99 AE 9E 78 30 A7 74 D3
000B:  A1 AA 8B 99 AE 9E 78 30 A7 74 D3
0016:  82 82 82 15 9C 9C 9E 38 40 3E
0020:  82 82 82 15 9C 9C 9E 38 40 3E

These are two blocks of calibration data (for redundancy), where the last byte of each is a checksum (all prior bytes plus 0x55). Nintendo's libraries will only use a calibration block if its checksum is correct, falling back to the second one if the first one is incorrect, and falling back to default values otherwise.

The first block contains calibration data related to the IR camera: the position of 4 reference values. Nintendo's code uses this to compute a center offset, scale, and possibly an angle offset. It also looks like the values could appear in any order; code exists to check for values in each quadrant (below and above x=512 and y=384).

  Bit
Byte 7 6 5 4 3 2 1 0
0x00 0x0B X 1<7:0>
0x01 0x0C Y 1<7:0>
0x02 0x0D Y 1<9:8> X 1<9:8> Y 2<9:8> X 2<9:8>
0x03 0x0E X 2<7:0>
0x04 0x0F Y 2<7:0>
0x05 0x10 X 3<7:0>
0x06 0x11 Y 3<7:0>
0x07 0x12 Y 3<9:8> X 3<9:8> Y 4<9:8> X 4<9:8>
0x08 0x13 X 4<7:0>
0x09 0x14 Y 4<7:0>
0x0A 0x15 Checksum

The data given above decodes to (0A1, 2AA), (399, 28B), (39E, 078), (0A7, 030).

Additional examples
  • b4 b8 b8 a0 ad b6 66 30 a5 61 18: (3B4, 2B8), (0A0, 2B8), (3B6, 066), (0A5, 030)
  • 70 5e 03 7b 50 83 9b b8 7a a5 e6: (070, 05E), (37B, 003), (383, 29B), (07A, 2B8)
  • 83 9b 80 7e 50 8a 4c 3b 8d 98 f7: (083, 29B), (07E, 080), (38A, 04C), (38D, 23B)
  • 6f a9 8b 7d ae 78 5d 03 83 67 e5: (06F, 2A9), (37D, 28B), (078, 05D), (383, 003)

The four bytes starting at 0x0016 and 0x0020 store the calibrated zero offsets for the accelerometer (high 8 bits of X,Y,Z in the first three bytes, low 2 bits packed in the fourth byte as --XXYYZZ). The four bytes at 0x001A and 0x24 store the force of gravity on those axes. The byte at 0x1e and 0x28 seems to be unused, but apparently used to contain flags related to the speaker volume and the rumble motor (Nintendo games will print a debug message related to it but do not seem to use the value elsewhere). On all checked remotes, that byte has been 0x40.

  Bit
Byte 7 6 5 4 3 2 1 0
0x16 0x20 0G X Acceleration<9:2>
0x17 0x21 0G Y Acceleration<9:2>
0x18 0x22 0G Z Acceleration<9:2>
0x19 0x23 0 0G X Acc<1:0> 0G Y Acc<1:0> 0G Z Acc<1:0>
0x1A 0x24 1G X Acceleration<9:2>
0x1B 0x25 1G Y Acceleration<9:2>
0x1C 0x26 1G Z Acceleration<9:2>
0x1D 0x27 0 1G X Acc<1:0> 1G Y Acc<1:0> 1G Z Acc<1:0>
0x1E 0x28 Motor Volume
0x1F 0x29 Checksum

At 0x16D0, there is some more unknown data:

16D0:  00 00 00 FF 11 EE 00 00 33 CC 44 BB 00 00 66 99
16E0:  77 88 00 00 2B 01 E8 13 00 00 00 00 00 00 00 00
16F0:  00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00

In contrast to the data at 0x0000, this data seems to differ in only a few bytes between different Wii Remotes.

The "User Data" section appears to start with a 0x38-byte header, repeated twice (so 0x70 bytes total). The header starts with an 8-byte timestamp (from TBL and TBU; 1/60750000ths of a second since January 1 2000). That is followed by a 34-byte (17 characters, or 16 excluding the null terminator) Unicode name (which does not necessarily match the game's name), followed by the 4-byte game ID, then the application type (the value at 0x80003184). Then there is a checksum (at 0x2f), which is the sum of bytes 0 through 0x2e plus 0x55. Bytes 0x30 through 0x37 do not seem to be used and were all 0.

Known memory ranges are listed below.

Start End Length Initial Value Use
0x0000 0x0029 0x002A See above Calibration values / pre-set data
0x002A 0x0FC9 0x0FA0 Zeroed User data / Unknown uses
0x0FCA 0x12B9 0x02f0 Zeroed Mii Data block 1
0x12BA 0x15A9 0x02f0 Zeroed Mii Data block 2
0x15AA 0x16CF 0x0126 Zeroed Unknown / Unused
0x16D0 0x16FF 0x0030 See above Unknown data

The top byte of the address is unused, which means memory is mirrored every 0x10000 bytes. Reading from unused addresses where the low 16 bits are >= 0x1700 will result in error returns.

Control Registers

The Wii Remote has several memory mapped register spaces corresponding to different peripherals in it. These include the Speaker, Extension Controllers, and the IR Camera.

Reminder
Remember to set bit 2 (0x04) on the first byte of the Output Report, otherwise you'll overwrite EEPROM memory!

The peripheral to access is selected by the first byte of the address, and the lower 16 bits specify the register to access within that peripheral. The lowest bit of the high byte is ignored, which means every peripheral is mirrored at its address + 0x10000. Known peripherals are listed below:

Start End Use
0xA20000 0xA20009 Speaker settings
0xA40000 0xA400FF Extension Controller settings and data
0xA60000 0xA600FF Wii Motion Plus settings and data
0xB00000 0xB00033 IR Camera settings

Most of these are also mirrored across the high bits of the individual peripheral. For example, the second byte of the address is ignored in the Extension controller address, which means any address of the form 0xA4xx00 will work (as will 0xA5xx00).

Input Features

The Wii Remote has two input features that are controlled directly by the Broadcom chip: a Three-Axis Accelerometer and 11 Buttons. Additionally, it has an IR Camera with an object tracking processor, and an expansion port that allows for external input features such as those contained in the Nunchuk and the Classic Controller (see Extension Controllers).

Buttons

The Wii Remote has 11 buttons on its front face, and one trigger-style button on the back. Of these, the Power button is special and is treated differently by the Wii Remote. All the other buttons are independantly accessible through a two-byte bitmask which is transmitted first in most Input Reports. A button will report a 1-bit if pressed, or a 0-bit otherwise. By default, these are sent only when the state of any button changes, in Data Reporting Mode 0x30. However, the Wii Remote may be configured to report the state of the buttons continuously; see Data Reporting.

Core Buttons

The Wii Remote has 11 buttons that are used as regular input devices: A, B (trigger), a 4-directional D-Pad, +, -, Home, 1, and 2. These are reported as bits in a two-byte bitmask. These are the assignments, in big-endian order:

Bit Mask First Byte Second Byte
0 0x01 D-Pad Left Two
1 0x02 D-Pad Right One
2 0x04 D-Pad Down B
3 0x08 D-Pad Up A
4 0x10 Plus Minus
5 0x20 Other uses Other uses
6 0x40 Other uses Other uses
7 0x80 Unknown Home

Power Button

When the Wii Remote is turned off, pressing the Power button will attempt to wake up the Wii that is synchronized to it. The mechanism for this is unknown, and it is handled entirely within the Wii's bluetooth module. When the Wii Remote is turned on and connected to a host, pressing and holding the Power button for a few seconds will turn the Wii Remote off and request disconnection from the host. The disconnection reason included with the Baseband (ACL) disconnection request indicates that the power button was pressed: REMOTE DEVICE TERMINATED CONNECTION DUE TO POWER OFF (0x15). Another possible value is REMOTE DEVICE TERMINATED CONNECTION DUE TO LOW RESOURCES (0x14), which indicates that the Wii Remote performed a controlled shut down due to a low battery condition.

Sync Button

The sync button is hidden under the battery cover. When the Sync button is pressed, the Wii remote will disconnect from whatever it is currently connected to, make itself discoverable, and accept pairing or connection requests for exactly 20 seconds (regardless of how long the button is held down for).

The "syncing" of a Wii Remote involves standard Bluetooth pairing. When the Sync button is pressed on the remote, it will accept pairing requests. The required PIN is the hosts's Bluetooth address, backwards (last byte first), in binary (6 bytes). Most current Bluetooth implementations don't deal with this correctly, as they usually consider the PIN to be a regular null-terminated ASCII string (no 00 bytes, etc) and most Bluetooth addresses will contain null bytes. Any further steps that need to be taken after the Wii Remote is paired have not been reverse engineered yet.

Once the Wii Remote is synced, when a button is pressed, it will actively seek out its paired host and try to connect to it, instead of the other way around. Establishing a connection can be done on PSM 0x11 for writing and PSM 0x13 for reading using the Bluetooth L2CAP protocol.

Button Hardware

The physical hardware of the buttons varies: there are membrane switches and microswitch click buttons. There has been some success soldering wires to the membrane switch contacts and actuating the switch through an external switch. The following table describes the physical hardware for each input.

Function Switch type Circuit board surface
A membrane top, SW9
B membrane bottom, SW8
- microswitch top, SW10
Home microswitch top, SW11
+ microswitch top, SW5
1 membrane top, SW7
2 membrane top, SW6
Up membrane top, SW4
Down membrane top, SW3
Left membrane top, SW1
Right membrane top, SW2
Sync bottom, SW12
Power top, SW13

Accelerometer

ADXL330 in a Wii remote
Coordinate system used by Wii Remote

The Wii Remote includes a three-axis linear accelerometer located on the top suface of the circuit board, slightly left of the large A button. The integrated circuit is the ADXL330 (data sheet), manufactured by Analog Devices. This device is physically rated to measure accelerations over a range of at least +/- 3g with 10% sensitivity.

Since the accelerometer actually measures the force exerted by a set of small proof masses inside of it with respect to its enclosure, the accelerometer measures linear acceleration in a free fall frame of reference. If the Wii Remote is in free fall, it will report zero acceleration. At rest, it will report an upward acceleration (+Z, when horizontal) equal to the acceleration due to gravity, g (approximately 9.8 m/s²) but in the opposite direction. This fact can be used to derive tilt from the acceleration outputs when the Wii Remote is reasonably still.

Normal Accelerometer Reporting

In all Data Reporting Modes which include Accelerometer data except for mode 0x3e/0x3f, the accelerometer data is reported as three consecutive bytes:

(a1) RR BB BB XX YY ZZ [...]

XX, YY, and ZZ are unsigned bytes representing the acceleration in each of the three axis, where zero acceleration is approximately 0x80. The coordinate system is shown in the diagram above (note that this is different from the coordinate system used by GlovePIE). Additionally, the BB BB Buttons bytes also include the LSBs of the acceleration values in the unused bits, according to the following table:

  Bit
Byte 7 6 5 4 3 2 1 0
0   X<1:0>  
1   Z<1> Y<1>  

Note that X has 10 bits of precision, while Y and Z only have 9. For consistency, they are assumed all to have a 10-bit range and the LSB is always set to zero for Y and Z.

Interleaved Accelerometer Reporting

In Data Reporting Mode 0x3e/0x3f, the accelerometer data is spread over two reports:

(a1) 3e BB BB XX [...]
(a1) 3f BB BB YY [...]

In this mode, the LSBs are not available. Instead, X and Y acceleration is reported as a single byte, and the Z value is encoded in the unused bits of the BB BB Buttons data as follows:

  Bit
Report ID Byte 7 6 5 4 3 2 1 0
0x3e 0   Z<5:4>  
0x3e 1   Z<7:6>  
0x3f 0   Z<1:0>  
0x3f 1   Z<3:2>  

IR Camera

Wii remote camera

The Wii Remote includes a 128x96 monochrome camera with built-in image processing. The camera looks through an infrared pass filter in the remote's plastic casing. The camera's built-in image processing is capable of tracking up to 4 moving objects, and these data are the only data available to the host. Raw pixel data is not available to the host, so the camera cannot be used to take a conventional picture. The built-in processor uses 8x subpixel analysis to provide 1024x768 resolution for the tracked points. The Sensor Bar that comes with the Wii includes two IR LED clusters at each end, which are tracked by the Wii Remote to provide pointing information. The distance between the centers of the LED clusters is 20 cm (as measured on one unit).

The IR Camera is enabled by setting bit 2 on output reports 0x13 and 0x1a:

(a2) 13 04
(a2) 1a 04

The first enables a 24MHz pixel clock on pin 7 of the camera. The second pulls pin 4 low - probably an active-low enable.

Mechanical Characteristics

The camera component is mounted on the bottom surface of the circuit board. The camera module itself is mounted in a socket perpendicular to the circuit board; to remove just the camera module, no desoldering is required. The process is as follows:

First, orient the camera so that you are looking into the lens with the PCB horizontal and below the lens.. There are four metal clips, two on each of the vertical sides of the socket. Use something tiny to slide between each metal clip and the camera module: maybe wire wrap wire? Then look at the back of the camera module, opposite the lens. There is a small rectangular hole in the middle of each vertical side of the socket. Use a pin or something to pry/press the camera module out.

Once the camera module is free of its socket, it may be further disassembled by gently prising up the tiny PCB with gold contacts; this is gently glued to the module's structure, but will come loose without damage. At this point you have three pieces: the camera socket, still attached to the Wiimote PCB, the camera module housing, complete with lens and dichroic filter (of unknown optical properties), and a tiny PCB with the camera chip and eight gold contacts on the bottom.

Optical Characteristics

The IR camera has an effective field of view is about 33 degrees horizontally and 23 degrees vertically (as measured on one unit). With the IR-pass filter intact, 940nm sources are detected with approximately twice the intensity of equivalent 850nm sources, but are not resolved as well at close distances. If the filter is removed, it can track any bright object. However, the IR filter referred to here is not only the dark plastic window of the wiimote but also a teensy slab of dichroic-coated glass inside the camera module. One may operate the wiimote having installed neither, one or the other, or both filters.

Initialization

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

The following procedure should be followed to turn on the IR Camera:

  1. Enable IR Camera (Send 0x04 to Output Report 0x13)
  2. Enable IR Camera 2 (Send 0x04 to Output Report 0x1a)
  3. Write 0x08 to register 0xb00030
  4. Write Sensitivity Block 1 to registers at 0xb00000
  5. Write Sensitivity Block 2 to registers at 0xb0001a
  6. Write Mode Number to register 0xb00033
  7. Write 0x08 to register 0xb00030 (again)

After these steps, the Wii Remote will be in one of 3 states: IR camera on but not taking data, IR camera on and taking data and half sensitivity, IR camera on and taking data at full sensitivity. Which state you end up in appears to be pretty much random. Repeat the steps until you're in the desired state. To avoid the random state put a delay of at least 50ms between every single byte transmission.

The Wii preforms these steps slightly different, differences in bold:

  1. Enable IR Pixel Clock (send 0x06 to Output Report 0x13)
  2. Enable IR Logic (send 0x06 to Output Report 0x1A)
  3. Write 0x01 to register 0xb00030
  4. Write Sensitivity Block 1 to registers at 0xb00000
  5. Write Sensitivity Block 2 to registers at 0xb0001a
  6. Write Mode Number to register 0xb00033
  7. Write 0x08 to register 0xb00030 (again)

Adding bit 0x02 to reports 0x13 and 0x1a is a request for acknowledgement (if set, wiimote will respond with report 0x22).

Sensitivity Settings

Sensitivity is controlled by two configuration blocks, 9 bytes and 2 bytes long. The following settings are known to work:

Block 1 Block 2 Notes
00 00 00 00 00 00 90 00 C0 40 00 Suggested by Marcan
00 00 00 00 00 00 FF 00 0C 00 00 Suggested by Kestrel (max sensitivity)
00 00 00 00 00 00 90 00 41 40 00 Suggested by inio (high sensitivity)
02 00 00 71 01 00 64 00 fe fd 05 Wii level 1
02 00 00 71 01 00 96 00 b4 b3 04 Wii level 2
02 00 00 71 01 00 aa 00 64 63 03 Wii level 3 (Suggested by Cliff)
02 00 00 71 01 00 c8 00 36 35 03 Wii level 4
07 00 00 71 01 00 72 00 20 1f 03 Wii level 5

The last byte of Block 1 determines the intensity sensitivity, with increasing values reducing the sensitivity. Both bytes of Block 2 must be zero for the full sensitivity range to be available. Setting the sensitivity as high as possible, without unwanted light being tracked, is recommended to achieve the highest subpixel resolution. As the sensitivity is reduced, the subpixel resolution also reduces, approaching the true sensor resolution of 128x96.

Data Formats

The IR Camera can return different sets of data describing the objects it is tracking. When the IR camera identifies an object, it assigns it to the first available object slot. If an object moves out of view, its slot is marked as empty (returns 0xFF data), but other objects retain their slots. For example, if the camera is tracking two objects and the first moves out of view, the data returned will be [empty, second object, empty, empty]. With more than four objects visible, the camera is prone to rapidly switching between some of them. This could allow perception of more than four objects, at a reduced response speed and reliability.

Mode Mode Number
Basic 1
Extended 3
Full 5

The data format MUST match the number of bytes available in the Reporting Mode selected. Even choosing a mode with space for more bytes than necessary will not work, it has to be an exact match.

Basic Mode

In Basic Mode, the IR Camera returns 10 bytes of data corresponding to the X and Y locations of each of the four dots. Each location is encoded in 10 bits and has a range of 0-1023 for the X dimension, and 0-767 for the Y dimension. Each pair of dots is packed into 5 bytes, and two of these are transmitted for a total of 4 dots and 10 bytes.

This is the data format for a pair of objects:

  Bit
Byte 7 6 5 4 3 2 1 0
0 X1<7:0>
1 Y1<7:0>
2 Y1<9:8> X1<9:8> Y2<9:8> X2<9:8>
3 X2<7:0>
4 Y2<7:0>

Extended Mode

In Extended Mode, the IR Camera returns the same data as it does in Basic Mode, plus a rough size value for each object. The data is returned as 12 bytes, three bytes per object. Size has a range of 0-15.

This is the data format for each object:

  Bit
Byte 7 6 5 4 3 2 1 0
0 X<7:0>
1 Y<7:0>
2 Y<9:8> X<9:8> S<3:0>

Full Mode

In Full Mode, the IR Camera returns even more data, 9 bytes per object for a total of 36 bytes for all four. The data is split up between two input reports of 18 bytes each (see Data Reporting Mode 0x3e/0x3f). The first three bytes of each object are the same as the extended mode, and are followed by the bounding box of the pixels included in the blob along with a deeper intensity value. The data format of each object is:

  Bit
Byte 7 6 5 4 3 2 1 0
0 X<7:0>
1 Y<7:0>
2 Y<9:8> X<9:8> S<3:0>
3 0 X min<6:0>
4 0 Y min<6:0>
5 0 X max<6:0>
6 0 Y max<6:0>
7 0
8 Intensity<7:0>

Feedback Features

The Wii Remote sports three feedback features: Player LEDs, Rumble, and the Speaker.

Player LEDs

Wii remote player LEDs

There are four blue LEDs on the front face of the Wii Remote. During discovery and before initialization, these LEDs blink at a fixed rate. The number of blinking LEDs is proportional to the battery voltage, indicating battery charge (all four are lit for newly charged batteries, and only the first is lit if the batteries are low and should be replaced).

During gameplay with the Wii, one LED is lit to indicate the player number assigned to the Wii Remote. However, the LEDs are independently controllable by the host, and can be set to display any pattern. They can also be modulated at a moderately high speed, enabling some brightness control at the cost of a lot of Bluetooth bandwidth. Sigma-delta modulation works reasonably well for this.

The LEDs can be controlled by sending a report with ID 0x11:

(a2) 11 LL

The high nybble of LL controls the four LEDs. Bit 4 of LL controls the first LED, and bit 7 controls the last:

Bit Mask LEDs
4 0x10
·
··
···
····
5 0x20
·
··
···
····
6 0x40
·
··
···
····
7 0x80
·
··
···
····

Turning off all LEDs for an extended period of time is discouraged, as it might lead the user to believe the Wii Remote is turned off and disconnected, when in fact it is active.

The LEDs are surface mount parts, driven at 2.66 VDC.

Rumble

Wii remote rumble motor

The Wii Remote includes a rumble feature, which is implemented as a small motor attached to an off-center weight. It will cause the controller to vibrate when activated.

The rumble motor is primarilly controlled via the report with ID 0x10:

(a2) 10 RR

Setting RR to 1 enables rumble, and setting it to 0 disables rumble. However, note that the rumble motor can be turned on or off through any of the Output Reports, not just 0x10. Setting the LSB (bit 0) of the first byte of any output report will activate the rumble motor, and unsetting it will deactivate it. For example, the following report will turn the rumble motor on:

(a2) 11 01

However, this will also have the side-effect of turning off all LEDs. The rumble bit needs to be set properly with every single report sent, to avoid inadvertently turning the rumble motor off.

Different photos of the rumble motor hardware show different markings. One example is SEM 8728DA. The Wii Remote drives it at 3.3 VDC and it draws 35 mA. It would be reasonable to think that the rumble motor could be removed and the motor replaced with another device with equal voltage and equal or less current draw.

Speaker

The Wii Remote has a small low-quality 21mm piezo-electric speaker, used for short sound effects during gameplay. The sound is streamed directly from the host, and the speaker has some adjustable parameters.

The speaker is controlled by using three output reports, together with a section of the register address space of the Wii Remote.

Report 0x14 is used to enable or disable the speaker. Setting bit 2 will enable the speaker, and clearing it will disable it. For example, to enable the speaker, send:

(a2) 14 04

Report 0x19 is used to mute or unmute the speaker, and works identically to report 0x14. 0x04 will mute the speaker, and 0x00 will unmute it.

Report 0x18 is used to send speaker data. 1-20 bytes may be sent at once:

(a2) 18 LL DD DD DD DD DD DD DD DD DD DD DD DD DD DD DD DD DD DD DD DD

LL specifies the data length, shifted left by three bits. The DD bytes are the speaker data. To fullfill the report length requirements, the data must be padded if it is less than 20 bytes long. Sound data must be sent at the proper rate.

Initialization Sequence

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

The following sequence will initialize the speaker:

  1. Enable speaker (Send 0x04 to Output Report 0x14)
  2. Mute speaker (Send 0x04 to Output Report 0x19)
  3. Write 0x01 to register 0xa20009
  4. Write 0x08 to register 0xa20001
  5. Write 7-byte configuration to registers 0xa20001-0xa20008
  6. Write 0x01 to register 0xa20008
  7. Unmute speaker (Send 0x00 to Output Report 0x19)

Speaker Configuration

7 bytes control the speaker settings, including volume. The full purpose of these bytes is not known, but the following values seem to produce some sound:

00 FF RR RR VV 00 00

RR RR specify the sample rate (little-endian format), using the following formulae:

pcm_sample_rate = 12000000 / rate_value adpcm_sample_rate = 6000000 / rate_value

The standard value is 0x7d0, for 3000Hz 4-bit PCM

FF configures the data format. Setting it to 0x40 configures the speaker to use signed 8-bit PCM, while setting it to 0x00 configures it to use 4-bit Yamaha ADPCM. VV specifies the volume, which has a range of 0x00-0xFF for 8-bit mode, and 0x00-0x40 for 4-bit mode.

This configuration can be used to play 4-bit ADPCM sound at 3000Hz:

00 00 D0 07 40 00 00

This configuration can be used to play 8-bit PCM sound at 1500Hz sample rate:

00 40 40 1f 40 00 00

It looks like the best compromise between sample rate and slow bluetooth chipsets / drivers is playing 8-bit PCM at 2000Hz, so a new 20-byte chunk of audio data is sent every 10 milliseconds.

00 40 70 17 60 00 00

Sound Data Format

The Wii Remote can use multiple sound formats at multiple sampling rates. PC drivers currently seem unable to keep up with the higher rates.

The 4-bit ADPCM is Yamaha ADPCM (for example, as implemented in ffmpeg).

8-bit signed PCM mode works, but when in 8-bit mode the sampling frequency must be made so low that the audio quality is pretty bad.

Extension Controllers

Wiimote/Extension Controllers

Notes

See Also

Acknowledgements

Some of the information here is based on the documentation at Wiili