Source-Changes-HG archive
[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index][Old Index]
[src/trunk]: src/sys/dev/usb Improvements:
details: https://anonhg.NetBSD.org/src/rev/121b9a0ca6e8
branches: trunk
changeset: 464329:121b9a0ca6e8
user: maxv <maxv%NetBSD.org@localhost>
date: Thu Oct 03 05:13:23 2019 +0000
description:
Improvements:
- Don't process packets if the USB device is detached. Contrary to the
other HCIs, vHCI has no timeout, so we never collect the pending
packets, and must drop them synchronously.
- Fix refcounting bug in vhci_device_ctrl_abort.
- Implement vhci_activate.
- Add a few KASSERTs.
diffstat:
sys/dev/usb/vhci.c | 63 ++++++++++++++++++++++++++++++++++++++---------------
1 files changed, 45 insertions(+), 18 deletions(-)
diffs (186 lines):
diff -r 68694c15ace3 -r 121b9a0ca6e8 sys/dev/usb/vhci.c
--- a/sys/dev/usb/vhci.c Thu Oct 03 05:06:29 2019 +0000
+++ b/sys/dev/usb/vhci.c Thu Oct 03 05:13:23 2019 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: vhci.c,v 1.2 2019/09/14 12:32:08 maxv Exp $ */
+/* $NetBSD: vhci.c,v 1.3 2019/10/03 05:13:23 maxv Exp $ */
/*
* Copyright (c) 2019 The NetBSD Foundation, Inc.
@@ -30,7 +30,7 @@
*/
#include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: vhci.c,v 1.2 2019/09/14 12:32:08 maxv Exp $");
+__KERNEL_RCSID(0, "$NetBSD: vhci.c,v 1.3 2019/10/03 05:13:23 maxv Exp $");
#ifdef _KERNEL_OPT
#include "opt_usb.h"
@@ -239,11 +239,10 @@
TAILQ_INSERT_TAIL(&vxfer->pkts, pkt, xferlist);
/* Insert in the port. */
- mutex_enter(&port->lock);
+ KASSERT(mutex_owned(&port->lock));
TAILQ_INSERT_TAIL(reqlist, req, portlist);
if (pkt != NULL)
TAILQ_INSERT_TAIL(pktlist, pkt, portlist);
- mutex_exit(&port->lock);
}
static void
@@ -508,7 +507,7 @@
vhci_port_t *port;
bool polling = sc->sc_bus.ub_usepolling;
bool isread = (req->bmRequestType & UT_READ) != 0;
- int portno;
+ int portno, ret;
KASSERT(xfer->ux_rqflags & URQ_REQUEST);
KASSERT(dev->ud_myhsport != NULL);
@@ -524,13 +523,21 @@
if (!polling)
mutex_enter(&sc->sc_lock);
- xfer->ux_status = USBD_IN_PROGRESS;
+
+ mutex_enter(&port->lock);
+ if (port->status & UPS_PORT_ENABLED) {
+ xfer->ux_status = USBD_IN_PROGRESS;
+ vhci_pkt_create(port, xfer, isread);
+ ret = USBD_IN_PROGRESS;
+ } else {
+ ret = USBD_IOERROR;
+ }
+ mutex_exit(&port->lock);
+
if (!polling)
mutex_exit(&sc->sc_lock);
- vhci_pkt_create(port, xfer, isread);
-
- return USBD_IN_PROGRESS;
+ return ret;
}
static void
@@ -554,7 +561,7 @@
return;
mutex_enter(&port->lock);
- for (; vxfer->refcnt > 0; vxfer->refcnt--) {
+ while (vxfer->refcnt > 0) {
pkt = TAILQ_FIRST(&vxfer->pkts);
KASSERT(pkt != NULL);
vhci_pkt_destroy(sc, pkt);
@@ -708,9 +715,11 @@
mutex_enter(&sc->sc_lock);
+ mutex_enter(&port->lock);
port->status = UPS_CURRENT_CONNECT_STATUS | UPS_PORT_ENABLED |
UPS_PORT_POWER;
port->change = UPS_C_CONNECT_STATUS | UPS_C_PORT_RESET;
+ mutex_exit(&port->lock);
xfer = sc->sc_intrxfer;
@@ -741,7 +750,7 @@
vhci_xfer_t *vxfer;
KASSERT(mutex_owned(&sc->sc_lock));
- mutex_enter(&port->lock);
+ KASSERT(mutex_owned(&port->lock));
TAILQ_INIT(&vxferlist);
@@ -769,12 +778,9 @@
struct usbd_xfer *xfer = &vxfer->xfer;
TAILQ_REMOVE(&vxferlist, vxfer, freelist);
- xfer->ux_actlen = xfer->ux_length;
- xfer->ux_status = USBD_NORMAL_COMPLETION;
+ xfer->ux_status = USBD_TIMEOUT;
usb_transfer_complete(xfer);
}
-
- mutex_exit(&port->lock);
}
static int
@@ -797,6 +803,8 @@
return ENOBUFS;
}
+ mutex_enter(&port->lock);
+
port->status = 0;
port->change = UPS_C_CONNECT_STATUS | UPS_C_PORT_RESET;
@@ -808,8 +816,9 @@
usb_transfer_complete(xfer);
vhci_port_flush(sc, port);
+
+ mutex_exit(&port->lock);
mutex_exit(&sc->sc_lock);
-
return 0;
}
@@ -905,11 +914,13 @@
{
struct vhci_ioc_usb_detach args;
vhci_fd_t *vfd = fp->f_data;
+ int ret __diagused;
KASSERT(vfd != NULL);
args.port = vfd->port;
- vhci_usb_detach(vfd, &args);
+ ret = vhci_usb_detach(vfd, &args);
+ KASSERT(ret == 0);
kmem_free(vfd, sizeof(*vfd));
fp->f_data = NULL;
@@ -1022,6 +1033,7 @@
TAILQ_FOREACH_SAFE(pkt, pktlist, portlist, nxt) {
vxfer = (vhci_xfer_t *)pkt->xfer;
buf = pkt->buf + pkt->cursor;
+ KASSERT(pkt->size >= pkt->cursor);
size = uimin(uio->uio_resid, pkt->size - pkt->cursor);
KASSERT(vxfer->xfer.ux_status == USBD_IN_PROGRESS);
@@ -1087,9 +1099,10 @@
static int vhci_match(device_t, cfdata_t, void *);
static void vhci_attach(device_t, device_t, void *);
+static int vhci_activate(device_t, enum devact);
CFATTACH_DECL_NEW(vhci, sizeof(vhci_softc_t), vhci_match, vhci_attach,
- NULL, NULL);
+ NULL, vhci_activate);
void
vhciattach(int nunits)
@@ -1114,6 +1127,20 @@
}
static int
+vhci_activate(device_t self, enum devact act)
+{
+ vhci_softc_t *sc = device_private(self);
+
+ switch (act) {
+ case DVACT_DEACTIVATE:
+ sc->sc_dying = 1;
+ return 0;
+ default:
+ return EOPNOTSUPP;
+ }
+}
+
+static int
vhci_match(device_t parent, cfdata_t match, void *aux)
{
return 1;
Home |
Main Index |
Thread Index |
Old Index