diff options
author | Laxman Dewangan <ldewangan@nvidia.com> | 2012-01-09 15:05:11 -0500 |
---|---|---|
committer | Olof Johansson <olof@lixom.net> | 2012-02-06 21:24:59 -0500 |
commit | cb3732d0dc9df198c889a26210b6b27bc51a1c4a (patch) | |
tree | b95a5a952d14897c093d09af778dcafa39cf756e | |
parent | 941b8db1df8bfc29a88fc8e3e203289d84a3f64d (diff) |
ARM: tegra: Pause DMA when reading transfer count
In order to read an accurate channel transfer count
from the APB DMA engine, the DMA controller must be
paused first.
Signed-off-by: Laxman Dewangan <ldewangan@nvidia.com>
Acked-by: Stephen Warren <swarren@nvidia.com>
Tested-by: Stephen Warren <swarren@nvidia.com>
Signed-off-by: Olof Johansson <olof@lixom.net>
-rw-r--r-- | arch/arm/mach-tegra/dma.c | 116 |
1 files changed, 74 insertions, 42 deletions
diff --git a/arch/arm/mach-tegra/dma.c b/arch/arm/mach-tegra/dma.c index 122e46770d93..998c55ddca42 100644 --- a/arch/arm/mach-tegra/dma.c +++ b/arch/arm/mach-tegra/dma.c | |||
@@ -135,6 +135,7 @@ struct tegra_dma_channel { | |||
135 | 135 | ||
136 | static bool tegra_dma_initialized; | 136 | static bool tegra_dma_initialized; |
137 | static DEFINE_MUTEX(tegra_dma_lock); | 137 | static DEFINE_MUTEX(tegra_dma_lock); |
138 | static DEFINE_SPINLOCK(enable_lock); | ||
138 | 139 | ||
139 | static DECLARE_BITMAP(channel_usage, NV_DMA_MAX_CHANNELS); | 140 | static DECLARE_BITMAP(channel_usage, NV_DMA_MAX_CHANNELS); |
140 | static struct tegra_dma_channel dma_channels[NV_DMA_MAX_CHANNELS]; | 141 | static struct tegra_dma_channel dma_channels[NV_DMA_MAX_CHANNELS]; |
@@ -200,18 +201,82 @@ static int tegra_dma_cancel(struct tegra_dma_channel *ch) | |||
200 | return 0; | 201 | return 0; |
201 | } | 202 | } |
202 | 203 | ||
204 | static unsigned int get_channel_status(struct tegra_dma_channel *ch, | ||
205 | struct tegra_dma_req *req, bool is_stop_dma) | ||
206 | { | ||
207 | void __iomem *addr = IO_ADDRESS(TEGRA_APB_DMA_BASE); | ||
208 | unsigned int status; | ||
209 | |||
210 | if (is_stop_dma) { | ||
211 | /* | ||
212 | * STOP the DMA and get the transfer count. | ||
213 | * Getting the transfer count is tricky. | ||
214 | * - Globally disable DMA on all channels | ||
215 | * - Read the channel's status register to know the number | ||
216 | * of pending bytes to be transfered. | ||
217 | * - Stop the dma channel | ||
218 | * - Globally re-enable DMA to resume other transfers | ||
219 | */ | ||
220 | spin_lock(&enable_lock); | ||
221 | writel(0, addr + APB_DMA_GEN); | ||
222 | udelay(20); | ||
223 | status = readl(ch->addr + APB_DMA_CHAN_STA); | ||
224 | tegra_dma_stop(ch); | ||
225 | writel(GEN_ENABLE, addr + APB_DMA_GEN); | ||
226 | spin_unlock(&enable_lock); | ||
227 | if (status & STA_ISE_EOC) { | ||
228 | pr_err("Got Dma Int here clearing"); | ||
229 | writel(status, ch->addr + APB_DMA_CHAN_STA); | ||
230 | } | ||
231 | req->status = TEGRA_DMA_REQ_ERROR_ABORTED; | ||
232 | } else { | ||
233 | status = readl(ch->addr + APB_DMA_CHAN_STA); | ||
234 | } | ||
235 | return status; | ||
236 | } | ||
237 | |||
238 | /* should be called with the channel lock held */ | ||
239 | static unsigned int dma_active_count(struct tegra_dma_channel *ch, | ||
240 | struct tegra_dma_req *req, unsigned int status) | ||
241 | { | ||
242 | unsigned int to_transfer; | ||
243 | unsigned int req_transfer_count; | ||
244 | unsigned int bytes_transferred; | ||
245 | |||
246 | to_transfer = ((status & STA_COUNT_MASK) >> STA_COUNT_SHIFT) + 1; | ||
247 | req_transfer_count = ch->req_transfer_count + 1; | ||
248 | bytes_transferred = req_transfer_count; | ||
249 | if (status & STA_BUSY) | ||
250 | bytes_transferred -= to_transfer; | ||
251 | /* | ||
252 | * In continuous transfer mode, DMA only tracks the count of the | ||
253 | * half DMA buffer. So, if the DMA already finished half the DMA | ||
254 | * then add the half buffer to the completed count. | ||
255 | */ | ||
256 | if (ch->mode & TEGRA_DMA_MODE_CONTINOUS) { | ||
257 | if (req->buffer_status == TEGRA_DMA_REQ_BUF_STATUS_HALF_FULL) | ||
258 | bytes_transferred += req_transfer_count; | ||
259 | if (status & STA_ISE_EOC) | ||
260 | bytes_transferred += req_transfer_count; | ||
261 | } | ||
262 | bytes_transferred *= 4; | ||
263 | return bytes_transferred; | ||
264 | } | ||
265 | |||
203 | int tegra_dma_dequeue_req(struct tegra_dma_channel *ch, | 266 | int tegra_dma_dequeue_req(struct tegra_dma_channel *ch, |
204 | struct tegra_dma_req *_req) | 267 | struct tegra_dma_req *_req) |
205 | { | 268 | { |
206 | unsigned int csr; | ||
207 | unsigned int status; | 269 | unsigned int status; |
208 | struct tegra_dma_req *req = NULL; | 270 | struct tegra_dma_req *req = NULL; |
209 | int found = 0; | 271 | int found = 0; |
210 | unsigned long irq_flags; | 272 | unsigned long irq_flags; |
211 | int to_transfer; | 273 | int stop = 0; |
212 | int req_transfer_count; | ||
213 | 274 | ||
214 | spin_lock_irqsave(&ch->lock, irq_flags); | 275 | spin_lock_irqsave(&ch->lock, irq_flags); |
276 | |||
277 | if (list_entry(ch->list.next, struct tegra_dma_req, node) == _req) | ||
278 | stop = 1; | ||
279 | |||
215 | list_for_each_entry(req, &ch->list, node) { | 280 | list_for_each_entry(req, &ch->list, node) { |
216 | if (req == _req) { | 281 | if (req == _req) { |
217 | list_del(&req->node); | 282 | list_del(&req->node); |
@@ -224,47 +289,12 @@ int tegra_dma_dequeue_req(struct tegra_dma_channel *ch, | |||
224 | return 0; | 289 | return 0; |
225 | } | 290 | } |
226 | 291 | ||
227 | /* STOP the DMA and get the transfer count. | 292 | if (!stop) |
228 | * Getting the transfer count is tricky. | 293 | goto skip_stop_dma; |
229 | * - Change the source selector to invalid to stop the DMA from | ||
230 | * FIFO to memory. | ||
231 | * - Read the status register to know the number of pending | ||
232 | * bytes to be transferred. | ||
233 | * - Finally stop or program the DMA to the next buffer in the | ||
234 | * list. | ||
235 | */ | ||
236 | csr = readl(ch->addr + APB_DMA_CHAN_CSR); | ||
237 | csr &= ~CSR_REQ_SEL_MASK; | ||
238 | csr |= CSR_REQ_SEL_INVALID; | ||
239 | writel(csr, ch->addr + APB_DMA_CHAN_CSR); | ||
240 | |||
241 | /* Get the transfer count */ | ||
242 | status = readl(ch->addr + APB_DMA_CHAN_STA); | ||
243 | to_transfer = (status & STA_COUNT_MASK) >> STA_COUNT_SHIFT; | ||
244 | req_transfer_count = ch->req_transfer_count; | ||
245 | req_transfer_count += 1; | ||
246 | to_transfer += 1; | ||
247 | 294 | ||
248 | req->bytes_transferred = req_transfer_count; | 295 | status = get_channel_status(ch, req, true); |
296 | req->bytes_transferred = dma_active_count(ch, req, status); | ||
249 | 297 | ||
250 | if (status & STA_BUSY) | ||
251 | req->bytes_transferred -= to_transfer; | ||
252 | |||
253 | /* In continuous transfer mode, DMA only tracks the count of the | ||
254 | * half DMA buffer. So, if the DMA already finished half the DMA | ||
255 | * then add the half buffer to the completed count. | ||
256 | * | ||
257 | * FIXME: There can be a race here. What if the req to | ||
258 | * dequue happens at the same time as the DMA just moved to | ||
259 | * the new buffer and SW didn't yet received the interrupt? | ||
260 | */ | ||
261 | if (ch->mode & TEGRA_DMA_MODE_CONTINOUS) | ||
262 | if (req->buffer_status == TEGRA_DMA_REQ_BUF_STATUS_HALF_FULL) | ||
263 | req->bytes_transferred += req_transfer_count; | ||
264 | |||
265 | req->bytes_transferred *= 4; | ||
266 | |||
267 | tegra_dma_stop(ch); | ||
268 | if (!list_empty(&ch->list)) { | 298 | if (!list_empty(&ch->list)) { |
269 | /* if the list is not empty, queue the next request */ | 299 | /* if the list is not empty, queue the next request */ |
270 | struct tegra_dma_req *next_req; | 300 | struct tegra_dma_req *next_req; |
@@ -272,6 +302,8 @@ int tegra_dma_dequeue_req(struct tegra_dma_channel *ch, | |||
272 | typeof(*next_req), node); | 302 | typeof(*next_req), node); |
273 | tegra_dma_update_hw(ch, next_req); | 303 | tegra_dma_update_hw(ch, next_req); |
274 | } | 304 | } |
305 | |||
306 | skip_stop_dma: | ||
275 | req->status = -TEGRA_DMA_REQ_ERROR_ABORTED; | 307 | req->status = -TEGRA_DMA_REQ_ERROR_ABORTED; |
276 | 308 | ||
277 | spin_unlock_irqrestore(&ch->lock, irq_flags); | 309 | spin_unlock_irqrestore(&ch->lock, irq_flags); |