// Copyright 2019, Collabora, Ltd. // SPDX-License-Identifier: BSL-1.0 /*! * @file * @brief Hid implementation based on hidraw. * @author Jakob Bornecrantz * @author Ryan Pavlik * @ingroup aux_os */ #include "os_hid.h" #ifdef XRT_OS_LINUX #include "util/u_misc.h" #include #include #include #include #include #include #include #include struct hid_hidraw { struct os_hid_device base; int fd; }; static int os_hidraw_read(struct os_hid_device *ohdev, uint8_t *data, size_t length, int milliseconds) { struct hid_hidraw *hrdev = (struct hid_hidraw *)ohdev; struct pollfd fds; int ret; if (milliseconds >= 0) { fds.fd = hrdev->fd; fds.events = POLLIN; fds.revents = 0; ret = poll(&fds, 1, milliseconds); if (ret == -1 || ret == 0) { // Error or timeout. return ret; } if (fds.revents & (POLLERR | POLLHUP | POLLNVAL)) { // Device disconnect? return -1; } } ret = read(hrdev->fd, data, length); if (ret < 0 && (errno == EAGAIN || errno == EINPROGRESS)) { // Process most likely received a signal. ret = 0; } return ret; } static int os_hidraw_write(struct os_hid_device *ohdev, const uint8_t *data, size_t length) { struct hid_hidraw *hrdev = (struct hid_hidraw *)ohdev; return write(hrdev->fd, data, length); } static int os_hidraw_get_feature(struct os_hid_device *ohdev, uint8_t report_num, uint8_t *data, size_t length) { struct hid_hidraw *hrdev = (struct hid_hidraw *)ohdev; // The ioctl expects the report number in the first byte of the buffer, // but it will overwrite it. data[0] = report_num; return ioctl(hrdev->fd, HIDIOCGFEATURE(length), data); } static int os_hidraw_set_feature(struct os_hid_device *ohdev, const uint8_t *data, size_t length) { struct hid_hidraw *hrdev = (struct hid_hidraw *)ohdev; return ioctl(hrdev->fd, HIDIOCSFEATURE(length), data); } static void os_hidraw_destroy(struct os_hid_device *ohdev) { struct hid_hidraw *hrdev = (struct hid_hidraw *)ohdev; close(hrdev->fd); free(hrdev); } int os_hid_open_hidraw(const char *path, struct os_hid_device **out_hid) { struct hid_hidraw *hrdev = U_TYPED_CALLOC(struct hid_hidraw); hrdev->base.read = os_hidraw_read; hrdev->base.write = os_hidraw_write; hrdev->base.get_feature = os_hidraw_get_feature; hrdev->base.set_feature = os_hidraw_set_feature; hrdev->base.destroy = os_hidraw_destroy; hrdev->fd = open(path, O_RDWR); if (hrdev->fd < 0) { free(hrdev); return -1; } *out_hid = &hrdev->base; return 0; } #endif