Fixed doubled report ID for Bluetooth Classic devices on Linux

This commit is contained in:
Sam Lantinga 2020-11-13 13:17:04 -08:00
parent 8cd4f364db
commit 30371f2d4d
1 changed files with 17 additions and 13 deletions
src/hidapi/linux

View File

@ -86,7 +86,7 @@ struct hid_device_ {
int device_handle; int device_handle;
int blocking; int blocking;
int uses_numbered_reports; int uses_numbered_reports;
int is_bluetooth; int needs_ble_hack;
}; };
@ -119,7 +119,7 @@ static hid_device *new_hid_device(void)
dev->device_handle = -1; dev->device_handle = -1;
dev->blocking = 1; dev->blocking = 1;
dev->uses_numbered_reports = 0; dev->uses_numbered_reports = 0;
dev->is_bluetooth = 0; dev->needs_ble_hack = 0;
return dev; return dev;
} }
@ -269,12 +269,12 @@ next_line:
return (found_id && found_name && found_serial); return (found_id && found_name && found_serial);
} }
static int is_bluetooth(hid_device *dev) static int is_BLE(hid_device *dev)
{ {
struct udev *udev; struct udev *udev;
struct udev_device *udev_dev, *hid_dev; struct udev_device *udev_dev, *hid_dev;
struct stat s; struct stat s;
int ret = -1; int ret;
/* Create the udev object */ /* Create the udev object */
udev = udev_new(); udev = udev_new();
@ -284,13 +284,13 @@ static int is_bluetooth(hid_device *dev)
} }
/* Get the dev_t (major/minor numbers) from the file handle. */ /* Get the dev_t (major/minor numbers) from the file handle. */
ret = fstat(dev->device_handle, &s); if (fstat(dev->device_handle, &s) < 0) {
if (-1 == ret) {
udev_unref(udev); udev_unref(udev);
return ret; return -1;
} }
/* Open a udev device from the dev_t. 'c' means character device. */ /* Open a udev device from the dev_t. 'c' means character device. */
ret = 0;
udev_dev = udev_device_new_from_devnum(udev, 'c', s.st_rdev); udev_dev = udev_device_new_from_devnum(udev, 'c', s.st_rdev);
if (udev_dev) { if (udev_dev) {
hid_dev = udev_device_get_parent_with_subsystem_devtype( hid_dev = udev_device_get_parent_with_subsystem_devtype(
@ -314,7 +314,12 @@ static int is_bluetooth(hid_device *dev)
free(serial_number_utf8); free(serial_number_utf8);
free(product_name_utf8); free(product_name_utf8);
ret = (bus_type == BUS_BLUETOOTH); if (bus_type == BUS_BLUETOOTH) {
/* Right now the Steam Controller is the only BLE device that we send feature reports to */
if (dev_vid == 0x28de /* Valve */) {
ret = 1;
}
}
/* hid_dev doesn't need to be (and can't be) unref'd. /* hid_dev doesn't need to be (and can't be) unref'd.
I'm not sure why, but it'll throw double-free() errors. */ I'm not sure why, but it'll throw double-free() errors. */
@ -327,7 +332,6 @@ static int is_bluetooth(hid_device *dev)
return ret; return ret;
} }
static int get_device_string(hid_device *dev, enum device_string_id key, wchar_t *string, size_t maxlen) static int get_device_string(hid_device *dev, enum device_string_id key, wchar_t *string, size_t maxlen)
{ {
struct udev *udev; struct udev *udev;
@ -741,7 +745,7 @@ hid_device * HID_API_EXPORT hid_open_path(const char *path, int bExclusive)
rpt_desc.size); rpt_desc.size);
} }
dev->is_bluetooth = (is_bluetooth(dev) == 1); dev->needs_ble_hack = (is_BLE(dev) == 1);
return dev; return dev;
} }
@ -840,8 +844,8 @@ int HID_API_EXPORT hid_get_feature_report(hid_device *dev, unsigned char *data,
{ {
int res; int res;
/* It looks like HIDIOCGFEATURE() on Bluetooth devices doesn't return the report number */ /* It looks like HIDIOCGFEATURE() on Bluetooth LE devices doesn't return the report number */
if (dev->is_bluetooth) { if (dev->needs_ble_hack) {
data[1] = data[0]; data[1] = data[0];
++data; ++data;
--length; --length;
@ -849,7 +853,7 @@ int HID_API_EXPORT hid_get_feature_report(hid_device *dev, unsigned char *data,
res = ioctl(dev->device_handle, HIDIOCGFEATURE(length), data); res = ioctl(dev->device_handle, HIDIOCGFEATURE(length), data);
if (res < 0) if (res < 0)
perror("ioctl (GFEATURE)"); perror("ioctl (GFEATURE)");
else if (dev->is_bluetooth) else if (dev->needs_ble_hack)
++res; ++res;
return res; return res;