844 lines
22 KiB
C
844 lines
22 KiB
C
|
/*
|
||
|
* Written by Oron Peled <oron@actcom.co.il>
|
||
|
* Copyright (C) 2008, 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 <sys/time.h>
|
||
|
#include <sys/types.h>
|
||
|
#include <sys/ipc.h>
|
||
|
#include <sys/sem.h>
|
||
|
#include <usb.h>
|
||
|
#include <xtalk/debug.h>
|
||
|
#include <xtalk/xusb.h>
|
||
|
#include <autoconfig.h>
|
||
|
#include "xusb_common.h"
|
||
|
|
||
|
#define DBG_MASK 0x01
|
||
|
#define TIMEOUT 500
|
||
|
#define MAX_RETRIES 10
|
||
|
|
||
|
#define EP_IS_IN(ep) (((ep) & USB_ENDPOINT_DIR_MASK) == USB_ENDPOINT_IN)
|
||
|
#define EP_IS_OUT(ep) (((ep) & USB_ENDPOINT_DIR_MASK) == USB_ENDPOINT_OUT)
|
||
|
|
||
|
struct libusb_implementation {
|
||
|
struct usb_device *dev;
|
||
|
usb_dev_handle *handle;
|
||
|
struct usb_config_descriptor *config_desc;
|
||
|
};
|
||
|
|
||
|
static void xusb_init();
|
||
|
|
||
|
/*
|
||
|
* USB handling
|
||
|
*/
|
||
|
|
||
|
static int get_usb_string(struct xusb_device *xusb_device, uint8_t item, char *buf)
|
||
|
{
|
||
|
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 = usb_get_string_simple(xusb_device->impl->handle, item, tmp, BUFSIZ);
|
||
|
if (ret <= 0)
|
||
|
return ret;
|
||
|
return snprintf(buf, BUFSIZ, "%s", tmp);
|
||
|
}
|
||
|
|
||
|
static const struct usb_interface_descriptor *get_iface_descriptor(
|
||
|
const struct xusb_device *xusb_device, int i)
|
||
|
{
|
||
|
const struct usb_config_descriptor *config_desc;
|
||
|
const struct usb_interface *interface;
|
||
|
const struct usb_interface_descriptor *iface_desc;
|
||
|
int num_altsetting;
|
||
|
|
||
|
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];
|
||
|
assert(interface != NULL);
|
||
|
iface_desc = interface->altsetting;
|
||
|
num_altsetting = interface->num_altsetting;
|
||
|
assert(num_altsetting != 0);
|
||
|
assert(iface_desc != NULL);
|
||
|
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)
|
||
|
{
|
||
|
const struct usb_device_descriptor *dev_desc;
|
||
|
const struct usb_interface_descriptor *iface_desc;
|
||
|
struct xusb_iface *iface = xusb_device->interfaces[interface_num];
|
||
|
int ret;
|
||
|
|
||
|
dev_desc = &xusb_device->impl->dev->descriptor;
|
||
|
assert(dev_desc);
|
||
|
ret = GET_USB_STRING(xusb_device, dev_desc, iManufacturer);
|
||
|
if (ret < 0) {
|
||
|
XUSB_ERR(iface, "Failed reading iManufacturer string: %s\n",
|
||
|
usb_strerror());
|
||
|
return 0;
|
||
|
}
|
||
|
ret = GET_USB_STRING(xusb_device, dev_desc, iProduct);
|
||
|
if (ret < 0) {
|
||
|
XUSB_ERR(iface, "Failed reading iProduct string: %s\n",
|
||
|
usb_strerror());
|
||
|
return 0;
|
||
|
}
|
||
|
ret = GET_USB_STRING(xusb_device, dev_desc, iSerialNumber);
|
||
|
if (ret < 0) {
|
||
|
XUSB_ERR(iface, "Failed reading iSerialNumber string: %s\n",
|
||
|
usb_strerror());
|
||
|
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);
|
||
|
if (ret < 0) {
|
||
|
XUSB_ERR(iface, "Failed reading iInterface string: %s\n",
|
||
|
usb_strerror());
|
||
|
return 0;
|
||
|
}
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
static int xusb_open(struct xusb_device *xusb_device)
|
||
|
{
|
||
|
int ret = 0;
|
||
|
|
||
|
assert(xusb_device);
|
||
|
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;
|
||
|
}
|
||
|
xusb_device->impl->handle = usb_open(xusb_device->impl->dev);
|
||
|
if (!xusb_device->impl->handle) {
|
||
|
ERR("%s: Failed to open usb device: %s\n",
|
||
|
xusb_device->devpath_tail,
|
||
|
usb_strerror());
|
||
|
xusb_device->impl->handle = NULL;
|
||
|
goto out;
|
||
|
}
|
||
|
return 1;
|
||
|
out:
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
void xusb_release(struct xusb_iface *iface)
|
||
|
{
|
||
|
if (iface && iface->is_claimed) {
|
||
|
usb_dev_handle *handle;
|
||
|
int ret;
|
||
|
|
||
|
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;
|
||
|
}
|
||
|
ret = usb_release_interface(handle, iface->interface_num);
|
||
|
if (ret < 0)
|
||
|
XUSB_ERR(iface, "Releasing interface: %s\n",
|
||
|
usb_strerror());
|
||
|
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 = usb_clear_halt(xusb_device->impl->handle, ep);
|
||
|
if (ret < 0) {
|
||
|
XUSB_ERR(xusb_iface, "Clearing output endpoint 0x%02X: %s\n",
|
||
|
ep, usb_strerror());
|
||
|
goto out;
|
||
|
}
|
||
|
ep = EP_IN(xusb_iface);
|
||
|
ret = usb_clear_halt(xusb_device->impl->handle, ep);
|
||
|
if (ret < 0) {
|
||
|
XUSB_ERR(xusb_iface, "Clearing input endpoint 0x%02X: %s\n",
|
||
|
ep, usb_strerror());
|
||
|
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) {
|
||
|
XUSB_ERR(iface, "device closed\n");
|
||
|
ret = -ENXIO;
|
||
|
goto failed;
|
||
|
}
|
||
|
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 = usb_claim_interface(xusb_device->impl->handle, iface->interface_num);
|
||
|
if (ret < 0) {
|
||
|
XUSB_ERR(iface, "usb_claim_interface: %s\n", usb_strerror());
|
||
|
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;
|
||
|
ret = xusb_clear_halt(iface);
|
||
|
if (ret < 0)
|
||
|
goto failed;
|
||
|
ret = xusb_flushread(iface);
|
||
|
if (ret < 0) {
|
||
|
XUSB_ERR(iface, "xusb_flushread failed: %s\n", usb_strerror());
|
||
|
goto failed;
|
||
|
}
|
||
|
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));
|
||
|
*xusb_iface = iface;
|
||
|
return 0;
|
||
|
failed:
|
||
|
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;
|
||
|
}
|
||
|
}
|
||
|
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 usb_config_descriptor *config_desc;
|
||
|
const struct usb_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);
|
||
|
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 = iface_desc->endpoint[ep_idx].wMaxPacketSize;
|
||
|
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 usb_device *dev,
|
||
|
const struct xusb_spec *spec,
|
||
|
xusb_filter_t filterfunc,
|
||
|
void *data)
|
||
|
{
|
||
|
struct usb_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;
|
||
|
/*
|
||
|
* Get information from the usb_device
|
||
|
*/
|
||
|
dev_desc = &dev->descriptor;
|
||
|
if (!dev_desc) {
|
||
|
ERR("usb device without a device descriptor\n");
|
||
|
goto fail;
|
||
|
}
|
||
|
xusb_device->idVendor = dev_desc->idVendor;
|
||
|
xusb_device->idProduct = dev_desc->idProduct;
|
||
|
sscanf(dev->bus->dirname, "%d", &xusb_device->bus_num);
|
||
|
sscanf(dev->filename, "%d", &xusb_device->device_num);
|
||
|
snprintf(xusb_device->devpath_tail, PATH_MAX, "%03d/%03d",
|
||
|
xusb_device->bus_num, xusb_device->device_num);
|
||
|
if (!match_device(xusb_device, spec)) {
|
||
|
DBG("[%04X:%04X] did not match\n",
|
||
|
xusb_device->idVendor, xusb_device->idProduct);
|
||
|
goto fail;
|
||
|
}
|
||
|
xusb_device->impl->config_desc = dev->config;
|
||
|
if (!xusb_device->impl->config_desc) {
|
||
|
ERR("usb device without a configuration descriptor\n");
|
||
|
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;
|
||
|
}
|
||
|
if (!xusb_open(xusb_device)) {
|
||
|
ERR("%s: Failed opening device: %04X:%04X\n",
|
||
|
xusb_device->devpath_tail,
|
||
|
xusb_device->idVendor,
|
||
|
xusb_device->idProduct);
|
||
|
goto fail;
|
||
|
}
|
||
|
DBG("%s: %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)
|
||
|
{
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
struct xusb_device *xusb_find_bypath(const char *path)
|
||
|
{
|
||
|
struct usb_bus *bus;
|
||
|
struct usb_device *dev;
|
||
|
char devpath_tail[PATH_MAX];
|
||
|
struct xusb_spec *spec;
|
||
|
|
||
|
DBG("path='%s'\n", path);
|
||
|
spec = calloc(sizeof(*spec), 1);
|
||
|
if (!spec) {
|
||
|
ERR("Failed allocating spec\n");
|
||
|
goto failed;
|
||
|
}
|
||
|
xusb_init();
|
||
|
for (bus = usb_get_busses(); bus; bus = bus->next) {
|
||
|
for (dev = bus->devices; dev; dev = dev->next) {
|
||
|
struct usb_device_descriptor *dev_desc;
|
||
|
struct xusb_device *xusb_device = NULL;
|
||
|
|
||
|
dev_desc = &dev->descriptor;
|
||
|
assert(dev_desc);
|
||
|
DBG("usb:%s/%s: ID=%04X:%04X\n",
|
||
|
dev->bus->dirname,
|
||
|
dev->filename,
|
||
|
dev_desc->idVendor,
|
||
|
dev_desc->idProduct);
|
||
|
snprintf(devpath_tail, PATH_MAX, "%3s/%3s",
|
||
|
dev->bus->dirname, dev->filename);
|
||
|
if (!match_devpath(path, devpath_tail))
|
||
|
continue;
|
||
|
DBG("Found: usb:%s/%s: ID=%04X:%04X\n",
|
||
|
dev->bus->dirname,
|
||
|
dev->filename,
|
||
|
dev_desc->idVendor,
|
||
|
dev_desc->idProduct);
|
||
|
xusb_init_spec(spec, "<BYPATH>",
|
||
|
dev_desc->idVendor, dev_desc->idProduct);
|
||
|
xusb_device = xusb_new(dev, spec, NULL, NULL);
|
||
|
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;
|
||
|
struct usb_bus *bus;
|
||
|
struct usb_device *dev;
|
||
|
|
||
|
DBG("specs(%d)\n", numspecs);
|
||
|
xlist = xlist_new(NULL);
|
||
|
if (!xlist) {
|
||
|
ERR("Failed allocation new xlist");
|
||
|
goto failed;
|
||
|
}
|
||
|
xusb_init();
|
||
|
for (bus = usb_get_busses(); bus; bus = bus->next) {
|
||
|
for (dev = bus->devices; dev; dev = dev->next) {
|
||
|
struct usb_device_descriptor *dev_desc;
|
||
|
struct xlist_node *item;
|
||
|
int i;
|
||
|
|
||
|
dev_desc = &dev->descriptor;
|
||
|
assert(dev_desc);
|
||
|
DBG("usb:%s/%s: ID=%04X:%04X\n",
|
||
|
dev->bus->dirname,
|
||
|
dev->filename,
|
||
|
dev_desc->idVendor,
|
||
|
dev_desc->idProduct);
|
||
|
for (i = 0; i < numspecs; i++) {
|
||
|
struct xusb_device *xusb_device;
|
||
|
const struct xusb_spec *sp = &specs[i];
|
||
|
|
||
|
xusb_device = xusb_new(dev, sp, filterfunc, data);
|
||
|
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;
|
||
|
failed:
|
||
|
if (xlist)
|
||
|
xlist_destroy(xlist, NULL);
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
void xusb_showinfo(const struct xusb_device *xusb_device)
|
||
|
{
|
||
|
struct usb_device_descriptor *dev_desc;
|
||
|
struct usb_device *dev;
|
||
|
const struct xusb_iface **piface;
|
||
|
|
||
|
assert(xusb_device);
|
||
|
dev = xusb_device->impl->dev;
|
||
|
assert(dev);
|
||
|
dev_desc = &dev->descriptor;
|
||
|
assert(dev_desc);
|
||
|
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: [%s/%s] (%s)\n",
|
||
|
dev->bus->dirname,
|
||
|
dev->filename,
|
||
|
(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);
|
||
|
usb_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 usb_interface_descriptor *iface_desc;
|
||
|
const struct usb_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 USB_ENDPOINT_TYPE_CONTROL:
|
||
|
case USB_ENDPOINT_TYPE_ISOCHRONOUS:
|
||
|
break;
|
||
|
case USB_ENDPOINT_TYPE_BULK:
|
||
|
ret = XUSB_TT_BULK;
|
||
|
break;
|
||
|
case USB_ENDPOINT_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 ep_out = EP_OUT(iface);
|
||
|
int retries = 0;
|
||
|
int ret;
|
||
|
|
||
|
dump_packet(LOG_DEBUG, DBG_MASK, __func__, buf, len);
|
||
|
if (!xusb_device->impl->handle) {
|
||
|
XUSB_ERR(iface, "device closed\n");
|
||
|
return -ENXIO;
|
||
|
}
|
||
|
if (!EP_IS_OUT(ep_out)) {
|
||
|
XUSB_ERR(iface, "%s called with an input endpoint 0x%x\n",
|
||
|
__func__, ep_out);
|
||
|
return -EINVAL;
|
||
|
}
|
||
|
retry_write:
|
||
|
switch (iface->transfer_type) {
|
||
|
case XUSB_TT_BULK:
|
||
|
ret = usb_bulk_write(xusb_device->impl->handle, ep_out, (char *)buf, len, timeout);
|
||
|
break;
|
||
|
case XUSB_TT_INTERRUPT:
|
||
|
ret = usb_interrupt_write(xusb_device->impl->handle, ep_out, (char *)buf, len, timeout);
|
||
|
break;
|
||
|
default:
|
||
|
ret = -EINVAL;
|
||
|
break;
|
||
|
}
|
||
|
if (ret < 0) {
|
||
|
/*
|
||
|
* If the device was gone, it may be the
|
||
|
* result of renumeration. Ignore it.
|
||
|
*/
|
||
|
if (ret != -ENODEV) {
|
||
|
XUSB_ERR(iface, "write to endpoint 0x%x failed: (%d) %s\n",
|
||
|
ep_out, ret, usb_strerror());
|
||
|
dump_packet(LOG_ERR, DBG_MASK, "xusb_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 ret;
|
||
|
}
|
||
|
if (!ret) {
|
||
|
|
||
|
XUSB_ERR(iface, "write to endpoint 0x%x short write[%d]: (%d)\n",
|
||
|
ep_out, retries, ret);
|
||
|
if (retries++ > MAX_RETRIES)
|
||
|
return -EFAULT;
|
||
|
usleep(100);
|
||
|
goto retry_write;
|
||
|
}
|
||
|
if (ret != len) {
|
||
|
XUSB_ERR(iface, "write to endpoint 0x%x short write: (%d) %s\n",
|
||
|
ep_out, ret, usb_strerror());
|
||
|
dump_packet(LOG_ERR, DBG_MASK, "xusb_send[ERR]", buf, len);
|
||
|
return -EFAULT;
|
||
|
}
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
int xusb_recv(struct xusb_iface *iface, char *buf, size_t len, int timeout)
|
||
|
{
|
||
|
struct xusb_device *xusb_device = iface->xusb_device;
|
||
|
int ep_in = EP_IN(iface);
|
||
|
int retries = 0;
|
||
|
int ret;
|
||
|
|
||
|
if (!xusb_device->impl->handle) {
|
||
|
XUSB_ERR(iface, "device closed\n");
|
||
|
return -ENXIO;
|
||
|
}
|
||
|
if (!EP_IS_IN(ep_in)) {
|
||
|
XUSB_ERR(iface, "called with an output endpoint 0x%x\n", ep_in);
|
||
|
return -EINVAL;
|
||
|
}
|
||
|
retry_read:
|
||
|
switch (iface->transfer_type) {
|
||
|
case XUSB_TT_BULK:
|
||
|
ret = usb_bulk_read(xusb_device->impl->handle, ep_in, buf, len, timeout);
|
||
|
break;
|
||
|
case XUSB_TT_INTERRUPT:
|
||
|
ret = usb_interrupt_read(xusb_device->impl->handle, ep_in, buf, 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, usb_strerror());
|
||
|
memset(buf, 0, len);
|
||
|
return ret;
|
||
|
}
|
||
|
if (!ret) {
|
||
|
XUSB_ERR(iface, "read to endpoint 0x%x short read[%d]: (%d)\n",
|
||
|
ep_in, retries, ret);
|
||
|
if (retries++ > MAX_RETRIES)
|
||
|
return -EFAULT;
|
||
|
usleep(100);
|
||
|
goto retry_read;
|
||
|
}
|
||
|
dump_packet(LOG_DEBUG, DBG_MASK, __func__, buf, ret);
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* Serialize calls to usb_find_busses()/usb_find_devices()
|
||
|
*/
|
||
|
|
||
|
static const key_t SEM_KEY = 0x1a2b3c4d;
|
||
|
static int semid = -1; /* Failure */
|
||
|
|
||
|
static void xusb_lock_usb()
|
||
|
{
|
||
|
struct sembuf sembuf;
|
||
|
|
||
|
while (semid < 0) {
|
||
|
/* Maybe it was already created? */
|
||
|
semid = semget(SEM_KEY, 1, 0);
|
||
|
if (semid < 0) {
|
||
|
/* No, let's create ourselves */
|
||
|
semid = semget(SEM_KEY, 1, IPC_CREAT | IPC_EXCL | 0644);
|
||
|
if (semid < 0) {
|
||
|
/* Someone else won the race to create it */
|
||
|
if (errno != ENOENT)
|
||
|
ERR("%s: semget() failed: %s\n",
|
||
|
__func__, strerror(errno));
|
||
|
/* Retry */
|
||
|
continue;
|
||
|
}
|
||
|
/* Initialize */
|
||
|
if (semctl(semid, 0, SETVAL, 1) < 0)
|
||
|
ERR("%s: SETVAL() failed: %s\n",
|
||
|
__func__, strerror(errno));
|
||
|
}
|
||
|
}
|
||
|
DBG("%d: LOCKING\n", getpid());
|
||
|
sembuf.sem_num = 0;
|
||
|
sembuf.sem_op = -1;
|
||
|
sembuf.sem_flg = SEM_UNDO;
|
||
|
if (semop(semid, &sembuf, 1) < 0)
|
||
|
ERR("%s: semop() failed: %s\n", __func__, strerror(errno));
|
||
|
DBG("%d: LOCKED\n", getpid());
|
||
|
}
|
||
|
|
||
|
static void xusb_unlock_usb()
|
||
|
{
|
||
|
struct sembuf sembuf;
|
||
|
|
||
|
DBG("%d: UNLOCKING\n", getpid());
|
||
|
sembuf.sem_num = 0;
|
||
|
sembuf.sem_op = 1;
|
||
|
sembuf.sem_flg = SEM_UNDO;
|
||
|
if (semop(semid, &sembuf, 1) < 0)
|
||
|
ERR("%s: semop() failed: %s\n", __func__, strerror(errno));
|
||
|
DBG("%d: UNLOCKED\n", getpid());
|
||
|
}
|
||
|
|
||
|
static int initizalized;
|
||
|
|
||
|
void __attribute__((constructor)) xusb_init(void)
|
||
|
{
|
||
|
xtalk_parse_options();
|
||
|
if (!getenv("XUSB_NOLOCK"))
|
||
|
xusb_lock_usb();
|
||
|
if (!initizalized) {
|
||
|
usb_init();
|
||
|
initizalized = 1;
|
||
|
}
|
||
|
usb_find_busses();
|
||
|
usb_find_devices();
|
||
|
if (!getenv("XUSB_NOLOCK"))
|
||
|
xusb_unlock_usb();
|
||
|
}
|