On Tue, Apr 08, 2014 at 11:05:20AM +0800, Xiao Jin wrote:
> We find two problems on acm tty write delayed mechanism.

Then you should split this into two patches.

> (1) When acm resume, the delayed wb will be started. But now
> only one write can be saved during acm suspend. More acm write
> may be abandoned.

Why not simply return 0 in write and use the tty buffering rather than
implement another buffer in cdc_acm?

> (2) acm tty port ASYNCB_INITIALIZED flag will be cleared when
> close. If acm resume callback run after ASYNCB_INITIALIZED flag
> cleared, there will have no chance for delayed write to start.
> That lead to acm_wb.use can't be cleared. If user space open
> acm tty again and try to setd, tty will be blocked in
> tty_wait_until_sent for ever.
>
> This patch have two modification.
> (1) use list_head to save the write acm_wb during acm suspend.
> It can ensure no acm write abandoned.
> (2) enable flush buffer callback when acm tty close. acm delayed
> wb will start before acm port shutdown callback, it make sure
> the delayed acm_wb.use to be cleared. The patch also clear
> acm_wb.use and acm.transmitting when port shutdown.

This is not the right way to do this. See below.

> Signed-off-by: xiao jin <jin.x...@intel.com>
> Reviewed-by: David Cohen <david.a.co...@linux.intel.com>
> ---
>   drivers/usb/class/cdc-acm.c |   56 
> ++++++++++++++++++++++++++++++-------------
>   drivers/usb/class/cdc-acm.h |    3 ++-
>   2 files changed, 42 insertions(+), 17 deletions(-)
> 
> diff --git a/drivers/usb/class/cdc-acm.c b/drivers/usb/class/cdc-acm.c
> index 900f7ff..cfe00b2 100644
> --- a/drivers/usb/class/cdc-acm.c
> +++ b/drivers/usb/class/cdc-acm.c
> @@ -217,6 +217,24 @@ static int acm_start_wb(struct acm *acm, struct 
> acm_wb *wb)
>   }
> 
>   /*
> + * Delayed write.
> + */
> +static int acm_start_delayed_wb(struct acm *acm)
> +{
> +     struct acm_wb *wb, *n_wb;
> +     LIST_HEAD(delay_wb_list);
> +
> +     spin_lock_irq(&acm->write_lock);
> +     list_replace_init(&acm->delayed_wb_list, &delay_wb_list);
> +     spin_unlock_irq(&acm->write_lock);
> +     list_for_each_entry_safe(wb, n_wb,
> +                     &delay_wb_list, delay) {
> +             list_del(&wb->delay);
> +             acm_start_wb(acm, wb);
> +     }
> +}
> +
> +/*
>    * attributes exported through sysfs
>    */
>   static ssize_t show_caps
> @@ -580,8 +598,11 @@ static void acm_port_shutdown(struct tty_port *port)
>               usb_autopm_get_interface(acm->control);
>               acm_set_control(acm, acm->ctrlout = 0);
>               usb_kill_urb(acm->ctrlurb);
> -             for (i = 0; i < ACM_NW; i++)
> +             acm->transmitting = 0;

No need to reset transmitting which is balanced by usb_kill_urb below.

> +             for (i = 0; i < ACM_NW; i++) {
>                       usb_kill_urb(acm->wb[i].urb);
> +                     acm->wb[i].use = 0;

Same here: use is generally cleared by the completion handler -- you
only need to handle any delayed urb.

> +             }
>               for (i = 0; i < acm->rx_buflimit; i++)
>                       usb_kill_urb(acm->read_urbs[i]);
>               acm->control->needs_remote_wakeup = 0;
> @@ -608,6 +629,8 @@ static void acm_tty_close(struct tty_struct *tty, 
> struct file *filp)
>   {
>       struct acm *acm = tty->driver_data;
>       dev_dbg(&acm->control->dev, "%s\n", __func__);
> +     /* Set flow_stopped to enable flush buffer*/
> +     tty->flow_stopped = 1;

You shouldn't abuse the flow_stopped flag. Simply clear the delayed urb
in shutdown (rather than try to use flush_buffer).

>       tty_port_close(&acm->port, tty, filp);
>   }
> 
> @@ -646,10 +669,7 @@ static int acm_tty_write(struct tty_struct *tty,
> 
>       usb_autopm_get_interface_async(acm->control);
>       if (acm->susp_count) {
> -             if (!acm->delayed_wb)
> -                     acm->delayed_wb = wb;
> -             else
> -                     usb_autopm_put_interface_async(acm->control);
> +             list_add_tail(&wb->delay, &acm->delayed_wb_list);
>               spin_unlock_irqrestore(&acm->write_lock, flags);
>               return count;   /* A white lie */
>       }
> @@ -688,6 +708,18 @@ static int acm_tty_chars_in_buffer(struct 
> tty_struct *tty)
>       return (ACM_NW - acm_wb_is_avail(acm)) * acm->writesize;
>   }
> 
> +static void acm_tty_flush_buffer(struct tty_struct *tty)
> +{
> +     struct acm *acm = tty->driver_data;
> +
> +     /* flush delayed write buffer */
> +     if (!acm->disconnected) {
> +             usb_autopm_get_interface(acm->control);
> +             acm_start_delayed_wb(acm);
> +             usb_autopm_put_interface(acm->control);
> +     }
> +}
> +

