ref: 8ed13fe6643eab17ef3f5cff75830d3a7c1bc715
parent: f001ddfdb598f9a588b22c029a5ade9030324c4b
author: cinap_lenrek <[email protected]>
date: Tue Aug 1 21:45:30 EDT 2017
usbxhci: have to serialize and set read pointer for endpoint stop command
--- a/sys/src/9/pc/usbxhci.c
+++ b/sys/src/9/pc/usbxhci.c
@@ -175,6 +175,8 @@
u32int *ctx;
u32int *doorbell;
+ int stopped;
+
Wait *pending;
Lock;
};
@@ -302,6 +304,7 @@
r->slot = nil;
r->doorbell = nil;
r->pending = nil;
+ r->stopped = 0;
r->shift = shift;
r->mask = (1<<shift)-1;
r->rp = r->wp = 0;
@@ -474,30 +477,30 @@
}
static void
-queuetd(Ring *ring, u32int c, u32int s, u64int p, Wait *w)
+queuetd(Ring *r, u32int c, u32int s, u64int p, Wait *w)
{
u32int *td, x;
- x = ring->wp++;
- if((x & ring->mask) == ring->mask){
- td = ring->base + 4*(x & ring->mask);
- *(u64int*)td = PADDR(ring->base);
+ x = r->wp++;
+ if((x & r->mask) == r->mask){
+ td = r->base + 4*(x & r->mask);
+ *(u64int*)td = PADDR(r->base);
td[2] = 0;
- td[3] = ((~x>>ring->shift)&1) | (1<<1) | TR_LINK;
- x = ring->wp++;
+ td[3] = ((~x>>r->shift)&1) | (1<<1) | TR_LINK;
+ x = r->wp++;
}
- td = ring->base + 4*(x & ring->mask);
+ td = r->base + 4*(x & r->mask);
if(w != nil){
- memset(w->er, 0, 4*4);
- w->ring = ring;
+ w->er[0] = w->er[1] = w->er[2] = w->er[3] = 0;
+ w->ring = r;
w->td = td;
w->z = &up->sleep;
- w->next = ring->pending;
- ring->pending = w;
+ w->next = r->pending;
+ r->pending = w;
}
*(u64int*)td = p;
td[2] = s;
- td[3] = ((~x>>ring->shift)&1) | c;
+ td[3] = ((~x>>r->shift)&1) | c;
}
static char *ccerrtab[] = {
@@ -556,22 +559,45 @@
return ((Wait*)a)->z == nil;
}
+static void
+flushring(Ring *r)
+{
+ Rendez *z;
+ Wait *w;
+
+ while((w = r->pending) != nil){
+ r->pending = w->next;
+ w->next = nil;
+ if((z = w->z) != nil){
+ w->z = nil;
+ wakeup(z);
+ }
+ }
+}
+
static char*
ctlrcmd(Ctlr *ctlr, u32int c, u32int s, u64int p, u32int *er);
static char*
-waittd(Ctlr *ctlr, Wait *w, int tmout, u32int *er)
+waittd(Ctlr *ctlr, Wait *w, int tmout, QLock *q)
{
- Ring *ring = w->ring;
+ Ring *r = w->ring;
coherence();
- *ring->doorbell = ring->id;
+ *r->doorbell = r->id;
while(waserror()){
- if(ring == ctlr->cr)
- ctlr->opr[CRCR] |= CA;
- else
- ctlrcmd(ctlr, CR_STOPEP | (ring->id<<16) | (ring->slot->id<<24), 0, 0, nil);
+ if(q != nil)
+ qlock(q);
+ if(!r->stopped) {
+ if(r == ctlr->cr)
+ ctlr->opr[CRCR] |= CA;
+ else
+ ctlrcmd(ctlr, CR_STOPEP | (r->id<<16) | (r->slot->id<<24), 0, 0, nil);
+ r->stopped = 1;
+ }
+ if(q != nil)
+ qunlock(q);
tmout = 0;
}
if(tmout > 0){
@@ -584,8 +610,12 @@
}
poperror();
- if(er != nil)
- memmove(er, w->er, 4*4);
+ if(r->stopped) {
+ ilock(r);
+ flushring(r);
+ iunlock(r);
+ }
+
return ccerrstr(w->er[2]>>24);
}
@@ -593,12 +623,21 @@
ctlrcmd(Ctlr *ctlr, u32int c, u32int s, u64int p, u32int *er)
{
Wait ws[1];
+ char *err;
ilock(ctlr->cr);
+ if(ctlr->cr->stopped){
+ flushring(ctlr->cr);
+ ctlr->cr->stopped = 0;
+ }
queuetd(ctlr->cr, c, s, p, ws);
iunlock(ctlr->cr);
- return waittd(ctlr, ws, 5000, er);
+ err = waittd(ctlr, ws, 5000, nil);
+ if(er != nil)
+ memmove(er, ws->er, 4*4);
+
+ return err;
}
static void
@@ -682,6 +721,10 @@
case ER_HCE:
iprint("xhci: host controller error: %ux %ux %ux %ux\n",
td[0], td[1], td[2], td[3]);
+ ilock(ctlr->cr);
+ ctlr->cr->stopped = 1;
+ flushring(ctlr->cr);
+ iunlock(ctlr->cr);
break;
case ER_PORTSC:
break;
@@ -1133,21 +1176,27 @@
char *err;
switch(r->ctx[0]&7){
- case 0: /* disabled */
- case 1: /* running */
- case 3: /* stopped */
- break;
case 2: /* halted */
case 4: /* error */
+ r->stopped = 1;
+
+ err = ctlrcmd(r->slot->ctlr, CR_RESETEP | (r->id<<16) | (r->slot->id<<24), 0, 0, nil);
+ if(err != nil)
+ return err;
+ }
+
+ if(r->stopped){
ilock(r);
+ flushring(r);
r->rp = r->wp;
qp = PADDR(&r->base[4*(r->wp & r->mask)]) | ((~r->wp>>r->shift) & 1);
iunlock(r);
- err = ctlrcmd(r->slot->ctlr, CR_RESETEP | (r->id<<16) | (r->slot->id<<24), 0, 0, nil);
- ctlrcmd(r->slot->ctlr, CR_SETTRDQP | (r->id<<16) | (r->slot->id<<24), 0, qp, nil);
+ err = ctlrcmd(r->slot->ctlr, CR_SETTRDQP | (r->id<<16) | (r->slot->id<<24), 0, qp, nil);
if(err != nil)
return err;
+
+ r->stopped = 0;
}
if(r->wp - r->rp >= r->mask)
@@ -1210,7 +1259,7 @@
iunlock(io->ring);
qunlock(io);
- if((err = waittd((Ctlr*)ep->hp->aux, ws, ep->tmout, nil)) != nil)
+ if((err = waittd((Ctlr*)ep->hp->aux, ws, ep->tmout, io)) != nil)
error(err);
n -= (ws->er[2] & 0xFFFFFF);
@@ -1243,8 +1292,13 @@
return n;
io = (Epio*)ep->aux + OREAD;
+ ring = io[OWRITE-OREAD].ring;
qlock(io);
if(waserror()){
+ ilock(ring);
+ ring->pending = nil;
+ iunlock(ring);
+
qunlock(io);
nexterror();
}
@@ -1264,8 +1318,6 @@
io->b->wp += len;
}
}
-
- ring = io[OWRITE-OREAD].ring;
if((err = unstall(ring)) != nil)
error(err);
@@ -1304,6 +1356,7 @@
}
if((err = waittd((Ctlr*)ep->hp->aux, &ws[2], ep->tmout, nil)) != nil)
error(err);
+
qunlock(io);
poperror();
@@ -1337,7 +1390,7 @@
iunlock(io->ring);
qunlock(io);
- if((err = waittd((Ctlr*)ep->hp->aux, ws, ep->tmout, nil)) != nil)
+ if((err = waittd((Ctlr*)ep->hp->aux, ws, ep->tmout, io)) != nil)
error(err);
return n;