dahdi-tools/xpp/xtalk/xusb_libusbx.c

827 lines
22 KiB
C

/*
* Written by Oron Peled <oron@actcom.co.il>
* 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 <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <assert.h>
#include <errno.h>
#include <stdarg.h>
#include <syslog.h>
#include <arpa/inet.h>
#include <libusb.h>
#include <xtalk/debug.h>
#include <xtalk/xusb.h>
#include <autoconfig.h>
#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;
}
DBG("Found: %04x:%04x %s\n",
dev_desc.idVendor, dev_desc.idProduct, devpath_tail);
xusb_init_spec(spec, "<BYPATH>",
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, "<EMPTY>", 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;
}