]> git.proxmox.com Git - mirror_ubuntu-artful-kernel.git/blobdiff - drivers/usb/serial/io_ti.c
Merge tag 'tty-3.9-rc1' of git://git.kernel.org/pub/scm/linux/kernel/git/gregkh/tty
[mirror_ubuntu-artful-kernel.git] / drivers / usb / serial / io_ti.c
index b5ab0a54b6b6624195419de00a6054f075c682bd..c23776679f7029a37598e5cea826060670be7e64 100644 (file)
@@ -521,65 +521,6 @@ exit_is_tx_active:
        return bytes_left;
 }
 
-static void chase_port(struct edgeport_port *port, unsigned long timeout,
-                                                               int flush)
-{
-       int baud_rate;
-       struct tty_struct *tty = tty_port_tty_get(&port->port->port);
-       struct usb_serial *serial = port->port->serial;
-       wait_queue_t wait;
-       unsigned long flags;
-
-       if (!tty)
-               return;
-
-       if (!timeout)
-               timeout = (HZ * EDGE_CLOSING_WAIT)/100;
-
-       /* wait for data to drain from the buffer */
-       spin_lock_irqsave(&port->ep_lock, flags);
-       init_waitqueue_entry(&wait, current);
-       add_wait_queue(&tty->write_wait, &wait);
-       for (;;) {
-               set_current_state(TASK_INTERRUPTIBLE);
-               if (kfifo_len(&port->write_fifo) == 0
-               || timeout == 0 || signal_pending(current)
-               || serial->disconnected)
-                       /* disconnect */
-                       break;
-               spin_unlock_irqrestore(&port->ep_lock, flags);
-               timeout = schedule_timeout(timeout);
-               spin_lock_irqsave(&port->ep_lock, flags);
-       }
-       set_current_state(TASK_RUNNING);
-       remove_wait_queue(&tty->write_wait, &wait);
-       if (flush)
-               kfifo_reset_out(&port->write_fifo);
-       spin_unlock_irqrestore(&port->ep_lock, flags);
-       tty_kref_put(tty);
-
-       /* wait for data to drain from the device */
-       timeout += jiffies;
-       while ((long)(jiffies - timeout) < 0 && !signal_pending(current)
-                                               && !serial->disconnected) {
-               /* not disconnected */
-               if (!tx_active(port))
-                       break;
-               msleep(10);
-       }
-
-       /* disconnected */
-       if (serial->disconnected)
-               return;
-
-       /* wait one more character time, based on baud rate */
-       /* (tx_active doesn't seem to wait for the last byte) */
-       baud_rate = port->baud_rate;
-       if (baud_rate == 0)
-               baud_rate = 50;
-       msleep(max(1, DIV_ROUND_UP(10000, baud_rate)));
-}
-
 static int choose_config(struct usb_device *dev)
 {
        /*
@@ -1936,6 +1877,8 @@ static int edge_open(struct tty_struct *tty, struct usb_serial_port *port)
 
        ++edge_serial->num_ports_open;
 
+       port->port.drain_delay = 1;
+
        goto release_es_lock;
 
 unlink_int_urb:
@@ -1951,6 +1894,7 @@ static void edge_close(struct usb_serial_port *port)
        struct edgeport_serial *edge_serial;
        struct edgeport_port *edge_port;
        struct usb_serial *serial = port->serial;
+       unsigned long flags;
        int port_number;
 
        edge_serial = usb_get_serial_data(port->serial);
@@ -1962,12 +1906,12 @@ static void edge_close(struct usb_serial_port *port)
         * this flag and dump add read data */
        edge_port->close_pending = 1;
 
-       /* chase the port close and flush */
-       chase_port(edge_port, (HZ * closing_wait) / 100, 1);
-
        usb_kill_urb(port->read_urb);
        usb_kill_urb(port->write_urb);
        edge_port->ep_write_urb_in_use = 0;
+       spin_lock_irqsave(&edge_port->ep_lock, flags);
+       kfifo_reset_out(&edge_port->write_fifo);
+       spin_unlock_irqrestore(&edge_port->ep_lock, flags);
 
        /* assuming we can still talk to the device,
         * send a close port command to it */
@@ -2093,16 +2037,21 @@ static int edge_chars_in_buffer(struct tty_struct *tty)
        struct edgeport_port *edge_port = usb_get_serial_port_data(port);
        int chars = 0;
        unsigned long flags;
+       int ret;
 
        if (edge_port == NULL)
                return 0;
-       if (edge_port->close_pending == 1)
-               return 0;
 
        spin_lock_irqsave(&edge_port->ep_lock, flags);
        chars = kfifo_len(&edge_port->write_fifo);
        spin_unlock_irqrestore(&edge_port->ep_lock, flags);
 
+       if (!chars) {
+               ret = tx_active(edge_port);
+               if (ret > 0)
+                       chars = ret;
+       }
+
        dev_dbg(&port->dev, "%s - returns %d\n", __func__, chars);
        return chars;
 }
@@ -2440,10 +2389,15 @@ static int get_serial_info(struct edgeport_port *edge_port,
                                struct serial_struct __user *retinfo)
 {
        struct serial_struct tmp;
+       unsigned cwait;
 
        if (!retinfo)
                return -EFAULT;
 
+       cwait = edge_port->port->port.closing_wait;
+       if (cwait != ASYNC_CLOSING_WAIT_NONE)
+               cwait = jiffies_to_msecs(closing_wait) / 10;
+
        memset(&tmp, 0, sizeof(tmp));
 
        tmp.type                = PORT_16550A;
@@ -2454,7 +2408,7 @@ static int get_serial_info(struct edgeport_port *edge_port,
        tmp.xmit_fifo_size      = edge_port->port->bulk_out_size;
        tmp.baud_base           = 9600;
        tmp.close_delay         = 5*HZ;
-       tmp.closing_wait        = closing_wait;
+       tmp.closing_wait        = cwait;
 
        if (copy_to_user(retinfo, &tmp, sizeof(*retinfo)))
                return -EFAULT;
@@ -2509,8 +2463,7 @@ static void edge_break(struct tty_struct *tty, int break_state)
        int status;
        int bv = 0;     /* Off */
 
-       /* chase the port close */
-       chase_port(edge_port, 0, 0);
+       tty_wait_until_sent(tty, 0);
 
        if (break_state == -1)
                bv = 1; /* On */
@@ -2583,6 +2536,8 @@ static int edge_port_probe(struct usb_serial_port *port)
                return ret;
        }
 
+       port->port.closing_wait = msecs_to_jiffies(closing_wait * 10);
+
        return 0;
 }