diff options
-rw-r--r-- | drivers/usb/serial/generic.c | 47 | ||||
-rw-r--r-- | drivers/usb/serial/usb-serial.c | 1 | ||||
-rw-r--r-- | include/linux/usb/serial.h | 5 |
3 files changed, 40 insertions, 13 deletions
diff --git a/drivers/usb/serial/generic.c b/drivers/usb/serial/generic.c index ad4823bbfa19..1a134f9c64f3 100644 --- a/drivers/usb/serial/generic.c +++ b/drivers/usb/serial/generic.c | |||
@@ -167,12 +167,35 @@ void usb_serial_generic_close(struct usb_serial_port *port) | |||
167 | } | 167 | } |
168 | EXPORT_SYMBOL_GPL(usb_serial_generic_close); | 168 | EXPORT_SYMBOL_GPL(usb_serial_generic_close); |
169 | 169 | ||
170 | int usb_serial_generic_prepare_write_buffer(struct usb_serial_port *port, | ||
171 | void **dest, size_t size, const void *src, size_t count) | ||
172 | { | ||
173 | if (!*dest) { | ||
174 | size = count; | ||
175 | *dest = kmalloc(count, GFP_ATOMIC); | ||
176 | if (!*dest) { | ||
177 | dev_err(&port->dev, "%s - could not allocate buffer\n", | ||
178 | __func__); | ||
179 | return -ENOMEM; | ||
180 | } | ||
181 | } | ||
182 | if (src) { | ||
183 | count = size; | ||
184 | memcpy(*dest, src, size); | ||
185 | } else { | ||
186 | count = kfifo_out_locked(&port->write_fifo, *dest, size, | ||
187 | &port->lock); | ||
188 | } | ||
189 | return count; | ||
190 | } | ||
191 | EXPORT_SYMBOL_GPL(usb_serial_generic_prepare_write_buffer); | ||
192 | |||
170 | static int usb_serial_multi_urb_write(struct tty_struct *tty, | 193 | static int usb_serial_multi_urb_write(struct tty_struct *tty, |
171 | struct usb_serial_port *port, const unsigned char *buf, int count) | 194 | struct usb_serial_port *port, const unsigned char *buf, int count) |
172 | { | 195 | { |
173 | unsigned long flags; | 196 | unsigned long flags; |
174 | struct urb *urb; | 197 | struct urb *urb; |
175 | unsigned char *buffer; | 198 | void *buffer; |
176 | int status; | 199 | int status; |
177 | 200 | ||
178 | spin_lock_irqsave(&port->lock, flags); | 201 | spin_lock_irqsave(&port->lock, flags); |
@@ -191,16 +214,14 @@ static int usb_serial_multi_urb_write(struct tty_struct *tty, | |||
191 | goto err_urb; | 214 | goto err_urb; |
192 | } | 215 | } |
193 | 216 | ||
217 | buffer = NULL; | ||
194 | count = min_t(int, count, PAGE_SIZE); | 218 | count = min_t(int, count, PAGE_SIZE); |
195 | buffer = kmalloc(count, GFP_ATOMIC); | 219 | count = port->serial->type->prepare_write_buffer(port, &buffer, 0, |
196 | if (!buffer) { | 220 | buf, count); |
197 | dev_err(&port->dev, "%s - could not allocate buffer\n", | 221 | if (count < 0) { |
198 | __func__); | 222 | status = count; |
199 | status = -ENOMEM; | ||
200 | goto err_buf; | 223 | goto err_buf; |
201 | } | 224 | } |
202 | |||
203 | memcpy(buffer, buf, count); | ||
204 | usb_serial_debug_data(debug, &port->dev, __func__, count, buffer); | 225 | usb_serial_debug_data(debug, &port->dev, __func__, count, buffer); |
205 | usb_fill_bulk_urb(urb, port->serial->dev, | 226 | usb_fill_bulk_urb(urb, port->serial->dev, |
206 | usb_sndbulkpipe(port->serial->dev, | 227 | usb_sndbulkpipe(port->serial->dev, |
@@ -242,7 +263,6 @@ err_urb: | |||
242 | */ | 263 | */ |
243 | static int usb_serial_generic_write_start(struct usb_serial_port *port) | 264 | static int usb_serial_generic_write_start(struct usb_serial_port *port) |
244 | { | 265 | { |
245 | unsigned char *data; | ||
246 | int result; | 266 | int result; |
247 | int count; | 267 | int count; |
248 | unsigned long flags; | 268 | unsigned long flags; |
@@ -255,10 +275,11 @@ static int usb_serial_generic_write_start(struct usb_serial_port *port) | |||
255 | port->write_urb_busy = 1; | 275 | port->write_urb_busy = 1; |
256 | spin_unlock_irqrestore(&port->lock, flags); | 276 | spin_unlock_irqrestore(&port->lock, flags); |
257 | 277 | ||
258 | data = port->write_urb->transfer_buffer; | 278 | count = port->serial->type->prepare_write_buffer(port, |
259 | count = kfifo_out_locked(&port->write_fifo, data, port->bulk_out_size, &port->lock); | 279 | &port->write_urb->transfer_buffer, |
260 | usb_serial_debug_data(debug, &port->dev, __func__, count, data); | 280 | port->bulk_out_size, NULL, 0); |
261 | 281 | usb_serial_debug_data(debug, &port->dev, __func__, | |
282 | count, port->write_urb->transfer_buffer); | ||
262 | port->write_urb->transfer_buffer_length = count; | 283 | port->write_urb->transfer_buffer_length = count; |
263 | 284 | ||
264 | /* send the data out the bulk port */ | 285 | /* send the data out the bulk port */ |
diff --git a/drivers/usb/serial/usb-serial.c b/drivers/usb/serial/usb-serial.c index 1b924425089f..8249fd8381fb 100644 --- a/drivers/usb/serial/usb-serial.c +++ b/drivers/usb/serial/usb-serial.c | |||
@@ -1299,6 +1299,7 @@ static void fixup_generic(struct usb_serial_driver *device) | |||
1299 | set_to_generic_if_null(device, disconnect); | 1299 | set_to_generic_if_null(device, disconnect); |
1300 | set_to_generic_if_null(device, release); | 1300 | set_to_generic_if_null(device, release); |
1301 | set_to_generic_if_null(device, process_read_urb); | 1301 | set_to_generic_if_null(device, process_read_urb); |
1302 | set_to_generic_if_null(device, prepare_write_buffer); | ||
1302 | } | 1303 | } |
1303 | 1304 | ||
1304 | int usb_serial_register(struct usb_serial_driver *driver) | 1305 | int usb_serial_register(struct usb_serial_driver *driver) |
diff --git a/include/linux/usb/serial.h b/include/linux/usb/serial.h index 2a3283761600..a4c99ea390e7 100644 --- a/include/linux/usb/serial.h +++ b/include/linux/usb/serial.h | |||
@@ -277,6 +277,9 @@ struct usb_serial_driver { | |||
277 | void (*write_bulk_callback)(struct urb *urb); | 277 | void (*write_bulk_callback)(struct urb *urb); |
278 | /* Called by the generic read bulk callback */ | 278 | /* Called by the generic read bulk callback */ |
279 | void (*process_read_urb)(struct urb *urb); | 279 | void (*process_read_urb)(struct urb *urb); |
280 | /* Called by the generic write implementation */ | ||
281 | int (*prepare_write_buffer)(struct usb_serial_port *port, | ||
282 | void **dest, size_t size, const void *src, size_t count); | ||
280 | }; | 283 | }; |
281 | #define to_usb_serial_driver(d) \ | 284 | #define to_usb_serial_driver(d) \ |
282 | container_of(d, struct usb_serial_driver, driver) | 285 | container_of(d, struct usb_serial_driver, driver) |
@@ -329,6 +332,8 @@ extern void usb_serial_generic_deregister(void); | |||
329 | extern int usb_serial_generic_submit_read_urb(struct usb_serial_port *port, | 332 | extern int usb_serial_generic_submit_read_urb(struct usb_serial_port *port, |
330 | gfp_t mem_flags); | 333 | gfp_t mem_flags); |
331 | extern void usb_serial_generic_process_read_urb(struct urb *urb); | 334 | extern void usb_serial_generic_process_read_urb(struct urb *urb); |
335 | extern int usb_serial_generic_prepare_write_buffer(struct usb_serial_port *port, | ||
336 | void **dest, size_t size, const void *src, size_t count); | ||
332 | extern int usb_serial_handle_sysrq_char(struct tty_struct *tty, | 337 | extern int usb_serial_handle_sysrq_char(struct tty_struct *tty, |
333 | struct usb_serial_port *port, | 338 | struct usb_serial_port *port, |
334 | unsigned int ch); | 339 | unsigned int ch); |