USB: opticon: fix abuse of interface data
authorJohan Hovold <jhovold@gmail.com>
Wed, 25 Apr 2012 13:56:29 +0000 (15:56 +0200)
committerGreg Kroah-Hartman <gregkh@linuxfoundation.org>
Wed, 25 Apr 2012 18:37:18 +0000 (11:37 -0700)
Fix abuse of interface data which was used to signal device disconnect.

Use the usb_serial disconnect flag and mutex where appropriate.

Note that tiocmget does not need to check for disconnect as it does not
access the device.

Signed-off-by: Johan Hovold <jhovold@gmail.com>
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
drivers/usb/serial/opticon.c

index 82cc9d202b832f95b4d2920ebbaff798ae890167..d31e997ebd27d70ac0b09c0f15b19b01e75e4ba7 100644 (file)
@@ -401,8 +401,6 @@ static int opticon_tiocmget(struct tty_struct *tty)
        int result = 0;
 
        dbg("%s - port %d", __func__, port->number);
-       if (!usb_get_intfdata(port->serial->interface))
-               return -ENODEV;
 
        spin_lock_irqsave(&priv->lock, flags);
        if (priv->rts)
@@ -419,13 +417,13 @@ static int opticon_tiocmset(struct tty_struct *tty,
                           unsigned int set, unsigned int clear)
 {
        struct usb_serial_port *port = tty->driver_data;
+       struct usb_serial *serial = port->serial;
        struct opticon_private *priv = usb_get_serial_data(port->serial);
        unsigned long flags;
        bool rts;
        bool changed = false;
+       int ret;
 
-       if (!usb_get_intfdata(port->serial->interface))
-               return -ENODEV;
        /* We only support RTS so we only handle that */
        spin_lock_irqsave(&priv->lock, flags);
 
@@ -441,7 +439,14 @@ static int opticon_tiocmset(struct tty_struct *tty,
                return 0;
 
        /* Send the new RTS state to the connected device */
-       return send_control_msg(port, CONTROL_RTS, !rts);
+       mutex_lock(&serial->disc_mutex);
+       if (!serial->disconnected)
+               ret = send_control_msg(port, CONTROL_RTS, !rts);
+       else
+               ret = -ENODEV;
+       mutex_unlock(&serial->disc_mutex);
+
+       return ret;
 }
 
 static int get_serial_info(struct opticon_private *priv,