forked from mirrors/linux
		
	USB: support for autosuspend in sierra while online
This implements support for autosuspend in the sierra driver while online. Remote wakeup is used for reception. Transmission is facilitated with a queue and the asynchronous autopm mechanism. To prevent races a private flag for opened ports and a counter of running transmissions needs to be added. Signed-off-by: Oliver Neukum <oliver@neukum.org> Tested-by: Elina Pasheva <epasheva@sierrawireless.com> Signed-off-by: Greg Kroah-Hartman <gregkh@suse.de>
This commit is contained in:
		
							parent
							
								
									ad45f1dc83
								
							
						
					
					
						commit
						e6929a9020
					
				
					 1 changed files with 152 additions and 5 deletions
				
			
		| 
						 | 
					@ -51,6 +51,12 @@ struct sierra_iface_info {
 | 
				
			||||||
	const u8  *ifaceinfo;	/* pointer to the array holding the numbers */
 | 
						const u8  *ifaceinfo;	/* pointer to the array holding the numbers */
 | 
				
			||||||
};
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					struct sierra_intf_private {
 | 
				
			||||||
 | 
						spinlock_t susp_lock;
 | 
				
			||||||
 | 
						unsigned int suspended:1;
 | 
				
			||||||
 | 
						int in_flight;
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
static int sierra_set_power_state(struct usb_device *udev, __u16 swiState)
 | 
					static int sierra_set_power_state(struct usb_device *udev, __u16 swiState)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	int result;
 | 
						int result;
 | 
				
			||||||
| 
						 | 
					@ -144,6 +150,7 @@ static int sierra_probe(struct usb_serial *serial,
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	int result = 0;
 | 
						int result = 0;
 | 
				
			||||||
	struct usb_device *udev;
 | 
						struct usb_device *udev;
 | 
				
			||||||
 | 
						struct sierra_intf_private *data;
 | 
				
			||||||
	u8 ifnum;
 | 
						u8 ifnum;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	udev = serial->dev;
 | 
						udev = serial->dev;
 | 
				
			||||||
| 
						 | 
					@ -171,6 +178,11 @@ static int sierra_probe(struct usb_serial *serial,
 | 
				
			||||||
		return -ENODEV;
 | 
							return -ENODEV;
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						data = serial->private = kzalloc(sizeof(struct sierra_intf_private), GFP_KERNEL);
 | 
				
			||||||
 | 
						if (!data)
 | 
				
			||||||
 | 
							return -ENOMEM;
 | 
				
			||||||
 | 
						spin_lock_init(&data->susp_lock);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	return result;
 | 
						return result;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -261,13 +273,18 @@ static struct usb_driver sierra_driver = {
 | 
				
			||||||
	.name       = "sierra",
 | 
						.name       = "sierra",
 | 
				
			||||||
	.probe      = usb_serial_probe,
 | 
						.probe      = usb_serial_probe,
 | 
				
			||||||
	.disconnect = usb_serial_disconnect,
 | 
						.disconnect = usb_serial_disconnect,
 | 
				
			||||||
 | 
						.suspend    = usb_serial_suspend,
 | 
				
			||||||
 | 
						.resume     = usb_serial_resume,
 | 
				
			||||||
	.id_table   = id_table,
 | 
						.id_table   = id_table,
 | 
				
			||||||
	.no_dynamic_id = 	1,
 | 
						.no_dynamic_id = 	1,
 | 
				
			||||||
 | 
						.supports_autosuspend =	1,
 | 
				
			||||||
};
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
struct sierra_port_private {
 | 
					struct sierra_port_private {
 | 
				
			||||||
	spinlock_t lock;	/* lock the structure */
 | 
						spinlock_t lock;	/* lock the structure */
 | 
				
			||||||
	int outstanding_urbs;	/* number of out urbs in flight */
 | 
						int outstanding_urbs;	/* number of out urbs in flight */
 | 
				
			||||||
 | 
						struct usb_anchor active;
 | 
				
			||||||
 | 
						struct usb_anchor delayed;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	/* Input endpoints and buffers for this port */
 | 
						/* Input endpoints and buffers for this port */
 | 
				
			||||||
	struct urb *in_urbs[N_IN_URB];
 | 
						struct urb *in_urbs[N_IN_URB];
 | 
				
			||||||
| 
						 | 
					@ -279,6 +296,8 @@ struct sierra_port_private {
 | 
				
			||||||
	int dsr_state;
 | 
						int dsr_state;
 | 
				
			||||||
	int dcd_state;
 | 
						int dcd_state;
 | 
				
			||||||
	int ri_state;
 | 
						int ri_state;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						unsigned int opened:1;
 | 
				
			||||||
};
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
static int sierra_send_setup(struct usb_serial_port *port)
 | 
					static int sierra_send_setup(struct usb_serial_port *port)
 | 
				
			||||||
| 
						 | 
					@ -390,21 +409,25 @@ static void sierra_outdat_callback(struct urb *urb)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	struct usb_serial_port *port = urb->context;
 | 
						struct usb_serial_port *port = urb->context;
 | 
				
			||||||
	struct sierra_port_private *portdata = usb_get_serial_port_data(port);
 | 
						struct sierra_port_private *portdata = usb_get_serial_port_data(port);
 | 
				
			||||||
 | 
						struct sierra_intf_private *intfdata;
 | 
				
			||||||
	int status = urb->status;
 | 
						int status = urb->status;
 | 
				
			||||||
	unsigned long flags;
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
	dev_dbg(&port->dev, "%s - port %d\n", __func__, port->number);
 | 
						dev_dbg(&port->dev, "%s - port %d\n", __func__, port->number);
 | 
				
			||||||
 | 
						intfdata = port->serial->private;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	/* free up the transfer buffer, as usb_free_urb() does not do this */
 | 
						/* free up the transfer buffer, as usb_free_urb() does not do this */
 | 
				
			||||||
	kfree(urb->transfer_buffer);
 | 
						kfree(urb->transfer_buffer);
 | 
				
			||||||
 | 
						usb_autopm_put_interface_async(port->serial->interface);
 | 
				
			||||||
	if (status)
 | 
						if (status)
 | 
				
			||||||
		dev_dbg(&port->dev, "%s - nonzero write bulk status "
 | 
							dev_dbg(&port->dev, "%s - nonzero write bulk status "
 | 
				
			||||||
		    "received: %d\n", __func__, status);
 | 
							    "received: %d\n", __func__, status);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	spin_lock_irqsave(&portdata->lock, flags);
 | 
						spin_lock(&portdata->lock);
 | 
				
			||||||
	--portdata->outstanding_urbs;
 | 
						--portdata->outstanding_urbs;
 | 
				
			||||||
	spin_unlock_irqrestore(&portdata->lock, flags);
 | 
						spin_unlock(&portdata->lock);
 | 
				
			||||||
 | 
						spin_lock(&intfdata->susp_lock);
 | 
				
			||||||
 | 
						--intfdata->in_flight;
 | 
				
			||||||
 | 
						spin_unlock(&intfdata->susp_lock);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	usb_serial_port_softint(port);
 | 
						usb_serial_port_softint(port);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
| 
						 | 
					@ -414,6 +437,7 @@ static int sierra_write(struct tty_struct *tty, struct usb_serial_port *port,
 | 
				
			||||||
					const unsigned char *buf, int count)
 | 
										const unsigned char *buf, int count)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	struct sierra_port_private *portdata = usb_get_serial_port_data(port);
 | 
						struct sierra_port_private *portdata = usb_get_serial_port_data(port);
 | 
				
			||||||
 | 
						struct sierra_intf_private *intfdata;
 | 
				
			||||||
	struct usb_serial *serial = port->serial;
 | 
						struct usb_serial *serial = port->serial;
 | 
				
			||||||
	unsigned long flags;
 | 
						unsigned long flags;
 | 
				
			||||||
	unsigned char *buffer;
 | 
						unsigned char *buffer;
 | 
				
			||||||
| 
						 | 
					@ -426,9 +450,9 @@ static int sierra_write(struct tty_struct *tty, struct usb_serial_port *port,
 | 
				
			||||||
		return 0;
 | 
							return 0;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	portdata = usb_get_serial_port_data(port);
 | 
						portdata = usb_get_serial_port_data(port);
 | 
				
			||||||
 | 
						intfdata = serial->private;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	dev_dbg(&port->dev, "%s: write (%zd bytes)\n", __func__, writesize);
 | 
						dev_dbg(&port->dev, "%s: write (%zd bytes)\n", __func__, writesize);
 | 
				
			||||||
 | 
					 | 
				
			||||||
	spin_lock_irqsave(&portdata->lock, flags);
 | 
						spin_lock_irqsave(&portdata->lock, flags);
 | 
				
			||||||
	dev_dbg(&port->dev, "%s - outstanding_urbs: %d\n", __func__,
 | 
						dev_dbg(&port->dev, "%s - outstanding_urbs: %d\n", __func__,
 | 
				
			||||||
		portdata->outstanding_urbs);
 | 
							portdata->outstanding_urbs);
 | 
				
			||||||
| 
						 | 
					@ -442,6 +466,14 @@ static int sierra_write(struct tty_struct *tty, struct usb_serial_port *port,
 | 
				
			||||||
		portdata->outstanding_urbs);
 | 
							portdata->outstanding_urbs);
 | 
				
			||||||
	spin_unlock_irqrestore(&portdata->lock, flags);
 | 
						spin_unlock_irqrestore(&portdata->lock, flags);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						retval = usb_autopm_get_interface_async(serial->interface);
 | 
				
			||||||
 | 
						if (retval < 0) {
 | 
				
			||||||
 | 
							spin_lock_irqsave(&portdata->lock, flags);
 | 
				
			||||||
 | 
							portdata->outstanding_urbs--;
 | 
				
			||||||
 | 
							spin_unlock_irqrestore(&portdata->lock, flags);
 | 
				
			||||||
 | 
							goto error_simple;
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	buffer = kmalloc(writesize, GFP_ATOMIC);
 | 
						buffer = kmalloc(writesize, GFP_ATOMIC);
 | 
				
			||||||
	if (!buffer) {
 | 
						if (!buffer) {
 | 
				
			||||||
		dev_err(&port->dev, "out of memory\n");
 | 
							dev_err(&port->dev, "out of memory\n");
 | 
				
			||||||
| 
						 | 
					@ -468,14 +500,29 @@ static int sierra_write(struct tty_struct *tty, struct usb_serial_port *port,
 | 
				
			||||||
	/* Handle the need to send a zero length packet */
 | 
						/* Handle the need to send a zero length packet */
 | 
				
			||||||
	urb->transfer_flags |= URB_ZERO_PACKET;
 | 
						urb->transfer_flags |= URB_ZERO_PACKET;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						spin_lock_irqsave(&intfdata->susp_lock, flags);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if (intfdata->suspended) {
 | 
				
			||||||
 | 
							usb_anchor_urb(urb, &portdata->delayed);
 | 
				
			||||||
 | 
							spin_unlock_irqrestore(&intfdata->susp_lock, flags);
 | 
				
			||||||
 | 
							goto skip_power;
 | 
				
			||||||
 | 
						} else {
 | 
				
			||||||
 | 
							usb_anchor_urb(urb, &portdata->active);
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
	/* send it down the pipe */
 | 
						/* send it down the pipe */
 | 
				
			||||||
	retval = usb_submit_urb(urb, GFP_ATOMIC);
 | 
						retval = usb_submit_urb(urb, GFP_ATOMIC);
 | 
				
			||||||
	if (retval) {
 | 
						if (retval) {
 | 
				
			||||||
 | 
							usb_unanchor_urb(urb);
 | 
				
			||||||
 | 
							spin_unlock_irqrestore(&intfdata->susp_lock, flags);
 | 
				
			||||||
		dev_err(&port->dev, "%s - usb_submit_urb(write bulk) failed "
 | 
							dev_err(&port->dev, "%s - usb_submit_urb(write bulk) failed "
 | 
				
			||||||
			"with status = %d\n", __func__, retval);
 | 
								"with status = %d\n", __func__, retval);
 | 
				
			||||||
		goto error;
 | 
							goto error;
 | 
				
			||||||
 | 
						} else {
 | 
				
			||||||
 | 
							intfdata->in_flight++;
 | 
				
			||||||
 | 
							spin_unlock_irqrestore(&intfdata->susp_lock, flags);
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					skip_power:
 | 
				
			||||||
	/* we are done with this urb, so let the host driver
 | 
						/* we are done with this urb, so let the host driver
 | 
				
			||||||
	 * really free it when it is finished with it */
 | 
						 * really free it when it is finished with it */
 | 
				
			||||||
	usb_free_urb(urb);
 | 
						usb_free_urb(urb);
 | 
				
			||||||
| 
						 | 
					@ -491,6 +538,8 @@ static int sierra_write(struct tty_struct *tty, struct usb_serial_port *port,
 | 
				
			||||||
	dev_dbg(&port->dev, "%s - 2. outstanding_urbs: %d\n", __func__,
 | 
						dev_dbg(&port->dev, "%s - 2. outstanding_urbs: %d\n", __func__,
 | 
				
			||||||
		portdata->outstanding_urbs);
 | 
							portdata->outstanding_urbs);
 | 
				
			||||||
	spin_unlock_irqrestore(&portdata->lock, flags);
 | 
						spin_unlock_irqrestore(&portdata->lock, flags);
 | 
				
			||||||
 | 
						usb_autopm_put_interface_async(serial->interface);
 | 
				
			||||||
 | 
					error_simple:
 | 
				
			||||||
	return retval;
 | 
						return retval;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -530,6 +579,7 @@ static void sierra_indat_callback(struct urb *urb)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	/* Resubmit urb so we continue receiving */
 | 
						/* Resubmit urb so we continue receiving */
 | 
				
			||||||
	if (port->port.count && status != -ESHUTDOWN && status != -EPERM) {
 | 
						if (port->port.count && status != -ESHUTDOWN && status != -EPERM) {
 | 
				
			||||||
 | 
							usb_mark_last_busy(port->serial->dev);
 | 
				
			||||||
		err = usb_submit_urb(urb, GFP_ATOMIC);
 | 
							err = usb_submit_urb(urb, GFP_ATOMIC);
 | 
				
			||||||
		if (err)
 | 
							if (err)
 | 
				
			||||||
			dev_err(&port->dev, "resubmit read urb failed."
 | 
								dev_err(&port->dev, "resubmit read urb failed."
 | 
				
			||||||
| 
						 | 
					@ -591,6 +641,7 @@ static void sierra_instat_callback(struct urb *urb)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	/* Resubmit urb so we continue receiving IRQ data */
 | 
						/* Resubmit urb so we continue receiving IRQ data */
 | 
				
			||||||
	if (port->port.count && status != -ESHUTDOWN && status != -ENOENT) {
 | 
						if (port->port.count && status != -ESHUTDOWN && status != -ENOENT) {
 | 
				
			||||||
 | 
							usb_mark_last_busy(serial->dev);
 | 
				
			||||||
		urb->dev = serial->dev;
 | 
							urb->dev = serial->dev;
 | 
				
			||||||
		err = usb_submit_urb(urb, GFP_ATOMIC);
 | 
							err = usb_submit_urb(urb, GFP_ATOMIC);
 | 
				
			||||||
		if (err)
 | 
							if (err)
 | 
				
			||||||
| 
						 | 
					@ -711,6 +762,8 @@ static void sierra_close(struct usb_serial_port *port)
 | 
				
			||||||
	int i;
 | 
						int i;
 | 
				
			||||||
	struct usb_serial *serial = port->serial;
 | 
						struct usb_serial *serial = port->serial;
 | 
				
			||||||
	struct sierra_port_private *portdata;
 | 
						struct sierra_port_private *portdata;
 | 
				
			||||||
 | 
						struct sierra_intf_private *intfdata = port->serial->private;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	dev_dbg(&port->dev, "%s\n", __func__);
 | 
						dev_dbg(&port->dev, "%s\n", __func__);
 | 
				
			||||||
	portdata = usb_get_serial_port_data(port);
 | 
						portdata = usb_get_serial_port_data(port);
 | 
				
			||||||
| 
						 | 
					@ -723,6 +776,10 @@ static void sierra_close(struct usb_serial_port *port)
 | 
				
			||||||
		if (!serial->disconnected)
 | 
							if (!serial->disconnected)
 | 
				
			||||||
			sierra_send_setup(port);
 | 
								sierra_send_setup(port);
 | 
				
			||||||
		mutex_unlock(&serial->disc_mutex);
 | 
							mutex_unlock(&serial->disc_mutex);
 | 
				
			||||||
 | 
							spin_lock_irq(&intfdata->susp_lock);
 | 
				
			||||||
 | 
							portdata->opened = 0;
 | 
				
			||||||
 | 
							spin_unlock_irq(&intfdata->susp_lock);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		/* Stop reading urbs */
 | 
							/* Stop reading urbs */
 | 
				
			||||||
		sierra_stop_rx_urbs(port);
 | 
							sierra_stop_rx_urbs(port);
 | 
				
			||||||
| 
						 | 
					@ -731,6 +788,8 @@ static void sierra_close(struct usb_serial_port *port)
 | 
				
			||||||
			sierra_release_urb(portdata->in_urbs[i]);
 | 
								sierra_release_urb(portdata->in_urbs[i]);
 | 
				
			||||||
			portdata->in_urbs[i] = NULL;
 | 
								portdata->in_urbs[i] = NULL;
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
 | 
							usb_autopm_get_interface(serial->interface);
 | 
				
			||||||
 | 
							serial->interface->needs_remote_wakeup = 0;
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -738,6 +797,7 @@ static int sierra_open(struct tty_struct *tty, struct usb_serial_port *port)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	struct sierra_port_private *portdata;
 | 
						struct sierra_port_private *portdata;
 | 
				
			||||||
	struct usb_serial *serial = port->serial;
 | 
						struct usb_serial *serial = port->serial;
 | 
				
			||||||
 | 
						struct sierra_intf_private *intfdata = serial->private;
 | 
				
			||||||
	int i;
 | 
						int i;
 | 
				
			||||||
	int err;
 | 
						int err;
 | 
				
			||||||
	int endpoint;
 | 
						int endpoint;
 | 
				
			||||||
| 
						 | 
					@ -771,6 +831,12 @@ static int sierra_open(struct tty_struct *tty, struct usb_serial_port *port)
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
	sierra_send_setup(port);
 | 
						sierra_send_setup(port);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						serial->interface->needs_remote_wakeup = 1;
 | 
				
			||||||
 | 
						spin_lock_irq(&intfdata->susp_lock);
 | 
				
			||||||
 | 
						portdata->opened = 1;
 | 
				
			||||||
 | 
						spin_unlock_irq(&intfdata->susp_lock);
 | 
				
			||||||
 | 
						usb_autopm_put_interface(serial->interface);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	return 0;
 | 
						return 0;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -818,6 +884,8 @@ static int sierra_startup(struct usb_serial *serial)
 | 
				
			||||||
			return -ENOMEM;
 | 
								return -ENOMEM;
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
		spin_lock_init(&portdata->lock);
 | 
							spin_lock_init(&portdata->lock);
 | 
				
			||||||
 | 
							init_usb_anchor(&portdata->active);
 | 
				
			||||||
 | 
							init_usb_anchor(&portdata->delayed);
 | 
				
			||||||
		/* Set the port private data pointer */
 | 
							/* Set the port private data pointer */
 | 
				
			||||||
		usb_set_serial_port_data(port, portdata);
 | 
							usb_set_serial_port_data(port, portdata);
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
| 
						 | 
					@ -844,6 +912,83 @@ static void sierra_release(struct usb_serial *serial)
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					static void stop_read_write_urbs(struct usb_serial *serial)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
						int i, j;
 | 
				
			||||||
 | 
						struct usb_serial_port *port;
 | 
				
			||||||
 | 
						struct sierra_port_private *portdata;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						/* Stop reading/writing urbs */
 | 
				
			||||||
 | 
						for (i = 0; i < serial->num_ports; ++i) {
 | 
				
			||||||
 | 
							port = serial->port[i];
 | 
				
			||||||
 | 
							portdata = usb_get_serial_port_data(port);
 | 
				
			||||||
 | 
							for (j = 0; j < N_IN_URB; j++)
 | 
				
			||||||
 | 
								usb_kill_urb(portdata->in_urbs[j]);
 | 
				
			||||||
 | 
							usb_kill_anchored_urbs(&portdata->active);
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					static int sierra_suspend(struct usb_serial *serial, pm_message_t message)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
						struct sierra_intf_private *intfdata;
 | 
				
			||||||
 | 
						int b;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if (serial->dev->auto_pm) {
 | 
				
			||||||
 | 
							intfdata = serial->private;
 | 
				
			||||||
 | 
							spin_lock_irq(&intfdata->susp_lock);
 | 
				
			||||||
 | 
							b = intfdata->in_flight;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							if (b) {
 | 
				
			||||||
 | 
								spin_unlock_irq(&intfdata->susp_lock);
 | 
				
			||||||
 | 
								return -EBUSY;
 | 
				
			||||||
 | 
							} else {
 | 
				
			||||||
 | 
								intfdata->suspended = 1;
 | 
				
			||||||
 | 
								spin_unlock_irq(&intfdata->susp_lock);
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						stop_read_write_urbs(serial);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return 0;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					static int sierra_resume(struct usb_serial *serial)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
						struct usb_serial_port *port;
 | 
				
			||||||
 | 
						struct sierra_intf_private *intfdata = serial->private;
 | 
				
			||||||
 | 
						struct sierra_port_private *portdata;
 | 
				
			||||||
 | 
						struct urb *urb;
 | 
				
			||||||
 | 
						int ec = 0;
 | 
				
			||||||
 | 
						int i, err;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						spin_lock_irq(&intfdata->susp_lock);
 | 
				
			||||||
 | 
						for (i = 0; i < serial->num_ports; i++) {
 | 
				
			||||||
 | 
							port = serial->port[i];
 | 
				
			||||||
 | 
							portdata = usb_get_serial_port_data(port);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							while ((urb = usb_get_from_anchor(&portdata->delayed))) {
 | 
				
			||||||
 | 
								usb_anchor_urb(urb, &portdata->active);
 | 
				
			||||||
 | 
								intfdata->in_flight++;
 | 
				
			||||||
 | 
								err = usb_submit_urb(urb, GFP_ATOMIC);
 | 
				
			||||||
 | 
								if (err < 0) {
 | 
				
			||||||
 | 
									intfdata->in_flight--;
 | 
				
			||||||
 | 
									usb_unanchor_urb(urb);
 | 
				
			||||||
 | 
									usb_scuttle_anchored_urbs(&portdata->delayed);
 | 
				
			||||||
 | 
									break;
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							if (portdata->opened) {
 | 
				
			||||||
 | 
								err = sierra_submit_rx_urbs(port, GFP_ATOMIC);
 | 
				
			||||||
 | 
								if (err)
 | 
				
			||||||
 | 
									ec++;
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						intfdata->suspended = 0;
 | 
				
			||||||
 | 
						spin_unlock_irq(&intfdata->susp_lock);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return ec ? -EIO : 0;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
static struct usb_serial_driver sierra_device = {
 | 
					static struct usb_serial_driver sierra_device = {
 | 
				
			||||||
	.driver = {
 | 
						.driver = {
 | 
				
			||||||
		.owner =	THIS_MODULE,
 | 
							.owner =	THIS_MODULE,
 | 
				
			||||||
| 
						 | 
					@ -864,6 +1009,8 @@ static struct usb_serial_driver sierra_device = {
 | 
				
			||||||
	.tiocmset          = sierra_tiocmset,
 | 
						.tiocmset          = sierra_tiocmset,
 | 
				
			||||||
	.attach            = sierra_startup,
 | 
						.attach            = sierra_startup,
 | 
				
			||||||
	.release           = sierra_release,
 | 
						.release           = sierra_release,
 | 
				
			||||||
 | 
						.suspend	   = sierra_suspend,
 | 
				
			||||||
 | 
						.resume		   = sierra_resume,
 | 
				
			||||||
	.read_int_callback = sierra_instat_callback,
 | 
						.read_int_callback = sierra_instat_callback,
 | 
				
			||||||
};
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
		Loading…
	
		Reference in a new issue