mirror of
https://github.com/felis/USB_Host_Shield_2.0.git
synced 2024-03-22 11:31:26 +01:00
Now able to pair with Wiimote
This commit is contained in:
parent
02719f4c92
commit
ebb333de9a
5 changed files with 180 additions and 73 deletions
101
BTD.cpp
101
BTD.cpp
|
@ -427,12 +427,18 @@ void BTD::HCI_event_task() {
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case EV_PIN_CODE_REQUEST:
|
case EV_PIN_CODE_REQUEST:
|
||||||
if(btdPin != NULL) {
|
if(pairWithWii) {
|
||||||
|
#ifdef DEBUG
|
||||||
|
Notify(PSTR("\r\nPairing with wiimote"));
|
||||||
|
#endif
|
||||||
|
hci_pin_code_request_reply();
|
||||||
|
}
|
||||||
|
else if(btdPin != NULL) {
|
||||||
#ifdef DEBUG
|
#ifdef DEBUG
|
||||||
Notify(PSTR("\r\nBluetooth pin is set too: "));
|
Notify(PSTR("\r\nBluetooth pin is set too: "));
|
||||||
Serial.print(btdPin);
|
Serial.print(btdPin);
|
||||||
#endif
|
#endif
|
||||||
hci_pin_code_request_reply(btdPin);
|
hci_pin_code_request_reply();
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
#ifdef DEBUG
|
#ifdef DEBUG
|
||||||
|
@ -449,14 +455,21 @@ void BTD::HCI_event_task() {
|
||||||
hci_link_key_request_negative_reply();
|
hci_link_key_request_negative_reply();
|
||||||
break;
|
break;
|
||||||
|
|
||||||
/* We will just ignore the following events */
|
case EV_AUTHENTICATION_COMPLETE:
|
||||||
|
if(pairWithWii) {
|
||||||
|
#ifdef DEBUG
|
||||||
|
Notify(PSTR("\r\nPairing successful"));
|
||||||
|
#endif
|
||||||
|
connectToWii = true; // Only send the ACL data to the Wii service
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
/* We will just ignore the following events */
|
||||||
case EV_NUM_COMPLETE_PKT:
|
case EV_NUM_COMPLETE_PKT:
|
||||||
case EV_ROLE_CHANGED:
|
case EV_ROLE_CHANGED:
|
||||||
case EV_PAGE_SCAN_REP_MODE:
|
case EV_PAGE_SCAN_REP_MODE:
|
||||||
case EV_LOOPBACK_COMMAND:
|
case EV_LOOPBACK_COMMAND:
|
||||||
case EV_DATA_BUFFER_OVERFLOW:
|
case EV_DATA_BUFFER_OVERFLOW:
|
||||||
case EV_CHANGE_CONNECTION_LINK:
|
case EV_CHANGE_CONNECTION_LINK:
|
||||||
case EV_AUTHENTICATION_COMPLETE:
|
|
||||||
case EV_MAX_SLOTS_CHANGE:
|
case EV_MAX_SLOTS_CHANGE:
|
||||||
case EV_QOS_SETUP_COMPLETE:
|
case EV_QOS_SETUP_COMPLETE:
|
||||||
case EV_LINK_KEY_NOTIFICATION:
|
case EV_LINK_KEY_NOTIFICATION:
|
||||||
|
@ -552,16 +565,12 @@ void BTD::HCI_task() {
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case HCI_CHECK_WII_SERVICE:
|
case HCI_CHECK_WII_SERVICE:
|
||||||
if(wiiServiceID != -1) { // Check if it should try to connect to a wiimote
|
if(pairWithWii) { // Check if it should try to connect to a wiimote
|
||||||
if(disc_bdaddr[5] == 0 && disc_bdaddr[4] == 0 && disc_bdaddr[3] == 0 && disc_bdaddr[2] == 0 && disc_bdaddr[1] == 0 && disc_bdaddr[0] == 0) {
|
|
||||||
#ifdef DEBUG
|
#ifdef DEBUG
|
||||||
Notify(PSTR("\r\nStarting inquiry\r\nPress 1 & 2 on the Wiimote"));
|
Notify(PSTR("\r\nStarting inquiry\r\nPress 1 & 2 on the Wiimote"));
|
||||||
#endif
|
#endif
|
||||||
hci_inquiry();
|
hci_inquiry();
|
||||||
hci_state = HCI_INQUIRY_STATE;
|
hci_state = HCI_INQUIRY_STATE;
|
||||||
}
|
|
||||||
else
|
|
||||||
hci_state = HCI_CONNECT_WII_STATE;
|
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
hci_state = HCI_SCANNING_STATE; // Don't try to connect to a Wiimote
|
hci_state = HCI_SCANNING_STATE; // Don't try to connect to a Wiimote
|
||||||
|
@ -572,23 +581,16 @@ void BTD::HCI_task() {
|
||||||
hci_inquiry_cancel(); // Stop inquiry
|
hci_inquiry_cancel(); // Stop inquiry
|
||||||
#ifdef DEBUG
|
#ifdef DEBUG
|
||||||
Notify(PSTR("\r\nWiimote found"));
|
Notify(PSTR("\r\nWiimote found"));
|
||||||
Notify(PSTR("\r\nCreate the instance like so to connect automatically:"));
|
Notify(PSTR("\r\nNow just create the instance like so:"));
|
||||||
Notify(PSTR("\r\nWII Wii(&Btd,"));
|
Notify(PSTR("\r\nWII Wii(&Btd);"));
|
||||||
for(int8_t i = 5; i>0;i--) {
|
Notify(PSTR("\r\nAnd then press any button on the Wiimote"));
|
||||||
Notify(PSTR("0x"));
|
|
||||||
PrintHex<uint8_t>(disc_bdaddr[i]);
|
|
||||||
Notify(PSTR(","));
|
|
||||||
}
|
|
||||||
Notify(PSTR("0x"));
|
|
||||||
PrintHex<uint8_t>(disc_bdaddr[0]);
|
|
||||||
Notify(PSTR(");"));
|
|
||||||
#endif
|
#endif
|
||||||
hci_state = HCI_CONNECT_WII_STATE;
|
hci_state = HCI_CONNECT_WII_STATE;
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case HCI_CONNECT_WII_STATE:
|
case HCI_CONNECT_WII_STATE:
|
||||||
if(!hci_wii_found || hci_cmd_complete) {
|
if(hci_cmd_complete) {
|
||||||
#ifdef DEBUG
|
#ifdef DEBUG
|
||||||
Notify(PSTR("\r\nConnecting to Wiimote"));
|
Notify(PSTR("\r\nConnecting to Wiimote"));
|
||||||
#endif
|
#endif
|
||||||
|
@ -602,8 +604,9 @@ void BTD::HCI_task() {
|
||||||
if(hci_connect_complete) {
|
if(hci_connect_complete) {
|
||||||
#ifdef DEBUG
|
#ifdef DEBUG
|
||||||
Notify(PSTR("\r\nConnected to Wiimote"));
|
Notify(PSTR("\r\nConnected to Wiimote"));
|
||||||
#endif
|
#endif
|
||||||
connectToWii = true; // Only send the ACL data to the Wii service
|
hci_authentication_request(); // This will start the pairing with the wiimote
|
||||||
|
//connectToWii = true;
|
||||||
hci_state = HCI_SCANNING_STATE;
|
hci_state = HCI_SCANNING_STATE;
|
||||||
} else {
|
} else {
|
||||||
#ifdef DEBUG
|
#ifdef DEBUG
|
||||||
|
@ -615,7 +618,7 @@ void BTD::HCI_task() {
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case HCI_SCANNING_STATE:
|
case HCI_SCANNING_STATE:
|
||||||
if(!connectToWii) {
|
if(!connectToWii && !pairWithWii) {
|
||||||
#ifdef DEBUG
|
#ifdef DEBUG
|
||||||
Notify(PSTR("\r\nWait For Incoming Connection Request"));
|
Notify(PSTR("\r\nWait For Incoming Connection Request"));
|
||||||
#endif
|
#endif
|
||||||
|
@ -663,6 +666,10 @@ void BTD::HCI_task() {
|
||||||
PrintHex<uint8_t>(disc_bdaddr[0]);
|
PrintHex<uint8_t>(disc_bdaddr[0]);
|
||||||
#endif
|
#endif
|
||||||
l2capConnectionClaimed = false;
|
l2capConnectionClaimed = false;
|
||||||
|
if(strncmp((const char *)remote_name, "Nintendo", 8) == 0) {
|
||||||
|
Serial.print("\r\nWiimote is connecting");
|
||||||
|
incomingWii = true;
|
||||||
|
}
|
||||||
hci_event_flag = 0;
|
hci_event_flag = 0;
|
||||||
hci_state = HCI_DONE_STATE;
|
hci_state = HCI_DONE_STATE;
|
||||||
}
|
}
|
||||||
|
@ -849,7 +856,7 @@ void BTD::hci_connect() {
|
||||||
|
|
||||||
HCI_Command(hcibuf, 16);
|
HCI_Command(hcibuf, 16);
|
||||||
}
|
}
|
||||||
void BTD::hci_pin_code_request_reply(const char* key) {
|
void BTD::hci_pin_code_request_reply() {
|
||||||
hcibuf[0] = 0x0D; // HCI OCF = 0D
|
hcibuf[0] = 0x0D; // HCI OCF = 0D
|
||||||
hcibuf[1] = 0x01 << 2; // HCI OGF = 1
|
hcibuf[1] = 0x01 << 2; // HCI OGF = 1
|
||||||
hcibuf[2] = 0x17; // parameter length 23
|
hcibuf[2] = 0x17; // parameter length 23
|
||||||
|
@ -858,14 +865,25 @@ void BTD::hci_pin_code_request_reply(const char* key) {
|
||||||
hcibuf[5] = disc_bdaddr[2];
|
hcibuf[5] = disc_bdaddr[2];
|
||||||
hcibuf[6] = disc_bdaddr[3];
|
hcibuf[6] = disc_bdaddr[3];
|
||||||
hcibuf[7] = disc_bdaddr[4];
|
hcibuf[7] = disc_bdaddr[4];
|
||||||
hcibuf[8] = disc_bdaddr[5];
|
hcibuf[8] = disc_bdaddr[5];
|
||||||
hcibuf[9] = strlen(key); // Length of key
|
if(pairWithWii) {
|
||||||
uint8_t i;
|
hcibuf[9] = 6; // Pin length is the length of the bt address
|
||||||
for(i = 0; i < strlen(key); i++) // The maximum size of the key is 16
|
hcibuf[10] = disc_bdaddr[5]; // The pin is the Wiimotes bt address backwards
|
||||||
hcibuf[i+10] = key[i];
|
hcibuf[11] = disc_bdaddr[4];
|
||||||
for(;i < 16; i++)
|
hcibuf[12] = disc_bdaddr[3];
|
||||||
hcibuf[i+10] = 0x00; // The rest should be 0
|
hcibuf[13] = disc_bdaddr[2];
|
||||||
|
hcibuf[14] = disc_bdaddr[1];
|
||||||
|
hcibuf[15] = disc_bdaddr[0];
|
||||||
|
for(uint8_t i = 16; i < 26; i++)
|
||||||
|
hcibuf[i] = 0x00; // The rest should be 0
|
||||||
|
} else {
|
||||||
|
hcibuf[9] = strlen(btdPin); // Length of key
|
||||||
|
uint8_t i;
|
||||||
|
for(i = 0; i < strlen(btdPin); i++) // The maximum size of the key is 16
|
||||||
|
hcibuf[i+10] = btdPin[i];
|
||||||
|
for(;i < 16; i++)
|
||||||
|
hcibuf[i+10] = 0x00; // The rest should be 0
|
||||||
|
}
|
||||||
HCI_Command(hcibuf, 26);
|
HCI_Command(hcibuf, 26);
|
||||||
}
|
}
|
||||||
void BTD::hci_pin_code_negative_request_reply() {
|
void BTD::hci_pin_code_negative_request_reply() {
|
||||||
|
@ -894,6 +912,15 @@ void BTD::hci_link_key_request_negative_reply() {
|
||||||
|
|
||||||
HCI_Command(hcibuf, 9);
|
HCI_Command(hcibuf, 9);
|
||||||
}
|
}
|
||||||
|
void BTD::hci_authentication_request() {
|
||||||
|
hcibuf[0] = 0x11; // HCI OCF = 11
|
||||||
|
hcibuf[1] = 0x01 << 2; // HCI OGF = 1
|
||||||
|
hcibuf[2] = 0x02; // parameter length = 2
|
||||||
|
hcibuf[3] = (uint8_t)(hci_handle & 0xFF);//connection handle - low byte
|
||||||
|
hcibuf[4] = (uint8_t)((hci_handle >> 8) & 0x0F);//connection handle - high byte
|
||||||
|
|
||||||
|
HCI_Command(hcibuf, 5);
|
||||||
|
}
|
||||||
void BTD::hci_disconnect(uint16_t handle) { // This is called by the different services
|
void BTD::hci_disconnect(uint16_t handle) { // This is called by the different services
|
||||||
hci_event_flag &= ~HCI_FLAG_DISCONN_COMPLETE;
|
hci_event_flag &= ~HCI_FLAG_DISCONN_COMPLETE;
|
||||||
hcibuf[0] = 0x06; // HCI OCF = 6
|
hcibuf[0] = 0x06; // HCI OCF = 6
|
||||||
|
|
10
BTD.h
10
BTD.h
|
@ -44,7 +44,7 @@
|
||||||
#define HCI_SET_NAME_STATE 4
|
#define HCI_SET_NAME_STATE 4
|
||||||
#define HCI_CHECK_WII_SERVICE 5
|
#define HCI_CHECK_WII_SERVICE 5
|
||||||
|
|
||||||
#define HCI_INQUIRY_STATE 6 // These three states are only used if it should connect to a Wii controller
|
#define HCI_INQUIRY_STATE 6 // These three states are only used if it should pair and connect to a Wii controller
|
||||||
#define HCI_CONNECT_WII_STATE 7
|
#define HCI_CONNECT_WII_STATE 7
|
||||||
#define HCI_CONNECTED_WII_STATE 8
|
#define HCI_CONNECTED_WII_STATE 8
|
||||||
|
|
||||||
|
@ -180,9 +180,12 @@ public:
|
||||||
uint8_t disc_bdaddr[6]; // Last incoming devices Bluetooth address
|
uint8_t disc_bdaddr[6]; // Last incoming devices Bluetooth address
|
||||||
uint8_t remote_name[30]; // First 30 chars of last remote name
|
uint8_t remote_name[30]; // First 30 chars of last remote name
|
||||||
uint8_t hci_version;
|
uint8_t hci_version;
|
||||||
|
|
||||||
int8_t wiiServiceID; // Stores the service ID of the Wii service
|
int8_t wiiServiceID; // Stores the service ID of the Wii service
|
||||||
|
|
||||||
bool connectToWii; // Used to only send the ACL data to the wiimote
|
bool connectToWii; // Used to only send the ACL data to the wiimote
|
||||||
|
bool incomingWii;
|
||||||
|
bool pairWithWii;
|
||||||
|
|
||||||
/* HCI Commands */
|
/* HCI Commands */
|
||||||
void HCI_Command(uint8_t* data, uint16_t nbytes);
|
void HCI_Command(uint8_t* data, uint16_t nbytes);
|
||||||
|
@ -195,9 +198,10 @@ public:
|
||||||
void hci_accept_connection();
|
void hci_accept_connection();
|
||||||
void hci_write_scan_disable();
|
void hci_write_scan_disable();
|
||||||
void hci_disconnect(uint16_t handle);
|
void hci_disconnect(uint16_t handle);
|
||||||
void hci_pin_code_request_reply(const char* key);
|
void hci_pin_code_request_reply();
|
||||||
void hci_pin_code_negative_request_reply();
|
void hci_pin_code_negative_request_reply();
|
||||||
void hci_link_key_request_negative_reply();
|
void hci_link_key_request_negative_reply();
|
||||||
|
void hci_authentication_request();
|
||||||
void hci_inquiry();
|
void hci_inquiry();
|
||||||
void hci_inquiry_cancel();
|
void hci_inquiry_cancel();
|
||||||
void hci_connect();
|
void hci_connect();
|
||||||
|
|
98
Wii.cpp
98
Wii.cpp
|
@ -20,18 +20,13 @@
|
||||||
//#define EXTRADEBUG // Uncomment to get even more debugging data
|
//#define EXTRADEBUG // Uncomment to get even more debugging data
|
||||||
//#define PRINTREPORT // Uncomment to print the report send by the Wii controllers
|
//#define PRINTREPORT // Uncomment to print the report send by the Wii controllers
|
||||||
|
|
||||||
WII::WII(BTD *p, uint8_t btadr5, uint8_t btadr4, uint8_t btadr3, uint8_t btadr2, uint8_t btadr1, uint8_t btadr0):
|
WII::WII(BTD *p, bool pair):
|
||||||
pBtd(p) // pointer to USB class instance - mandatory
|
pBtd(p) // pointer to USB class instance - mandatory
|
||||||
{
|
{
|
||||||
if (pBtd)
|
if (pBtd)
|
||||||
pBtd->wiiServiceID = pBtd->registerServiceClass(this); // Register it as a Bluetooth service
|
pBtd->wiiServiceID = pBtd->registerServiceClass(this); // Register it as a Bluetooth service
|
||||||
|
|
||||||
pBtd->disc_bdaddr[5] = btadr5; // Change to your dongle's Bluetooth address instead
|
pBtd->pairWithWii = pair;
|
||||||
pBtd->disc_bdaddr[4] = btadr4;
|
|
||||||
pBtd->disc_bdaddr[3] = btadr3;
|
|
||||||
pBtd->disc_bdaddr[2] = btadr2;
|
|
||||||
pBtd->disc_bdaddr[1] = btadr1;
|
|
||||||
pBtd->disc_bdaddr[0] = btadr0;
|
|
||||||
|
|
||||||
HIDBuffer[0] = 0xA2;// HID BT DATA_request (0x50) | Report Type (Output 0x02)
|
HIDBuffer[0] = 0xA2;// HID BT DATA_request (0x50) | Report Type (Output 0x02)
|
||||||
|
|
||||||
|
@ -47,6 +42,7 @@ void WII::Reset() {
|
||||||
wiimoteConnected = false;
|
wiimoteConnected = false;
|
||||||
nunchuckConnected = false;
|
nunchuckConnected = false;
|
||||||
motionPlusConnected = false;
|
motionPlusConnected = false;
|
||||||
|
activateNunchuck = false;
|
||||||
l2cap_event_flag = 0; // Reset flags
|
l2cap_event_flag = 0; // Reset flags
|
||||||
l2cap_state = L2CAP_WAIT;
|
l2cap_state = L2CAP_WAIT;
|
||||||
}
|
}
|
||||||
|
@ -59,7 +55,7 @@ void WII::disconnect() { // Use this void to disconnect any of the controllers
|
||||||
}
|
}
|
||||||
|
|
||||||
void WII::ACLData(uint8_t* l2capinbuf) {
|
void WII::ACLData(uint8_t* l2capinbuf) {
|
||||||
if (((l2capinbuf[0] | (l2capinbuf[1] << 8)) == (hci_handle | 0x2000))) { //acl_handle_ok
|
if (((l2capinbuf[0] | (l2capinbuf[1] << 8)) == (hci_handle | 0x2000)) || pBtd->incomingWii) { // acl_handle_ok or it's a new connection
|
||||||
if ((l2capinbuf[6] | (l2capinbuf[7] << 8)) == 0x0001) { //l2cap_control - Channel ID for ACL-U
|
if ((l2capinbuf[6] | (l2capinbuf[7] << 8)) == 0x0001) { //l2cap_control - Channel ID for ACL-U
|
||||||
if (l2capinbuf[8] == L2CAP_CMD_COMMAND_REJECT) {
|
if (l2capinbuf[8] == L2CAP_CMD_COMMAND_REJECT) {
|
||||||
#ifdef DEBUG
|
#ifdef DEBUG
|
||||||
|
@ -76,7 +72,7 @@ void WII::ACLData(uint8_t* l2capinbuf) {
|
||||||
Notify(PSTR(" "));
|
Notify(PSTR(" "));
|
||||||
PrintHex<uint8_t>(l2capinbuf[14]);
|
PrintHex<uint8_t>(l2capinbuf[14]);
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
else if (l2capinbuf[8] == L2CAP_CMD_CONNECTION_RESPONSE) {
|
else if (l2capinbuf[8] == L2CAP_CMD_CONNECTION_RESPONSE) {
|
||||||
if (((l2capinbuf[16] | (l2capinbuf[17] << 8)) == 0x0000) && ((l2capinbuf[18] | (l2capinbuf[19] << 8)) == SUCCESSFUL)) { // Success
|
if (((l2capinbuf[16] | (l2capinbuf[17] << 8)) == 0x0000) && ((l2capinbuf[18] | (l2capinbuf[19] << 8)) == SUCCESSFUL)) { // Success
|
||||||
if (l2capinbuf[14] == control_dcid[0] && l2capinbuf[15] == control_dcid[1]) { // Success
|
if (l2capinbuf[14] == control_dcid[0] && l2capinbuf[15] == control_dcid[1]) { // Success
|
||||||
|
@ -94,6 +90,32 @@ void WII::ACLData(uint8_t* l2capinbuf) {
|
||||||
l2cap_event_flag |= L2CAP_FLAG_INTERRUPT_CONNECTED;
|
l2cap_event_flag |= L2CAP_FLAG_INTERRUPT_CONNECTED;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
else if (l2capinbuf[8] == L2CAP_CMD_CONNECTION_REQUEST) {
|
||||||
|
#ifdef EXTRADEBUG
|
||||||
|
Notify(PSTR("\r\nL2CAP Connection Request - PSM: "));
|
||||||
|
PrintHex<uint8_t>(l2capinbuf[13]);
|
||||||
|
Notify(PSTR(" "));
|
||||||
|
PrintHex<uint8_t>(l2capinbuf[12]);
|
||||||
|
Notify(PSTR(" SCID: "));
|
||||||
|
PrintHex<uint8_t>(l2capinbuf[15]);
|
||||||
|
Notify(PSTR(" "));
|
||||||
|
PrintHex<uint8_t>(l2capinbuf[14]);
|
||||||
|
Notify(PSTR(" Identifier: "));
|
||||||
|
PrintHex<uint8_t>(l2capinbuf[9]);
|
||||||
|
#endif
|
||||||
|
if ((l2capinbuf[12] | (l2capinbuf[13] << 8)) == HID_CTRL_PSM) {
|
||||||
|
identifier = l2capinbuf[9];
|
||||||
|
control_scid[0] = l2capinbuf[14];
|
||||||
|
control_scid[1] = l2capinbuf[15];
|
||||||
|
l2cap_event_flag |= L2CAP_FLAG_CONNECTION_CONTROL_REQUEST;
|
||||||
|
}
|
||||||
|
else if ((l2capinbuf[12] | (l2capinbuf[13] << 8)) == HID_INTR_PSM) {
|
||||||
|
identifier = l2capinbuf[9];
|
||||||
|
interrupt_scid[0] = l2capinbuf[14];
|
||||||
|
interrupt_scid[1] = l2capinbuf[15];
|
||||||
|
l2cap_event_flag |= L2CAP_FLAG_CONNECTION_INTERRUPT_REQUEST;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
else if (l2capinbuf[8] == L2CAP_CMD_CONFIG_RESPONSE) {
|
else if (l2capinbuf[8] == L2CAP_CMD_CONFIG_RESPONSE) {
|
||||||
if ((l2capinbuf[16] | (l2capinbuf[17] << 8)) == 0x0000) { // Success
|
if ((l2capinbuf[16] | (l2capinbuf[17] << 8)) == 0x0000) { // Success
|
||||||
|
@ -410,6 +432,33 @@ void WII::ACLData(uint8_t* l2capinbuf) {
|
||||||
}
|
}
|
||||||
void WII::L2CAP_task() {
|
void WII::L2CAP_task() {
|
||||||
switch (l2cap_state) {
|
switch (l2cap_state) {
|
||||||
|
/* These states are used if the Wiimote is the host */
|
||||||
|
case L2CAP_CONTROL_SUCCESS:
|
||||||
|
if (l2cap_config_success_control_flag) {
|
||||||
|
#ifdef DEBUG
|
||||||
|
Notify(PSTR("\r\nHID Control Successfully Configured"));
|
||||||
|
#endif
|
||||||
|
l2cap_state = L2CAP_INTERRUPT_SETUP;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
|
case L2CAP_INTERRUPT_SETUP:
|
||||||
|
if (l2cap_connection_request_interrupt_flag) {
|
||||||
|
#ifdef DEBUG
|
||||||
|
Notify(PSTR("\r\nHID Interrupt Incoming Connection Request"));
|
||||||
|
#endif
|
||||||
|
pBtd->l2cap_connection_response(hci_handle,identifier, interrupt_dcid, interrupt_scid, PENDING);
|
||||||
|
delay(1);
|
||||||
|
pBtd->l2cap_connection_response(hci_handle,identifier, interrupt_dcid, interrupt_scid, SUCCESSFUL);
|
||||||
|
identifier++;
|
||||||
|
delay(1);
|
||||||
|
pBtd->l2cap_config_request(hci_handle,identifier, interrupt_scid);
|
||||||
|
|
||||||
|
l2cap_state = L2CAP_INTERRUPT_CONFIG_REQUEST;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
|
/* These states are used if the Arduino is the host */
|
||||||
case L2CAP_CONTROL_CONNECT_REQUEST:
|
case L2CAP_CONTROL_CONNECT_REQUEST:
|
||||||
if (l2cap_connected_control_flag) {
|
if (l2cap_connected_control_flag) {
|
||||||
#ifdef DEBUG
|
#ifdef DEBUG
|
||||||
|
@ -443,18 +492,18 @@ void WII::L2CAP_task() {
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
|
||||||
|
|
||||||
case L2CAP_INTERRUPT_CONFIG_REQUEST:
|
case L2CAP_INTERRUPT_CONFIG_REQUEST:
|
||||||
if(l2cap_config_success_interrupt_flag) {
|
if(l2cap_config_success_interrupt_flag) { // Now the HID channels is established
|
||||||
#ifdef DEBUG
|
#ifdef DEBUG
|
||||||
Notify(PSTR("\r\nHID Channels Established"));
|
Notify(PSTR("\r\nHID Channels Established"));
|
||||||
#endif
|
#endif
|
||||||
pBtd->connectToWii = false;
|
pBtd->connectToWii = false;
|
||||||
|
pBtd->pairWithWii = false;
|
||||||
wiimoteConnected = true;
|
wiimoteConnected = true;
|
||||||
stateCounter = 0;
|
stateCounter = 0;
|
||||||
l2cap_state = L2CAP_CHECK_MOTION_PLUS_STATE;
|
l2cap_state = L2CAP_CHECK_MOTION_PLUS_STATE;
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
|
||||||
/* The next states are in run() */
|
/* The next states are in run() */
|
||||||
|
|
||||||
|
@ -493,10 +542,21 @@ void WII::Run() {
|
||||||
identifier = 0;
|
identifier = 0;
|
||||||
pBtd->l2cap_connection_request(hci_handle,identifier,control_dcid,HID_CTRL_PSM);
|
pBtd->l2cap_connection_request(hci_handle,identifier,control_dcid,HID_CTRL_PSM);
|
||||||
l2cap_state = L2CAP_CONTROL_CONNECT_REQUEST;
|
l2cap_state = L2CAP_CONTROL_CONNECT_REQUEST;
|
||||||
|
} else if (l2cap_connection_request_control_flag) {
|
||||||
|
hci_handle = pBtd->hci_handle;
|
||||||
|
pBtd->incomingWii = false;
|
||||||
|
#ifdef DEBUG
|
||||||
|
Notify(PSTR("\r\nHID Control Incoming Connection Request"));
|
||||||
|
#endif
|
||||||
|
pBtd->l2cap_connection_response(hci_handle,identifier, control_dcid, control_scid, PENDING);
|
||||||
|
delay(1);
|
||||||
|
pBtd->l2cap_connection_response(hci_handle,identifier, control_dcid, control_scid, SUCCESSFUL);
|
||||||
|
identifier++;
|
||||||
|
delay(1);
|
||||||
|
pBtd->l2cap_config_request(hci_handle,identifier, control_scid);
|
||||||
|
l2cap_state = L2CAP_CONTROL_SUCCESS;
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
case L2CAP_CHECK_MOTION_PLUS_STATE:
|
case L2CAP_CHECK_MOTION_PLUS_STATE:
|
||||||
#ifdef DEBUG
|
#ifdef DEBUG
|
||||||
|
@ -552,7 +612,7 @@ void WII::Run() {
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case L2CAP_INIT_MOTION_PLUS_STATE:
|
case L2CAP_INIT_MOTION_PLUS_STATE: {
|
||||||
stateCounter++;
|
stateCounter++;
|
||||||
if(stateCounter == 1)
|
if(stateCounter == 1)
|
||||||
initMotionPlus();
|
initMotionPlus();
|
||||||
|
@ -566,14 +626,16 @@ void WII::Run() {
|
||||||
l2cap_state = L2CAP_LED_STATE;
|
l2cap_state = L2CAP_LED_STATE;
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
case L2CAP_LED_STATE:
|
case L2CAP_LED_STATE: {
|
||||||
if(nunchuck_connected_flag)
|
if(nunchuck_connected_flag)
|
||||||
nunchuckConnected = true;
|
nunchuckConnected = true;
|
||||||
setLedStatus();
|
setLedStatus();
|
||||||
l2cap_state = L2CAP_DONE;
|
l2cap_state = L2CAP_DONE;
|
||||||
break;
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
case L2CAP_DONE:
|
case L2CAP_DONE:
|
||||||
if(unknownExtensionConnected) {
|
if(unknownExtensionConnected) {
|
||||||
#ifdef DEBUG
|
#ifdef DEBUG
|
||||||
|
|
41
Wii.h
41
Wii.h
|
@ -22,19 +22,26 @@
|
||||||
|
|
||||||
/* Bluetooth L2CAP states for L2CAP_task() */
|
/* Bluetooth L2CAP states for L2CAP_task() */
|
||||||
#define L2CAP_WAIT 0
|
#define L2CAP_WAIT 0
|
||||||
#define L2CAP_CONTROL_CONNECT_REQUEST 1
|
|
||||||
#define L2CAP_CONTROL_CONFIG_REQUEST 2
|
|
||||||
#define L2CAP_INTERRUPT_CONNECT_REQUEST 3
|
|
||||||
#define L2CAP_INTERRUPT_CONFIG_REQUEST 4
|
|
||||||
|
|
||||||
#define L2CAP_CHECK_MOTION_PLUS_STATE 5
|
// These states are used if the Wiimote is the host
|
||||||
#define L2CAP_CHECK_EXTENSION_STATE 6
|
#define L2CAP_CONTROL_SUCCESS 1
|
||||||
#define L2CAP_INIT_MOTION_PLUS_STATE 7
|
#define L2CAP_INTERRUPT_SETUP 2
|
||||||
|
|
||||||
#define L2CAP_LED_STATE 8
|
// These states are used if the Arduino is the host
|
||||||
#define L2CAP_DONE 9
|
#define L2CAP_CONTROL_CONNECT_REQUEST 3
|
||||||
#define L2CAP_INTERRUPT_DISCONNECT 10
|
#define L2CAP_CONTROL_CONFIG_REQUEST 4
|
||||||
#define L2CAP_CONTROL_DISCONNECT 11
|
#define L2CAP_INTERRUPT_CONNECT_REQUEST 5
|
||||||
|
|
||||||
|
#define L2CAP_INTERRUPT_CONFIG_REQUEST 6
|
||||||
|
|
||||||
|
#define L2CAP_CHECK_MOTION_PLUS_STATE 7
|
||||||
|
#define L2CAP_CHECK_EXTENSION_STATE 8
|
||||||
|
#define L2CAP_INIT_MOTION_PLUS_STATE 9
|
||||||
|
#define L2CAP_LED_STATE 10
|
||||||
|
#define L2CAP_DONE 11
|
||||||
|
|
||||||
|
#define L2CAP_INTERRUPT_DISCONNECT 12
|
||||||
|
#define L2CAP_CONTROL_DISCONNECT 13
|
||||||
|
|
||||||
/* L2CAP event flags */
|
/* L2CAP event flags */
|
||||||
#define L2CAP_FLAG_CONTROL_CONNECTED 0x001
|
#define L2CAP_FLAG_CONTROL_CONNECTED 0x001
|
||||||
|
@ -43,6 +50,8 @@
|
||||||
#define L2CAP_FLAG_CONFIG_INTERRUPT_SUCCESS 0x008
|
#define L2CAP_FLAG_CONFIG_INTERRUPT_SUCCESS 0x008
|
||||||
#define L2CAP_FLAG_DISCONNECT_CONTROL_RESPONSE 0x040
|
#define L2CAP_FLAG_DISCONNECT_CONTROL_RESPONSE 0x040
|
||||||
#define L2CAP_FLAG_DISCONNECT_INTERRUPT_RESPONSE 0x080
|
#define L2CAP_FLAG_DISCONNECT_INTERRUPT_RESPONSE 0x080
|
||||||
|
#define L2CAP_FLAG_CONNECTION_CONTROL_REQUEST 0x100
|
||||||
|
#define L2CAP_FLAG_CONNECTION_INTERRUPT_REQUEST 0x200
|
||||||
|
|
||||||
/* Macros for L2CAP event flag tests */
|
/* Macros for L2CAP event flag tests */
|
||||||
#define l2cap_connected_control_flag (l2cap_event_flag & L2CAP_FLAG_CONTROL_CONNECTED)
|
#define l2cap_connected_control_flag (l2cap_event_flag & L2CAP_FLAG_CONTROL_CONNECTED)
|
||||||
|
@ -51,14 +60,18 @@
|
||||||
#define l2cap_config_success_interrupt_flag (l2cap_event_flag & L2CAP_FLAG_CONFIG_INTERRUPT_SUCCESS)
|
#define l2cap_config_success_interrupt_flag (l2cap_event_flag & L2CAP_FLAG_CONFIG_INTERRUPT_SUCCESS)
|
||||||
#define l2cap_disconnect_response_control_flag (l2cap_event_flag & L2CAP_FLAG_DISCONNECT_CONTROL_RESPONSE)
|
#define l2cap_disconnect_response_control_flag (l2cap_event_flag & L2CAP_FLAG_DISCONNECT_CONTROL_RESPONSE)
|
||||||
#define l2cap_disconnect_response_interrupt_flag (l2cap_event_flag & L2CAP_FLAG_DISCONNECT_INTERRUPT_RESPONSE)
|
#define l2cap_disconnect_response_interrupt_flag (l2cap_event_flag & L2CAP_FLAG_DISCONNECT_INTERRUPT_RESPONSE)
|
||||||
|
#define l2cap_connection_request_control_flag (l2cap_event_flag & L2CAP_FLAG_CONNECTION_CONTROL_REQUEST)
|
||||||
|
#define l2cap_connection_request_interrupt_flag (l2cap_event_flag & L2CAP_FLAG_CONNECTION_INTERRUPT_REQUEST)
|
||||||
|
|
||||||
/* Wii event flags */
|
/* Wii event flags */
|
||||||
#define WII_FLAG_MOTION_PLUS_CONNECTED 0x100
|
#define WII_FLAG_MOTION_PLUS_CONNECTED 0x400
|
||||||
#define WII_FLAG_NUNCHUCK_CONNECTED 0x200
|
#define WII_FLAG_NUNCHUCK_CONNECTED 0x800
|
||||||
|
|
||||||
#define motion_plus_connected_flag (l2cap_event_flag & WII_FLAG_MOTION_PLUS_CONNECTED)
|
#define motion_plus_connected_flag (l2cap_event_flag & WII_FLAG_MOTION_PLUS_CONNECTED)
|
||||||
#define nunchuck_connected_flag (l2cap_event_flag & WII_FLAG_NUNCHUCK_CONNECTED)
|
#define nunchuck_connected_flag (l2cap_event_flag & WII_FLAG_NUNCHUCK_CONNECTED)
|
||||||
|
|
||||||
|
#define PAIR 1
|
||||||
|
|
||||||
enum LED {
|
enum LED {
|
||||||
LED1 = 0x10,
|
LED1 = 0x10,
|
||||||
LED2 = 0x20,
|
LED2 = 0x20,
|
||||||
|
@ -97,7 +110,7 @@ enum AnalogHat {
|
||||||
|
|
||||||
class WII : public BluetoothService {
|
class WII : public BluetoothService {
|
||||||
public:
|
public:
|
||||||
WII(BTD *pBtd, uint8_t btadr5=0, uint8_t btadr4=0, uint8_t btadr3=0, uint8_t btadr2=0, uint8_t btadr1=0, uint8_t btadr0=0);
|
WII(BTD *p, bool pair=false);
|
||||||
|
|
||||||
// BluetoothService implementation
|
// BluetoothService implementation
|
||||||
virtual void ACLData(uint8_t* ACLData); // Used to pass acldata to the services
|
virtual void ACLData(uint8_t* ACLData); // Used to pass acldata to the services
|
||||||
|
|
|
@ -222,4 +222,5 @@ motionPlusConnected KEYWORD2
|
||||||
setRumbleToggle KEYWORD2
|
setRumbleToggle KEYWORD2
|
||||||
getPitch KEYWORD2
|
getPitch KEYWORD2
|
||||||
getRoll KEYWORD2
|
getRoll KEYWORD2
|
||||||
getYaw KEYWORD2
|
getYaw KEYWORD2
|
||||||
|
PAIR KEYWORD2
|
Loading…
Reference in a new issue