Again, don't use flush_buffer for this (it's used to discard output
buffers).

Johan

>   static void acm_tty_throttle(struct tty_struct *tty)
>   {
>       struct acm *acm = tty->driver_data;
> @@ -1346,6 +1378,7 @@ made_compressed_probe:
>               snd->urb->transfer_flags |= URB_NO_TRANSFER_DMA_MAP;
>               snd->instance = acm;
>       }
> +     INIT_LIST_HEAD(&acm->delayed_wb_list);
> 
>       usb_set_intfdata(intf, acm);
> 
> @@ -1540,7 +1573,6 @@ static int acm_suspend(struct usb_interface *intf, 
> pm_message_t message)
>   static int acm_resume(struct usb_interface *intf)
>   {
>       struct acm *acm = usb_get_intfdata(intf);
> -     struct acm_wb *wb;
>       int rv = 0;
>       int cnt;
> 
> @@ -1555,16 +1587,7 @@ static int acm_resume(struct usb_interface *intf)
>       if (test_bit(ASYNCB_INITIALIZED, &acm->port.flags)) {
>               rv = usb_submit_urb(acm->ctrlurb, GFP_NOIO);
> 
> -             spin_lock_irq(&acm->write_lock);
> -             if (acm->delayed_wb) {
> -                     wb = acm->delayed_wb;
> -                     acm->delayed_wb = NULL;
> -                     spin_unlock_irq(&acm->write_lock);
> -                     acm_start_wb(acm, wb);
> -             } else {
> -                     spin_unlock_irq(&acm->write_lock);
> -             }
> -
> +             acm_start_delayed_wb(acm);
>               /*
>                * delayed error checking because we must
>                * do the write path at all cost
> @@ -1823,6 +1846,7 @@ static const struct tty_operations acm_ops = {
>       .throttle =             acm_tty_throttle,
>       .unthrottle =           acm_tty_unthrottle,
>       .chars_in_buffer =      acm_tty_chars_in_buffer,
> +     .flush_buffer =         acm_tty_flush_buffer,
>       .break_ctl =            acm_tty_break_ctl,
>       .set_termios =          acm_tty_set_termios,
>       .tiocmget =             acm_tty_tiocmget,
> diff --git a/drivers/usb/class/cdc-acm.h b/drivers/usb/class/cdc-acm.h
> index e38dc78..42d6427 100644
> --- a/drivers/usb/class/cdc-acm.h
> +++ b/drivers/usb/class/cdc-acm.h
> @@ -69,6 +69,7 @@ struct acm_wb {
>       int use;
>       struct urb              *urb;
>       struct acm              *instance;
> +     struct list_head        delay;
>   };
> 
>   struct acm_rb {
> @@ -120,7 +121,7 @@ struct acm {
>       unsigned int throttled:1;                       /* actually throttled */
>       unsigned int throttle_req:1;                    /* throttle requested */
>       u8 bInterval;
> -     struct acm_wb *delayed_wb;                      /* write queued for a 
> device about to be 
> woken */
> +     struct list_head delayed_wb_list;               /* delayed wb list */
>   };
> 
>   #define CDC_DATA_INTERFACE_TYPE     0x0a
--
To unsubscribe from this list: send the line "unsubscribe linux-kernel" in
the body of a message to majord...@vger.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html
Please read the FAQ at  http://www.tux.org/lkml/

Reply via email to