Re: [PATCH] cdc-acm: some enhancement on acm delayed write

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



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.xiao@xxxxxxxxx>
> Reviewed-by: David Cohen <david.a.cohen@xxxxxxxxxxxxxxx>
> ---
>   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-usb" in
the body of a message to majordomo@xxxxxxxxxxxxxxx
More majordomo info at  http://vger.kernel.org/majordomo-info.html




[Index of Archives]     [Linux Media]     [Linux Input]     [Linux Audio Users]     [Yosemite News]     [Linux Kernel]     [Linux SCSI]     [Old Linux USB Devel Archive]

  Powered by Linux