Source-Changes-HG archive
[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index][Old Index]
[src/trunk]: src/sys/dev/usb ugen(4): Prevent ugenopen while ugen_set_config ...
details: https://anonhg.NetBSD.org/src/rev/1b31be3a0412
branches: trunk
changeset: 1023428:1b31be3a0412
user: riastradh <riastradh%NetBSD.org@localhost>
date: Tue Sep 07 10:42:48 2021 +0000
description:
ugen(4): Prevent ugenopen while ugen_set_config is in progress.
(except on the control endpoint)
Although we hold the kernel lock (which we should eventually change),
we may sleep in usbd_set_config_no at which point ugenopen might
happen and start making use of endpoint state which we'll stomp all
over once usbd_set_config_no returns. Setting sc_is_open[endpt]
while we wait prevents this.
diffstat:
sys/dev/usb/ugen.c | 32 +++++++++++++++++++++++++-------
1 files changed, 25 insertions(+), 7 deletions(-)
diffs (80 lines):
diff -r 3b55d08338ed -r 1b31be3a0412 sys/dev/usb/ugen.c
--- a/sys/dev/usb/ugen.c Tue Sep 07 10:42:34 2021 +0000
+++ b/sys/dev/usb/ugen.c Tue Sep 07 10:42:48 2021 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: ugen.c,v 1.160 2021/09/07 10:42:34 riastradh Exp $ */
+/* $NetBSD: ugen.c,v 1.161 2021/09/07 10:42:48 riastradh Exp $ */
/*
* Copyright (c) 1998, 2004 The NetBSD Foundation, Inc.
@@ -37,7 +37,7 @@
#include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: ugen.c,v 1.160 2021/09/07 10:42:34 riastradh Exp $");
+__KERNEL_RCSID(0, "$NetBSD: ugen.c,v 1.161 2021/09/07 10:42:48 riastradh Exp $");
#ifdef _KERNEL_OPT
#include "opt_compat_netbsd.h"
@@ -455,6 +455,12 @@
device_xname(sc->sc_dev), endptno));
return USBD_IN_USE;
}
+
+ /* Prevent opening while we're setting the config. */
+ for (endptno = 1; endptno < USB_MAX_ENDPOINTS; endptno++) {
+ KASSERT(!sc->sc_is_open[endptno]);
+ sc->sc_is_open[endptno] = 1;
+ }
}
/* Avoid setting the current value. */
@@ -462,23 +468,23 @@
if (!cdesc || cdesc->bConfigurationValue != configno) {
err = usbd_set_config_no(dev, configno, 1);
if (err)
- return err;
+ goto out;
}
ugen_clear_endpoints(sc);
err = usbd_interface_count(dev, &niface);
if (err)
- return err;
+ goto out;
for (ifaceno = 0; ifaceno < niface; ifaceno++) {
DPRINTFN(1,("ugen_set_config: ifaceno %d\n", ifaceno));
err = usbd_device2interface_handle(dev, ifaceno, &iface);
if (err)
- return err;
+ goto out;
err = usbd_endpoint_count(iface, &nendpt);
if (err)
- return err;
+ goto out;
for (endptno = 0; endptno < nendpt; endptno++) {
ed = usbd_interface2endpoint_descriptor(iface,endptno);
KASSERT(ed != NULL);
@@ -494,7 +500,19 @@
sce->iface = iface;
}
}
- return USBD_NORMAL_COMPLETION;
+ err = USBD_NORMAL_COMPLETION;
+
+out: if (chkopen) {
+ /*
+ * Allow open again now that we're done trying to set
+ * the config.
+ */
+ for (endptno = 1; endptno < USB_MAX_ENDPOINTS; endptno++) {
+ KASSERT(sc->sc_is_open[endptno]);
+ sc->sc_is_open[endptno] = 0;
+ }
+ }
+ return err;
}
static int
Home |
Main Index |
Thread Index |
Old Index