summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--sys/dev/usb/ucom.c225
1 files changed, 110 insertions, 115 deletions
diff --git a/sys/dev/usb/ucom.c b/sys/dev/usb/ucom.c
index 97793f96eb2..7b719408e7b 100644
--- a/sys/dev/usb/ucom.c
+++ b/sys/dev/usb/ucom.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: ucom.c,v 1.22 2005/01/10 08:17:49 dlg Exp $ */
+/* $OpenBSD: ucom.c,v 1.23 2005/01/19 07:49:35 dlg Exp $ */
/* $NetBSD: ucom.c,v 1.49 2003/01/01 00:10:25 thorpej Exp $ */
/*
@@ -118,7 +118,8 @@ struct ucom_softc {
u_char sc_cua;
- u_char sc_opening; /* lock during open */
+ struct lock sc_lock; /* lock during open */
+ int sc_open;
int sc_refcnt;
u_char sc_dying; /* disconnecting */
};
@@ -138,9 +139,26 @@ Static void ucomreadcb(usbd_xfer_handle, usbd_private_handle, usbd_status);
Static void ucomwritecb(usbd_xfer_handle, usbd_private_handle, usbd_status);
Static void tiocm_to_ucom(struct ucom_softc *, u_long, int);
Static int ucom_to_tiocm(struct ucom_softc *);
+Static void ucom_lock(struct ucom_softc *);
+Static void ucom_unlock(struct ucom_softc *);
USB_DECLARE_DRIVER(ucom);
+Static void
+ucom_lock(struct ucom_softc *sc)
+{
+ sc->sc_refcnt++;
+ usb_lockmgr(&sc->sc_lock, LK_EXCLUSIVE, NULL, curproc);
+}
+
+Static void
+ucom_unlock(struct ucom_softc *sc)
+{
+ usb_lockmgr(&sc->sc_lock, LK_RELEASE, NULL, curproc);
+ if (--sc->sc_refcnt < 0)
+ usb_detach_wakeup(USBDEV(sc->sc_dev));
+}
+
USB_MATCH(ucom)
{
return (1);
@@ -176,6 +194,9 @@ USB_ATTACH(ucom)
sc->sc_tty = tp;
sc->sc_cua = 0;
+ lockinit(&sc->sc_lock, PZERO, "ucomlk", 0, LK_CANRECURSE);
+ sc->sc_open = 0;
+
USB_ATTACH_SUCCESS_RETURN;
}
@@ -286,75 +307,24 @@ ucomopen(dev_t dev, int flag, int mode, usb_proc_ptr p)
if (ISSET(sc->sc_dev.dv_flags, DVF_ACTIVE) == 0)
return (ENXIO);
- tp = sc->sc_tty;
- DPRINTF(("ucomopen: unit=%d, tp=%p\n", unit, tp));
-
- /*
- * Do the following if this is a first open.
- */
- while (sc->sc_opening)
- tsleep(&sc->sc_opening, PRIBIO, "ucomop", 0);
-
- if (sc->sc_dying) {
- return (EIO);
- }
- sc->sc_opening = 1;
-
- tp->t_dev = dev;
- if (!ISSET(tp->t_state, TS_ISOPEN)) {
- SET(tp->t_state, TS_WOPEN);
+ /* open the pipes if this is the first open */
+ ucom_lock(sc);
+ if (sc->sc_open++ == 0) {
+ s = splusb();
if (sc->sc_methods->ucom_open != NULL) {
error = sc->sc_methods->ucom_open(sc->sc_parent,
sc->sc_portno);
if (error) {
ucom_cleanup(sc);
- sc->sc_opening = 0;
- wakeup(&sc->sc_opening);
splx(s);
+ ucom_unlock(sc);
return (error);
}
}
ucom_status_change(sc);
- /*
- * Initialize the termios status to the defaults. Add in the
- * sticky bits from TIOCSFLAGS.
- */
- t.c_ispeed = 0;
- t.c_ospeed = TTYDEF_SPEED;
- t.c_cflag = TTYDEF_CFLAG;
- if (ISSET(sc->sc_swflags, TIOCFLAG_CLOCAL))
- SET(t.c_cflag, CLOCAL);
- if (ISSET(sc->sc_swflags, TIOCFLAG_CRTSCTS))
- SET(t.c_cflag, CRTSCTS);
- if (ISSET(sc->sc_swflags, TIOCFLAG_MDMBUF))
- SET(t.c_cflag, MDMBUF);
-
- /* Make sure ucomparam() will do something. */
- tp->t_ospeed = 0;
- (void) ucomparam(tp, &t);
- tp->t_iflag = TTYDEF_IFLAG;
- tp->t_oflag = TTYDEF_OFLAG;
- tp->t_lflag = TTYDEF_LFLAG;
- ttychars(tp);
- ttsetwater(tp);
-
- s = spltty();
-
- /*
- * Turn on DTR. We must always do this, even if carrier is not
- * present, because otherwise we'd have to use TIOCSDTR
- * immediately after setting CLOCAL, which applications do not
- * expect. We always assert DTR while the device is open
- * unless explicitly requested to deassert it.
- */
- //ucom_dtr(sc, 1);
-
- /* XXX CLR(sc->sc_rx_flags, RX_ANY_BLOCK);*/
- ucom_hwiflow(sc);
-
DPRINTF(("ucomopen: open pipes in=%d out=%d\n",
sc->sc_bulkin_no, sc->sc_bulkout_no));
@@ -408,30 +378,72 @@ ucomopen(dev_t dev, int flag, int mode, usb_proc_ptr p)
ucomstartread(sc);
+ splx(s);
+ }
+ ucom_unlock(sc);
+
+ s = spltty();
+ tp = sc->sc_tty;
+ splx(s);
+
+ DPRINTF(("ucomopen: unit=%d, tp=%p\n", unit, tp));
+
+ tp->t_dev = dev;
+ if (!ISSET(tp->t_state, TS_ISOPEN)) {
+ SET(tp->t_state, TS_WOPEN);
+ ttychars(tp);
+
+ /*
+ * Initialize the termios status to the defaults. Add in the
+ * sticky bits from TIOCSFLAGS.
+ */
+ t.c_ispeed = 0;
+ t.c_ospeed = TTYDEF_SPEED;
+ t.c_cflag = TTYDEF_CFLAG;
+ if (ISSET(sc->sc_swflags, TIOCFLAG_CLOCAL))
+ SET(t.c_cflag, CLOCAL);
+ if (ISSET(sc->sc_swflags, TIOCFLAG_CRTSCTS))
+ SET(t.c_cflag, CRTSCTS);
+ if (ISSET(sc->sc_swflags, TIOCFLAG_MDMBUF))
+ SET(t.c_cflag, MDMBUF);
+
+ /* Make sure ucomparam() will do something. */
+ tp->t_ospeed = 0;
+ (void) ucomparam(tp, &t);
+ tp->t_iflag = TTYDEF_IFLAG;
+ tp->t_oflag = TTYDEF_OFLAG;
+ tp->t_lflag = TTYDEF_LFLAG;
+
+ s = spltty();
+ ttsetwater(tp);
+
+ /* XXX CLR(sc->sc_rx_flags, RX_ANY_BLOCK);*/
+ ucom_hwiflow(sc);
+
if (ISSET(sc->sc_swflags, TIOCFLAG_SOFTCAR) || UCOMCUA(dev) ||
ISSET(sc->sc_msr, UMSR_DCD) || ISSET(tp->t_cflag, MDMBUF))
SET(tp->t_state, TS_CARR_ON);
else
CLR(tp->t_state, TS_CARR_ON);
-
- } else if (ISSET(tp->t_state, TS_XCLUDE) && p->p_ucred->cr_uid != 0)
- return (EBUSY);
- else
+ } else if (ISSET(tp->t_state, TS_XCLUDE) && p->p_ucred->cr_uid != 0) {
+ error = EBUSY;
+ goto bad;
+ } else
s = spltty();
if (UCOMCUA(dev)) {
if (ISSET(tp->t_state, TS_ISOPEN)) {
/* Someone is already dialed in */
- splx(s);
- return (EBUSY);
+ error = EBUSY;
+ goto bad1;
}
sc->sc_cua = 1;
} else {
/* tty (not cua) device, wait for carrier */
if (ISSET(flag, O_NONBLOCK)) {
if (sc->sc_cua) {
- splx(s);
- return (EBUSY);
+ error = EBUSY;
+ goto bad1;
}
} else {
while (sc->sc_cua || (!ISSET(tp->t_cflag, CLOCAL) &&
@@ -447,21 +459,13 @@ ucomopen(dev_t dev, int flag, int mode, usb_proc_ptr p)
*/
if (error && ISSET(tp->t_state, TS_WOPEN)) {
CLR(tp->t_state, TS_WOPEN);
- if (!sc->sc_cua &&
- !ISSET(tp->t_state, TS_ISOPEN))
- ucom_cleanup(sc);
- splx(s);
- return (error);
+ goto bad1;
}
}
}
}
-
- sc->sc_opening = 0;
- wakeup(&sc->sc_opening);
splx(s);
-
error = ttyopen(UCOMUNIT(dev), tp);
if (error)
goto bad;
@@ -485,19 +489,16 @@ fail_1:
usbd_close_pipe(sc->sc_bulkin_pipe);
sc->sc_bulkin_pipe = NULL;
fail_0:
- sc->sc_opening = 0;
- wakeup(&sc->sc_opening);
splx(s);
+ ucom_unlock(sc);
return (error);
+bad1:
+ splx(s);
bad:
- if (!ISSET(tp->t_state, TS_ISOPEN)) {
- /*
- * We failed to open the device, and nobody else had it opened.
- * Clean up the state as appropriate.
- */
- ucom_cleanup(sc);
- }
+ ucom_lock(sc);
+ ucom_cleanup(sc);
+ ucom_unlock(sc);
return (error);
}
@@ -513,18 +514,11 @@ ucomclose(dev_t dev, int flag, int mode, usb_proc_ptr p)
if (!ISSET(tp->t_state, TS_ISOPEN))
return (0);
- sc->sc_refcnt++;
+ ucom_lock(sc);
(*LINESW(tp, l_close))(tp, flag);
s = spltty();
- if (!ISSET(tp->t_state, TS_WOPEN)) {
- /*
- * Although we got a last close, the device may still be in
- * use; e.g. if this was the dialout node, and there are still
- * processes waiting for carrier on the non-dialout node.
- */
- ucom_cleanup(sc);
- }
+ ucom_cleanup(sc);
CLR(tp->t_state, TS_BUSY | TS_FLUSH);
sc->sc_cua = 0;
splx(s);
@@ -533,8 +527,7 @@ ucomclose(dev_t dev, int flag, int mode, usb_proc_ptr p)
if (sc->sc_methods->ucom_close != NULL)
sc->sc_methods->ucom_close(sc->sc_parent, sc->sc_portno);
- if (--sc->sc_refcnt < 0)
- usb_detach_wakeup(USBDEV(sc->sc_dev));
+ ucom_unlock(sc);
return (0);
}
@@ -1109,26 +1102,28 @@ ucomreadcb(usbd_xfer_handle xfer, usbd_private_handle p, usbd_status status)
Static void
ucom_cleanup(struct ucom_softc *sc)
{
- 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) {
- usbd_free_xfer(sc->sc_ixfer);
- sc->sc_ixfer = NULL;
- }
- if (sc->sc_oxfer != NULL) {
- usbd_free_xfer(sc->sc_oxfer);
- sc->sc_oxfer = NULL;
+ if (--sc->sc_open == 0) {
+ 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) {
+ usbd_free_xfer(sc->sc_ixfer);
+ sc->sc_ixfer = NULL;
+ }
+ if (sc->sc_oxfer != NULL) {
+ usbd_free_xfer(sc->sc_oxfer);
+ sc->sc_oxfer = NULL;
+ }
}
}