usb: maintain async packet list per endpoint

Maintain a list of async packets per endpoint.  With the current code
the list will never receive more than a single item.  I think you can
guess what the future plan is though ;)

Signed-off-by: Gerd Hoffmann <kraxel@redhat.com>
This commit is contained in:
Gerd Hoffmann 2012-01-12 14:26:13 +01:00
parent 079d0b7f1e
commit db4be873d3
2 changed files with 108 additions and 28 deletions

127
hw/usb.c
View File

@ -279,6 +279,28 @@ USBDevice *usb_find_device(USBPort *port, uint8_t addr)
return usb_device_find_device(dev, addr); return usb_device_find_device(dev, addr);
} }
static int usb_process_one(USBPacket *p)
{
USBDevice *dev = p->ep->dev;
if (p->ep->nr == 0) {
/* control pipe */
switch (p->pid) {
case USB_TOKEN_SETUP:
return do_token_setup(dev, p);
case USB_TOKEN_IN:
return do_token_in(dev, p);
case USB_TOKEN_OUT:
return do_token_out(dev, p);
default:
return USB_RET_STALL;
}
} else {
/* data pipe */
return usb_device_handle_data(dev, p);
}
}
/* Hand over a packet to a device for processing. Return value /* Hand over a packet to a device for processing. Return value
USB_RET_ASYNC indicates the processing isn't finished yet, the USB_RET_ASYNC indicates the processing isn't finished yet, the
driver will call usb_packet_complete() when done processing it. */ driver will call usb_packet_complete() when done processing it. */
@ -292,30 +314,21 @@ int usb_handle_packet(USBDevice *dev, USBPacket *p)
assert(dev == p->ep->dev); assert(dev == p->ep->dev);
assert(dev->state == USB_STATE_DEFAULT); assert(dev->state == USB_STATE_DEFAULT);
assert(p->state == USB_PACKET_SETUP); assert(p->state == USB_PACKET_SETUP);
assert(p->ep != NULL);
if (p->ep->nr == 0) { if (QTAILQ_EMPTY(&p->ep->queue)) {
/* control pipe */ ret = usb_process_one(p);
switch (p->pid) { if (ret == USB_RET_ASYNC) {
case USB_TOKEN_SETUP: usb_packet_set_state(p, USB_PACKET_ASYNC);
ret = do_token_setup(dev, p); QTAILQ_INSERT_TAIL(&p->ep->queue, p, queue);
break; } else {
case USB_TOKEN_IN: p->result = ret;
ret = do_token_in(dev, p); usb_packet_set_state(p, USB_PACKET_COMPLETE);
break;
case USB_TOKEN_OUT:
ret = do_token_out(dev, p);
break;
default:
ret = USB_RET_STALL;
break;
} }
} else { } else {
/* data pipe */ ret = USB_RET_ASYNC;
ret = usb_device_handle_data(dev, p); usb_packet_set_state(p, USB_PACKET_QUEUED);
} QTAILQ_INSERT_TAIL(&p->ep->queue, p, queue);
if (ret == USB_RET_ASYNC) {
p->state = USB_PACKET_ASYNC;
} }
return ret; return ret;
} }
@ -325,9 +338,28 @@ int usb_handle_packet(USBDevice *dev, USBPacket *p)
handle_packet. */ handle_packet. */
void usb_packet_complete(USBDevice *dev, USBPacket *p) void usb_packet_complete(USBDevice *dev, USBPacket *p)
{ {
USBEndpoint *ep = p->ep;
int ret;
assert(p->state == USB_PACKET_ASYNC); assert(p->state == USB_PACKET_ASYNC);
p->state = USB_PACKET_COMPLETE; assert(QTAILQ_FIRST(&ep->queue) == p);
usb_packet_set_state(p, USB_PACKET_COMPLETE);
QTAILQ_REMOVE(&ep->queue, p, queue);
dev->port->ops->complete(dev->port, p); dev->port->ops->complete(dev->port, p);
while (!QTAILQ_EMPTY(&ep->queue)) {
p = QTAILQ_FIRST(&ep->queue);
assert(p->state == USB_PACKET_QUEUED);
ret = usb_process_one(p);
if (ret == USB_RET_ASYNC) {
usb_packet_set_state(p, USB_PACKET_ASYNC);
break;
}
p->result = ret;
usb_packet_set_state(p, USB_PACKET_COMPLETE);
QTAILQ_REMOVE(&ep->queue, p, queue);
dev->port->ops->complete(dev->port, p);
}
} }
/* Cancel an active packet. The packed must have been deferred by /* Cancel an active packet. The packed must have been deferred by
@ -335,9 +367,13 @@ void usb_packet_complete(USBDevice *dev, USBPacket *p)
completed. */ completed. */
void usb_cancel_packet(USBPacket * p) void usb_cancel_packet(USBPacket * p)
{ {
assert(p->state == USB_PACKET_ASYNC); bool callback = (p->state == USB_PACKET_ASYNC);
p->state = USB_PACKET_CANCELED; assert(usb_packet_is_inflight(p));
usb_device_cancel_packet(p->ep->dev, p); usb_packet_set_state(p, USB_PACKET_CANCELED);
QTAILQ_REMOVE(&p->ep->queue, p, queue);
if (callback) {
usb_device_cancel_packet(p->ep->dev, p);
}
} }
@ -346,14 +382,50 @@ void usb_packet_init(USBPacket *p)
qemu_iovec_init(&p->iov, 1); qemu_iovec_init(&p->iov, 1);
} }
void usb_packet_set_state(USBPacket *p, USBPacketState state)
{
#ifdef DEBUG
static const char *name[] = {
[USB_PACKET_UNDEFINED] = "undef",
[USB_PACKET_SETUP] = "setup",
[USB_PACKET_QUEUED] = "queued",
[USB_PACKET_ASYNC] = "async",
[USB_PACKET_COMPLETE] = "complete",
[USB_PACKET_CANCELED] = "canceled",
};
static const char *rets[] = {
[-USB_RET_NODEV] = "NODEV",
[-USB_RET_NAK] = "NAK",
[-USB_RET_STALL] = "STALL",
[-USB_RET_BABBLE] = "BABBLE",
[-USB_RET_ASYNC] = "ASYNC",
};
char add[16] = "";
if (state == USB_PACKET_COMPLETE) {
if (p->result < 0) {
snprintf(add, sizeof(add), " - %s", rets[-p->result]);
} else {
snprintf(add, sizeof(add), " - %d", p->result);
}
}
fprintf(stderr, "bus %s, port %s, dev %d, ep %d: packet %p: %s -> %s%s\n",
p->ep->dev->qdev.parent_bus->name,
p->ep->dev->port->path,
p->ep->dev->addr, p->ep->nr,
p, name[p->state], name[state], add);
#endif
p->state = state;
}
void usb_packet_setup(USBPacket *p, int pid, USBEndpoint *ep) void usb_packet_setup(USBPacket *p, int pid, USBEndpoint *ep)
{ {
assert(!usb_packet_is_inflight(p)); assert(!usb_packet_is_inflight(p));
p->state = USB_PACKET_SETUP;
p->pid = pid; p->pid = pid;
p->ep = ep; p->ep = ep;
p->result = 0; p->result = 0;
qemu_iovec_reset(&p->iov); qemu_iovec_reset(&p->iov);
usb_packet_set_state(p, USB_PACKET_SETUP);
} }
void usb_packet_addbuf(USBPacket *p, void *ptr, size_t len) void usb_packet_addbuf(USBPacket *p, void *ptr, size_t len)
@ -404,6 +476,7 @@ void usb_ep_init(USBDevice *dev)
dev->ep_ctl.type = USB_ENDPOINT_XFER_CONTROL; dev->ep_ctl.type = USB_ENDPOINT_XFER_CONTROL;
dev->ep_ctl.ifnum = 0; dev->ep_ctl.ifnum = 0;
dev->ep_ctl.dev = dev; dev->ep_ctl.dev = dev;
QTAILQ_INIT(&dev->ep_ctl.queue);
for (ep = 0; ep < USB_MAX_ENDPOINTS; ep++) { for (ep = 0; ep < USB_MAX_ENDPOINTS; ep++) {
dev->ep_in[ep].nr = ep + 1; dev->ep_in[ep].nr = ep + 1;
dev->ep_out[ep].nr = ep + 1; dev->ep_out[ep].nr = ep + 1;
@ -415,6 +488,8 @@ void usb_ep_init(USBDevice *dev)
dev->ep_out[ep].ifnum = 0; dev->ep_out[ep].ifnum = 0;
dev->ep_in[ep].dev = dev; dev->ep_in[ep].dev = dev;
dev->ep_out[ep].dev = dev; dev->ep_out[ep].dev = dev;
QTAILQ_INIT(&dev->ep_in[ep].queue);
QTAILQ_INIT(&dev->ep_out[ep].queue);
} }
} }

