Source-Changes-HG archive
[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index][Old Index]
[src/trunk]: src/sys/dev/usb Replace the workque for status changes with an u...
details: https://anonhg.NetBSD.org/src/rev/9bc0a8b47c00
branches: trunk
changeset: 327974:9bc0a8b47c00
user: martin <martin%NetBSD.org@localhost>
date: Sat Mar 22 20:56:04 2014 +0000
description:
Replace the workque for status changes with an usb task queue (hint from
Nick Hudson) - simpler for this, and avoids some races.
diffstat:
sys/dev/usb/umcs.c | 47 ++++++++++++++++++++---------------------------
1 files changed, 20 insertions(+), 27 deletions(-)
diffs (157 lines):
diff -r 87f358a6ce38 -r 9bc0a8b47c00 sys/dev/usb/umcs.c
--- a/sys/dev/usb/umcs.c Sat Mar 22 20:36:45 2014 +0000
+++ b/sys/dev/usb/umcs.c Sat Mar 22 20:56:04 2014 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: umcs.c,v 1.4 2014/03/17 21:21:57 martin Exp $ */
+/* $NetBSD: umcs.c,v 1.5 2014/03/22 20:56:04 martin Exp $ */
/* $FreeBSD: head/sys/dev/usb/serial/umcs.c 260559 2014-01-12 11:44:28Z hselasky $ */
/*-
@@ -41,7 +41,7 @@
*
*/
#include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: umcs.c,v 1.4 2014/03/17 21:21:57 martin Exp $");
+__KERNEL_RCSID(0, "$NetBSD: umcs.c,v 1.5 2014/03/22 20:56:04 martin Exp $");
#include <sys/param.h>
#include <sys/systm.h>
@@ -51,7 +51,6 @@
#include <sys/tty.h>
#include <sys/device.h>
#include <sys/kmem.h>
-#include <sys/workqueue.h>
#include <dev/usb/usb.h>
#include <dev/usb/usbdi.h>
@@ -79,7 +78,9 @@
*/
struct umcs7840_softc_oneport {
device_t sc_port_ucom; /* ucom subdevice */
- uint32_t sc_port_changed; /* call ucom_status_change() */
+ volatile uint32_t /* changes for this port have */
+ sc_port_changed; /* been signaled,
+ call ucom_status_change() */
unsigned int sc_port_phys; /* physical port number */
uint8_t sc_port_lcr; /* local line control register */
uint8_t sc_port_mcr; /* local modem control register */
@@ -92,8 +93,7 @@
usbd_pipe_handle sc_intr_pipe; /* interrupt pipe */
uint8_t *sc_intr_buf; /* buffer for interrupt xfer */
unsigned int sc_intr_buflen; /* size of buffer */
- struct workqueue *sc_change_wq; /* workqueue for status changes */
- struct work sc_work; /* work for said workqueue. */
+ struct usb_task sc_change_task; /* async status changes */
struct umcs7840_softc_oneport sc_ports[UMCS7840_MAX_PORTS];
/* data for each port */
uint8_t sc_numports; /* number of ports (subunits) */
@@ -114,7 +114,7 @@
static void umcs7840_attach(device_t, device_t, void *);
static int umcs7840_detach(device_t, int);
static void umcs7840_intr(usbd_xfer_handle xfer, usbd_private_handle priv, usbd_status status);
-static void umcs7840_change_worker(struct work *w, void *arg);
+static void umcs7840_change_task(void *arg);
static int umcs7840_activate(device_t, enum devact);
static void umcs7840_childdet(device_t, device_t);
@@ -294,11 +294,8 @@
return;
}
- if (workqueue_create(&sc->sc_change_wq, "umcsq",
- umcs7840_change_worker, sc, PRI_NONE, IPL_USB, WQ_MPSAFE)) {
- aprint_error_dev(self, "workqueue creation failed\n");
- return;
- }
+ usb_init_task(&sc->sc_change_task, umcs7840_change_task, sc,
+ USB_TASKQ_MPSAFE);
usbd_add_drv_event(USB_EVENT_DRIVER_ATTACH, sc->sc_udev,
sc->sc_dev);
@@ -534,8 +531,7 @@
kmem_free(sc->sc_intr_buf, sc->sc_intr_buflen);
sc->sc_intr_pipe = NULL;
}
- if (sc->sc_change_wq != NULL)
- workqueue_destroy(sc->sc_change_wq);
+ usb_rem_task(sc->sc_udev, &sc->sc_change_task);
/* detach children */
for (i = 0; i < sc->sc_numports; i++) {
@@ -856,8 +852,6 @@
int ctrlreg = umcs7840_reg_ctrl(pn);
uint8_t data;
- atomic_swap_32(&sc->sc_ports[portno].sc_port_changed, 0);
-
if (sc->sc_dying)
return;
@@ -879,13 +873,12 @@
struct umcs7840_softc *sc = priv;
u_char *buf = sc->sc_intr_buf;
int actlen;
- int subunit, found;
+ int subunit;
if (status == USBD_NOT_STARTED || status == USBD_CANCELLED
|| status == USBD_IOERROR)
return;
- found = 0;
if (status != USBD_NORMAL_COMPLETION) {
aprint_error_dev(sc->sc_dev,
"umcs7840_intr: abnormal status: %s\n",
@@ -896,6 +889,7 @@
usbd_get_xfer_status(xfer, NULL, NULL, &actlen, NULL);
if (actlen == 5 || actlen == 13) {
+ usb_rem_task(sc->sc_udev, &sc->sc_change_task);
/* Check status of all ports */
for (subunit = 0; subunit < sc->sc_numports; subunit++) {
uint8_t pn = sc->sc_ports[subunit].sc_port_phys;
@@ -910,19 +904,17 @@
case MCS7840_UART_ISR_RXHASDATA:
case MCS7840_UART_ISR_RXTIMEOUT:
case MCS7840_UART_ISR_MSCHANGE:
- atomic_swap_32(
- &sc->sc_ports[subunit].sc_port_changed,
- 1);
- found++;
+ sc->sc_ports[subunit].sc_port_changed = 1;
break;
default:
/* Do nothing */
break;
}
}
- if (found)
- workqueue_enqueue(sc->sc_change_wq, &sc->sc_work,
- NULL);
+
+ membar_exit();
+ usb_add_task(sc->sc_udev, &sc->sc_change_task,
+ USB_TASKQ_DRIVER);
} else {
aprint_error_dev(sc->sc_dev,
"Invalid interrupt data length %d", actlen);
@@ -930,16 +922,17 @@
}
static void
-umcs7840_change_worker(struct work *w, void *arg)
+umcs7840_change_task(void *arg)
{
struct umcs7840_softc *sc = arg;
int i;
for (i = 0; i < sc->sc_numports; i++) {
if (sc->sc_ports[i].sc_port_changed) {
+ sc->sc_ports[i].sc_port_changed = 0;
+ membar_exit();
ucom_status_change(device_private(
sc->sc_ports[i].sc_port_ucom));
- atomic_swap_32(&sc->sc_ports[i].sc_port_changed, 0);
}
}
}
Home |
Main Index |
Thread Index |
Old Index