diff options
author | Theo de Raadt <deraadt@cvs.openbsd.org> | 2011-07-03 19:38:51 +0000 |
---|---|---|
committer | Theo de Raadt <deraadt@cvs.openbsd.org> | 2011-07-03 19:38:51 +0000 |
commit | 000f5319ca0a4abea494ade0cdd45c92856d4eb8 (patch) | |
tree | 72336300431d09f0c4f6551cbabf65c2e3679c04 /sys/dev/usb | |
parent | e6caee1842efde93c3f8fd1ead5aab263999015c (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
Diffstat (limited to 'sys/dev/usb')
-rw-r--r-- | sys/dev/usb/ucom.c | 66 |
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; } } |