Source-Changes-HG archive
[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index][Old Index]
[src/trunk]: src/sys/dev/usb Be more careful with transfer descriptors and cr...
details: https://anonhg.NetBSD.org/src/rev/c80a70c199c4
branches: trunk
changeset: 933463:c80a70c199c4
user: skrll <skrll%NetBSD.org@localhost>
date: Mon May 25 13:55:31 2020 +0000
description:
Be more careful with transfer descriptors and crossing page boundaries
Enable USBMALLOC_MULTISEG
diffstat:
sys/dev/usb/uhci.c | 39 ++++++++++++++++++++++++++++++++-------
1 files changed, 32 insertions(+), 7 deletions(-)
diffs (122 lines):
diff -r 995bf9aadc54 -r c80a70c199c4 sys/dev/usb/uhci.c
--- a/sys/dev/usb/uhci.c Mon May 25 11:11:52 2020 +0000
+++ b/sys/dev/usb/uhci.c Mon May 25 13:55:31 2020 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: uhci.c,v 1.301 2020/05/15 06:15:42 skrll Exp $ */
+/* $NetBSD: uhci.c,v 1.302 2020/05/25 13:55:31 skrll Exp $ */
/*
* Copyright (c) 1998, 2004, 2011, 2012 The NetBSD Foundation, Inc.
@@ -42,7 +42,7 @@
*/
#include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: uhci.c,v 1.301 2020/05/15 06:15:42 skrll Exp $");
+__KERNEL_RCSID(0, "$NetBSD: uhci.c,v 1.302 2020/05/25 13:55:31 skrll Exp $");
#ifdef _KERNEL_OPT
#include "opt_usb.h"
@@ -582,6 +582,7 @@
sc->sc_bus.ub_methods = &uhci_bus_methods;
sc->sc_bus.ub_pipesize = sizeof(struct uhci_pipe);
sc->sc_bus.ub_usedma = true;
+ sc->sc_bus.ub_dmaflags = USBMALLOC_MULTISEG;
UHCICMD(sc, UHCI_CMD_MAXP); /* Assume 64 byte packets at frame end */
@@ -2009,6 +2010,17 @@
return EINVAL;
}
size_t ntd = howmany(len, maxp);
+ /*
+ * if our transfer is bigger than PAGE_SIZE and maxp not a factor of
+ * PAGE_SIZE then we will need another TD per page.
+ */
+ if (len > PAGE_SIZE && (PAGE_SIZE % maxp) != 0) {
+ ntd += howmany(len, PAGE_SIZE);
+ }
+
+ /*
+ * Might need one more TD if we're writing a ZLP
+ */
if (!rd && (flags & USBD_FORCE_SHORT_XFER)) {
ntd++;
}
@@ -2080,7 +2092,7 @@
int tog = *toggle;
int maxp;
uint32_t status;
- size_t i;
+ size_t i, offs;
UHCIHIST_FUNC(); UHCIHIST_CALLED();
DPRINTFN(8, "xfer=%#jx len %jd isread %jd toggle %jd", (uintptr_t)xfer,
@@ -2102,9 +2114,15 @@
usb_syncmem(dma, 0, len,
isread ? BUS_DMASYNC_PREREAD : BUS_DMASYNC_PREWRITE);
std = prev = NULL;
- for (i = 0; len != 0 && i < uxfer->ux_nstd; i++, prev = std) {
+ for (offs = i = 0; len != 0 && i < uxfer->ux_nstd; i++, prev = std) {
int l = len;
std = uxfer->ux_stds[i];
+
+ const bus_addr_t sbp = DMAADDR(dma, offs);
+ const bus_addr_t ebp = DMAADDR(dma, offs + l - 1);
+ if (((sbp ^ ebp) & ~PAGE_MASK) != 0)
+ l = PAGE_SIZE - (DMAADDR(dma, offs) & PAGE_MASK);
+
if (l > maxp)
l = maxp;
@@ -2129,7 +2147,7 @@
UHCI_TD_SET_DT(tog) |
UHCI_TD_SET_MAXLEN(l)
);
- std->td.td_buffer = htole32(DMAADDR(dma, i * maxp));
+ std->td.td_buffer = htole32(DMAADDR(dma, offs));
std->link.std = NULL;
@@ -2137,6 +2155,7 @@
BUS_DMASYNC_PREWRITE | BUS_DMASYNC_PREREAD);
tog ^= 1;
+ offs += l;
len -= l;
}
KASSERTMSG(len == 0, "xfer %p alen %d len %d mps %d ux_nqtd %zu i %zu",
@@ -2933,17 +2952,20 @@
xfer->ux_status = USBD_IN_PROGRESS;
ux->ux_curframe = next;
- buf = DMAADDR(&xfer->ux_dmabuf, 0);
offs = 0;
status = UHCI_TD_ZERO_ACTLEN(UHCI_TD_SET_ERRCNT(0) |
UHCI_TD_ACTIVE |
UHCI_TD_IOS);
nframes = xfer->ux_nframes;
for (i = 0; i < nframes; i++) {
+ buf = DMAADDR(&xfer->ux_dmabuf, offs);
std = isoc->stds[next];
if (++next >= UHCI_VFRAMELIST_COUNT)
next = 0;
len = xfer->ux_frlengths[i];
+
+ KASSERTMSG(len <= __SHIFTOUT_MASK(UHCI_TD_MAXLEN_MASK),
+ "len %d", len);
std->td.td_buffer = htole32(buf);
usb_syncmem(&xfer->ux_dmabuf, offs, len,
rd ? BUS_DMASYNC_PREREAD : BUS_DMASYNC_PREWRITE);
@@ -2962,8 +2984,11 @@
DPRINTF("--- dump end ---", 0, 0, 0, 0);
}
#endif
- buf += len;
offs += len;
+ const bus_addr_t bend __diagused =
+ DMAADDR(&xfer->ux_dmabuf, offs - 1);
+
+ KASSERT(((buf ^ bend) & ~PAGE_MASK) == 0);
}
isoc->next = next;
isoc->inuse += xfer->ux_nframes;
Home |
Main Index |
Thread Index |
Old Index