Commit 5dce9555 authored by Paul Zimmerman's avatar Paul Zimmerman Committed by Greg Kroah-Hartman

usb: dwc2: handle DMA buffer unmapping sanely

The driver's handling of DMA buffers for non-aligned transfers
was kind of nuts. For IN transfers, it left the URB DMA buffer
mapped until the transfer completed, then synced it, copied the
data from the bounce buffer, then synced it again.

Instead of that, just call usb_hcd_unmap_urb_for_dma() to unmap
the buffer before starting the transfer. Then no syncing is
required when doing the copy. This should also allow handling of
other types of mappings besides just dma_map_single() ones.

Also reduce the size of the bounce buffer allocation for Isoc
endpoints to 3K, since that's the largest possible transfer size.

Tested on Raspberry Pi and Altera SOCFPGA.
Signed-off-by: default avatarPaul Zimmerman <paulz@synopsys.com>
Signed-off-by: default avatarGreg Kroah-Hartman <gregkh@linuxfoundation.org>
parent e8f8c14d
...@@ -697,29 +697,45 @@ static void *dwc2_hc_init_xfer(struct dwc2_hsotg *hsotg, ...@@ -697,29 +697,45 @@ static void *dwc2_hc_init_xfer(struct dwc2_hsotg *hsotg,
} }
static int dwc2_hc_setup_align_buf(struct dwc2_hsotg *hsotg, struct dwc2_qh *qh, static int dwc2_hc_setup_align_buf(struct dwc2_hsotg *hsotg, struct dwc2_qh *qh,
struct dwc2_host_chan *chan, void *bufptr) struct dwc2_host_chan *chan,
struct dwc2_hcd_urb *urb, void *bufptr)
{ {
u32 buf_size; u32 buf_size;
struct urb *usb_urb;
if (chan->ep_type != USB_ENDPOINT_XFER_ISOC) struct usb_hcd *hcd;
buf_size = hsotg->core_params->max_transfer_size;
else
buf_size = 4096;
if (!qh->dw_align_buf) { if (!qh->dw_align_buf) {
if (chan->ep_type != USB_ENDPOINT_XFER_ISOC)
buf_size = hsotg->core_params->max_transfer_size;
else
/* 3072 = 3 max-size Isoc packets */
buf_size = 3072;
qh->dw_align_buf = dma_alloc_coherent(hsotg->dev, buf_size, qh->dw_align_buf = dma_alloc_coherent(hsotg->dev, buf_size,
&qh->dw_align_buf_dma, &qh->dw_align_buf_dma,
GFP_ATOMIC); GFP_ATOMIC);
if (!qh->dw_align_buf) if (!qh->dw_align_buf)
return -ENOMEM; return -ENOMEM;
qh->dw_align_buf_size = buf_size;
} }
if (!chan->ep_is_in && chan->xfer_len) { if (chan->xfer_len) {
dma_sync_single_for_cpu(hsotg->dev, chan->xfer_dma, buf_size, dev_vdbg(hsotg->dev, "%s(): non-aligned buffer\n", __func__);
DMA_TO_DEVICE); usb_urb = urb->priv;
memcpy(qh->dw_align_buf, bufptr, chan->xfer_len);
dma_sync_single_for_device(hsotg->dev, chan->xfer_dma, buf_size, if (usb_urb) {
DMA_TO_DEVICE); if (usb_urb->transfer_flags &
(URB_SETUP_MAP_SINGLE | URB_DMA_MAP_SG |
URB_DMA_MAP_PAGE | URB_DMA_MAP_SINGLE)) {
hcd = dwc2_hsotg_to_hcd(hsotg);
usb_hcd_unmap_urb_for_dma(hcd, usb_urb);
}
if (!chan->ep_is_in)
memcpy(qh->dw_align_buf, bufptr,
chan->xfer_len);
} else {
dev_warn(hsotg->dev, "no URB in dwc2_urb\n");
}
} }
chan->align_buf = qh->dw_align_buf_dma; chan->align_buf = qh->dw_align_buf_dma;
...@@ -828,7 +844,7 @@ static int dwc2_assign_and_init_hc(struct dwc2_hsotg *hsotg, struct dwc2_qh *qh) ...@@ -828,7 +844,7 @@ static int dwc2_assign_and_init_hc(struct dwc2_hsotg *hsotg, struct dwc2_qh *qh)
/* Non DWORD-aligned buffer case */ /* Non DWORD-aligned buffer case */
if (bufptr) { if (bufptr) {
dev_vdbg(hsotg->dev, "Non-aligned buffer\n"); dev_vdbg(hsotg->dev, "Non-aligned buffer\n");
if (dwc2_hc_setup_align_buf(hsotg, qh, chan, bufptr)) { if (dwc2_hc_setup_align_buf(hsotg, qh, chan, urb, bufptr)) {
dev_err(hsotg->dev, dev_err(hsotg->dev,
"%s: Failed to allocate memory to handle non-dword aligned buffer\n", "%s: Failed to allocate memory to handle non-dword aligned buffer\n",
__func__); __func__);
......
...@@ -243,7 +243,8 @@ enum dwc2_transaction_type { ...@@ -243,7 +243,8 @@ enum dwc2_transaction_type {
* @ntd: Actual number of transfer descriptors in a list * @ntd: Actual number of transfer descriptors in a list
* @dw_align_buf: Used instead of original buffer if its physical address * @dw_align_buf: Used instead of original buffer if its physical address
* is not dword-aligned * is not dword-aligned
* @dw_align_buf_dma: DMA address for align_buf * @dw_align_buf_size: Size of dw_align_buf
* @dw_align_buf_dma: DMA address for dw_align_buf
* @qtd_list: List of QTDs for this QH * @qtd_list: List of QTDs for this QH
* @channel: Host channel currently processing transfers for this QH * @channel: Host channel currently processing transfers for this QH
* @qh_list_entry: Entry for QH in either the periodic or non-periodic * @qh_list_entry: Entry for QH in either the periodic or non-periodic
...@@ -276,6 +277,7 @@ struct dwc2_qh { ...@@ -276,6 +277,7 @@ struct dwc2_qh {
u16 start_split_frame; u16 start_split_frame;
u16 ntd; u16 ntd;
u8 *dw_align_buf; u8 *dw_align_buf;
int dw_align_buf_size;
dma_addr_t dw_align_buf_dma; dma_addr_t dw_align_buf_dma;
struct list_head qtd_list; struct list_head qtd_list;
struct dwc2_host_chan *channel; struct dwc2_host_chan *channel;
......
...@@ -465,12 +465,8 @@ static int dwc2_update_urb_state(struct dwc2_hsotg *hsotg, ...@@ -465,12 +465,8 @@ static int dwc2_update_urb_state(struct dwc2_hsotg *hsotg,
/* Non DWORD-aligned buffer case handling */ /* Non DWORD-aligned buffer case handling */
if (chan->align_buf && xfer_length && chan->ep_is_in) { if (chan->align_buf && xfer_length && chan->ep_is_in) {
dev_vdbg(hsotg->dev, "%s(): non-aligned buffer\n", __func__); dev_vdbg(hsotg->dev, "%s(): non-aligned buffer\n", __func__);
dma_sync_single_for_cpu(hsotg->dev, urb->dma, urb->length,
DMA_FROM_DEVICE);
memcpy(urb->buf + urb->actual_length, chan->qh->dw_align_buf, memcpy(urb->buf + urb->actual_length, chan->qh->dw_align_buf,
xfer_length); xfer_length);
dma_sync_single_for_device(hsotg->dev, urb->dma, urb->length,
DMA_FROM_DEVICE);
} }
dev_vdbg(hsotg->dev, "urb->actual_length=%d xfer_length=%d\n", dev_vdbg(hsotg->dev, "urb->actual_length=%d xfer_length=%d\n",
...@@ -560,14 +556,9 @@ static enum dwc2_halt_status dwc2_update_isoc_urb_state( ...@@ -560,14 +556,9 @@ static enum dwc2_halt_status dwc2_update_isoc_urb_state(
chan->ep_is_in) { chan->ep_is_in) {
dev_vdbg(hsotg->dev, "%s(): non-aligned buffer\n", dev_vdbg(hsotg->dev, "%s(): non-aligned buffer\n",
__func__); __func__);
dma_sync_single_for_cpu(hsotg->dev, urb->dma,
urb->length, DMA_FROM_DEVICE);
memcpy(urb->buf + frame_desc->offset + memcpy(urb->buf + frame_desc->offset +
qtd->isoc_split_offset, chan->qh->dw_align_buf, qtd->isoc_split_offset, chan->qh->dw_align_buf,
frame_desc->actual_length); frame_desc->actual_length);
dma_sync_single_for_device(hsotg->dev, urb->dma,
urb->length,
DMA_FROM_DEVICE);
} }
break; break;
case DWC2_HC_XFER_FRAME_OVERRUN: case DWC2_HC_XFER_FRAME_OVERRUN:
...@@ -594,14 +585,9 @@ static enum dwc2_halt_status dwc2_update_isoc_urb_state( ...@@ -594,14 +585,9 @@ static enum dwc2_halt_status dwc2_update_isoc_urb_state(
chan->ep_is_in) { chan->ep_is_in) {
dev_vdbg(hsotg->dev, "%s(): non-aligned buffer\n", dev_vdbg(hsotg->dev, "%s(): non-aligned buffer\n",
__func__); __func__);
dma_sync_single_for_cpu(hsotg->dev, urb->dma,
urb->length, DMA_FROM_DEVICE);
memcpy(urb->buf + frame_desc->offset + memcpy(urb->buf + frame_desc->offset +
qtd->isoc_split_offset, chan->qh->dw_align_buf, qtd->isoc_split_offset, chan->qh->dw_align_buf,
frame_desc->actual_length); frame_desc->actual_length);
dma_sync_single_for_device(hsotg->dev, urb->dma,
urb->length,
DMA_FROM_DEVICE);
} }
/* Skip whole frame */ /* Skip whole frame */
...@@ -937,12 +923,8 @@ static int dwc2_xfercomp_isoc_split_in(struct dwc2_hsotg *hsotg, ...@@ -937,12 +923,8 @@ static int dwc2_xfercomp_isoc_split_in(struct dwc2_hsotg *hsotg,
if (chan->align_buf) { if (chan->align_buf) {
dev_vdbg(hsotg->dev, "%s(): non-aligned buffer\n", __func__); dev_vdbg(hsotg->dev, "%s(): non-aligned buffer\n", __func__);
dma_sync_single_for_cpu(hsotg->dev, qtd->urb->dma,
qtd->urb->length, DMA_FROM_DEVICE);
memcpy(qtd->urb->buf + frame_desc->offset + memcpy(qtd->urb->buf + frame_desc->offset +
qtd->isoc_split_offset, chan->qh->dw_align_buf, len); qtd->isoc_split_offset, chan->qh->dw_align_buf, len);
dma_sync_single_for_device(hsotg->dev, qtd->urb->dma,
qtd->urb->length, DMA_FROM_DEVICE);
} }
qtd->isoc_split_offset += len; qtd->isoc_split_offset += len;
...@@ -1170,12 +1152,8 @@ static void dwc2_update_urb_state_abn(struct dwc2_hsotg *hsotg, ...@@ -1170,12 +1152,8 @@ static void dwc2_update_urb_state_abn(struct dwc2_hsotg *hsotg,
/* Non DWORD-aligned buffer case handling */ /* Non DWORD-aligned buffer case handling */
if (chan->align_buf && xfer_length && chan->ep_is_in) { if (chan->align_buf && xfer_length && chan->ep_is_in) {
dev_vdbg(hsotg->dev, "%s(): non-aligned buffer\n", __func__); dev_vdbg(hsotg->dev, "%s(): non-aligned buffer\n", __func__);
dma_sync_single_for_cpu(hsotg->dev, urb->dma, urb->length,
DMA_FROM_DEVICE);
memcpy(urb->buf + urb->actual_length, chan->qh->dw_align_buf, memcpy(urb->buf + urb->actual_length, chan->qh->dw_align_buf,
xfer_length); xfer_length);
dma_sync_single_for_device(hsotg->dev, urb->dma, urb->length,
DMA_FROM_DEVICE);
} }
urb->actual_length += xfer_length; urb->actual_length += xfer_length;
......
...@@ -229,19 +229,11 @@ static struct dwc2_qh *dwc2_hcd_qh_create(struct dwc2_hsotg *hsotg, ...@@ -229,19 +229,11 @@ static struct dwc2_qh *dwc2_hcd_qh_create(struct dwc2_hsotg *hsotg,
*/ */
void dwc2_hcd_qh_free(struct dwc2_hsotg *hsotg, struct dwc2_qh *qh) void dwc2_hcd_qh_free(struct dwc2_hsotg *hsotg, struct dwc2_qh *qh)
{ {
u32 buf_size; if (hsotg->core_params->dma_desc_enable > 0)
if (hsotg->core_params->dma_desc_enable > 0) {
dwc2_hcd_qh_free_ddma(hsotg, qh); dwc2_hcd_qh_free_ddma(hsotg, qh);
} else if (qh->dw_align_buf) { else if (qh->dw_align_buf)
if (qh->ep_type == USB_ENDPOINT_XFER_ISOC) dma_free_coherent(hsotg->dev, qh->dw_align_buf_size,
buf_size = 4096; qh->dw_align_buf, qh->dw_align_buf_dma);
else
buf_size = hsotg->core_params->max_transfer_size;
dma_free_coherent(hsotg->dev, buf_size, qh->dw_align_buf,
qh->dw_align_buf_dma);
}
kfree(qh); kfree(qh);
} }
......
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