|
|
5544c1 |
From 82508212e19333e527b64fb76cd1bc016afacc8a Mon Sep 17 00:00:00 2001
|
|
|
c8dfc6 |
From: Gerd Hoffmann <kraxel@redhat.com>
|
|
|
c8dfc6 |
Date: Fri, 24 Aug 2012 14:13:08 +0200
|
|
|
5544c1 |
Subject: [PATCH] xhci: iso xfer support
|
|
|
c8dfc6 |
|
|
|
c8dfc6 |
Add support for iso transfers.
|
|
|
c8dfc6 |
|
|
|
c8dfc6 |
Signed-off-by: Gerd Hoffmann <kraxel@redhat.com>
|
|
|
c8dfc6 |
---
|
|
|
c8dfc6 |
hw/usb/hcd-xhci.c | 117 ++++++++++++++++++++++++++++++++++++++++++++++--------
|
|
|
c8dfc6 |
1 file changed, 101 insertions(+), 16 deletions(-)
|
|
|
c8dfc6 |
|
|
|
c8dfc6 |
diff --git a/hw/usb/hcd-xhci.c b/hw/usb/hcd-xhci.c
|
|
|
5544c1 |
index f5ba6a4..b313330 100644
|
|
|
c8dfc6 |
--- a/hw/usb/hcd-xhci.c
|
|
|
c8dfc6 |
+++ b/hw/usb/hcd-xhci.c
|
|
|
c8dfc6 |
@@ -325,9 +325,15 @@ typedef struct XHCITransfer {
|
|
|
c8dfc6 |
unsigned int pkts;
|
|
|
c8dfc6 |
unsigned int pktsize;
|
|
|
c8dfc6 |
unsigned int cur_pkt;
|
|
|
c8dfc6 |
+
|
|
|
c8dfc6 |
+ uint64_t mfindex_kick;
|
|
|
c8dfc6 |
} XHCITransfer;
|
|
|
c8dfc6 |
|
|
|
c8dfc6 |
typedef struct XHCIEPContext {
|
|
|
c8dfc6 |
+ XHCIState *xhci;
|
|
|
c8dfc6 |
+ unsigned int slotid;
|
|
|
c8dfc6 |
+ unsigned int epid;
|
|
|
c8dfc6 |
+
|
|
|
c8dfc6 |
XHCIRing ring;
|
|
|
c8dfc6 |
unsigned int next_xfer;
|
|
|
c8dfc6 |
unsigned int comp_xfer;
|
|
|
c8dfc6 |
@@ -337,6 +343,11 @@ typedef struct XHCIEPContext {
|
|
|
c8dfc6 |
dma_addr_t pctx;
|
|
|
c8dfc6 |
unsigned int max_psize;
|
|
|
c8dfc6 |
uint32_t state;
|
|
|
c8dfc6 |
+
|
|
|
c8dfc6 |
+ /* iso xfer scheduling */
|
|
|
c8dfc6 |
+ unsigned int interval;
|
|
|
c8dfc6 |
+ int64_t mfindex_last;
|
|
|
c8dfc6 |
+ QEMUTimer *kick_timer;
|
|
|
c8dfc6 |
} XHCIEPContext;
|
|
|
c8dfc6 |
|
|
|
c8dfc6 |
typedef struct XHCISlot {
|
|
|
c8dfc6 |
@@ -856,6 +867,12 @@ static void xhci_set_ep_state(XHCIState *xhci, XHCIEPContext *epctx,
|
|
|
c8dfc6 |
epctx->state = state;
|
|
|
c8dfc6 |
}
|
|
|
c8dfc6 |
|
|
|
c8dfc6 |
+static void xhci_ep_kick_timer(void *opaque)
|
|
|
c8dfc6 |
+{
|
|
|
c8dfc6 |
+ XHCIEPContext *epctx = opaque;
|
|
|
c8dfc6 |
+ xhci_kick_ep(epctx->xhci, epctx->slotid, epctx->epid);
|
|
|
c8dfc6 |
+}
|
|
|
c8dfc6 |
+
|
|
|
c8dfc6 |
static TRBCCode xhci_enable_ep(XHCIState *xhci, unsigned int slotid,
|
|
|
c8dfc6 |
unsigned int epid, dma_addr_t pctx,
|
|
|
c8dfc6 |
uint32_t *ctx)
|
|
|
c8dfc6 |
@@ -877,6 +894,9 @@ static TRBCCode xhci_enable_ep(XHCIState *xhci, unsigned int slotid,
|
|
|
c8dfc6 |
|
|
|
c8dfc6 |
epctx = g_malloc(sizeof(XHCIEPContext));
|
|
|
c8dfc6 |
memset(epctx, 0, sizeof(XHCIEPContext));
|
|
|
c8dfc6 |
+ epctx->xhci = xhci;
|
|
|
c8dfc6 |
+ epctx->slotid = slotid;
|
|
|
c8dfc6 |
+ epctx->epid = epid;
|
|
|
c8dfc6 |
|
|
|
c8dfc6 |
slot->eps[epid-1] = epctx;
|
|
|
c8dfc6 |
|
|
|
c8dfc6 |
@@ -895,6 +915,10 @@ static TRBCCode xhci_enable_ep(XHCIState *xhci, unsigned int slotid,
|
|
|
c8dfc6 |
usb_packet_init(&epctx->transfers[i].packet);
|
|
|
c8dfc6 |
}
|
|
|
c8dfc6 |
|
|
|
c8dfc6 |
+ epctx->interval = 1 << (ctx[0] >> 16) & 0xff;
|
|
|
c8dfc6 |
+ epctx->mfindex_last = 0;
|
|
|
c8dfc6 |
+ epctx->kick_timer = qemu_new_timer_ns(vm_clock, xhci_ep_kick_timer, epctx);
|
|
|
c8dfc6 |
+
|
|
|
c8dfc6 |
epctx->state = EP_RUNNING;
|
|
|
c8dfc6 |
ctx[0] &= ~EP_STATE_MASK;
|
|
|
c8dfc6 |
ctx[0] |= EP_RUNNING;
|
|
|
c8dfc6 |
@@ -934,6 +958,7 @@ static int xhci_ep_nuke_xfers(XHCIState *xhci, unsigned int slotid,
|
|
|
c8dfc6 |
if (t->running_retry) {
|
|
|
c8dfc6 |
t->running_retry = 0;
|
|
|
c8dfc6 |
epctx->retry = NULL;
|
|
|
c8dfc6 |
+ qemu_del_timer(epctx->kick_timer);
|
|
|
c8dfc6 |
}
|
|
|
c8dfc6 |
if (t->trbs) {
|
|
|
c8dfc6 |
g_free(t->trbs);
|
|
|
c8dfc6 |
@@ -969,6 +994,7 @@ static TRBCCode xhci_disable_ep(XHCIState *xhci, unsigned int slotid,
|
|
|
c8dfc6 |
|
|
|
c8dfc6 |
xhci_set_ep_state(xhci, epctx, EP_DISABLED);
|
|
|
c8dfc6 |
|
|
|
c8dfc6 |
+ qemu_free_timer(epctx->kick_timer);
|
|
|
c8dfc6 |
g_free(epctx);
|
|
|
c8dfc6 |
slot->eps[epid-1] = NULL;
|
|
|
c8dfc6 |
|
|
|
5544c1 |
@@ -1376,29 +1402,70 @@ static int xhci_fire_ctl_transfer(XHCIState *xhci, XHCITransfer *xfer)
|
|
|
c8dfc6 |
return 0;
|
|
|
c8dfc6 |
}
|
|
|
c8dfc6 |
|
|
|
c8dfc6 |
+static void xhci_calc_iso_kick(XHCIState *xhci, XHCITransfer *xfer,
|
|
|
c8dfc6 |
+ XHCIEPContext *epctx, uint64_t mfindex)
|
|
|
c8dfc6 |
+{
|
|
|
c8dfc6 |
+ if (xfer->trbs[0].control & TRB_TR_SIA) {
|
|
|
c8dfc6 |
+ uint64_t asap = ((mfindex + epctx->interval - 1) &
|
|
|
c8dfc6 |
+ ~(epctx->interval-1));
|
|
|
c8dfc6 |
+ if (asap >= epctx->mfindex_last &&
|
|
|
c8dfc6 |
+ asap <= epctx->mfindex_last + epctx->interval * 4) {
|
|
|
c8dfc6 |
+ xfer->mfindex_kick = epctx->mfindex_last + epctx->interval;
|
|
|
c8dfc6 |
+ } else {
|
|
|
c8dfc6 |
+ xfer->mfindex_kick = asap;
|
|
|
c8dfc6 |
+ }
|
|
|
c8dfc6 |
+ } else {
|
|
|
c8dfc6 |
+ xfer->mfindex_kick = (xfer->trbs[0].control >> TRB_TR_FRAMEID_SHIFT)
|
|
|
c8dfc6 |
+ & TRB_TR_FRAMEID_MASK;
|
|
|
c8dfc6 |
+ xfer->mfindex_kick |= mfindex & ~0x3fff;
|
|
|
c8dfc6 |
+ if (xfer->mfindex_kick < mfindex) {
|
|
|
c8dfc6 |
+ xfer->mfindex_kick += 0x4000;
|
|
|
c8dfc6 |
+ }
|
|
|
c8dfc6 |
+ }
|
|
|
c8dfc6 |
+}
|
|
|
c8dfc6 |
+
|
|
|
c8dfc6 |
+static void xhci_check_iso_kick(XHCIState *xhci, XHCITransfer *xfer,
|
|
|
c8dfc6 |
+ XHCIEPContext *epctx, uint64_t mfindex)
|
|
|
c8dfc6 |
+{
|
|
|
c8dfc6 |
+ if (xfer->mfindex_kick > mfindex) {
|
|
|
c8dfc6 |
+ qemu_mod_timer(epctx->kick_timer, qemu_get_clock_ns(vm_clock) +
|
|
|
c8dfc6 |
+ (xfer->mfindex_kick - mfindex) * 125000);
|
|
|
c8dfc6 |
+ xfer->running_retry = 1;
|
|
|
c8dfc6 |
+ } else {
|
|
|
c8dfc6 |
+ epctx->mfindex_last = xfer->mfindex_kick;
|
|
|
c8dfc6 |
+ qemu_del_timer(epctx->kick_timer);
|
|
|
c8dfc6 |
+ xfer->running_retry = 0;
|
|
|
c8dfc6 |
+ }
|
|
|
c8dfc6 |
+}
|
|
|
c8dfc6 |
+
|
|
|
c8dfc6 |
+
|
|
|
c8dfc6 |
static int xhci_submit(XHCIState *xhci, XHCITransfer *xfer, XHCIEPContext *epctx)
|
|
|
c8dfc6 |
{
|
|
|
c8dfc6 |
+ uint64_t mfindex;
|
|
|
c8dfc6 |
int ret;
|
|
|
c8dfc6 |
|
|
|
c8dfc6 |
DPRINTF("xhci_submit(slotid=%d,epid=%d)\n", xfer->slotid, xfer->epid);
|
|
|
c8dfc6 |
|
|
|
c8dfc6 |
xfer->in_xfer = epctx->type>>2;
|
|
|
c8dfc6 |
|
|
|
c8dfc6 |
- if (epctx->type == ET_ISO_IN || epctx->type == ET_ISO_OUT) {
|
|
|
c8dfc6 |
- xfer->pkts = 1;
|
|
|
c8dfc6 |
- } else {
|
|
|
c8dfc6 |
- xfer->pkts = 0;
|
|
|
c8dfc6 |
- }
|
|
|
c8dfc6 |
-
|
|
|
c8dfc6 |
switch(epctx->type) {
|
|
|
c8dfc6 |
case ET_INTR_OUT:
|
|
|
c8dfc6 |
case ET_INTR_IN:
|
|
|
c8dfc6 |
case ET_BULK_OUT:
|
|
|
c8dfc6 |
case ET_BULK_IN:
|
|
|
c8dfc6 |
+ xfer->pkts = 0;
|
|
|
c8dfc6 |
+ xfer->iso_xfer = false;
|
|
|
c8dfc6 |
break;
|
|
|
c8dfc6 |
case ET_ISO_OUT:
|
|
|
c8dfc6 |
case ET_ISO_IN:
|
|
|
c8dfc6 |
- FIXME();
|
|
|
c8dfc6 |
+ xfer->pkts = 1;
|
|
|
c8dfc6 |
+ xfer->iso_xfer = true;
|
|
|
c8dfc6 |
+ mfindex = xhci_mfindex_get(xhci);
|
|
|
c8dfc6 |
+ xhci_calc_iso_kick(xhci, xfer, epctx, mfindex);
|
|
|
c8dfc6 |
+ xhci_check_iso_kick(xhci, xfer, epctx, mfindex);
|
|
|
c8dfc6 |
+ if (xfer->running_retry) {
|
|
|
c8dfc6 |
+ return -1;
|
|
|
c8dfc6 |
+ }
|
|
|
c8dfc6 |
break;
|
|
|
c8dfc6 |
default:
|
|
|
c8dfc6 |
fprintf(stderr, "xhci: unknown or unhandled EP "
|
|
|
5544c1 |
@@ -1428,6 +1495,7 @@ static int xhci_fire_transfer(XHCIState *xhci, XHCITransfer *xfer, XHCIEPContext
|
|
|
c8dfc6 |
static void xhci_kick_ep(XHCIState *xhci, unsigned int slotid, unsigned int epid)
|
|
|
c8dfc6 |
{
|
|
|
c8dfc6 |
XHCIEPContext *epctx;
|
|
|
c8dfc6 |
+ uint64_t mfindex;
|
|
|
c8dfc6 |
int length;
|
|
|
c8dfc6 |
int i;
|
|
|
c8dfc6 |
|
|
|
5544c1 |
@@ -1447,20 +1515,35 @@ static void xhci_kick_ep(XHCIState *xhci, unsigned int slotid, unsigned int epid
|
|
|
c8dfc6 |
}
|
|
|
c8dfc6 |
|
|
|
c8dfc6 |
if (epctx->retry) {
|
|
|
c8dfc6 |
- /* retry nak'ed transfer */
|
|
|
c8dfc6 |
XHCITransfer *xfer = epctx->retry;
|
|
|
c8dfc6 |
int result;
|
|
|
c8dfc6 |
|
|
|
c8dfc6 |
trace_usb_xhci_xfer_retry(xfer);
|
|
|
c8dfc6 |
assert(xfer->running_retry);
|
|
|
c8dfc6 |
- if (xhci_setup_packet(xfer) < 0) {
|
|
|
c8dfc6 |
- return;
|
|
|
c8dfc6 |
- }
|
|
|
c8dfc6 |
- result = usb_handle_packet(xfer->packet.ep->dev, &xfer->packet);
|
|
|
c8dfc6 |
- if (result == USB_RET_NAK) {
|
|
|
c8dfc6 |
- return;
|
|
|
c8dfc6 |
+ if (xfer->iso_xfer) {
|
|
|
c8dfc6 |
+ /* retry delayed iso transfer */
|
|
|
c8dfc6 |
+ mfindex = xhci_mfindex_get(xhci);
|
|
|
c8dfc6 |
+ xhci_check_iso_kick(xhci, xfer, epctx, mfindex);
|
|
|
c8dfc6 |
+ if (xfer->running_retry) {
|
|
|
c8dfc6 |
+ return;
|
|
|
c8dfc6 |
+ }
|
|
|
c8dfc6 |
+ if (xhci_setup_packet(xfer) < 0) {
|
|
|
c8dfc6 |
+ return;
|
|
|
c8dfc6 |
+ }
|
|
|
c8dfc6 |
+ result = usb_handle_packet(xfer->packet.ep->dev, &xfer->packet);
|
|
|
c8dfc6 |
+ assert(result != USB_RET_NAK);
|
|
|
c8dfc6 |
+ xhci_complete_packet(xfer, result);
|
|
|
c8dfc6 |
+ } else {
|
|
|
c8dfc6 |
+ /* retry nak'ed transfer */
|
|
|
c8dfc6 |
+ if (xhci_setup_packet(xfer) < 0) {
|
|
|
c8dfc6 |
+ return;
|
|
|
c8dfc6 |
+ }
|
|
|
c8dfc6 |
+ result = usb_handle_packet(xfer->packet.ep->dev, &xfer->packet);
|
|
|
c8dfc6 |
+ if (result == USB_RET_NAK) {
|
|
|
c8dfc6 |
+ return;
|
|
|
c8dfc6 |
+ }
|
|
|
c8dfc6 |
+ xhci_complete_packet(xfer, result);
|
|
|
c8dfc6 |
}
|
|
|
c8dfc6 |
- xhci_complete_packet(xfer, result);
|
|
|
c8dfc6 |
assert(!xfer->running_retry);
|
|
|
c8dfc6 |
epctx->retry = NULL;
|
|
|
c8dfc6 |
}
|
|
|
5544c1 |
@@ -1512,7 +1595,9 @@ static void xhci_kick_ep(XHCIState *xhci, unsigned int slotid, unsigned int epid
|
|
|
c8dfc6 |
if (xhci_fire_transfer(xhci, xfer, epctx) >= 0) {
|
|
|
c8dfc6 |
epctx->next_xfer = (epctx->next_xfer + 1) % TD_QUEUE;
|
|
|
c8dfc6 |
} else {
|
|
|
c8dfc6 |
- fprintf(stderr, "xhci: error firing data transfer\n");
|
|
|
c8dfc6 |
+ if (!xfer->iso_xfer) {
|
|
|
c8dfc6 |
+ fprintf(stderr, "xhci: error firing data transfer\n");
|
|
|
c8dfc6 |
+ }
|
|
|
c8dfc6 |
}
|
|
|
c8dfc6 |
}
|
|
|
c8dfc6 |
|
|
|
c8dfc6 |
--
|
|
|
5544c1 |
1.7.12.1
|
|
|
c8dfc6 |
|