2012-05-26 01:48:12 +02:00
/* Copyright (C) 2012 Kristian Lauszus, TKJ Electronics. All rights reserved.
This software may be distributed and modified under the terms of the GNU
General Public License version 2 ( GPL2 ) as published by the Free Software
Foundation and appearing in the file GPL2 . TXT included in the packaging of
this file . Please note that GPL2 Section 2 [ b ] requires that all works based
on this software must also be made publicly available under the terms of
the GPL2 ( " Copyleft " ) .
Contact information
- - - - - - - - - - - - - - - - - - -
Kristian Lauszus , TKJ Electronics
Web : http : //www.tkjelectronics.com
e - mail : kristianl @ tkjelectronics . com
*/
# include "PS3USB.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
2012-08-04 19:08:43 +02:00
const uint8_t PS3_REPORT_BUFFER [ ] PROGMEM = {
2012-05-26 01:48:12 +02:00
0x00 , 0x00 , 0x00 , 0x00 , 0x00 ,
0x00 , 0x00 , 0x00 , 0x00 , 0x00 ,
0xff , 0x27 , 0x10 , 0x00 , 0x32 ,
0xff , 0x27 , 0x10 , 0x00 , 0x32 ,
0xff , 0x27 , 0x10 , 0x00 , 0x32 ,
0xff , 0x27 , 0x10 , 0x00 , 0x32 ,
0x00 , 0x00 , 0x00 , 0x00 , 0x00 ,
0x00 , 0x00 , 0x00 , 0x00 , 0x00 ,
0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00
} ;
2012-08-04 19:08:43 +02:00
const uint8_t MOVE_REPORT_BUFFER [ ] PROGMEM = {
2012-05-26 01:48:12 +02:00
0x02 , 0x00 , // Always 0x02, 0x00,
0x00 , 0x00 , 0x00 , // r, g, b,
0x00 , // Always 0x00,
0x00 // Rumble
} ;
PS3USB : : PS3USB ( 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
bPollEnable ( false ) // don't start polling before dongle is connected
{
2012-08-04 19:08:43 +02:00
for ( uint8_t i = 0 ; i < PS3_MAX_ENDPOINTS ; i + + ) {
2012-05-26 01:48:12 +02:00
epInfo [ i ] . epAddr = 0 ;
epInfo [ i ] . maxPktSize = ( i ) ? 0 : 8 ;
epInfo [ i ] . epAttribs = 0 ;
epInfo [ i ] . bmNakPower = ( i ) ? USB_NAK_NOWAIT : USB_NAK_MAX_POWER ;
}
if ( pUsb ) // register in USB subsystem
pUsb - > RegisterDeviceClass ( 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 ;
}
2012-08-04 19:08:43 +02:00
uint8_t PS3USB : : Init ( uint8_t parent , uint8_t port , bool lowspeed ) {
2012-05-26 01:48:12 +02:00
uint8_t buf [ sizeof ( USB_DEVICE_DESCRIPTOR ) ] ;
uint8_t rcode ;
UsbDevice * p = NULL ;
EpInfo * oldep_ptr = NULL ;
uint16_t PID ;
uint16_t VID ;
// get memory address of USB device address pool
AddressPool & addrPool = pUsb - > GetAddressPool ( ) ;
# ifdef EXTRADEBUG
2012-05-27 16:12:27 +02:00
Notify ( PSTR ( " \r \n PS3USB Init " ) ) ;
2012-05-26 01:48:12 +02:00
# endif
// check if address has already been assigned to an instance
2012-08-04 19:08:43 +02:00
if ( bAddress ) {
2012-05-26 01:48:12 +02:00
# ifdef DEBUG
Notify ( PSTR ( " \r \n Address in use " ) ) ;
# endif
return USB_ERROR_CLASS_INSTANCE_ALREADY_IN_USE ;
}
// Get pointer to pseudo device with address 0 assigned
p = addrPool . GetUsbDevicePtr ( 0 ) ;
2012-08-04 19:08:43 +02:00
if ( ! p ) {
2012-05-26 01:48:12 +02:00
# ifdef DEBUG
Notify ( PSTR ( " \r \n Address not found " ) ) ;
# endif
return USB_ERROR_ADDRESS_NOT_FOUND_IN_POOL ;
}
2012-08-04 19:08:43 +02:00
if ( ! p - > epinfo ) {
2012-05-26 01:48:12 +02:00
# ifdef DEBUG
Notify ( PSTR ( " \r \n epinfo is null " ) ) ;
# endif
return USB_ERROR_EPINFO_IS_NULL ;
}
// Save old pointer to EP_RECORD of address 0
oldep_ptr = p - > epinfo ;
// Temporary assign new pointer to epInfo to p->epinfo in order to avoid toggle inconsistence
p - > epinfo = epInfo ;
p - > lowspeed = lowspeed ;
// Get device descriptor
rcode = pUsb - > getDevDescr ( 0 , 0 , sizeof ( USB_DEVICE_DESCRIPTOR ) , ( uint8_t * ) buf ) ; // Get device descriptor - addr, ep, nbytes, data
2012-08-08 17:32:45 +02:00
VID = ( ( USB_DEVICE_DESCRIPTOR * ) buf ) - > idVendor ;
PID = ( ( USB_DEVICE_DESCRIPTOR * ) buf ) - > idProduct ;
if ( VID ! = PS3_VID | | ( PID ! = PS3_PID & & PID ! = PS3NAVIGATION_PID & & PID ! = PS3MOVE_PID ) )
goto FailUnknownDevice ;
2012-05-26 01:48:12 +02:00
// Restore p->epinfo
p - > epinfo = oldep_ptr ;
if ( rcode )
goto FailGetDevDescr ;
// Allocate new address according to device class
bAddress = addrPool . AllocAddress ( parent , false , port ) ;
if ( ! bAddress )
return USB_ERROR_OUT_OF_ADDRESS_SPACE_IN_POOL ;
// Extract Max Packet Size from device descriptor
epInfo [ 0 ] . maxPktSize = ( uint8_t ) ( ( USB_DEVICE_DESCRIPTOR * ) buf ) - > bMaxPacketSize0 ;
// Assign new address to the device
rcode = pUsb - > setAddr ( 0 , 0 , bAddress ) ;
2012-08-04 19:08:43 +02:00
if ( rcode ) {
2012-05-26 01:48:12 +02:00
p - > lowspeed = false ;
addrPool . FreeAddress ( bAddress ) ;
bAddress = 0 ;
# ifdef DEBUG
Notify ( PSTR ( " \r \n setAddr: " ) ) ;
# endif
PrintHex < uint8_t > ( rcode ) ;
return rcode ;
}
# ifdef EXTRADEBUG
Notify ( PSTR ( " \r \n Addr: " ) ) ;
PrintHex < uint8_t > ( bAddress ) ;
# endif
p - > lowspeed = false ;
//get pointer to assigned address record
p = addrPool . GetUsbDevicePtr ( bAddress ) ;
if ( ! p )
return USB_ERROR_ADDRESS_NOT_FOUND_IN_POOL ;
p - > lowspeed = lowspeed ;
// Assign epInfo to epinfo pointer - only EP0 is known
rcode = pUsb - > setEpInfoEntry ( bAddress , 1 , epInfo ) ;
if ( rcode )
goto FailSetDevTblEntry ;
2012-08-08 17:32:45 +02:00
/* 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 */
2012-05-26 01:48:12 +02:00
2012-08-08 17:32:45 +02:00
/* Initialize data structures for endpoints of device */
epInfo [ PS3_OUTPUT_PIPE ] . epAddr = 0x02 ; // PS3 output endpoint
epInfo [ PS3_OUTPUT_PIPE ] . epAttribs = EP_INTERRUPT ;
epInfo [ PS3_OUTPUT_PIPE ] . bmNakPower = USB_NAK_NOWAIT ; // Only poll once for interrupt endpoints
epInfo [ PS3_OUTPUT_PIPE ] . maxPktSize = EP_MAXPKTSIZE ;
epInfo [ PS3_OUTPUT_PIPE ] . bmSndToggle = bmSNDTOG0 ;
epInfo [ PS3_OUTPUT_PIPE ] . bmRcvToggle = bmRCVTOG0 ;
epInfo [ PS3_INPUT_PIPE ] . epAddr = 0x01 ; // PS3 report endpoint
epInfo [ PS3_INPUT_PIPE ] . epAttribs = EP_INTERRUPT ;
epInfo [ PS3_INPUT_PIPE ] . bmNakPower = USB_NAK_NOWAIT ; // Only poll once for interrupt endpoints
epInfo [ PS3_INPUT_PIPE ] . maxPktSize = EP_MAXPKTSIZE ;
epInfo [ PS3_INPUT_PIPE ] . bmSndToggle = bmSNDTOG0 ;
epInfo [ PS3_INPUT_PIPE ] . bmRcvToggle = bmRCVTOG0 ;
2012-05-26 01:48:12 +02:00
2012-08-08 17:32:45 +02:00
rcode = pUsb - > setEpInfoEntry ( bAddress , 3 , epInfo ) ;
if ( rcode )
goto FailSetDevTblEntry ;
2012-05-26 01:48:12 +02:00
2012-08-08 17:32:45 +02:00
delay ( 200 ) ; //Give time for address change
2012-05-26 01:48:12 +02:00
2012-08-08 17:32:45 +02:00
rcode = pUsb - > setConf ( bAddress , epInfo [ PS3_CONTROL_PIPE ] . epAddr , 1 ) ;
if ( rcode )
goto FailSetConf ;
2012-05-26 01:48:12 +02:00
2012-08-08 17:32:45 +02:00
if ( PID = = PS3_PID | | PID = = PS3NAVIGATION_PID ) {
if ( PID = = PS3_PID ) {
2012-05-26 01:48:12 +02:00
# ifdef DEBUG
2012-08-08 17:32:45 +02:00
Notify ( PSTR ( " \r \n Dualshock 3 Controller Connected " ) ) ;
2012-05-26 01:48:12 +02:00
# endif
2012-08-08 17:32:45 +02:00
PS3Connected = true ;
} else { // must be a navigation controller
2012-05-26 01:48:12 +02:00
# ifdef DEBUG
2012-08-08 17:32:45 +02:00
Notify ( PSTR ( " \r \n Navigation Controller Connected " ) ) ;
2012-05-26 01:48:12 +02:00
# endif
2012-08-08 17:32:45 +02:00
PS3NavigationConnected = true ;
}
/* Set internal bluetooth address and request for data */
setBdaddr ( my_bdaddr ) ;
enable_sixaxis ( ) ;
setLedOn ( LED1 ) ;
2012-05-26 01:48:12 +02:00
2012-08-08 17:32:45 +02:00
// Needed for PS3 Dualshock and Navigation commands to work
for ( uint8_t i = 0 ; i < PS3_REPORT_BUFFER_SIZE ; i + + )
writeBuf [ i ] = pgm_read_byte ( & PS3_REPORT_BUFFER [ i ] ) ;
2012-05-26 01:48:12 +02:00
2012-08-08 17:32:45 +02:00
for ( uint8_t i = 6 ; i < 10 ; i + + )
readBuf [ i ] = 0x7F ; // Set the analog joystick values to center position
}
else { // must be a Motion controller
2012-05-26 01:48:12 +02:00
# ifdef DEBUG
2012-08-08 17:32:45 +02:00
Notify ( PSTR ( " \r \n Motion Controller Connected " ) ) ;
2012-05-26 01:48:12 +02:00
# endif
2012-08-08 17:32:45 +02:00
PS3MoveConnected = true ;
setMoveBdaddr ( my_bdaddr ) ; // Set internal bluetooth address
moveSetBulb ( Red ) ;
2012-05-26 01:48:12 +02:00
2012-08-08 17:32:45 +02:00
// Needed for Move commands to work
for ( uint8_t i = 0 ; i < MOVE_REPORT_BUFFER_SIZE ; i + + )
writeBuf [ i ] = pgm_read_byte ( & MOVE_REPORT_BUFFER [ i ] ) ;
2012-05-26 01:48:12 +02:00
}
bPollEnable = true ;
Notify ( PSTR ( " \r \n " ) ) ;
2012-05-27 16:12:27 +02:00
timer = millis ( ) ;
2012-05-26 01:48:12 +02:00
return 0 ; // successful configuration
/* diagnostic messages */
FailGetDevDescr :
# ifdef DEBUG
Notify ( PSTR ( " \r \n getDevDescr: " ) ) ;
# endif
goto Fail ;
FailSetDevTblEntry :
# ifdef DEBUG
Notify ( PSTR ( " \r \n setDevTblEn: " ) ) ;
# endif
goto Fail ;
FailSetConf :
# ifdef DEBUG
Notify ( PSTR ( " \r \n setConf: " ) ) ;
# endif
goto Fail ;
FailUnknownDevice :
# ifdef DEBUG
Notify ( PSTR ( " \r \n Unknown Device Connected - VID: " ) ) ;
PrintHex < uint16_t > ( VID ) ;
Notify ( PSTR ( " PID: " ) ) ;
PrintHex < uint16_t > ( PID ) ;
# endif
2012-08-08 17:32:45 +02:00
rcode = USB_DEV_CONFIG_ERROR_DEVICE_NOT_SUPPORTED ;
2012-05-26 01:48:12 +02:00
goto Fail ;
Fail :
# ifdef DEBUG
Notify ( PSTR ( " \r \n PS3 Init Failed, error code: " ) ) ;
2012-08-08 17:32:45 +02:00
Serial . print ( rcode , HEX ) ;
2012-05-26 01:48:12 +02:00
# endif
Release ( ) ;
return rcode ;
}
/* Performs a cleanup after failed Init() attempt */
2012-08-04 19:08:43 +02:00
uint8_t PS3USB : : Release ( ) {
2012-05-26 01:48:12 +02:00
PS3Connected = false ;
PS3MoveConnected = false ;
PS3NavigationConnected = false ;
pUsb - > GetAddressPool ( ) . FreeAddress ( bAddress ) ;
bAddress = 0 ;
bPollEnable = false ;
return 0 ;
}
2012-08-04 19:08:43 +02:00
uint8_t PS3USB : : Poll ( ) {
2012-05-26 01:48:12 +02:00
if ( ! bPollEnable )
return 0 ;
if ( PS3Connected | | PS3NavigationConnected ) {
uint16_t BUFFER_SIZE = EP_MAXPKTSIZE ;
pUsb - > inTransfer ( bAddress , epInfo [ PS3_INPUT_PIPE ] . epAddr , & BUFFER_SIZE , readBuf ) ; // input on endpoint 1
if ( millis ( ) - timer > 100 ) { // Loop 100ms before processing data
readReport ( ) ;
# ifdef PRINTREPORT
printReport ( ) ; // Uncomment "#define PRINTREPORT" to print the report send by the PS3 Controllers
# endif
}
}
else if ( PS3MoveConnected ) { // One can only set the color of the bulb, set the rumble, set and get the bluetooth address and calibrate the magnetometer via USB
if ( millis ( ) - timer > 4000 ) // Send at least every 4th second
{
Move_Command ( writeBuf , MOVE_REPORT_BUFFER_SIZE ) ; // The Bulb and rumble values, has to be written again and again, for it to stay turned on
timer = millis ( ) ;
}
}
return 0 ;
}
2012-08-04 19:08:43 +02:00
void PS3USB : : readReport ( ) {
2012-05-26 01:48:12 +02:00
if ( readBuf = = NULL )
return ;
2012-05-30 23:55:07 +02:00
ButtonState = ( uint32_t ) ( readBuf [ 2 ] | ( ( uint16_t ) readBuf [ 3 ] < < 8 ) | ( ( uint32_t ) readBuf [ 4 ] < < 16 ) ) ;
2012-05-26 01:48:12 +02:00
//Notify(PSTR("\r\nButtonState");
//PrintHex<uint32_t>(ButtonState);
2012-08-04 19:08:43 +02:00
if ( ButtonState ! = OldButtonState ) {
2012-05-26 01:48:12 +02:00
buttonChanged = true ;
if ( ButtonState ! = 0x00 ) {
buttonPressed = true ;
buttonReleased = false ;
} else {
buttonPressed = false ;
buttonReleased = true ;
}
}
2012-08-04 19:08:43 +02:00
else {
2012-05-26 01:48:12 +02:00
buttonChanged = false ;
buttonPressed = false ;
buttonReleased = false ;
}
OldButtonState = ButtonState ;
}
2012-08-04 19:08:43 +02:00
void PS3USB : : printReport ( ) { //Uncomment "#define PRINTREPORT" to print the report send by the PS3 Controllers
2012-05-26 01:48:12 +02:00
if ( readBuf = = NULL )
return ;
for ( uint8_t i = 0 ; i < PS3_REPORT_BUFFER_SIZE ; i + + )
{
PrintHex < uint8_t > ( readBuf [ i ] ) ;
Serial . print ( " " ) ;
}
Serial . println ( " " ) ;
}
2012-08-04 19:08:43 +02:00
bool PS3USB : : getButton ( Button b ) {
2012-05-26 01:48:12 +02:00
if ( readBuf = = NULL )
return false ;
if ( ( readBuf [ ( uint16_t ) b > > 8 ] & ( ( uint8_t ) b & 0xff ) ) > 0 )
return true ;
else
return false ;
}
2012-08-04 19:08:43 +02:00
uint8_t PS3USB : : getAnalogButton ( AnalogButton a ) {
2012-05-26 01:48:12 +02:00
if ( readBuf = = NULL )
return 0 ;
return ( uint8_t ) ( readBuf [ ( uint16_t ) a ] ) ;
}
2012-08-04 19:08:43 +02:00
uint8_t PS3USB : : getAnalogHat ( AnalogHat a ) {
2012-05-26 01:48:12 +02:00
if ( readBuf = = NULL )
return 0 ;
return ( uint8_t ) ( readBuf [ ( uint16_t ) a ] ) ;
}
2012-08-04 19:08:43 +02:00
uint16_t PS3USB : : getSensor ( Sensor a ) {
2012-05-26 01:48:12 +02:00
if ( readBuf = = NULL )
return 0 ;
return ( ( readBuf [ ( uint16_t ) a ] < < 8 ) | readBuf [ ( uint16_t ) a + 1 ] ) ;
}
2012-05-27 16:12:27 +02:00
double PS3USB : : getAngle ( Angle a ) {
2012-05-26 01:48:12 +02:00
if ( PS3Connected ) {
2012-05-27 16:12:27 +02:00
double accXval ;
double accYval ;
double accZval ;
2012-05-26 01:48:12 +02:00
// Data for the Kionix KXPC4 used in the DualShock 3
const double zeroG = 511.5 ; // 1.65/3.3*1023 (1,65V)
accXval = - ( ( double ) getSensor ( aX ) - zeroG ) ;
accYval = - ( ( double ) getSensor ( aY ) - zeroG ) ;
accZval = - ( ( double ) getSensor ( aZ ) - zeroG ) ;
2012-05-27 16:12:27 +02:00
// 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 ( accYval , accZval ) + PI ) * RAD_TO_DEG ;
return angle ;
} else {
double angle = ( atan2 ( accXval , accZval ) + PI ) * RAD_TO_DEG ;
return angle ;
}
} else
return 0 ;
2012-05-26 01:48:12 +02:00
}
2012-08-04 19:08:43 +02:00
bool PS3USB : : getStatus ( Status c ) {
2012-05-26 01:48:12 +02:00
if ( readBuf = = NULL )
return false ;
if ( readBuf [ ( uint16_t ) c > > 8 ] = = ( ( uint8_t ) c & 0xff ) )
return true ;
return false ;
}
2012-08-04 19:08:43 +02:00
String PS3USB : : getStatusString ( ) {
2012-05-26 01:48:12 +02:00
if ( PS3Connected | | PS3NavigationConnected )
{
char statusOutput [ 100 ] ;
strcpy ( statusOutput , " ConnectionStatus: " ) ;
if ( getStatus ( Plugged ) ) strcat ( statusOutput , " Plugged " ) ;
else if ( getStatus ( Unplugged ) ) strcat ( statusOutput , " Unplugged " ) ;
else strcat ( statusOutput , " Error " ) ;
strcat ( statusOutput , " - PowerRating: " ) ;
2012-05-27 16:12:27 +02:00
2012-05-26 01:48:12 +02:00
if ( getStatus ( Charging ) ) strcat ( statusOutput , " Charging " ) ;
else if ( getStatus ( NotCharging ) ) strcat ( statusOutput , " Not Charging " ) ;
else if ( getStatus ( Shutdown ) ) strcat ( statusOutput , " Shutdown " ) ;
else if ( getStatus ( Dying ) ) strcat ( statusOutput , " Dying " ) ;
else if ( getStatus ( Low ) ) strcat ( statusOutput , " Low " ) ;
else if ( getStatus ( High ) ) strcat ( statusOutput , " High " ) ;
else if ( getStatus ( Full ) ) strcat ( statusOutput , " Full " ) ;
else strcat ( statusOutput , " Error " ) ;
strcat ( statusOutput , " - WirelessStatus: " ) ;
if ( getStatus ( CableRumble ) ) strcat ( statusOutput , " Cable - Rumble is on " ) ;
else if ( getStatus ( Cable ) ) strcat ( statusOutput , " Cable - Rumble is off " ) ;
else if ( getStatus ( BluetoothRumble ) ) strcat ( statusOutput , " Bluetooth - Rumble is on " ) ;
else if ( getStatus ( Bluetooth ) ) strcat ( statusOutput , " Bluetooth - Rumble is off " ) ;
else strcat ( statusOutput , " Error " ) ;
return statusOutput ;
}
}
/* Playstation Sixaxis Dualshock and Navigation Controller commands */
2012-08-04 19:08:43 +02:00
void PS3USB : : PS3_Command ( uint8_t * data , uint16_t nbytes ) {
2012-05-26 01:48:12 +02:00
//bmRequest = Host to device (0x00) | Class (0x20) | Interface (0x01) = 0x21, bRequest = Set Report (0x09), Report ID (0x01), Report Type (Output 0x02), interface (0x00), datalength, datalength, data)
pUsb - > ctrlReq ( bAddress , epInfo [ PS3_CONTROL_PIPE ] . epAddr , bmREQ_HID_OUT , HID_REQUEST_SET_REPORT , 0x01 , 0x02 , 0x00 , nbytes , nbytes , data , NULL ) ;
}
2012-08-04 19:08:43 +02:00
void PS3USB : : setAllOff ( ) {
2012-05-26 01:48:12 +02:00
for ( uint8_t i = 0 ; i < PS3_REPORT_BUFFER_SIZE ; i + + )
writeBuf [ i ] = pgm_read_byte ( & PS3_REPORT_BUFFER [ i ] ) ; // Reset buffer
PS3_Command ( writeBuf , PS3_REPORT_BUFFER_SIZE ) ;
}
2012-08-04 19:08:43 +02:00
void PS3USB : : setRumbleOff ( ) {
2012-05-26 01:48:12 +02:00
writeBuf [ 1 ] = 0x00 ;
writeBuf [ 2 ] = 0x00 ; //low mode off
writeBuf [ 3 ] = 0x00 ;
writeBuf [ 4 ] = 0x00 ; //high mode off
PS3_Command ( writeBuf , PS3_REPORT_BUFFER_SIZE ) ;
}
2012-08-04 19:08:43 +02:00
void PS3USB : : setRumbleOn ( Rumble mode ) {
2012-05-26 01:48:12 +02:00
/* Still not totally sure how it works, maybe something like this instead?
* 3 - duration_right
* 4 - power_right
* 5 - duration_left
* 6 - power_left
*/
if ( ( mode & 0x30 ) > 0 )
{
writeBuf [ 1 ] = 0xfe ;
writeBuf [ 3 ] = 0xfe ;
if ( mode = = RumbleHigh )
{
writeBuf [ 2 ] = 0 ; //low mode off
writeBuf [ 4 ] = 0xff ; //high mode on
}
else
{
writeBuf [ 2 ] = 0xff ; //low mode on
writeBuf [ 4 ] = 0 ; //high mode off
}
PS3_Command ( writeBuf , PS3_REPORT_BUFFER_SIZE ) ;
}
}
2012-08-04 19:08:43 +02:00
void PS3USB : : setLedOff ( LED a ) {
2012-05-26 01:48:12 +02:00
writeBuf [ 9 ] & = ~ ( ( uint8_t ) ( ( ( uint16_t ) a & 0x0f ) < < 1 ) ) ;
PS3_Command ( writeBuf , PS3_REPORT_BUFFER_SIZE ) ;
}
2012-08-04 19:08:43 +02:00
void PS3USB : : setLedOn ( LED a ) {
2012-05-26 01:48:12 +02:00
writeBuf [ 9 ] | = ( uint8_t ) ( ( ( uint16_t ) a & 0x0f ) < < 1 ) ;
PS3_Command ( writeBuf , PS3_REPORT_BUFFER_SIZE ) ;
}
2012-08-04 19:08:43 +02:00
void PS3USB : : setLedToggle ( LED a ) {
2012-05-26 01:48:12 +02:00
writeBuf [ 9 ] ^ = ( uint8_t ) ( ( ( uint16_t ) a & 0x0f ) < < 1 ) ;
PS3_Command ( writeBuf , PS3_REPORT_BUFFER_SIZE ) ;
}
2012-08-04 19:08:43 +02:00
void PS3USB : : setBdaddr ( uint8_t * BDADDR ) {
2012-05-26 01:48:12 +02:00
/* 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 ] = 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
Notify ( PSTR ( " \r \n Bluetooth Address was set to: " ) ) ;
for ( int8_t i = 5 ; i > 0 ; i - - )
{
PrintHex < uint8_t > ( my_bdaddr [ i ] ) ;
Serial . print ( " : " ) ;
}
PrintHex < uint8_t > ( my_bdaddr [ 0 ] ) ;
# endif
return ;
}
2012-08-04 19:08:43 +02:00
void PS3USB : : enable_sixaxis ( ) { //Command used to enable the Dualshock 3 and Navigation controller to send data via USB
2012-05-26 01:48:12 +02:00
uint8_t cmd_buf [ 4 ] ;
cmd_buf [ 0 ] = 0x42 ; // Special PS3 Controller enable commands
cmd_buf [ 1 ] = 0x0c ;
cmd_buf [ 2 ] = 0x00 ;
cmd_buf [ 3 ] = 0x00 ;
//bmRequest = Host to device (0x00) | Class (0x20) | Interface (0x01) = 0x21, bRequest = Set Report (0x09), Report ID (0xF4), Report Type (Feature 0x03), interface (0x00), datalength, datalength, data)
pUsb - > ctrlReq ( bAddress , epInfo [ PS3_CONTROL_PIPE ] . epAddr , bmREQ_HID_OUT , HID_REQUEST_SET_REPORT , 0xF4 , 0x03 , 0x00 , 4 , 4 , cmd_buf , NULL ) ;
}
/* Playstation Move Controller commands */
2012-08-04 19:08:43 +02:00
void PS3USB : : Move_Command ( uint8_t * data , uint16_t nbytes ) {
2012-05-26 01:48:12 +02:00
pUsb - > outTransfer ( bAddress , epInfo [ PS3_OUTPUT_PIPE ] . epAddr , nbytes , data ) ;
}
2012-08-04 19:08:43 +02:00
void PS3USB : : moveSetBulb ( uint8_t r , uint8_t g , uint8_t b ) { //Use this to set the Color using RGB values
2012-05-26 01:48:12 +02:00
// set the Bulb's values into the write buffer
writeBuf [ 2 ] = r ;
writeBuf [ 3 ] = g ;
writeBuf [ 4 ] = b ;
Move_Command ( writeBuf , MOVE_REPORT_BUFFER_SIZE ) ;
}
2012-08-04 19:08:43 +02:00
void PS3USB : : moveSetBulb ( Colors color ) { //Use this to set the Color using the predefined colors in "enums.h"
2012-05-26 01:48:12 +02:00
moveSetBulb ( ( uint8_t ) ( color > > 16 ) , ( uint8_t ) ( color > > 8 ) , ( uint8_t ) ( color ) ) ;
}
2012-08-04 19:08:43 +02:00
void PS3USB : : moveSetRumble ( uint8_t rumble ) {
2012-05-26 01:48:12 +02:00
# ifdef DEBUG
if ( rumble < 64 & & rumble ! = 0 ) // The rumble value has to at least 64, or approximately 25% (64/255*100)
Notify ( PSTR ( " \r \n The rumble value has to at least 64, or approximately 25% " ) ) ;
# endif
//set the rumble value into the write buffer
writeBuf [ 6 ] = rumble ;
Move_Command ( writeBuf , MOVE_REPORT_BUFFER_SIZE ) ;
}
2012-08-04 19:08:43 +02:00
void PS3USB : : setMoveBdaddr ( uint8_t * BDADDR ) {
2012-05-26 01:48:12 +02:00
/* Set the internal bluetooth address */
uint8_t buf [ 11 ] ;
buf [ 0 ] = 0x05 ;
buf [ 7 ] = 0x10 ;
buf [ 8 ] = 0x01 ;
buf [ 9 ] = 0x02 ;
buf [ 10 ] = 0x12 ;
for ( uint8_t i = 0 ; i < 6 ; i + + )
buf [ i + 1 ] = 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_HID_OUT , HID_REQUEST_SET_REPORT , 0x05 , 0x03 , 0x00 , 11 , 11 , buf , NULL ) ;
# ifdef DEBUG
Notify ( PSTR ( " \r \n Bluetooth Address was set to: " ) ) ;
for ( int8_t i = 5 ; i > 0 ; i - - )
{
PrintHex < uint8_t > ( my_bdaddr [ i ] ) ;
Serial . print ( " : " ) ;
}
PrintHex < uint8_t > ( my_bdaddr [ 0 ] ) ;
# endif
return ;
}