|
|
|
|
|
|
|
|
|
|
|
// Simple test of USB Host Joystick |
|
|
|
|
|
// |
|
|
|
|
|
// This example is in the public domain |
|
|
|
|
|
|
|
|
|
|
|
#include "USBHost_t36.h" |
|
|
|
|
|
|
|
|
|
|
|
USBHost myusb; |
|
|
|
|
|
USBHub hub1(myusb); |
|
|
|
|
|
USBHIDParser hid1(myusb); |
|
|
|
|
|
USBHIDParser hid2(myusb); |
|
|
|
|
|
USBHIDParser hid3(myusb); |
|
|
|
|
|
USBHIDParser hid4(myusb); |
|
|
|
|
|
#define COUNT_JOYSTICKS 4 |
|
|
|
|
|
JoystickController joysticks[COUNT_JOYSTICKS](myusb); |
|
|
|
|
|
int user_axis[64]; |
|
|
|
|
|
uint32_t buttons_prev = 0; |
|
|
|
|
|
|
|
|
|
|
|
USBDriver *drivers[] = {&hub1, &joysticks[0], &joysticks[1], &joysticks[2], &joysticks[3], &hid1, &hid2, &hid3, &hid4}; |
|
|
|
|
|
#define CNT_DEVICES (sizeof(drivers)/sizeof(drivers[0])) |
|
|
|
|
|
const char * driver_names[CNT_DEVICES] = {"Hub1", "joystick[0D]", "joystick[1D]", "joystick[2D]", "joystick[3D]", "HID1", "HID2", "HID3", "HID4"}; |
|
|
|
|
|
bool driver_active[CNT_DEVICES] = {false, false, false, false}; |
|
|
|
|
|
|
|
|
|
|
|
// Lets also look at HID Input devices |
|
|
|
|
|
USBHIDInput *hiddrivers[] = {&joysticks[0], &joysticks[1], &joysticks[2], &joysticks[3]}; |
|
|
|
|
|
#define CNT_HIDDEVICES (sizeof(hiddrivers)/sizeof(hiddrivers[0])) |
|
|
|
|
|
const char * hid_driver_names[CNT_DEVICES] = {"joystick[0H]", "joystick[1H]", "joystick[2H]", "joystick[3H]"}; |
|
|
|
|
|
bool hid_driver_active[CNT_DEVICES] = {false}; |
|
|
|
|
|
bool show_changed_only = false; |
|
|
|
|
|
|
|
|
|
|
|
uint8_t joystick_left_trigger_value[COUNT_JOYSTICKS] = {0}; |
|
|
|
|
|
uint8_t joystick_right_trigger_value[COUNT_JOYSTICKS] = {0}; |
|
|
|
|
|
uint64_t joystick_full_notify_mask = (uint64_t) - 1; |
|
|
|
|
|
|
|
|
|
|
|
//============================================================================= |
|
|
|
|
|
// Setup |
|
|
|
|
|
//============================================================================= |
|
|
|
|
|
void setup() |
|
|
|
|
|
{ |
|
|
|
|
|
while (!Serial) ; // wait for Arduino Serial Monitor |
|
|
|
|
|
Serial.println("\n\nUSB Host Joystick Testing"); |
|
|
|
|
|
myusb.begin(); |
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
//============================================================================= |
|
|
|
|
|
// loop |
|
|
|
|
|
//============================================================================= |
|
|
|
|
|
void loop() |
|
|
|
|
|
{ |
|
|
|
|
|
myusb.Task(); |
|
|
|
|
|
PrintDeviceListChanges(); |
|
|
|
|
|
|
|
|
|
|
|
if (Serial.available()) { |
|
|
|
|
|
int ch = Serial.read(); // get the first char. |
|
|
|
|
|
while (Serial.read() != -1) ; |
|
|
|
|
|
if ((ch == 'b') || (ch == 'B')) { |
|
|
|
|
|
Serial.println("Only notify on Basic Axis changes"); |
|
|
|
|
|
for (int joystick_index = 0; joystick_index < COUNT_JOYSTICKS; joystick_index++) |
|
|
|
|
|
joysticks[joystick_index].axisChangeNotifyMask(0x3ff); |
|
|
|
|
|
} else if ((ch == 'f') || (ch == 'F')) { |
|
|
|
|
|
Serial.println("Only notify on Full Axis changes"); |
|
|
|
|
|
for (int joystick_index = 0; joystick_index < COUNT_JOYSTICKS; joystick_index++) |
|
|
|
|
|
joysticks[joystick_index].axisChangeNotifyMask(joystick_full_notify_mask); |
|
|
|
|
|
|
|
|
|
|
|
} else { |
|
|
|
|
|
if (show_changed_only) { |
|
|
|
|
|
show_changed_only = false; |
|
|
|
|
|
Serial.println("\n*** Show All fields mode ***"); |
|
|
|
|
|
} else { |
|
|
|
|
|
show_changed_only = true; |
|
|
|
|
|
Serial.println("\n*** Show only changed fields mode ***"); |
|
|
|
|
|
} |
|
|
|
|
|
} |
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
for (int joystick_index = 0; joystick_index < COUNT_JOYSTICKS; joystick_index++) { |
|
|
|
|
|
if (joysticks[joystick_index].available()) { |
|
|
|
|
|
uint64_t axis_mask = joysticks[joystick_index].axisMask(); |
|
|
|
|
|
uint64_t axis_changed_mask = joysticks[joystick_index].axisChangedMask(); |
|
|
|
|
|
uint32_t buttons = joysticks[joystick_index].getButtons(); |
|
|
|
|
|
Serial.printf("Joystick(%d): buttons = %x", joystick_index, buttons); |
|
|
|
|
|
//Serial.printf(" AMasks: %x %x:%x", axis_mask, (uint32_t)(user_axis_mask >> 32), (uint32_t)(user_axis_mask & 0xffffffff)); |
|
|
|
|
|
//Serial.printf(" M: %lx %lx", axis_mask, joysticks[joystick_index].axisChangedMask()); |
|
|
|
|
|
if (show_changed_only) { |
|
|
|
|
|
for (uint8_t i = 0; axis_changed_mask != 0; i++, axis_changed_mask >>= 1) { |
|
|
|
|
|
if (axis_changed_mask & 1) { |
|
|
|
|
|
Serial.printf(" %d:%d", i, joysticks[joystick_index].getAxis(i)); |
|
|
|
|
|
} |
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
} else { |
|
|
|
|
|
for (uint8_t i = 0; axis_mask != 0; i++, axis_mask >>= 1) { |
|
|
|
|
|
if (axis_mask & 1) { |
|
|
|
|
|
Serial.printf(" %d:%d", i, joysticks[joystick_index].getAxis(i)); |
|
|
|
|
|
} |
|
|
|
|
|
} |
|
|
|
|
|
} |
|
|
|
|
|
uint8_t ltv; |
|
|
|
|
|
uint8_t rtv; |
|
|
|
|
|
switch (joysticks[joystick_index].joystickType) { |
|
|
|
|
|
default: |
|
|
|
|
|
break; |
|
|
|
|
|
case JoystickController::PS4: |
|
|
|
|
|
ltv = joysticks[joystick_index].getAxis(3); |
|
|
|
|
|
rtv = joysticks[joystick_index].getAxis(4); |
|
|
|
|
|
if ((ltv != joystick_left_trigger_value[joystick_index]) || (rtv != joystick_right_trigger_value[joystick_index])) { |
|
|
|
|
|
joystick_left_trigger_value[joystick_index] = ltv; |
|
|
|
|
|
joystick_right_trigger_value[joystick_index] = rtv; |
|
|
|
|
|
joysticks[joystick_index].setRumble(ltv, rtv); |
|
|
|
|
|
} |
|
|
|
|
|
break; |
|
|
|
|
|
|
|
|
|
|
|
case JoystickController::PS3: |
|
|
|
|
|
ltv = joysticks[joystick_index].getAxis(18); |
|
|
|
|
|
rtv = joysticks[joystick_index].getAxis(19); |
|
|
|
|
|
if ((ltv != joystick_left_trigger_value[joystick_index]) || (rtv != joystick_right_trigger_value[joystick_index])) { |
|
|
|
|
|
joystick_left_trigger_value[joystick_index] = ltv; |
|
|
|
|
|
joystick_right_trigger_value[joystick_index] = rtv; |
|
|
|
|
|
joysticks[joystick_index].setRumble(ltv, rtv, 50); |
|
|
|
|
|
} |
|
|
|
|
|
break; |
|
|
|
|
|
|
|
|
|
|
|
case JoystickController::XBOXONE: |
|
|
|
|
|
case JoystickController::XBOX360: |
|
|
|
|
|
ltv = joysticks[joystick_index].getAxis(4); |
|
|
|
|
|
rtv = joysticks[joystick_index].getAxis(5); |
|
|
|
|
|
if ((ltv != joystick_left_trigger_value[joystick_index]) || (rtv != joystick_right_trigger_value[joystick_index])) { |
|
|
|
|
|
joystick_left_trigger_value[joystick_index] = ltv; |
|
|
|
|
|
joystick_right_trigger_value[joystick_index] = rtv; |
|
|
|
|
|
joysticks[joystick_index].setRumble(ltv, rtv); |
|
|
|
|
|
Serial.printf(" Set Rumble %d %d", ltv, rtv); |
|
|
|
|
|
} |
|
|
|
|
|
break; |
|
|
|
|
|
} |
|
|
|
|
|
if (buttons != buttons_prev) { |
|
|
|
|
|
if (joysticks[joystick_index].joystickType == JoystickController::PS3) { |
|
|
|
|
|
joysticks[joystick_index].setLEDs((buttons >> 12) & 0xf); // try to get to TRI/CIR/X/SQuare |
|
|
|
|
|
} else { |
|
|
|
|
|
uint8_t lr = (buttons & 1) ? 0xff : 0; |
|
|
|
|
|
uint8_t lg = (buttons & 2) ? 0xff : 0; |
|
|
|
|
|
uint8_t lb = (buttons & 4) ? 0xff : 0; |
|
|
|
|
|
joysticks[joystick_index].setLEDs(lr, lg, lb); |
|
|
|
|
|
} |
|
|
|
|
|
buttons_prev = buttons; |
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
Serial.println(); |
|
|
|
|
|
joysticks[joystick_index].joystickDataClear(); |
|
|
|
|
|
} |
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
//============================================================================= |
|
|
|
|
|
// Show when devices are added or removed |
|
|
|
|
|
//============================================================================= |
|
|
|
|
|
void PrintDeviceListChanges() { |
|
|
|
|
|
for (uint8_t i = 0; i < CNT_DEVICES; i++) { |
|
|
|
|
|
if (*drivers[i] != driver_active[i]) { |
|
|
|
|
|
if (driver_active[i]) { |
|
|
|
|
|
Serial.printf("*** Device %s - disconnected ***\n", driver_names[i]); |
|
|
|
|
|
driver_active[i] = false; |
|
|
|
|
|
} else { |
|
|
|
|
|
Serial.printf("*** Device %s %x:%x - connected ***\n", driver_names[i], drivers[i]->idVendor(), drivers[i]->idProduct()); |
|
|
|
|
|
driver_active[i] = true; |
|
|
|
|
|
|
|
|
|
|
|
const uint8_t *psz = drivers[i]->manufacturer(); |
|
|
|
|
|
if (psz && *psz) Serial.printf(" manufacturer: %s\n", psz); |
|
|
|
|
|
psz = drivers[i]->product(); |
|
|
|
|
|
if (psz && *psz) Serial.printf(" product: %s\n", psz); |
|
|
|
|
|
psz = drivers[i]->serialNumber(); |
|
|
|
|
|
if (psz && *psz) Serial.printf(" Serial: %s\n", psz); |
|
|
|
|
|
} |
|
|
|
|
|
} |
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
for (uint8_t i = 0; i < CNT_HIDDEVICES; i++) { |
|
|
|
|
|
if (*hiddrivers[i] != hid_driver_active[i]) { |
|
|
|
|
|
if (hid_driver_active[i]) { |
|
|
|
|
|
Serial.printf("*** HID Device %s - disconnected ***\n", hid_driver_names[i]); |
|
|
|
|
|
hid_driver_active[i] = false; |
|
|
|
|
|
} else { |
|
|
|
|
|
Serial.printf("*** HID Device %s %x:%x - connected ***\n", hid_driver_names[i], hiddrivers[i]->idVendor(), hiddrivers[i]->idProduct()); |
|
|
|
|
|
hid_driver_active[i] = true; |
|
|
|
|
|
|
|
|
|
|
|
const uint8_t *psz = hiddrivers[i]->manufacturer(); |
|
|
|
|
|
if (psz && *psz) Serial.printf(" manufacturer: %s\n", psz); |
|
|
|
|
|
psz = hiddrivers[i]->product(); |
|
|
|
|
|
if (psz && *psz) Serial.printf(" product: %s\n", psz); |
|
|
|
|
|
psz = hiddrivers[i]->serialNumber(); |
|
|
|
|
|
if (psz && *psz) Serial.printf(" Serial: %s\n", psz); |
|
|
|
|
|
} |
|
|
|
|
|
} |
|
|
|
|
|
} |
|
|
|
|
|
} |