Based on Peter Barrett\'s work on BlueUSB, I added support for the PS3 Sixaxis controller (both USB and Bluetooth). When connecting a Sixaxis via USB, it will be paired with the (hardcoded) MAC address of my Bluetooth dongle.

Dependencies:   mbed

Dependents:   PS3_BlueUSB_downstate

Committer:
BartJanssens
Date:
Tue Apr 26 16:09:17 2011 +0000
Revision:
0:99a111b75cb4

        

Who changed what in which revision?

UserRevisionLine numberNew contents of line
BartJanssens 0:99a111b75cb4 1
BartJanssens 0:99a111b75cb4 2 /*
BartJanssens 0:99a111b75cb4 3 Copyright (c) 2010 Peter Barrett
BartJanssens 0:99a111b75cb4 4
BartJanssens 0:99a111b75cb4 5 Permission is hereby granted, free of charge, to any person obtaining a copy
BartJanssens 0:99a111b75cb4 6 of this software and associated documentation files (the "Software"), to deal
BartJanssens 0:99a111b75cb4 7 in the Software without restriction, including without limitation the rights
BartJanssens 0:99a111b75cb4 8 to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
BartJanssens 0:99a111b75cb4 9 copies of the Software, and to permit persons to whom the Software is
BartJanssens 0:99a111b75cb4 10 furnished to do so, subject to the following conditions:
BartJanssens 0:99a111b75cb4 11
BartJanssens 0:99a111b75cb4 12 The above copyright notice and this permission notice shall be included in
BartJanssens 0:99a111b75cb4 13 all copies or substantial portions of the Software.
BartJanssens 0:99a111b75cb4 14
BartJanssens 0:99a111b75cb4 15 THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
BartJanssens 0:99a111b75cb4 16 IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
BartJanssens 0:99a111b75cb4 17 FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
BartJanssens 0:99a111b75cb4 18 AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
BartJanssens 0:99a111b75cb4 19 LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
BartJanssens 0:99a111b75cb4 20 OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
BartJanssens 0:99a111b75cb4 21 THE SOFTWARE.
BartJanssens 0:99a111b75cb4 22 */
BartJanssens 0:99a111b75cb4 23
BartJanssens 0:99a111b75cb4 24 /*
BartJanssens 0:99a111b75cb4 25 Tue Apr 26 2011 Bart Janssens: added PS3 Bluetooth support
BartJanssens 0:99a111b75cb4 26 */
BartJanssens 0:99a111b75cb4 27
BartJanssens 0:99a111b75cb4 28 #include <stdio.h>
BartJanssens 0:99a111b75cb4 29 #include <stdlib.h>
BartJanssens 0:99a111b75cb4 30 #include <string.h>
BartJanssens 0:99a111b75cb4 31
BartJanssens 0:99a111b75cb4 32 #include "Utils.h"
BartJanssens 0:99a111b75cb4 33 #include "USBHost.h"
BartJanssens 0:99a111b75cb4 34 #include "hci.h"
BartJanssens 0:99a111b75cb4 35 #include "ps3.h"
BartJanssens 0:99a111b75cb4 36
BartJanssens 0:99a111b75cb4 37 #include "mbed.h"
BartJanssens 0:99a111b75cb4 38
BartJanssens 0:99a111b75cb4 39
BartJanssens 0:99a111b75cb4 40 void printf(const BD_ADDR* addr)
BartJanssens 0:99a111b75cb4 41 {
BartJanssens 0:99a111b75cb4 42 const u8* a = addr->addr;
BartJanssens 0:99a111b75cb4 43 printf("%02X:%02X:%02X:%02X:%02X:%02X",a[5],a[4],a[3],a[2],a[1],a[0]);
BartJanssens 0:99a111b75cb4 44 }
BartJanssens 0:99a111b75cb4 45
BartJanssens 0:99a111b75cb4 46 #define MAX_HCL_SIZE 260
BartJanssens 0:99a111b75cb4 47 #define MAX_ACL_SIZE 400
BartJanssens 0:99a111b75cb4 48
BartJanssens 0:99a111b75cb4 49 class HCITransportUSB : public HCITransport
BartJanssens 0:99a111b75cb4 50 {
BartJanssens 0:99a111b75cb4 51 int _device;
BartJanssens 0:99a111b75cb4 52 u8* _hciBuffer;
BartJanssens 0:99a111b75cb4 53 u8* _aclBuffer;
BartJanssens 0:99a111b75cb4 54
BartJanssens 0:99a111b75cb4 55 public:
BartJanssens 0:99a111b75cb4 56 void Open(int device, u8* hciBuffer, u8* aclBuffer)
BartJanssens 0:99a111b75cb4 57 {
BartJanssens 0:99a111b75cb4 58 _device = device;
BartJanssens 0:99a111b75cb4 59 _hciBuffer = hciBuffer;
BartJanssens 0:99a111b75cb4 60 _aclBuffer = aclBuffer;
BartJanssens 0:99a111b75cb4 61 USBInterruptTransfer(_device,0x81,_hciBuffer,MAX_HCL_SIZE,HciCallback,this);
BartJanssens 0:99a111b75cb4 62 USBBulkTransfer(_device,0x82,_aclBuffer,MAX_ACL_SIZE,AclCallback,this);
BartJanssens 0:99a111b75cb4 63 }
BartJanssens 0:99a111b75cb4 64
BartJanssens 0:99a111b75cb4 65 static void HciCallback(int device, int endpoint, int status, u8* data, int len, void* userData)
BartJanssens 0:99a111b75cb4 66 {
BartJanssens 0:99a111b75cb4 67 HCI* t = ((HCITransportUSB*)userData)->_target;
BartJanssens 0:99a111b75cb4 68 if (t)
BartJanssens 0:99a111b75cb4 69 t->HCIRecv(data,len);
BartJanssens 0:99a111b75cb4 70 USBInterruptTransfer(device,0x81,data,MAX_HCL_SIZE,HciCallback,userData);
BartJanssens 0:99a111b75cb4 71 }
BartJanssens 0:99a111b75cb4 72
BartJanssens 0:99a111b75cb4 73 static void AclCallback(int device, int endpoint, int status, u8* data, int len, void* userData)
BartJanssens 0:99a111b75cb4 74 {
BartJanssens 0:99a111b75cb4 75 HCI* t = ((HCITransportUSB*)userData)->_target;
BartJanssens 0:99a111b75cb4 76 if (t)
BartJanssens 0:99a111b75cb4 77 t->ACLRecv(data,len);
BartJanssens 0:99a111b75cb4 78 USBBulkTransfer(device,0x82,data,MAX_ACL_SIZE,AclCallback,userData);
BartJanssens 0:99a111b75cb4 79 }
BartJanssens 0:99a111b75cb4 80
BartJanssens 0:99a111b75cb4 81 virtual void HCISend(const u8* data, int len)
BartJanssens 0:99a111b75cb4 82 {
BartJanssens 0:99a111b75cb4 83 USBControlTransfer(_device,REQUEST_TYPE_CLASS, 0, 0, 0,(u8*)data,len);
BartJanssens 0:99a111b75cb4 84 }
BartJanssens 0:99a111b75cb4 85
BartJanssens 0:99a111b75cb4 86 virtual void ACLSend(const u8* data, int len)
BartJanssens 0:99a111b75cb4 87 {
BartJanssens 0:99a111b75cb4 88 USBBulkTransfer(_device,0x02,(u8*)data,len);
BartJanssens 0:99a111b75cb4 89 }
BartJanssens 0:99a111b75cb4 90 };
BartJanssens 0:99a111b75cb4 91
BartJanssens 0:99a111b75cb4 92
BartJanssens 0:99a111b75cb4 93 #define WII_REMOTE 0x042500
BartJanssens 0:99a111b75cb4 94 #define PS3_REMOTE 0x080500
BartJanssens 0:99a111b75cb4 95
BartJanssens 0:99a111b75cb4 96 class HIDBluetooth
BartJanssens 0:99a111b75cb4 97 {
BartJanssens 0:99a111b75cb4 98 int _control; // Sockets for control (out) and interrupt (in)
BartJanssens 0:99a111b75cb4 99 int _interrupt;
BartJanssens 0:99a111b75cb4 100 int _devClass;
BartJanssens 0:99a111b75cb4 101 BD_ADDR _addr;
BartJanssens 0:99a111b75cb4 102 u8 _pad[2]; // Struct align
BartJanssens 0:99a111b75cb4 103 int _ready;
BartJanssens 0:99a111b75cb4 104 Timeout _timeout;
BartJanssens 0:99a111b75cb4 105 int _count;
BartJanssens 0:99a111b75cb4 106
BartJanssens 0:99a111b75cb4 107 public:
BartJanssens 0:99a111b75cb4 108 HIDBluetooth() : _control(0),_interrupt(0),_devClass(0), _ready(1) {};
BartJanssens 0:99a111b75cb4 109
BartJanssens 0:99a111b75cb4 110
BartJanssens 0:99a111b75cb4 111 bool InUse()
BartJanssens 0:99a111b75cb4 112 {
BartJanssens 0:99a111b75cb4 113 return _control != 0;
BartJanssens 0:99a111b75cb4 114 }
BartJanssens 0:99a111b75cb4 115
BartJanssens 0:99a111b75cb4 116 void attimeout()
BartJanssens 0:99a111b75cb4 117 {
BartJanssens 0:99a111b75cb4 118 printf("Timeout reached\r\n");
BartJanssens 0:99a111b75cb4 119 }
BartJanssens 0:99a111b75cb4 120
BartJanssens 0:99a111b75cb4 121 static void OnHidInterrupt(int socket, SocketState state,const u8* data, int len, void* userData)
BartJanssens 0:99a111b75cb4 122 {
BartJanssens 0:99a111b75cb4 123 HIDBluetooth* t = (HIDBluetooth*)userData;
BartJanssens 0:99a111b75cb4 124 t->_ready = 0;
BartJanssens 0:99a111b75cb4 125 if (data)
BartJanssens 0:99a111b75cb4 126 {
BartJanssens 0:99a111b75cb4 127 //printf("devClass = %06X \r\n",t->_devClass);
BartJanssens 0:99a111b75cb4 128 if (t->_devClass == WII_REMOTE && data[1] == 0x30)
BartJanssens 0:99a111b75cb4 129 {
BartJanssens 0:99a111b75cb4 130 printf("================wii====================\r\n");
BartJanssens 0:99a111b75cb4 131 t->WIILed();
BartJanssens 0:99a111b75cb4 132 t->WIIHid(); // ask for accelerometer
BartJanssens 0:99a111b75cb4 133 t->_devClass = 0;
BartJanssens 0:99a111b75cb4 134
BartJanssens 0:99a111b75cb4 135
BartJanssens 0:99a111b75cb4 136 const u8* d = data;
BartJanssens 0:99a111b75cb4 137 switch (d[1])
BartJanssens 0:99a111b75cb4 138 {
BartJanssens 0:99a111b75cb4 139 case 0x02:
BartJanssens 0:99a111b75cb4 140 {
BartJanssens 0:99a111b75cb4 141 int x = (signed char)d[3];
BartJanssens 0:99a111b75cb4 142 int y = (signed char)d[4];
BartJanssens 0:99a111b75cb4 143 printf("Mouse %2X dx:%d dy:%d\r\n",d[2],x,y);
BartJanssens 0:99a111b75cb4 144 }
BartJanssens 0:99a111b75cb4 145 break;
BartJanssens 0:99a111b75cb4 146
BartJanssens 0:99a111b75cb4 147 case 0x37: // Accelerometer http://wiki.wiimoteproject.com/Reports
BartJanssens 0:99a111b75cb4 148 {
BartJanssens 0:99a111b75cb4 149 int pad = (d[2] & 0x9F) | ((d[3] & 0x9F) << 8);
BartJanssens 0:99a111b75cb4 150 int x = (d[2] & 0x60) >> 5 | d[4] << 2;
BartJanssens 0:99a111b75cb4 151 int y = (d[3] & 0x20) >> 4 | d[5] << 2;
BartJanssens 0:99a111b75cb4 152 int z = (d[3] & 0x40) >> 5 | d[6] << 2;
BartJanssens 0:99a111b75cb4 153 printf("WII %04X %d %d %d\r\n",pad,x,y,z);
BartJanssens 0:99a111b75cb4 154 }
BartJanssens 0:99a111b75cb4 155 break;
BartJanssens 0:99a111b75cb4 156 default:
BartJanssens 0:99a111b75cb4 157 printHex(data,len);
BartJanssens 0:99a111b75cb4 158 }
BartJanssens 0:99a111b75cb4 159 }
BartJanssens 0:99a111b75cb4 160 if (t->_devClass == PS3_REMOTE)
BartJanssens 0:99a111b75cb4 161 {
BartJanssens 0:99a111b75cb4 162 t->_count ++;
BartJanssens 0:99a111b75cb4 163 if (t->_count == 25) t->_count = 1;
BartJanssens 0:99a111b75cb4 164 ParsePs3Result((data + 1), sizeof(ps3report),t->_count);
BartJanssens 0:99a111b75cb4 165 }
BartJanssens 0:99a111b75cb4 166 else {
BartJanssens 0:99a111b75cb4 167 printf("Not yet implemented \r\n");
BartJanssens 0:99a111b75cb4 168
BartJanssens 0:99a111b75cb4 169 }
BartJanssens 0:99a111b75cb4 170 }
BartJanssens 0:99a111b75cb4 171
BartJanssens 0:99a111b75cb4 172 }
BartJanssens 0:99a111b75cb4 173
BartJanssens 0:99a111b75cb4 174 static void OnHidControl(int socket, SocketState state, const u8* data, int len, void* userData)
BartJanssens 0:99a111b75cb4 175 {
BartJanssens 0:99a111b75cb4 176 //HIDBluetooth* t = (HIDBluetooth*)userData;
BartJanssens 0:99a111b75cb4 177
BartJanssens 0:99a111b75cb4 178 //printf("OnHidControl\r\n");
BartJanssens 0:99a111b75cb4 179
BartJanssens 0:99a111b75cb4 180 }
BartJanssens 0:99a111b75cb4 181
BartJanssens 0:99a111b75cb4 182 static void OnAcceptCtrlSocket(int socket, SocketState state, const u8* data, int len, void* userData)
BartJanssens 0:99a111b75cb4 183 {
BartJanssens 0:99a111b75cb4 184 HIDBluetooth* t = (HIDBluetooth*)userData;
BartJanssens 0:99a111b75cb4 185
BartJanssens 0:99a111b75cb4 186 t->_control = socket;
BartJanssens 0:99a111b75cb4 187
BartJanssens 0:99a111b75cb4 188 //printf("Ctrl Socket number = %d \r\n", socket);
BartJanssens 0:99a111b75cb4 189
BartJanssens 0:99a111b75cb4 190 Socket_Accept(socket,OnHidControl,userData);
BartJanssens 0:99a111b75cb4 191 u8 enable[6] = {
BartJanssens 0:99a111b75cb4 192 0x53, /* HIDP_TRANS_SET_REPORT | HIDP_DATA_RTYPE_FEATURE */
BartJanssens 0:99a111b75cb4 193 0xf4, 0x42, 0x03, 0x00, 0x00 };
BartJanssens 0:99a111b75cb4 194 Socket_Send(socket,enable,6);
BartJanssens 0:99a111b75cb4 195
BartJanssens 0:99a111b75cb4 196
BartJanssens 0:99a111b75cb4 197 }
BartJanssens 0:99a111b75cb4 198
BartJanssens 0:99a111b75cb4 199 static void OnAcceptDataSocket(int socket, SocketState state, const u8* data, int len, void* userData)
BartJanssens 0:99a111b75cb4 200 {
BartJanssens 0:99a111b75cb4 201 HIDBluetooth* t = (HIDBluetooth*)userData;
BartJanssens 0:99a111b75cb4 202 t->_interrupt = socket;
BartJanssens 0:99a111b75cb4 203
BartJanssens 0:99a111b75cb4 204 printf("OnAcceptDataSocket: Data Socket accept here \r\n");
BartJanssens 0:99a111b75cb4 205 printf("OnAcceptDataSocket: Data Socket number = %d \r\n", socket);
BartJanssens 0:99a111b75cb4 206
BartJanssens 0:99a111b75cb4 207 //printf("OnAcceptDataSocket: Ctrl Socket = %d Data Socket accept = %d \r\n", t->_control, t->_interrupt);
BartJanssens 0:99a111b75cb4 208
BartJanssens 0:99a111b75cb4 209 Socket_Accept(socket,OnHidInterrupt,userData);
BartJanssens 0:99a111b75cb4 210
BartJanssens 0:99a111b75cb4 211 //if (data)
BartJanssens 0:99a111b75cb4 212 // printHex(data,len);
BartJanssens 0:99a111b75cb4 213 }
BartJanssens 0:99a111b75cb4 214
BartJanssens 0:99a111b75cb4 215 void Open(BD_ADDR* bdAddr, inquiry_info* info)
BartJanssens 0:99a111b75cb4 216 {
BartJanssens 0:99a111b75cb4 217 printf("L2CAPAddr size %d\r\n",sizeof(L2CAPAddr));
BartJanssens 0:99a111b75cb4 218 _addr = *bdAddr;
BartJanssens 0:99a111b75cb4 219 L2CAPAddr sockAddr;
BartJanssens 0:99a111b75cb4 220 sockAddr.bdaddr = _addr;
BartJanssens 0:99a111b75cb4 221 sockAddr.psm = L2CAP_PSM_HID_INTR;
BartJanssens 0:99a111b75cb4 222 printf("Socket_Open size %d\r\n",sizeof(L2CAPAddr));
BartJanssens 0:99a111b75cb4 223 _interrupt = Socket_Open(SOCKET_L2CAP,&sockAddr.hdr,OnHidInterrupt,this);
BartJanssens 0:99a111b75cb4 224 sockAddr.psm = L2CAP_PSM_HID_CNTL;
BartJanssens 0:99a111b75cb4 225 _control = Socket_Open(SOCKET_L2CAP,&sockAddr.hdr,OnHidControl,this);
BartJanssens 0:99a111b75cb4 226
BartJanssens 0:99a111b75cb4 227 printfBytes("OPEN DEVICE CLASS",info->dev_class,3);
BartJanssens 0:99a111b75cb4 228 _devClass = (info->dev_class[0] << 16) | (info->dev_class[1] << 8) | info->dev_class[2];
BartJanssens 0:99a111b75cb4 229 }
BartJanssens 0:99a111b75cb4 230
BartJanssens 0:99a111b75cb4 231 void Listen(BD_ADDR* bdAddr, inquiry_info* info)
BartJanssens 0:99a111b75cb4 232 {
BartJanssens 0:99a111b75cb4 233 int result;
BartJanssens 0:99a111b75cb4 234 //printf("L2CAPAddr size %d\r\n",sizeof(L2CAPAddr));
BartJanssens 0:99a111b75cb4 235 _addr = *bdAddr;
BartJanssens 0:99a111b75cb4 236 L2CAPAddr sockAddr;
BartJanssens 0:99a111b75cb4 237 sockAddr.bdaddr = _addr;
BartJanssens 0:99a111b75cb4 238
BartJanssens 0:99a111b75cb4 239 _count = 1;
BartJanssens 0:99a111b75cb4 240 _ready = 1;
BartJanssens 0:99a111b75cb4 241
BartJanssens 0:99a111b75cb4 242 // set a buffer for the led&rumble report
BartJanssens 0:99a111b75cb4 243 u8 abuffer[37] = {
BartJanssens 0:99a111b75cb4 244 0x52, /* HIDP_TRANS_SET_REPORT | HIDP_DATA_RTYPE_OUTPUT */
BartJanssens 0:99a111b75cb4 245 0x01,
BartJanssens 0:99a111b75cb4 246 0x00, 0x00, 0x00, 0x00, 0x00,
BartJanssens 0:99a111b75cb4 247 0x00, 0x00, 0x00, 0x00, 0x1E,
BartJanssens 0:99a111b75cb4 248 0xff, 0x27, 0x10, 0x00, 0x32,
BartJanssens 0:99a111b75cb4 249 0xff, 0x27, 0x10, 0x00, 0x32,
BartJanssens 0:99a111b75cb4 250 0xff, 0x27, 0x10, 0x00, 0x32,
BartJanssens 0:99a111b75cb4 251 0xff, 0x27, 0x10, 0x00, 0x32,
BartJanssens 0:99a111b75cb4 252 0x00, 0x00, 0x00, 0x00, 0x00,
BartJanssens 0:99a111b75cb4 253 };
BartJanssens 0:99a111b75cb4 254 memcpy(_ledrumble,abuffer,37);
BartJanssens 0:99a111b75cb4 255
BartJanssens 0:99a111b75cb4 256 result = Socket_Listen(SOCKET_L2CAP,L2CAP_PSM_HID_CNTL,OnAcceptCtrlSocket,this);
BartJanssens 0:99a111b75cb4 257 printf("listen return code ctrl socket = %d \r\n", result);
BartJanssens 0:99a111b75cb4 258
BartJanssens 0:99a111b75cb4 259
BartJanssens 0:99a111b75cb4 260 result = Socket_Listen(SOCKET_L2CAP,L2CAP_PSM_HID_INTR,OnAcceptDataSocket,this);
BartJanssens 0:99a111b75cb4 261 printf("listen return code data socket = %d \r\n", result);
BartJanssens 0:99a111b75cb4 262
BartJanssens 0:99a111b75cb4 263 printfBytes("OPEN DEVICE CLASS",info->dev_class,3);
BartJanssens 0:99a111b75cb4 264 _devClass = (info->dev_class[0] << 16) | (info->dev_class[1] << 8) | info->dev_class[2];
BartJanssens 0:99a111b75cb4 265
BartJanssens 0:99a111b75cb4 266 while (_ready){ // wait till we receive data from PS3Hid
BartJanssens 0:99a111b75cb4 267 USBLoop();
BartJanssens 0:99a111b75cb4 268 }
BartJanssens 0:99a111b75cb4 269 USBLoop();
BartJanssens 0:99a111b75cb4 270
BartJanssens 0:99a111b75cb4 271
BartJanssens 0:99a111b75cb4 272
BartJanssens 0:99a111b75cb4 273 }
BartJanssens 0:99a111b75cb4 274
BartJanssens 0:99a111b75cb4 275 void Close()
BartJanssens 0:99a111b75cb4 276 {
BartJanssens 0:99a111b75cb4 277 if (_control)
BartJanssens 0:99a111b75cb4 278 Socket_Close(_control);
BartJanssens 0:99a111b75cb4 279 if (_interrupt)
BartJanssens 0:99a111b75cb4 280 Socket_Close(_interrupt);
BartJanssens 0:99a111b75cb4 281 _control = _interrupt = 0;
BartJanssens 0:99a111b75cb4 282 }
BartJanssens 0:99a111b75cb4 283
BartJanssens 0:99a111b75cb4 284 void WIILed(int id = 0x10)
BartJanssens 0:99a111b75cb4 285 {
BartJanssens 0:99a111b75cb4 286 u8 led[3] = {0x52, 0x11, id};
BartJanssens 0:99a111b75cb4 287 if (_control)
BartJanssens 0:99a111b75cb4 288 Socket_Send(_control,led,3);
BartJanssens 0:99a111b75cb4 289 }
BartJanssens 0:99a111b75cb4 290
BartJanssens 0:99a111b75cb4 291 void WIIHid(int report = 0x37)
BartJanssens 0:99a111b75cb4 292 {
BartJanssens 0:99a111b75cb4 293 u8 hid[4] = { 0x52, 0x12, 0x00, report };
BartJanssens 0:99a111b75cb4 294 if (_control != -1)
BartJanssens 0:99a111b75cb4 295 Socket_Send(_control,hid,4);
BartJanssens 0:99a111b75cb4 296 }
BartJanssens 0:99a111b75cb4 297
BartJanssens 0:99a111b75cb4 298
BartJanssens 0:99a111b75cb4 299
BartJanssens 0:99a111b75cb4 300 void Ps3Hid_Led(int i)
BartJanssens 0:99a111b75cb4 301 {
BartJanssens 0:99a111b75cb4 302 printf("Ps3Hid led %d\r\n",i);
BartJanssens 0:99a111b75cb4 303 u8 ledpattern[7] = {0x02, 0x04, 0x08, 0x10, 0x12, 0x14, 0x18 };
BartJanssens 0:99a111b75cb4 304 u8 buf[37];
BartJanssens 0:99a111b75cb4 305
BartJanssens 0:99a111b75cb4 306 if (i < 7) _ledrumble[11] = ledpattern[i];
BartJanssens 0:99a111b75cb4 307 memcpy(buf, _ledrumble, 37);
BartJanssens 0:99a111b75cb4 308
BartJanssens 0:99a111b75cb4 309 if (_control != -1)
BartJanssens 0:99a111b75cb4 310 Socket_Send(_control,buf,37);
BartJanssens 0:99a111b75cb4 311 wait_ms(4);
BartJanssens 0:99a111b75cb4 312 }
BartJanssens 0:99a111b75cb4 313
BartJanssens 0:99a111b75cb4 314 void Ps3Hid_Rumble(u8 duration_right, u8 power_right, u8 duration_left, u8 power_left )
BartJanssens 0:99a111b75cb4 315 {
BartJanssens 0:99a111b75cb4 316 printf("Ps3Hid rumble \r\n");
BartJanssens 0:99a111b75cb4 317 u8 buf[37];
BartJanssens 0:99a111b75cb4 318
BartJanssens 0:99a111b75cb4 319 memcpy(buf, _ledrumble, 37);
BartJanssens 0:99a111b75cb4 320 buf[3] = duration_right;
BartJanssens 0:99a111b75cb4 321 buf[4] = power_right;
BartJanssens 0:99a111b75cb4 322 buf[5] = duration_left;
BartJanssens 0:99a111b75cb4 323 buf[6] = power_left;
BartJanssens 0:99a111b75cb4 324
BartJanssens 0:99a111b75cb4 325 if (_control != -1)
BartJanssens 0:99a111b75cb4 326 Socket_Send(_control,buf,37);
BartJanssens 0:99a111b75cb4 327 wait_ms(4);
BartJanssens 0:99a111b75cb4 328 }
BartJanssens 0:99a111b75cb4 329
BartJanssens 0:99a111b75cb4 330 int CheckHID()
BartJanssens 0:99a111b75cb4 331 {
BartJanssens 0:99a111b75cb4 332 printf("CheckHID \r\n");
BartJanssens 0:99a111b75cb4 333 printf("Ctrl = %d Intr = %d \r\n", _control, _interrupt);
BartJanssens 0:99a111b75cb4 334 if (_control < 1) {
BartJanssens 0:99a111b75cb4 335 printf("Ps3 not ready \r\n");
BartJanssens 0:99a111b75cb4 336 return 1;
BartJanssens 0:99a111b75cb4 337 } else {
BartJanssens 0:99a111b75cb4 338 printf("Ps3 ready %d \r\n",_control);
BartJanssens 0:99a111b75cb4 339 return 0;
BartJanssens 0:99a111b75cb4 340 }
BartJanssens 0:99a111b75cb4 341 }
BartJanssens 0:99a111b75cb4 342 private:
BartJanssens 0:99a111b75cb4 343 u8 _ledrumble[37] ;
BartJanssens 0:99a111b75cb4 344 };
BartJanssens 0:99a111b75cb4 345
BartJanssens 0:99a111b75cb4 346
BartJanssens 0:99a111b75cb4 347 HCI* gHCI = 0;
BartJanssens 0:99a111b75cb4 348
BartJanssens 0:99a111b75cb4 349 #define MAX_HID_DEVICES 8
BartJanssens 0:99a111b75cb4 350
BartJanssens 0:99a111b75cb4 351 int GetConsoleChar();
BartJanssens 0:99a111b75cb4 352 class ShellApp
BartJanssens 0:99a111b75cb4 353 {
BartJanssens 0:99a111b75cb4 354 char _line[64];
BartJanssens 0:99a111b75cb4 355 HIDBluetooth _hids[MAX_HID_DEVICES];
BartJanssens 0:99a111b75cb4 356
BartJanssens 0:99a111b75cb4 357 public:
BartJanssens 0:99a111b75cb4 358 void Ready()
BartJanssens 0:99a111b75cb4 359 {
BartJanssens 0:99a111b75cb4 360 printf("HIDBluetooth %d\r\n",sizeof(HIDBluetooth));
BartJanssens 0:99a111b75cb4 361 memset(_hids,0,sizeof(_hids));
BartJanssens 0:99a111b75cb4 362 //Inquiry();
BartJanssens 0:99a111b75cb4 363 Scan();
BartJanssens 0:99a111b75cb4 364 }
BartJanssens 0:99a111b75cb4 365
BartJanssens 0:99a111b75cb4 366 // We have connected to a device
BartJanssens 0:99a111b75cb4 367 void ConnectionComplete(HCI* hci, connection_info* info)
BartJanssens 0:99a111b75cb4 368 {
BartJanssens 0:99a111b75cb4 369 printf("ConnectionComplete ");
BartJanssens 0:99a111b75cb4 370 BD_ADDR* a = &info->bdaddr;
BartJanssens 0:99a111b75cb4 371 printf(a);
BartJanssens 0:99a111b75cb4 372 BTDevice* bt = hci->Find(a);
BartJanssens 0:99a111b75cb4 373 HIDBluetooth* hid = NewHIDBluetooth();
BartJanssens 0:99a111b75cb4 374 printf("%08x %08x\r\n",bt,hid);
BartJanssens 0:99a111b75cb4 375 if (hid)
BartJanssens 0:99a111b75cb4 376 hid->Listen(a,&bt->_info); // use Listen for PS3, Open for WII
BartJanssens 0:99a111b75cb4 377 hid->Ps3Hid_Led(0); // set led 1
BartJanssens 0:99a111b75cb4 378 hid->Ps3Hid_Rumble(0x20,0xff,0x20,0xff); // rumble
BartJanssens 0:99a111b75cb4 379
BartJanssens 0:99a111b75cb4 380 }
BartJanssens 0:99a111b75cb4 381
BartJanssens 0:99a111b75cb4 382 HIDBluetooth* NewHIDBluetooth()
BartJanssens 0:99a111b75cb4 383 {
BartJanssens 0:99a111b75cb4 384 for (int i = 0; i < MAX_HID_DEVICES; i++)
BartJanssens 0:99a111b75cb4 385 if (!_hids[i].InUse())
BartJanssens 0:99a111b75cb4 386 return _hids+i;
BartJanssens 0:99a111b75cb4 387 return 0;
BartJanssens 0:99a111b75cb4 388 }
BartJanssens 0:99a111b75cb4 389
BartJanssens 0:99a111b75cb4 390 void ConnectDevices()
BartJanssens 0:99a111b75cb4 391 {
BartJanssens 0:99a111b75cb4 392 BTDevice* devs[8];
BartJanssens 0:99a111b75cb4 393 int count = gHCI->GetDevices(devs,8);
BartJanssens 0:99a111b75cb4 394 for (int i = 0; i < count; i++)
BartJanssens 0:99a111b75cb4 395 {
BartJanssens 0:99a111b75cb4 396 printfBytes("DEVICE CLASS",devs[i]->_info.dev_class,3);
BartJanssens 0:99a111b75cb4 397 if (devs[i]->_handle == 0)
BartJanssens 0:99a111b75cb4 398 {
BartJanssens 0:99a111b75cb4 399 BD_ADDR* bd = &devs[i]->_info.bdaddr;
BartJanssens 0:99a111b75cb4 400 printf("Connecting to ");
BartJanssens 0:99a111b75cb4 401 printf(bd);
BartJanssens 0:99a111b75cb4 402 printf("\r\n");
BartJanssens 0:99a111b75cb4 403 gHCI->CreateConnection(bd);
BartJanssens 0:99a111b75cb4 404 }
BartJanssens 0:99a111b75cb4 405 }
BartJanssens 0:99a111b75cb4 406 }
BartJanssens 0:99a111b75cb4 407
BartJanssens 0:99a111b75cb4 408 const char* ReadLine()
BartJanssens 0:99a111b75cb4 409 {
BartJanssens 0:99a111b75cb4 410 int i;
BartJanssens 0:99a111b75cb4 411 for (i = 0; i < 255; )
BartJanssens 0:99a111b75cb4 412 {
BartJanssens 0:99a111b75cb4 413 USBLoop();
BartJanssens 0:99a111b75cb4 414 int c = GetConsoleChar();
BartJanssens 0:99a111b75cb4 415 if (c == -1)
BartJanssens 0:99a111b75cb4 416 continue;
BartJanssens 0:99a111b75cb4 417 if (c == '\n' || c == 13)
BartJanssens 0:99a111b75cb4 418 break;
BartJanssens 0:99a111b75cb4 419 _line[i++] = c;
BartJanssens 0:99a111b75cb4 420 }
BartJanssens 0:99a111b75cb4 421 _line[i] = 0;
BartJanssens 0:99a111b75cb4 422 return _line;
BartJanssens 0:99a111b75cb4 423 }
BartJanssens 0:99a111b75cb4 424
BartJanssens 0:99a111b75cb4 425 void Inquiry()
BartJanssens 0:99a111b75cb4 426 {
BartJanssens 0:99a111b75cb4 427 printf("Inquiry..\r\n");
BartJanssens 0:99a111b75cb4 428 gHCI->Inquiry();
BartJanssens 0:99a111b75cb4 429 }
BartJanssens 0:99a111b75cb4 430
BartJanssens 0:99a111b75cb4 431 void List()
BartJanssens 0:99a111b75cb4 432 {
BartJanssens 0:99a111b75cb4 433 #if 0
BartJanssens 0:99a111b75cb4 434 printf("%d devices\r\n",_deviceCount);
BartJanssens 0:99a111b75cb4 435 for (int i = 0; i < _deviceCount; i++)
BartJanssens 0:99a111b75cb4 436 {
BartJanssens 0:99a111b75cb4 437 printf(&_devices[i].info.bdaddr);
BartJanssens 0:99a111b75cb4 438 printf("\r\n");
BartJanssens 0:99a111b75cb4 439 }
BartJanssens 0:99a111b75cb4 440 #endif
BartJanssens 0:99a111b75cb4 441 }
BartJanssens 0:99a111b75cb4 442
BartJanssens 0:99a111b75cb4 443 void Scan()
BartJanssens 0:99a111b75cb4 444 {
BartJanssens 0:99a111b75cb4 445 printf("Scanning...\r\n");
BartJanssens 0:99a111b75cb4 446 gHCI->WriteScanEnable();
BartJanssens 0:99a111b75cb4 447 }
BartJanssens 0:99a111b75cb4 448
BartJanssens 0:99a111b75cb4 449 void Connect()
BartJanssens 0:99a111b75cb4 450 {
BartJanssens 0:99a111b75cb4 451 ConnectDevices();
BartJanssens 0:99a111b75cb4 452 }
BartJanssens 0:99a111b75cb4 453
BartJanssens 0:99a111b75cb4 454
BartJanssens 0:99a111b75cb4 455 void Disconnect()
BartJanssens 0:99a111b75cb4 456 {
BartJanssens 0:99a111b75cb4 457 gHCI->DisconnectAll();
BartJanssens 0:99a111b75cb4 458 }
BartJanssens 0:99a111b75cb4 459
BartJanssens 0:99a111b75cb4 460 void CloseMouse()
BartJanssens 0:99a111b75cb4 461 {
BartJanssens 0:99a111b75cb4 462 }
BartJanssens 0:99a111b75cb4 463
BartJanssens 0:99a111b75cb4 464 void Quit()
BartJanssens 0:99a111b75cb4 465 {
BartJanssens 0:99a111b75cb4 466 CloseMouse();
BartJanssens 0:99a111b75cb4 467 }
BartJanssens 0:99a111b75cb4 468
BartJanssens 0:99a111b75cb4 469 void Run()
BartJanssens 0:99a111b75cb4 470 {
BartJanssens 0:99a111b75cb4 471 for(;;)
BartJanssens 0:99a111b75cb4 472 {
BartJanssens 0:99a111b75cb4 473 const char* cmd = ReadLine();
BartJanssens 0:99a111b75cb4 474 if (strcmp(cmd,"scan") == 0 || strcmp(cmd,"inquiry") == 0)
BartJanssens 0:99a111b75cb4 475 Inquiry();
BartJanssens 0:99a111b75cb4 476 else if (strcmp(cmd,"ls") == 0)
BartJanssens 0:99a111b75cb4 477 List();
BartJanssens 0:99a111b75cb4 478 else if (strcmp(cmd,"connect") == 0)
BartJanssens 0:99a111b75cb4 479 Connect();
BartJanssens 0:99a111b75cb4 480 else if (strcmp(cmd,"disconnect") == 0)
BartJanssens 0:99a111b75cb4 481 Disconnect();
BartJanssens 0:99a111b75cb4 482 else if (strcmp(cmd,"q")== 0)
BartJanssens 0:99a111b75cb4 483 {
BartJanssens 0:99a111b75cb4 484 Quit();
BartJanssens 0:99a111b75cb4 485 break;
BartJanssens 0:99a111b75cb4 486 } else {
BartJanssens 0:99a111b75cb4 487 printf("eh? %s\r\n",cmd);
BartJanssens 0:99a111b75cb4 488 }
BartJanssens 0:99a111b75cb4 489 }
BartJanssens 0:99a111b75cb4 490 }
BartJanssens 0:99a111b75cb4 491 };
BartJanssens 0:99a111b75cb4 492
BartJanssens 0:99a111b75cb4 493 // Instance
BartJanssens 0:99a111b75cb4 494 ShellApp gApp;
BartJanssens 0:99a111b75cb4 495
BartJanssens 0:99a111b75cb4 496 static int HciCallback(HCI* hci, HCI_CALLBACK_EVENT evt, const u8* data, int len)
BartJanssens 0:99a111b75cb4 497 {
BartJanssens 0:99a111b75cb4 498 switch (evt)
BartJanssens 0:99a111b75cb4 499 {
BartJanssens 0:99a111b75cb4 500 case CALLBACK_READY:
BartJanssens 0:99a111b75cb4 501 printf("CALLBACK_READY\r\n");
BartJanssens 0:99a111b75cb4 502 gApp.Ready();
BartJanssens 0:99a111b75cb4 503 break;
BartJanssens 0:99a111b75cb4 504
BartJanssens 0:99a111b75cb4 505 case CALLBACK_INQUIRY_RESULT:
BartJanssens 0:99a111b75cb4 506 printf("CALLBACK_INQUIRY_RESULT ");
BartJanssens 0:99a111b75cb4 507 printf((BD_ADDR*)data);
BartJanssens 0:99a111b75cb4 508 printf("\r\n");
BartJanssens 0:99a111b75cb4 509 break;
BartJanssens 0:99a111b75cb4 510
BartJanssens 0:99a111b75cb4 511 case CALLBACK_INQUIRY_DONE:
BartJanssens 0:99a111b75cb4 512 printf("CALLBACK_INQUIRY_DONE\r\n");
BartJanssens 0:99a111b75cb4 513 gApp.ConnectDevices();
BartJanssens 0:99a111b75cb4 514 break;
BartJanssens 0:99a111b75cb4 515
BartJanssens 0:99a111b75cb4 516 case CALLBACK_REMOTE_NAME:
BartJanssens 0:99a111b75cb4 517 {
BartJanssens 0:99a111b75cb4 518 BD_ADDR* addr = (BD_ADDR*)data;
BartJanssens 0:99a111b75cb4 519 const char* name = (const char*)(data + 6);
BartJanssens 0:99a111b75cb4 520 printf(addr);
BartJanssens 0:99a111b75cb4 521 printf(" % s\r\n",name);
BartJanssens 0:99a111b75cb4 522 }
BartJanssens 0:99a111b75cb4 523 break;
BartJanssens 0:99a111b75cb4 524
BartJanssens 0:99a111b75cb4 525 case CALLBACK_CONNECTION_COMPLETE:
BartJanssens 0:99a111b75cb4 526 gApp.ConnectionComplete(hci,(connection_info*)data);
BartJanssens 0:99a111b75cb4 527 break;
BartJanssens 0:99a111b75cb4 528 };
BartJanssens 0:99a111b75cb4 529 return 0;
BartJanssens 0:99a111b75cb4 530 }
BartJanssens 0:99a111b75cb4 531
BartJanssens 0:99a111b75cb4 532 // these should be placed in the DMA SRAM
BartJanssens 0:99a111b75cb4 533 typedef struct
BartJanssens 0:99a111b75cb4 534 {
BartJanssens 0:99a111b75cb4 535 u8 _hciBuffer[MAX_HCL_SIZE];
BartJanssens 0:99a111b75cb4 536 u8 _aclBuffer[MAX_ACL_SIZE];
BartJanssens 0:99a111b75cb4 537 } SRAMPlacement;
BartJanssens 0:99a111b75cb4 538
BartJanssens 0:99a111b75cb4 539 HCITransportUSB _HCITransportUSB;
BartJanssens 0:99a111b75cb4 540 HCI _HCI;
BartJanssens 0:99a111b75cb4 541
BartJanssens 0:99a111b75cb4 542 u8* USBGetBuffer(u32* len);
BartJanssens 0:99a111b75cb4 543 int OnBluetoothInsert(int device)
BartJanssens 0:99a111b75cb4 544 {
BartJanssens 0:99a111b75cb4 545 printf("Bluetooth inserted of %d\r\n",device);
BartJanssens 0:99a111b75cb4 546 u32 sramLen;
BartJanssens 0:99a111b75cb4 547 u8* sram = USBGetBuffer(&sramLen);
BartJanssens 0:99a111b75cb4 548 sram = (u8*)(((u32)sram + 1023) & ~1023);
BartJanssens 0:99a111b75cb4 549 SRAMPlacement* s = (SRAMPlacement*)sram;
BartJanssens 0:99a111b75cb4 550 _HCITransportUSB.Open(device,s->_hciBuffer,s->_aclBuffer);
BartJanssens 0:99a111b75cb4 551 _HCI.Open(&_HCITransportUSB,HciCallback);
BartJanssens 0:99a111b75cb4 552 RegisterSocketHandler(SOCKET_L2CAP,&_HCI);
BartJanssens 0:99a111b75cb4 553 gHCI = &_HCI;
BartJanssens 0:99a111b75cb4 554 //gApp.Inquiry();
BartJanssens 0:99a111b75cb4 555 //gApp.Scan();
BartJanssens 0:99a111b75cb4 556 gApp.Connect();
BartJanssens 0:99a111b75cb4 557 return 0;
BartJanssens 0:99a111b75cb4 558 }
BartJanssens 0:99a111b75cb4 559
BartJanssens 0:99a111b75cb4 560 void TestShell()
BartJanssens 0:99a111b75cb4 561 {
BartJanssens 0:99a111b75cb4 562 USBInit();
BartJanssens 0:99a111b75cb4 563 gApp.Run();
BartJanssens 0:99a111b75cb4 564 }