Mike R / Mbed 2 deprecated Pinscape_Controller_V2

Dependencies:   mbed FastIO FastPWM USBDevice

Fork of Pinscape_Controller by Mike R

Committer:
mjr
Date:
Thu Jan 23 04:09:24 2020 +0000
Revision:
105:6a25bbfae1e4
Parent:
99:8139b0c274f4
Child:
108:bd5d4bd4383b
Fix AEDR-8300 reverse orientation option

Who changed what in which revision?

UserRevisionLine numberNew contents of line
mjr 3:3514575d4f86 1 /* Copyright (c) 2010-2011 mbed.org, MIT License
mjr 3:3514575d4f86 2 * Modified Mouse code for Joystick - WH 2012
mjr 3:3514575d4f86 3 *
mjr 3:3514575d4f86 4 * Permission is hereby granted, free of charge, to any person obtaining a copy of this software
mjr 3:3514575d4f86 5 * and associated documentation files (the "Software"), to deal in the Software without
mjr 3:3514575d4f86 6 * restriction, including without limitation the rights to use, copy, modify, merge, publish,
mjr 3:3514575d4f86 7 * distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the
mjr 3:3514575d4f86 8 * Software is furnished to do so, subject to the following conditions:
mjr 3:3514575d4f86 9 *
mjr 3:3514575d4f86 10 * The above copyright notice and this permission notice shall be included in all copies or
mjr 3:3514575d4f86 11 * substantial portions of the Software.
mjr 3:3514575d4f86 12 *
mjr 3:3514575d4f86 13 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING
mjr 3:3514575d4f86 14 * BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
mjr 3:3514575d4f86 15 * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM,
mjr 3:3514575d4f86 16 * DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
mjr 3:3514575d4f86 17 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
mjr 3:3514575d4f86 18 */
mjr 3:3514575d4f86 19
mjr 3:3514575d4f86 20 #include "stdint.h"
mjr 3:3514575d4f86 21 #include "USBJoystick.h"
mjr 21:5048e16cc9ef 22
mjr 21:5048e16cc9ef 23 #include "config.h" // Pinscape configuration
mjr 21:5048e16cc9ef 24
mjr 35:e959ffba78fd 25
mjr 35:e959ffba78fd 26
mjr 48:058ace2aed1d 27 // Maximum report sizes
mjr 77:0b96f6867312 28 const int MAX_REPORT_JS_TX = USBJoystick::reportLen;
mjr 48:058ace2aed1d 29 const int MAX_REPORT_JS_RX = 8;
mjr 63:5cd1a5f3a41b 30 const int MAX_REPORT_KB_TX = 8;
mjr 63:5cd1a5f3a41b 31 const int MAX_REPORT_KB_RX = 4;
mjr 48:058ace2aed1d 32
mjr 11:bd9da7088e6e 33 bool USBJoystick::update(int16_t x, int16_t y, int16_t z, uint32_t buttons, uint16_t status)
mjr 3:3514575d4f86 34 {
mjr 3:3514575d4f86 35 _x = x;
mjr 3:3514575d4f86 36 _y = y;
mjr 3:3514575d4f86 37 _z = z;
mjr 11:bd9da7088e6e 38 _buttonsLo = (uint16_t)(buttons & 0xffff);
mjr 11:bd9da7088e6e 39 _buttonsHi = (uint16_t)((buttons >> 16) & 0xffff);
mjr 10:976666ffa4ef 40 _status = status;
mjr 3:3514575d4f86 41
mjr 3:3514575d4f86 42 // send the report
mjr 3:3514575d4f86 43 return update();
mjr 3:3514575d4f86 44 }
mjr 35:e959ffba78fd 45
mjr 11:bd9da7088e6e 46 bool USBJoystick::update()
mjr 11:bd9da7088e6e 47 {
mjr 3:3514575d4f86 48 HID_REPORT report;
mjr 63:5cd1a5f3a41b 49
mjr 3:3514575d4f86 50 // Fill the report according to the Joystick Descriptor
mjr 6:cc35eb643e8f 51 #define put(idx, val) (report.data[idx] = (val) & 0xff, report.data[(idx)+1] = ((val) >> 8) & 0xff)
mjr 53:9b2611964afc 52 #define putbe(idx, val) (report.data[(idx)+1] = (val) & 0xff, report.data[idx] = ((val) >> 8) & 0xff)
mjr 40:cc0d9814522b 53 #define putl(idx, val) (put(idx, val), put((idx)+2, (val) >> 16))
mjr 54:fd77a6b2f76c 54 #define putlbe(idx, val) (putbe((idx)+2, val), putbe(idx, (val) >> 16))
mjr 77:0b96f6867312 55 #define put64(idx, val) (putl(idx, val), putl((idx)+4, (val) >> 32))
mjr 63:5cd1a5f3a41b 56 put(0, _status);
mjr 63:5cd1a5f3a41b 57 put(2, 0); // second word of status - zero in high bit identifies as normal joystick report
mjr 63:5cd1a5f3a41b 58 put(4, _buttonsLo);
mjr 63:5cd1a5f3a41b 59 put(6, _buttonsHi);
mjr 63:5cd1a5f3a41b 60 put(8, _x);
mjr 63:5cd1a5f3a41b 61 put(10, _y);
mjr 63:5cd1a5f3a41b 62 put(12, _z);
mjr 21:5048e16cc9ef 63
mjr 21:5048e16cc9ef 64 // important: keep reportLen in sync with the actual byte length of
mjr 21:5048e16cc9ef 65 // the reports we build here
mjr 63:5cd1a5f3a41b 66 report.length = reportLen;
mjr 3:3514575d4f86 67
mjr 5:a70c0bce770d 68 // send the report
mjr 10:976666ffa4ef 69 return sendTO(&report, 100);
mjr 10:976666ffa4ef 70 }
mjr 10:976666ffa4ef 71
mjr 35:e959ffba78fd 72 bool USBJoystick::kbUpdate(uint8_t data[8])
mjr 35:e959ffba78fd 73 {
mjr 35:e959ffba78fd 74 // set up the report
mjr 35:e959ffba78fd 75 HID_REPORT report;
mjr 35:e959ffba78fd 76 report.data[0] = REPORT_ID_KB; // report ID = keyboard
mjr 35:e959ffba78fd 77 memcpy(&report.data[1], data, 8); // copy the kb report data
mjr 35:e959ffba78fd 78 report.length = 9; // length = ID prefix + kb report length
mjr 35:e959ffba78fd 79
mjr 35:e959ffba78fd 80 // send it to endpoint 4 (the keyboard interface endpoint)
mjr 35:e959ffba78fd 81 return writeTO(EP4IN, report.data, report.length, MAX_PACKET_SIZE_EPINT, 100);
mjr 35:e959ffba78fd 82 }
mjr 35:e959ffba78fd 83
mjr 35:e959ffba78fd 84 bool USBJoystick::mediaUpdate(uint8_t data)
mjr 35:e959ffba78fd 85 {
mjr 35:e959ffba78fd 86 // set up the report
mjr 35:e959ffba78fd 87 HID_REPORT report;
mjr 35:e959ffba78fd 88 report.data[0] = REPORT_ID_MEDIA; // report ID = media
mjr 35:e959ffba78fd 89 report.data[1] = data; // key pressed bits
mjr 35:e959ffba78fd 90 report.length = 2;
mjr 35:e959ffba78fd 91
mjr 35:e959ffba78fd 92 // send it
mjr 35:e959ffba78fd 93 return writeTO(EP4IN, report.data, report.length, MAX_PACKET_SIZE_EPINT, 100);
mjr 35:e959ffba78fd 94 }
mjr 35:e959ffba78fd 95
mjr 52:8298b2a73eb2 96 bool USBJoystick::sendPlungerStatus(
mjr 87:8d35c74403af 97 int npix, int plungerPos, int flags, uint32_t avgScanTime, uint32_t processingTime)
mjr 52:8298b2a73eb2 98 {
mjr 52:8298b2a73eb2 99 HID_REPORT report;
mjr 52:8298b2a73eb2 100
mjr 52:8298b2a73eb2 101 // Set the special status bits to indicate it's an extended
mjr 52:8298b2a73eb2 102 // exposure report.
mjr 63:5cd1a5f3a41b 103 put(0, 0x87FF);
mjr 52:8298b2a73eb2 104
mjr 52:8298b2a73eb2 105 // start at the second byte
mjr 63:5cd1a5f3a41b 106 int ofs = 2;
mjr 52:8298b2a73eb2 107
mjr 52:8298b2a73eb2 108 // write the report subtype (0) to byte 2
mjr 52:8298b2a73eb2 109 report.data[ofs++] = 0;
mjr 52:8298b2a73eb2 110
mjr 52:8298b2a73eb2 111 // write the number of pixels to bytes 3-4
mjr 52:8298b2a73eb2 112 put(ofs, uint16_t(npix));
mjr 52:8298b2a73eb2 113 ofs += 2;
mjr 52:8298b2a73eb2 114
mjr 87:8d35c74403af 115 // write the detected plunger position to bytes 5-6
mjr 87:8d35c74403af 116 put(ofs, uint16_t(plungerPos));
mjr 52:8298b2a73eb2 117 ofs += 2;
mjr 52:8298b2a73eb2 118
mjr 86:e30a1f60f783 119 // Add the calibration mode flag if applicable
mjr 52:8298b2a73eb2 120 extern bool plungerCalMode;
mjr 86:e30a1f60f783 121 if (plungerCalMode) flags |= 0x04;
mjr 86:e30a1f60f783 122
mjr 86:e30a1f60f783 123 // write the flags to byte 7
mjr 52:8298b2a73eb2 124 report.data[ofs++] = flags;
mjr 52:8298b2a73eb2 125
mjr 52:8298b2a73eb2 126 // write the average scan time in 10us intervals to bytes 8-10
mjr 52:8298b2a73eb2 127 uint32_t t = uint32_t(avgScanTime / 10);
mjr 52:8298b2a73eb2 128 report.data[ofs++] = t & 0xff;
mjr 52:8298b2a73eb2 129 report.data[ofs++] = (t >> 8) & 0xff;
mjr 52:8298b2a73eb2 130 report.data[ofs++] = (t >> 16) & 0xff;
mjr 52:8298b2a73eb2 131
mjr 52:8298b2a73eb2 132 // write the processing time to bytes 11-13
mjr 52:8298b2a73eb2 133 t = uint32_t(processingTime / 10);
mjr 52:8298b2a73eb2 134 report.data[ofs++] = t & 0xff;
mjr 52:8298b2a73eb2 135 report.data[ofs++] = (t >> 8) & 0xff;
mjr 52:8298b2a73eb2 136 report.data[ofs++] = (t >> 16) & 0xff;
mjr 52:8298b2a73eb2 137
mjr 52:8298b2a73eb2 138 // send the report
mjr 63:5cd1a5f3a41b 139 report.length = reportLen;
mjr 52:8298b2a73eb2 140 return sendTO(&report, 100);
mjr 52:8298b2a73eb2 141 }
mjr 52:8298b2a73eb2 142
mjr 86:e30a1f60f783 143 bool USBJoystick::sendPlungerStatus2(
mjr 86:e30a1f60f783 144 int nativeScale,
mjr 86:e30a1f60f783 145 int jitterLo, int jitterHi, int rawPos,
mjr 86:e30a1f60f783 146 int axcTime)
mjr 86:e30a1f60f783 147 {
mjr 86:e30a1f60f783 148 HID_REPORT report;
mjr 86:e30a1f60f783 149 memset(report.data, 0, sizeof(report.data));
mjr 86:e30a1f60f783 150
mjr 86:e30a1f60f783 151 // Set the special status bits to indicate it's an extended
mjr 86:e30a1f60f783 152 // exposure report.
mjr 86:e30a1f60f783 153 put(0, 0x87FF);
mjr 86:e30a1f60f783 154
mjr 86:e30a1f60f783 155 // start at the second byte
mjr 86:e30a1f60f783 156 int ofs = 2;
mjr 86:e30a1f60f783 157
mjr 86:e30a1f60f783 158 // write the report subtype (1) to byte 2
mjr 86:e30a1f60f783 159 report.data[ofs++] = 1;
mjr 86:e30a1f60f783 160
mjr 86:e30a1f60f783 161 // write the native scale to bytes 3:4
mjr 86:e30a1f60f783 162 put(ofs, uint16_t(nativeScale));
mjr 86:e30a1f60f783 163 ofs += 2;
mjr 86:e30a1f60f783 164
mjr 86:e30a1f60f783 165 // limit the jitter filter bounds to the native scale
mjr 86:e30a1f60f783 166 if (jitterLo < 0)
mjr 86:e30a1f60f783 167 jitterLo = 0;
mjr 86:e30a1f60f783 168 else if (jitterLo > nativeScale)
mjr 86:e30a1f60f783 169 jitterLo = nativeScale;
mjr 86:e30a1f60f783 170 if (jitterHi < 0)
mjr 86:e30a1f60f783 171 jitterHi = 0;
mjr 86:e30a1f60f783 172 else if (jitterHi > nativeScale)
mjr 86:e30a1f60f783 173 jitterHi = nativeScale;
mjr 86:e30a1f60f783 174
mjr 86:e30a1f60f783 175 // write the jitter filter window bounds to 5:6 and 7:8
mjr 86:e30a1f60f783 176 put(ofs, uint16_t(jitterLo));
mjr 86:e30a1f60f783 177 ofs += 2;
mjr 86:e30a1f60f783 178 put(ofs, uint16_t(jitterHi));
mjr 86:e30a1f60f783 179 ofs += 2;
mjr 86:e30a1f60f783 180
mjr 86:e30a1f60f783 181 // add the raw position
mjr 86:e30a1f60f783 182 put(ofs, uint16_t(rawPos));
mjr 86:e30a1f60f783 183 ofs += 2;
mjr 86:e30a1f60f783 184
mjr 86:e30a1f60f783 185 // add the auto-exposure time
mjr 86:e30a1f60f783 186 put(ofs, uint16_t(axcTime));
mjr 86:e30a1f60f783 187 ofs += 2;
mjr 86:e30a1f60f783 188
mjr 86:e30a1f60f783 189 // send the report
mjr 86:e30a1f60f783 190 report.length = reportLen;
mjr 86:e30a1f60f783 191 return sendTO(&report, 100);
mjr 86:e30a1f60f783 192 }
mjr 86:e30a1f60f783 193
mjr 87:8d35c74403af 194 bool USBJoystick::sendPlungerStatusBarcode(
mjr 87:8d35c74403af 195 int nbits, int codetype, int startOfs, int pixPerBit, int raw, int mask)
mjr 87:8d35c74403af 196 {
mjr 87:8d35c74403af 197 HID_REPORT report;
mjr 87:8d35c74403af 198 memset(report.data, 0, sizeof(report.data));
mjr 87:8d35c74403af 199
mjr 87:8d35c74403af 200 // Set the special status bits to indicate it's an extended
mjr 87:8d35c74403af 201 // exposure report.
mjr 87:8d35c74403af 202 put(0, 0x87FF);
mjr 87:8d35c74403af 203
mjr 87:8d35c74403af 204 // start at the second byte
mjr 87:8d35c74403af 205 int ofs = 2;
mjr 87:8d35c74403af 206
mjr 87:8d35c74403af 207 // write the report subtype (2) to byte 2
mjr 87:8d35c74403af 208 report.data[ofs++] = 2;
mjr 87:8d35c74403af 209
mjr 87:8d35c74403af 210 // write the bit count and code type
mjr 87:8d35c74403af 211 report.data[ofs++] = nbits;
mjr 87:8d35c74403af 212 report.data[ofs++] = codetype;
mjr 87:8d35c74403af 213
mjr 87:8d35c74403af 214 // write the bar code starting pixel offset
mjr 87:8d35c74403af 215 put(ofs, uint16_t(startOfs));
mjr 87:8d35c74403af 216 ofs += 2;
mjr 87:8d35c74403af 217
mjr 87:8d35c74403af 218 // write the pixel width per bit
mjr 87:8d35c74403af 219 report.data[ofs++] = pixPerBit;
mjr 87:8d35c74403af 220
mjr 87:8d35c74403af 221 // write the raw bar code and success bit mask
mjr 87:8d35c74403af 222 put(ofs, uint16_t(raw));
mjr 87:8d35c74403af 223 ofs += 2;
mjr 87:8d35c74403af 224 put(ofs, uint16_t(mask));
mjr 87:8d35c74403af 225 ofs += 2;
mjr 87:8d35c74403af 226
mjr 87:8d35c74403af 227 // send the report
mjr 87:8d35c74403af 228 report.length = reportLen;
mjr 87:8d35c74403af 229 return sendTO(&report, 100);
mjr 87:8d35c74403af 230 }
mjr 87:8d35c74403af 231
mjr 87:8d35c74403af 232
mjr 52:8298b2a73eb2 233 bool USBJoystick::sendPlungerPix(int &idx, int npix, const uint8_t *pix)
mjr 10:976666ffa4ef 234 {
mjr 10:976666ffa4ef 235 HID_REPORT report;
mjr 10:976666ffa4ef 236
mjr 10:976666ffa4ef 237 // Set the special status bits to indicate it's an exposure report.
mjr 10:976666ffa4ef 238 // The high 5 bits of the status word are set to 10000, and the
mjr 10:976666ffa4ef 239 // low 11 bits are the current pixel index.
mjr 10:976666ffa4ef 240 uint16_t s = idx | 0x8000;
mjr 63:5cd1a5f3a41b 241 put(0, s);
mjr 25:e22b88bd783a 242
mjr 25:e22b88bd783a 243 // start at the second byte
mjr 63:5cd1a5f3a41b 244 int ofs = 2;
mjr 25:e22b88bd783a 245
mjr 47:df7a88cd249c 246 // now fill out the remaining bytes with exposure values
mjr 63:5cd1a5f3a41b 247 report.length = reportLen;
mjr 47:df7a88cd249c 248 for ( ; ofs < report.length ; ++ofs)
mjr 47:df7a88cd249c 249 report.data[ofs] = (idx < npix ? pix[idx++] : 0);
mjr 10:976666ffa4ef 250
mjr 10:976666ffa4ef 251 // send the report
mjr 35:e959ffba78fd 252 return sendTO(&report, 100);
mjr 3:3514575d4f86 253 }
mjr 9:fd65b0a94720 254
mjr 53:9b2611964afc 255 bool USBJoystick::reportID(int index)
mjr 40:cc0d9814522b 256 {
mjr 40:cc0d9814522b 257 HID_REPORT report;
mjr 40:cc0d9814522b 258
mjr 40:cc0d9814522b 259 // initially fill the report with zeros
mjr 40:cc0d9814522b 260 memset(report.data, 0, sizeof(report.data));
mjr 40:cc0d9814522b 261
mjr 40:cc0d9814522b 262 // Set the special status bits to indicate that it's an ID report
mjr 40:cc0d9814522b 263 uint16_t s = 0x9000;
mjr 63:5cd1a5f3a41b 264 put(0, s);
mjr 40:cc0d9814522b 265
mjr 53:9b2611964afc 266 // add the requested ID index
mjr 63:5cd1a5f3a41b 267 report.data[2] = (uint8_t)index;
mjr 53:9b2611964afc 268
mjr 53:9b2611964afc 269 // figure out which ID we're reporting
mjr 53:9b2611964afc 270 switch (index)
mjr 53:9b2611964afc 271 {
mjr 53:9b2611964afc 272 case 1:
mjr 53:9b2611964afc 273 // KL25Z CPU ID
mjr 63:5cd1a5f3a41b 274 putbe(3, SIM->UIDMH);
mjr 63:5cd1a5f3a41b 275 putlbe(5, SIM->UIDML);
mjr 63:5cd1a5f3a41b 276 putlbe(9, SIM->UIDL);
mjr 53:9b2611964afc 277 break;
mjr 53:9b2611964afc 278
mjr 53:9b2611964afc 279 case 2:
mjr 53:9b2611964afc 280 // OpenSDA ID. Copy the low-order 80 bits of the OpenSDA ID.
mjr 53:9b2611964afc 281 // (The stored value is 128 bits = 16 bytes; we only want the last
mjr 53:9b2611964afc 282 // 80 bits = 10 bytes. So skip ahead 16 and back up 10 to get
mjr 53:9b2611964afc 283 // the starting point.)
mjr 53:9b2611964afc 284 extern const char *getOpenSDAID();
mjr 63:5cd1a5f3a41b 285 memcpy(&report.data[3], getOpenSDAID() + 16 - 10, 10);
mjr 53:9b2611964afc 286 break;
mjr 53:9b2611964afc 287 }
mjr 53:9b2611964afc 288
mjr 53:9b2611964afc 289 // send the report
mjr 63:5cd1a5f3a41b 290 report.length = reportLen;
mjr 53:9b2611964afc 291 return sendTO(&report, 100);
mjr 53:9b2611964afc 292 }
mjr 53:9b2611964afc 293
mjr 53:9b2611964afc 294 bool USBJoystick::reportBuildInfo(const char *date)
mjr 53:9b2611964afc 295 {
mjr 53:9b2611964afc 296 HID_REPORT report;
mjr 53:9b2611964afc 297
mjr 53:9b2611964afc 298 // initially fill the report with zeros
mjr 53:9b2611964afc 299 memset(report.data, 0, sizeof(report.data));
mjr 53:9b2611964afc 300
mjr 53:9b2611964afc 301 // Set the special status bits to indicate that it's a build
mjr 53:9b2611964afc 302 // info report
mjr 53:9b2611964afc 303 uint16_t s = 0xA000;
mjr 63:5cd1a5f3a41b 304 put(0, s);
mjr 53:9b2611964afc 305
mjr 53:9b2611964afc 306 // Parse the date. This is given in the standard __DATE__ " " __TIME
mjr 53:9b2611964afc 307 // macro format, "Mon dd yyyy hh:mm:ss" (e.g., "Feb 16 2016 12:15:06").
mjr 53:9b2611964afc 308 static const char mon[][4] = {
mjr 53:9b2611964afc 309 "Jan", "Feb", "Mar", "Apr", "May", "Jun",
mjr 53:9b2611964afc 310 "Jul", "Aug", "Sep", "Oct", "Nov", "Dec"
mjr 53:9b2611964afc 311 };
mjr 53:9b2611964afc 312 long dd = (atol(date + 7) * 10000L) // YYYY0000
mjr 53:9b2611964afc 313 + (atol(date + 4)); // 000000DD
mjr 53:9b2611964afc 314 for (int i = 0 ; i < 12 ; ++i)
mjr 53:9b2611964afc 315 {
mjr 53:9b2611964afc 316 if (memcmp(mon[i], date, 3) == 0)
mjr 53:9b2611964afc 317 {
mjr 53:9b2611964afc 318 dd += (i+1)*100; // 0000MM00
mjr 53:9b2611964afc 319 break;
mjr 53:9b2611964afc 320 }
mjr 53:9b2611964afc 321 }
mjr 53:9b2611964afc 322
mjr 53:9b2611964afc 323 // parse the time into a long formatted as decimal HHMMSS (e.g.,
mjr 53:9b2611964afc 324 // "12:15:06" turns into 121506 decimal)
mjr 53:9b2611964afc 325 long tt = (atol(date+12)*10000)
mjr 53:9b2611964afc 326 + (atol(date+15)*100)
mjr 53:9b2611964afc 327 + (atol(date+18));
mjr 53:9b2611964afc 328
mjr 53:9b2611964afc 329 // store the build date and time
mjr 63:5cd1a5f3a41b 330 putl(2, dd);
mjr 63:5cd1a5f3a41b 331 putl(6, tt);
mjr 40:cc0d9814522b 332
mjr 40:cc0d9814522b 333 // send the report
mjr 63:5cd1a5f3a41b 334 report.length = reportLen;
mjr 40:cc0d9814522b 335 return sendTO(&report, 100);
mjr 40:cc0d9814522b 336 }
mjr 40:cc0d9814522b 337
mjr 52:8298b2a73eb2 338 bool USBJoystick::reportConfigVar(const uint8_t *data)
mjr 52:8298b2a73eb2 339 {
mjr 52:8298b2a73eb2 340 HID_REPORT report;
mjr 52:8298b2a73eb2 341
mjr 52:8298b2a73eb2 342 // initially fill the report with zeros
mjr 52:8298b2a73eb2 343 memset(report.data, 0, sizeof(report.data));
mjr 52:8298b2a73eb2 344
mjr 52:8298b2a73eb2 345 // Set the special status bits to indicate that it's a config
mjr 52:8298b2a73eb2 346 // variable report
mjr 52:8298b2a73eb2 347 uint16_t s = 0x9800;
mjr 63:5cd1a5f3a41b 348 put(0, s);
mjr 52:8298b2a73eb2 349
mjr 52:8298b2a73eb2 350 // Copy the variable data (7 bytes, starting with the variable ID)
mjr 63:5cd1a5f3a41b 351 memcpy(report.data + 2, data, 7);
mjr 52:8298b2a73eb2 352
mjr 52:8298b2a73eb2 353 // send the report
mjr 63:5cd1a5f3a41b 354 report.length = reportLen;
mjr 52:8298b2a73eb2 355 return sendTO(&report, 100);
mjr 52:8298b2a73eb2 356 }
mjr 52:8298b2a73eb2 357
mjr 52:8298b2a73eb2 358 bool USBJoystick::reportConfig(
mjr 52:8298b2a73eb2 359 int numOutputs, int unitNo,
mjr 52:8298b2a73eb2 360 int plungerZero, int plungerMax, int plungerRlsTime,
mjr 92:f264fbaa1be5 361 bool configured, bool sbxpbx, bool newAccelFeatures,
mjr 92:f264fbaa1be5 362 bool flashStatusFeature, bool reportTimingFeatures,
mjr 99:8139b0c274f4 363 bool chimeLogicFeature, size_t freeHeapBytes)
mjr 33:d832bcab089e 364 {
mjr 33:d832bcab089e 365 HID_REPORT report;
mjr 33:d832bcab089e 366
mjr 33:d832bcab089e 367 // initially fill the report with zeros
mjr 33:d832bcab089e 368 memset(report.data, 0, sizeof(report.data));
mjr 33:d832bcab089e 369
mjr 33:d832bcab089e 370 // Set the special status bits to indicate that it's a config report.
mjr 33:d832bcab089e 371 uint16_t s = 0x8800;
mjr 63:5cd1a5f3a41b 372 put(0, s);
mjr 33:d832bcab089e 373
mjr 33:d832bcab089e 374 // write the number of configured outputs
mjr 63:5cd1a5f3a41b 375 put(2, numOutputs);
mjr 33:d832bcab089e 376
mjr 33:d832bcab089e 377 // write the unit number
mjr 63:5cd1a5f3a41b 378 put(4, unitNo);
mjr 33:d832bcab089e 379
mjr 35:e959ffba78fd 380 // write the plunger zero and max values
mjr 63:5cd1a5f3a41b 381 put(6, plungerZero);
mjr 63:5cd1a5f3a41b 382 put(8, plungerMax);
mjr 63:5cd1a5f3a41b 383 report.data[10] = uint8_t(plungerRlsTime);
mjr 35:e959ffba78fd 384
mjr 40:cc0d9814522b 385 // write the status bits:
mjr 40:cc0d9814522b 386 // 0x01 -> configuration loaded
mjr 75:677892300e7a 387 // 0x02 -> SBX/PBX protocol extensions supported
mjr 92:f264fbaa1be5 388 // 0x04 -> new accelerometer features supported
mjr 92:f264fbaa1be5 389 // 0x08 -> flash status feature supported
mjr 92:f264fbaa1be5 390 // 0x10 -> joystick report timing features supported
mjr 99:8139b0c274f4 391 // 0x20 -> chime logic feature supported
mjr 75:677892300e7a 392 report.data[11] =
mjr 75:677892300e7a 393 (configured ? 0x01 : 0x00)
mjr 78:1e00b3fa11af 394 | (sbxpbx ? 0x02 : 0x00)
mjr 82:4f6209cb5c33 395 | (newAccelFeatures ? 0x04 : 0x00)
mjr 92:f264fbaa1be5 396 | (flashStatusFeature ? 0x08 : 0x00)
mjr 98:4df3c0f7e707 397 | (reportTimingFeatures ? 0x10 : 0x00)
mjr 99:8139b0c274f4 398 | (chimeLogicFeature ? 0x20 : 0x00);
mjr 40:cc0d9814522b 399
mjr 73:4e8ce0b18915 400 // write the free heap space
mjr 73:4e8ce0b18915 401 put(12, freeHeapBytes);
mjr 77:0b96f6867312 402
mjr 33:d832bcab089e 403 // send the report
mjr 63:5cd1a5f3a41b 404 report.length = reportLen;
mjr 35:e959ffba78fd 405 return sendTO(&report, 100);
mjr 33:d832bcab089e 406 }
mjr 33:d832bcab089e 407
mjr 77:0b96f6867312 408 // report physical button status
mjr 73:4e8ce0b18915 409 bool USBJoystick::reportButtonStatus(int numButtons, const uint8_t *state)
mjr 73:4e8ce0b18915 410 {
mjr 77:0b96f6867312 411 // initially fill the report with zeros
mjr 73:4e8ce0b18915 412 HID_REPORT report;
mjr 73:4e8ce0b18915 413 memset(report.data, 0, sizeof(report.data));
mjr 73:4e8ce0b18915 414
mjr 77:0b96f6867312 415 // set the special status bits to indicate that it's a buton report
mjr 73:4e8ce0b18915 416 uint16_t s = 0xA100;
mjr 73:4e8ce0b18915 417 put(0, s);
mjr 73:4e8ce0b18915 418
mjr 73:4e8ce0b18915 419 // write the number of buttons
mjr 77:0b96f6867312 420 report.data[2] = uint8_t(numButtons);
mjr 73:4e8ce0b18915 421
mjr 73:4e8ce0b18915 422 // Write the buttons - these are packed into ceil(numButtons/8) bytes.
mjr 73:4e8ce0b18915 423 size_t btnBytes = (numButtons+7)/8;
mjr 73:4e8ce0b18915 424 if (btnBytes + 3 > reportLen) btnBytes = reportLen - 3;
mjr 73:4e8ce0b18915 425 memcpy(&report.data[3], state, btnBytes);
mjr 73:4e8ce0b18915 426
mjr 73:4e8ce0b18915 427 // send the report
mjr 73:4e8ce0b18915 428 report.length = reportLen;
mjr 73:4e8ce0b18915 429 return sendTO(&report, 100);
mjr 73:4e8ce0b18915 430 }
mjr 73:4e8ce0b18915 431
mjr 77:0b96f6867312 432 // report raw IR timing codes (for learning mode)
mjr 77:0b96f6867312 433 bool USBJoystick::reportRawIR(int n, const uint16_t *data)
mjr 77:0b96f6867312 434 {
mjr 77:0b96f6867312 435 // initially fill the report with zeros
mjr 77:0b96f6867312 436 HID_REPORT report;
mjr 77:0b96f6867312 437 memset(report.data, 0, sizeof(report.data));
mjr 77:0b96f6867312 438
mjr 77:0b96f6867312 439 // set the special status bits to indicate that it's an IR report
mjr 77:0b96f6867312 440 uint16_t s = 0xA200;
mjr 77:0b96f6867312 441 put(0, s);
mjr 77:0b96f6867312 442
mjr 77:0b96f6867312 443 // limit the number of items reported to the available space
mjr 77:0b96f6867312 444 if (n > maxRawIR)
mjr 77:0b96f6867312 445 n = maxRawIR;
mjr 77:0b96f6867312 446
mjr 77:0b96f6867312 447 // write the number of codes
mjr 77:0b96f6867312 448 report.data[2] = uint8_t(n);
mjr 77:0b96f6867312 449
mjr 77:0b96f6867312 450 // write the codes
mjr 77:0b96f6867312 451 for (int i = 0, ofs = 3 ; i < n ; ++i, ofs += 2)
mjr 77:0b96f6867312 452 put(ofs, data[i]);
mjr 77:0b96f6867312 453
mjr 77:0b96f6867312 454 // send the report
mjr 77:0b96f6867312 455 report.length = reportLen;
mjr 77:0b96f6867312 456 return sendTO(&report, 100);
mjr 77:0b96f6867312 457 }
mjr 77:0b96f6867312 458
mjr 77:0b96f6867312 459 // report a decoded IR command
mjr 77:0b96f6867312 460 bool USBJoystick::reportIRCode(uint8_t pro, uint8_t flags, uint64_t code)
mjr 77:0b96f6867312 461 {
mjr 77:0b96f6867312 462 // initially fill the report with zeros
mjr 77:0b96f6867312 463 HID_REPORT report;
mjr 77:0b96f6867312 464 memset(report.data, 0, sizeof(report.data));
mjr 77:0b96f6867312 465
mjr 77:0b96f6867312 466 // set the special status bits to indicate that it's an IR report
mjr 77:0b96f6867312 467 uint16_t s = 0xA200;
mjr 77:0b96f6867312 468 put(0, s);
mjr 77:0b96f6867312 469
mjr 77:0b96f6867312 470 // set the raw count to 0xFF to flag that it's a decoded command
mjr 77:0b96f6867312 471 report.data[2] = 0xFF;
mjr 77:0b96f6867312 472
mjr 77:0b96f6867312 473 // write the data
mjr 77:0b96f6867312 474 report.data[3] = pro;
mjr 77:0b96f6867312 475 report.data[4] = flags;
mjr 77:0b96f6867312 476 put64(5, code);
mjr 77:0b96f6867312 477
mjr 77:0b96f6867312 478 // send the report
mjr 77:0b96f6867312 479 report.length = reportLen;
mjr 77:0b96f6867312 480 return sendTO(&report, 100);
mjr 77:0b96f6867312 481 }
mjr 73:4e8ce0b18915 482
mjr 33:d832bcab089e 483 bool USBJoystick::move(int16_t x, int16_t y)
mjr 33:d832bcab089e 484 {
mjr 3:3514575d4f86 485 _x = x;
mjr 3:3514575d4f86 486 _y = y;
mjr 3:3514575d4f86 487 return update();
mjr 3:3514575d4f86 488 }
mjr 3:3514575d4f86 489
mjr 33:d832bcab089e 490 bool USBJoystick::setZ(int16_t z)
mjr 33:d832bcab089e 491 {
mjr 3:3514575d4f86 492 _z = z;
mjr 3:3514575d4f86 493 return update();
mjr 3:3514575d4f86 494 }
mjr 3:3514575d4f86 495
mjr 33:d832bcab089e 496 bool USBJoystick::buttons(uint32_t buttons)
mjr 33:d832bcab089e 497 {
mjr 11:bd9da7088e6e 498 _buttonsLo = (uint16_t)(buttons & 0xffff);
mjr 11:bd9da7088e6e 499 _buttonsHi = (uint16_t)((buttons >> 16) & 0xffff);
mjr 3:3514575d4f86 500 return update();
mjr 3:3514575d4f86 501 }
mjr 21:5048e16cc9ef 502
mjr 21:5048e16cc9ef 503 bool USBJoystick::updateStatus(uint32_t status)
mjr 21:5048e16cc9ef 504 {
mjr 21:5048e16cc9ef 505 HID_REPORT report;
mjr 21:5048e16cc9ef 506
mjr 63:5cd1a5f3a41b 507 // Fill the report according to the Joystick Descriptor
mjr 63:5cd1a5f3a41b 508 memset(report.data, 0, reportLen);
mjr 63:5cd1a5f3a41b 509 put(0, status);
mjr 63:5cd1a5f3a41b 510 report.length = reportLen;
mjr 21:5048e16cc9ef 511
mjr 21:5048e16cc9ef 512 // send the report
mjr 21:5048e16cc9ef 513 return sendTO(&report, 100);
mjr 21:5048e16cc9ef 514 }
mjr 21:5048e16cc9ef 515
mjr 3:3514575d4f86 516 void USBJoystick::_init() {
mjr 3:3514575d4f86 517
mjr 3:3514575d4f86 518 _x = 0;
mjr 3:3514575d4f86 519 _y = 0;
mjr 3:3514575d4f86 520 _z = 0;
mjr 11:bd9da7088e6e 521 _buttonsLo = 0x0000;
mjr 11:bd9da7088e6e 522 _buttonsHi = 0x0000;
mjr 9:fd65b0a94720 523 _status = 0;
mjr 3:3514575d4f86 524 }
mjr 3:3514575d4f86 525
mjr 3:3514575d4f86 526
mjr 35:e959ffba78fd 527 // --------------------------------------------------------------------------
mjr 35:e959ffba78fd 528 //
mjr 63:5cd1a5f3a41b 529 // USB HID Report Descriptor - Joystick
mjr 35:e959ffba78fd 530 //
mjr 63:5cd1a5f3a41b 531 static const uint8_t reportDescriptorJS[] =
mjr 63:5cd1a5f3a41b 532 {
mjr 63:5cd1a5f3a41b 533 USAGE_PAGE(1), 0x01, // Generic desktop
mjr 63:5cd1a5f3a41b 534 USAGE(1), 0x04, // Joystick
mjr 63:5cd1a5f3a41b 535 COLLECTION(1), 0x01, // Application
mjr 90:aa4e571da8e8 536
mjr 63:5cd1a5f3a41b 537 // input report (device to host)
mjr 63:5cd1a5f3a41b 538 USAGE_PAGE(1), 0x06, // generic device controls - for config status
mjr 63:5cd1a5f3a41b 539 USAGE(1), 0x00, // undefined device control
mjr 63:5cd1a5f3a41b 540 LOGICAL_MINIMUM(1), 0x00, // 8-bit values
mjr 63:5cd1a5f3a41b 541 LOGICAL_MAXIMUM(1), 0xFF,
mjr 63:5cd1a5f3a41b 542 REPORT_SIZE(1), 0x08, // 8 bits per report
mjr 63:5cd1a5f3a41b 543 REPORT_COUNT(1), 0x04, // 4 reports (4 bytes)
mjr 63:5cd1a5f3a41b 544 INPUT(1), 0x02, // Data, Variable, Absolute
mjr 35:e959ffba78fd 545
mjr 63:5cd1a5f3a41b 546 USAGE_PAGE(1), 0x09, // Buttons
mjr 63:5cd1a5f3a41b 547 USAGE_MINIMUM(1), 0x01, // { buttons }
mjr 63:5cd1a5f3a41b 548 USAGE_MAXIMUM(1), 0x20, // { 1-32 }
mjr 63:5cd1a5f3a41b 549 LOGICAL_MINIMUM(1), 0x00, // 1-bit buttons - 0...
mjr 63:5cd1a5f3a41b 550 LOGICAL_MAXIMUM(1), 0x01, // ...to 1
mjr 63:5cd1a5f3a41b 551 REPORT_SIZE(1), 0x01, // 1 bit per report
mjr 63:5cd1a5f3a41b 552 REPORT_COUNT(1), 0x20, // 32 reports
mjr 63:5cd1a5f3a41b 553 UNIT_EXPONENT(1), 0x00, // Unit_Exponent (0)
mjr 63:5cd1a5f3a41b 554 UNIT(1), 0x00, // Unit (None)
mjr 63:5cd1a5f3a41b 555 INPUT(1), 0x02, // Data, Variable, Absolute
mjr 63:5cd1a5f3a41b 556
mjr 63:5cd1a5f3a41b 557 USAGE_PAGE(1), 0x01, // Generic desktop
mjr 63:5cd1a5f3a41b 558 USAGE(1), 0x30, // X axis
mjr 63:5cd1a5f3a41b 559 USAGE(1), 0x31, // Y axis
mjr 63:5cd1a5f3a41b 560 USAGE(1), 0x32, // Z axis
mjr 63:5cd1a5f3a41b 561 LOGICAL_MINIMUM(2), 0x00,0xF0, // each value ranges -4096
mjr 63:5cd1a5f3a41b 562 LOGICAL_MAXIMUM(2), 0x00,0x10, // ...to +4096
mjr 63:5cd1a5f3a41b 563 REPORT_SIZE(1), 0x10, // 16 bits per report
mjr 63:5cd1a5f3a41b 564 REPORT_COUNT(1), 0x03, // 3 reports (X, Y, Z)
mjr 63:5cd1a5f3a41b 565 INPUT(1), 0x02, // Data, Variable, Absolute
mjr 63:5cd1a5f3a41b 566
mjr 63:5cd1a5f3a41b 567 // output report (host to device)
mjr 63:5cd1a5f3a41b 568 REPORT_SIZE(1), 0x08, // 8 bits per report
mjr 63:5cd1a5f3a41b 569 REPORT_COUNT(1), 0x08, // output report count - 8-byte LedWiz format
mjr 63:5cd1a5f3a41b 570 0x09, 0x01, // usage
mjr 63:5cd1a5f3a41b 571 0x91, 0x01, // Output (array)
mjr 35:e959ffba78fd 572
mjr 35:e959ffba78fd 573 END_COLLECTION(0)
mjr 35:e959ffba78fd 574 };
mjr 35:e959ffba78fd 575
mjr 90:aa4e571da8e8 576 // Joystick report descriptor with "R" axis reports. This version
mjr 90:aa4e571da8e8 577 // uses Rx and Ry for the accelerometer readings and Rz for the
mjr 90:aa4e571da8e8 578 // plunger, instead of the standard X/Y/Z axes. This can be used
mjr 90:aa4e571da8e8 579 // to avoid conflicts with other devices reporting on the normal
mjr 90:aa4e571da8e8 580 // X/Y/Z axes.
mjr 90:aa4e571da8e8 581 static const uint8_t reportDescriptorJS_RXRYRZ[] =
mjr 90:aa4e571da8e8 582 {
mjr 90:aa4e571da8e8 583 USAGE_PAGE(1), 0x01, // Generic desktop
mjr 90:aa4e571da8e8 584 USAGE(1), 0x04, // Joystick
mjr 90:aa4e571da8e8 585 COLLECTION(1), 0x01, // Application
mjr 90:aa4e571da8e8 586
mjr 90:aa4e571da8e8 587 // input report (device to host)
mjr 90:aa4e571da8e8 588 USAGE_PAGE(1), 0x06, // generic device controls - for config status
mjr 90:aa4e571da8e8 589 USAGE(1), 0x00, // undefined device control
mjr 90:aa4e571da8e8 590 LOGICAL_MINIMUM(1), 0x00, // 8-bit values
mjr 90:aa4e571da8e8 591 LOGICAL_MAXIMUM(1), 0xFF,
mjr 90:aa4e571da8e8 592 REPORT_SIZE(1), 0x08, // 8 bits per report
mjr 90:aa4e571da8e8 593 REPORT_COUNT(1), 0x04, // 4 reports (4 bytes)
mjr 90:aa4e571da8e8 594 INPUT(1), 0x02, // Data, Variable, Absolute
mjr 90:aa4e571da8e8 595
mjr 90:aa4e571da8e8 596 USAGE_PAGE(1), 0x09, // Buttons
mjr 90:aa4e571da8e8 597 USAGE_MINIMUM(1), 0x01, // { buttons }
mjr 90:aa4e571da8e8 598 USAGE_MAXIMUM(1), 0x20, // { 1-32 }
mjr 90:aa4e571da8e8 599 LOGICAL_MINIMUM(1), 0x00, // 1-bit buttons - 0...
mjr 90:aa4e571da8e8 600 LOGICAL_MAXIMUM(1), 0x01, // ...to 1
mjr 90:aa4e571da8e8 601 REPORT_SIZE(1), 0x01, // 1 bit per report
mjr 90:aa4e571da8e8 602 REPORT_COUNT(1), 0x20, // 32 reports
mjr 90:aa4e571da8e8 603 UNIT_EXPONENT(1), 0x00, // Unit_Exponent (0)
mjr 90:aa4e571da8e8 604 UNIT(1), 0x00, // Unit (None)
mjr 90:aa4e571da8e8 605 INPUT(1), 0x02, // Data, Variable, Absolute
mjr 90:aa4e571da8e8 606
mjr 90:aa4e571da8e8 607 USAGE_PAGE(1), 0x01, // Generic desktop
mjr 90:aa4e571da8e8 608 USAGE(1), 0x33, // Rx axis ("X rotation")
mjr 90:aa4e571da8e8 609 USAGE(1), 0x34, // Ry axis
mjr 90:aa4e571da8e8 610 USAGE(1), 0x35, // Rz axis
mjr 90:aa4e571da8e8 611 LOGICAL_MINIMUM(2), 0x00,0xF0, // each value ranges -4096
mjr 90:aa4e571da8e8 612 LOGICAL_MAXIMUM(2), 0x00,0x10, // ...to +4096
mjr 90:aa4e571da8e8 613 REPORT_SIZE(1), 0x10, // 16 bits per report
mjr 90:aa4e571da8e8 614 REPORT_COUNT(1), 0x03, // 3 reports (X, Y, Z)
mjr 90:aa4e571da8e8 615 INPUT(1), 0x02, // Data, Variable, Absolute
mjr 90:aa4e571da8e8 616
mjr 90:aa4e571da8e8 617 // output report (host to device)
mjr 90:aa4e571da8e8 618 REPORT_SIZE(1), 0x08, // 8 bits per report
mjr 90:aa4e571da8e8 619 REPORT_COUNT(1), 0x08, // output report count - 8-byte LedWiz format
mjr 90:aa4e571da8e8 620 0x09, 0x01, // usage
mjr 90:aa4e571da8e8 621 0x91, 0x01, // Output (array)
mjr 90:aa4e571da8e8 622
mjr 90:aa4e571da8e8 623 END_COLLECTION(0)
mjr 90:aa4e571da8e8 624 };
mjr 90:aa4e571da8e8 625
mjr 90:aa4e571da8e8 626
mjr 63:5cd1a5f3a41b 627 //
mjr 63:5cd1a5f3a41b 628 // USB HID Report Descriptor - Keyboard/Media Control
mjr 63:5cd1a5f3a41b 629 //
mjr 48:058ace2aed1d 630 static const uint8_t reportDescriptorKB[] =
mjr 35:e959ffba78fd 631 {
mjr 63:5cd1a5f3a41b 632 USAGE_PAGE(1), 0x01, // Generic Desktop
mjr 63:5cd1a5f3a41b 633 USAGE(1), 0x06, // Keyboard
mjr 63:5cd1a5f3a41b 634 COLLECTION(1), 0x01, // Application
mjr 63:5cd1a5f3a41b 635 REPORT_ID(1), REPORT_ID_KB,
mjr 63:5cd1a5f3a41b 636
mjr 63:5cd1a5f3a41b 637 USAGE_PAGE(1), 0x07, // Key Codes
mjr 63:5cd1a5f3a41b 638 USAGE_MINIMUM(1), 0xE0,
mjr 63:5cd1a5f3a41b 639 USAGE_MAXIMUM(1), 0xE7,
mjr 63:5cd1a5f3a41b 640 LOGICAL_MINIMUM(1), 0x00,
mjr 63:5cd1a5f3a41b 641 LOGICAL_MAXIMUM(1), 0x01,
mjr 63:5cd1a5f3a41b 642 REPORT_SIZE(1), 0x01,
mjr 63:5cd1a5f3a41b 643 REPORT_COUNT(1), 0x08,
mjr 63:5cd1a5f3a41b 644 INPUT(1), 0x02, // Data, Variable, Absolute
mjr 63:5cd1a5f3a41b 645 REPORT_COUNT(1), 0x01,
mjr 63:5cd1a5f3a41b 646 REPORT_SIZE(1), 0x08,
mjr 63:5cd1a5f3a41b 647 INPUT(1), 0x01, // Constant
mjr 63:5cd1a5f3a41b 648
mjr 63:5cd1a5f3a41b 649 REPORT_COUNT(1), 0x05,
mjr 63:5cd1a5f3a41b 650 REPORT_SIZE(1), 0x01,
mjr 63:5cd1a5f3a41b 651 USAGE_PAGE(1), 0x08, // LEDs
mjr 63:5cd1a5f3a41b 652 USAGE_MINIMUM(1), 0x01,
mjr 63:5cd1a5f3a41b 653 USAGE_MAXIMUM(1), 0x05,
mjr 63:5cd1a5f3a41b 654 OUTPUT(1), 0x02, // Data, Variable, Absolute
mjr 63:5cd1a5f3a41b 655 REPORT_COUNT(1), 0x01,
mjr 63:5cd1a5f3a41b 656 REPORT_SIZE(1), 0x03,
mjr 63:5cd1a5f3a41b 657 OUTPUT(1), 0x01, // Constant
mjr 63:5cd1a5f3a41b 658
mjr 63:5cd1a5f3a41b 659 REPORT_COUNT(1), 0x06,
mjr 63:5cd1a5f3a41b 660 REPORT_SIZE(1), 0x08,
mjr 63:5cd1a5f3a41b 661 LOGICAL_MINIMUM(1), 0x00,
mjr 68:998faf685b00 662 LOGICAL_MAXIMUM(1), 0xA4,
mjr 63:5cd1a5f3a41b 663 USAGE_PAGE(1), 0x07, // Key Codes
mjr 63:5cd1a5f3a41b 664 USAGE_MINIMUM(1), 0x00,
mjr 68:998faf685b00 665 USAGE_MAXIMUM(1), 0xA4,
mjr 63:5cd1a5f3a41b 666 INPUT(1), 0x00, // Data, Array
mjr 63:5cd1a5f3a41b 667 END_COLLECTION(0),
mjr 63:5cd1a5f3a41b 668
mjr 63:5cd1a5f3a41b 669 // Media Control
mjr 63:5cd1a5f3a41b 670 USAGE_PAGE(1), 0x0C,
mjr 63:5cd1a5f3a41b 671 USAGE(1), 0x01,
mjr 63:5cd1a5f3a41b 672 COLLECTION(1), 0x01,
mjr 63:5cd1a5f3a41b 673 REPORT_ID(1), REPORT_ID_MEDIA,
mjr 63:5cd1a5f3a41b 674 USAGE_PAGE(1), 0x0C,
mjr 63:5cd1a5f3a41b 675 LOGICAL_MINIMUM(1), 0x00,
mjr 63:5cd1a5f3a41b 676 LOGICAL_MAXIMUM(1), 0x01,
mjr 63:5cd1a5f3a41b 677 REPORT_SIZE(1), 0x01,
mjr 63:5cd1a5f3a41b 678 REPORT_COUNT(1), 0x07,
mjr 63:5cd1a5f3a41b 679 USAGE(1), 0xE2, // Mute -> 0x01
mjr 63:5cd1a5f3a41b 680 USAGE(1), 0xE9, // Volume Up -> 0x02
mjr 63:5cd1a5f3a41b 681 USAGE(1), 0xEA, // Volume Down -> 0x04
mjr 63:5cd1a5f3a41b 682 USAGE(1), 0xB5, // Next Track -> 0x08
mjr 63:5cd1a5f3a41b 683 USAGE(1), 0xB6, // Previous Track -> 0x10
mjr 63:5cd1a5f3a41b 684 USAGE(1), 0xB7, // Stop -> 0x20
mjr 63:5cd1a5f3a41b 685 USAGE(1), 0xCD, // Play / Pause -> 0x40
mjr 63:5cd1a5f3a41b 686 INPUT(1), 0x02, // Input (Data, Variable, Absolute) -> 0x80
mjr 63:5cd1a5f3a41b 687 REPORT_COUNT(1), 0x01,
mjr 63:5cd1a5f3a41b 688 INPUT(1), 0x01,
mjr 63:5cd1a5f3a41b 689 END_COLLECTION(0),
mjr 35:e959ffba78fd 690 };
mjr 29:582472d0bc57 691
mjr 63:5cd1a5f3a41b 692 //
mjr 63:5cd1a5f3a41b 693 // USB HID Report Descriptor - LedWiz only, with no joystick or keyboard
mjr 63:5cd1a5f3a41b 694 // input reporting
mjr 63:5cd1a5f3a41b 695 //
mjr 63:5cd1a5f3a41b 696 static const uint8_t reportDescriptorLW[] =
mjr 63:5cd1a5f3a41b 697 {
mjr 63:5cd1a5f3a41b 698 USAGE_PAGE(1), 0x01, // Generic desktop
mjr 63:5cd1a5f3a41b 699 USAGE(1), 0x00, // Undefined
mjr 63:5cd1a5f3a41b 700
mjr 63:5cd1a5f3a41b 701 COLLECTION(1), 0x01, // Application
mjr 63:5cd1a5f3a41b 702
mjr 63:5cd1a5f3a41b 703 // input report (device to host)
mjr 63:5cd1a5f3a41b 704 USAGE_PAGE(1), 0x06, // generic device controls - for config status
mjr 63:5cd1a5f3a41b 705 USAGE(1), 0x00, // undefined device control
mjr 63:5cd1a5f3a41b 706 LOGICAL_MINIMUM(1), 0x00, // 8-bit values
mjr 63:5cd1a5f3a41b 707 LOGICAL_MAXIMUM(1), 0xFF,
mjr 63:5cd1a5f3a41b 708 REPORT_SIZE(1), 0x08, // 8 bits per report
mjr 77:0b96f6867312 709 REPORT_COUNT(1), USBJoystick::reportLen, // standard report length (same as if we were in joystick mode)
mjr 63:5cd1a5f3a41b 710 INPUT(1), 0x02, // Data, Variable, Absolute
mjr 63:5cd1a5f3a41b 711
mjr 63:5cd1a5f3a41b 712 // output report (host to device)
mjr 63:5cd1a5f3a41b 713 REPORT_SIZE(1), 0x08, // 8 bits per report
mjr 63:5cd1a5f3a41b 714 REPORT_COUNT(1), 0x08, // output report count (LEDWiz messages)
mjr 63:5cd1a5f3a41b 715 0x09, 0x01, // usage
mjr 63:5cd1a5f3a41b 716 0x91, 0x01, // Output (array)
mjr 63:5cd1a5f3a41b 717
mjr 63:5cd1a5f3a41b 718 END_COLLECTION(0)
mjr 35:e959ffba78fd 719 };
mjr 35:e959ffba78fd 720
mjr 63:5cd1a5f3a41b 721
mjr 54:fd77a6b2f76c 722 const uint8_t *USBJoystick::reportDesc(int idx, uint16_t &len)
mjr 35:e959ffba78fd 723 {
mjr 63:5cd1a5f3a41b 724 switch (idx)
mjr 35:e959ffba78fd 725 {
mjr 63:5cd1a5f3a41b 726 case 0:
mjr 63:5cd1a5f3a41b 727 // If the joystick is enabled, this is the joystick.
mjr 63:5cd1a5f3a41b 728 // Otherwise, it's the plain LedWiz control interface.
mjr 63:5cd1a5f3a41b 729 if (enableJoystick)
mjr 63:5cd1a5f3a41b 730 {
mjr 90:aa4e571da8e8 731 switch (axisFormat)
mjr 90:aa4e571da8e8 732 {
mjr 90:aa4e571da8e8 733 case AXIS_FORMAT_XYZ:
mjr 90:aa4e571da8e8 734 default:
mjr 90:aa4e571da8e8 735 len = sizeof(reportDescriptorJS);
mjr 90:aa4e571da8e8 736 return reportDescriptorJS;
mjr 90:aa4e571da8e8 737
mjr 90:aa4e571da8e8 738 case AXIS_FORMAT_RXRYRZ:
mjr 90:aa4e571da8e8 739 len = sizeof(reportDescriptorJS_RXRYRZ);
mjr 90:aa4e571da8e8 740 return reportDescriptorJS_RXRYRZ;
mjr 90:aa4e571da8e8 741 }
mjr 63:5cd1a5f3a41b 742 }
mjr 63:5cd1a5f3a41b 743 else
mjr 63:5cd1a5f3a41b 744 {
mjr 63:5cd1a5f3a41b 745 len = sizeof(reportDescriptorLW);
mjr 63:5cd1a5f3a41b 746 return reportDescriptorLW;
mjr 63:5cd1a5f3a41b 747 }
mjr 63:5cd1a5f3a41b 748
mjr 63:5cd1a5f3a41b 749 case 1:
mjr 63:5cd1a5f3a41b 750 // This is the keyboard, if enabled.
mjr 63:5cd1a5f3a41b 751 if (useKB)
mjr 63:5cd1a5f3a41b 752 {
mjr 63:5cd1a5f3a41b 753 len = sizeof(reportDescriptorKB);
mjr 63:5cd1a5f3a41b 754 return reportDescriptorKB;
mjr 63:5cd1a5f3a41b 755 }
mjr 63:5cd1a5f3a41b 756 else
mjr 63:5cd1a5f3a41b 757 {
mjr 63:5cd1a5f3a41b 758 len = 0;
mjr 63:5cd1a5f3a41b 759 return 0;
mjr 63:5cd1a5f3a41b 760 }
mjr 63:5cd1a5f3a41b 761
mjr 63:5cd1a5f3a41b 762 default:
mjr 63:5cd1a5f3a41b 763 // Unknown interface ID
mjr 54:fd77a6b2f76c 764 len = 0;
mjr 48:058ace2aed1d 765 return 0;
mjr 35:e959ffba78fd 766 }
mjr 35:e959ffba78fd 767 }
mjr 3:3514575d4f86 768
mjr 48:058ace2aed1d 769 const uint8_t *USBJoystick::stringImanufacturerDesc() {
mjr 48:058ace2aed1d 770 static const uint8_t stringImanufacturerDescriptor[] = {
mjr 61:3c7e6e9ec355 771 0x0E, /* bLength */
mjr 61:3c7e6e9ec355 772 STRING_DESCRIPTOR, /* bDescriptorType 0x03 (String Descriptor) */
mjr 61:3c7e6e9ec355 773 'm',0,'j',0,'r',0,'n',0,'e',0,'t',0 /* bString iManufacturer - mjrnet */
mjr 3:3514575d4f86 774 };
mjr 3:3514575d4f86 775 return stringImanufacturerDescriptor;
mjr 3:3514575d4f86 776 }
mjr 3:3514575d4f86 777
mjr 54:fd77a6b2f76c 778 const uint8_t *USBJoystick::stringIserialDesc()
mjr 54:fd77a6b2f76c 779 {
mjr 90:aa4e571da8e8 780 // set up a buffer with space for the length prefix byte, descriptor type
mjr 90:aa4e571da8e8 781 // byte, and serial number (as a wide-character string)
mjr 90:aa4e571da8e8 782 const int numChars = 3 + 16 + 1 + 1 + 3;
mjr 61:3c7e6e9ec355 783 static uint8_t buf[2 + numChars*2];
mjr 54:fd77a6b2f76c 784 uint8_t *dst = buf;
mjr 90:aa4e571da8e8 785
mjr 90:aa4e571da8e8 786 // store a placeholder for the length, followed by the descriptor type byte
mjr 90:aa4e571da8e8 787 *dst++ = 0;
mjr 54:fd77a6b2f76c 788 *dst++ = STRING_DESCRIPTOR;
mjr 54:fd77a6b2f76c 789
mjr 54:fd77a6b2f76c 790 // Create an ASCII version of our unique serial number string:
mjr 54:fd77a6b2f76c 791 //
mjr 90:aa4e571da8e8 792 // PSCxxxxxxxxxxxxxxxxi[a]vvv
mjr 54:fd77a6b2f76c 793 //
mjr 54:fd77a6b2f76c 794 // where:
mjr 54:fd77a6b2f76c 795 //
mjr 54:fd77a6b2f76c 796 // xxx... = decimal representation of low 64 bits of CPU ID (16 hex digits)
mjr 54:fd77a6b2f76c 797 // i = interface type: first character is J if joystick is enabled,
mjr 54:fd77a6b2f76c 798 // L = LedWiz/control interface only, no input
mjr 54:fd77a6b2f76c 799 // J = Joystick + LedWiz
mjr 54:fd77a6b2f76c 800 // K = Keyboard + LedWiz
mjr 54:fd77a6b2f76c 801 // C = Joystick + Keyboard + LedWiz ("C" for combo)
mjr 90:aa4e571da8e8 802 // a = joystick axis types:
mjr 90:aa4e571da8e8 803 // <empty> = X,Y,Z, or no joystick interface at all
mjr 90:aa4e571da8e8 804 // A = Rx,Ry,Rz
mjr 61:3c7e6e9ec355 805 // vvv = version suffix
mjr 54:fd77a6b2f76c 806 //
mjr 54:fd77a6b2f76c 807 // The suffix for the interface type resolves a problem on some Windows systems
mjr 54:fd77a6b2f76c 808 // when switching between interface types. Windows can cache device information
mjr 54:fd77a6b2f76c 809 // that includes the interface descriptors, and it won't recognize a change in
mjr 54:fd77a6b2f76c 810 // the interfaces once the information is cached, causing connection failures.
mjr 54:fd77a6b2f76c 811 // The cache key includes the device serial number, though, so this can be
mjr 54:fd77a6b2f76c 812 // resolved by changing the serial number when the interface setup changes.
mjr 61:3c7e6e9ec355 813 char xbuf[numChars + 1];
mjr 54:fd77a6b2f76c 814 uint32_t x = SIM->UIDML;
mjr 54:fd77a6b2f76c 815 static char ifcCode[] = "LJKC";
mjr 90:aa4e571da8e8 816 static const char *axisCode[] = { "", "A" };
mjr 90:aa4e571da8e8 817 sprintf(xbuf, "PSC%08lX%08lX%c%s008",
mjr 54:fd77a6b2f76c 818 SIM->UIDML,
mjr 54:fd77a6b2f76c 819 SIM->UIDL,
mjr 90:aa4e571da8e8 820 ifcCode[(enableJoystick ? 0x01 : 0x00) | (useKB ? 0x02 : 0x00)],
mjr 90:aa4e571da8e8 821 axisCode[(enableJoystick ? axisFormat : 0)]);
mjr 54:fd77a6b2f76c 822
mjr 54:fd77a6b2f76c 823 // copy the ascii bytes into the descriptor buffer, converting to unicode
mjr 54:fd77a6b2f76c 824 // 16-bit little-endian characters
mjr 54:fd77a6b2f76c 825 for (char *src = xbuf ; *src != '\0' && dst < buf + sizeof(buf) ; )
mjr 54:fd77a6b2f76c 826 {
mjr 54:fd77a6b2f76c 827 *dst++ = *src++;
mjr 54:fd77a6b2f76c 828 *dst++ = '\0';
mjr 54:fd77a6b2f76c 829 }
mjr 54:fd77a6b2f76c 830
mjr 90:aa4e571da8e8 831 // store the final length (in bytes) in the length prefix byte
mjr 90:aa4e571da8e8 832 buf[0] = dst - buf;
mjr 90:aa4e571da8e8 833
mjr 54:fd77a6b2f76c 834 // return the buffer
mjr 54:fd77a6b2f76c 835 return buf;
mjr 3:3514575d4f86 836 }
mjr 3:3514575d4f86 837
mjr 48:058ace2aed1d 838 const uint8_t *USBJoystick::stringIproductDesc() {
mjr 48:058ace2aed1d 839 static const uint8_t stringIproductDescriptor[] = {
mjr 9:fd65b0a94720 840 0x28, /*bLength*/
mjr 3:3514575d4f86 841 STRING_DESCRIPTOR, /*bDescriptorType 0x03*/
mjr 3:3514575d4f86 842 'P',0,'i',0,'n',0,'s',0,'c',0,'a',0,'p',0,'e',0,
mjr 3:3514575d4f86 843 ' ',0,'C',0,'o',0,'n',0,'t',0,'r',0,'o',0,'l',0,
mjr 3:3514575d4f86 844 'l',0,'e',0,'r',0 /*String iProduct */
mjr 3:3514575d4f86 845 };
mjr 3:3514575d4f86 846 return stringIproductDescriptor;
mjr 3:3514575d4f86 847 }
mjr 35:e959ffba78fd 848
mjr 35:e959ffba78fd 849 #define DEFAULT_CONFIGURATION (1)
mjr 35:e959ffba78fd 850
mjr 48:058ace2aed1d 851 const uint8_t *USBJoystick::configurationDesc()
mjr 35:e959ffba78fd 852 {
mjr 63:5cd1a5f3a41b 853 int rptlen0 = reportDescLength(0);
mjr 63:5cd1a5f3a41b 854 int rptlen1 = reportDescLength(1);
mjr 63:5cd1a5f3a41b 855 if (useKB)
mjr 35:e959ffba78fd 856 {
mjr 63:5cd1a5f3a41b 857 const int cfglenKB =
mjr 63:5cd1a5f3a41b 858 ((1 * CONFIGURATION_DESCRIPTOR_LENGTH)
mjr 63:5cd1a5f3a41b 859 + (2 * INTERFACE_DESCRIPTOR_LENGTH)
mjr 63:5cd1a5f3a41b 860 + (2 * HID_DESCRIPTOR_LENGTH)
mjr 63:5cd1a5f3a41b 861 + (4 * ENDPOINT_DESCRIPTOR_LENGTH));
mjr 63:5cd1a5f3a41b 862 static uint8_t configurationDescriptorWithKB[] =
mjr 63:5cd1a5f3a41b 863 {
mjr 63:5cd1a5f3a41b 864 CONFIGURATION_DESCRIPTOR_LENGTH,// bLength
mjr 63:5cd1a5f3a41b 865 CONFIGURATION_DESCRIPTOR, // bDescriptorType
mjr 63:5cd1a5f3a41b 866 LSB(cfglenKB), // wTotalLength (LSB)
mjr 63:5cd1a5f3a41b 867 MSB(cfglenKB), // wTotalLength (MSB)
mjr 63:5cd1a5f3a41b 868 0x02, // bNumInterfaces - TWO INTERFACES (JOYSTICK + KEYBOARD)
mjr 63:5cd1a5f3a41b 869 DEFAULT_CONFIGURATION, // bConfigurationValue
mjr 63:5cd1a5f3a41b 870 0x00, // iConfiguration
mjr 63:5cd1a5f3a41b 871 C_RESERVED | C_SELF_POWERED, // bmAttributes
mjr 63:5cd1a5f3a41b 872 C_POWER(0), // bMaxPower
mjr 63:5cd1a5f3a41b 873
mjr 63:5cd1a5f3a41b 874 // ***** INTERFACE 0 - JOYSTICK/LEDWIZ ******
mjr 63:5cd1a5f3a41b 875 INTERFACE_DESCRIPTOR_LENGTH, // bLength
mjr 63:5cd1a5f3a41b 876 INTERFACE_DESCRIPTOR, // bDescriptorType
mjr 63:5cd1a5f3a41b 877 0x00, // bInterfaceNumber
mjr 63:5cd1a5f3a41b 878 0x00, // bAlternateSetting
mjr 63:5cd1a5f3a41b 879 0x02, // bNumEndpoints
mjr 63:5cd1a5f3a41b 880 HID_CLASS, // bInterfaceClass
mjr 63:5cd1a5f3a41b 881 HID_SUBCLASS_NONE, // bInterfaceSubClass
mjr 63:5cd1a5f3a41b 882 HID_PROTOCOL_NONE, // bInterfaceProtocol
mjr 63:5cd1a5f3a41b 883 0x00, // iInterface
mjr 63:5cd1a5f3a41b 884
mjr 63:5cd1a5f3a41b 885 HID_DESCRIPTOR_LENGTH, // bLength
mjr 63:5cd1a5f3a41b 886 HID_DESCRIPTOR, // bDescriptorType
mjr 63:5cd1a5f3a41b 887 LSB(HID_VERSION_1_11), // bcdHID (LSB)
mjr 63:5cd1a5f3a41b 888 MSB(HID_VERSION_1_11), // bcdHID (MSB)
mjr 63:5cd1a5f3a41b 889 0x00, // bCountryCode
mjr 63:5cd1a5f3a41b 890 0x01, // bNumDescriptors
mjr 63:5cd1a5f3a41b 891 REPORT_DESCRIPTOR, // bDescriptorType
mjr 63:5cd1a5f3a41b 892 LSB(rptlen0), // wDescriptorLength (LSB)
mjr 63:5cd1a5f3a41b 893 MSB(rptlen0), // wDescriptorLength (MSB)
mjr 63:5cd1a5f3a41b 894
mjr 63:5cd1a5f3a41b 895 ENDPOINT_DESCRIPTOR_LENGTH, // bLength
mjr 63:5cd1a5f3a41b 896 ENDPOINT_DESCRIPTOR, // bDescriptorType
mjr 63:5cd1a5f3a41b 897 PHY_TO_DESC(EPINT_IN), // bEndpointAddress - EPINT == EP1
mjr 63:5cd1a5f3a41b 898 E_INTERRUPT, // bmAttributes
mjr 63:5cd1a5f3a41b 899 LSB(MAX_PACKET_SIZE_EPINT), // wMaxPacketSize (LSB)
mjr 63:5cd1a5f3a41b 900 MSB(MAX_PACKET_SIZE_EPINT), // wMaxPacketSize (MSB)
mjr 63:5cd1a5f3a41b 901 1, // bInterval (milliseconds)
mjr 63:5cd1a5f3a41b 902
mjr 63:5cd1a5f3a41b 903 ENDPOINT_DESCRIPTOR_LENGTH, // bLength
mjr 63:5cd1a5f3a41b 904 ENDPOINT_DESCRIPTOR, // bDescriptorType
mjr 63:5cd1a5f3a41b 905 PHY_TO_DESC(EPINT_OUT), // bEndpointAddress - EPINT == EP1
mjr 63:5cd1a5f3a41b 906 E_INTERRUPT, // bmAttributes
mjr 63:5cd1a5f3a41b 907 LSB(MAX_PACKET_SIZE_EPINT), // wMaxPacketSize (LSB)
mjr 63:5cd1a5f3a41b 908 MSB(MAX_PACKET_SIZE_EPINT), // wMaxPacketSize (MSB)
mjr 63:5cd1a5f3a41b 909 1, // bInterval (milliseconds)
mjr 63:5cd1a5f3a41b 910
mjr 63:5cd1a5f3a41b 911 // ****** INTERFACE 1 - KEYBOARD ******
mjr 63:5cd1a5f3a41b 912 INTERFACE_DESCRIPTOR_LENGTH, // bLength
mjr 63:5cd1a5f3a41b 913 INTERFACE_DESCRIPTOR, // bDescriptorType
mjr 63:5cd1a5f3a41b 914 0x01, // bInterfaceNumber
mjr 63:5cd1a5f3a41b 915 0x00, // bAlternateSetting
mjr 63:5cd1a5f3a41b 916 0x02, // bNumEndpoints
mjr 63:5cd1a5f3a41b 917 HID_CLASS, // bInterfaceClass
mjr 63:5cd1a5f3a41b 918 HID_SUBCLASS_BOOT, // bInterfaceSubClass
mjr 63:5cd1a5f3a41b 919 HID_PROTOCOL_KB, // bInterfaceProtocol
mjr 63:5cd1a5f3a41b 920 0x00, // iInterface
mjr 63:5cd1a5f3a41b 921
mjr 63:5cd1a5f3a41b 922 HID_DESCRIPTOR_LENGTH, // bLength
mjr 63:5cd1a5f3a41b 923 HID_DESCRIPTOR, // bDescriptorType
mjr 63:5cd1a5f3a41b 924 LSB(HID_VERSION_1_11), // bcdHID (LSB)
mjr 63:5cd1a5f3a41b 925 MSB(HID_VERSION_1_11), // bcdHID (MSB)
mjr 63:5cd1a5f3a41b 926 0x00, // bCountryCode
mjr 63:5cd1a5f3a41b 927 0x01, // bNumDescriptors
mjr 63:5cd1a5f3a41b 928 REPORT_DESCRIPTOR, // bDescriptorType
mjr 63:5cd1a5f3a41b 929 LSB(rptlen1), // wDescriptorLength (LSB)
mjr 63:5cd1a5f3a41b 930 MSB(rptlen1), // wDescriptorLength (MSB)
mjr 63:5cd1a5f3a41b 931
mjr 63:5cd1a5f3a41b 932 ENDPOINT_DESCRIPTOR_LENGTH, // bLength
mjr 63:5cd1a5f3a41b 933 ENDPOINT_DESCRIPTOR, // bDescriptorType
mjr 63:5cd1a5f3a41b 934 PHY_TO_DESC(EP4IN), // bEndpointAddress
mjr 63:5cd1a5f3a41b 935 E_INTERRUPT, // bmAttributes
mjr 63:5cd1a5f3a41b 936 LSB(MAX_PACKET_SIZE_EPINT), // wMaxPacketSize (LSB)
mjr 63:5cd1a5f3a41b 937 MSB(MAX_PACKET_SIZE_EPINT), // wMaxPacketSize (MSB)
mjr 63:5cd1a5f3a41b 938 1, // bInterval (milliseconds)
mjr 63:5cd1a5f3a41b 939
mjr 63:5cd1a5f3a41b 940 ENDPOINT_DESCRIPTOR_LENGTH, // bLength
mjr 63:5cd1a5f3a41b 941 ENDPOINT_DESCRIPTOR, // bDescriptorType
mjr 63:5cd1a5f3a41b 942 PHY_TO_DESC(EP4OUT), // bEndpointAddress
mjr 63:5cd1a5f3a41b 943 E_INTERRUPT, // bmAttributes
mjr 63:5cd1a5f3a41b 944 LSB(MAX_PACKET_SIZE_EPINT), // wMaxPacketSize (LSB)
mjr 63:5cd1a5f3a41b 945 MSB(MAX_PACKET_SIZE_EPINT), // wMaxPacketSize (MSB)
mjr 63:5cd1a5f3a41b 946 1, // bInterval (milliseconds)
mjr 61:3c7e6e9ec355 947
mjr 63:5cd1a5f3a41b 948 };
mjr 63:5cd1a5f3a41b 949
mjr 63:5cd1a5f3a41b 950 // Keyboard + joystick interfaces
mjr 63:5cd1a5f3a41b 951 return configurationDescriptorWithKB;
mjr 63:5cd1a5f3a41b 952 }
mjr 63:5cd1a5f3a41b 953 else
mjr 63:5cd1a5f3a41b 954 {
mjr 63:5cd1a5f3a41b 955 // No keyboard - joystick interface only
mjr 63:5cd1a5f3a41b 956 const int cfglenNoKB =
mjr 63:5cd1a5f3a41b 957 ((1 * CONFIGURATION_DESCRIPTOR_LENGTH)
mjr 63:5cd1a5f3a41b 958 + (1 * INTERFACE_DESCRIPTOR_LENGTH)
mjr 63:5cd1a5f3a41b 959 + (1 * HID_DESCRIPTOR_LENGTH)
mjr 63:5cd1a5f3a41b 960 + (2 * ENDPOINT_DESCRIPTOR_LENGTH));
mjr 63:5cd1a5f3a41b 961 static uint8_t configurationDescriptorNoKB[] =
mjr 63:5cd1a5f3a41b 962 {
mjr 63:5cd1a5f3a41b 963 CONFIGURATION_DESCRIPTOR_LENGTH,// bLength
mjr 63:5cd1a5f3a41b 964 CONFIGURATION_DESCRIPTOR, // bDescriptorType
mjr 63:5cd1a5f3a41b 965 LSB(cfglenNoKB), // wTotalLength (LSB)
mjr 63:5cd1a5f3a41b 966 MSB(cfglenNoKB), // wTotalLength (MSB)
mjr 63:5cd1a5f3a41b 967 0x01, // bNumInterfaces
mjr 63:5cd1a5f3a41b 968 DEFAULT_CONFIGURATION, // bConfigurationValue
mjr 63:5cd1a5f3a41b 969 0x00, // iConfiguration
mjr 63:5cd1a5f3a41b 970 C_RESERVED | C_SELF_POWERED, // bmAttributes
mjr 63:5cd1a5f3a41b 971 C_POWER(0), // bMaxPower
mjr 63:5cd1a5f3a41b 972
mjr 63:5cd1a5f3a41b 973 INTERFACE_DESCRIPTOR_LENGTH, // bLength
mjr 63:5cd1a5f3a41b 974 INTERFACE_DESCRIPTOR, // bDescriptorType
mjr 63:5cd1a5f3a41b 975 0x00, // bInterfaceNumber
mjr 63:5cd1a5f3a41b 976 0x00, // bAlternateSetting
mjr 63:5cd1a5f3a41b 977 0x02, // bNumEndpoints
mjr 63:5cd1a5f3a41b 978 HID_CLASS, // bInterfaceClass
mjr 63:5cd1a5f3a41b 979 HID_SUBCLASS_NONE, // bInterfaceSubClass
mjr 63:5cd1a5f3a41b 980 HID_PROTOCOL_NONE, // bInterfaceProtocol (keyboard)
mjr 63:5cd1a5f3a41b 981 0x00, // iInterface
mjr 63:5cd1a5f3a41b 982
mjr 63:5cd1a5f3a41b 983 HID_DESCRIPTOR_LENGTH, // bLength
mjr 63:5cd1a5f3a41b 984 HID_DESCRIPTOR, // bDescriptorType
mjr 63:5cd1a5f3a41b 985 LSB(HID_VERSION_1_11), // bcdHID (LSB)
mjr 63:5cd1a5f3a41b 986 MSB(HID_VERSION_1_11), // bcdHID (MSB)
mjr 63:5cd1a5f3a41b 987 0x00, // bCountryCode
mjr 63:5cd1a5f3a41b 988 0x01, // bNumDescriptors
mjr 63:5cd1a5f3a41b 989 REPORT_DESCRIPTOR, // bDescriptorType
mjr 63:5cd1a5f3a41b 990 (uint8_t)(LSB(rptlen0)), // wDescriptorLength (LSB)
mjr 63:5cd1a5f3a41b 991 (uint8_t)(MSB(rptlen0)), // wDescriptorLength (MSB)
mjr 63:5cd1a5f3a41b 992
mjr 63:5cd1a5f3a41b 993 ENDPOINT_DESCRIPTOR_LENGTH, // bLength
mjr 63:5cd1a5f3a41b 994 ENDPOINT_DESCRIPTOR, // bDescriptorType
mjr 63:5cd1a5f3a41b 995 PHY_TO_DESC(EPINT_IN), // bEndpointAddress
mjr 63:5cd1a5f3a41b 996 E_INTERRUPT, // bmAttributes
mjr 63:5cd1a5f3a41b 997 LSB(MAX_PACKET_SIZE_EPINT), // wMaxPacketSize (LSB)
mjr 63:5cd1a5f3a41b 998 MSB(MAX_PACKET_SIZE_EPINT), // wMaxPacketSize (MSB)
mjr 63:5cd1a5f3a41b 999 1, // bInterval (milliseconds)
mjr 63:5cd1a5f3a41b 1000
mjr 63:5cd1a5f3a41b 1001 ENDPOINT_DESCRIPTOR_LENGTH, // bLength
mjr 63:5cd1a5f3a41b 1002 ENDPOINT_DESCRIPTOR, // bDescriptorType
mjr 63:5cd1a5f3a41b 1003 PHY_TO_DESC(EPINT_OUT), // bEndpointAddress
mjr 63:5cd1a5f3a41b 1004 E_INTERRUPT, // bmAttributes
mjr 63:5cd1a5f3a41b 1005 LSB(MAX_PACKET_SIZE_EPINT), // wMaxPacketSize (LSB)
mjr 63:5cd1a5f3a41b 1006 MSB(MAX_PACKET_SIZE_EPINT), // wMaxPacketSize (MSB)
mjr 63:5cd1a5f3a41b 1007 1 // bInterval (milliseconds)
mjr 63:5cd1a5f3a41b 1008 };
mjr 63:5cd1a5f3a41b 1009
mjr 63:5cd1a5f3a41b 1010 return configurationDescriptorNoKB;
mjr 63:5cd1a5f3a41b 1011 }
mjr 35:e959ffba78fd 1012 }
mjr 35:e959ffba78fd 1013
mjr 35:e959ffba78fd 1014 // Set the configuration. We need to set up the endpoints for
mjr 35:e959ffba78fd 1015 // our active interfaces.
mjr 35:e959ffba78fd 1016 bool USBJoystick::USBCallback_setConfiguration(uint8_t configuration)
mjr 35:e959ffba78fd 1017 {
mjr 35:e959ffba78fd 1018 // we only have one valid configuration
mjr 35:e959ffba78fd 1019 if (configuration != DEFAULT_CONFIGURATION)
mjr 35:e959ffba78fd 1020 return false;
mjr 35:e959ffba78fd 1021
mjr 63:5cd1a5f3a41b 1022 // Configure endpoint 1 - we use this in all cases, for either
mjr 63:5cd1a5f3a41b 1023 // the combined joystick/ledwiz interface or just the ledwiz interface
mjr 48:058ace2aed1d 1024 addEndpoint(EPINT_IN, MAX_REPORT_JS_TX + 1);
mjr 48:058ace2aed1d 1025 addEndpoint(EPINT_OUT, MAX_REPORT_JS_RX + 1);
mjr 48:058ace2aed1d 1026 readStart(EPINT_OUT, MAX_REPORT_JS_TX + 1);
mjr 63:5cd1a5f3a41b 1027
mjr 63:5cd1a5f3a41b 1028 // if the keyboard is enabled, configure endpoint 4 for the kb interface
mjr 63:5cd1a5f3a41b 1029 if (useKB)
mjr 63:5cd1a5f3a41b 1030 {
mjr 63:5cd1a5f3a41b 1031 addEndpoint(EP4IN, MAX_REPORT_KB_TX + 1);
mjr 63:5cd1a5f3a41b 1032 addEndpoint(EP4OUT, MAX_REPORT_KB_RX + 1);
mjr 63:5cd1a5f3a41b 1033 readStart(EP4OUT, MAX_REPORT_KB_TX + 1);
mjr 63:5cd1a5f3a41b 1034 }
mjr 35:e959ffba78fd 1035
mjr 35:e959ffba78fd 1036 // success
mjr 35:e959ffba78fd 1037 return true;
mjr 35:e959ffba78fd 1038 }
mjr 35:e959ffba78fd 1039
mjr 38:091e511ce8a0 1040 // Handle incoming messages on the joystick/LedWiz interface = endpoint 1.
mjr 38:091e511ce8a0 1041 // This interface receives LedWiz protocol commands and commands using our
mjr 38:091e511ce8a0 1042 // custom LedWiz protocol extensions.
mjr 38:091e511ce8a0 1043 //
mjr 38:091e511ce8a0 1044 // We simply queue the messages in our circular buffer for processing in
mjr 38:091e511ce8a0 1045 // the main loop. The circular buffer object is designed for safe access
mjr 38:091e511ce8a0 1046 // from the interrupt handler using the rule that only the interrupt
mjr 38:091e511ce8a0 1047 // handler can change the write pointer, and only the regular code can
mjr 38:091e511ce8a0 1048 // change the read pointer.
mjr 38:091e511ce8a0 1049 bool USBJoystick::EP1_OUT_callback()
mjr 38:091e511ce8a0 1050 {
mjr 38:091e511ce8a0 1051 // Read this message
mjr 63:5cd1a5f3a41b 1052 union {
mjr 63:5cd1a5f3a41b 1053 LedWizMsg msg;
mjr 63:5cd1a5f3a41b 1054 uint8_t buf[MAX_HID_REPORT_SIZE];
mjr 63:5cd1a5f3a41b 1055 } buf;
mjr 38:091e511ce8a0 1056 uint32_t bytesRead = 0;
mjr 63:5cd1a5f3a41b 1057 USBDevice::readEP(EP1OUT, buf.buf, &bytesRead, MAX_HID_REPORT_SIZE);
mjr 38:091e511ce8a0 1058
mjr 63:5cd1a5f3a41b 1059 // if it's the right length, queue it to our circular buffer
mjr 63:5cd1a5f3a41b 1060 if (bytesRead == 8)
mjr 63:5cd1a5f3a41b 1061 lwbuf.write(buf.msg);
mjr 38:091e511ce8a0 1062
mjr 38:091e511ce8a0 1063 // start the next read
mjr 39:b3815a1c3802 1064 return readStart(EP1OUT, MAX_HID_REPORT_SIZE);
mjr 38:091e511ce8a0 1065 }
mjr 63:5cd1a5f3a41b 1066
mjr 63:5cd1a5f3a41b 1067 // Handle incoming messages on the keyboard interface = endpoint 4.
mjr 63:5cd1a5f3a41b 1068 // The host uses this to send updates for the keyboard indicator LEDs
mjr 63:5cd1a5f3a41b 1069 // (caps lock, num lock, etc). We don't do anything with these, but
mjr 63:5cd1a5f3a41b 1070 // we have to read them to keep the pipe open.
mjr 63:5cd1a5f3a41b 1071 bool USBJoystick::EP4_OUT_callback()
mjr 63:5cd1a5f3a41b 1072 {
mjr 63:5cd1a5f3a41b 1073 // read this message
mjr 63:5cd1a5f3a41b 1074 uint32_t bytesRead = 0;
mjr 63:5cd1a5f3a41b 1075 uint8_t led[MAX_HID_REPORT_SIZE];
mjr 63:5cd1a5f3a41b 1076 USBDevice::readEP(EP4OUT, led, &bytesRead, MAX_HID_REPORT_SIZE);
mjr 63:5cd1a5f3a41b 1077
mjr 63:5cd1a5f3a41b 1078 // start the next read
mjr 63:5cd1a5f3a41b 1079 return readStart(EP4OUT, MAX_HID_REPORT_SIZE);
mjr 63:5cd1a5f3a41b 1080 }
mjr 63:5cd1a5f3a41b 1081