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 OHCI_PAGE limitations in oh...



details:   https://anonhg.NetBSD.org/src/rev/da4da2bfbfb4
branches:  trunk
changeset: 972334:da4da2bfbfb4
user:      skrll <skrll%NetBSD.org@localhost>
date:      Sun May 24 07:42:51 2020 +0000

description:
Be more careful with OHCI_PAGE limitations in ohci_reset_std_chain and
ohci_device_isoc_enter

Enable USBMALLOC_MULTISEG

diffstat:

 sys/dev/usb/ohci.c |  86 +++++++++++++++++++++++++++++++++++++----------------
 1 files changed, 59 insertions(+), 27 deletions(-)

diffs (163 lines):

diff -r 2dff2362af44 -r da4da2bfbfb4 sys/dev/usb/ohci.c
--- a/sys/dev/usb/ohci.c        Sun May 24 07:17:34 2020 +0000
+++ b/sys/dev/usb/ohci.c        Sun May 24 07:42:51 2020 +0000
@@ -1,4 +1,4 @@
-/*     $NetBSD: ohci.c,v 1.307 2020/05/19 19:09:43 jakllsch Exp $      */
+/*     $NetBSD: ohci.c,v 1.308 2020/05/24 07:42:51 skrll Exp $ */
 
 /*
  * Copyright (c) 1998, 2004, 2005, 2012 The NetBSD Foundation, Inc.
@@ -41,7 +41,7 @@
  */
 
 #include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: ohci.c,v 1.307 2020/05/19 19:09:43 jakllsch Exp $");
+__KERNEL_RCSID(0, "$NetBSD: ohci.c,v 1.308 2020/05/24 07:42:51 skrll Exp $");
 
 #ifdef _KERNEL_OPT
 #include "opt_usb.h"
@@ -618,26 +618,28 @@
                        next = ox->ox_stds[j++];
                KASSERT(next != cur);
 
-               curlen = 0;
-               const ohci_physaddr_t sdataphys = DMAADDR(dma, curoffs);
-               ohci_physaddr_t edataphys = DMAADDR(dma, curoffs + len - 1);
-
-               const ohci_physaddr_t sphyspg = OHCI_PAGE(sdataphys);
-               ohci_physaddr_t ephyspg = OHCI_PAGE(edataphys);
+               curlen = len;
                /*
-                * The OHCI hardware can handle at most one page
-                * crossing per TD
+                * The OHCI hardware can handle at most one page crossing per
+                * TD.  That is, 2 * OHCI_PAGE_SIZE as a maximum.  Limit the
+                * length in this TD accordingly.
                 */
-               curlen = len;
-               if (sphyspg != ephyspg &&
-                   sphyspg + OHCI_PAGE_SIZE != ephyspg) {
-                       /* must use multiple TDs, fill as much as possible. */
-                       curlen = 2 * OHCI_PAGE_SIZE -
-                           OHCI_PAGE_OFFSET(sdataphys);
-                       /* the length must be a multiple of the max size */
+               const ohci_physaddr_t sdataphys = DMAADDR(dma, curoffs);
+
+               int maxlen = (2 * OHCI_PAGE_SIZE) - OHCI_PAGE_OFFSET(sdataphys);
+               if (curlen > maxlen) {
+                       curlen = maxlen;
+
+                       /*
+                        * the length must be a multiple of
+                        * the max size
+                        */
                        curlen -= curlen % mps;
-                       edataphys = DMAADDR(dma, curoffs + curlen - 1);
                }
+
+               const int edataoffs = curoffs + curlen - 1;
+               const ohci_physaddr_t edataphys = DMAADDR(dma, edataoffs);
+
                KASSERT(curlen != 0);
                DPRINTFN(4, "sdataphys=0x%08jx edataphys=0x%08jx "
                    "len=%jd curlen=%jd", sdataphys, edataphys, len, curlen);
@@ -811,6 +813,7 @@
        }
        sc->sc_bus.ub_revision = USBREV_1_0;
        sc->sc_bus.ub_usedma = true;
+       sc->sc_bus.ub_dmaflags = USBMALLOC_MULTISEG;
 
        /* XXX determine alignment by R/W */
        /* Allocate the HCCA area. */
@@ -3447,8 +3450,9 @@
        ohci_softc_t *sc = OHCI_XFER2SC(xfer);
        ohci_soft_ed_t *sed = opipe->sed;
        ohci_soft_itd_t *sitd, *nsitd, *tail;
