ref: d46099e3afa9681a4f98e0d2574c2ad65820ed7e
parent: adfb0e9371752ffb8d9995d4678e6bfa24c75f60
author: cinap_lenrek <[email protected]>
date: Sun Jul 30 23:07:14 EDT 2017
usbehci: fix crash in cancelisoio() for highspeed device due to wrong pollival
--- a/sys/src/9/port/usbehci.c
+++ b/sys/src/9/port/usbehci.c
@@ -2000,7 +2000,7 @@
iunlock(ctlr);
qunlock(iso);
poperror();
- diprint("uhci: episoread: %#p %uld bytes err '%s'\n", iso, tot, iso->err);
+ diprint("ehci: episoread: %#p %uld bytes err '%s'\n", iso, tot, iso->err);
if(iso->err != nil)
error(iso->err);
return tot;
@@ -2691,6 +2691,7 @@
ulong frno;
Ctlr *ctlr;
+ iso->hs = 0;
ctlr = ep->hp->aux;
left = 0;
ltd = nil;
@@ -2742,9 +2743,7 @@
Ctlr *ctlr;
iso->hs = 1;
- ival = 1;
- if(ep->pollival > 8)
- ival = ep->pollival/8;
+ ival = (ep->pollival+7)/8;
ctlr = ep->hp->aux;
left = 0;
ltd = nil;
@@ -2807,19 +2806,17 @@
iso->state = Qidle;
coherence();
iso->debug = ep->debug;
- ival = ep->pollival;
tpf = 1;
+ ival = ep->pollival;
if(ep->dev->speed == Highspeed){
tpf = 8;
- if(ival <= 8)
- ival = 1;
- else
- ival /= 8;
+ ival = (ival+7)/8;
}
- assert(ival != 0);
+ if(ival < 1)
+ error("bad pollival");
iso->nframes = Nisoframes / ival;
if(iso->nframes < 3)
- error("uhci isoopen bug"); /* we need at least 3 tds */
+ error("ehci isoopen bug"); /* we need at least 3 tds */
iso->maxsize = ep->ntds * ep->maxpkt;
if(ctlr->load + ep->load > 800)
print("usb: ehci: bandwidth may be exceeded\n");
@@ -2871,7 +2868,7 @@
for(w = 0; w < n; w++){
frno = iso->td0frno;
woff = w * Nisoframes;
- for(i = 0; i < iso->nframes ; i++){
+ for(i = 0; i < iso->nframes; i++){
assert(woff+frno < ctlr->nframes);
assert(iso->tdps[frno] != nil);
if(ep->dev->speed == Highspeed)
@@ -2881,7 +2878,7 @@
ctlr->frames[woff+frno] = PADDR(iso->tdps[frno])
|Lsitd;
coherence();
- frno = TRUNC(frno+ep->pollival, Nisoframes);
+ frno = TRUNC(frno+ival, Nisoframes);
}
}
coherence();
@@ -3002,7 +2999,7 @@
}
static void
-cancelisoio(Ctlr *ctlr, Isoio *iso, int pollival, ulong load)
+cancelisoio(Ctlr *ctlr, Isoio *iso, int ival, ulong load)
{
int frno, i, n, t, w, woff;
ulong *lp, *tp;
@@ -3031,6 +3028,8 @@
panic("cancleiso: not found");
*il = iso->next;
+ if(iso->hs)
+ ival = (ival+7)/8;
frno = iso->td0frno;
for(i = 0; i < iso->nframes; i++){
tp = iso->tdps[frno];
@@ -3062,7 +3061,7 @@
}
}
coherence();
- frno = TRUNC(frno+pollival, Nisoframes);
+ frno = TRUNC(frno+ival, Nisoframes);
}
iunlock(ctlr);
@@ -3085,7 +3084,7 @@
else
sitdfree(ctlr, iso->sitdps[frno]);
iso->tdps[frno] = nil;
- frno = TRUNC(frno+pollival, Nisoframes);
+ frno = TRUNC(frno+ival, Nisoframes);
}
free(iso->tdps);
iso->tdps = nil;