Commit d571b592 authored by Mauro Carvalho Chehab's avatar Mauro Carvalho Chehab

media: em28xx: don't use coherent buffer for DMA transfers

While coherent memory is cheap on x86, it may cause performance
impacts on other archs. As we don't have any good reason to
use it, let's change the logic by allocating memory via kmalloc()
and letting the USB core to do the DMA mapping and memory free
for us.

While here, also fixes an issue that it was not de-allocating
memories if something gets wrong during memory block
allocation.
Signed-off-by: default avatarMauro Carvalho Chehab <mchehab@s-opensource.com>
parent 94448e21
...@@ -821,7 +821,6 @@ void em28xx_uninit_usb_xfer(struct em28xx *dev, enum em28xx_mode mode) ...@@ -821,7 +821,6 @@ void em28xx_uninit_usb_xfer(struct em28xx *dev, enum em28xx_mode mode)
{ {
struct urb *urb; struct urb *urb;
struct em28xx_usb_bufs *usb_bufs; struct em28xx_usb_bufs *usb_bufs;
struct usb_device *udev = interface_to_usbdev(dev->intf);
int i; int i;
em28xx_isocdbg("em28xx: called em28xx_uninit_usb_xfer in mode %d\n", em28xx_isocdbg("em28xx: called em28xx_uninit_usb_xfer in mode %d\n",
...@@ -840,23 +839,16 @@ void em28xx_uninit_usb_xfer(struct em28xx *dev, enum em28xx_mode mode) ...@@ -840,23 +839,16 @@ void em28xx_uninit_usb_xfer(struct em28xx *dev, enum em28xx_mode mode)
else else
usb_unlink_urb(urb); usb_unlink_urb(urb);
if (usb_bufs->transfer_buffer[i]) {
usb_free_coherent(udev,
urb->transfer_buffer_length,
usb_bufs->transfer_buffer[i],
urb->transfer_dma);
}
usb_free_urb(urb); usb_free_urb(urb);
usb_bufs->urb[i] = NULL; usb_bufs->urb[i] = NULL;
} }
usb_bufs->transfer_buffer[i] = NULL;
} }
kfree(usb_bufs->urb); kfree(usb_bufs->urb);
kfree(usb_bufs->transfer_buffer); kfree(usb_bufs->buf);
usb_bufs->urb = NULL; usb_bufs->urb = NULL;
usb_bufs->transfer_buffer = NULL; usb_bufs->buf = NULL;
usb_bufs->num_bufs = 0; usb_bufs->num_bufs = 0;
em28xx_capture_start(dev, 0); em28xx_capture_start(dev, 0);
...@@ -933,14 +925,13 @@ int em28xx_alloc_urbs(struct em28xx *dev, enum em28xx_mode mode, int xfer_bulk, ...@@ -933,14 +925,13 @@ int em28xx_alloc_urbs(struct em28xx *dev, enum em28xx_mode mode, int xfer_bulk,
usb_bufs->num_bufs = num_bufs; usb_bufs->num_bufs = num_bufs;
usb_bufs->urb = kzalloc(sizeof(void *)*num_bufs, GFP_KERNEL); usb_bufs->urb = kcalloc(sizeof(void *), num_bufs, GFP_KERNEL);
if (!usb_bufs->urb) if (!usb_bufs->urb)
return -ENOMEM; return -ENOMEM;
usb_bufs->transfer_buffer = kzalloc(sizeof(void *)*num_bufs, usb_bufs->buf = kcalloc(sizeof(void *), num_bufs, GFP_KERNEL);
GFP_KERNEL); if (!usb_bufs->buf) {
if (!usb_bufs->transfer_buffer) { kfree(usb_bufs->buf);
kfree(usb_bufs->urb);
return -ENOMEM; return -ENOMEM;
} }
...@@ -963,37 +954,41 @@ int em28xx_alloc_urbs(struct em28xx *dev, enum em28xx_mode mode, int xfer_bulk, ...@@ -963,37 +954,41 @@ int em28xx_alloc_urbs(struct em28xx *dev, enum em28xx_mode mode, int xfer_bulk,
} }
usb_bufs->urb[i] = urb; usb_bufs->urb[i] = urb;
usb_bufs->transfer_buffer[i] = usb_alloc_coherent(udev, usb_bufs->buf[i] = kzalloc(sb_size, GFP_KERNEL);
sb_size, GFP_KERNEL, &urb->transfer_dma); if (!usb_bufs->buf[i]) {
if (!usb_bufs->transfer_buffer[i]) {
dev_err(&dev->intf->dev, dev_err(&dev->intf->dev,
"unable to allocate %i bytes for transfer buffer %i%s\n", "unable to allocate %i bytes for transfer buffer %i%s\n",
sb_size, i, sb_size, i,
in_interrupt() ? " while in int" : ""); in_interrupt() ? " while in int" : "");
em28xx_uninit_usb_xfer(dev, mode); em28xx_uninit_usb_xfer(dev, mode);
for (i--; i >= 0; i--)
kfree(usb_bufs->buf[i]);
kfree(usb_bufs->buf);
usb_bufs->buf = NULL;
return -ENOMEM; return -ENOMEM;
} }
memset(usb_bufs->transfer_buffer[i], 0, sb_size);
urb->transfer_flags = URB_FREE_BUFFER;
if (xfer_bulk) { /* bulk */ if (xfer_bulk) { /* bulk */
pipe = usb_rcvbulkpipe(udev, pipe = usb_rcvbulkpipe(udev,
mode == EM28XX_ANALOG_MODE ? mode == EM28XX_ANALOG_MODE ?
dev->analog_ep_bulk : dev->analog_ep_bulk :
dev->dvb_ep_bulk); dev->dvb_ep_bulk);
usb_fill_bulk_urb(urb, udev, pipe, usb_fill_bulk_urb(urb, udev, pipe, usb_bufs->buf[i],
usb_bufs->transfer_buffer[i], sb_size, sb_size, em28xx_irq_callback, dev);
em28xx_irq_callback, dev);
urb->transfer_flags = URB_NO_TRANSFER_DMA_MAP;
} else { /* isoc */ } else { /* isoc */
pipe = usb_rcvisocpipe(udev, pipe = usb_rcvisocpipe(udev,
mode == EM28XX_ANALOG_MODE ? mode == EM28XX_ANALOG_MODE ?
dev->analog_ep_isoc : dev->analog_ep_isoc :
dev->dvb_ep_isoc); dev->dvb_ep_isoc);
usb_fill_int_urb(urb, udev, pipe, usb_fill_int_urb(urb, udev, pipe, usb_bufs->buf[i],
usb_bufs->transfer_buffer[i], sb_size, sb_size, em28xx_irq_callback, dev, 1);
em28xx_irq_callback, dev, 1); urb->transfer_flags |= URB_ISO_ASAP;
urb->transfer_flags = URB_ISO_ASAP |
URB_NO_TRANSFER_DMA_MAP;
k = 0; k = 0;
for (j = 0; j < usb_bufs->num_packets; j++) { for (j = 0; j < usb_bufs->num_packets; j++) {
urb->iso_frame_desc[j].offset = k; urb->iso_frame_desc[j].offset = k;
......
...@@ -242,7 +242,7 @@ struct em28xx_usb_bufs { ...@@ -242,7 +242,7 @@ struct em28xx_usb_bufs {
struct urb **urb; struct urb **urb;
/* transfer buffers for isoc/bulk transfer */ /* transfer buffers for isoc/bulk transfer */
char **transfer_buffer; char **buf;
}; };
struct em28xx_usb_ctl { struct em28xx_usb_ctl {
......
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