-       ohci_physaddr_t buf, offs, noffs, bp0;
+       ohci_physaddr_t buf, offs, bp0, bp1;
        int i, ncur, nframes;
+       size_t boff, frlen;
 
        OHCIHIST_FUNC(); OHCIHIST_CALLED();
        DPRINTFN(5, "xfer=%#jx", (uintptr_t)xfer, 0, 0, 0);
@@ -3486,17 +3490,44 @@
        opipe->tail.itd = ox->ox_sitds[0];
        ox->ox_sitds[0] = sitd;
 
+       boff = 0;
        buf = DMAADDR(&xfer->ux_dmabuf, 0);
-       bp0 = OHCI_PAGE(buf);
+       bp0 = bp1 = OHCI_PAGE(buf);
        offs = OHCI_PAGE_OFFSET(buf);
+
+       ohci_physaddr_t end = bp0;      /* XXX stupid GCC */
+
        nframes = xfer->ux_nframes;
        xfer->ux_hcpriv = sitd;
        size_t j = 1;
        for (i = ncur = 0; i < nframes; i++, ncur++) {
-               noffs = offs + xfer->ux_frlengths[i];
-               if (ncur == OHCI_ITD_NOFFSET || /* all offsets used */
-                   OHCI_PAGE(buf + noffs) > bp0 + OHCI_PAGE_SIZE) { /* too many page crossings */
-
+               frlen = xfer->ux_frlengths[i];
+
+               DPRINTFN(1, "frame=%jd ux_frlengths[%jd]=%jd", i, i,
+                   xfer->ux_frlengths[i], 0);
+               /*
+                * XXXNH: The loop assumes this is never true, because
+                * incrementing 'i' assumes all the ux_frlengths[i] is covered.
+                */
+               if (frlen > 2 * OHCI_PAGE_SIZE - offs)
+                       frlen = 2 * OHCI_PAGE_SIZE - offs;
+
+               boff += frlen;
+               buf = DMAADDR(&xfer->ux_dmabuf, boff);
+               ohci_physaddr_t noffs = OHCI_PAGE_OFFSET(buf);
+
+               ohci_physaddr_t nend = DMAADDR(&xfer->ux_dmabuf, boff - 1);
+               const ohci_physaddr_t nep = OHCI_PAGE(nend);
+
+               /* Note the first page crossing in bp1 */
+               if (bp0 == bp1 && bp1 != nep)
+                       bp1 = nep;
+
+               DPRINTFN(1, "ncur=%jd bp0=%#jx bp1=%#jx nend=%#jx",
+                   ncur, bp0, bp1, nend);
+
+               /* all offsets used or too many page crossings */
+               if (ncur == OHCI_ITD_NOFFSET || (bp0 != bp1 && bp1 != nep)) {
                        /* Allocate next ITD */
                        nsitd = ox->ox_sitds[j++];
                        KASSERT(nsitd != NULL);
@@ -3510,7 +3541,7 @@
                                OHCI_ITD_SET_FC(ncur));
                        sitd->itd.itd_bp0 = HTOO32(bp0);
                        sitd->itd.itd_nextitd = HTOO32(nsitd->physaddr);
-                       sitd->itd.itd_be = HTOO32(bp0 + offs - 1);
+                       sitd->itd.itd_be = HTOO32(end);
                        sitd->nextitd = nsitd;
                        sitd->xfer = xfer;
                        sitd->flags = 0;
@@ -3523,10 +3554,11 @@
 
                        sitd = nsitd;
                        isoc->next = isoc->next + ncur;
-                       bp0 = OHCI_PAGE(buf + offs);
+                       bp0 = bp1 = OHCI_PAGE(buf);
                        ncur = 0;
                }
                sitd->itd.itd_offset[ncur] = HTOO16(OHCI_ITD_MK_OFFS(offs));
+               end = nend;
                offs = noffs;
        }
        KASSERT(j <= ox->ox_nsitd);
@@ -3547,7 +3579,7 @@
                OHCI_ITD_SET_FC(ncur));
        sitd->itd.itd_bp0 = HTOO32(bp0);
        sitd->itd.itd_nextitd = HTOO32(tail->physaddr);
-       sitd->itd.itd_be = HTOO32(bp0 + offs - 1);
+       sitd->itd.itd_be = HTOO32(end);
        sitd->nextitd = tail;
        sitd->xfer = xfer;
        sitd->flags = OHCI_CALL_DONE;



Home | Main Index | Thread Index | Old Index