From 79dd1d6952d8087b473383a2e2205377a827c7dc Mon Sep 17 00:00:00 2001 From: Kristian Lauszus Date: Wed, 11 Apr 2012 01:48:46 +0200 Subject: [PATCH 1/9] Minor edit --- examples/adk/ArduinoBlinkLED/ArduinoBlinkLED.ino | 9 ++++----- 1 file changed, 4 insertions(+), 5 deletions(-) diff --git a/examples/adk/ArduinoBlinkLED/ArduinoBlinkLED.ino b/examples/adk/ArduinoBlinkLED/ArduinoBlinkLED.ino index 303335af..19433c61 100644 --- a/examples/adk/ArduinoBlinkLED/ArduinoBlinkLED.ino +++ b/examples/adk/ArduinoBlinkLED/ArduinoBlinkLED.ino @@ -12,7 +12,7 @@ ADK adk(&Usb,"TKJElectronics", // Manufacturer Name "http://www.tkjelectronics.dk/uploads/ArduinoBlinkLED.apk", // URL (web page to visit if no installed apps support the accessory) "123456789"); // Serial Number (optional) -#define LED_RED 13 // Pin 13 is occupied by the SCK pin on a normal Arduino (Uno, Duemilanove etc.), so use a different pin +#define LED 13 // Pin 13 is occupied by the SCK pin on a normal Arduino (Uno, Duemilanove etc.), so use a different pin void setup() { @@ -22,7 +22,7 @@ void setup() Serial.print("\r\nOSCOKIRQ failed to assert"); while(1); //halt } - pinMode(LED_RED, OUTPUT); + pinMode(LED, OUTPUT); } void loop() @@ -37,10 +37,9 @@ void loop() if(len > 0) { Serial.print(F("\r\nData Packet: ")); Serial.print(msg[0]); - digitalWrite(LED_RED,msg[0] ? HIGH : LOW); + digitalWrite(LED,msg[0] ? HIGH : LOW); } } else - digitalWrite(LED_RED, LOW); - delay(10); + digitalWrite(LED, LOW); } From d4fcf68de4eebb25c97f710246b017b10558d6b1 Mon Sep 17 00:00:00 2001 From: Kristian Lauszus Date: Wed, 11 Apr 2012 01:52:43 +0200 Subject: [PATCH 2/9] Minor revision The major difference is the potential support for different kind of dongles --- PS3BT.cpp | 686 ++++++++++++++++++++++++++++----------------------- PS3BT.h | 200 ++++++++------- keywords.txt | 7 +- 3 files changed, 495 insertions(+), 398 deletions(-) diff --git a/PS3BT.cpp b/PS3BT.cpp index c55df46f..363237bb 100644 --- a/PS3BT.cpp +++ b/PS3BT.cpp @@ -19,6 +19,10 @@ #define DEBUG // Uncomment to print data for debugging //#define PRINTREPORT // Uncomment to print the report send by the PS3 Controllers +const uint8_t PS3BT::BTD_EVENT_PIPE = 1; +const uint8_t PS3BT::BTD_DATAIN_PIPE = 2; +const uint8_t PS3BT::BTD_DATAOUT_PIPE = 3; + prog_char OUTPUT_REPORT_BUFFER[] PROGMEM = { 0x00, 0x00, 0x00, 0x00, 0x00, @@ -33,23 +37,24 @@ prog_char OUTPUT_REPORT_BUFFER[] PROGMEM = }; PS3BT::PS3BT(USB *p): - pUsb(p), //pointer to USB class instance - mandatory - bAddress(0), //device address - mandatory - bPollEnable(false) //don't start polling before dongle is connected + pUsb(p), // pointer to USB class instance - mandatory + bAddress(0), // device address - mandatory + bNumEP(1), // if config descriptor needs to be parsed + qNextPollTime(0), + bPollEnable(false) // don't start polling before dongle is connected { for(uint8_t i=0; iRegisterDeviceClass(this); //set devConfig[] entry - my_bdaddr[5] = 0x00;//Change to your dongle's Bluetooth address instead + my_bdaddr[5] = 0x00; // Change to your dongle's Bluetooth address instead my_bdaddr[4] = 0x1F; my_bdaddr[3] = 0x81; my_bdaddr[2] = 0x00; @@ -59,11 +64,11 @@ PS3BT::PS3BT(USB *p): uint8_t PS3BT::Init(uint8_t parent, uint8_t port, bool lowspeed) { - const uint8_t constBufSize = sizeof(USB_DEVICE_DESCRIPTOR); - uint8_t buf[constBufSize]; + uint8_t buf[sizeof(USB_DEVICE_DESCRIPTOR)]; uint8_t rcode; UsbDevice *p = NULL; - EpInfo *oldep_ptr = NULL; + EpInfo *oldep_ptr = NULL; + uint8_t num_of_conf; // number of configurations uint16_t PID; uint16_t VID; @@ -109,14 +114,13 @@ uint8_t PS3BT::Init(uint8_t parent, uint8_t port, bool lowspeed) p->lowspeed = lowspeed; // Get device descriptor - rcode = pUsb->getDevDescr(0, 0, constBufSize, (uint8_t*)buf);// Get device descriptor - addr, ep, nbytes, data + rcode = pUsb->getDevDescr(0, 0, sizeof(USB_DEVICE_DESCRIPTOR), (uint8_t*)buf);// Get device descriptor - addr, ep, nbytes, data // Restore p->epinfo p->epinfo = oldep_ptr; - if( rcode ){ + if(rcode) goto FailGetDevDescr; - } // Allocate new address according to device class bAddress = addrPool.AllocAddress(parent, false, port); @@ -148,27 +152,25 @@ uint8_t PS3BT::Init(uint8_t parent, uint8_t port, bool lowspeed) //get pointer to assigned address record p = addrPool.GetUsbDevicePtr(bAddress); if (!p) - { return USB_ERROR_ADDRESS_NOT_FOUND_IN_POOL; - } - p->lowspeed = lowspeed; + p->lowspeed = lowspeed; // Assign epInfo to epinfo pointer - only EP0 is known rcode = pUsb->setEpInfoEntry(bAddress, 1, epInfo); - if (rcode) - { + if (rcode) goto FailSetDevTblEntry; - } + num_of_conf = ((USB_DEVICE_DESCRIPTOR*)buf)->bNumConfigurations; VID = ((USB_DEVICE_DESCRIPTOR*)buf)->idVendor; PID = ((USB_DEVICE_DESCRIPTOR*)buf)->idProduct; - if(VID == CSR_VID && PID == CSR_PID) + if((VID == CSR_VID || VID == ISSC_VID) && (PID == CSR_PID || PID == ISSC_PID)) { #ifdef DEBUG Notify(PSTR("\r\nBluetooth Dongle Connected")); #endif + //Needed for PS3 Dualshock Controller commands to work via bluetooth for (uint8_t i = 0; i < OUTPUT_REPORT_BUFFER_SIZE; i++) HIDBuffer[i + 2] = pgm_read_byte(&OUTPUT_REPORT_BUFFER[i]);//First two bytes reserved for report type and ID @@ -186,33 +188,37 @@ uint8_t PS3BT::Init(uint8_t parent, uint8_t port, bool lowspeed) interrupt_dcid[0] = 0x41;//0x0041 interrupt_dcid[1] = 0x00; - /* Initialize data structures for endpoints of device */ - epInfo[ CSR_EVENT_PIPE ].epAddr = 0x01; // Bluetooth event endpoint - epInfo[ CSR_EVENT_PIPE ].epAttribs = EP_INTERRUPT; - epInfo[ CSR_EVENT_PIPE ].bmNakPower = USB_NAK_NOWAIT; // Only poll once for interrupt endpoints - epInfo[ CSR_EVENT_PIPE ].maxPktSize = INT_MAXPKTSIZE; - epInfo[ CSR_EVENT_PIPE ].bmSndToggle = bmSNDTOG0; - epInfo[ CSR_EVENT_PIPE ].bmRcvToggle = bmRCVTOG0; - epInfo[ CSR_DATAIN_PIPE ].epAddr = 0x02; // Bluetoth data endpoint - epInfo[ CSR_DATAIN_PIPE ].epAttribs = EP_BULK; - epInfo[ CSR_DATAIN_PIPE ].bmNakPower = USB_NAK_NOWAIT; - epInfo[ CSR_DATAIN_PIPE ].maxPktSize = BULK_MAXPKTSIZE; - epInfo[ CSR_DATAIN_PIPE ].bmSndToggle = bmSNDTOG0; - epInfo[ CSR_DATAIN_PIPE ].bmRcvToggle = bmRCVTOG0; - epInfo[ CSR_DATAOUT_PIPE ].epAddr = 0x02; // Bluetooth data endpoint - epInfo[ CSR_DATAOUT_PIPE ].epAttribs = EP_BULK; - epInfo[ CSR_DATAOUT_PIPE ].bmNakPower = USB_NAK_NOWAIT; - epInfo[ CSR_DATAOUT_PIPE ].maxPktSize = BULK_MAXPKTSIZE; - epInfo[ CSR_DATAOUT_PIPE ].bmSndToggle = bmSNDTOG0; - epInfo[ CSR_DATAOUT_PIPE ].bmRcvToggle = bmRCVTOG0; + //check if attached device is a Bluetooth dongle and fill endpoint data structure + //first interface in the configuration must have Bluetooth assigned Class/Subclass/Protocol + //and 3 endpoints - interrupt-IN, bulk-IN, bulk-OUT, + //not necessarily in this order + for (uint8_t i=0; i confDescrParser(this); + rcode = pUsb->getConfDescr(bAddress, 0, i, &confDescrParser); + if( rcode ) { + goto FailGetConfDescr; + } + if( bNumEP > 3 ) { //all endpoints extracted + break; + } + } // for (uint8_t i=0; isetEpInfoEntry(bAddress, 3, epInfo); + if (bNumEP < PS3_MAX_ENDPOINTS) { + Notify(PSTR("\r\nBluetooth dongle is not supported")); + goto Fail; + } + + // Assign epInfo to epinfo pointer - this time all 3 endpoins + rcode = pUsb->setEpInfoEntry(bAddress, bNumEP, epInfo); if( rcode ) - goto FailSetDevTblEntry; + goto FailSetDevTblEntry; - delay(200);//Give time for address change + delay(200); // Give time for address change - rcode = pUsb->setConf(bAddress, epInfo[ CSR_CONTROL_PIPE ].epAddr, bConfigurationValue);//bConfigurationValue = 0x01 + //rcode = pUsb->setConf(bAddress, epInfo[ CSR_CONTROL_PIPE ].epAddr, bConfigurationValue);//bConfigurationValue = 0x01 + + // Set Configuration Value + rcode = pUsb->setConf(bAddress, epInfo[ BTD_CONTROL_PIPE ].epAddr, bConfNum); if( rcode ) goto FailSetConf; @@ -222,12 +228,16 @@ uint8_t PS3BT::Init(uint8_t parent, uint8_t port, bool lowspeed) #ifdef DEBUG Notify(PSTR("\r\nCSR Initialized")); #endif - delay(200); - + + watingForConnection = false; bPollEnable = true; } else if((VID == PS3_VID || VID == PS3NAVIGATION_VID || VID == PS3MOVE_VID) && (PID == PS3_PID || PID == PS3NAVIGATION_PID || PID == PS3MOVE_PID)) { + /*The application will work in reduced host mode, so we can save program and data + memory space. After verifying the PID and VID we will use known values for the + configuration values for device, interface, endpoints and HID for the PS3 Controllers */ + /* Initialize data structures for endpoints of device */ epInfo[ PS3_OUTPUT_PIPE ].epAddr = 0x02; // PS3 output endpoint epInfo[ PS3_OUTPUT_PIPE ].epAttribs = EP_INTERRUPT; @@ -241,8 +251,7 @@ uint8_t PS3BT::Init(uint8_t parent, uint8_t port, bool lowspeed) epInfo[ PS3_INPUT_PIPE ].maxPktSize = EP_MAXPKTSIZE; epInfo[ PS3_INPUT_PIPE ].bmSndToggle = bmSNDTOG0; epInfo[ PS3_INPUT_PIPE ].bmRcvToggle = bmRCVTOG0; - - + rcode = pUsb->setEpInfoEntry(bAddress, 3, epInfo); if( rcode ) goto FailSetDevTblEntry; @@ -292,6 +301,11 @@ uint8_t PS3BT::Init(uint8_t parent, uint8_t port, bool lowspeed) Notify(PSTR("\r\nsetDevTblEn:")); #endif goto Fail; + FailGetConfDescr: + #ifdef DEBUG + Notify(PSTR("\r\ngetConf:")); + #endif + goto Fail; FailSetConf: #ifdef DEBUG Notify(PSTR("\r\nsetConf:")); @@ -313,6 +327,55 @@ uint8_t PS3BT::Init(uint8_t parent, uint8_t port, bool lowspeed) Release(); return rcode; } +/* Extracts interrupt-IN, bulk-IN, bulk-OUT endpoint information from config descriptor */ +void PS3BT::EndpointXtract(uint8_t conf, uint8_t iface, uint8_t alt, uint8_t proto, const USB_ENDPOINT_DESCRIPTOR *pep) +{ + //ErrorMessage(PSTR("Conf.Val"),conf); + //ErrorMessage(PSTR("Iface Num"),iface); + //ErrorMessage(PSTR("Alt.Set"),alt); + + if(alt) // wrong interface - by BT spec, no alt setting + return; + + bConfNum = conf; + uint8_t index; + + if ((pep->bmAttributes & 0x03) == 3 && (pep->bEndpointAddress & 0x80) == 0x80) //Interrupt In endpoint found + index = BTD_EVENT_PIPE; + + else { + if ((pep->bmAttributes & 0x02) == 2) //bulk endpoint found + index = ((pep->bEndpointAddress & 0x80) == 0x80) ? BTD_DATAIN_PIPE : BTD_DATAOUT_PIPE; + else + return; + } + + //Fill the rest of endpoint data structure + epInfo[index].epAddr = (pep->bEndpointAddress & 0x0F); + epInfo[index].maxPktSize = (uint8_t)pep->wMaxPacketSize; + //PrintEndpointDescriptor(pep); + if(pollInterval < pep->bInterval) // Set the polling interval as the largest polling interval obtained from endpoints + pollInterval = pep->bInterval; + bNumEP++; + return; +} +void PS3BT::PrintEndpointDescriptor( const USB_ENDPOINT_DESCRIPTOR* ep_ptr ) +{ + Notify(PSTR("Endpoint descriptor:")); + Notify(PSTR("\r\nLength:\t\t")); + PrintHex(ep_ptr->bLength); + Notify(PSTR("\r\nType:\t\t")); + PrintHex(ep_ptr->bDescriptorType); + Notify(PSTR("\r\nAddress:\t")); + PrintHex(ep_ptr->bEndpointAddress); + Notify(PSTR("\r\nAttributes:\t")); + PrintHex(ep_ptr->bmAttributes); + Notify(PSTR("\r\nMaxPktSize:\t")); + PrintHex(ep_ptr->wMaxPacketSize); + Notify(PSTR("\r\nPoll Intrv:\t")); + PrintHex(ep_ptr->bInterval); + Notify(PSTR("\r\n")); +} /* Performs a cleanup after failed Init() attempt */ uint8_t PS3BT::Release() @@ -320,16 +383,18 @@ uint8_t PS3BT::Release() pUsb->GetAddressPool().FreeAddress(bAddress); bAddress = 0; bPollEnable = false; + bNumEP = 1; // must have to be reset to 1 return 0; } uint8_t PS3BT::Poll() { if (!bPollEnable) return 0; - - HCI_event_task(); // poll the HCI event pipe - ACL_event_task(); // start polling the ACL input pipe too, though discard data until connected - + if (qNextPollTime <= millis()) { // Don't poll if shorter than polling interval + HCI_event_task(); // poll the HCI event pipe + ACL_event_task(); // start polling the ACL input pipe too, though discard data until connected + } + qNextPollTime = millis() + pollInterval; // Poll time return 0; } void PS3BT::setBdaddr(uint8_t* BDADDR) @@ -346,7 +411,7 @@ void PS3BT::setBdaddr(uint8_t* BDADDR) buf[i+2] = my_bdaddr[5 - i];//Copy into buffer, has to be written reversed //bmRequest = Host to device (0x00) | Class (0x20) | Interface (0x01) = 0x21, bRequest = Set Report (0x09), Report ID (0xF5), Report Type (Feature 0x03), interface (0x00), datalength, datalength, data) - pUsb->ctrlReq(bAddress,epInfo[PS3_CONTROL_PIPE].epAddr, bmREQ_HIDOUT, HID_REQUEST_SET_REPORT, 0xF5, 0x03, 0x00, 8, 8, buf, NULL); + pUsb->ctrlReq(bAddress,epInfo[PS3_CONTROL_PIPE].epAddr, bmREQ_HID_OUT, HID_REQUEST_SET_REPORT, 0xF5, 0x03, 0x00, 8, 8, buf, NULL); #ifdef DEBUG Notify(PSTR("\r\nBluetooth Address was set to: ")); for(int8_t i = 5; i > 0; i--) @@ -376,7 +441,7 @@ void PS3BT::setMoveBdaddr(uint8_t* BDADDR) buf[i + 1] = my_bdaddr[i]; //bmRequest = Host to device (0x00) | Class (0x20) | Interface (0x01) = 0x21, bRequest = Set Report (0x09), Report ID (0x05), Report Type (Feature 0x03), interface (0x00), datalength, datalength, data) - pUsb->ctrlReq(bAddress,epInfo[PS3_CONTROL_PIPE].epAddr, bmREQ_HIDOUT, HID_REQUEST_SET_REPORT, 0x05, 0x03, 0x00,11,11, buf, NULL); + pUsb->ctrlReq(bAddress,epInfo[PS3_CONTROL_PIPE].epAddr, bmREQ_HID_OUT, HID_REQUEST_SET_REPORT, 0x05, 0x03, 0x00,11,11, buf, NULL); #ifdef DEBUG Notify(PSTR("\r\nBluetooth Address was set to: ")); for(int8_t i = 5; i > 0; i--) @@ -411,17 +476,13 @@ uint8_t PS3BT::getAnalogHat(AnalogHat a) } uint32_t PS3BT::getSensor(Sensor a) { + if (l2capinbuf == NULL) + return 0; if (a == aX || a == aY || a == aZ || a == gZ) - { - if (l2capinbuf == NULL) - return 0; return ((l2capinbuf[(uint16_t)a] << 8) | l2capinbuf[(uint16_t)a + 1]); - } else if (a == mXmove || a == mYmove || a == mZmove) - { - //Might not be correct, haven't tested it yet - if (l2capinbuf == NULL) - return 0; + { + // Might not be correct, haven't tested it yet if (a == mXmove) return ((l2capinbuf[(uint16_t)a + 1] << 0x04) | (l2capinbuf[(uint16_t)a] << 0x0C)); //return (((unsigned char)l2capinbuf[(unsigned int)a + 1]) | (((unsigned char)l2capinbuf[(unsigned int)a] & 0x0F)) << 8); @@ -442,12 +503,11 @@ uint32_t PS3BT::getSensor(Sensor a) } else { - if (l2capinbuf == NULL) - return 0; + return (((l2capinbuf[(uint16_t)a + 1] << 8) | l2capinbuf[(uint16_t)a]) - 0x8000); } } -double PS3BT::getAngle(Angle a, boolean resolution)//Boolean indicate if 360-degrees resolution is used or not - set false if you want to use both axis +double PS3BT::getAngle(Angle a, boolean resolution) // Boolean indicate if 360-degrees resolution is used or not - set false if you want to use both axis { double accXin; double accXval; @@ -592,88 +652,99 @@ void PS3BT::disconnect()//Use this void to disconnect any of the controllers void PS3BT::HCI_event_task() { /* check the event pipe*/ - uint16_t MAX_BUFFER_SIZE = BULK_MAXPKTSIZE; - pUsb->inTransfer(bAddress, epInfo[ CSR_EVENT_PIPE ].epAddr, &MAX_BUFFER_SIZE, hcibuf); // input on endpoint 1 - switch (hcibuf[0]) //switch on event type + uint16_t MAX_BUFFER_SIZE = BULK_MAXPKTSIZE; // Request more than 16 bytes anyway, the inTransfer routine will take care of this + uint8_t rcode = pUsb->inTransfer(bAddress, epInfo[ BTD_EVENT_PIPE ].epAddr, &MAX_BUFFER_SIZE, hcibuf); // input on endpoint 1 + if(!rcode || rcode == hrNAK) // Check for errors { - case EV_COMMAND_COMPLETE: - hci_event_flag |= HCI_FLAG_CMD_COMPLETE; // set command complete flag - if((hcibuf[3] == 0x09) && (hcibuf[4] == 0x10))// parameters from read local bluetooth address - { - for (uint8_t i = 0; i < 6; i++) - my_bdaddr[i] = hcibuf[6 + i]; - } - break; + switch (hcibuf[0]) //switch on event type + { + case EV_COMMAND_COMPLETE: + hci_event_flag |= HCI_FLAG_CMD_COMPLETE; // set command complete flag + if((hcibuf[3] == 0x09) && (hcibuf[4] == 0x10))// parameters from read local bluetooth address + { + for (uint8_t i = 0; i < 6; i++) + my_bdaddr[i] = hcibuf[6 + i]; + } + break; - case EV_COMMAND_STATUS: - //hci_command_packets = hcibuf[3]; // update flow control - hci_event_flag |= HCI_FLAG_CMD_STATUS; //set status flag - if(hcibuf[2]) // show status on serial if not OK - { - #ifdef DEBUG - Notify(PSTR("\r\nHCI Command Failed: ")); - PrintHex(hcibuf[2]); - Serial.print(" "); - PrintHex(hcibuf[4]); - Serial.print(" "); - PrintHex(hcibuf[5]); - #endif - } - break; + case EV_COMMAND_STATUS: + if(hcibuf[2]) // show status on serial if not OK + { + #ifdef DEBUG + Notify(PSTR("\r\nHCI Command Failed: ")); + PrintHex(hcibuf[2]); + Serial.print(" "); + PrintHex(hcibuf[4]); + Serial.print(" "); + PrintHex(hcibuf[5]); + #endif + } + break; - case EV_CONNECT_COMPLETE: - hci_event_flag |= HCI_FLAG_CONN_COMPLETE; // set connection complete flag - if (!hcibuf[2]) // check if connected OK - { - hci_handle = hcibuf[3] | hcibuf[4] << 8; //store the handle for the ACL connection - hci_event_flag |= HCI_FLAG_CONNECT_OK; //set connection OK flag - } - break; + case EV_CONNECT_COMPLETE: + if (!hcibuf[2]) // check if connected OK + { + hci_handle = hcibuf[3] | hcibuf[4] << 8; //store the handle for the ACL connection + hci_event_flag |= HCI_FLAG_CONN_COMPLETE; // set connection complete flag + } + break; - case EV_DISCONNECT_COMPLETE: - hci_event_flag |= HCI_FLAG_DISCONN_COMPLETE; //set disconnect commend complete flag - if (!hcibuf[2]) // check if disconnected OK - hci_event_flag &= ~(HCI_FLAG_CONNECT_OK); //clear connection OK flag - break; + case EV_DISCONNECT_COMPLETE: + if (!hcibuf[2]) // check if disconnected OK + { + hci_event_flag |= HCI_FLAG_DISCONN_COMPLETE; //set disconnect commend complete flag + hci_event_flag &= ~HCI_FLAG_CONN_COMPLETE; // clear connection complete flag + } + break; - case EV_NUM_COMPLETE_PKT: - break; - case EV_REMOTE_NAME_COMPLETE: - for (uint8_t i = 0; i < 30; i++) - remote_name[i] = hcibuf[9 + i]; //store first 30 bytes - hci_event_flag |=HCI_FLAG_REMOTE_NAME_COMPLETE; - break; + case EV_NUM_COMPLETE_PKT: + break; - case EV_INCOMING_CONNECT: - disc_bdaddr[0] = hcibuf[2]; - disc_bdaddr[1] = hcibuf[3]; - disc_bdaddr[2] = hcibuf[4]; - disc_bdaddr[3] = hcibuf[5]; - disc_bdaddr[4] = hcibuf[6]; - disc_bdaddr[5] = hcibuf[7]; - hci_event_flag |=HCI_FLAG_INCOMING_REQUEST; - break; + case EV_REMOTE_NAME_COMPLETE: + if (!hcibuf[2]) // check if reading is OK + { + for (uint8_t i = 0; i < 30; i++) + remote_name[i] = hcibuf[9 + i]; //store first 30 bytes + hci_event_flag |= HCI_FLAG_REMOTE_NAME_COMPLETE; + } + break; - case EV_ROLE_CHANGED: - /* - #ifdef DEBUG - Notify(PSTR("\r\nRole Changed")); - #endif - */ - break; - default: - /* - #ifdef DEBUG - if(hcibuf[0] != 0x00) - { - Notify(PSTR("\r\nUnmanaged Event: ")); - PrintHex(hcibuf[0]); - } - #endif - */ - break; - } // switch - HCI_task(); + case EV_INCOMING_CONNECT: + disc_bdaddr[0] = hcibuf[2]; + disc_bdaddr[1] = hcibuf[3]; + disc_bdaddr[2] = hcibuf[4]; + disc_bdaddr[3] = hcibuf[5]; + disc_bdaddr[4] = hcibuf[6]; + disc_bdaddr[5] = hcibuf[7]; + hci_event_flag |= HCI_FLAG_INCOMING_REQUEST; + break; + + case EV_ROLE_CHANGED: + /* + #ifdef DEBUG + Notify(PSTR("\r\nRole Changed")); + #endif + */ + break; + + default: + /* + #ifdef DEBUG + if(hcibuf[0] != 0x00) + { + Notify(PSTR("\r\nUnmanaged Event: ")); + PrintHex(hcibuf[0]); + } + #endif + */ + break; + } // switch + HCI_task(); + } + else { + Notify(PSTR("\r\nHCI event error: ")); + PrintHex(rcode); + } } /* Poll Bluetooth and print result */ @@ -727,14 +798,16 @@ void PS3BT::HCI_task() case HCI_SCANNING_STATE: #ifdef DEBUG Notify(PSTR("\r\nWait For Incoming Connection Request")); - #endif + #endif hci_write_scan_enable(); + watingForConnection = true; hci_state = HCI_CONNECT_IN_STATE; break; case HCI_CONNECT_IN_STATE: if(hci_incoming_connect_request) { + watingForConnection = false; #ifdef DEBUG Notify(PSTR("\r\nIncoming Request")); #endif @@ -819,8 +892,8 @@ void PS3BT::HCI_task() for (uint8_t i = 0; i < OUTPUT_REPORT_BUFFER_SIZE; i++) HIDBuffer[i + 2] = pgm_read_byte(&OUTPUT_REPORT_BUFFER[i]);//First two bytes reserved for report type and ID - for (uint8_t i = 2; i < HIDMOVEBUFFERSIZE; i++) - HIDMoveBuffer[i] = 0; + for (uint8_t i = 2; i < HID_BUFFERSIZE; i++) + HIDMoveBuffer[i] = 0; l2cap_state = L2CAP_EV_WAIT; hci_state = HCI_SCANNING_STATE; @@ -834,148 +907,155 @@ void PS3BT::HCI_task() void PS3BT::ACL_event_task() { uint16_t MAX_BUFFER_SIZE = BULK_MAXPKTSIZE; - pUsb->inTransfer(bAddress, epInfo[ CSR_DATAIN_PIPE ].epAddr, &MAX_BUFFER_SIZE, l2capinbuf); // input on endpoint 2 - if (((l2capinbuf[0] | (l2capinbuf[1] << 8)) == (hci_handle | 0x2000)))//acl_handle_ok - { - if ((l2capinbuf[6] | (l2capinbuf[7] << 8)) == 0x0001)//l2cap_control - Channel ID for ACL-U - { - /* - if (l2capinbuf[8] != 0x00) - { - Serial.print("\r\nL2CAP Signaling Command - 0x"); - PrintHex(l2capoutbuf[8]); - } - */ - if (l2capinbuf[8] == L2CAP_CMD_COMMAND_REJECT) - { - #ifdef DEBUG - Notify(PSTR("\r\nL2CAP Command Rejected - Reason: ")); - PrintHex(l2capinbuf[13]); - Serial.print(" "); - PrintHex(l2capinbuf[12]); - Serial.print(" Data: "); - PrintHex(l2capinbuf[17]); - Serial.print(" "); - PrintHex(l2capinbuf[16]); - Serial.print(" "); - PrintHex(l2capinbuf[15]); - Serial.print(" "); - PrintHex(l2capinbuf[14]); - #endif - } - else if (l2capinbuf[8] == L2CAP_CMD_CONNECTION_REQUEST) - { + uint8_t rcode = pUsb->inTransfer(bAddress, epInfo[ BTD_DATAIN_PIPE ].epAddr, &MAX_BUFFER_SIZE, l2capinbuf); // input on endpoint 2 + if(!rcode || rcode == hrNAK) // Check for errors + { + if (((l2capinbuf[0] | (l2capinbuf[1] << 8)) == (hci_handle | 0x2000)))//acl_handle_ok + { + if ((l2capinbuf[6] | (l2capinbuf[7] << 8)) == 0x0001)//l2cap_control - Channel ID for ACL-U + { /* - Notify(PSTR("\r\nPSM: ")); - PrintHex(l2capinbuf[13]); - Serial.print(" "); - PrintHex(l2capinbuf[12]); - Serial.print(" "); - - Notify(PSTR(" SCID: ")); - PrintHex(l2capinbuf[15]); - Serial.print(" "); - PrintHex(l2capinbuf[14]); - - Notify(PSTR(" Identifier: ")); - PrintHex(l2capinbuf[9]); - */ - if ((l2capinbuf[13] | l2capinbuf[12]) == L2CAP_PSM_HID_CTRL) - { - identifier = l2capinbuf[9]; - control_scid[0] = l2capinbuf[14]; - control_scid[1] = l2capinbuf[15]; - l2cap_event_flag |= L2CAP_EV_CONTROL_CONNECTION_REQUEST; - } - else if ((l2capinbuf[13] | l2capinbuf[12]) == L2CAP_PSM_HID_INTR) - { - identifier = l2capinbuf[9]; - interrupt_scid[0] = l2capinbuf[14]; - interrupt_scid[1] = l2capinbuf[15]; - l2cap_event_flag |= L2CAP_EV_INTERRUPT_CONNECTION_REQUEST; - } - } - else if (l2capinbuf[8] == L2CAP_CMD_CONFIG_RESPONSE) - { - if (l2capinbuf[12] == control_dcid[0] && l2capinbuf[13] == control_dcid[1]) - { - if ((l2capinbuf[16] | (l2capinbuf[17] << 8)) == 0x0000)//Success - { - //Serial.print("\r\nHID Control Configuration Complete"); - l2cap_event_flag |= L2CAP_EV_CONTROL_CONFIG_SUCCESS; - } - } - else if (l2capinbuf[12] == interrupt_dcid[0] && l2capinbuf[13] == interrupt_dcid[1]) - { - if ((l2capinbuf[16] | (l2capinbuf[17] << 8)) == 0x0000)//Success - { - //Serial.print("\r\nHID Interrupt Configuration Complete"); - l2cap_event_flag |= L2CAP_EV_INTERRUPT_CONFIG_SUCCESS; - } - } - } - else if (l2capinbuf[8] == L2CAP_CMD_CONFIG_REQUEST) - { - if (l2capinbuf[12] == control_dcid[0] && l2capinbuf[13] == control_dcid[1]) - { - //Serial.print("\r\nHID Control Configuration Request"); - identifier = l2capinbuf[9]; - l2cap_event_flag |= L2CAP_EV_CONTROL_CONFIG_REQUEST; - } - else if (l2capinbuf[12] == interrupt_dcid[0] && l2capinbuf[13] == interrupt_dcid[1]) - { - //Serial.print("\r\nHID Interrupt Configuration Request"); - identifier = l2capinbuf[9]; - l2cap_event_flag |= L2CAP_EV_INTERRUPT_CONFIG_REQUEST; - } - } - else if (l2capinbuf[8] == L2CAP_CMD_DISCONNECT_REQUEST) - { - if (l2capinbuf[12] == control_dcid[0] && l2capinbuf[13] == control_dcid[1]) + if (l2capinbuf[8] != 0x00) + { + Serial.print("\r\nL2CAP Signaling Command - 0x"); + PrintHex(l2capoutbuf[8]); + } + */ + if (l2capinbuf[8] == L2CAP_CMD_COMMAND_REJECT) { #ifdef DEBUG - Notify(PSTR("\r\nDisconnected Request: Disconnected Control")); + Notify(PSTR("\r\nL2CAP Command Rejected - Reason: ")); + PrintHex(l2capinbuf[13]); + Serial.print(" "); + PrintHex(l2capinbuf[12]); + Serial.print(" Data: "); + PrintHex(l2capinbuf[17]); + Serial.print(" "); + PrintHex(l2capinbuf[16]); + Serial.print(" "); + PrintHex(l2capinbuf[15]); + Serial.print(" "); + PrintHex(l2capinbuf[14]); #endif - identifier = l2capinbuf[9]; - l2cap_disconnection_response(identifier,control_dcid,control_scid); } - else if (l2capinbuf[12] == interrupt_dcid[0] && l2capinbuf[13] == interrupt_dcid[1]) + else if (l2capinbuf[8] == L2CAP_CMD_CONNECTION_REQUEST) + { + /* + Notify(PSTR("\r\nPSM: ")); + PrintHex(l2capinbuf[13]); + Serial.print(" "); + PrintHex(l2capinbuf[12]); + Serial.print(" "); + + Notify(PSTR(" SCID: ")); + PrintHex(l2capinbuf[15]); + Serial.print(" "); + PrintHex(l2capinbuf[14]); + + Notify(PSTR(" Identifier: ")); + PrintHex(l2capinbuf[9]); + */ + if ((l2capinbuf[13] | l2capinbuf[12]) == L2CAP_PSM_HID_CTRL) + { + identifier = l2capinbuf[9]; + control_scid[0] = l2capinbuf[14]; + control_scid[1] = l2capinbuf[15]; + l2cap_event_flag |= L2CAP_EV_CONTROL_CONNECTION_REQUEST; + } + else if ((l2capinbuf[13] | l2capinbuf[12]) == L2CAP_PSM_HID_INTR) + { + identifier = l2capinbuf[9]; + interrupt_scid[0] = l2capinbuf[14]; + interrupt_scid[1] = l2capinbuf[15]; + l2cap_event_flag |= L2CAP_EV_INTERRUPT_CONNECTION_REQUEST; + } + } + else if (l2capinbuf[8] == L2CAP_CMD_CONFIG_RESPONSE) { - #ifdef DEBUG - Notify(PSTR("\r\nDisconnected Request: Disconnected Interrupt")); - #endif - identifier = l2capinbuf[9]; - l2cap_disconnection_response(identifier,interrupt_dcid,interrupt_scid); + if (l2capinbuf[12] == control_dcid[0] && l2capinbuf[13] == control_dcid[1]) + { + if ((l2capinbuf[16] | (l2capinbuf[17] << 8)) == 0x0000)//Success + { + //Serial.print("\r\nHID Control Configuration Complete"); + l2cap_event_flag |= L2CAP_EV_CONTROL_CONFIG_SUCCESS; + } + } + else if (l2capinbuf[12] == interrupt_dcid[0] && l2capinbuf[13] == interrupt_dcid[1]) + { + if ((l2capinbuf[16] | (l2capinbuf[17] << 8)) == 0x0000)//Success + { + //Serial.print("\r\nHID Interrupt Configuration Complete"); + l2cap_event_flag |= L2CAP_EV_INTERRUPT_CONFIG_SUCCESS; + } + } + } + else if (l2capinbuf[8] == L2CAP_CMD_CONFIG_REQUEST) + { + if (l2capinbuf[12] == control_dcid[0] && l2capinbuf[13] == control_dcid[1]) + { + //Serial.print("\r\nHID Control Configuration Request"); + identifier = l2capinbuf[9]; + l2cap_event_flag |= L2CAP_EV_CONTROL_CONFIG_REQUEST; + } + else if (l2capinbuf[12] == interrupt_dcid[0] && l2capinbuf[13] == interrupt_dcid[1]) + { + //Serial.print("\r\nHID Interrupt Configuration Request"); + identifier = l2capinbuf[9]; + l2cap_event_flag |= L2CAP_EV_INTERRUPT_CONFIG_REQUEST; + } + } + else if (l2capinbuf[8] == L2CAP_CMD_DISCONNECT_REQUEST) + { + if (l2capinbuf[12] == control_dcid[0] && l2capinbuf[13] == control_dcid[1]) + { + #ifdef DEBUG + Notify(PSTR("\r\nDisconnected Request: Disconnected Control")); + #endif + identifier = l2capinbuf[9]; + l2cap_disconnection_response(identifier,control_dcid,control_scid); + } + else if (l2capinbuf[12] == interrupt_dcid[0] && l2capinbuf[13] == interrupt_dcid[1]) + { + #ifdef DEBUG + Notify(PSTR("\r\nDisconnected Request: Disconnected Interrupt")); + #endif + identifier = l2capinbuf[9]; + l2cap_disconnection_response(identifier,interrupt_dcid,interrupt_scid); + } + } + else if (l2capinbuf[8] == L2CAP_CMD_DISCONNECT_RESPONSE) + { + if (l2capinbuf[12] == control_scid[0] && l2capinbuf[13] == control_scid[1]) + { + //Serial.print("\r\nDisconnected Response: Disconnected Control"); + identifier = l2capinbuf[9]; + l2cap_event_flag |= L2CAP_EV_CONTROL_DISCONNECT_RESPONSE; + } + else if (l2capinbuf[12] == interrupt_scid[0] && l2capinbuf[13] == interrupt_scid[1]) + { + //Serial.print("\r\nDisconnected Response: Disconnected Interrupt"); + identifier = l2capinbuf[9]; + l2cap_event_flag |= L2CAP_EV_INTERRUPT_DISCONNECT_RESPONSE; + } + } + } + else if (l2capinbuf[6] == interrupt_dcid[0] && l2capinbuf[7] == interrupt_dcid[1])//l2cap_interrupt + { + //Serial.print("\r\nL2CAP Interrupt"); + if(PS3BTConnected || PS3MoveBTConnected || PS3NavigationBTConnected) + { + readReport(); + #ifdef PRINTREPORT + printReport(); //Uncomment "#define PRINTREPORT" to print the report send by the PS3 Controllers + #endif } } - else if (l2capinbuf[8] == L2CAP_CMD_DISCONNECT_RESPONSE) - { - if (l2capinbuf[12] == control_scid[0] && l2capinbuf[13] == control_scid[1]) - { - //Serial.print("\r\nDisconnected Response: Disconnected Control"); - identifier = l2capinbuf[9]; - l2cap_event_flag |= L2CAP_EV_CONTROL_DISCONNECT_RESPONSE; - } - else if (l2capinbuf[12] == interrupt_scid[0] && l2capinbuf[13] == interrupt_scid[1]) - { - //Serial.print("\r\nDisconnected Response: Disconnected Interrupt"); - identifier = l2capinbuf[9]; - l2cap_event_flag |= L2CAP_EV_INTERRUPT_DISCONNECT_RESPONSE; - } - } - } - else if (l2capinbuf[6] == interrupt_dcid[0] && l2capinbuf[7] == interrupt_dcid[1])//l2cap_interrupt - { - //Serial.print("\r\nL2CAP Interrupt"); - if(PS3BTConnected || PS3MoveBTConnected || PS3NavigationBTConnected) - { - readReport(); - #ifdef PRINTREPORT - printReport(); //Uncomment "#define PRINTREPORT" to print the report send by the PS3 Controllers - #endif - } - } - L2CAP_task(); + L2CAP_task(); + } + } + else { + Notify(PSTR("\r\nACL data in error: ")); + PrintHex(rcode); } } void PS3BT::L2CAP_task() @@ -1110,7 +1190,7 @@ void PS3BT::L2CAP_task() dtimeBulbRumble = millis() - timerBulbRumble; if (dtimeBulbRumble > 4000)//Send at least every 4th second { - HIDMove_Command(HIDMoveBuffer, HIDMOVEBUFFERSIZE);//The Bulb and rumble values, has to be written again and again, for it to stay turned on + HIDMove_Command(HIDMoveBuffer, HID_BUFFERSIZE);//The Bulb and rumble values, has to be written again and again, for it to stay turned on timerBulbRumble = millis(); } } @@ -1159,17 +1239,21 @@ void PS3BT::readReport() if(ButtonState != OldButtonState) { - ButtonChanged = true; - if(ButtonState != 0x00) - ButtonPressed = true; - else - ButtonPressed = false; + buttonChanged = true; + if(ButtonState != 0x00) { + buttonPressed = true; + buttonReleased = false; + } else { + buttonPressed = false; + buttonReleased = true; + } } else { - ButtonChanged = false; - ButtonPressed = false; + buttonChanged = false; + buttonPressed = false; + buttonReleased = false; } OldButtonState = ButtonState; @@ -1197,7 +1281,7 @@ void PS3BT::printReport() //Uncomment "#define PRINTREPORT" to print the report void PS3BT::HCI_Command(uint8_t* data, uint16_t nbytes) { hci_event_flag &= ~HCI_FLAG_CMD_COMPLETE; - pUsb->ctrlReq(bAddress, epInfo[ CSR_CONTROL_PIPE ].epAddr, bmREQ_HCI_OUT, 0x00, 0x00, 0x00 ,0x00, nbytes, nbytes, data, NULL); + pUsb->ctrlReq(bAddress, epInfo[ BTD_CONTROL_PIPE ].epAddr, bmREQ_HCI_OUT, 0x00, 0x00, 0x00 ,0x00, nbytes, nbytes, data, NULL); } void PS3BT::hci_reset() @@ -1210,6 +1294,7 @@ void PS3BT::hci_reset() } void PS3BT::hci_write_scan_enable() { + hci_event_flag &= ~HCI_FLAG_INCOMING_REQUEST; hcibuf[0] = 0x1A; // HCI OCF = 1A hcibuf[1] = 0x03 << 2; // HCI OGF = 3 hcibuf[2] = 0x01;// parameter length = 1 @@ -1232,10 +1317,7 @@ void PS3BT::hci_read_bdaddr() HCI_Command(hcibuf, 3); } void PS3BT::hci_accept_connection() -{ - hci_event_flag |= HCI_FLAG_CONNECT_OK; - hci_event_flag &= ~(HCI_FLAG_INCOMING_REQUEST); - +{ hcibuf[0] = 0x09; // HCI OCF = 9 hcibuf[1] = 0x01 << 2; // HCI OGF = 1 hcibuf[2] = 0x07; // parameter length 7 @@ -1251,7 +1333,7 @@ void PS3BT::hci_accept_connection() } void PS3BT::hci_remote_name() { - hci_event_flag &= ~(HCI_FLAG_REMOTE_NAME_COMPLETE); + hci_event_flag &= ~HCI_FLAG_REMOTE_NAME_COMPLETE; hcibuf[0] = 0x19; // HCI OCF = 19 hcibuf[1] = 0x01 << 2; // HCI OGF = 1 hcibuf[2] = 0x0A; // parameter length = 10 @@ -1298,7 +1380,7 @@ void PS3BT::L2CAP_Command(uint8_t* data, uint16_t nbytes) for (uint16_t i = 0; i < nbytes; i++)//L2CAP C-frame buf[8 + i] = data[i]; - uint8_t rcode = pUsb->outTransfer(bAddress, epInfo[ CSR_DATAOUT_PIPE ].epAddr, (8 + nbytes), buf); + uint8_t rcode = pUsb->outTransfer(bAddress, epInfo[ BTD_DATAOUT_PIPE ].epAddr, (8 + nbytes), buf); if(rcode) { #ifdef DEBUG @@ -1319,7 +1401,7 @@ void PS3BT::l2cap_connection_response(uint8_t rxid, uint8_t dcid[], uint8_t scid l2capoutbuf[7] = scid[1]; l2capoutbuf[8] = result;// Result: Pending or Success l2capoutbuf[9] = 0x00; - l2capoutbuf[10] = 0x00;//No further information + l2capoutbuf[10] = 0x00;// No further information l2capoutbuf[11] = 0x00; L2CAP_Command(l2capoutbuf, 12); @@ -1432,7 +1514,7 @@ void PS3BT::HID_Command(uint8_t* data, uint16_t nbytes) if (dtimeHID <= 250)// Check if is has been more than 250ms since last command delay((uint32_t)(250 - dtimeHID));//There have to be a delay between commands - pUsb->outTransfer(bAddress, epInfo[ CSR_DATAOUT_PIPE ].epAddr, (8 + nbytes), buf); + pUsb->outTransfer(bAddress, epInfo[ BTD_DATAOUT_PIPE ].epAddr, (8 + nbytes), buf); timerHID = millis(); } @@ -1441,7 +1523,7 @@ void PS3BT::setAllOff() for (uint8_t i = 0; i < OUTPUT_REPORT_BUFFER_SIZE; i++) HIDBuffer[i + 2] = pgm_read_byte(&OUTPUT_REPORT_BUFFER[i]);//First two bytes reserved for report type and ID - HID_Command(HIDBuffer, OUTPUT_REPORT_BUFFER_SIZE + 2); + HID_Command(HIDBuffer, HID_BUFFERSIZE); } void PS3BT::setRumbleOff() { @@ -1450,7 +1532,7 @@ void PS3BT::setRumbleOff() HIDBuffer[5] = 0x00; HIDBuffer[6] = 0x00;//high mode off - HID_Command(HIDBuffer, OUTPUT_REPORT_BUFFER_SIZE + 2); + HID_Command(HIDBuffer, HID_BUFFERSIZE); } void PS3BT::setRumbleOn(Rumble mode) { @@ -1476,7 +1558,7 @@ void PS3BT::setRumbleOn(Rumble mode) HIDBuffer[6] = 0;//high mode off } - HID_Command(HIDBuffer, OUTPUT_REPORT_BUFFER_SIZE + 2); + HID_Command(HIDBuffer, HID_BUFFERSIZE); } } void PS3BT::setLedOff(LED a) @@ -1487,18 +1569,18 @@ void PS3BT::setLedOff(LED a) //set the LED into the write buffer HIDBuffer[11] = (uint8_t)((uint8_t)(((uint16_t)a & 0x0f) << 1) ^ HIDBuffer[11]); - HID_Command(HIDBuffer, OUTPUT_REPORT_BUFFER_SIZE + 2); + HID_Command(HIDBuffer, HID_BUFFERSIZE); } } void PS3BT::setLedOn(LED a) { HIDBuffer[11] = (uint8_t)((uint8_t)(((uint16_t)a & 0x0f) << 1) | HIDBuffer[11]); - HID_Command(HIDBuffer, OUTPUT_REPORT_BUFFER_SIZE + 2); + HID_Command(HIDBuffer, HID_BUFFERSIZE); } void PS3BT::enable_sixaxis()//Command used to enable the Dualshock 3 and Navigation controller to send data via USB { - uint8_t cmd_buf[12]; + uint8_t cmd_buf[6]; cmd_buf[0] = 0x53;// HID BT Set_report (0x50) | Report Type (Feature 0x03) cmd_buf[1] = 0xF4;// Report ID cmd_buf[2] = 0x42;// Special PS3 Controller enable commands @@ -1530,7 +1612,7 @@ void PS3BT::HIDMove_Command(uint8_t* data,uint16_t nbytes) if (dtimeHID <= 250)// Check if is has been less than 200ms since last command delay((uint32_t)(250 - dtimeHID));//There have to be a delay between commands - pUsb->outTransfer(bAddress, epInfo[ CSR_DATAOUT_PIPE ].epAddr, (8 + nbytes), buf); + pUsb->outTransfer(bAddress, epInfo[ BTD_DATAOUT_PIPE ].epAddr, (8 + nbytes), buf); timerHID = millis(); } @@ -1541,7 +1623,7 @@ void PS3BT::moveSetBulb(uint8_t r, uint8_t g, uint8_t b)//Use this to set the Co HIDMoveBuffer[4] = g; HIDMoveBuffer[5] = b; - HIDMove_Command(HIDMoveBuffer, HIDMOVEBUFFERSIZE); + HIDMove_Command(HIDMoveBuffer, HID_BUFFERSIZE); } void PS3BT::moveSetBulb(Colors color)//Use this to set the Color using the predefined colors in "enums.h" { @@ -1550,12 +1632,12 @@ void PS3BT::moveSetBulb(Colors color)//Use this to set the Color using the prede HIDMoveBuffer[4] = (uint8_t)(color >> 8); HIDMoveBuffer[5] = (uint8_t)(color); - HIDMove_Command(HIDMoveBuffer, HIDMOVEBUFFERSIZE); + HIDMove_Command(HIDMoveBuffer, HID_BUFFERSIZE); } void PS3BT::moveSetRumble(uint8_t rumble) { //set the rumble value into the write buffer HIDMoveBuffer[7] = rumble; - HIDMove_Command(HIDMoveBuffer, HIDMOVEBUFFERSIZE); + HIDMove_Command(HIDMoveBuffer, HID_BUFFERSIZE); } \ No newline at end of file diff --git a/PS3BT.h b/PS3BT.h index 189d0cb6..75fee898 100644 --- a/PS3BT.h +++ b/PS3BT.h @@ -25,115 +25,105 @@ #endif #include "Usb.h" - -/*The application will work in reduced host mode, so we can save program and data - memory space. After verifying the PID and VID we will use known values for the - configuration values for device, interface, endpoints and HID */ +#include "confdescparser.h" /* CSR Bluetooth data taken from descriptors */ -#define INT_MAXPKTSIZE 16 // max size for HCI data -#define BULK_MAXPKTSIZE 64 // max size for ACL data +#define INT_MAXPKTSIZE 16 // max size for HCI data +#define BULK_MAXPKTSIZE 64 // max size for ACL data /* PS3 data taken from descriptors */ -#define EP_MAXPKTSIZE 64 // max size for data via USB +#define EP_MAXPKTSIZE 64 // max size for data via USB /* Endpoint types */ -#define EP_INTERRUPT 0x03 -#define EP_BULK 0x02 +#define EP_INTERRUPT 0x03 -#define CSR_CONTROL_PIPE 0 // names we give to the 4 pipes -#define CSR_EVENT_PIPE 1 -#define CSR_DATAIN_PIPE 2 -#define CSR_DATAOUT_PIPE 3 - -#define PS3_CONTROL_PIPE 0 // names we give to the 3 pipes -#define PS3_OUTPUT_PIPE 1 -#define PS3_INPUT_PIPE 2 +/* Names we give to the 3 ps3 pipes - this is only used for setting the bluetooth address into the ps3 controllers */ +#define PS3_CONTROL_PIPE 0 +#define PS3_OUTPUT_PIPE 1 +#define PS3_INPUT_PIPE 2 //PID and VID of the different devices -#define CSR_VID 0x0A12 //Cambridge Silicon Radio Ltd. -#define CSR_PID 0x0001 //Bluetooth HCI Device -#define PS3_VID 0x054C //Sony Corporation -#define PS3_PID 0x0268 //PS3 Controller DualShock 3 -#define PS3NAVIGATION_VID 0x054C //Sony Corporation -#define PS3NAVIGATION_PID 0x042F //Navigation controller -#define PS3MOVE_VID 0x054C //Sony Corporation -#define PS3MOVE_PID 0x03D5 //Motion controller +#define CSR_VID 0x0A12 // Cambridge Silicon Radio Ltd. +#define CSR_PID 0x0001 // Bluetooth HCI Device +#define ISSC_VID 0x1131 // Integrated System Solution Corp. +#define ISSC_PID 0x1004 // Bluetooth Device +#define PS3_VID 0x054C // Sony Corporation +#define PS3_PID 0x0268 // PS3 Controller DualShock 3 +#define PS3NAVIGATION_VID 0x054C // Sony Corporation +#define PS3NAVIGATION_PID 0x042F // Navigation controller +#define PS3MOVE_VID 0x054C // Sony Corporation +#define PS3MOVE_PID 0x03D5 // Motion controller -#define HIDMOVEBUFFERSIZE 50 // size of the buffer for the Playstation Motion Controller -#define OUTPUT_REPORT_BUFFER_SIZE 48 //Size of the output report buffer for the controllers +#define HID_BUFFERSIZE 50 // size of the buffer for the Playstation Motion Controller +#define OUTPUT_REPORT_BUFFER_SIZE 48 //Size of the output report buffer for the controllers // used in control endpoint header for HCI Commands #define bmREQ_HCI_OUT USB_SETUP_HOST_TO_DEVICE|USB_SETUP_TYPE_CLASS|USB_SETUP_RECIPIENT_DEVICE // used in control endpoint header for HID Commands -#define bmREQ_HIDOUT USB_SETUP_HOST_TO_DEVICE|USB_SETUP_TYPE_CLASS|USB_SETUP_RECIPIENT_INTERFACE -#define HID_REQUEST_SET_REPORT 0x09 +#define bmREQ_HID_OUT USB_SETUP_HOST_TO_DEVICE|USB_SETUP_TYPE_CLASS|USB_SETUP_RECIPIENT_INTERFACE +#define HID_REQUEST_SET_REPORT 0x09 /* Bluetooth HCI states for hci_task() */ -#define HCI_INIT_STATE 0 -#define HCI_RESET_STATE 1 -#define HCI_BDADDR_STATE 2 -#define HCI_SCANNING_STATE 3 -#define HCI_CONNECT_IN_STATE 4 -#define HCI_REMOTE_NAME_STATE 5 -#define HCI_CONNECTED_STATE 6 -#define HCI_DISABLE_SCAN 7 -#define HCI_DONE_STATE 8 -#define HCI_DISCONNECT_STATE 9 +#define HCI_INIT_STATE 0 +#define HCI_RESET_STATE 1 +#define HCI_BDADDR_STATE 2 +#define HCI_SCANNING_STATE 3 +#define HCI_CONNECT_IN_STATE 4 +#define HCI_REMOTE_NAME_STATE 5 +#define HCI_CONNECTED_STATE 6 +#define HCI_DISABLE_SCAN 7 +#define HCI_DONE_STATE 8 +#define HCI_DISCONNECT_STATE 9 /* HCI event flags*/ -#define HCI_FLAG_CMD_COMPLETE 0x01 -#define HCI_FLAG_CMD_STATUS 0x02 -#define HCI_FLAG_CONN_COMPLETE 0x04 -#define HCI_FLAG_DISCONN_COMPLETE 0x08 -#define HCI_FLAG_CONNECT_OK 0x10 -#define HCI_FLAG_REMOTE_NAME_COMPLETE 0x20 -#define HCI_FLAG_INCOMING_REQUEST 0x40 +#define HCI_FLAG_CMD_COMPLETE 0x01 +#define HCI_FLAG_CONN_COMPLETE 0x02 +#define HCI_FLAG_DISCONN_COMPLETE 0x04 +#define HCI_FLAG_REMOTE_NAME_COMPLETE 0x08 +#define HCI_FLAG_INCOMING_REQUEST 0x10 /*Macros for HCI event flag tests */ #define hci_cmd_complete (hci_event_flag & HCI_FLAG_CMD_COMPLETE) -#define hci_cmd_status (hci_event_flag & HCI_FLAG_CMD_STATUS) #define hci_connect_complete (hci_event_flag & HCI_FLAG_CONN_COMPLETE) #define hci_disconnect_complete (hci_event_flag & HCI_FLAG_DISCONN_COMPLETE) -#define hci_connect_ok (hci_event_flag & HCI_FLAG_CONNECT_OK) #define hci_remote_name_complete (hci_event_flag & HCI_FLAG_REMOTE_NAME_COMPLETE) #define hci_incoming_connect_request (hci_event_flag & HCI_FLAG_INCOMING_REQUEST) /* HCI Events managed */ -#define EV_COMMAND_COMPLETE 0x0E -#define EV_COMMAND_STATUS 0x0F -#define EV_CONNECT_COMPLETE 0x03 -#define EV_DISCONNECT_COMPLETE 0x05 -#define EV_NUM_COMPLETE_PKT 0x13 -#define EV_INQUIRY_COMPLETE 0x01 -#define EV_INQUIRY_RESULT 0x02 -#define EV_REMOTE_NAME_COMPLETE 0x07 -#define EV_INCOMING_CONNECT 0x04 -#define EV_ROLE_CHANGED 0x12 +#define EV_COMMAND_COMPLETE 0x0E +#define EV_COMMAND_STATUS 0x0F +#define EV_CONNECT_COMPLETE 0x03 +#define EV_DISCONNECT_COMPLETE 0x05 +#define EV_NUM_COMPLETE_PKT 0x13 +#define EV_INQUIRY_COMPLETE 0x01 +#define EV_INQUIRY_RESULT 0x02 +#define EV_REMOTE_NAME_COMPLETE 0x07 +#define EV_INCOMING_CONNECT 0x04 +#define EV_ROLE_CHANGED 0x12 /* Bluetooth L2CAP states for L2CAP_task() */ -#define L2CAP_EV_WAIT 0 -#define L2CAP_EV_CONTROL_SETUP 1 -#define L2CAP_EV_CONTROL_REQUEST 2 -#define L2CAP_EV_CONTROL_SUCCESS 3 -#define L2CAP_EV_INTERRUPT_SETUP 4 -#define L2CAP_EV_INTERRUPT_REQUEST 5 -#define L2CAP_EV_INTERRUPT_SUCCESS 6 -#define L2CAP_EV_HID_ENABLE_SIXAXIS 7 -#define L2CAP_EV_L2CAP_DONE 8 -#define L2CAP_EV_INTERRUPT_DISCONNECT 9 -#define L2CAP_EV_CONTROL_DISCONNECT 10 +#define L2CAP_EV_WAIT 0 +#define L2CAP_EV_CONTROL_SETUP 1 +#define L2CAP_EV_CONTROL_REQUEST 2 +#define L2CAP_EV_CONTROL_SUCCESS 3 +#define L2CAP_EV_INTERRUPT_SETUP 4 +#define L2CAP_EV_INTERRUPT_REQUEST 5 +#define L2CAP_EV_INTERRUPT_SUCCESS 6 +#define L2CAP_EV_HID_ENABLE_SIXAXIS 7 +#define L2CAP_EV_L2CAP_DONE 8 +#define L2CAP_EV_INTERRUPT_DISCONNECT 9 +#define L2CAP_EV_CONTROL_DISCONNECT 10 /* L2CAP event flags */ -#define L2CAP_EV_CONTROL_CONNECTION_REQUEST 0x01 -#define L2CAP_EV_CONTROL_CONFIG_REQUEST 0x02 -#define L2CAP_EV_CONTROL_CONFIG_SUCCESS 0x04 -#define L2CAP_EV_INTERRUPT_CONNECTION_REQUEST 0x08 -#define L2CAP_EV_INTERRUPT_CONFIG_REQUEST 0x10 -#define L2CAP_EV_INTERRUPT_CONFIG_SUCCESS 0x20 -#define L2CAP_EV_CONTROL_DISCONNECT_RESPONSE 0x40 -#define L2CAP_EV_INTERRUPT_DISCONNECT_RESPONSE 0x80 +#define L2CAP_EV_CONTROL_CONNECTION_REQUEST 0x01 +#define L2CAP_EV_CONTROL_CONFIG_REQUEST 0x02 +#define L2CAP_EV_CONTROL_CONFIG_SUCCESS 0x04 +#define L2CAP_EV_INTERRUPT_CONNECTION_REQUEST 0x08 +#define L2CAP_EV_INTERRUPT_CONFIG_REQUEST 0x10 +#define L2CAP_EV_INTERRUPT_CONFIG_SUCCESS 0x20 +#define L2CAP_EV_CONTROL_DISCONNECT_RESPONSE 0x40 +#define L2CAP_EV_INTERRUPT_DISCONNECT_RESPONSE 0x80 /*Macros for L2CAP event flag tests */ #define l2cap_control_connection_request (l2cap_event_flag & L2CAP_EV_CONTROL_CONNECTION_REQUEST) @@ -146,25 +136,27 @@ #define l2cap_interrupt_disconnect_response (l2cap_event_flag & L2CAP_EV_INTERRUPT_DISCONNECT_RESPONSE) /* L2CAP signaling commands */ -#define L2CAP_CMD_COMMAND_REJECT 0x01 -#define L2CAP_CMD_CONNECTION_REQUEST 0x02 -#define L2CAP_CMD_CONNECTION_RESPONSE 0x03 -#define L2CAP_CMD_CONFIG_REQUEST 0x04 -#define L2CAP_CMD_CONFIG_RESPONSE 0x05 -#define L2CAP_CMD_DISCONNECT_REQUEST 0x06 -#define L2CAP_CMD_DISCONNECT_RESPONSE 0x07 +#define L2CAP_CMD_COMMAND_REJECT 0x01 +#define L2CAP_CMD_CONNECTION_REQUEST 0x02 +#define L2CAP_CMD_CONNECTION_RESPONSE 0x03 +#define L2CAP_CMD_CONFIG_REQUEST 0x04 +#define L2CAP_CMD_CONFIG_RESPONSE 0x05 +#define L2CAP_CMD_DISCONNECT_REQUEST 0x06 +#define L2CAP_CMD_DISCONNECT_RESPONSE 0x07 /* Bluetooth L2CAP PSM */ #define L2CAP_PSM_HID_CTRL 0x11 // HID_Control #define L2CAP_PSM_HID_INTR 0x13 // HID_Interrupt // Used For Connection Response - Remember to Include High Byte -#define PENDING 0x01 -#define SUCCESSFUL 0x00 +#define PENDING 0x01 +#define SUCCESSFUL 0x00 #define bConfigurationValue 0x01 // Used to set configuration -#define PS3_MAX_ENDPOINTS 4 +#define PS3_MAX_ENDPOINTS 4 +#define WI_SUBCLASS_RF 0x01 +#define WI_PROTOCOL_BT 0x01 enum LED { @@ -321,7 +313,7 @@ enum Rumble RumbleLow = 0x20, }; -class PS3BT : public USBDeviceConfig +class PS3BT : public USBDeviceConfig, public UsbConfigXtracter { public: PS3BT(USB *pUsb); @@ -331,6 +323,12 @@ public: virtual uint8_t Release(); virtual uint8_t Poll(); virtual uint8_t GetAddress() { return bAddress; }; + virtual bool isReady() { return bPollEnable; }; + + // UsbConfigXtracter implementation + virtual void EndpointXtract(uint8_t conf, uint8_t iface, uint8_t alt, uint8_t proto, const USB_ENDPOINT_DESCRIPTOR *ep); + + bool isWatingForConnection() { return watingForConnection; }; // Use this to indicate when it is ready for a incoming connection void setBdaddr(uint8_t* BDADDR); void setMoveBdaddr(uint8_t* BDADDR); @@ -360,21 +358,35 @@ public: bool PS3BTConnected;// Variable used to indicate if the normal playstation controller is successfully connected bool PS3MoveBTConnected;// Variable used to indicate if the move controller is successfully connected bool PS3NavigationBTConnected;// Variable used to indicate if the navigation controller is successfully connected - bool ButtonChanged;//Indicate if a button has been changed - bool ButtonPressed;//Indicate if a button has been pressed + bool buttonChanged;//Indicate if a button has been changed + bool buttonPressed;//Indicate if a button has been pressed + bool buttonReleased;//Indicate if a button has been pressed protected: /* mandatory members */ USB *pUsb; - uint8_t bAddress; + uint8_t bAddress; // device address EpInfo epInfo[PS3_MAX_ENDPOINTS]; //endpoint info structure + uint8_t bConfNum; // configuration number + uint8_t bNumEP; // total number of endpoints in the configuration + uint32_t qNextPollTime; // next poll time + + #define BTD_CONTROL_PIPE 0 // Bluetooth dongles control endpoint + static const uint8_t BTD_EVENT_PIPE; // HCI event endpoint index + static const uint8_t BTD_DATAIN_PIPE; // ACL In endpoint index + static const uint8_t BTD_DATAOUT_PIPE; // ACL Out endpoint index + + void PrintEndpointDescriptor(const USB_ENDPOINT_DESCRIPTOR* ep_ptr); + private: bool bPollEnable; + uint8_t pollInterval; + bool watingForConnection; /*variables filled from HCI event management */ int16_t hci_handle; - uint8_t disc_bdaddr[6]; // maximum of three discovered devices + uint8_t disc_bdaddr[6]; // the bluetooth address is always 6 bytes uint8_t remote_name[30]; // first 30 chars of remote name /* variables used by high level HCI task */ @@ -397,8 +409,8 @@ private: uint8_t hcibuf[BULK_MAXPKTSIZE];//General purpose buffer for hci data uint8_t l2capinbuf[BULK_MAXPKTSIZE];//General purpose buffer for l2cap in data uint8_t l2capoutbuf[BULK_MAXPKTSIZE];//General purpose buffer for l2cap out data - uint8_t HIDBuffer[BULK_MAXPKTSIZE];// Used to store HID commands - uint8_t HIDMoveBuffer[HIDMOVEBUFFERSIZE];// Used to store HID commands for the Move controller + uint8_t HIDBuffer[HID_BUFFERSIZE];// Used to store HID commands + uint8_t HIDMoveBuffer[HID_BUFFERSIZE];// Used to store HID commands for the Move controller /* L2CAP Channels */ uint8_t control_scid[2];// L2CAP source CID for HID_Control diff --git a/keywords.txt b/keywords.txt index c95dcdf5..378fc91a 100644 --- a/keywords.txt +++ b/keywords.txt @@ -34,8 +34,11 @@ moveSetRumble KEYWORD2 PS3BTConnected KEYWORD2 PS3MoveBTConnected KEYWORD2 PS3NavigationBTConnected KEYWORD2 -ButtonChanged KEYWORD2 -ButtonPressed KEYWORD2 +buttonChanged KEYWORD2 +buttonPressed KEYWORD2 +buttonReleased KEYWORD2 + +isWatingForConnection KEYWORD2 ################################################ # Constants and enums (LITERAL1) From f16e34819943375e119c5d1e26ebf899d37b5554 Mon Sep 17 00:00:00 2001 From: Kristian Lauszus Date: Wed, 11 Apr 2012 02:18:25 +0200 Subject: [PATCH 3/9] Minor edit --- PS3BT.cpp | 18 +++++++----------- PS3BT.h | 2 -- 2 files changed, 7 insertions(+), 13 deletions(-) diff --git a/PS3BT.cpp b/PS3BT.cpp index 363237bb..e2ca504d 100644 --- a/PS3BT.cpp +++ b/PS3BT.cpp @@ -195,12 +195,10 @@ uint8_t PS3BT::Init(uint8_t parent, uint8_t port, bool lowspeed) for (uint8_t i=0; i confDescrParser(this); rcode = pUsb->getConfDescr(bAddress, 0, i, &confDescrParser); - if( rcode ) { + if(rcode) goto FailGetConfDescr; - } - if( bNumEP > 3 ) { //all endpoints extracted + if( bNumEP > 3 ) //all endpoints extracted break; - } } // for (uint8_t i=0; isetConf(bAddress, epInfo[ CSR_CONTROL_PIPE ].epAddr, bConfigurationValue);//bConfigurationValue = 0x01 + delay(200); // Give time for address change // Set Configuration Value rcode = pUsb->setConf(bAddress, epInfo[ BTD_CONTROL_PIPE ].epAddr, bConfNum); @@ -259,7 +255,7 @@ uint8_t PS3BT::Init(uint8_t parent, uint8_t port, bool lowspeed) delay(200);//Give time for address change - rcode = pUsb->setConf(bAddress, epInfo[ PS3_CONTROL_PIPE ].epAddr, bConfigurationValue);//bConfigurationValue = 0x01 + rcode = pUsb->setConf(bAddress, epInfo[ PS3_CONTROL_PIPE ].epAddr, 1); if( rcode ) goto FailSetConf; @@ -1134,9 +1130,7 @@ void PS3BT::L2CAP_task() l2cap_state = L2CAP_EV_HID_ENABLE_SIXAXIS; } break; - case L2CAP_EV_HID_ENABLE_SIXAXIS: - delay(1000);//There has to be a delay before sending the commands - + case L2CAP_EV_HID_ENABLE_SIXAXIS: for (uint8_t i = 0; i < BULK_MAXPKTSIZE; i++)//Reset l2cap in buffer as it sometimes read it as a button has been pressed l2capinbuf[i] = 0; ButtonState = 0; @@ -1144,6 +1138,7 @@ void PS3BT::L2CAP_task() if (remote_name[0] == 'P')//First letter in PLAYSTATION(R)3 Controller ('P') - 0x50 { + delay(1000);//There has to be a delay before sending the commands enable_sixaxis(); for (uint8_t i = 15; i < 19; i++) @@ -1158,6 +1153,7 @@ void PS3BT::L2CAP_task() } else if (remote_name[0] == 'N')//First letter in Navigation Controller ('N') - 0x4E { + delay(1000);//There has to be a delay before sending the commands enable_sixaxis(); for (uint8_t i = 15; i < 19; i++) diff --git a/PS3BT.h b/PS3BT.h index 75fee898..e15802a3 100644 --- a/PS3BT.h +++ b/PS3BT.h @@ -152,8 +152,6 @@ #define PENDING 0x01 #define SUCCESSFUL 0x00 -#define bConfigurationValue 0x01 // Used to set configuration - #define PS3_MAX_ENDPOINTS 4 #define WI_SUBCLASS_RF 0x01 #define WI_PROTOCOL_BT 0x01 From f6aef1461963e2508606996cbfb76c332bce2128 Mon Sep 17 00:00:00 2001 From: Kristian Lauszus Date: Wed, 11 Apr 2012 02:18:35 +0200 Subject: [PATCH 4/9] Fixed lowercase --- examples/PS3BT/PS3BT.ino | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/examples/PS3BT/PS3BT.ino b/examples/PS3BT/PS3BT.ino index ba8179ce..f82df2f6 100644 --- a/examples/PS3BT/PS3BT.ino +++ b/examples/PS3BT/PS3BT.ino @@ -58,7 +58,7 @@ void loop() Serial.println(""); } - if(BT.ButtonPressed) + if(BT.buttonPressed) { Serial.print(F("PS3 Controller")); @@ -96,7 +96,7 @@ void loop() if(BT.getButton(L1)) Serial.print(F(" - L1")); //if(BT.getButton(L2)) - //Serial.print(F(" - L2")); + //Serial.print(F(" - L2")); if(BT.getButton(L3)) Serial.print(F(" - L3")); if(BT.getButton(R1)) @@ -121,7 +121,7 @@ void loop() if(BT.getAnalogButton(T_MOVE_ANALOG) > 0) { Serial.print(F("T: ")); Serial.println(BT.getAnalogButton(T_MOVE_ANALOG), DEC); - } if(BT.ButtonPressed) { + } if(BT.buttonPressed) { Serial.print(F("PS3 Move Controller")); if(BT.getButton(PS_MOVE)) { @@ -176,4 +176,4 @@ void loop() Serial.println(templow); } } -} \ No newline at end of file +} From a90c2ab4dadf67febd2a2d378f2c2274586cf39a Mon Sep 17 00:00:00 2001 From: Kristian Lauszus Date: Wed, 11 Apr 2012 16:48:55 +0200 Subject: [PATCH 5/9] Added EXTRADEBUG --- PS3BT.cpp | 99 ++++++++++++++++++++++++++++++++----------------------- PS3BT.h | 5 +++ 2 files changed, 63 insertions(+), 41 deletions(-) diff --git a/PS3BT.cpp b/PS3BT.cpp index e2ca504d..d09b7daf 100644 --- a/PS3BT.cpp +++ b/PS3BT.cpp @@ -17,6 +17,7 @@ #include "PS3BT.h" #define DEBUG // Uncomment to print data for debugging +//#define EXTRADEBUG // Uncomment to get even more debugging data //#define PRINTREPORT // Uncomment to print the report send by the PS3 Controllers const uint8_t PS3BT::BTD_EVENT_PIPE = 1; @@ -74,9 +75,9 @@ uint8_t PS3BT::Init(uint8_t parent, uint8_t port, bool lowspeed) // get memory address of USB device address pool AddressPool &addrPool = pUsb->GetAddressPool(); - - //Notify(PSTR("\r\nPS3BT Init"); - + #ifdef EXTRADEBUG + Notify(PSTR("\r\nPS3BT Init")); + #endif // check if address has already been assigned to an instance if (bAddress) { @@ -144,9 +145,10 @@ uint8_t PS3BT::Init(uint8_t parent, uint8_t port, bool lowspeed) PrintHex(rcode); return rcode; } - //Notify(PSTR("\r\nAddr: ")); - //PrintHex(bAddress); - + #ifdef EXTRADEBUG + Notify(PSTR("\r\nAddr: ")); + PrintHex(bAddress); + #endif p->lowspeed = false; //get pointer to assigned address record @@ -202,7 +204,9 @@ uint8_t PS3BT::Init(uint8_t parent, uint8_t port, bool lowspeed) } // for (uint8_t i=0; isetConf(bAddress, epInfo[ PS3_CONTROL_PIPE ].epAddr, 1); if( rcode ) goto FailSetConf; @@ -349,7 +352,9 @@ void PS3BT::EndpointXtract(uint8_t conf, uint8_t iface, uint8_t alt, uint8_t pro //Fill the rest of endpoint data structure epInfo[index].epAddr = (pep->bEndpointAddress & 0x0F); epInfo[index].maxPktSize = (uint8_t)pep->wMaxPacketSize; - //PrintEndpointDescriptor(pep); + #ifdef EXTRADEBUG + PrintEndpointDescriptor(pep); + #endif if(pollInterval < pep->bInterval) // Set the polling interval as the largest polling interval obtained from endpoints pollInterval = pep->bInterval; bNumEP++; @@ -357,7 +362,7 @@ void PS3BT::EndpointXtract(uint8_t conf, uint8_t iface, uint8_t alt, uint8_t pro } void PS3BT::PrintEndpointDescriptor( const USB_ENDPOINT_DESCRIPTOR* ep_ptr ) { - Notify(PSTR("Endpoint descriptor:")); + Notify(PSTR("\r\nEndpoint descriptor:")); Notify(PSTR("\r\nLength:\t\t")); PrintHex(ep_ptr->bLength); Notify(PSTR("\r\nType:\t\t")); @@ -370,7 +375,6 @@ void PS3BT::PrintEndpointDescriptor( const USB_ENDPOINT_DESCRIPTOR* ep_ptr ) PrintHex(ep_ptr->wMaxPacketSize); Notify(PSTR("\r\nPoll Intrv:\t")); PrintHex(ep_ptr->bInterval); - Notify(PSTR("\r\n")); } /* Performs a cleanup after failed Init() attempt */ @@ -715,31 +719,42 @@ void PS3BT::HCI_event_task() hci_event_flag |= HCI_FLAG_INCOMING_REQUEST; break; + /* We will just ignore the following events */ case EV_ROLE_CHANGED: - /* - #ifdef DEBUG - Notify(PSTR("\r\nRole Changed")); - #endif - */ + break; + + case EV_PAGE_SCAN_REP_MODE: + break; + + case EV_LOOPBACK_COMMAND: + break; + + case EV_DATA_BUFFER_OVERFLOW: + break; + + case EV_CHANGE_CONNECTION_LINK: + break; + + case EV_AUTHENTICATION_COMPLETE: break; default: - /* - #ifdef DEBUG + #ifdef EXTRADEBUG if(hcibuf[0] != 0x00) { Notify(PSTR("\r\nUnmanaged Event: ")); PrintHex(hcibuf[0]); } #endif - */ break; } // switch HCI_task(); } else { + #ifdef EXTRADEBUG Notify(PSTR("\r\nHCI event error: ")); PrintHex(rcode); + #endif } } @@ -841,7 +856,7 @@ void PS3BT::HCI_task() } PrintHex(disc_bdaddr[0]); #endif - hci_write_scan_disable();//Only allow one controller + hci_write_scan_disable(); // Only allow one controller hci_state = HCI_DISABLE_SCAN; } break; @@ -875,8 +890,8 @@ void PS3BT::HCI_task() } PrintHex(disc_bdaddr[0]); #endif - l2cap_event_flag = 0;//Clear all flags - hci_event_flag = 0;//Clear all flags + l2cap_event_flag = 0; // Clear all flags + hci_event_flag = 0; // Clear all flags //Reset all buffers for (uint8_t i = 0; i < BULK_MAXPKTSIZE; i++) @@ -887,9 +902,9 @@ void PS3BT::HCI_task() l2capoutbuf[i] = 0; for (uint8_t i = 0; i < OUTPUT_REPORT_BUFFER_SIZE; i++) - HIDBuffer[i + 2] = pgm_read_byte(&OUTPUT_REPORT_BUFFER[i]);//First two bytes reserved for report type and ID - for (uint8_t i = 2; i < HID_BUFFERSIZE; i++) - HIDMoveBuffer[i] = 0; + HIDBuffer[i + 2] = pgm_read_byte(&OUTPUT_REPORT_BUFFER[i]); // First two bytes reserved for report type and ID + for (uint8_t i = 0; i < OUTPUT_REPORT_BUFFER_SIZE; i++) + HIDMoveBuffer[i + 2] = 0; // First two bytes reserved for report type and ID l2cap_state = L2CAP_EV_WAIT; hci_state = HCI_SCANNING_STATE; @@ -909,14 +924,14 @@ void PS3BT::ACL_event_task() if (((l2capinbuf[0] | (l2capinbuf[1] << 8)) == (hci_handle | 0x2000)))//acl_handle_ok { if ((l2capinbuf[6] | (l2capinbuf[7] << 8)) == 0x0001)//l2cap_control - Channel ID for ACL-U - { + { /* - if (l2capinbuf[8] != 0x00) - { + if (l2capinbuf[8] != 0x00) + { Serial.print("\r\nL2CAP Signaling Command - 0x"); PrintHex(l2capoutbuf[8]); } - */ + */ if (l2capinbuf[8] == L2CAP_CMD_COMMAND_REJECT) { #ifdef DEBUG @@ -937,20 +952,20 @@ void PS3BT::ACL_event_task() else if (l2capinbuf[8] == L2CAP_CMD_CONNECTION_REQUEST) { /* - Notify(PSTR("\r\nPSM: ")); - PrintHex(l2capinbuf[13]); - Serial.print(" "); - PrintHex(l2capinbuf[12]); - Serial.print(" "); + Notify(PSTR("\r\nL2CAP Connection Request - PSM: ")); + PrintHex(l2capinbuf[13]); + Serial.print(" "); + PrintHex(l2capinbuf[12]); + Serial.print(" "); - Notify(PSTR(" SCID: ")); - PrintHex(l2capinbuf[15]); - Serial.print(" "); - PrintHex(l2capinbuf[14]); + Notify(PSTR(" SCID: ")); + PrintHex(l2capinbuf[15]); + Serial.print(" "); + PrintHex(l2capinbuf[14]); - Notify(PSTR(" Identifier: ")); - PrintHex(l2capinbuf[9]); - */ + Notify(PSTR(" Identifier: ")); + PrintHex(l2capinbuf[9]); + */ if ((l2capinbuf[13] | l2capinbuf[12]) == L2CAP_PSM_HID_CTRL) { identifier = l2capinbuf[9]; @@ -1050,8 +1065,10 @@ void PS3BT::ACL_event_task() } } else { + #ifdef EXTRADEBUG Notify(PSTR("\r\nACL data in error: ")); PrintHex(rcode); + #endif } } void PS3BT::L2CAP_task() diff --git a/PS3BT.h b/PS3BT.h index e15802a3..1807ba19 100644 --- a/PS3BT.h +++ b/PS3BT.h @@ -101,6 +101,11 @@ #define EV_REMOTE_NAME_COMPLETE 0x07 #define EV_INCOMING_CONNECT 0x04 #define EV_ROLE_CHANGED 0x12 +#define EV_PAGE_SCAN_REP_MODE 0x20 +#define EV_DATA_BUFFER_OVERFLOW 0x1A +#define EV_LOOPBACK_COMMAND 0x19 +#define EV_CHANGE_CONNECTION_LINK 0x09 +#define EV_AUTHENTICATION_COMPLETE 0x06 /* Bluetooth L2CAP states for L2CAP_task() */ #define L2CAP_EV_WAIT 0 From da6eb99a2fa87159f7b80d82232c2c8e8a51956a Mon Sep 17 00:00:00 2001 From: Kristian Lauszus Date: Thu, 12 Apr 2012 23:15:18 +0200 Subject: [PATCH 6/9] Minor revision --- PS3BT.cpp | 348 +++++++++++++++++++++------------------ PS3BT.h | 14 +- examples/PS3BT/PS3BT.ino | 38 +++-- 3 files changed, 217 insertions(+), 183 deletions(-) diff --git a/PS3BT.cpp b/PS3BT.cpp index d09b7daf..38706f46 100644 --- a/PS3BT.cpp +++ b/PS3BT.cpp @@ -37,12 +37,38 @@ prog_char OUTPUT_REPORT_BUFFER[] PROGMEM = 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }; -PS3BT::PS3BT(USB *p): - pUsb(p), // pointer to USB class instance - mandatory - bAddress(0), // device address - mandatory - bNumEP(1), // if config descriptor needs to be parsed - qNextPollTime(0), - bPollEnable(false) // don't start polling before dongle is connected +PS3BT::PS3BT(USB *p, uint8_t btadr5 = 0, uint8_t btadr4 = 0, uint8_t btadr3 = 0, uint8_t btadr2 = 0, uint8_t btadr1 = 0, uint8_t btadr0 = 0): +pUsb(p), // pointer to USB class instance - mandatory +bAddress(0), // device address - mandatory +bNumEP(1), // if config descriptor needs to be parsed +qNextPollTime(0), +bPollEnable(false) // don't start polling before dongle is connected +{ + for(uint8_t i=0; iRegisterDeviceClass(this); //set devConfig[] entry + + my_bdaddr[5] = btadr5; // Change to your dongle's Bluetooth address instead + my_bdaddr[4] = btadr4; + my_bdaddr[3] = btadr3; + my_bdaddr[2] = btadr2; + my_bdaddr[1] = btadr1; + my_bdaddr[0] = btadr0; +} + +PS3BT::PS3BT(USB *p): +pUsb(p), // pointer to USB class instance - mandatory +bAddress(0), // device address - mandatory +bNumEP(1), // if config descriptor needs to be parsed +qNextPollTime(0), +bPollEnable(false) // don't start polling before dongle is connected { for(uint8_t i=0; iRegisterDeviceClass(this); //set devConfig[] entry - - my_bdaddr[5] = 0x00; // Change to your dongle's Bluetooth address instead - my_bdaddr[4] = 0x1F; - my_bdaddr[3] = 0x81; - my_bdaddr[2] = 0x00; - my_bdaddr[1] = 0x08; - my_bdaddr[0] = 0x30; } uint8_t PS3BT::Init(uint8_t parent, uint8_t port, bool lowspeed) @@ -162,79 +181,13 @@ uint8_t PS3BT::Init(uint8_t parent, uint8_t port, bool lowspeed) rcode = pUsb->setEpInfoEntry(bAddress, 1, epInfo); if (rcode) goto FailSetDevTblEntry; - + num_of_conf = ((USB_DEVICE_DESCRIPTOR*)buf)->bNumConfigurations; VID = ((USB_DEVICE_DESCRIPTOR*)buf)->idVendor; PID = ((USB_DEVICE_DESCRIPTOR*)buf)->idProduct; - if((VID == CSR_VID || VID == ISSC_VID) && (PID == CSR_PID || PID == ISSC_PID)) - { - #ifdef DEBUG - Notify(PSTR("\r\nBluetooth Dongle Connected")); - #endif - - //Needed for PS3 Dualshock Controller commands to work via bluetooth - for (uint8_t i = 0; i < OUTPUT_REPORT_BUFFER_SIZE; i++) - HIDBuffer[i + 2] = pgm_read_byte(&OUTPUT_REPORT_BUFFER[i]);//First two bytes reserved for report type and ID - - HIDBuffer[0] = 0x52;// HID BT Set_report (0x50) | Report Type (Output 0x02) - HIDBuffer[1] = 0x01;// Report ID - - //Needed for PS3 Move Controller commands to work via bluetooth - HIDMoveBuffer[0] = 0xA2;// HID BT DATA_request (0xA0) | Report Type (Output 0x02) - HIDMoveBuffer[1] = 0x02;// Report ID - - /* Set device cid for the control and intterrupt channelse - LSB */ - control_dcid[0] = 0x40;//0x0040 - control_dcid[1] = 0x00; - interrupt_dcid[0] = 0x41;//0x0041 - interrupt_dcid[1] = 0x00; - - //check if attached device is a Bluetooth dongle and fill endpoint data structure - //first interface in the configuration must have Bluetooth assigned Class/Subclass/Protocol - //and 3 endpoints - interrupt-IN, bulk-IN, bulk-OUT, - //not necessarily in this order - for (uint8_t i=0; i confDescrParser(this); - rcode = pUsb->getConfDescr(bAddress, 0, i, &confDescrParser); - if(rcode) - goto FailGetConfDescr; - if( bNumEP > 3 ) //all endpoints extracted - break; - } // for (uint8_t i=0; isetEpInfoEntry(bAddress, bNumEP, epInfo); - if( rcode ) - goto FailSetDevTblEntry; - - delay(200); // Give time for address change - - // Set Configuration Value - rcode = pUsb->setConf(bAddress, epInfo[ BTD_CONTROL_PIPE ].epAddr, bConfNum); - if( rcode ) - goto FailSetConf; - - hci_state = HCI_INIT_STATE; - hci_counter = 0; - l2cap_state = L2CAP_EV_WAIT; - #ifdef DEBUG - Notify(PSTR("\r\nCSR Initialized")); - #endif - - watingForConnection = false; - bPollEnable = true; - } - else if((VID == PS3_VID || VID == PS3NAVIGATION_VID || VID == PS3MOVE_VID) && (PID == PS3_PID || PID == PS3NAVIGATION_PID || PID == PS3MOVE_PID)) - { - /*The application will work in reduced host mode, so we can save program and data + if(VID == PS3_VID && (PID == PS3_PID || PID == PS3NAVIGATION_PID || PID == PS3MOVE_PID)) { + /* The application will work in reduced host mode, so we can save program and data memory space. After verifying the PID and VID we will use known values for the configuration values for device, interface, endpoints and HID for the PS3 Controllers */ @@ -251,20 +204,20 @@ uint8_t PS3BT::Init(uint8_t parent, uint8_t port, bool lowspeed) epInfo[ PS3_INPUT_PIPE ].maxPktSize = EP_MAXPKTSIZE; epInfo[ PS3_INPUT_PIPE ].bmSndToggle = bmSNDTOG0; epInfo[ PS3_INPUT_PIPE ].bmRcvToggle = bmRCVTOG0; - + rcode = pUsb->setEpInfoEntry(bAddress, 3, epInfo); if( rcode ) goto FailSetDevTblEntry; delay(200);//Give time for address change - + rcode = pUsb->setConf(bAddress, epInfo[ PS3_CONTROL_PIPE ].epAddr, 1); if( rcode ) goto FailSetConf; - if((VID == PS3_VID || VID == PS3NAVIGATION_VID) && (PID == PS3_PID || PID == PS3NAVIGATION_PID)) + if(PID == PS3_PID || PID == PS3NAVIGATION_PID) { - if(VID == PS3_VID && PID == PS3_PID) { + if(PID == PS3_PID) { #ifdef DEBUG Notify(PSTR("\r\nDualshock 3 Controller Connected")); #endif @@ -285,46 +238,112 @@ uint8_t PS3BT::Init(uint8_t parent, uint8_t port, bool lowspeed) } } else - goto FailUnknownDevice; - + { + //check if attached device is a Bluetooth dongle and fill endpoint data structure + //first interface in the configuration must have Bluetooth assigned Class/Subclass/Protocol + //and 3 endpoints - interrupt-IN, bulk-IN, bulk-OUT, + //not necessarily in this order + for (uint8_t i=0; i confDescrParser(this); + rcode = pUsb->getConfDescr(bAddress, 0, i, &confDescrParser); + if(rcode) + goto FailGetConfDescr; + if(bNumEP > 3) //all endpoints extracted + break; + } // for (uint8_t i=0; isetEpInfoEntry(bAddress, bNumEP, epInfo); + if(rcode) + goto FailSetDevTblEntry; + + delay(200); // Give time for address change + + // Set Configuration Value + rcode = pUsb->setConf(bAddress, epInfo[ BTD_CONTROL_PIPE ].epAddr, bConfNum); + if(rcode) + goto FailSetConf; + + if(VID == CSR_VID && PID == CSR_PID) { + if((uint16_t)((USB_DEVICE_DESCRIPTOR*)buf)->bcdDevice < 0x1915) { // I don't know the exact number, plese let me know if you do + #ifdef DEBUG + Notify(PSTR("\r\nYour dongle may not support reading the analog buttons, sensors and status\r\nYour Revision ID is: 0x")); + PrintHex((uint16_t)((USB_DEVICE_DESCRIPTOR*)buf)->bcdDevice); + Notify(PSTR("\r\nBut should be at least 0x1915\r\nThis usually means that it doesn't support Bluetooth Version 2.0+EDR")); + #endif + } + } + + //Needed for PS3 Dualshock Controller commands to work via bluetooth + for (uint8_t i = 0; i < OUTPUT_REPORT_BUFFER_SIZE; i++) + HIDBuffer[i + 2] = pgm_read_byte(&OUTPUT_REPORT_BUFFER[i]);//First two bytes reserved for report type and ID + + HIDBuffer[0] = 0x52;// HID BT Set_report (0x50) | Report Type (Output 0x02) + HIDBuffer[1] = 0x01;// Report ID + + //Needed for PS3 Move Controller commands to work via bluetooth + HIDMoveBuffer[0] = 0xA2;// HID BT DATA_request (0xA0) | Report Type (Output 0x02) + HIDMoveBuffer[1] = 0x02;// Report ID + + /* Set device cid for the control and intterrupt channelse - LSB */ + control_dcid[0] = 0x40;//0x0040 + control_dcid[1] = 0x00; + interrupt_dcid[0] = 0x41;//0x0041 + interrupt_dcid[1] = 0x00; + + hci_num_reset_loops = 10; // only loop 10 times before trying to send the hci reset command + + hci_state = HCI_INIT_STATE; + hci_counter = 0; + l2cap_state = L2CAP_EV_WAIT; + #ifdef DEBUG + Notify(PSTR("\r\nBluetooth Dongle Initialized")); + #endif + + watingForConnection = false; + bPollEnable = true; + } return 0; //successful configuration - /* diagnostic messages */ - FailGetDevDescr: - #ifdef DEBUG - Notify(PSTR("\r\ngetDevDescr:")); - #endif - goto Fail; - FailSetDevTblEntry: - #ifdef DEBUG - Notify(PSTR("\r\nsetDevTblEn:")); - #endif - goto Fail; - FailGetConfDescr: - #ifdef DEBUG - Notify(PSTR("\r\ngetConf:")); - #endif - goto Fail; - FailSetConf: - #ifdef DEBUG - Notify(PSTR("\r\nsetConf:")); - #endif - goto Fail; - FailUnknownDevice: - #ifdef DEBUG - Notify(PSTR("\r\nUnknown Device Connected - VID: ")); - PrintHex(VID); - Notify(PSTR(" PID: ")); - PrintHex(PID); - #endif - goto Fail; - Fail: - #ifdef DEBUG - Notify(PSTR("\r\nPS3 Init Failed, error code: ")); - Serial.print(rcode); - #endif - Release(); - return rcode; +/* diagnostic messages */ +FailGetDevDescr: + #ifdef DEBUG + Notify(PSTR("\r\ngetDevDescr:")); + #endif + goto Fail; +FailSetDevTblEntry: + #ifdef DEBUG + Notify(PSTR("\r\nsetDevTblEn:")); + #endif + goto Fail; +FailGetConfDescr: + #ifdef DEBUG + Notify(PSTR("\r\ngetConf:")); + #endif + goto Fail; +FailSetConf: + #ifdef DEBUG + Notify(PSTR("\r\nsetConf:")); + #endif + goto Fail; +FailUnknownDevice: + #ifdef DEBUG + Notify(PSTR("\r\nUnknown Device Connected - VID: ")); + PrintHex(VID); + Notify(PSTR(" PID: ")); + PrintHex(PID); + #endif + goto Fail; +Fail: + #ifdef DEBUG + Notify(PSTR("\r\nPS3 Init Failed, error code: ")); + Serial.print(rcode); + #endif + Release(); + return rcode; } /* Extracts interrupt-IN, bulk-IN, bulk-OUT endpoint information from config descriptor */ void PS3BT::EndpointXtract(uint8_t conf, uint8_t iface, uint8_t alt, uint8_t proto, const USB_ENDPOINT_DESCRIPTOR *pep) @@ -341,7 +360,7 @@ void PS3BT::EndpointXtract(uint8_t conf, uint8_t iface, uint8_t alt, uint8_t pro if ((pep->bmAttributes & 0x03) == 3 && (pep->bEndpointAddress & 0x80) == 0x80) //Interrupt In endpoint found index = BTD_EVENT_PIPE; - + else { if ((pep->bmAttributes & 0x02) == 2) //bulk endpoint found index = ((pep->bEndpointAddress & 0x80) == 0x80) ? BTD_DATAIN_PIPE : BTD_DATAOUT_PIPE; @@ -352,7 +371,7 @@ void PS3BT::EndpointXtract(uint8_t conf, uint8_t iface, uint8_t alt, uint8_t pro //Fill the rest of endpoint data structure epInfo[index].epAddr = (pep->bEndpointAddress & 0x0F); epInfo[index].maxPktSize = (uint8_t)pep->wMaxPacketSize; - #ifdef EXTRADEBUG + #ifdef EXTRADEBUG PrintEndpointDescriptor(pep); #endif if(pollInterval < pep->bInterval) // Set the polling interval as the largest polling interval obtained from endpoints @@ -402,14 +421,14 @@ void PS3BT::setBdaddr(uint8_t* BDADDR) /* Store the bluetooth address */ for(uint8_t i = 0; i <6;i++) my_bdaddr[i] = BDADDR[i]; - + /* Set the internal bluetooth address */ uint8_t buf[8]; buf[0] = 0x01; buf[1] = 0x00; for (uint8_t i = 0; i < 6; i++) buf[i+2] = my_bdaddr[5 - i];//Copy into buffer, has to be written reversed - + //bmRequest = Host to device (0x00) | Class (0x20) | Interface (0x01) = 0x21, bRequest = Set Report (0x09), Report ID (0xF5), Report Type (Feature 0x03), interface (0x00), datalength, datalength, data) pUsb->ctrlReq(bAddress,epInfo[PS3_CONTROL_PIPE].epAddr, bmREQ_HID_OUT, HID_REQUEST_SET_REPORT, 0xF5, 0x03, 0x00, 8, 8, buf, NULL); #ifdef DEBUG @@ -511,11 +530,11 @@ double PS3BT::getAngle(Angle a, boolean resolution) // Boolean indicate if 360-d { double accXin; double accXval; - double Pitch; + double angleX; double accYin; double accYval; - double Roll; + double angleY; double accZin; double accZval; @@ -543,36 +562,36 @@ double PS3BT::getAngle(Angle a, boolean resolution) // Boolean indicate if 360-d { //the result will come out as radians, so it is multiplied by 180/pi, to convert to degrees //In the end it is minus by 90, so its 0 degrees when in horizontal postion - Pitch = acos(accXval / R) * 180 / PI - 90; + angleX = acos(accXval / R) * 180 / PI - 90; if(resolution) { if (accZval < 0)//Convert to 360 degrees resolution - set resolution false if you need both pitch and roll { - if (Pitch < 0) - Pitch = -180 - Pitch; + if (angleX < 0) + angleX = -180 - angleX; else - Pitch = 180 - Pitch; + angleX = 180 - angleX; } } - return Pitch; + return angleX; } else { //the result will come out as radians, so it is multiplied by 180/pi, to convert to degrees //In the end it is minus by 90, so its 0 degrees when in horizontal postion - Roll = acos(accYval / R) * 180 / PI - 90; + angleY = acos(accYval / R) * 180 / PI - 90; if(resolution) { if (accZval < 0)//Convert to 360 degrees resolution - set resolution false if you need both pitch and roll { - if (Roll < 0) - Roll = -180 - Roll; + if (angleY < 0) + angleY = -180 - angleY; else - Roll = 180 - Roll; + angleY = 180 - angleY; } } - return Roll; + return angleY; } } bool PS3BT::getStatus(Status c) @@ -666,7 +685,7 @@ void PS3BT::HCI_event_task() my_bdaddr[i] = hcibuf[6 + i]; } break; - + case EV_COMMAND_STATUS: if(hcibuf[2]) // show status on serial if not OK { @@ -680,7 +699,7 @@ void PS3BT::HCI_event_task() #endif } break; - + case EV_CONNECT_COMPLETE: if (!hcibuf[2]) // check if connected OK { @@ -696,7 +715,7 @@ void PS3BT::HCI_event_task() hci_event_flag &= ~HCI_FLAG_CONN_COMPLETE; // clear connection complete flag } break; - + case EV_NUM_COMPLETE_PKT: break; @@ -725,29 +744,29 @@ void PS3BT::HCI_event_task() case EV_PAGE_SCAN_REP_MODE: break; - + case EV_LOOPBACK_COMMAND: break; - + case EV_DATA_BUFFER_OVERFLOW: break; case EV_CHANGE_CONNECTION_LINK: break; - + case EV_AUTHENTICATION_COMPLETE: break; default: - #ifdef EXTRADEBUG - if(hcibuf[0] != 0x00) - { + #ifdef EXTRADEBUG + if(hcibuf[0] != 0x00) + { Notify(PSTR("\r\nUnmanaged Event: ")); PrintHex(hcibuf[0]); - } - #endif + } + #endif break; - } // switch + } // switch HCI_task(); } else { @@ -764,7 +783,7 @@ void PS3BT::HCI_task() switch (hci_state){ case HCI_INIT_STATE: hci_counter++; - if (hci_counter > 1000) // wait until we have looped 1000 times to clear any old events + if (hci_counter > hci_num_reset_loops) // wait until we have looped x times to clear any old events { hci_reset(); hci_state = HCI_RESET_STATE; @@ -782,8 +801,11 @@ void PS3BT::HCI_task() hci_state = HCI_BDADDR_STATE; hci_read_bdaddr(); } - else if (hci_counter > 1000) + else if (hci_counter > hci_num_reset_loops) { + hci_num_reset_loops *= 10; + if(hci_num_reset_loops > 2000) + hci_num_reset_loops = 2000; #ifdef DEBUG Notify(PSTR("\r\nNo response to HCI Reset")); #endif @@ -924,13 +946,13 @@ void PS3BT::ACL_event_task() if (((l2capinbuf[0] | (l2capinbuf[1] << 8)) == (hci_handle | 0x2000)))//acl_handle_ok { if ((l2capinbuf[6] | (l2capinbuf[7] << 8)) == 0x0001)//l2cap_control - Channel ID for ACL-U - { + { /* if (l2capinbuf[8] != 0x00) { Serial.print("\r\nL2CAP Signaling Command - 0x"); PrintHex(l2capoutbuf[8]); - } + } */ if (l2capinbuf[8] == L2CAP_CMD_COMMAND_REJECT) { @@ -950,19 +972,19 @@ void PS3BT::ACL_event_task() #endif } else if (l2capinbuf[8] == L2CAP_CMD_CONNECTION_REQUEST) - { + { /* Notify(PSTR("\r\nL2CAP Connection Request - PSM: ")); PrintHex(l2capinbuf[13]); Serial.print(" "); PrintHex(l2capinbuf[12]); Serial.print(" "); - + Notify(PSTR(" SCID: ")); PrintHex(l2capinbuf[15]); Serial.print(" "); PrintHex(l2capinbuf[14]); - + Notify(PSTR(" Identifier: ")); PrintHex(l2capinbuf[9]); */ @@ -1058,7 +1080,7 @@ void PS3BT::ACL_event_task() readReport(); #ifdef PRINTREPORT printReport(); //Uncomment "#define PRINTREPORT" to print the report send by the PS3 Controllers - #endif + #endif } } L2CAP_task(); @@ -1261,14 +1283,14 @@ void PS3BT::readReport() buttonReleased = true; } } - + else { buttonChanged = false; buttonPressed = false; buttonReleased = false; } - + OldButtonState = ButtonState; } } @@ -1392,7 +1414,7 @@ void PS3BT::L2CAP_Command(uint8_t* data, uint16_t nbytes) for (uint16_t i = 0; i < nbytes; i++)//L2CAP C-frame buf[8 + i] = data[i]; - + uint8_t rcode = pUsb->outTransfer(bAddress, epInfo[ BTD_DATAOUT_PIPE ].epAddr, (8 + nbytes), buf); if(rcode) { diff --git a/PS3BT.h b/PS3BT.h index 1807ba19..2e6df6f2 100644 --- a/PS3BT.h +++ b/PS3BT.h @@ -44,14 +44,10 @@ //PID and VID of the different devices #define CSR_VID 0x0A12 // Cambridge Silicon Radio Ltd. -#define CSR_PID 0x0001 // Bluetooth HCI Device -#define ISSC_VID 0x1131 // Integrated System Solution Corp. -#define ISSC_PID 0x1004 // Bluetooth Device +#define CSR_PID 0x0001 // Bluetooth HCI Device #define PS3_VID 0x054C // Sony Corporation #define PS3_PID 0x0268 // PS3 Controller DualShock 3 -#define PS3NAVIGATION_VID 0x054C // Sony Corporation #define PS3NAVIGATION_PID 0x042F // Navigation controller -#define PS3MOVE_VID 0x054C // Sony Corporation #define PS3MOVE_PID 0x03D5 // Motion controller #define HID_BUFFERSIZE 50 // size of the buffer for the Playstation Motion Controller @@ -157,10 +153,12 @@ #define PENDING 0x01 #define SUCCESSFUL 0x00 -#define PS3_MAX_ENDPOINTS 4 +// Used to determine if it is a Bluetooth dongle #define WI_SUBCLASS_RF 0x01 #define WI_PROTOCOL_BT 0x01 +#define PS3_MAX_ENDPOINTS 4 + enum LED { LED1 = 0x01, @@ -319,6 +317,7 @@ enum Rumble class PS3BT : public USBDeviceConfig, public UsbConfigXtracter { public: + PS3BT(USB *pUsb, uint8_t btadr5, uint8_t btadr4, uint8_t btadr3, uint8_t btadr2, uint8_t btadr1, uint8_t btadr0); PS3BT(USB *pUsb); // USBDeviceConfig implementation @@ -332,7 +331,7 @@ public: virtual void EndpointXtract(uint8_t conf, uint8_t iface, uint8_t alt, uint8_t proto, const USB_ENDPOINT_DESCRIPTOR *ep); bool isWatingForConnection() { return watingForConnection; }; // Use this to indicate when it is ready for a incoming connection - + void setBdaddr(uint8_t* BDADDR); void setMoveBdaddr(uint8_t* BDADDR); @@ -395,6 +394,7 @@ private: /* variables used by high level HCI task */ uint8_t hci_state; //current state of bluetooth hci connection uint16_t hci_counter; // counter used for bluetooth hci reset loops + uint8_t hci_num_reset_loops; // this value indicate how many times it should read before trying to reset uint16_t hci_event_flag;// hci flags of received bluetooth events /* variables used by high level L2CAP task */ diff --git a/examples/PS3BT/PS3BT.ino b/examples/PS3BT/PS3BT.ino index f82df2f6..667b30ea 100644 --- a/examples/PS3BT/PS3BT.ino +++ b/examples/PS3BT/PS3BT.ino @@ -6,9 +6,12 @@ #include USB Usb; -PS3BT BT(&Usb); +/* You can create the instance of the class in two ways */ +PS3BT BT(&Usb); // This will just create the instance +//PS3BT BT(&Usb,0x00,0x15,0x83,0x3D,0x0A,0x57); // This will also store the bluetooth address - this can be obtained from the dongle when running the sketch boolean printTemperature; +boolean printAngle; void setup() { @@ -28,19 +31,19 @@ void loop() if(BT.getAnalogHat(LeftHatX) > 137 || BT.getAnalogHat(LeftHatX) < 117 || BT.getAnalogHat(LeftHatY) > 137 || BT.getAnalogHat(LeftHatY) < 117 || BT.getAnalogHat(RightHatX) > 137 || BT.getAnalogHat(RightHatX) < 117 || BT.getAnalogHat(RightHatY) > 137 || BT.getAnalogHat(RightHatY) < 117) { if(BT.getAnalogHat(LeftHatX) > 137 || BT.getAnalogHat(LeftHatX) < 117) { Serial.print(F("LeftHatX: ")); - Serial.print(BT.getAnalogHat(LeftHatX), DEC); + Serial.print(BT.getAnalogHat(LeftHatX)); Serial.print("\t"); } if(BT.getAnalogHat(LeftHatY) > 137 || BT.getAnalogHat(LeftHatY) < 117) { Serial.print(F("LeftHatY: ")); - Serial.print(BT.getAnalogHat(LeftHatY), DEC); + Serial.print(BT.getAnalogHat(LeftHatY)); Serial.print("\t"); } if(BT.getAnalogHat(RightHatX) > 137 || BT.getAnalogHat(RightHatX) < 117) { Serial.print(F("RightHatX: ")); - Serial.print(BT.getAnalogHat(RightHatX), DEC); + Serial.print(BT.getAnalogHat(RightHatX)); Serial.print("\t"); } if(BT.getAnalogHat(RightHatY) > 137 || BT.getAnalogHat(RightHatY) < 117) { Serial.print(F("RightHatY: ")); - Serial.print(BT.getAnalogHat(RightHatY), DEC); + Serial.print(BT.getAnalogHat(RightHatY)); } Serial.println(""); } @@ -49,11 +52,11 @@ void loop() if(BT.getAnalogButton(L2_ANALOG) > 0 || BT.getAnalogButton(R2_ANALOG) > 0) { if(BT.getAnalogButton(L2_ANALOG) > 0) { Serial.print(F("L2: ")); - Serial.print(BT.getAnalogButton(L2_ANALOG), DEC); + Serial.print(BT.getAnalogButton(L2_ANALOG)); Serial.print("\t"); } if(BT.getAnalogButton(R2_ANALOG) > 0) { Serial.print(F("R2: ")); - Serial.print(BT.getAnalogButton(R2_ANALOG), DEC); + Serial.print(BT.getAnalogButton(R2_ANALOG)); } Serial.println(""); } @@ -109,18 +112,27 @@ void loop() if(BT.getButton(SELECT)) { Serial.print(F(" - Select - ")); Serial.print(BT.getStatusString()); - } if(BT.getButton(START)) - Serial.print(F(" - Start")); - + } if(BT.getButton(START)) { + Serial.print(F(" - Start")); + printAngle = !printAngle; + while(BT.getButton(START)) + Usb.Task(); + } Serial.println(""); } - } + } + if(printAngle) { + Serial.print(F("Pitch: ")); + Serial.print(BT.getAngle(Pitch,false)); + Serial.print(F("\tRoll: ")); + Serial.println(BT.getAngle(Roll,false)); + } } else if(BT.PS3MoveBTConnected) { if(BT.getAnalogButton(T_MOVE_ANALOG) > 0) { Serial.print(F("T: ")); - Serial.println(BT.getAnalogButton(T_MOVE_ANALOG), DEC); + Serial.println(BT.getAnalogButton(T_MOVE_ANALOG)); } if(BT.buttonPressed) { Serial.print(F("PS3 Move Controller")); @@ -161,7 +173,7 @@ void loop() if(printTemperature) { String templow; String temphigh; - String input = String(BT.getSensor(tempMove), DEC); + String input = String(BT.getSensor(tempMove)); if (input.length() > 3) { temphigh = input.substring(0, 2); From fd58db268052cff97daa7d34097e78dc471c180d Mon Sep 17 00:00:00 2001 From: Kristian Lauszus Date: Sun, 15 Apr 2012 01:42:16 +0200 Subject: [PATCH 7/9] Removed all delays --- PS3BT.cpp | 146 +++++++++++++++++++++++++----------------------------- PS3BT.h | 13 ++--- 2 files changed, 75 insertions(+), 84 deletions(-) diff --git a/PS3BT.cpp b/PS3BT.cpp index 38706f46..885651c7 100644 --- a/PS3BT.cpp +++ b/PS3BT.cpp @@ -265,17 +265,7 @@ uint8_t PS3BT::Init(uint8_t parent, uint8_t port, bool lowspeed) // Set Configuration Value rcode = pUsb->setConf(bAddress, epInfo[ BTD_CONTROL_PIPE ].epAddr, bConfNum); if(rcode) - goto FailSetConf; - - if(VID == CSR_VID && PID == CSR_PID) { - if((uint16_t)((USB_DEVICE_DESCRIPTOR*)buf)->bcdDevice < 0x1915) { // I don't know the exact number, plese let me know if you do - #ifdef DEBUG - Notify(PSTR("\r\nYour dongle may not support reading the analog buttons, sensors and status\r\nYour Revision ID is: 0x")); - PrintHex((uint16_t)((USB_DEVICE_DESCRIPTOR*)buf)->bcdDevice); - Notify(PSTR("\r\nBut should be at least 0x1915\r\nThis usually means that it doesn't support Bluetooth Version 2.0+EDR")); - #endif - } - } + goto FailSetConf; //Needed for PS3 Dualshock Controller commands to work via bluetooth for (uint8_t i = 0; i < OUTPUT_REPORT_BUFFER_SIZE; i++) @@ -948,12 +938,12 @@ void PS3BT::ACL_event_task() if ((l2capinbuf[6] | (l2capinbuf[7] << 8)) == 0x0001)//l2cap_control - Channel ID for ACL-U { /* - if (l2capinbuf[8] != 0x00) - { - Serial.print("\r\nL2CAP Signaling Command - 0x"); - PrintHex(l2capoutbuf[8]); - } - */ + if (l2capinbuf[8] != 0x00) + { + Serial.print("\r\nL2CAP Signaling Command - 0x"); + PrintHex(l2capoutbuf[8]); + } + */ if (l2capinbuf[8] == L2CAP_CMD_COMMAND_REJECT) { #ifdef DEBUG @@ -974,20 +964,20 @@ void PS3BT::ACL_event_task() else if (l2capinbuf[8] == L2CAP_CMD_CONNECTION_REQUEST) { /* - Notify(PSTR("\r\nL2CAP Connection Request - PSM: ")); - PrintHex(l2capinbuf[13]); - Serial.print(" "); - PrintHex(l2capinbuf[12]); - Serial.print(" "); + Notify(PSTR("\r\nL2CAP Connection Request - PSM: ")); + PrintHex(l2capinbuf[13]); + Serial.print(" "); + PrintHex(l2capinbuf[12]); + Serial.print(" "); - Notify(PSTR(" SCID: ")); - PrintHex(l2capinbuf[15]); - Serial.print(" "); - PrintHex(l2capinbuf[14]); + Notify(PSTR(" SCID: ")); + PrintHex(l2capinbuf[15]); + Serial.print(" "); + PrintHex(l2capinbuf[14]); - Notify(PSTR(" Identifier: ")); - PrintHex(l2capinbuf[9]); - */ + Notify(PSTR(" Identifier: ")); + PrintHex(l2capinbuf[9]); + */ if ((l2capinbuf[13] | l2capinbuf[12]) == L2CAP_PSM_HID_CTRL) { identifier = l2capinbuf[9]; @@ -1165,58 +1155,58 @@ void PS3BT::L2CAP_task() { #ifdef DEBUG Notify(PSTR("\r\nHID Interrupt Successfully Configured")); - #endif - l2cap_state = L2CAP_EV_HID_ENABLE_SIXAXIS; + #endif + if(remote_name[0] == 'M') { // First letter in Motion Controller ('M') + for (uint8_t i = 0; i < BULK_MAXPKTSIZE; i++) // Reset l2cap in buffer as it sometimes read it as a button has been pressed + l2capinbuf[i] = 0; + ButtonState = 0; + OldButtonState = 0; + + l2cap_state = L2CAP_EV_HID_PS3_LED; + } else + l2cap_state = L2CAP_EV_HID_ENABLE_SIXAXIS; + timer = millis(); + } + break; + case L2CAP_EV_HID_ENABLE_SIXAXIS: + if(millis() - timer > 1000) { // loop 1 second before sending the command + for (uint8_t i = 0; i < BULK_MAXPKTSIZE; i++) // Reset l2cap in buffer as it sometimes read it as a button has been pressed + l2capinbuf[i] = 0; + ButtonState = 0; + OldButtonState = 0; + + enable_sixaxis(); + for (uint8_t i = 15; i < 19; i++) + l2capinbuf[i] = 0x7F; // Set the analog joystick values to center position + l2cap_state = L2CAP_EV_HID_PS3_LED; + timer = millis(); } break; - case L2CAP_EV_HID_ENABLE_SIXAXIS: - for (uint8_t i = 0; i < BULK_MAXPKTSIZE; i++)//Reset l2cap in buffer as it sometimes read it as a button has been pressed - l2capinbuf[i] = 0; - ButtonState = 0; - OldButtonState = 0; - if (remote_name[0] == 'P')//First letter in PLAYSTATION(R)3 Controller ('P') - 0x50 - { - delay(1000);//There has to be a delay before sending the commands - enable_sixaxis(); - - for (uint8_t i = 15; i < 19; i++) - l2capinbuf[i] = 0x7F;//Set the analog joystick values to center position - - delay(1000);//There has to be a delay before data can be read - setLedOn(LED1); - #ifdef DEBUG - Notify(PSTR("\r\nDualshock 3 Controller Enabled\r\n")); - #endif - PS3BTConnected = true; - } - else if (remote_name[0] == 'N')//First letter in Navigation Controller ('N') - 0x4E - { - delay(1000);//There has to be a delay before sending the commands - enable_sixaxis(); - - for (uint8_t i = 15; i < 19; i++) - l2capinbuf[i] = 0x7F;//Set the analog joystick values to center - - delay(1000);//There has to be a delay before data can be read - setLedOn(LED1);//This just turns LED constantly on, on the Navigation controller - #ifdef DEBUG - Notify(PSTR("\r\nNavigation Controller Enabled\r\n")); - #endif - PS3NavigationBTConnected = true; - } - else if (remote_name[0] == 'M')//First letter in Motion Controller ('M') - 0x4D - { - delay(1000);//There has to be a delay before data can be read - moveSetBulb(Red); - #ifdef DEBUG - Notify(PSTR("\r\nMotion Controller Enabled\r\n")); - #endif - PS3MoveBTConnected = true; - - timerBulbRumble = millis(); - } - l2cap_state = L2CAP_EV_L2CAP_DONE; + case L2CAP_EV_HID_PS3_LED: + if(millis() - timer > 1000) { // loop 1 second before sending the command + if (remote_name[0] == 'P') { // First letter in PLAYSTATION(R)3 Controller ('P') + setLedOn(LED1); + #ifdef DEBUG + Notify(PSTR("\r\nDualshock 3 Controller Enabled\r\n")); + #endif + PS3BTConnected = true; + } else if (remote_name[0] == 'N') { // First letter in Navigation Controller ('N') + setLedOn(LED1); // This just turns LED constantly on, on the Navigation controller + #ifdef DEBUG + Notify(PSTR("\r\nNavigation Controller Enabled\r\n")); + #endif + PS3NavigationBTConnected = true; + } else if(remote_name[0] == 'M') { // First letter in Motion Controller ('M') + moveSetBulb(Red); + timerBulbRumble = millis(); + #ifdef DEBUG + Notify(PSTR("\r\nMotion Controller Enabled\r\n")); + #endif + PS3MoveBTConnected = true; + } + l2cap_state = L2CAP_EV_L2CAP_DONE; + } break; case L2CAP_EV_L2CAP_DONE: diff --git a/PS3BT.h b/PS3BT.h index 2e6df6f2..08e7018b 100644 --- a/PS3BT.h +++ b/PS3BT.h @@ -43,8 +43,6 @@ #define PS3_INPUT_PIPE 2 //PID and VID of the different devices -#define CSR_VID 0x0A12 // Cambridge Silicon Radio Ltd. -#define CSR_PID 0x0001 // Bluetooth HCI Device #define PS3_VID 0x054C // Sony Corporation #define PS3_PID 0x0268 // PS3 Controller DualShock 3 #define PS3NAVIGATION_PID 0x042F // Navigation controller @@ -112,9 +110,10 @@ #define L2CAP_EV_INTERRUPT_REQUEST 5 #define L2CAP_EV_INTERRUPT_SUCCESS 6 #define L2CAP_EV_HID_ENABLE_SIXAXIS 7 -#define L2CAP_EV_L2CAP_DONE 8 -#define L2CAP_EV_INTERRUPT_DISCONNECT 9 -#define L2CAP_EV_CONTROL_DISCONNECT 10 +#define L2CAP_EV_HID_PS3_LED 8 +#define L2CAP_EV_L2CAP_DONE 9 +#define L2CAP_EV_INTERRUPT_DISCONNECT 10 +#define L2CAP_EV_CONTROL_DISCONNECT 11 /* L2CAP event flags */ #define L2CAP_EV_CONTROL_CONNECTION_REQUEST 0x01 @@ -395,12 +394,14 @@ private: uint8_t hci_state; //current state of bluetooth hci connection uint16_t hci_counter; // counter used for bluetooth hci reset loops uint8_t hci_num_reset_loops; // this value indicate how many times it should read before trying to reset - uint16_t hci_event_flag;// hci flags of received bluetooth events + uint16_t hci_event_flag;// hci flags of received bluetooth events /* variables used by high level L2CAP task */ uint8_t l2cap_state; uint16_t l2cap_event_flag;// l2cap flags of received bluetooth events + unsigned long timer; + uint32_t ButtonState; uint32_t OldButtonState; uint32_t timerHID;// timer used see if there has to be a delay before a new HID command From 98cf17c94732702b95b3d1f32163b5af9236442d Mon Sep 17 00:00:00 2001 From: Kristian Lauszus Date: Tue, 24 Apr 2012 01:35:43 +0200 Subject: [PATCH 8/9] Added 360 resolution --- PS3BT.cpp | 290 +++++++++++++++++---------------------- PS3BT.h | 8 +- examples/PS3BT/PS3BT.ino | 4 +- 3 files changed, 131 insertions(+), 171 deletions(-) diff --git a/PS3BT.cpp b/PS3BT.cpp index 885651c7..24fc9d7e 100644 --- a/PS3BT.cpp +++ b/PS3BT.cpp @@ -24,8 +24,7 @@ const uint8_t PS3BT::BTD_EVENT_PIPE = 1; const uint8_t PS3BT::BTD_DATAIN_PIPE = 2; const uint8_t PS3BT::BTD_DATAOUT_PIPE = 3; -prog_char OUTPUT_REPORT_BUFFER[] PROGMEM = -{ +prog_char OUTPUT_REPORT_BUFFER[] PROGMEM = { 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff, 0x27, 0x10, 0x00, 0x32, @@ -37,7 +36,7 @@ prog_char OUTPUT_REPORT_BUFFER[] PROGMEM = 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }; -PS3BT::PS3BT(USB *p, uint8_t btadr5 = 0, uint8_t btadr4 = 0, uint8_t btadr3 = 0, uint8_t btadr2 = 0, uint8_t btadr1 = 0, uint8_t btadr0 = 0): +PS3BT::PS3BT(USB *p, uint8_t btadr5, uint8_t btadr4, uint8_t btadr3, uint8_t btadr2, uint8_t btadr1, uint8_t btadr0): pUsb(p), // pointer to USB class instance - mandatory bAddress(0), // device address - mandatory bNumEP(1), // if config descriptor needs to be parsed @@ -94,15 +93,15 @@ uint8_t PS3BT::Init(uint8_t parent, uint8_t port, bool lowspeed) // get memory address of USB device address pool AddressPool &addrPool = pUsb->GetAddressPool(); - #ifdef EXTRADEBUG +#ifdef EXTRADEBUG Notify(PSTR("\r\nPS3BT Init")); - #endif +#endif // check if address has already been assigned to an instance if (bAddress) { - #ifdef DEBUG +#ifdef DEBUG Notify(PSTR("\r\nAddress in use")); - #endif +#endif return USB_ERROR_CLASS_INSTANCE_ALREADY_IN_USE; } @@ -111,17 +110,17 @@ uint8_t PS3BT::Init(uint8_t parent, uint8_t port, bool lowspeed) if (!p) { - #ifdef DEBUG +#ifdef DEBUG Notify(PSTR("\r\nAddress not found")); - #endif +#endif return USB_ERROR_ADDRESS_NOT_FOUND_IN_POOL; } if (!p->epinfo) { - #ifdef DEBUG +#ifdef DEBUG Notify(PSTR("\r\nepinfo is null")); - #endif +#endif return USB_ERROR_EPINFO_IS_NULL; } @@ -158,16 +157,16 @@ uint8_t PS3BT::Init(uint8_t parent, uint8_t port, bool lowspeed) p->lowspeed = false; addrPool.FreeAddress(bAddress); bAddress = 0; - #ifdef DEBUG +#ifdef DEBUG Notify(PSTR("\r\nsetAddr: ")); - #endif +#endif PrintHex(rcode); return rcode; } - #ifdef EXTRADEBUG +#ifdef EXTRADEBUG Notify(PSTR("\r\nAddr: ")); PrintHex(bAddress); - #endif +#endif p->lowspeed = false; //get pointer to assigned address record @@ -218,22 +217,22 @@ uint8_t PS3BT::Init(uint8_t parent, uint8_t port, bool lowspeed) if(PID == PS3_PID || PID == PS3NAVIGATION_PID) { if(PID == PS3_PID) { - #ifdef DEBUG +#ifdef DEBUG Notify(PSTR("\r\nDualshock 3 Controller Connected")); - #endif +#endif } else { // must be a navigation controller - #ifdef DEBUG +#ifdef DEBUG Notify(PSTR("\r\nNavigation Controller Connected")); - #endif +#endif } /* Set internal bluetooth address */ setBdaddr(my_bdaddr); } else // must be a Motion controller { - #ifdef DEBUG +#ifdef DEBUG Notify(PSTR("\r\nMotion Controller Connected")); - #endif +#endif setMoveBdaddr(my_bdaddr); } } @@ -289,49 +288,49 @@ uint8_t PS3BT::Init(uint8_t parent, uint8_t port, bool lowspeed) hci_state = HCI_INIT_STATE; hci_counter = 0; l2cap_state = L2CAP_EV_WAIT; - #ifdef DEBUG +#ifdef DEBUG Notify(PSTR("\r\nBluetooth Dongle Initialized")); - #endif +#endif watingForConnection = false; bPollEnable = true; } return 0; //successful configuration -/* diagnostic messages */ + /* diagnostic messages */ FailGetDevDescr: - #ifdef DEBUG +#ifdef DEBUG Notify(PSTR("\r\ngetDevDescr:")); - #endif +#endif goto Fail; FailSetDevTblEntry: - #ifdef DEBUG +#ifdef DEBUG Notify(PSTR("\r\nsetDevTblEn:")); - #endif +#endif goto Fail; FailGetConfDescr: - #ifdef DEBUG +#ifdef DEBUG Notify(PSTR("\r\ngetConf:")); - #endif +#endif goto Fail; FailSetConf: - #ifdef DEBUG +#ifdef DEBUG Notify(PSTR("\r\nsetConf:")); - #endif +#endif goto Fail; FailUnknownDevice: - #ifdef DEBUG +#ifdef DEBUG Notify(PSTR("\r\nUnknown Device Connected - VID: ")); PrintHex(VID); Notify(PSTR(" PID: ")); PrintHex(PID); - #endif +#endif goto Fail; Fail: - #ifdef DEBUG +#ifdef DEBUG Notify(PSTR("\r\nPS3 Init Failed, error code: ")); Serial.print(rcode); - #endif +#endif Release(); return rcode; } @@ -361,9 +360,9 @@ void PS3BT::EndpointXtract(uint8_t conf, uint8_t iface, uint8_t alt, uint8_t pro //Fill the rest of endpoint data structure epInfo[index].epAddr = (pep->bEndpointAddress & 0x0F); epInfo[index].maxPktSize = (uint8_t)pep->wMaxPacketSize; - #ifdef EXTRADEBUG +#ifdef EXTRADEBUG PrintEndpointDescriptor(pep); - #endif +#endif if(pollInterval < pep->bInterval) // Set the polling interval as the largest polling interval obtained from endpoints pollInterval = pep->bInterval; bNumEP++; @@ -421,7 +420,7 @@ void PS3BT::setBdaddr(uint8_t* BDADDR) //bmRequest = Host to device (0x00) | Class (0x20) | Interface (0x01) = 0x21, bRequest = Set Report (0x09), Report ID (0xF5), Report Type (Feature 0x03), interface (0x00), datalength, datalength, data) pUsb->ctrlReq(bAddress,epInfo[PS3_CONTROL_PIPE].epAddr, bmREQ_HID_OUT, HID_REQUEST_SET_REPORT, 0xF5, 0x03, 0x00, 8, 8, buf, NULL); - #ifdef DEBUG +#ifdef DEBUG Notify(PSTR("\r\nBluetooth Address was set to: ")); for(int8_t i = 5; i > 0; i--) { @@ -429,7 +428,7 @@ void PS3BT::setBdaddr(uint8_t* BDADDR) Serial.print(":"); } PrintHex(my_bdaddr[0]); - #endif +#endif return; } void PS3BT::setMoveBdaddr(uint8_t* BDADDR) @@ -451,7 +450,7 @@ void PS3BT::setMoveBdaddr(uint8_t* BDADDR) //bmRequest = Host to device (0x00) | Class (0x20) | Interface (0x01) = 0x21, bRequest = Set Report (0x09), Report ID (0x05), Report Type (Feature 0x03), interface (0x00), datalength, datalength, data) pUsb->ctrlReq(bAddress,epInfo[PS3_CONTROL_PIPE].epAddr, bmREQ_HID_OUT, HID_REQUEST_SET_REPORT, 0x05, 0x03, 0x00,11,11, buf, NULL); - #ifdef DEBUG +#ifdef DEBUG Notify(PSTR("\r\nBluetooth Address was set to: ")); for(int8_t i = 5; i > 0; i--) { @@ -459,7 +458,7 @@ void PS3BT::setMoveBdaddr(uint8_t* BDADDR) Serial.print(":"); } PrintHex(my_bdaddr[0]); - #endif +#endif return; } bool PS3BT::getButton(Button b) @@ -516,72 +515,33 @@ uint32_t PS3BT::getSensor(Sensor a) return (((l2capinbuf[(uint16_t)a + 1] << 8) | l2capinbuf[(uint16_t)a]) - 0x8000); } } -double PS3BT::getAngle(Angle a, boolean resolution) // Boolean indicate if 360-degrees resolution is used or not - set false if you want to use both axis -{ - double accXin; - double accXval; - double angleX; +double PS3BT::getAngle(Angle a) { + // Data for the Kionix KXPC4 used in DualShock 3 + const double sensivity = 204.6; // 0.66/3.3*1023 (660mV/g) + const double zeroG = 511.5; // 1.65/3.3*1023 (1,65V) - double accYin; - double accYval; - double angleY; - - double accZin; - double accZval; - - //Data for the Kionix KXPC4 used in DualShock 3 - double sensivity = 204.6;//0.66/3.3*1023 (660mV/g) - double zeroG = 511.5;//1.65/3.3*1023 (1,65V) - double R;//force vector - - accXin = getSensor(aX); - accXval = (zeroG - accXin) / sensivity;//Convert to g's - accXval *= 2; - - accYin = getSensor(aY); - accYval = (zeroG - accYin) / sensivity;//Convert to g's - accYval *= 2; - - accZin = getSensor(aZ); - accZval = (zeroG - accZin) / sensivity;//Convert to g's + double accXval = (zeroG - getSensor(aX)) / sensivity; // Convert to g's + accXval *= 2; + double accYval = (zeroG - getSensor(aY)) / sensivity; // Convert to g's + accYval *= 2; + double accZval = (zeroG - getSensor(aZ)) / sensivity; // Convert to g's accZval *= 2; - R = sqrt(pow(accXval, 2) + pow(accYval, 2) + pow(accZval, 2)); + double R = sqrt(accXval*accXval + accYval*accYval + accZval*accZval); + // convert read values to -π/2 to π/2 - Needed for atan2 + double angleX = acos(accXval/R)-0.5*PI; + double angleY = acos(accYval/R)-0.5*PI; + double angleZ = acos(accZval/R)-0.5*PI; - if (a == Pitch) - { - //the result will come out as radians, so it is multiplied by 180/pi, to convert to degrees - //In the end it is minus by 90, so its 0 degrees when in horizontal postion - angleX = acos(accXval / R) * 180 / PI - 90; - if(resolution) - { - if (accZval < 0)//Convert to 360 degrees resolution - set resolution false if you need both pitch and roll - { - if (angleX < 0) - angleX = -180 - angleX; - else - angleX = 180 - angleX; - } - } - return angleX; - - } - else - { - //the result will come out as radians, so it is multiplied by 180/pi, to convert to degrees - //In the end it is minus by 90, so its 0 degrees when in horizontal postion - angleY = acos(accYval / R) * 180 / PI - 90; - if(resolution) - { - if (accZval < 0)//Convert to 360 degrees resolution - set resolution false if you need both pitch and roll - { - if (angleY < 0) - angleY = -180 - angleY; - else - angleY = 180 - angleY; - } - } - return angleY; + // Convert to 360 degrees resolution + // atan2 outputs the value of -π to π (radians) + // We are then converting it to 0 to 2π and then to degrees + if (a == Pitch) { + double angle = (atan2(-angleY,-angleZ)+PI)*RAD_TO_DEG; + return angle; + } else { + double angle = (atan2(-angleX,-angleZ)+PI)*RAD_TO_DEG; + return angle; } } bool PS3BT::getStatus(Status c) @@ -679,14 +639,14 @@ void PS3BT::HCI_event_task() case EV_COMMAND_STATUS: if(hcibuf[2]) // show status on serial if not OK { - #ifdef DEBUG +#ifdef DEBUG Notify(PSTR("\r\nHCI Command Failed: ")); PrintHex(hcibuf[2]); Serial.print(" "); PrintHex(hcibuf[4]); Serial.print(" "); PrintHex(hcibuf[5]); - #endif +#endif } break; @@ -728,7 +688,7 @@ void PS3BT::HCI_event_task() hci_event_flag |= HCI_FLAG_INCOMING_REQUEST; break; - /* We will just ignore the following events */ + /* We will just ignore the following events */ case EV_ROLE_CHANGED: break; @@ -748,22 +708,22 @@ void PS3BT::HCI_event_task() break; default: - #ifdef EXTRADEBUG +#ifdef EXTRADEBUG if(hcibuf[0] != 0x00) { Notify(PSTR("\r\nUnmanaged Event: ")); PrintHex(hcibuf[0]); } - #endif +#endif break; } // switch HCI_task(); } else { - #ifdef EXTRADEBUG +#ifdef EXTRADEBUG Notify(PSTR("\r\nHCI event error: ")); PrintHex(rcode); - #endif +#endif } } @@ -785,9 +745,9 @@ void PS3BT::HCI_task() hci_counter++; if (hci_cmd_complete) { - #ifdef DEBUG +#ifdef DEBUG Notify(PSTR("\r\nHCI Reset complete")); - #endif +#endif hci_state = HCI_BDADDR_STATE; hci_read_bdaddr(); } @@ -796,9 +756,9 @@ void PS3BT::HCI_task() hci_num_reset_loops *= 10; if(hci_num_reset_loops > 2000) hci_num_reset_loops = 2000; - #ifdef DEBUG +#ifdef DEBUG Notify(PSTR("\r\nNo response to HCI Reset")); - #endif +#endif hci_state = HCI_INIT_STATE; hci_counter = 0; } @@ -806,7 +766,7 @@ void PS3BT::HCI_task() case HCI_BDADDR_STATE: if (hci_cmd_complete) { - #ifdef DEBUG +#ifdef DEBUG Notify(PSTR("\r\nLocal Bluetooth Address: ")); for(int8_t i = 5; i > 0;i--) { @@ -814,14 +774,14 @@ void PS3BT::HCI_task() Serial.print(":"); } PrintHex(my_bdaddr[0]); - #endif +#endif hci_state = HCI_SCANNING_STATE; } break; case HCI_SCANNING_STATE: - #ifdef DEBUG +#ifdef DEBUG Notify(PSTR("\r\nWait For Incoming Connection Request")); - #endif +#endif hci_write_scan_enable(); watingForConnection = true; hci_state = HCI_CONNECT_IN_STATE; @@ -831,9 +791,9 @@ void PS3BT::HCI_task() if(hci_incoming_connect_request) { watingForConnection = false; - #ifdef DEBUG +#ifdef DEBUG Notify(PSTR("\r\nIncoming Request")); - #endif +#endif hci_remote_name(); hci_state = HCI_REMOTE_NAME_STATE; } @@ -842,7 +802,7 @@ void PS3BT::HCI_task() case HCI_REMOTE_NAME_STATE: if(hci_remote_name_complete) { - #ifdef DEBUG +#ifdef DEBUG Notify(PSTR("\r\nRemote Name: ")); for (uint8_t i = 0; i < 30; i++) { @@ -850,7 +810,7 @@ void PS3BT::HCI_task() break; Serial.write(remote_name[i]); } - #endif +#endif hci_accept_connection(); hci_state = HCI_CONNECTED_STATE; } @@ -859,7 +819,7 @@ void PS3BT::HCI_task() case HCI_CONNECTED_STATE: if (hci_connect_complete) { - #ifdef DEBUG +#ifdef DEBUG Notify(PSTR("\r\nConnected to Device: ")); for(int8_t i = 5; i>0;i--) { @@ -867,7 +827,7 @@ void PS3BT::HCI_task() Serial.print(":"); } PrintHex(disc_bdaddr[0]); - #endif +#endif hci_write_scan_disable(); // Only allow one controller hci_state = HCI_DISABLE_SCAN; } @@ -876,9 +836,9 @@ void PS3BT::HCI_task() case HCI_DISABLE_SCAN: if (hci_cmd_complete) { - #ifdef DEBUG +#ifdef DEBUG Notify(PSTR("\r\nScan Disabled")); - #endif +#endif l2cap_event_flag = 0; l2cap_state = L2CAP_EV_CONTROL_SETUP; hci_state = HCI_DONE_STATE; @@ -893,7 +853,7 @@ void PS3BT::HCI_task() case HCI_DISCONNECT_STATE: if (hci_disconnect_complete) { - #ifdef DEBUG +#ifdef DEBUG Notify(PSTR("\r\nDisconnected from Device: ")); for(int8_t i = 5; i>0;i--) { @@ -901,7 +861,7 @@ void PS3BT::HCI_task() Serial.print(":"); } PrintHex(disc_bdaddr[0]); - #endif +#endif l2cap_event_flag = 0; // Clear all flags hci_event_flag = 0; // Clear all flags @@ -946,7 +906,7 @@ void PS3BT::ACL_event_task() */ if (l2capinbuf[8] == L2CAP_CMD_COMMAND_REJECT) { - #ifdef DEBUG +#ifdef DEBUG Notify(PSTR("\r\nL2CAP Command Rejected - Reason: ")); PrintHex(l2capinbuf[13]); Serial.print(" "); @@ -959,7 +919,7 @@ void PS3BT::ACL_event_task() PrintHex(l2capinbuf[15]); Serial.print(" "); PrintHex(l2capinbuf[14]); - #endif +#endif } else if (l2capinbuf[8] == L2CAP_CMD_CONNECTION_REQUEST) { @@ -1031,17 +991,17 @@ void PS3BT::ACL_event_task() { if (l2capinbuf[12] == control_dcid[0] && l2capinbuf[13] == control_dcid[1]) { - #ifdef DEBUG +#ifdef DEBUG Notify(PSTR("\r\nDisconnected Request: Disconnected Control")); - #endif +#endif identifier = l2capinbuf[9]; l2cap_disconnection_response(identifier,control_dcid,control_scid); } else if (l2capinbuf[12] == interrupt_dcid[0] && l2capinbuf[13] == interrupt_dcid[1]) { - #ifdef DEBUG +#ifdef DEBUG Notify(PSTR("\r\nDisconnected Request: Disconnected Interrupt")); - #endif +#endif identifier = l2capinbuf[9]; l2cap_disconnection_response(identifier,interrupt_dcid,interrupt_scid); } @@ -1068,19 +1028,19 @@ void PS3BT::ACL_event_task() if(PS3BTConnected || PS3MoveBTConnected || PS3NavigationBTConnected) { readReport(); - #ifdef PRINTREPORT +#ifdef PRINTREPORT printReport(); //Uncomment "#define PRINTREPORT" to print the report send by the PS3 Controllers - #endif +#endif } } L2CAP_task(); } } else { - #ifdef EXTRADEBUG +#ifdef EXTRADEBUG Notify(PSTR("\r\nACL data in error: ")); PrintHex(rcode); - #endif +#endif } } void PS3BT::L2CAP_task() @@ -1092,9 +1052,9 @@ void PS3BT::L2CAP_task() case L2CAP_EV_CONTROL_SETUP: if (l2cap_control_connection_request) { - #ifdef DEBUG +#ifdef DEBUG Notify(PSTR("\r\nHID Control Incoming Connection Request")); - #endif +#endif l2cap_connection_response(identifier, control_dcid, control_scid, PENDING); delay(1); l2cap_connection_response(identifier, control_dcid, control_scid, SUCCESSFUL); @@ -1107,9 +1067,9 @@ void PS3BT::L2CAP_task() case L2CAP_EV_CONTROL_REQUEST: if (l2cap_control_config_request) { - #ifdef DEBUG +#ifdef DEBUG Notify(PSTR("\r\nHID Control Configuration Request")); - #endif +#endif l2cap_config_response(identifier, control_scid); l2cap_state = L2CAP_EV_CONTROL_SUCCESS; } @@ -1118,18 +1078,18 @@ void PS3BT::L2CAP_task() case L2CAP_EV_CONTROL_SUCCESS: if (l2cap_control_config_success) { - #ifdef DEBUG +#ifdef DEBUG Notify(PSTR("\r\nHID Control Successfully Configured")); - #endif +#endif l2cap_state = L2CAP_EV_INTERRUPT_SETUP; } break; case L2CAP_EV_INTERRUPT_SETUP: if (l2cap_interrupt_connection_request) { - #ifdef DEBUG +#ifdef DEBUG Notify(PSTR("\r\nHID Interrupt Incoming Connection Request")); - #endif +#endif l2cap_connection_response(identifier, interrupt_dcid, interrupt_scid, PENDING); delay(1); l2cap_connection_response(identifier, interrupt_dcid, interrupt_scid, SUCCESSFUL); @@ -1143,9 +1103,9 @@ void PS3BT::L2CAP_task() case L2CAP_EV_INTERRUPT_REQUEST: if (l2cap_interrupt_config_request) { - #ifdef DEBUG +#ifdef DEBUG Notify(PSTR("\r\nHID Interrupt Configuration Request")); - #endif +#endif l2cap_config_response(identifier, interrupt_scid); l2cap_state = L2CAP_EV_INTERRUPT_SUCCESS; } @@ -1153,18 +1113,18 @@ void PS3BT::L2CAP_task() case L2CAP_EV_INTERRUPT_SUCCESS: if (l2cap_interrupt_config_success) { - #ifdef DEBUG +#ifdef DEBUG Notify(PSTR("\r\nHID Interrupt Successfully Configured")); - #endif +#endif if(remote_name[0] == 'M') { // First letter in Motion Controller ('M') for (uint8_t i = 0; i < BULK_MAXPKTSIZE; i++) // Reset l2cap in buffer as it sometimes read it as a button has been pressed l2capinbuf[i] = 0; ButtonState = 0; OldButtonState = 0; - + l2cap_state = L2CAP_EV_HID_PS3_LED; } else - l2cap_state = L2CAP_EV_HID_ENABLE_SIXAXIS; + l2cap_state = L2CAP_EV_HID_ENABLE_SIXAXIS; timer = millis(); } break; @@ -1187,22 +1147,22 @@ void PS3BT::L2CAP_task() if(millis() - timer > 1000) { // loop 1 second before sending the command if (remote_name[0] == 'P') { // First letter in PLAYSTATION(R)3 Controller ('P') setLedOn(LED1); - #ifdef DEBUG +#ifdef DEBUG Notify(PSTR("\r\nDualshock 3 Controller Enabled\r\n")); - #endif +#endif PS3BTConnected = true; } else if (remote_name[0] == 'N') { // First letter in Navigation Controller ('N') setLedOn(LED1); // This just turns LED constantly on, on the Navigation controller - #ifdef DEBUG +#ifdef DEBUG Notify(PSTR("\r\nNavigation Controller Enabled\r\n")); - #endif +#endif PS3NavigationBTConnected = true; } else if(remote_name[0] == 'M') { // First letter in Motion Controller ('M') moveSetBulb(Red); timerBulbRumble = millis(); - #ifdef DEBUG +#ifdef DEBUG Notify(PSTR("\r\nMotion Controller Enabled\r\n")); - #endif +#endif PS3MoveBTConnected = true; } l2cap_state = L2CAP_EV_L2CAP_DONE; @@ -1224,9 +1184,9 @@ void PS3BT::L2CAP_task() case L2CAP_EV_INTERRUPT_DISCONNECT: if (l2cap_interrupt_disconnect_response) { - #ifdef DEBUG +#ifdef DEBUG Notify(PSTR("\r\nDisconnected Interrupt Channel")); - #endif +#endif identifier++; l2cap_disconnection_request(identifier, control_dcid, control_scid); l2cap_state = L2CAP_EV_CONTROL_DISCONNECT; @@ -1236,9 +1196,9 @@ void PS3BT::L2CAP_task() case L2CAP_EV_CONTROL_DISCONNECT: if (l2cap_control_disconnect_response) { - #ifdef DEBUG +#ifdef DEBUG Notify(PSTR("\r\nDisconnected Control Channel")); - #endif +#endif hci_disconnect(); l2cap_state = L2CAP_EV_L2CAP_DONE; hci_state = HCI_DISCONNECT_STATE; @@ -1408,10 +1368,10 @@ void PS3BT::L2CAP_Command(uint8_t* data, uint16_t nbytes) uint8_t rcode = pUsb->outTransfer(bAddress, epInfo[ BTD_DATAOUT_PIPE ].epAddr, (8 + nbytes), buf); if(rcode) { - #ifdef DEBUG +#ifdef DEBUG Notify(PSTR("\r\nError sending message: 0x")); PrintHex(rcode); - #endif +#endif } } void PS3BT::l2cap_connection_response(uint8_t rxid, uint8_t dcid[], uint8_t scid[], uint8_t result) diff --git a/PS3BT.h b/PS3BT.h index 08e7018b..d355497a 100644 --- a/PS3BT.h +++ b/PS3BT.h @@ -339,7 +339,7 @@ public: uint8_t getAnalogButton(AnalogButton a); uint8_t getAnalogHat(AnalogHat a); uint32_t getSensor(Sensor a); - double getAngle(Angle a, boolean resolution); + double getAngle(Angle a); bool getStatus(Status c); String getStatusString(); void disconnect(); // use this void to disconnect any of the controllers @@ -386,15 +386,15 @@ private: bool watingForConnection; /*variables filled from HCI event management */ - int16_t hci_handle; + int16_t hci_handle; uint8_t disc_bdaddr[6]; // the bluetooth address is always 6 bytes uint8_t remote_name[30]; // first 30 chars of remote name /* variables used by high level HCI task */ uint8_t hci_state; //current state of bluetooth hci connection - uint16_t hci_counter; // counter used for bluetooth hci reset loops + uint16_t hci_counter; // counter used for bluetooth hci reset loops uint8_t hci_num_reset_loops; // this value indicate how many times it should read before trying to reset - uint16_t hci_event_flag;// hci flags of received bluetooth events + uint16_t hci_event_flag;// hci flags of received bluetooth events /* variables used by high level L2CAP task */ uint8_t l2cap_state; diff --git a/examples/PS3BT/PS3BT.ino b/examples/PS3BT/PS3BT.ino index 667b30ea..413b6d75 100644 --- a/examples/PS3BT/PS3BT.ino +++ b/examples/PS3BT/PS3BT.ino @@ -123,9 +123,9 @@ void loop() } if(printAngle) { Serial.print(F("Pitch: ")); - Serial.print(BT.getAngle(Pitch,false)); + Serial.print(BT.getAngle(Pitch)); Serial.print(F("\tRoll: ")); - Serial.println(BT.getAngle(Roll,false)); + Serial.println(BT.getAngle(Roll)); } } else if(BT.PS3MoveBTConnected) From d25414948fde1ad860dc91d26ae6e7a9bbb36ec8 Mon Sep 17 00:00:00 2001 From: Kristian Lauszus Date: Tue, 24 Apr 2012 22:49:34 +0200 Subject: [PATCH 9/9] Added 360 degrees resolution of angle calculated from accelerometer --- PS3BT.cpp | 79 +++++++++++++++++++++++++--------------- PS3BT.h | 2 +- examples/PS3BT/PS3BT.ino | 16 ++++++-- 3 files changed, 63 insertions(+), 34 deletions(-) diff --git a/PS3BT.cpp b/PS3BT.cpp index 24fc9d7e..92c8a640 100644 --- a/PS3BT.cpp +++ b/PS3BT.cpp @@ -482,7 +482,7 @@ uint8_t PS3BT::getAnalogHat(AnalogHat a) return 0; return (uint8_t)(l2capinbuf[(uint16_t)a]); } -uint32_t PS3BT::getSensor(Sensor a) +int32_t PS3BT::getSensor(Sensor a) { if (l2capinbuf == NULL) return 0; @@ -504,45 +504,64 @@ uint32_t PS3BT::getSensor(Sensor a) return 0; } else if (a == tempMove) - { - if (l2capinbuf == NULL) - return 0; - return (((l2capinbuf[(uint16_t)a + 1] & 0xF0) >> 4) | (l2capinbuf[(uint16_t)a] << 4)); - } - else - { - - return (((l2capinbuf[(uint16_t)a + 1] << 8) | l2capinbuf[(uint16_t)a]) - 0x8000); - } + return (((l2capinbuf[(uint16_t)a + 1] & 0xF0) >> 4) | (l2capinbuf[(uint16_t)a] << 4)); + else // aXmove, aYmove, aZmove, gXmove, gYmove and gZmove + return ((l2capinbuf[(uint16_t)a + 1] << 8) | l2capinbuf[(uint16_t)a]); } -double PS3BT::getAngle(Angle a) { - // Data for the Kionix KXPC4 used in DualShock 3 - const double sensivity = 204.6; // 0.66/3.3*1023 (660mV/g) - const double zeroG = 511.5; // 1.65/3.3*1023 (1,65V) +double PS3BT::getAngle(Angle a) { + double accXval; + double accYval; + double accZval; - double accXval = (zeroG - getSensor(aX)) / sensivity; // Convert to g's - accXval *= 2; - double accYval = (zeroG - getSensor(aY)) / sensivity; // Convert to g's - accYval *= 2; - double accZval = (zeroG - getSensor(aZ)) / sensivity; // Convert to g's - accZval *= 2; + if(PS3BTConnected) { + // Data for the Kionix KXPC4 used in the DualShock 3 + double sensivity = 204.6; // 0.66/3.3*1023 (660mV/g) + double zeroG = 511.5; // 1.65/3.3*1023 (1,65V) + accXval = ((double)getSensor(aX)-zeroG) / sensivity; // Convert to g's + accXval *= 2; + accYval = ((double)getSensor(aY)-zeroG) / sensivity; // Convert to g's + accYval *= 2; + accZval = ((double)getSensor(aZ)-zeroG) / sensivity; // Convert to g's + accZval *= 2; + } else if(PS3MoveBTConnected) { + // It's a Kionix KXSC4 inside the Motion controller + const uint16_t sensivity = 28285; // Find by experimenting + accXval = (double)getSensor(aXmove)/sensivity; + accYval = (double)getSensor(aYmove)/sensivity; + accZval = (double)getSensor(aZmove)/sensivity; + + if(accXval < -1) // Convert to g's + accXval = ((1+accXval)-(1-1.15))*(-1/0.15); + else if(accXval > 1) + accXval = ((1+accXval)-(1+1.15))*(-1/0.15); + + if(accYval < -1) // Convert to g's + accYval = ((1+accYval)-(1-1.15))*(-1/0.15); + else if(accYval > 1) + accYval = ((1+accYval)-(1+1.15))*(-1/0.15); + + if(accZval < -1) // Convert to g's + accZval = ((1+accZval)-(1-1.15))*(-1/0.15); + else if(accZval > 1) + accZval = ((1+accZval)-(1+1.15))*(-1/0.15); + } - double R = sqrt(accXval*accXval + accYval*accYval + accZval*accZval); - // convert read values to -π/2 to π/2 - Needed for atan2 - double angleX = acos(accXval/R)-0.5*PI; - double angleY = acos(accYval/R)-0.5*PI; - double angleZ = acos(accZval/R)-0.5*PI; + double R = sqrt(accXval*accXval + accYval*accYval + accZval*accZval); // Calculate the length of the force vector + // Normalize vectors + accXval = accXval/R; + accYval = accYval/R; + accZval = accZval/R; // Convert to 360 degrees resolution // atan2 outputs the value of -π to π (radians) - // We are then converting it to 0 to 2π and then to degrees + // We are then converting it to 0 to 2π and then to degrees if (a == Pitch) { - double angle = (atan2(-angleY,-angleZ)+PI)*RAD_TO_DEG; + double angle = (atan2(-accYval,-accZval)+PI)*RAD_TO_DEG; return angle; } else { - double angle = (atan2(-angleX,-angleZ)+PI)*RAD_TO_DEG; + double angle = (atan2(-accXval,-accZval)+PI)*RAD_TO_DEG; return angle; - } + } } bool PS3BT::getStatus(Status c) { diff --git a/PS3BT.h b/PS3BT.h index d355497a..8c1f43df 100644 --- a/PS3BT.h +++ b/PS3BT.h @@ -338,7 +338,7 @@ public: bool getButton(Button b); uint8_t getAnalogButton(AnalogButton a); uint8_t getAnalogHat(AnalogHat a); - uint32_t getSensor(Sensor a); + int32_t getSensor(Sensor a); double getAngle(Angle a); bool getStatus(Status c); String getStatusString(); diff --git a/examples/PS3BT/PS3BT.ino b/examples/PS3BT/PS3BT.ino index 413b6d75..2b6fbd95 100644 --- a/examples/PS3BT/PS3BT.ino +++ b/examples/PS3BT/PS3BT.ino @@ -142,10 +142,14 @@ void loop() } else { if(BT.getButton(SELECT_MOVE)) { Serial.print(F(" - Select")); - printTemperature = false; + printTemperature = !printTemperature; + while(BT.getButton(SELECT_MOVE)) + Usb.Task(); } if(BT.getButton(START_MOVE)) { Serial.print(F(" - Start")); - printTemperature = true; + printAngle = !printAngle; + while(BT.getButton(START_MOVE)) + Usb.Task(); } if(BT.getButton(TRIANGLE_MOVE)) { Serial.print(F(" - Triangle")); BT.moveSetBulb(Red); @@ -170,7 +174,13 @@ void loop() Serial.println(""); } } - if(printTemperature) { + if(printAngle) { + Serial.print(F("Pitch: ")); + Serial.print(BT.getAngle(Pitch)); + Serial.print(F("\tRoll: ")); + Serial.println(BT.getAngle(Roll)); + } + else if(printTemperature) { String templow; String temphigh; String input = String(BT.getSensor(tempMove));