summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorTheo de Raadt <deraadt@cvs.openbsd.org>2011-07-03 19:38:51 +0000
committerTheo de Raadt <deraadt@cvs.openbsd.org>2011-07-03 19:38:51 +0000
commit000f5319ca0a4abea494ade0cdd45c92856d4eb8 (patch)
tree72336300431d09f0c4f6551cbabf65c2e3679c04
parente6caee1842efde93c3f8fd1ead5aab263999015c (diff)
ucom(4) did not understand the last-close semantics. repair that, and
add a bit more locking in the open function. originally spotted by matthew, ok matthew miod
-rw-r--r--sys/dev/usb/ucom.c66
1 files changed, 31 insertions, 35 deletions
diff --git a/sys/dev/usb/ucom.c b/sys/dev/usb/ucom.c
index 6d02d518e79..aea1a43fdf4 100644
--- a/sys/dev/usb/ucom.c
+++ b/sys/dev/usb/ucom.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: ucom.c,v 1.53 2011/07/03 15:47:17 matthew Exp $ */
+/* $OpenBSD: ucom.c,v 1.54 2011/07/03 19:38:50 deraadt Exp $ */
/* $NetBSD: ucom.c,v 1.49 2003/01/01 00:10:25 thorpej Exp $ */
/*
@@ -210,8 +210,6 @@ ucom_attach(struct device *parent, struct device *self, void *aux)
sc->sc_cua = 0;
rw_init(&sc->sc_lock, "ucomlk");
-
- sc->sc_open = 0;
}
int
@@ -319,9 +317,8 @@ ucomopen(dev_t dev, int flag, int mode, struct proc *p)
/* open the pipes if this is the first open */
ucom_lock(sc);
- if (sc->sc_open++ == 0) {
- s = splusb();
-
+ s = splusb();
+ if (sc->sc_open == 0) {
DPRINTF(("ucomopen: open pipes in=%d out=%d\n",
sc->sc_bulkin_no, sc->sc_bulkout_no));
DPRINTF(("ucomopen: hid %p pipes in=%p out=%p\n",
@@ -400,12 +397,11 @@ ucomopen(dev_t dev, int flag, int mode, struct proc *p)
ucom_status_change(sc);
ucomstartread(sc);
-
- splx(s);
+ sc->sc_open = 1;
}
- ucom_unlock(sc);
-
+ splx(s);
s = spltty();
+ ucom_unlock(sc);
tp = sc->sc_tty;
splx(s);
@@ -554,8 +550,8 @@ ucomclose(dev_t dev, int flag, int mode, struct proc *p)
CLR(tp->t_state, TS_BUSY | TS_FLUSH);
sc->sc_cua = 0;
ttyclose(tp);
- ucom_cleanup(sc);
splx(s);
+ ucom_cleanup(sc);
if (sc->sc_methods->ucom_close != NULL)
sc->sc_methods->ucom_close(sc->sc_parent, sc->sc_portno);
@@ -1143,31 +1139,31 @@ ucomreadcb(usbd_xfer_handle xfer, usbd_private_handle p, usbd_status status)
void
ucom_cleanup(struct ucom_softc *sc)
{
- if (--sc->sc_open == 0) {
- DPRINTF(("ucom_cleanup: closing pipes\n"));
+ DPRINTF(("ucom_cleanup: closing pipes\n"));
- ucom_shutdown(sc);
- if (sc->sc_bulkin_pipe != NULL) {
- usbd_abort_pipe(sc->sc_bulkin_pipe);
- usbd_close_pipe(sc->sc_bulkin_pipe);
- sc->sc_bulkin_pipe = NULL;
- }
- if (sc->sc_bulkout_pipe != NULL) {
- usbd_abort_pipe(sc->sc_bulkout_pipe);
- usbd_close_pipe(sc->sc_bulkout_pipe);
- sc->sc_bulkout_pipe = NULL;
- }
- if (sc->sc_ixfer != NULL) {
- if (sc->sc_uhidev == NULL)
- usbd_free_xfer(sc->sc_ixfer);
- sc->sc_ixfer = NULL;
- }
- if (sc->sc_oxfer != NULL) {
- usbd_free_buffer(sc->sc_oxfer);
- if (sc->sc_uhidev == NULL)
- usbd_free_xfer(sc->sc_oxfer);
- sc->sc_oxfer = NULL;
- }
+ sc->sc_open = 0;
+
+ ucom_shutdown(sc);
+ if (sc->sc_bulkin_pipe != NULL) {
+ usbd_abort_pipe(sc->sc_bulkin_pipe);
+ usbd_close_pipe(sc->sc_bulkin_pipe);
+ sc->sc_bulkin_pipe = NULL;
+ }
+ if (sc->sc_bulkout_pipe != NULL) {
+ usbd_abort_pipe(sc->sc_bulkout_pipe);
+ usbd_close_pipe(sc->sc_bulkout_pipe);
+ sc->sc_bulkout_pipe = NULL;
+ }
+ if (sc->sc_ixfer != NULL) {
+ if (sc->sc_uhidev == NULL)
+ usbd_free_xfer(sc->sc_ixfer);
+ sc->sc_ixfer = NULL;
+ }
+ if (sc->sc_oxfer != NULL) {
+ usbd_free_buffer(sc->sc_oxfer);
+ if (sc->sc_uhidev == NULL)
+ usbd_free_xfer(sc->sc_oxfer);
+ sc->sc_oxfer = NULL;
}
}