| 1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396 |
- /*******************************************************
- HIDAPI - Multi-Platform library for
- communication with HID devices.
- Alan Ott
- Signal 11 Software
- libusb/hidapi Team
- Copyright 2022, All Rights Reserved.
- At the discretion of the user of this library,
- this software may be licensed under the terms of the
- GNU General Public License v3, a BSD-Style license, or the
- original HIDAPI license as outlined in the LICENSE.txt,
- LICENSE-gpl3.txt, LICENSE-bsd.txt, and LICENSE-orig.txt
- files located at the root of the source distribution.
- These files may also be found in the public source
- code repository located at:
- https://github.com/libusb/hidapi .
- ********************************************************/
- /* C */
- #include <stdio.h>
- #include <string.h>
- #include <stdlib.h>
- #include <locale.h>
- #include <errno.h>
- /* Unix */
- #include <unistd.h>
- #include <sys/types.h>
- #include <sys/stat.h>
- #include <sys/ioctl.h>
- #include <sys/utsname.h>
- #include <fcntl.h>
- #include <poll.h>
- /* Linux */
- #include <linux/hidraw.h>
- #include <linux/version.h>
- #include <linux/input.h>
- #include <libudev.h>
- #include "../hidapi/hidapi.h"
- #ifdef HIDAPI_ALLOW_BUILD_WORKAROUND_KERNEL_2_6_39
- /* This definitions first appeared in Linux Kernel 2.6.39 in linux/hidraw.h.
- hidapi doesn't support kernels older than that,
- so we don't define macros below explicitly, to fail builds on old kernels.
- For those who really need this as a workaround (e.g. to be able to build on old build machines),
- can workaround by defining the macro above.
- */
- #ifndef HIDIOCSFEATURE
- #define HIDIOCSFEATURE(len) _IOC(_IOC_WRITE|_IOC_READ, 'H', 0x06, len)
- #endif
- #ifndef HIDIOCGFEATURE
- #define HIDIOCGFEATURE(len) _IOC(_IOC_WRITE|_IOC_READ, 'H', 0x07, len)
- #endif
- #endif
- // HIDIOCGINPUT is not defined in Linux kernel headers < 5.11.
- // This definition is from hidraw.h in Linux >= 5.11.
- // https://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git/commit/?id=f43d3870cafa2a0f3854c1819c8385733db8f9ae
- #ifndef HIDIOCGINPUT
- #define HIDIOCGINPUT(len) _IOC(_IOC_WRITE|_IOC_READ, 'H', 0x0A, len)
- #endif
- struct hid_device_ {
- int device_handle;
- int blocking;
- int needs_ble_hack;
- wchar_t *last_error_str;
- struct hid_device_info* device_info;
- };
- static struct hid_api_version api_version = {
- .major = HID_API_VERSION_MAJOR,
- .minor = HID_API_VERSION_MINOR,
- .patch = HID_API_VERSION_PATCH
- };
- static wchar_t *last_global_error_str = NULL;
- static hid_device *new_hid_device(void)
- {
- hid_device *dev = (hid_device*) calloc(1, sizeof(hid_device));
- if (dev == NULL) {
- return NULL;
- }
- dev->device_handle = -1;
- dev->blocking = 1;
- dev->last_error_str = NULL;
- dev->device_info = NULL;
- return dev;
- }
- /* The caller must free the returned string with free(). */
- static wchar_t *utf8_to_wchar_t(const char *utf8)
- {
- wchar_t *ret = NULL;
- if (utf8) {
- size_t wlen = mbstowcs(NULL, utf8, 0);
- if ((size_t) -1 == wlen) {
- return wcsdup(L"");
- }
- ret = (wchar_t*) calloc(wlen+1, sizeof(wchar_t));
- if (ret == NULL) {
- /* as much as we can do at this point */
- return NULL;
- }
- mbstowcs(ret, utf8, wlen+1);
- ret[wlen] = 0x0000;
- }
- return ret;
- }
- /* Makes a copy of the given error message (and decoded according to the
- * currently locale) into the wide string pointer pointed by error_str.
- * The last stored error string is freed.
- * Use register_error_str(NULL) to free the error message completely. */
- static void register_error_str(wchar_t **error_str, const char *msg)
- {
- free(*error_str);
- *error_str = utf8_to_wchar_t(msg);
- }
- /* Semilar to register_error_str, but allows passing a format string with va_list args into this function. */
- static void register_error_str_vformat(wchar_t **error_str, const char *format, va_list args)
- {
- char msg[256];
- vsnprintf(msg, sizeof(msg), format, args);
- register_error_str(error_str, msg);
- }
- /* Set the last global error to be reported by hid_error(NULL).
- * The given error message will be copied (and decoded according to the
- * currently locale, so do not pass in string constants).
- * The last stored global error message is freed.
- * Use register_global_error(NULL) to indicate "no error". */
- static void register_global_error(const char *msg)
- {
- register_error_str(&last_global_error_str, msg);
- }
- /* Similar to register_global_error, but allows passing a format string into this function. */
- static void register_global_error_format(const char *format, ...)
- {
- va_list args;
- va_start(args, format);
- register_error_str_vformat(&last_global_error_str, format, args);
- va_end(args);
- }
- /* Set the last error for a device to be reported by hid_error(dev).
- * The given error message will be copied (and decoded according to the
- * currently locale, so do not pass in string constants).
- * The last stored device error message is freed.
- * Use register_device_error(dev, NULL) to indicate "no error". */
- static void register_device_error(hid_device *dev, const char *msg)
- {
- register_error_str(&dev->last_error_str, msg);
- }
- /* Similar to register_device_error, but you can pass a format string into this function. */
- static void register_device_error_format(hid_device *dev, const char *format, ...)
- {
- va_list args;
- va_start(args, format);
- register_error_str_vformat(&dev->last_error_str, format, args);
- va_end(args);
- }
- /* Get an attribute value from a udev_device and return it as a whar_t
- string. The returned string must be freed with free() when done.*/
- static wchar_t *copy_udev_string(struct udev_device *dev, const char *udev_name)
- {
- return utf8_to_wchar_t(udev_device_get_sysattr_value(dev, udev_name));
- }
- /*
- * Gets the size of the HID item at the given position
- * Returns 1 if successful, 0 if an invalid key
- * Sets data_len and key_size when successful
- */
- static int get_hid_item_size(__u8 *report_descriptor, unsigned int pos, __u32 size, int *data_len, int *key_size)
- {
- int key = report_descriptor[pos];
- int size_code;
- /*
- * This is a Long Item. The next byte contains the
- * length of the data section (value) for this key.
- * See the HID specification, version 1.11, section
- * 6.2.2.3, titled "Long Items."
- */
- if ((key & 0xf0) == 0xf0) {
- if (pos + 1 < size)
- {
- *data_len = report_descriptor[pos + 1];
- *key_size = 3;
- return 1;
- }
- *data_len = 0; /* malformed report */
- *key_size = 0;
- }
- /*
- * This is a Short Item. The bottom two bits of the
- * key contain the size code for the data section
- * (value) for this key. Refer to the HID
- * specification, version 1.11, section 6.2.2.2,
- * titled "Short Items."
- */
- size_code = key & 0x3;
- switch (size_code) {
- case 0:
- case 1:
- case 2:
- *data_len = size_code;
- *key_size = 1;
- return 1;
- case 3:
- *data_len = 4;
- *key_size = 1;
- return 1;
- default:
- /* Can't ever happen since size_code is & 0x3 */
- *data_len = 0;
- *key_size = 0;
- break;
- };
- /* malformed report */
- return 0;
- }
- /*
- * Get bytes from a HID Report Descriptor.
- * Only call with a num_bytes of 0, 1, 2, or 4.
- */
- static __u32 get_hid_report_bytes(__u8 *rpt, size_t len, size_t num_bytes, size_t cur)
- {
- /* Return if there aren't enough bytes. */
- if (cur + num_bytes >= len)
- return 0;
- if (num_bytes == 0)
- return 0;
- else if (num_bytes == 1)
- return rpt[cur + 1];
- else if (num_bytes == 2)
- return (rpt[cur + 2] * 256 + rpt[cur + 1]);
- else if (num_bytes == 4)
- return (
- rpt[cur + 4] * 0x01000000 +
- rpt[cur + 3] * 0x00010000 +
- rpt[cur + 2] * 0x00000100 +
- rpt[cur + 1] * 0x00000001
- );
- else
- return 0;
- }
- /*
- * Retrieves the device's Usage Page and Usage from the report descriptor.
- * The algorithm returns the current Usage Page/Usage pair whenever a new
- * Collection is found and a Usage Local Item is currently in scope.
- * Usage Local Items are consumed by each Main Item (See. 6.2.2.8).
- * The algorithm should give similar results as Apple's:
- * https://developer.apple.com/documentation/iokit/kiohiddeviceusagepairskey?language=objc
- * Physical Collections are also matched (macOS does the same).
- *
- * This function can be called repeatedly until it returns non-0
- * Usage is found. pos is the starting point (initially 0) and will be updated
- * to the next search position.
- *
- * The return value is 0 when a pair is found.
- * 1 when finished processing descriptor.
- * -1 on a malformed report.
- */
- static int get_next_hid_usage(__u8 *report_descriptor, __u32 size, unsigned int *pos, unsigned short *usage_page, unsigned short *usage)
- {
- int data_len, key_size;
- int initial = *pos == 0; /* Used to handle case where no top-level application collection is defined */
- int usage_pair_ready = 0;
- /* Usage is a Local Item, it must be set before each Main Item (Collection) before a pair is returned */
- int usage_found = 0;
- while (*pos < size) {
- int key = report_descriptor[*pos];
- int key_cmd = key & 0xfc;
- /* Determine data_len and key_size */
- if (!get_hid_item_size(report_descriptor, *pos, size, &data_len, &key_size))
- return -1; /* malformed report */
- switch (key_cmd) {
- case 0x4: /* Usage Page 6.2.2.7 (Global) */
- *usage_page = get_hid_report_bytes(report_descriptor, size, data_len, *pos);
- break;
- case 0x8: /* Usage 6.2.2.8 (Local) */
- *usage = get_hid_report_bytes(report_descriptor, size, data_len, *pos);
- usage_found = 1;
- break;
- case 0xa0: /* Collection 6.2.2.4 (Main) */
- /* A Usage Item (Local) must be found for the pair to be valid */
- if (usage_found)
- usage_pair_ready = 1;
- /* Usage is a Local Item, unset it */
- usage_found = 0;
- break;
- case 0x80: /* Input 6.2.2.4 (Main) */
- case 0x90: /* Output 6.2.2.4 (Main) */
- case 0xb0: /* Feature 6.2.2.4 (Main) */
- case 0xc0: /* End Collection 6.2.2.4 (Main) */
- /* Usage is a Local Item, unset it */
- usage_found = 0;
- break;
- }
- /* Skip over this key and its associated data */
- *pos += data_len + key_size;
- /* Return usage pair */
- if (usage_pair_ready)
- return 0;
- }
- /* If no top-level application collection is found and usage page/usage pair is found, pair is valid
- https://docs.microsoft.com/en-us/windows-hardware/drivers/hid/top-level-collections */
- if (initial && usage_found)
- return 0; /* success */
- return 1; /* finished processing */
- }
- /*
- * Retrieves the hidraw report descriptor from a file.
- * When using this form, <sysfs_path>/device/report_descriptor, elevated priviledges are not required.
- */
- static int get_hid_report_descriptor(const char *rpt_path, struct hidraw_report_descriptor *rpt_desc)
- {
- int rpt_handle;
- ssize_t res;
- rpt_handle = open(rpt_path, O_RDONLY | O_CLOEXEC);
- if (rpt_handle < 0) {
- register_global_error_format("open failed (%s): %s", rpt_path, strerror(errno));
- return -1;
- }
- /*
- * Read in the Report Descriptor
- * The sysfs file has a maximum size of 4096 (which is the same as HID_MAX_DESCRIPTOR_SIZE) so we should always
- * be ok when reading the descriptor.
- * In practice if the HID descriptor is any larger I suspect many other things will break.
- */
- memset(rpt_desc, 0x0, sizeof(*rpt_desc));
- res = read(rpt_handle, rpt_desc->value, HID_MAX_DESCRIPTOR_SIZE);
- if (res < 0) {
- register_global_error_format("read failed (%s): %s", rpt_path, strerror(errno));
- }
- rpt_desc->size = (__u32) res;
- close(rpt_handle);
- return (int) res;
- }
- /* return size of the descriptor, or -1 on failure */
- static int get_hid_report_descriptor_from_sysfs(const char *sysfs_path, struct hidraw_report_descriptor *rpt_desc)
- {
- int res = -1;
- /* Construct <sysfs_path>/device/report_descriptor */
- size_t rpt_path_len = strlen(sysfs_path) + 25 + 1;
- char* rpt_path = (char*) calloc(1, rpt_path_len);
- snprintf(rpt_path, rpt_path_len, "%s/device/report_descriptor", sysfs_path);
- res = get_hid_report_descriptor(rpt_path, rpt_desc);
- free(rpt_path);
- return res;
- }
- /* return non-zero if successfully parsed */
- static int parse_hid_vid_pid_from_uevent(const char *uevent, unsigned *bus_type, unsigned short *vendor_id, unsigned short *product_id)
- {
- char tmp[1024];
- size_t uevent_len = strlen(uevent);
- if (uevent_len > sizeof(tmp) - 1)
- uevent_len = sizeof(tmp) - 1;
- memcpy(tmp, uevent, uevent_len);
- tmp[uevent_len] = '\0';
- char *saveptr = NULL;
- char *line;
- char *key;
- char *value;
- line = strtok_r(tmp, "\n", &saveptr);
- while (line != NULL) {
- /* line: "KEY=value" */
- key = line;
- value = strchr(line, '=');
- if (!value) {
- goto next_line;
- }
- *value = '\0';
- value++;
- if (strcmp(key, "HID_ID") == 0) {
- /**
- * type vendor product
- * HID_ID=0003:000005AC:00008242
- **/
- int ret = sscanf(value, "%x:%hx:%hx", bus_type, vendor_id, product_id);
- if (ret == 3) {
- return 1;
- }
- }
- next_line:
- line = strtok_r(NULL, "\n", &saveptr);
- }
- register_global_error("Couldn't find/parse HID_ID");
- return 0;
- }
- /* return non-zero if successfully parsed */
- static int parse_hid_vid_pid_from_uevent_path(const char *uevent_path, unsigned *bus_type, unsigned short *vendor_id, unsigned short *product_id)
- {
- int handle;
- ssize_t res;
- handle = open(uevent_path, O_RDONLY | O_CLOEXEC);
- if (handle < 0) {
- register_global_error_format("open failed (%s): %s", uevent_path, strerror(errno));
- return 0;
- }
- char buf[1024];
- res = read(handle, buf, sizeof(buf) - 1); /* -1 for '\0' at the end */
- close(handle);
- if (res < 0) {
- register_global_error_format("read failed (%s): %s", uevent_path, strerror(errno));
- return 0;
- }
- buf[res] = '\0';
- return parse_hid_vid_pid_from_uevent(buf, bus_type, vendor_id, product_id);
- }
- /* return non-zero if successfully read/parsed */
- static int parse_hid_vid_pid_from_sysfs(const char *sysfs_path, unsigned *bus_type, unsigned short *vendor_id, unsigned short *product_id)
- {
- int res = 0;
- /* Construct <sysfs_path>/device/uevent */
- size_t uevent_path_len = strlen(sysfs_path) + 14 + 1;
- char* uevent_path = (char*) calloc(1, uevent_path_len);
- snprintf(uevent_path, uevent_path_len, "%s/device/uevent", sysfs_path);
- res = parse_hid_vid_pid_from_uevent_path(uevent_path, bus_type, vendor_id, product_id);
- free(uevent_path);
- return res;
- }
- static int get_hid_report_descriptor_from_hidraw(hid_device *dev, struct hidraw_report_descriptor *rpt_desc)
- {
- int desc_size = 0;
- /* Get Report Descriptor Size */
- int res = ioctl(dev->device_handle, HIDIOCGRDESCSIZE, &desc_size);
- if (res < 0) {
- register_device_error_format(dev, "ioctl(GRDESCSIZE): %s", strerror(errno));
- return res;
- }
- /* Get Report Descriptor */
- memset(rpt_desc, 0x0, sizeof(*rpt_desc));
- rpt_desc->size = desc_size;
- res = ioctl(dev->device_handle, HIDIOCGRDESC, rpt_desc);
- if (res < 0) {
- register_device_error_format(dev, "ioctl(GRDESC): %s", strerror(errno));
- }
- return res;
- }
- /*
- * The caller is responsible for free()ing the (newly-allocated) character
- * strings pointed to by serial_number_utf8 and product_name_utf8 after use.
- */
- static int parse_uevent_info(const char *uevent, unsigned *bus_type,
- unsigned short *vendor_id, unsigned short *product_id,
- char **serial_number_utf8, char **product_name_utf8)
- {
- char tmp[1024];
- if (!uevent) {
- return 0;
- }
- size_t uevent_len = strlen(uevent);
- if (uevent_len > sizeof(tmp) - 1)
- uevent_len = sizeof(tmp) - 1;
- memcpy(tmp, uevent, uevent_len);
- tmp[uevent_len] = '\0';
- char *saveptr = NULL;
- char *line;
- char *key;
- char *value;
- int found_id = 0;
- int found_serial = 0;
- int found_name = 0;
- line = strtok_r(tmp, "\n", &saveptr);
- while (line != NULL) {
- /* line: "KEY=value" */
- key = line;
- value = strchr(line, '=');
- if (!value) {
- goto next_line;
- }
- *value = '\0';
- value++;
- if (strcmp(key, "HID_ID") == 0) {
- /**
- * type vendor product
- * HID_ID=0003:000005AC:00008242
- **/
- int ret = sscanf(value, "%x:%hx:%hx", bus_type, vendor_id, product_id);
- if (ret == 3) {
- found_id = 1;
- }
- } else if (strcmp(key, "HID_NAME") == 0) {
- /* The caller has to free the product name */
- *product_name_utf8 = strdup(value);
- found_name = 1;
- } else if (strcmp(key, "HID_UNIQ") == 0) {
- /* The caller has to free the serial number */
- *serial_number_utf8 = strdup(value);
- found_serial = 1;
- }
- next_line:
- line = strtok_r(NULL, "\n", &saveptr);
- }
- return (found_id && found_name && found_serial);
- }
- static int is_BLE(hid_device *dev)
- {
- struct udev *udev;
- struct udev_device *udev_dev, *hid_dev;
- struct stat s;
- int ret;
- /* Create the udev object */
- udev = udev_new();
- if (!udev) {
- printf("Can't create udev\n");
- return -1;
- }
- /* Get the dev_t (major/minor numbers) from the file handle. */
- if (fstat(dev->device_handle, &s) < 0) {
- udev_unref(udev);
- return -1;
- }
- /* 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);
- if (udev_dev) {
- hid_dev = udev_device_get_parent_with_subsystem_devtype(
- udev_dev,
- "hid",
- NULL);
- if (hid_dev) {
- unsigned short dev_vid = 0;
- unsigned short dev_pid = 0;
- unsigned bus_type = 0;
- char *serial_number_utf8 = NULL;
- char *product_name_utf8 = NULL;
- parse_uevent_info(
- udev_device_get_sysattr_value(hid_dev, "uevent"),
- &bus_type,
- &dev_vid,
- &dev_pid,
- &serial_number_utf8,
- &product_name_utf8);
- free(serial_number_utf8);
- free(product_name_utf8);
- 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.
- I'm not sure why, but it'll throw double-free() errors. */
- }
- udev_device_unref(udev_dev);
- }
- udev_unref(udev);
- return ret;
- }
- static struct hid_device_info * create_device_info_for_device(struct udev_device *raw_dev)
- {
- struct hid_device_info *root = NULL;
- struct hid_device_info *cur_dev = NULL;
- const char *sysfs_path;
- const char *dev_path;
- const char *str;
- struct udev_device *hid_dev; /* The device's HID udev node. */
- struct udev_device *usb_dev; /* The device's USB udev node. */
- struct udev_device *intf_dev; /* The device's interface (in the USB sense). */
- unsigned short dev_vid;
- unsigned short dev_pid;
- char *serial_number_utf8 = NULL;
- char *product_name_utf8 = NULL;
- unsigned bus_type;
- int result;
- struct hidraw_report_descriptor report_desc;
- sysfs_path = udev_device_get_syspath(raw_dev);
- dev_path = udev_device_get_devnode(raw_dev);
- hid_dev = udev_device_get_parent_with_subsystem_devtype(
- raw_dev,
- "hid",
- NULL);
- if (!hid_dev) {
- /* Unable to find parent hid device. */
- goto end;
- }
- result = parse_uevent_info(
- udev_device_get_sysattr_value(hid_dev, "uevent"),
- &bus_type,
- &dev_vid,
- &dev_pid,
- &serial_number_utf8,
- &product_name_utf8);
- if (!result) {
- /* parse_uevent_info() failed for at least one field. */
- goto end;
- }
- /* Filter out unhandled devices right away */
- switch (bus_type) {
- case BUS_BLUETOOTH:
- case BUS_I2C:
- case BUS_USB:
- case BUS_SPI:
- break;
- default:
- goto end;
- }
- /* Create the record. */
- root = (struct hid_device_info*) calloc(1, sizeof(struct hid_device_info));
- if (!root)
- goto end;
- cur_dev = root;
- /* Fill out the record */
- cur_dev->next = NULL;
- cur_dev->path = dev_path? strdup(dev_path): NULL;
- /* VID/PID */
- cur_dev->vendor_id = dev_vid;
- cur_dev->product_id = dev_pid;
- /* Serial Number */
- cur_dev->serial_number = utf8_to_wchar_t(serial_number_utf8);
- /* Release Number */
- cur_dev->release_number = 0x0;
- /* Interface Number */
- cur_dev->interface_number = -1;
- switch (bus_type) {
- case BUS_USB:
- /* The device pointed to by raw_dev contains information about
- the hidraw device. In order to get information about the
- USB device, get the parent device with the
- subsystem/devtype pair of "usb"/"usb_device". This will
- be several levels up the tree, but the function will find
- it. */
- usb_dev = udev_device_get_parent_with_subsystem_devtype(
- raw_dev,
- "usb",
- "usb_device");
- /* uhid USB devices
- * Since this is a virtual hid interface, no USB information will
- * be available. */
- if (!usb_dev) {
- /* Manufacturer and Product strings */
- cur_dev->manufacturer_string = wcsdup(L"");
- cur_dev->product_string = utf8_to_wchar_t(product_name_utf8);
- break;
- }
- cur_dev->manufacturer_string = copy_udev_string(usb_dev, "manufacturer");
- cur_dev->product_string = copy_udev_string(usb_dev, "product");
- cur_dev->bus_type = HID_API_BUS_USB;
- str = udev_device_get_sysattr_value(usb_dev, "bcdDevice");
- cur_dev->release_number = (str)? strtol(str, NULL, 16): 0x0;
- /* Get a handle to the interface's udev node. */
- intf_dev = udev_device_get_parent_with_subsystem_devtype(
- raw_dev,
- "usb",
- "usb_interface");
- if (intf_dev) {
- str = udev_device_get_sysattr_value(intf_dev, "bInterfaceNumber");
- cur_dev->interface_number = (str)? strtol(str, NULL, 16): -1;
- }
- break;
- case BUS_BLUETOOTH:
- cur_dev->manufacturer_string = wcsdup(L"");
- cur_dev->product_string = utf8_to_wchar_t(product_name_utf8);
- cur_dev->bus_type = HID_API_BUS_BLUETOOTH;
- break;
- case BUS_I2C:
- cur_dev->manufacturer_string = wcsdup(L"");
- cur_dev->product_string = utf8_to_wchar_t(product_name_utf8);
- cur_dev->bus_type = HID_API_BUS_I2C;
- break;
- case BUS_SPI:
- cur_dev->manufacturer_string = wcsdup(L"");
- cur_dev->product_string = utf8_to_wchar_t(product_name_utf8);
- cur_dev->bus_type = HID_API_BUS_SPI;
- break;
- default:
- /* Unknown device type - this should never happen, as we
- * check for USB and Bluetooth devices above */
- break;
- }
- /* Usage Page and Usage */
- result = get_hid_report_descriptor_from_sysfs(sysfs_path, &report_desc);
- if (result >= 0) {
- unsigned short page = 0, usage = 0;
- unsigned int pos = 0;
- /*
- * Parse the first usage and usage page
- * out of the report descriptor.
- */
- if (!get_next_hid_usage(report_desc.value, report_desc.size, &pos, &page, &usage)) {
- cur_dev->usage_page = page;
- cur_dev->usage = usage;
- }
- /*
- * Parse any additional usage and usage pages
- * out of the report descriptor.
- */
- while (!get_next_hid_usage(report_desc.value, report_desc.size, &pos, &page, &usage)) {
- /* Create new record for additional usage pairs */
- struct hid_device_info *tmp = (struct hid_device_info*) calloc(1, sizeof(struct hid_device_info));
- struct hid_device_info *prev_dev = cur_dev;
- if (!tmp)
- continue;
- cur_dev->next = tmp;
- cur_dev = tmp;
- /* Update fields */
- cur_dev->path = dev_path? strdup(dev_path): NULL;
- cur_dev->vendor_id = dev_vid;
- cur_dev->product_id = dev_pid;
- cur_dev->serial_number = prev_dev->serial_number? wcsdup(prev_dev->serial_number): NULL;
- cur_dev->release_number = prev_dev->release_number;
- cur_dev->interface_number = prev_dev->interface_number;
- cur_dev->manufacturer_string = prev_dev->manufacturer_string? wcsdup(prev_dev->manufacturer_string): NULL;
- cur_dev->product_string = prev_dev->product_string? wcsdup(prev_dev->product_string): NULL;
- cur_dev->usage_page = page;
- cur_dev->usage = usage;
- cur_dev->bus_type = prev_dev->bus_type;
- }
- }
- end:
- free(serial_number_utf8);
- free(product_name_utf8);
- return root;
- }
- static struct hid_device_info * create_device_info_for_hid_device(hid_device *dev) {
- struct udev *udev;
- struct udev_device *udev_dev;
- struct stat s;
- int ret = -1;
- struct hid_device_info *root = NULL;
- register_device_error(dev, NULL);
- /* Get the dev_t (major/minor numbers) from the file handle. */
- ret = fstat(dev->device_handle, &s);
- if (-1 == ret) {
- register_device_error(dev, "Failed to stat device handle");
- return NULL;
- }
- /* Create the udev object */
- udev = udev_new();
- if (!udev) {
- register_device_error(dev, "Couldn't create udev context");
- return NULL;
- }
- /* Open a udev device from the dev_t. 'c' means character device. */
- udev_dev = udev_device_new_from_devnum(udev, 'c', s.st_rdev);
- if (udev_dev) {
- root = create_device_info_for_device(udev_dev);
- }
- if (!root) {
- /* TODO: have a better error reporting via create_device_info_for_device */
- register_device_error(dev, "Couldn't create hid_device_info");
- }
- udev_device_unref(udev_dev);
- udev_unref(udev);
- return root;
- }
- HID_API_EXPORT const struct hid_api_version* HID_API_CALL hid_version(void)
- {
- return &api_version;
- }
- HID_API_EXPORT const char* HID_API_CALL hid_version_str(void)
- {
- return HID_API_VERSION_STR;
- }
- int HID_API_EXPORT hid_init(void)
- {
- const char *locale;
- /* indicate no error */
- register_global_error(NULL);
- /* Set the locale if it's not set. */
- locale = setlocale(LC_CTYPE, NULL);
- if (!locale)
- setlocale(LC_CTYPE, "");
- return 0;
- }
- int HID_API_EXPORT hid_exit(void)
- {
- /* Free global error message */
- register_global_error(NULL);
- return 0;
- }
- struct hid_device_info HID_API_EXPORT *hid_enumerate(unsigned short vendor_id, unsigned short product_id)
- {
- struct udev *udev;
- struct udev_enumerate *enumerate;
- struct udev_list_entry *devices, *dev_list_entry;
- struct hid_device_info *root = NULL; /* return object */
- struct hid_device_info *cur_dev = NULL;
- hid_init();
- /* register_global_error: global error is reset by hid_init */
- /* Create the udev object */
- udev = udev_new();
- if (!udev) {
- register_global_error("Couldn't create udev context");
- return NULL;
- }
- /* Create a list of the devices in the 'hidraw' subsystem. */
- enumerate = udev_enumerate_new(udev);
- udev_enumerate_add_match_subsystem(enumerate, "hidraw");
- udev_enumerate_scan_devices(enumerate);
- devices = udev_enumerate_get_list_entry(enumerate);
- /* For each item, see if it matches the vid/pid, and if so
- create a udev_device record for it */
- udev_list_entry_foreach(dev_list_entry, devices) {
- const char *sysfs_path;
- unsigned short dev_vid = 0;
- unsigned short dev_pid = 0;
- unsigned bus_type = 0;
- struct udev_device *raw_dev; /* The device's hidraw udev node. */
- struct hid_device_info * tmp;
- /* Get the filename of the /sys entry for the device
- and create a udev_device object (dev) representing it */
- sysfs_path = udev_list_entry_get_name(dev_list_entry);
- if (!sysfs_path)
- continue;
- if (vendor_id != 0 || product_id != 0) {
- if (!parse_hid_vid_pid_from_sysfs(sysfs_path, &bus_type, &dev_vid, &dev_pid))
- continue;
- if (vendor_id != 0 && vendor_id != dev_vid)
- continue;
- if (product_id != 0 && product_id != dev_pid)
- continue;
- }
- #ifdef HIDAPI_IGNORE_DEVICE
- /* See if there are any devices we should skip in enumeration */
- if (!parse_hid_vid_pid_from_sysfs(sysfs_path, &bus_type, &dev_vid, &dev_pid))
- continue;
- if (HIDAPI_IGNORE_DEVICE(dev_vid, dev_pid)) {
- continue;
- }
- #endif
- raw_dev = udev_device_new_from_syspath(udev, sysfs_path);
- if (!raw_dev)
- continue;
- tmp = create_device_info_for_device(raw_dev);
- if (tmp) {
- if (cur_dev) {
- cur_dev->next = tmp;
- }
- else {
- root = tmp;
- }
- cur_dev = tmp;
- /* move the pointer to the tail of returnd list */
- while (cur_dev->next != NULL) {
- cur_dev = cur_dev->next;
- }
- }
- udev_device_unref(raw_dev);
- }
- /* Free the enumerator and udev objects. */
- udev_enumerate_unref(enumerate);
- udev_unref(udev);
- if (root == NULL) {
- if (vendor_id == 0 && product_id == 0) {
- register_global_error("No HID devices found in the system.");
- } else {
- register_global_error("No HID devices with requested VID/PID found in the system.");
- }
- }
- return root;
- }
- void HID_API_EXPORT hid_free_enumeration(struct hid_device_info *devs)
- {
- struct hid_device_info *d = devs;
- while (d) {
- struct hid_device_info *next = d->next;
- free(d->path);
- free(d->serial_number);
- free(d->manufacturer_string);
- free(d->product_string);
- free(d);
- d = next;
- }
- }
- hid_device * hid_open(unsigned short vendor_id, unsigned short product_id, const wchar_t *serial_number)
- {
- struct hid_device_info *devs, *cur_dev;
- const char *path_to_open = NULL;
- hid_device *handle = NULL;
- /* register_global_error: global error is reset by hid_enumerate/hid_init */
- devs = hid_enumerate(vendor_id, product_id);
- if (devs == NULL) {
- /* register_global_error: global error is already set by hid_enumerate */
- return NULL;
- }
- cur_dev = devs;
- while (cur_dev) {
- if (cur_dev->vendor_id == vendor_id &&
- cur_dev->product_id == product_id) {
- if (serial_number) {
- if (wcscmp(serial_number, cur_dev->serial_number) == 0) {
- path_to_open = cur_dev->path;
- break;
- }
- }
- else {
- path_to_open = cur_dev->path;
- break;
- }
- }
- cur_dev = cur_dev->next;
- }
- if (path_to_open) {
- /* Open the device */
- handle = hid_open_path(path_to_open);
- } else {
- register_global_error("Device with requested VID/PID/(SerialNumber) not found");
- }
- hid_free_enumeration(devs);
- return handle;
- }
- hid_device * HID_API_EXPORT hid_open_path(const char *path)
- {
- hid_device *dev = NULL;
- hid_init();
- /* register_global_error: global error is reset by hid_init */
- dev = new_hid_device();
- if (!dev) {
- register_global_error("Couldn't allocate memory");
- return NULL;
- }
- dev->device_handle = open(path, O_RDWR | O_CLOEXEC);
- if (dev->device_handle >= 0) {
- int res, desc_size = 0;
- /* Make sure this is a HIDRAW device - responds to HIDIOCGRDESCSIZE */
- res = ioctl(dev->device_handle, HIDIOCGRDESCSIZE, &desc_size);
- if (res < 0) {
- hid_close(dev);
- register_global_error_format("ioctl(GRDESCSIZE) error for '%s', not a HIDRAW device?: %s", path, strerror(errno));
- return NULL;
- }
- dev->needs_ble_hack = (is_BLE(dev) == 1);
- return dev;
- }
- else {
- /* Unable to open a device. */
- free(dev);
- register_global_error_format("Failed to open a device with path '%s': %s", path, strerror(errno));
- return NULL;
- }
- }
- int HID_API_EXPORT hid_write(hid_device *dev, const unsigned char *data, size_t length)
- {
- int bytes_written;
- if (!data || (length == 0)) {
- errno = EINVAL;
- register_device_error(dev, strerror(errno));
- return -1;
- }
- bytes_written = write(dev->device_handle, data, length);
- register_device_error(dev, (bytes_written == -1)? strerror(errno): NULL);
- return bytes_written;
- }
- int HID_API_EXPORT hid_read_timeout(hid_device *dev, unsigned char *data, size_t length, int milliseconds)
- {
- /* Set device error to none */
- register_device_error(dev, NULL);
- int bytes_read;
- if (milliseconds >= 0) {
- /* Milliseconds is either 0 (non-blocking) or > 0 (contains
- a valid timeout). In both cases we want to call poll()
- and wait for data to arrive. Don't rely on non-blocking
- operation (O_NONBLOCK) since some kernels don't seem to
- properly report device disconnection through read() when
- in non-blocking mode. */
- int ret;
- struct pollfd fds;
- fds.fd = dev->device_handle;
- fds.events = POLLIN;
- fds.revents = 0;
- ret = poll(&fds, 1, milliseconds);
- if (ret == 0) {
- /* Timeout */
- return ret;
- }
- if (ret == -1) {
- /* Error */
- register_device_error(dev, strerror(errno));
- return ret;
- }
- else {
- /* Check for errors on the file descriptor. This will
- indicate a device disconnection. */
- if (fds.revents & (POLLERR | POLLHUP | POLLNVAL)) {
- // We cannot use strerror() here as no -1 was returned from poll().
- register_device_error(dev, "hid_read_timeout: unexpected poll error (device disconnected)");
- return -1;
- }
- }
- }
- bytes_read = read(dev->device_handle, data, length);
- if (bytes_read < 0) {
- if (errno == EAGAIN || errno == EINPROGRESS)
- bytes_read = 0;
- else
- register_device_error(dev, strerror(errno));
- }
- return bytes_read;
- }
- int HID_API_EXPORT hid_read(hid_device *dev, unsigned char *data, size_t length)
- {
- return hid_read_timeout(dev, data, length, (dev->blocking)? -1: 0);
- }
- int HID_API_EXPORT hid_set_nonblocking(hid_device *dev, int nonblock)
- {
- /* Do all non-blocking in userspace using poll(), since it looks
- like there's a bug in the kernel in some versions where
- read() will not return -1 on disconnection of the USB device */
- dev->blocking = !nonblock;
- return 0; /* Success */
- }
- int HID_API_EXPORT hid_send_feature_report(hid_device *dev, const unsigned char *data, size_t length)
- {
- static const int MAX_RETRIES = 50;
- int retry;
- int res;
- register_device_error(dev, NULL);
- for (retry = 0; retry < MAX_RETRIES; ++retry) {
- res = ioctl(dev->device_handle, HIDIOCSFEATURE(length), data);
- if (res < 0 && errno == EPIPE) {
- /* Try again... */
- continue;
- }
- if (res < 0)
- register_device_error_format(dev, "ioctl (SFEATURE): %s", strerror(errno));
- break;
- }
- return res;
- }
- int HID_API_EXPORT hid_get_feature_report(hid_device *dev, unsigned char *data, size_t length)
- {
- int res;
- unsigned char report = data[0];
- register_device_error(dev, NULL);
- res = ioctl(dev->device_handle, HIDIOCGFEATURE(length), data);
- if (res < 0)
- register_device_error_format(dev, "ioctl (GFEATURE): %s", strerror(errno));
- else if (dev->needs_ble_hack) {
- /* Versions of BlueZ before 5.56 don't include the report in the data,
- * and versions of BlueZ >= 5.56 include 2 copies of the report.
- * We'll fix it so that there is a single copy of the report in both cases
- */
- if (data[0] == report && data[1] == report) {
- memmove(&data[0], &data[1], res);
- } else if (data[0] != report) {
- memmove(&data[1], &data[0], res);
- data[0] = report;
- ++res;
- }
- }
- return res;
- }
- int HID_API_EXPORT HID_API_CALL hid_get_input_report(hid_device *dev, unsigned char *data, size_t length)
- {
- int res;
- register_device_error(dev, NULL);
- res = ioctl(dev->device_handle, HIDIOCGINPUT(length), data);
- if (res < 0)
- register_device_error_format(dev, "ioctl (GINPUT): %s", strerror(errno));
- return res;
- }
- void HID_API_EXPORT hid_close(hid_device *dev)
- {
- if (!dev)
- return;
- close(dev->device_handle);
- /* Free the device error message */
- register_device_error(dev, NULL);
- hid_free_enumeration(dev->device_info);
- free(dev);
- }
- int HID_API_EXPORT_CALL hid_get_manufacturer_string(hid_device *dev, wchar_t *string, size_t maxlen)
- {
- if (!string || !maxlen) {
- register_device_error(dev, "Zero buffer/length");
- return -1;
- }
- struct hid_device_info *info = hid_get_device_info(dev);
- if (!info) {
- // hid_get_device_info will have set an error already
- return -1;
- }
- if (info->manufacturer_string) {
- wcsncpy(string, info->manufacturer_string, maxlen);
- string[maxlen - 1] = L'\0';
- }
- else {
- string[0] = L'\0';
- }
- return 0;
- }
- int HID_API_EXPORT_CALL hid_get_product_string(hid_device *dev, wchar_t *string, size_t maxlen)
- {
- if (!string || !maxlen) {
- register_device_error(dev, "Zero buffer/length");
- return -1;
- }
- struct hid_device_info *info = hid_get_device_info(dev);
- if (!info) {
- // hid_get_device_info will have set an error already
- return -1;
- }
- if (info->product_string) {
- wcsncpy(string, info->product_string, maxlen);
- string[maxlen - 1] = L'\0';
- }
- else {
- string[0] = L'\0';
- }
- return 0;
- }
- int HID_API_EXPORT_CALL hid_get_serial_number_string(hid_device *dev, wchar_t *string, size_t maxlen)
- {
- if (!string || !maxlen) {
- register_device_error(dev, "Zero buffer/length");
- return -1;
- }
- struct hid_device_info *info = hid_get_device_info(dev);
- if (!info) {
- // hid_get_device_info will have set an error already
- return -1;
- }
- if (info->serial_number) {
- wcsncpy(string, info->serial_number, maxlen);
- string[maxlen - 1] = L'\0';
- }
- else {
- string[0] = L'\0';
- }
- return 0;
- }
- HID_API_EXPORT struct hid_device_info *HID_API_CALL hid_get_device_info(hid_device *dev) {
- if (!dev->device_info) {
- // Lazy initialize device_info
- dev->device_info = create_device_info_for_hid_device(dev);
- }
- // create_device_info_for_hid_device will set an error if needed
- return dev->device_info;
- }
- int HID_API_EXPORT_CALL hid_get_indexed_string(hid_device *dev, int string_index, wchar_t *string, size_t maxlen)
- {
- (void)string_index;
- (void)string;
- (void)maxlen;
- register_device_error(dev, "hid_get_indexed_string: not supported by hidraw");
- return -1;
- }
- int HID_API_EXPORT_CALL hid_get_report_descriptor(hid_device *dev, unsigned char *buf, size_t buf_size)
- {
- struct hidraw_report_descriptor rpt_desc;
- int res = get_hid_report_descriptor_from_hidraw(dev, &rpt_desc);
- if (res < 0) {
- /* error already registered */
- return res;
- }
- if (rpt_desc.size < buf_size) {
- buf_size = (size_t) rpt_desc.size;
- }
- memcpy(buf, rpt_desc.value, buf_size);
- return (int) buf_size;
- }
- /* Passing in NULL means asking for the last global error message. */
- HID_API_EXPORT const wchar_t * HID_API_CALL hid_error(hid_device *dev)
- {
- if (dev) {
- if (dev->last_error_str == NULL)
- return L"Success";
- return dev->last_error_str;
- }
- if (last_global_error_str == NULL)
- return L"Success";
- return last_global_error_str;
- }
|