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 #include "stdlib.h"
BartJanssens 0:99a111b75cb4 25 #include "stdio.h"
BartJanssens 0:99a111b75cb4 26 #include "string.h"
BartJanssens 0:99a111b75cb4 27
BartJanssens 0:99a111b75cb4 28 #include "Utils.h"
BartJanssens 0:99a111b75cb4 29 #include "USBHost.h"
BartJanssens 0:99a111b75cb4 30
BartJanssens 0:99a111b75cb4 31
BartJanssens 0:99a111b75cb4 32 int MassStorage_ReadCapacity(int device, u32* blockCount, u32* blockSize);
BartJanssens 0:99a111b75cb4 33 int MassStorage_ReadBlock(int device, u32 block, u8* dst);
BartJanssens 0:99a111b75cb4 34 int MassStorage_WriteBlock(int device, u32 block, const u8* dst);
BartJanssens 0:99a111b75cb4 35
BartJanssens 0:99a111b75cb4 36
BartJanssens 0:99a111b75cb4 37 #define ERR_BAD_CSW_SIGNATURE -200
BartJanssens 0:99a111b75cb4 38
BartJanssens 0:99a111b75cb4 39 #define CBW_SIGNATURE 0x43425355
BartJanssens 0:99a111b75cb4 40 #define CSW_SIGNATURE 0x53425355
BartJanssens 0:99a111b75cb4 41
BartJanssens 0:99a111b75cb4 42 // Command Block
BartJanssens 0:99a111b75cb4 43 typedef struct
BartJanssens 0:99a111b75cb4 44 {
BartJanssens 0:99a111b75cb4 45 u32 Signature;
BartJanssens 0:99a111b75cb4 46 u32 Tag;
BartJanssens 0:99a111b75cb4 47 u32 TransferLength;
BartJanssens 0:99a111b75cb4 48 u8 Flags;
BartJanssens 0:99a111b75cb4 49 u8 LUN;
BartJanssens 0:99a111b75cb4 50 u8 CBLength;
BartJanssens 0:99a111b75cb4 51 u8 CB[16]; // only 6 really
BartJanssens 0:99a111b75cb4 52 } CBW;
BartJanssens 0:99a111b75cb4 53
BartJanssens 0:99a111b75cb4 54 // Status block
BartJanssens 0:99a111b75cb4 55 typedef struct
BartJanssens 0:99a111b75cb4 56 {
BartJanssens 0:99a111b75cb4 57 u32 Signature;
BartJanssens 0:99a111b75cb4 58 u32 Tag;
BartJanssens 0:99a111b75cb4 59 u32 DataResidue;
BartJanssens 0:99a111b75cb4 60 u8 Status;
BartJanssens 0:99a111b75cb4 61 } CSW;
BartJanssens 0:99a111b75cb4 62
BartJanssens 0:99a111b75cb4 63 int SCSIRequestSense(int device);
BartJanssens 0:99a111b75cb4 64
BartJanssens 0:99a111b75cb4 65 int DoSCSI(int device, const u8* cmd, int cmdLen, int flags, u8* data, u32 transferLen)
BartJanssens 0:99a111b75cb4 66 {
BartJanssens 0:99a111b75cb4 67 CBW cbw;
BartJanssens 0:99a111b75cb4 68 cbw.Signature = CBW_SIGNATURE;
BartJanssens 0:99a111b75cb4 69 cbw.Tag = 0;
BartJanssens 0:99a111b75cb4 70 cbw.TransferLength = transferLen;
BartJanssens 0:99a111b75cb4 71 cbw.Flags = flags;
BartJanssens 0:99a111b75cb4 72 cbw.LUN = 0;
BartJanssens 0:99a111b75cb4 73 cbw.CBLength = cmdLen;
BartJanssens 0:99a111b75cb4 74 memset(cbw.CB,0,sizeof(cbw.CB));
BartJanssens 0:99a111b75cb4 75 memcpy(cbw.CB,cmd,cmdLen);
BartJanssens 0:99a111b75cb4 76
BartJanssens 0:99a111b75cb4 77 int r;
BartJanssens 0:99a111b75cb4 78 r = USBBulkTransfer(device,0x01,(u8*)&cbw,31); // Send the command
BartJanssens 0:99a111b75cb4 79 if (r < 0)
BartJanssens 0:99a111b75cb4 80 return r;
BartJanssens 0:99a111b75cb4 81
BartJanssens 0:99a111b75cb4 82 if (data)
BartJanssens 0:99a111b75cb4 83 {
BartJanssens 0:99a111b75cb4 84 r = USBBulkTransfer(device,flags | 1,data,transferLen);
BartJanssens 0:99a111b75cb4 85 if (r < 0)
BartJanssens 0:99a111b75cb4 86 return r;
BartJanssens 0:99a111b75cb4 87 }
BartJanssens 0:99a111b75cb4 88
BartJanssens 0:99a111b75cb4 89 CSW csw;
BartJanssens 0:99a111b75cb4 90 csw.Signature = 0;
BartJanssens 0:99a111b75cb4 91 r = USBBulkTransfer(device,0x81,(u8*)&csw,13);
BartJanssens 0:99a111b75cb4 92 if (r < 0)
BartJanssens 0:99a111b75cb4 93 return r;
BartJanssens 0:99a111b75cb4 94
BartJanssens 0:99a111b75cb4 95 if (csw.Signature != CSW_SIGNATURE)
BartJanssens 0:99a111b75cb4 96 return ERR_BAD_CSW_SIGNATURE;
BartJanssens 0:99a111b75cb4 97
BartJanssens 0:99a111b75cb4 98 // ModeSense?
BartJanssens 0:99a111b75cb4 99 if (csw.Status == 1 && cmd[0] != 3)
BartJanssens 0:99a111b75cb4 100 return SCSIRequestSense(device);
BartJanssens 0:99a111b75cb4 101
BartJanssens 0:99a111b75cb4 102 return csw.Status;
BartJanssens 0:99a111b75cb4 103 }
BartJanssens 0:99a111b75cb4 104
BartJanssens 0:99a111b75cb4 105 int SCSITestUnitReady(int device)
BartJanssens 0:99a111b75cb4 106 {
BartJanssens 0:99a111b75cb4 107 u8 cmd[6];
BartJanssens 0:99a111b75cb4 108 memset(cmd,0,6);
BartJanssens 0:99a111b75cb4 109 return DoSCSI(device,cmd,6,DEVICE_TO_HOST,0,0);
BartJanssens 0:99a111b75cb4 110 }
BartJanssens 0:99a111b75cb4 111
BartJanssens 0:99a111b75cb4 112 int SCSIRequestSense(int device)
BartJanssens 0:99a111b75cb4 113 {
BartJanssens 0:99a111b75cb4 114 u8 cmd[6] = {0x03,0,0,0,18,0};
BartJanssens 0:99a111b75cb4 115 u8 result[18];
BartJanssens 0:99a111b75cb4 116 int r = DoSCSI(device,cmd,6,DEVICE_TO_HOST,result,18);
BartJanssens 0:99a111b75cb4 117 return r;
BartJanssens 0:99a111b75cb4 118 }
BartJanssens 0:99a111b75cb4 119
BartJanssens 0:99a111b75cb4 120 int SCSIInquiry(int device)
BartJanssens 0:99a111b75cb4 121 {
BartJanssens 0:99a111b75cb4 122 u8 cmd[6] = {0x12,0,0,0,36,0};
BartJanssens 0:99a111b75cb4 123 u8 result[36+2];
BartJanssens 0:99a111b75cb4 124 result[36] = '\n';
BartJanssens 0:99a111b75cb4 125 result[37] = 0;
BartJanssens 0:99a111b75cb4 126 int r = DoSCSI(device,cmd,6,DEVICE_TO_HOST,result,36);
BartJanssens 0:99a111b75cb4 127 if (r == 0)
BartJanssens 0:99a111b75cb4 128 printf((const char*)result + 8);
BartJanssens 0:99a111b75cb4 129 return r;
BartJanssens 0:99a111b75cb4 130 }
BartJanssens 0:99a111b75cb4 131
BartJanssens 0:99a111b75cb4 132 int SCSIReadCapacity(int device, u32* blockCount, u32* blockSize)
BartJanssens 0:99a111b75cb4 133 {
BartJanssens 0:99a111b75cb4 134 u8 cmd[10] = {0x25,0,0,0,8,0,0,0,0,0};
BartJanssens 0:99a111b75cb4 135 u8 result[8];
BartJanssens 0:99a111b75cb4 136 *blockSize = 0;
BartJanssens 0:99a111b75cb4 137 *blockCount = 0;
BartJanssens 0:99a111b75cb4 138 int r = DoSCSI(device,cmd,10,DEVICE_TO_HOST,result,8);
BartJanssens 0:99a111b75cb4 139 if (r == 0)
BartJanssens 0:99a111b75cb4 140 {
BartJanssens 0:99a111b75cb4 141 *blockCount = BE32(result);
BartJanssens 0:99a111b75cb4 142 *blockSize = BE32(result+4);
BartJanssens 0:99a111b75cb4 143 }
BartJanssens 0:99a111b75cb4 144 return r;
BartJanssens 0:99a111b75cb4 145 }
BartJanssens 0:99a111b75cb4 146
BartJanssens 0:99a111b75cb4 147 int SCSITransfer(int device, u32 blockAddr, u32 blockCount, u8* dst, u32 blockSize, int direction)
BartJanssens 0:99a111b75cb4 148 {
BartJanssens 0:99a111b75cb4 149 // USB hardware will only do 4k per transfer
BartJanssens 0:99a111b75cb4 150 while (blockCount*blockSize > 4096)
BartJanssens 0:99a111b75cb4 151 {
BartJanssens 0:99a111b75cb4 152 int count = 4096/blockSize;
BartJanssens 0:99a111b75cb4 153 int r = SCSITransfer(device,blockAddr,count,dst,blockSize,direction);
BartJanssens 0:99a111b75cb4 154 dst += count*blockSize;
BartJanssens 0:99a111b75cb4 155 blockAddr += count;
BartJanssens 0:99a111b75cb4 156 blockCount -= count;
BartJanssens 0:99a111b75cb4 157 }
BartJanssens 0:99a111b75cb4 158
BartJanssens 0:99a111b75cb4 159 u8 cmd[10];
BartJanssens 0:99a111b75cb4 160 memset(cmd,0,10);
BartJanssens 0:99a111b75cb4 161 cmd[0] = (direction == DEVICE_TO_HOST) ? 0x28 : 0x2A;
BartJanssens 0:99a111b75cb4 162 BE32(blockAddr,cmd+2);
BartJanssens 0:99a111b75cb4 163 BE16(blockCount,cmd+7);
BartJanssens 0:99a111b75cb4 164 return DoSCSI(device,cmd,10,direction,dst,blockSize*blockCount);
BartJanssens 0:99a111b75cb4 165 }
BartJanssens 0:99a111b75cb4 166
BartJanssens 0:99a111b75cb4 167 int MassStorage_ReadCapacity(int device, u32* blockCount, u32* blockSize)
BartJanssens 0:99a111b75cb4 168 {
BartJanssens 0:99a111b75cb4 169 return SCSIReadCapacity(device,blockCount,blockSize);
BartJanssens 0:99a111b75cb4 170 }
BartJanssens 0:99a111b75cb4 171
BartJanssens 0:99a111b75cb4 172 int MassStorage_Read(int device, u32 blockAddr, u32 blockCount, u8* dst, u32 blockSize = 512)
BartJanssens 0:99a111b75cb4 173 {
BartJanssens 0:99a111b75cb4 174 return SCSITransfer(device,blockAddr,blockCount,dst,blockSize,DEVICE_TO_HOST);
BartJanssens 0:99a111b75cb4 175 }
BartJanssens 0:99a111b75cb4 176
BartJanssens 0:99a111b75cb4 177 int MassStorage_Write(int device, u32 blockAddr, u32 blockCount, u8* dst, u32 blockSize = 512)
BartJanssens 0:99a111b75cb4 178 {
BartJanssens 0:99a111b75cb4 179 return SCSITransfer(device,blockAddr,blockCount,dst,blockSize,HOST_TO_DEVICE);
BartJanssens 0:99a111b75cb4 180 }