View File

@ -177,6 +177,7 @@ struct USBEndpoint {
uint8_t ifnum; uint8_t ifnum;
int max_packet_size; int max_packet_size;
USBDevice *dev; USBDevice *dev;
QTAILQ_HEAD(, USBPacket) queue;
}; };
/* definition of a USB device */ /* definition of a USB device */
@ -309,15 +310,16 @@ struct USBPort {
typedef void USBCallback(USBPacket * packet, void *opaque); typedef void USBCallback(USBPacket * packet, void *opaque);
/* Structure used to hold information about an active USB packet. */
typedef enum USBPacketState { typedef enum USBPacketState {
USB_PACKET_UNDEFINED = 0, USB_PACKET_UNDEFINED = 0,
USB_PACKET_SETUP, USB_PACKET_SETUP,
USB_PACKET_QUEUED,
USB_PACKET_ASYNC, USB_PACKET_ASYNC,
USB_PACKET_COMPLETE, USB_PACKET_COMPLETE,
USB_PACKET_CANCELED, USB_PACKET_CANCELED,
} USBPacketState; } USBPacketState;
/* Structure used to hold information about an active USB packet. */
struct USBPacket { struct USBPacket {
/* Data fields for use by the driver. */ /* Data fields for use by the driver. */
int pid; int pid;
@ -326,9 +328,11 @@ struct USBPacket {
int result; /* transfer length or USB_RET_* status code */ int result; /* transfer length or USB_RET_* status code */
/* Internal use by the USB layer. */ /* Internal use by the USB layer. */
USBPacketState state; USBPacketState state;
QTAILQ_ENTRY(USBPacket) queue;
}; };
void usb_packet_init(USBPacket *p); void usb_packet_init(USBPacket *p);
void usb_packet_set_state(USBPacket *p, USBPacketState state);
void usb_packet_setup(USBPacket *p, int pid, USBEndpoint *ep); void usb_packet_setup(USBPacket *p, int pid, USBEndpoint *ep);
void usb_packet_addbuf(USBPacket *p, void *ptr, size_t len); void usb_packet_addbuf(USBPacket *p, void *ptr, size_t len);
int usb_packet_map(USBPacket *p, QEMUSGList *sgl); int usb_packet_map(USBPacket *p, QEMUSGList *sgl);
@ -339,7 +343,8 @@ void usb_packet_cleanup(USBPacket *p);
static inline bool usb_packet_is_inflight(USBPacket *p) static inline bool usb_packet_is_inflight(USBPacket *p)
{ {
return p->state == USB_PACKET_ASYNC; return (p->state == USB_PACKET_QUEUED ||
p->state == USB_PACKET_ASYNC);
} }
USBDevice *usb_find_device(USBPort *port, uint8_t addr); USBDevice *usb_find_device(USBPort *port, uint8_t addr);