/* * Written by Oron Peled * Copyright (C) 2012, Xorcom * * All rights reserved. * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. * */ #include #include #include #include #include #include #include #include #include #include #include #include #include "xusb_common.h" #define DBG_MASK 0x01 #define TIMEOUT 500 #define EP_IS_IN(ep) (((ep) & LIBUSB_ENDPOINT_DIR_MASK) == LIBUSB_ENDPOINT_IN) #define EP_IS_OUT(ep) (((ep) & LIBUSB_ENDPOINT_DIR_MASK) == LIBUSB_ENDPOINT_OUT) struct libusb_implementation { struct libusb_device *dev; struct libusb_device_handle *handle; struct libusb_config_descriptor *config_desc; }; void __attribute__((constructor)) xusb_init(void) { int ret; xtalk_parse_options(); ret = libusb_init(NULL); if (ret < 0) { ERR("libusb_init() failed: ret=%d\n", ret); abort(); } } void __attribute__((destructor)) xusb_fini(void) { libusb_exit(NULL); } /* * Translate libusbx error codes to regular errno */ static int errno_map(int libusb_err) { int ret = -libusb_err; #define ERRNO_MAP(libusb, errno) [-(libusb)] = errno static const int error_codes[] = { ERRNO_MAP(LIBUSB_SUCCESS, 0), ERRNO_MAP(LIBUSB_ERROR_IO, EIO), ERRNO_MAP(LIBUSB_ERROR_INVALID_PARAM, EINVAL), ERRNO_MAP(LIBUSB_ERROR_ACCESS, EACCES), ERRNO_MAP(LIBUSB_ERROR_NO_DEVICE, ENODEV), ERRNO_MAP(LIBUSB_ERROR_NOT_FOUND, ENOENT), ERRNO_MAP(LIBUSB_ERROR_BUSY, EBUSY), ERRNO_MAP(LIBUSB_ERROR_TIMEOUT, ETIMEDOUT), ERRNO_MAP(LIBUSB_ERROR_OVERFLOW, EOVERFLOW), ERRNO_MAP(LIBUSB_ERROR_PIPE, EPIPE), ERRNO_MAP(LIBUSB_ERROR_INTERRUPTED, EINTR), ERRNO_MAP(LIBUSB_ERROR_NO_MEM, ENOMSG), ERRNO_MAP(LIBUSB_ERROR_NOT_SUPPORTED, ENOTSUP), ERRNO_MAP(LIBUSB_ERROR_OTHER, EPROTO), }; #undef ERRNO_MAP if (ret < 0 || ret > sizeof(error_codes)/sizeof(error_codes[0])) { ERR("%s: Bad return code %d\n", __func__, -ret); return -EPROTO; } return -(error_codes[ret]); } /* * USB handling */ static int get_usb_string(struct xusb_device *xusb_device, uint8_t item, char *buf) { unsigned char tmp[BUFSIZ]; int ret; if (!xusb_device->impl->handle) { ERR("%s: device closed\n", xusb_device->devpath_tail); return -ENXIO; } if (!item) return 0; ret = libusb_get_string_descriptor_ascii(xusb_device->impl->handle, item, tmp, BUFSIZ); if (ret <= 0) return errno_map(ret); return snprintf(buf, BUFSIZ, "%s", tmp); } static const struct libusb_interface_descriptor *get_iface_descriptor( const struct xusb_device *xusb_device, int i) { const struct libusb_config_descriptor *config_desc; const struct libusb_interface *interface; const struct libusb_interface_descriptor *iface_desc; assert(xusb_device); assert(xusb_device->impl); config_desc = xusb_device->impl->config_desc; assert(config_desc); assert(config_desc->bNumInterfaces < XUSB_MAX_INTERFACES); if (i >= XUSB_MAX_INTERFACES) return NULL; interface = &config_desc->interface[i]; iface_desc = interface->altsetting; return iface_desc; } #define GET_USB_STRING(xusb_device, from, item) \ get_usb_string((xusb_device), (from)->item, (xusb_device)->item) static int xusb_fill_strings(struct xusb_device *xusb_device, int interface_num) { struct libusb_device_descriptor dev_desc; const struct libusb_interface_descriptor *iface_desc; struct xusb_iface *iface = xusb_device->interfaces[interface_num]; int ret; assert(xusb_device); assert(xusb_device->impl); ret = libusb_get_device_descriptor(xusb_device->impl->dev, &dev_desc); if (ret) { XUSB_ERR(iface, "libusb_get_device_descriptor() failed: %s\n", libusb_error_name(ret)); return errno_map(ret); } ret = GET_USB_STRING(xusb_device, &dev_desc, iManufacturer); if (ret < 0) { XUSB_ERR(iface, "Failed reading iManufacturer string: %s\n", libusb_error_name(ret)); return 0; } ret = GET_USB_STRING(xusb_device, &dev_desc, iProduct); if (ret < 0) { XUSB_ERR(iface, "Failed reading iProduct string: %s\n", libusb_error_name(ret)); return 0; } ret = GET_USB_STRING(xusb_device, &dev_desc, iSerialNumber); if (ret < 0) { XUSB_ERR(iface, "Failed reading iSerialNumber string: %s\n", libusb_error_name(ret)); return 0; } iface_desc = get_iface_descriptor(xusb_device, interface_num); if (!iface_desc) { XUSB_ERR(iface, "Could not get interface descriptor of device\n"); return 0; } ret = get_usb_string(xusb_device, iface_desc->iInterface, iface->iInterface); if (ret < 0) { XUSB_ERR(iface, "Failed reading iInterface string: %s\n", libusb_error_name(ret)); return 0; } return 1; } static int xusb_open(struct xusb_device *xusb_device) { int ret = 0; DBG("%s\n", xusb_device->devpath_tail); if (xusb_device->impl->handle) { ERR("%s: already open\n", xusb_device->devpath_tail); ret = -EBUSY; goto out; } ret = libusb_open(xusb_device->impl->dev, &(xusb_device->impl->handle)); if (ret < 0) { ERR("%s: Failed to open usb device: %s\n", xusb_device->devpath_tail, libusb_error_name(ret)); ret = errno_map(ret); xusb_device->impl->handle = NULL; goto out; } out: return ret; } void xusb_release(struct xusb_iface *iface) { if (iface && iface->is_claimed) { struct libusb_device_handle *handle; assert(iface->xusb_device); handle = iface->xusb_device->impl->handle; XUSB_DBG(iface, "Releasing interface\n"); if (!handle) { XUSB_ERR(iface, "device closed\n"); iface->is_claimed = 0; return; } int ret = libusb_release_interface(handle, iface->interface_num); if (ret < 0) XUSB_ERR(iface, "Releasing interface: %s\n", libusb_error_name(ret)); iface->is_claimed = 0; } } static int xusb_clear_halt(struct xusb_iface *xusb_iface) { struct xusb_device *xusb_device; int ret = 0; int ep; xusb_device = xusb_iface->xusb_device; /* * WE DO NOT CALL HALT for problematic devices: * - It cause problem with our usb-dongle (cypress CY7C63803, interrupt driven) */ if (xusb_device->idVendor == 0xe4e4 && xusb_device->idProduct == 0x11a3) { XUSB_DBG(xusb_iface, "Skipping clear_halt()\n"); goto out; } if (!xtalk_option_use_clear_halt()) { XUSB_DBG(xusb_iface, "Don't use clear_halt()\n"); goto out; } ep = EP_OUT(xusb_iface); ret = libusb_clear_halt(xusb_device->impl->handle, ep); if (ret < 0) { XUSB_ERR(xusb_iface, "Clearing output endpoint 0x%02X: %s\n", ep, libusb_error_name(ret)); ret = errno_map(ret); goto out; } ep = EP_IN(xusb_iface); ret = libusb_clear_halt(xusb_device->impl->handle, ep); if (ret < 0) { XUSB_ERR(xusb_iface, "Clearing input endpoint 0x%02X: %s\n", ep, libusb_error_name(ret)); ret = errno_map(ret); goto out; } out: return ret; } int xusb_claim(struct xusb_device *xusb_device, unsigned int interface_num, struct xusb_iface **xusb_iface) { struct xusb_iface *iface = NULL; enum xusb_transfer_type iface_tt = XUSB_TT_ILLEGAL; int ret = 0; *xusb_iface = NULL; assert(xusb_device); if (!xusb_device->impl->handle) { ERR("%s: device closed\n", xusb_device->devpath_tail); return -ENXIO; } if (interface_num >= XUSB_MAX_INTERFACES) { ERR("%s: interface number %d is too big\n", xusb_device->devpath_tail, interface_num); ret = -EINVAL; goto failed; } iface = xusb_device->interfaces[interface_num]; if (!iface) { ERR("%s: No interface number %d\n", xusb_device->devpath_tail, interface_num); ret = -EINVAL; goto failed; } if (iface->is_claimed) { XUSB_ERR(iface, "Already claimed\n"); ret = -EBUSY; goto failed; } ret = libusb_claim_interface(xusb_device->impl->handle, iface->interface_num); if (ret < 0) { XUSB_ERR(iface, "libusb_claim_interface: %s\n", libusb_error_name(ret)); ret = errno_map(ret); goto failed; } iface->is_claimed = 1; iface_tt = xusb_transfer_type(iface); if (iface_tt == XUSB_TT_ILLEGAL) { ret = -ENOTSUP; goto failed; } iface->transfer_type = iface_tt; xusb_fill_strings(xusb_device, interface_num); XUSB_DBG(iface, "ID=%04X:%04X Manufacturer=[%s] Product=[%s] " "SerialNumber=[%s] Interface=[%s] TT=%s\n", xusb_device->idVendor, xusb_device->idProduct, xusb_device->iManufacturer, xusb_device->iProduct, xusb_device->iSerialNumber, iface->iInterface, xusb_tt_name(iface->transfer_type)); ret = xusb_clear_halt(iface); if (ret < 0) goto failed; ret = xusb_flushread(iface); if (ret < 0) { XUSB_ERR(iface, "xusb_flushread failed: %d\n", ret); goto failed; } *xusb_iface = iface; return 0; failed: if (iface) xusb_release(iface); return ret; } void xusb_destroy(struct xusb_device *xusb_device) { if (xusb_device) { struct xusb_iface **piface; struct libusb_implementation *impl; for (piface = xusb_device->interfaces; *piface; piface++) { xusb_destroy_interface(*piface); *piface = NULL; } impl = xusb_device->impl; if (impl) { if (impl->handle) { xusb_close(xusb_device); impl->handle = NULL; } if (impl->config_desc) libusb_free_config_descriptor(impl->config_desc); } DBG("%s: MEM: FREE device\n", xusb_device->devpath_tail); memset(xusb_device, 0, sizeof(*xusb_device)); free(xusb_device); xusb_device = NULL; } } static int init_interfaces(struct xusb_device *xusb_device) { const struct libusb_config_descriptor *config_desc; const struct libusb_interface_descriptor *iface_desc; struct xusb_iface *iface; int max_packet_size = 0; int packet_size; int if_idx; assert(xusb_device); assert(xusb_device->impl); config_desc = xusb_device->impl->config_desc; assert(config_desc->bNumInterfaces < XUSB_MAX_INTERFACES); for (if_idx = 0; if_idx < config_desc->bNumInterfaces; if_idx++) { int ep_idx; iface_desc = get_iface_descriptor(xusb_device, if_idx); if (iface_desc->bInterfaceNumber != if_idx) { ERR("%s: interface %d is number %d\n", xusb_device->devpath_tail, if_idx, iface_desc->bInterfaceNumber); return -EINVAL; } if (iface_desc->bNumEndpoints != 2) { ERR("%s: interface %d has %d endpoints\n", xusb_device->devpath_tail, if_idx, iface_desc->bNumEndpoints); return -EINVAL; } iface = calloc(sizeof(*iface), 1); if (!iface) { ERR("%s: interface %d -- out of memory\n", xusb_device->devpath_tail, if_idx); return -ENOMEM; } DBG("MEM: ALLOC interface: %p\n", iface); xusb_device->interfaces[if_idx] = iface; iface->xusb_device = xusb_device; iface->interface_num = iface_desc->bInterfaceNumber; /* Search Endpoints */ iface->ep_in = 0; iface->ep_out = 0; for (ep_idx = 0; ep_idx < iface_desc->bNumEndpoints; ep_idx++) { int ep_num; ep_num = iface_desc->endpoint[ep_idx].bEndpointAddress; packet_size = libusb_get_max_packet_size(xusb_device->impl->dev, ep_num); if (!max_packet_size || packet_size < max_packet_size) max_packet_size = packet_size; if (EP_IS_OUT(ep_num)) iface->ep_out = ep_num; if (EP_IS_IN(ep_num)) iface->ep_in = ep_num; } /* Validation */ if (!iface->ep_out) { ERR("%s[%d]: Missing output endpoint\n", xusb_device->devpath_tail, if_idx); return -EINVAL; } if (!iface->ep_in) { ERR("%s[%d]: Missing input endpoint\n", xusb_device->devpath_tail, if_idx); return -EINVAL; } iface->is_claimed = 0; XUSB_DBG(iface, "ep_out=0x%X ep_in=0x%X packet_size=%d\n", iface->ep_out, iface->ep_in, max_packet_size); } if (xusb_device->packet_size < max_packet_size) xusb_device->packet_size = max_packet_size; xusb_device->is_usb2 = (xusb_device->packet_size == 512); return 0; } static struct xusb_device *xusb_new(struct libusb_device *dev, const struct xusb_spec *spec) { struct libusb_device_descriptor dev_desc; int ret; struct xusb_device *xusb_device = NULL; xusb_device = calloc(sizeof(*xusb_device) + sizeof(struct libusb_implementation), 1); if (!xusb_device) { ERR("Out of memory"); goto fail; } DBG("MEM: ALLOC device: %p\n", xusb_device); /* Fill xusb_device */ xusb_device->impl = (void *)xusb_device + sizeof(*xusb_device); xusb_device->impl->dev = dev; xusb_device->spec = spec; xusb_device->bus_num = libusb_get_bus_number(dev); xusb_device->device_num = libusb_get_device_address(dev); snprintf(xusb_device->devpath_tail, PATH_MAX, "%03d/%03d", xusb_device->bus_num, xusb_device->device_num); /* * Opening the device */ ret = xusb_open(xusb_device); if (ret < 0) { ERR("%s: Failed to open usb device: %s\n", xusb_device->devpath_tail, strerror(-ret)); goto fail; } /* * Get information from the usb_device */ ret = libusb_get_device_descriptor(dev, &dev_desc); if (ret) { ERR("%s: libusb_get_device_descriptor() failed: %s\n", xusb_device->devpath_tail, libusb_error_name(ret)); goto fail; } xusb_device->idVendor = dev_desc.idVendor; xusb_device->idProduct = dev_desc.idProduct; if (!match_device(xusb_device, spec)) { DBG("[%04X:%04X] did not match\n", xusb_device->idVendor, xusb_device->idProduct); goto fail; } DBG("%s: process [%X:%X]\n", xusb_device->devpath_tail, xusb_device->idVendor, xusb_device->idProduct); ret = libusb_get_config_descriptor(dev, 0, &xusb_device->impl->config_desc); if (ret) { ERR("%s: libusb_get_config_descriptor() failed: %s\n", xusb_device->devpath_tail, libusb_error_name(ret)); goto fail; } ret = init_interfaces(xusb_device); if (ret < 0) { ERR("%s: init_interfaces() failed (ret = %d)\n", xusb_device->devpath_tail, ret); goto fail; } DBG("%s: Created %04X:%04X\n", xusb_device->devpath_tail, xusb_device->idVendor, xusb_device->idProduct); return xusb_device; fail: xusb_destroy(xusb_device); return NULL; } struct xusb_iface *xusb_find_iface(const char *devpath, int iface_num, int ep_out, int ep_in, struct xusb_spec *dummy_spec) { ERR("FIXME: Unimplemented yet\n"); return NULL; } struct xusb_device *xusb_find_bypath(const char *path) { struct xusb_spec *spec; libusb_device **list; ssize_t cnt; int i; DBG("path='%s'\n", path); spec = calloc(sizeof(*spec), 1); if (!spec) { ERR("Failed allocating spec\n"); goto failed; } cnt = libusb_get_device_list(NULL, &list); if (cnt < 0) { ERR("libusb_get_device_list() failed"); goto failed; } for (i = 0; i < cnt; i++) { struct libusb_device_descriptor dev_desc; libusb_device *dev = list[i]; struct xusb_device *xusb_device; char devpath_tail[PATH_MAX]; int bus_num; int device_num; int ret; bus_num = libusb_get_bus_number(dev); device_num = libusb_get_device_address(dev); snprintf(devpath_tail, PATH_MAX, "%03d/%03d", bus_num, device_num); if (!match_devpath(path, devpath_tail)) continue; ret = libusb_get_device_descriptor(dev, &dev_desc); if (ret < 0) { ERR("usb device without a device descriptor\n"); continue; } INFO("Found: %04x:%04x %s\n", dev_desc.idVendor, dev_desc.idProduct, devpath_tail); xusb_init_spec(spec, "", dev_desc.idVendor, dev_desc.idProduct); xusb_device = xusb_new(dev, spec); if (!xusb_device) { ERR("Failed creating xusb for %s\n", devpath_tail); xusb_init_spec(spec, "", 0, 0); continue; } return xusb_device; } failed: if (spec) free(spec); return NULL; } struct xlist_node *xusb_find_byproduct(const struct xusb_spec *specs, int numspecs, xusb_filter_t filterfunc, void *data) { struct xlist_node *xlist; libusb_device **list; ssize_t cnt; int i; DBG("specs(%d)\n", numspecs); xlist = xlist_new(NULL); if (!xlist) { ERR("Failed allocation new xlist"); goto cleanup; } cnt = libusb_get_device_list(NULL, &list); if (cnt < 0) { ERR("libusb_get_device_list() failed"); goto cleanup; } for (i = 0; i < cnt; i++) { struct libusb_device_descriptor dev_desc; libusb_device *dev = list[i]; struct xlist_node *item; int ret; int j; ret = libusb_get_device_descriptor(dev, &dev_desc); if (ret < 0) { ERR("usb device without a device descriptor\n"); continue; } for (j = 0; j < numspecs; j++) { struct xusb_device *xusb_device; const struct xusb_spec *sp = &specs[j]; xusb_device = xusb_new(dev, sp); if (!xusb_device) continue; if (filterfunc && !filterfunc(xusb_device, data)) { DBG("%s: %04X:%04X filtered out\n", xusb_device->devpath_tail, dev_desc.idVendor, dev_desc.idProduct); xusb_destroy(xusb_device); continue; } item = xlist_new(xusb_device); xlist_append_item(xlist, item); break; } } xusb_list_dump(xlist); return xlist; cleanup: if (xlist) xlist_destroy(xlist, NULL); return NULL; } void xusb_showinfo(const struct xusb_device *xusb_device) { struct libusb_device_descriptor dev_desc; struct libusb_device *dev; const struct xusb_iface **piface; int ret; assert(xusb_device); dev = xusb_device->impl->dev; ret = libusb_get_device_descriptor(dev, &dev_desc); if (ret < 0) { ERR("%s: usb device without a device descriptor\n", xusb_device->devpath_tail); return; } if (verbose <= LOG_INFO) { INFO("%s: [%04X:%04X] [%s / %s / %s]\n", xusb_device->devpath_tail, dev_desc.idVendor, dev_desc.idProduct, xusb_device->iManufacturer, xusb_device->iProduct, xusb_device->iSerialNumber); } else { printf("USB Bus/Device: [%03d/%03d] (%s)\n", xusb_device->bus_num, xusb_device->device_num, (xusb_device->impl->handle) ? "open" : "closed"); printf("USB Spec name: [%s]\n", xusb_device->spec->name); printf("USB iManufacturer: [%s]\n", xusb_device->iManufacturer); printf("USB iProduct: [%s]\n", xusb_device->iProduct); printf("USB iSerialNumber: [%s]\n", xusb_device->iSerialNumber); piface = (const struct xusb_iface **)xusb_device->interfaces; for (; *piface; piface++) { printf("USB Interface[%d]: ep_out=0x%02X ep_in=0x%02X claimed=%d [%s]\n", (*piface)->interface_num, (*piface)->ep_out, (*piface)->ep_in, (*piface)->is_claimed, (*piface)->iInterface); } } } int xusb_close(struct xusb_device *xusb_device) { if (xusb_device && xusb_device->impl && xusb_device->impl->handle) { assert(xusb_device->spec); assert(xusb_device->spec->name); DBG("%s: Closing device \"%s\"\n", xusb_device->devpath_tail, xusb_device->spec->name); libusb_close(xusb_device->impl->handle); xusb_device->impl->handle = NULL; } return 0; } enum xusb_transfer_type xusb_transfer_type(const struct xusb_iface *iface) { const struct xusb_device *xusb_device; const struct libusb_interface_descriptor *iface_desc; const struct libusb_endpoint_descriptor *ep; enum xusb_transfer_type ret = XUSB_TT_ILLEGAL; assert(iface); xusb_device = iface->xusb_device; assert(xusb_device); iface_desc = get_iface_descriptor(xusb_device, iface->interface_num); assert(iface_desc); ep = iface_desc->endpoint; assert(ep); switch (ep->bmAttributes) { case LIBUSB_TRANSFER_TYPE_CONTROL: case LIBUSB_TRANSFER_TYPE_ISOCHRONOUS: break; case LIBUSB_TRANSFER_TYPE_BULK: ret = XUSB_TT_BULK; break; case LIBUSB_TRANSFER_TYPE_INTERRUPT: ret = XUSB_TT_INTERRUPT; break; } return ret; } int xusb_send(struct xusb_iface *iface, const char *buf, int len, int timeout) { struct xusb_device *xusb_device = iface->xusb_device; int actual_len; int ep_out = EP_OUT(iface); int ret; if (!xusb_device->impl->handle) { ERR("%s: device closed\n", xusb_device->devpath_tail); return -ENXIO; } dump_packet(LOG_DEBUG, DBG_MASK, __func__, buf, len); if (!EP_IS_OUT(ep_out)) { XUSB_ERR(iface, "%s called with an input endpoint 0x%x\n", __func__, ep_out); return -EINVAL; } switch (iface->transfer_type) { case XUSB_TT_BULK: ret = libusb_bulk_transfer(xusb_device->impl->handle, ep_out, (unsigned char *)buf, len, &actual_len, timeout); break; case XUSB_TT_INTERRUPT: ret = libusb_interrupt_transfer(xusb_device->impl->handle, ep_out, (unsigned char *)buf, len, &actual_len, timeout); break; default: ret = -EAFNOSUPPORT; break; } if (ret) { /* * If the device was gone, it may be the * result of renumeration. Ignore it. */ if (ret != LIBUSB_ERROR_NO_DEVICE) { XUSB_ERR(iface, "write to endpoint 0x%x failed: (%d) %s\n", ep_out, ret, libusb_error_name(ret)); dump_packet(LOG_ERR, DBG_MASK, "xbus_send[ERR]", buf, len); /*exit(2);*/ } else { XUSB_DBG(iface, "write to endpoint 0x%x got ENODEV\n", ep_out); xusb_close(xusb_device); } return errno_map(ret); } else if (actual_len != len) { XUSB_ERR(iface, "write to endpoint 0x%x short write: (%d) %s\n", ep_out, ret, libusb_error_name(ret)); dump_packet(LOG_ERR, DBG_MASK, "xbus_send[ERR]", buf, len); return -EFAULT; } return actual_len; } int xusb_recv(struct xusb_iface *iface, char *buf, size_t len, int timeout) { struct xusb_device *xusb_device = iface->xusb_device; int actual_len; int ep_in = EP_IN(iface); int ret; if (!xusb_device->impl->handle) { ERR("%s: device closed\n", xusb_device->devpath_tail); return -ENXIO; } if (!EP_IS_IN(ep_in)) { XUSB_ERR(iface, "%s called with an output endpoint 0x%x\n", __func__, ep_in); return -EINVAL; } switch (iface->transfer_type) { case XUSB_TT_BULK: ret = libusb_bulk_transfer(xusb_device->impl->handle, ep_in, (unsigned char *)buf, len, &actual_len, timeout); break; case XUSB_TT_INTERRUPT: ret = libusb_interrupt_transfer(xusb_device->impl->handle, ep_in, (unsigned char *)buf, len, &actual_len, timeout); break; default: ret = -EAFNOSUPPORT; break; } if (ret < 0) { XUSB_DBG(iface, "read from endpoint 0x%x failed: (%d) %s\n", ep_in, ret, libusb_error_name(ret)); memset(buf, 0, len); return errno_map(ret); } dump_packet(LOG_DEBUG, DBG_MASK, __func__, buf, actual_len); return actual_len; }