2 * 8250_dma.c - DMA Engine API support for 8250.c
4 * Copyright (C) 2013 Intel Corporation
6 * This program is free software; you can redistribute it and/or modify
7 * it under the terms of the GNU General Public License as published by
8 * the Free Software Foundation; either version 2 of the License, or
9 * (at your option) any later version.
11 #include <linux/tty.h>
12 #include <linux/tty_flip.h>
13 #include <linux/serial_reg.h>
14 #include <linux/dma-mapping.h>
18 static void __dma_tx_complete(void *param
)
20 struct uart_8250_port
*p
= param
;
21 struct uart_8250_dma
*dma
= p
->dma
;
22 struct circ_buf
*xmit
= &p
->port
.state
->xmit
;
25 dma_sync_single_for_cpu(dma
->txchan
->device
->dev
, dma
->tx_addr
,
26 UART_XMIT_SIZE
, DMA_TO_DEVICE
);
28 spin_lock_irqsave(&p
->port
.lock
, flags
);
32 xmit
->tail
+= dma
->tx_size
;
33 xmit
->tail
&= UART_XMIT_SIZE
- 1;
34 p
->port
.icount
.tx
+= dma
->tx_size
;
36 if (uart_circ_chars_pending(xmit
) < WAKEUP_CHARS
)
37 uart_write_wakeup(&p
->port
);
39 if (!uart_circ_empty(xmit
) && !uart_tx_stopped(&p
->port
))
42 spin_unlock_irqrestore(&p
->port
.lock
, flags
);
45 static void __dma_rx_complete(void *param
)
47 struct uart_8250_port
*p
= param
;
48 struct uart_8250_dma
*dma
= p
->dma
;
49 struct tty_port
*tty_port
= &p
->port
.state
->port
;
50 struct dma_tx_state state
;
53 dma_sync_single_for_cpu(dma
->rxchan
->device
->dev
, dma
->rx_addr
,
54 dma
->rx_size
, DMA_FROM_DEVICE
);
56 dmaengine_tx_status(dma
->rxchan
, dma
->rx_cookie
, &state
);
57 dmaengine_terminate_all(dma
->rxchan
);
59 count
= dma
->rx_size
- state
.residue
;
61 tty_insert_flip_string(tty_port
, dma
->rx_buf
, count
);
62 p
->port
.icount
.rx
+= count
;
64 tty_flip_buffer_push(tty_port
);
67 int serial8250_tx_dma(struct uart_8250_port
*p
)
69 struct uart_8250_dma
*dma
= p
->dma
;
70 struct circ_buf
*xmit
= &p
->port
.state
->xmit
;
71 struct dma_async_tx_descriptor
*desc
;
73 if (uart_tx_stopped(&p
->port
) || dma
->tx_running
||
74 uart_circ_empty(xmit
))
77 dma
->tx_size
= CIRC_CNT_TO_END(xmit
->head
, xmit
->tail
, UART_XMIT_SIZE
);
79 desc
= dmaengine_prep_slave_single(dma
->txchan
,
80 dma
->tx_addr
+ xmit
->tail
,
81 dma
->tx_size
, DMA_MEM_TO_DEV
,
82 DMA_PREP_INTERRUPT
| DMA_CTRL_ACK
);
88 desc
->callback
= __dma_tx_complete
;
89 desc
->callback_param
= p
;
91 dma
->tx_cookie
= dmaengine_submit(desc
);
93 dma_sync_single_for_device(dma
->txchan
->device
->dev
, dma
->tx_addr
,
94 UART_XMIT_SIZE
, DMA_TO_DEVICE
);
96 dma_async_issue_pending(dma
->txchan
);
100 EXPORT_SYMBOL_GPL(serial8250_tx_dma
);
102 int serial8250_rx_dma(struct uart_8250_port
*p
, unsigned int iir
)
104 struct uart_8250_dma
*dma
= p
->dma
;
105 struct dma_async_tx_descriptor
*desc
;
106 struct dma_tx_state state
;
109 dma_status
= dmaengine_tx_status(dma
->rxchan
, dma
->rx_cookie
, &state
);
111 switch (iir
& 0x3f) {
113 /* 8250_core handles errors and break interrupts */
115 case UART_IIR_RX_TIMEOUT
:
117 * If RCVR FIFO trigger level was not reached, complete the
118 * transfer and let 8250_core copy the remaining data.
120 if (dma_status
== DMA_IN_PROGRESS
) {
121 dmaengine_pause(dma
->rxchan
);
122 __dma_rx_complete(p
);
132 desc
= dmaengine_prep_slave_single(dma
->rxchan
, dma
->rx_addr
,
133 dma
->rx_size
, DMA_DEV_TO_MEM
,
134 DMA_PREP_INTERRUPT
| DMA_CTRL_ACK
);
138 desc
->callback
= __dma_rx_complete
;
139 desc
->callback_param
= p
;
141 dma
->rx_cookie
= dmaengine_submit(desc
);
143 dma_sync_single_for_device(dma
->rxchan
->device
->dev
, dma
->rx_addr
,
144 dma
->rx_size
, DMA_FROM_DEVICE
);
146 dma_async_issue_pending(dma
->rxchan
);
150 EXPORT_SYMBOL_GPL(serial8250_rx_dma
);
152 int serial8250_request_dma(struct uart_8250_port
*p
)
154 struct uart_8250_dma
*dma
= p
->dma
;
157 /* Default slave configuration parameters */
158 dma
->rxconf
.direction
= DMA_DEV_TO_MEM
;
159 dma
->rxconf
.src_addr_width
= DMA_SLAVE_BUSWIDTH_1_BYTE
;
160 dma
->rxconf
.src_addr
= p
->port
.mapbase
+ UART_RX
;
162 dma
->txconf
.direction
= DMA_MEM_TO_DEV
;
163 dma
->txconf
.dst_addr_width
= DMA_SLAVE_BUSWIDTH_1_BYTE
;
164 dma
->txconf
.dst_addr
= p
->port
.mapbase
+ UART_TX
;
167 dma_cap_set(DMA_SLAVE
, mask
);
169 /* Get a channel for RX */
170 dma
->rxchan
= dma_request_slave_channel_compat(mask
,
171 dma
->fn
, dma
->rx_param
,
176 dmaengine_slave_config(dma
->rxchan
, &dma
->rxconf
);
178 /* Get a channel for TX */
179 dma
->txchan
= dma_request_slave_channel_compat(mask
,
180 dma
->fn
, dma
->tx_param
,
183 dma_release_channel(dma
->rxchan
);
187 dmaengine_slave_config(dma
->txchan
, &dma
->txconf
);
191 dma
->rx_size
= PAGE_SIZE
;
193 dma
->rx_buf
= dma_alloc_coherent(dma
->rxchan
->device
->dev
, dma
->rx_size
,
194 &dma
->rx_addr
, GFP_KERNEL
);
196 dma_release_channel(dma
->rxchan
);
197 dma_release_channel(dma
->txchan
);
202 dma
->tx_addr
= dma_map_single(dma
->txchan
->device
->dev
,
203 p
->port
.state
->xmit
.buf
,
207 dev_dbg_ratelimited(p
->port
.dev
, "got both dma channels\n");
211 EXPORT_SYMBOL_GPL(serial8250_request_dma
);
213 void serial8250_release_dma(struct uart_8250_port
*p
)
215 struct uart_8250_dma
*dma
= p
->dma
;
220 /* Release RX resources */
221 dmaengine_terminate_all(dma
->rxchan
);
222 dma_free_coherent(dma
->rxchan
->device
->dev
, dma
->rx_size
, dma
->rx_buf
,
224 dma_release_channel(dma
->rxchan
);
227 /* Release TX resources */
228 dmaengine_terminate_all(dma
->txchan
);
229 dma_unmap_single(dma
->txchan
->device
->dev
, dma
->tx_addr
,
230 UART_XMIT_SIZE
, DMA_TO_DEVICE
);
231 dma_release_channel(dma
->txchan
);
235 dev_dbg_ratelimited(p
->port
.dev
, "dma channels released\n");
237 EXPORT_SYMBOL_GPL(serial8250_release_dma
);