Commit 8e8dce06 authored by David VomLehn's avatar David VomLehn Committed by Greg Kroah-Hartman

USB: use kfifo to buffer usb-generic serial writes

When do_output_char() attempts to write a carriage return/line feed sequence,
it first checks to see how much buffer room is available. If there are at least
two characters free, it will write the carriage return/line feed with two calls
to tty_put_char(). It calls the tty_operation functions write() for devices that
don't support the tty_operations function put_char(). If the USB generic serial
device's write URB is not in use, it will return the buffer size when asked how
much room is available. The write() of the carriage return will cause it to mark
the write URB busy, so the subsequent write() of the line feed will be ignored.

This patch uses the kfifo infrastructure to implement a write FIFO that
accurately returns the amount of space available in the buffer.
Signed-off-by: default avatarDavid VomLehn <dvomlehn@cisco.com>
Signed-off-by: default avatarGreg Kroah-Hartman <gregkh@suse.de>
parent 74aee796
...@@ -19,7 +19,7 @@ ...@@ -19,7 +19,7 @@
#include <linux/usb.h> #include <linux/usb.h>
#include <linux/usb/serial.h> #include <linux/usb/serial.h>
#include <linux/uaccess.h> #include <linux/uaccess.h>
#include <linux/kfifo.h>
static int debug; static int debug;
...@@ -166,24 +166,6 @@ static void generic_cleanup(struct usb_serial_port *port) ...@@ -166,24 +166,6 @@ static void generic_cleanup(struct usb_serial_port *port)
} }
} }
int usb_serial_generic_resume(struct usb_serial *serial)
{
struct usb_serial_port *port;
int i, c = 0, r;
for (i = 0; i < serial->num_ports; i++) {
port = serial->port[i];
if (port->port.count && port->read_urb) {
r = usb_submit_urb(port->read_urb, GFP_NOIO);
if (r < 0)
c++;
}
}
return c ? -EIO : 0;
}
EXPORT_SYMBOL_GPL(usb_serial_generic_resume);
void usb_serial_generic_close(struct usb_serial_port *port) void usb_serial_generic_close(struct usb_serial_port *port)
{ {
dbg("%s - port %d", __func__, port->number); dbg("%s - port %d", __func__, port->number);
...@@ -272,42 +254,38 @@ static int usb_serial_multi_urb_write(struct tty_struct *tty, ...@@ -272,42 +254,38 @@ static int usb_serial_multi_urb_write(struct tty_struct *tty,
return bwrite; return bwrite;
} }
int usb_serial_generic_write(struct tty_struct *tty, /**
struct usb_serial_port *port, const unsigned char *buf, int count) * usb_serial_generic_write_start - kick off an URB write
* @port: Pointer to the &struct usb_serial_port data
*
* Returns the number of bytes queued on success. This will be zero if there
* was nothing to send. Otherwise, it returns a negative errno value
*/
static int usb_serial_generic_write_start(struct usb_serial_port *port)
{ {
struct usb_serial *serial = port->serial; struct usb_serial *serial = port->serial;
int result;
unsigned char *data; unsigned char *data;
int result;
dbg("%s - port %d", __func__, port->number); int count;
if (count == 0) {
dbg("%s - write request of 0 bytes", __func__);
return 0;
}
/* only do something if we have a bulk out endpoint */
if (serial->num_bulk_out) {
unsigned long flags; unsigned long flags;
bool start_io;
if (serial->type->max_in_flight_urbs) /* Atomically determine whether we can and need to start a USB
return usb_serial_multi_urb_write(tty, port, * operation. */
buf, count);
spin_lock_irqsave(&port->lock, flags); spin_lock_irqsave(&port->lock, flags);
if (port->write_urb_busy) { if (port->write_urb_busy)
spin_unlock_irqrestore(&port->lock, flags); start_io = false;
dbg("%s - already writing", __func__); else {
return 0; start_io = (__kfifo_len(port->write_fifo) != 0);
port->write_urb_busy = start_io;
} }
port->write_urb_busy = 1;
spin_unlock_irqrestore(&port->lock, flags); spin_unlock_irqrestore(&port->lock, flags);
count = (count > port->bulk_out_size) ? if (!start_io)
port->bulk_out_size : count; return 0;
memcpy(port->write_urb->transfer_buffer, buf, count);
data = port->write_urb->transfer_buffer; data = port->write_urb->transfer_buffer;
count = kfifo_get(port->write_fifo, data, port->bulk_out_size);
usb_serial_debug_data(debug, &port->dev, __func__, count, data); usb_serial_debug_data(debug, &port->dev, __func__, count, data);
/* set up our urb */ /* set up our urb */
...@@ -321,7 +299,6 @@ int usb_serial_generic_write(struct tty_struct *tty, ...@@ -321,7 +299,6 @@ int usb_serial_generic_write(struct tty_struct *tty,
port); port);
/* send the data out the bulk port */ /* send the data out the bulk port */
port->write_urb_busy = 1;
result = usb_submit_urb(port->write_urb, GFP_ATOMIC); result = usb_submit_urb(port->write_urb, GFP_ATOMIC);
if (result) { if (result) {
dev_err(&port->dev, dev_err(&port->dev,
...@@ -334,10 +311,47 @@ int usb_serial_generic_write(struct tty_struct *tty, ...@@ -334,10 +311,47 @@ int usb_serial_generic_write(struct tty_struct *tty,
result = count; result = count;
return result; return result;
}
/**
* usb_serial_generic_write - generic write function for serial USB devices
* @tty: Pointer to &struct tty_struct for the device
* @port: Pointer to the &usb_serial_port structure for the device
* @buf: Pointer to the data to write
* @count: Number of bytes to write
*
* Returns the number of characters actually written, which may be anything
* from zero to @count. If an error occurs, it returns the negative errno
* value.
*/
int usb_serial_generic_write(struct tty_struct *tty,
struct usb_serial_port *port, const unsigned char *buf, int count)
{
struct usb_serial *serial = port->serial;
int result;
dbg("%s - port %d", __func__, port->number);
if (count == 0) {
dbg("%s - write request of 0 bytes", __func__);
return 0;
} }
/* no bulk out, so return 0 bytes written */ /* only do something if we have a bulk out endpoint */
if (!serial->num_bulk_out)
return 0; return 0;
if (serial->type->max_in_flight_urbs)
return usb_serial_multi_urb_write(tty, port,
buf, count);
count = kfifo_put(port->write_fifo, buf, count);
result = usb_serial_generic_write_start(port);
if (result >= 0)
result = count;
return result;
} }
EXPORT_SYMBOL_GPL(usb_serial_generic_write); EXPORT_SYMBOL_GPL(usb_serial_generic_write);
...@@ -355,9 +369,8 @@ int usb_serial_generic_write_room(struct tty_struct *tty) ...@@ -355,9 +369,8 @@ int usb_serial_generic_write_room(struct tty_struct *tty)
room = port->bulk_out_size * room = port->bulk_out_size *
(serial->type->max_in_flight_urbs - (serial->type->max_in_flight_urbs -
port->urbs_in_flight); port->urbs_in_flight);
} else if (serial->num_bulk_out && !(port->write_urb_busy)) { } else if (serial->num_bulk_out)
room = port->bulk_out_size; room = port->write_fifo->size - __kfifo_len(port->write_fifo);
}
spin_unlock_irqrestore(&port->lock, flags); spin_unlock_irqrestore(&port->lock, flags);
dbg("%s - returns %d", __func__, room); dbg("%s - returns %d", __func__, room);
...@@ -377,11 +390,8 @@ int usb_serial_generic_chars_in_buffer(struct tty_struct *tty) ...@@ -377,11 +390,8 @@ int usb_serial_generic_chars_in_buffer(struct tty_struct *tty)
spin_lock_irqsave(&port->lock, flags); spin_lock_irqsave(&port->lock, flags);
chars = port->tx_bytes_flight; chars = port->tx_bytes_flight;
spin_unlock_irqrestore(&port->lock, flags); spin_unlock_irqrestore(&port->lock, flags);
} else if (serial->num_bulk_out) { } else if (serial->num_bulk_out)
/* FIXME: Locking */ chars = kfifo_len(port->write_fifo);
if (port->write_urb_busy)
chars = port->write_urb->transfer_buffer_length;
}
dbg("%s - returns %d", __func__, chars); dbg("%s - returns %d", __func__, chars);
return chars; return chars;
...@@ -485,16 +495,23 @@ void usb_serial_generic_write_bulk_callback(struct urb *urb) ...@@ -485,16 +495,23 @@ void usb_serial_generic_write_bulk_callback(struct urb *urb)
if (port->urbs_in_flight < 0) if (port->urbs_in_flight < 0)
port->urbs_in_flight = 0; port->urbs_in_flight = 0;
spin_unlock_irqrestore(&port->lock, flags); spin_unlock_irqrestore(&port->lock, flags);
if (status) {
dbg("%s - nonzero multi-urb write bulk status "
"received: %d", __func__, status);
return;
}
} else { } else {
/* Handle the case for single urb mode */
port->write_urb_busy = 0; port->write_urb_busy = 0;
}
if (status) { if (status) {
dbg("%s - nonzero write bulk status received: %d", dbg("%s - nonzero multi-urb write bulk status "
__func__, status); "received: %d", __func__, status);
return; kfifo_reset(port->write_fifo);
} else
usb_serial_generic_write_start(port);
} }
usb_serial_port_softint(port); usb_serial_port_softint(port);
} }
EXPORT_SYMBOL_GPL(usb_serial_generic_write_bulk_callback); EXPORT_SYMBOL_GPL(usb_serial_generic_write_bulk_callback);
...@@ -559,6 +576,33 @@ int usb_serial_handle_break(struct usb_serial_port *port) ...@@ -559,6 +576,33 @@ int usb_serial_handle_break(struct usb_serial_port *port)
} }
EXPORT_SYMBOL_GPL(usb_serial_handle_break); EXPORT_SYMBOL_GPL(usb_serial_handle_break);
int usb_serial_generic_resume(struct usb_serial *serial)
{
struct usb_serial_port *port;
int i, c = 0, r;
for (i = 0; i < serial->num_ports; i++) {
port = serial->port[i];
if (!port->port.count)
continue;
if (port->read_urb) {
r = usb_submit_urb(port->read_urb, GFP_NOIO);
if (r < 0)
c++;
}
if (port->write_urb) {
r = usb_serial_generic_write_start(port);
if (r < 0)
c++;
}
}
return c ? -EIO : 0;
}
EXPORT_SYMBOL_GPL(usb_serial_generic_resume);
void usb_serial_generic_disconnect(struct usb_serial *serial) void usb_serial_generic_disconnect(struct usb_serial *serial)
{ {
int i; int i;
......
...@@ -35,6 +35,7 @@ ...@@ -35,6 +35,7 @@
#include <linux/serial.h> #include <linux/serial.h>
#include <linux/usb.h> #include <linux/usb.h>
#include <linux/usb/serial.h> #include <linux/usb/serial.h>
#include <linux/kfifo.h>
#include "pl2303.h" #include "pl2303.h"
/* /*
...@@ -625,6 +626,8 @@ static void port_release(struct device *dev) ...@@ -625,6 +626,8 @@ static void port_release(struct device *dev)
usb_free_urb(port->write_urb); usb_free_urb(port->write_urb);
usb_free_urb(port->interrupt_in_urb); usb_free_urb(port->interrupt_in_urb);
usb_free_urb(port->interrupt_out_urb); usb_free_urb(port->interrupt_out_urb);
if (!IS_ERR(port->write_fifo) && port->write_fifo)
kfifo_free(port->write_fifo);
kfree(port->bulk_in_buffer); kfree(port->bulk_in_buffer);
kfree(port->bulk_out_buffer); kfree(port->bulk_out_buffer);
kfree(port->interrupt_in_buffer); kfree(port->interrupt_in_buffer);
...@@ -964,6 +967,10 @@ int usb_serial_probe(struct usb_interface *interface, ...@@ -964,6 +967,10 @@ int usb_serial_probe(struct usb_interface *interface,
dev_err(&interface->dev, "No free urbs available\n"); dev_err(&interface->dev, "No free urbs available\n");
goto probe_error; goto probe_error;
} }
port->write_fifo = kfifo_alloc(PAGE_SIZE, GFP_KERNEL,
&port->lock);
if (IS_ERR(port->write_fifo))
goto probe_error;
buffer_size = le16_to_cpu(endpoint->wMaxPacketSize); buffer_size = le16_to_cpu(endpoint->wMaxPacketSize);
port->bulk_out_size = buffer_size; port->bulk_out_size = buffer_size;
port->bulk_out_endpointAddress = endpoint->bEndpointAddress; port->bulk_out_endpointAddress = endpoint->bEndpointAddress;
......
...@@ -59,6 +59,7 @@ enum port_dev_state { ...@@ -59,6 +59,7 @@ enum port_dev_state {
* @bulk_out_buffer: pointer to the bulk out buffer for this port. * @bulk_out_buffer: pointer to the bulk out buffer for this port.
* @bulk_out_size: the size of the bulk_out_buffer, in bytes. * @bulk_out_size: the size of the bulk_out_buffer, in bytes.
* @write_urb: pointer to the bulk out struct urb for this port. * @write_urb: pointer to the bulk out struct urb for this port.
* @write_fifo: kfifo used to buffer outgoing data
* @write_urb_busy: port`s writing status * @write_urb_busy: port`s writing status
* @bulk_out_endpointAddress: endpoint address for the bulk out pipe for this * @bulk_out_endpointAddress: endpoint address for the bulk out pipe for this
* port. * port.
...@@ -96,6 +97,7 @@ struct usb_serial_port { ...@@ -96,6 +97,7 @@ struct usb_serial_port {
unsigned char *bulk_out_buffer; unsigned char *bulk_out_buffer;
int bulk_out_size; int bulk_out_size;
struct urb *write_urb; struct urb *write_urb;
struct kfifo *write_fifo;
int write_urb_busy; int write_urb_busy;
__u8 bulk_out_endpointAddress; __u8 bulk_out_endpointAddress;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment