2014-05-26 16:47:47 +02:00
|
|
|
/*
|
|
|
|
* Implementation of DDEkit related calls/data
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <string.h> /* memset */
|
|
|
|
|
|
|
|
#include <ddekit/usb.h>
|
|
|
|
|
|
|
|
#include <usb/hcd_ddekit.h>
|
2014-05-26 16:47:56 +02:00
|
|
|
#include <usb/hcd_interface.h>
|
2014-05-26 16:47:47 +02:00
|
|
|
#include <usb/usb_common.h>
|
|
|
|
|
|
|
|
|
|
|
|
/*===========================================================================*
|
|
|
|
* Local declarations *
|
|
|
|
*===========================================================================*/
|
|
|
|
/*
|
|
|
|
* In this file "struct ddekit_usb_dev" equals "hcd_device_state"
|
|
|
|
* */
|
|
|
|
struct ddekit_usb_device_id;
|
|
|
|
struct ddekit_usb_urb;
|
|
|
|
struct ddekit_usb_dev;
|
|
|
|
|
2014-06-13 12:50:13 +02:00
|
|
|
/* Translates DDEKit UBR to one used by HCD */
|
|
|
|
static void hcd_decode_urb(hcd_urb *, struct ddekit_usb_urb *);
|
|
|
|
static void hcd_encode_urb(hcd_urb *, struct ddekit_usb_urb *);
|
|
|
|
|
|
|
|
|
2014-05-26 16:47:47 +02:00
|
|
|
/*===========================================================================*
|
|
|
|
* Global definitions *
|
|
|
|
*===========================================================================*/
|
|
|
|
ddekit_usb_completion_cb completion_cb = NULL;
|
|
|
|
ddekit_usb_connect_cb connect_cb = NULL;
|
|
|
|
ddekit_usb_disconnect_cb disconnect_cb = NULL;
|
|
|
|
|
|
|
|
|
|
|
|
/*===========================================================================*
|
|
|
|
* Implementation for usb_server.c *
|
|
|
|
*===========================================================================*/
|
|
|
|
|
|
|
|
/*===========================================================================*
|
|
|
|
* _ddekit_usb_get_manufacturer *
|
|
|
|
*===========================================================================*/
|
|
|
|
char *
|
|
|
|
_ddekit_usb_get_manufacturer(struct ddekit_usb_dev * ddev)
|
|
|
|
{
|
|
|
|
static const char mfg[] = "UNKNOWN";
|
|
|
|
DEBUG_DUMP;
|
2014-05-26 16:47:56 +02:00
|
|
|
/* TODO: UNUSED for argument won't work */
|
2014-05-26 16:47:47 +02:00
|
|
|
((void)ddev);
|
|
|
|
return (char *)mfg;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*===========================================================================*
|
|
|
|
* _ddekit_usb_get_product *
|
|
|
|
*===========================================================================*/
|
|
|
|
char *
|
|
|
|
_ddekit_usb_get_product(struct ddekit_usb_dev * ddev)
|
|
|
|
{
|
|
|
|
static const char prod[] = "UNKNOWN";
|
|
|
|
DEBUG_DUMP;
|
2014-05-26 16:47:56 +02:00
|
|
|
/* TODO: UNUSED for argument won't work */
|
2014-05-26 16:47:47 +02:00
|
|
|
((void)ddev);
|
|
|
|
return (char *)prod;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*===========================================================================*
|
|
|
|
* _ddekit_usb_get_serial *
|
|
|
|
*===========================================================================*/
|
|
|
|
char *
|
|
|
|
_ddekit_usb_get_serial(struct ddekit_usb_dev * ddev)
|
|
|
|
{
|
|
|
|
static const char serial[] = "UNKNOWN";
|
|
|
|
DEBUG_DUMP;
|
2014-05-26 16:47:56 +02:00
|
|
|
/* TODO: UNUSED for argument won't work */
|
2014-05-26 16:47:47 +02:00
|
|
|
((void)ddev);
|
|
|
|
return (char *)serial;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*===========================================================================*
|
|
|
|
* _ddekit_usb_get_device_desc *
|
|
|
|
*===========================================================================*/
|
|
|
|
struct usb_device_descriptor *
|
|
|
|
_ddekit_usb_get_device_desc(struct ddekit_usb_dev * ddev)
|
|
|
|
{
|
|
|
|
hcd_device_state * dev;
|
|
|
|
|
|
|
|
DEBUG_DUMP;
|
|
|
|
|
|
|
|
dev = (hcd_device_state *)ddev;
|
|
|
|
|
|
|
|
return (struct usb_device_descriptor *)
|
|
|
|
(&(dev->config_tree.descriptor));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*===========================================================================*
|
|
|
|
* _ddekit_usb_get_interface_desc *
|
|
|
|
*===========================================================================*/
|
|
|
|
struct usb_interface_descriptor *
|
|
|
|
_ddekit_usb_get_interface_desc(struct ddekit_usb_dev * ddev, int inum)
|
|
|
|
{
|
|
|
|
hcd_device_state * dev;
|
|
|
|
|
|
|
|
DEBUG_DUMP;
|
|
|
|
|
|
|
|
dev = (hcd_device_state *)ddev;
|
|
|
|
|
|
|
|
return (struct usb_interface_descriptor *)
|
|
|
|
(&(dev->config_tree.interface[inum].descriptor));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*===========================================================================*
|
|
|
|
* Implementation for <ddekit/usb.h> *
|
|
|
|
*===========================================================================*/
|
|
|
|
|
|
|
|
/*===========================================================================*
|
|
|
|
* ddekit_usb_dev_set_data *
|
|
|
|
*===========================================================================*/
|
|
|
|
int
|
|
|
|
ddekit_usb_dev_set_data(struct ddekit_usb_dev * dev, void * data)
|
|
|
|
{
|
|
|
|
hcd_device_state * hcd_dev;
|
|
|
|
|
|
|
|
DEBUG_DUMP;
|
|
|
|
|
|
|
|
hcd_dev = (hcd_device_state *)dev;
|
|
|
|
|
|
|
|
hcd_dev->data = data;
|
|
|
|
|
|
|
|
return EXIT_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*===========================================================================*
|
|
|
|
* ddekit_usb_dev_get_data *
|
|
|
|
*===========================================================================*/
|
|
|
|
void *
|
|
|
|
ddekit_usb_dev_get_data(struct ddekit_usb_dev * dev)
|
|
|
|
{
|
|
|
|
hcd_device_state * hcd_dev;
|
|
|
|
|
|
|
|
DEBUG_DUMP;
|
|
|
|
|
|
|
|
hcd_dev = (hcd_device_state *)dev;
|
|
|
|
|
|
|
|
return hcd_dev->data;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-06-13 12:50:13 +02:00
|
|
|
/* TODO: This was in 'ddekit/usb.h' header file, but is not used anywhere */
|
2014-05-26 16:47:47 +02:00
|
|
|
#if 0
|
|
|
|
/*===========================================================================*
|
|
|
|
* ddekit_usb_get_device_id *
|
|
|
|
*===========================================================================*/
|
|
|
|
void
|
|
|
|
ddekit_usb_get_device_id(struct ddekit_usb_dev * dev,
|
|
|
|
struct ddekit_usb_device_id * id)
|
|
|
|
{
|
|
|
|
DEBUG_DUMP;
|
2014-05-26 16:47:56 +02:00
|
|
|
/* TODO: UNUSED for argument won't work */
|
2014-05-26 16:47:47 +02:00
|
|
|
((void)dev);
|
|
|
|
((void)id);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
/*===========================================================================*
|
|
|
|
* ddekit_usb_submit_urb *
|
|
|
|
*===========================================================================*/
|
|
|
|
int
|
|
|
|
ddekit_usb_submit_urb(struct ddekit_usb_urb * d_urb)
|
|
|
|
{
|
|
|
|
hcd_device_state * dev;
|
2014-05-26 16:47:56 +02:00
|
|
|
hcd_driver_state * drv;
|
2014-05-26 16:47:47 +02:00
|
|
|
|
|
|
|
DEBUG_DUMP;
|
|
|
|
|
2014-06-13 12:50:13 +02:00
|
|
|
/* Retrieve info on device/driver state from DDEKit's USB */
|
|
|
|
dev = (hcd_device_state *)(d_urb->dev);
|
2014-05-26 16:47:56 +02:00
|
|
|
drv = (hcd_driver_state *)(dev->driver);
|
2014-05-26 16:47:47 +02:00
|
|
|
|
2014-06-13 12:50:13 +02:00
|
|
|
/* Remember original URB */
|
|
|
|
dev->urb.original_urb = (void *)d_urb;
|
|
|
|
|
|
|
|
/* TODO: URB's should be queued somewhere if DDEKit is not changed */
|
|
|
|
/* Turn DDEKit URB format to one that is easier to handle by HCD, also
|
|
|
|
* check if URB is valid */
|
|
|
|
hcd_decode_urb(&(dev->urb), d_urb);
|
2014-05-26 16:47:56 +02:00
|
|
|
|
2014-06-13 12:50:13 +02:00
|
|
|
/* Start handling URB event */
|
|
|
|
drv->current_event = HCD_EVENT_URB;
|
2014-05-26 16:47:56 +02:00
|
|
|
hcd_handle_event(drv);
|
2014-05-26 16:47:47 +02:00
|
|
|
|
|
|
|
return EXIT_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*===========================================================================*
|
|
|
|
* ddekit_usb_cancle_urb *
|
|
|
|
*===========================================================================*/
|
|
|
|
int
|
|
|
|
ddekit_usb_cancle_urb(struct ddekit_usb_urb * d_urb)
|
|
|
|
{
|
|
|
|
DEBUG_DUMP;
|
2014-05-26 16:47:56 +02:00
|
|
|
/* TODO: UNUSED for argument won't work */
|
2014-05-26 16:47:47 +02:00
|
|
|
((void)d_urb);
|
|
|
|
return EXIT_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*===========================================================================*
|
|
|
|
* ddekit_usb_init *
|
|
|
|
*===========================================================================*/
|
|
|
|
int
|
|
|
|
ddekit_usb_init(struct ddekit_usb_driver * drv,
|
|
|
|
ddekit_usb_malloc_fn * _m,
|
|
|
|
ddekit_usb_free_fn * _f)
|
|
|
|
{
|
|
|
|
DEBUG_DUMP;
|
|
|
|
|
|
|
|
completion_cb = drv->completion;
|
|
|
|
connect_cb = drv->connect;
|
|
|
|
disconnect_cb = drv->disconnect;
|
|
|
|
|
|
|
|
*_m = malloc;
|
|
|
|
*_f = free;
|
|
|
|
|
|
|
|
return EXIT_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*===========================================================================*
|
|
|
|
* hcd_connect_cb *
|
|
|
|
*===========================================================================*/
|
2014-06-13 12:50:13 +02:00
|
|
|
void
|
|
|
|
hcd_connect_cb(hcd_device_state * dev)
|
2014-05-26 16:47:47 +02:00
|
|
|
{
|
|
|
|
unsigned int if_bitmask;
|
|
|
|
|
|
|
|
DEBUG_DUMP;
|
|
|
|
|
2014-06-13 12:50:13 +02:00
|
|
|
/* TODO: Magic numbers like in ddekit/devman */
|
2014-05-26 16:47:47 +02:00
|
|
|
/* Each bit starting from 0, represents valid interface */
|
|
|
|
if_bitmask = 0xFFFFFFFF >> (32 - dev->config_tree.num_interfaces);
|
|
|
|
|
|
|
|
USB_DBG("Interfaces %d, mask %08X",
|
|
|
|
dev->config_tree.num_interfaces,
|
|
|
|
if_bitmask);
|
|
|
|
|
|
|
|
connect_cb((struct ddekit_usb_dev *)dev, (int)if_bitmask);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*===========================================================================*
|
|
|
|
* hcd_disconnect_cb *
|
|
|
|
*===========================================================================*/
|
2014-06-13 12:50:13 +02:00
|
|
|
void
|
|
|
|
hcd_disconnect_cb(hcd_device_state * dev)
|
2014-05-26 16:47:47 +02:00
|
|
|
{
|
|
|
|
DEBUG_DUMP;
|
|
|
|
|
|
|
|
disconnect_cb((struct ddekit_usb_dev *)dev);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*===========================================================================*
|
|
|
|
* hcd_completion_cb *
|
|
|
|
*===========================================================================*/
|
2014-06-13 12:50:13 +02:00
|
|
|
void
|
|
|
|
hcd_completion_cb(hcd_urb * urb)
|
|
|
|
{
|
|
|
|
struct ddekit_usb_urb * d_urb;
|
|
|
|
|
|
|
|
DEBUG_DUMP;
|
|
|
|
|
|
|
|
/* Recollect original URB */
|
|
|
|
d_urb = (struct ddekit_usb_urb *)urb->original_urb;
|
|
|
|
|
|
|
|
/* Turn HCD URB format to one handled by DDEKit */
|
|
|
|
hcd_encode_urb(urb, d_urb);
|
|
|
|
|
|
|
|
completion_cb(d_urb->priv);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*===========================================================================*
|
|
|
|
* hcd_decode_urb *
|
|
|
|
*===========================================================================*/
|
|
|
|
static void
|
|
|
|
hcd_decode_urb(hcd_urb * urb, struct ddekit_usb_urb * dde_urb)
|
|
|
|
{
|
|
|
|
DEBUG_DUMP;
|
|
|
|
|
|
|
|
/* No UBR error initially */
|
|
|
|
urb->inout_status = EXIT_SUCCESS;
|
|
|
|
|
|
|
|
/* Check transfer direction */
|
|
|
|
switch (dde_urb->direction) {
|
|
|
|
case DDEKIT_USB_IN:
|
|
|
|
urb->direction = HCD_DIRECTION_IN;
|
|
|
|
break;
|
|
|
|
case DDEKIT_USB_OUT:
|
|
|
|
urb->direction = HCD_DIRECTION_OUT;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
USB_MSG("URB direction error");
|
|
|
|
goto URB_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Check transfer type */
|
|
|
|
switch (dde_urb->type) {
|
|
|
|
case DDEKIT_USB_TRANSFER_ISO:
|
|
|
|
urb->type = HCD_TRANSFER_ISOCHRONOUS;
|
|
|
|
break;
|
|
|
|
case DDEKIT_USB_TRANSFER_INT:
|
|
|
|
urb->type = HCD_TRANSFER_INTERRUPT;
|
|
|
|
break;
|
|
|
|
case DDEKIT_USB_TRANSFER_CTL:
|
|
|
|
urb->type = HCD_TRANSFER_CONTROL;
|
|
|
|
break;
|
|
|
|
case DDEKIT_USB_TRANSFER_BLK:
|
|
|
|
urb->type = HCD_TRANSFER_BULK;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
USB_MSG("URB type error");
|
|
|
|
goto URB_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Check transfer endpoint validity */
|
|
|
|
if ((dde_urb->endpoint <= (int)HCD_LAST_EP) &&
|
|
|
|
(dde_urb->endpoint >= (int)HCD_DEFAULT_EP))
|
|
|
|
urb->endpoint = (hcd_reg1)dde_urb->endpoint;
|
|
|
|
else {
|
|
|
|
USB_MSG("URB endpoint error");
|
|
|
|
goto URB_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Check transfer interval validity */
|
|
|
|
if ((dde_urb->interval <= (int)HCD_HIGHEST_INTERVAL) &&
|
|
|
|
(dde_urb->interval >= (int)HCD_LOWEST_INTERVAL))
|
|
|
|
urb->interval = (hcd_reg1)dde_urb->interval;
|
|
|
|
else {
|
|
|
|
USB_MSG("URB interval error");
|
|
|
|
goto URB_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* TODO: Alignment of setup packet. Can DDE client guarantee that? */
|
|
|
|
/* Transfer data assignment */
|
|
|
|
urb->inout_data = (void *)dde_urb->data;
|
|
|
|
urb->in_setup = (hcd_ctrlrequest *)dde_urb->setup_packet;
|
|
|
|
|
|
|
|
/* TODO: Sane size check? */
|
|
|
|
urb->in_size = (hcd_reg4)dde_urb->size;
|
|
|
|
|
|
|
|
/* Buffer validity check */
|
|
|
|
if ((NULL == urb->inout_data) && (NULL == urb->in_setup)) {
|
|
|
|
USB_MSG("URB buffer error");
|
|
|
|
goto URB_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Remember device and check for NULL */
|
|
|
|
if (NULL == (urb->target_device = (hcd_device_state *)dde_urb->dev)) {
|
|
|
|
USB_MSG("URB device pointer error");
|
|
|
|
goto URB_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Decoding completed */
|
|
|
|
return;
|
|
|
|
|
|
|
|
URB_ERROR:
|
|
|
|
urb->inout_status = EXIT_FAILURE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*===========================================================================*
|
|
|
|
* hcd_encode_urb *
|
|
|
|
*===========================================================================*/
|
|
|
|
static void
|
|
|
|
hcd_encode_urb(hcd_urb * urb, struct ddekit_usb_urb * dde_urb)
|
2014-05-26 16:47:47 +02:00
|
|
|
{
|
|
|
|
DEBUG_DUMP;
|
|
|
|
|
2014-06-13 12:50:13 +02:00
|
|
|
/* Rewrite output for DDEKit part */
|
|
|
|
dde_urb->actual_length = urb->out_size;
|
|
|
|
dde_urb->status = urb->inout_status;
|
2014-05-26 16:47:47 +02:00
|
|
|
}
|