blob: 00f8e566cf12141b0e38162015ed6d5e57794183 [file] [log] [blame]
Russell King7bedaa52012-04-13 12:10:24 +01001/*
2 * OMAP DMAengine support
3 *
4 * This program is free software; you can redistribute it and/or modify
5 * it under the terms of the GNU General Public License version 2 as
6 * published by the Free Software Foundation.
7 */
Russell Kingfa3ad862013-11-02 17:07:09 +00008#include <linux/delay.h>
Russell King7bedaa52012-04-13 12:10:24 +01009#include <linux/dmaengine.h>
10#include <linux/dma-mapping.h>
11#include <linux/err.h>
12#include <linux/init.h>
13#include <linux/interrupt.h>
14#include <linux/list.h>
15#include <linux/module.h>
16#include <linux/omap-dma.h>
17#include <linux/platform_device.h>
18#include <linux/slab.h>
19#include <linux/spinlock.h>
Jon Hunter8d306622013-02-26 12:27:24 -060020#include <linux/of_dma.h>
21#include <linux/of_device.h>
Russell King7bedaa52012-04-13 12:10:24 +010022
23#include "virt-dma.h"
Tony Lindgren7d7e1eb2012-08-27 17:43:01 -070024
Russell King7bedaa52012-04-13 12:10:24 +010025struct omap_dmadev {
26 struct dma_device ddev;
27 spinlock_t lock;
28 struct tasklet_struct task;
29 struct list_head pending;
Russell King596c4712013-12-10 11:08:01 +000030 void __iomem *base;
31 const struct omap_dma_reg *reg_map;
Russell King1b416c42013-11-02 13:00:03 +000032 struct omap_system_dma_plat_info *plat;
Russell King7bedaa52012-04-13 12:10:24 +010033};
34
35struct omap_chan {
36 struct virt_dma_chan vc;
37 struct list_head node;
Russell King596c4712013-12-10 11:08:01 +000038 void __iomem *channel_base;
39 const struct omap_dma_reg *reg_map;
Russell King7bedaa52012-04-13 12:10:24 +010040
41 struct dma_slave_config cfg;
42 unsigned dma_sig;
Russell King3a774ea2012-06-21 10:40:15 +010043 bool cyclic;
Peter Ujfalusi2dcdf572012-09-14 15:05:45 +030044 bool paused;
Russell King7bedaa52012-04-13 12:10:24 +010045
46 int dma_ch;
47 struct omap_desc *desc;
48 unsigned sgidx;
49};
50
51struct omap_sg {
52 dma_addr_t addr;
53 uint32_t en; /* number of elements (24-bit) */
54 uint32_t fn; /* number of frames (16-bit) */
55};
56
57struct omap_desc {
58 struct virt_dma_desc vd;
59 enum dma_transfer_direction dir;
60 dma_addr_t dev_addr;
61
Russell King7c836bc2012-06-18 16:45:19 +010062 int16_t fi; /* for OMAP_DMA_SYNC_PACKET */
Russell King90438262013-11-02 19:57:06 +000063 uint8_t es; /* CSDP_DATA_TYPE_xxx */
Russell King3ed4d182013-11-02 19:16:09 +000064 uint32_t ccr; /* CCR value */
Russell King965aeb4d2013-11-06 17:12:30 +000065 uint16_t clnk_ctrl; /* CLNK_CTRL value */
Russell Kingfa3ad862013-11-02 17:07:09 +000066 uint16_t cicr; /* CICR value */
Russell King2f0d13b2013-11-02 18:51:53 +000067 uint32_t csdp; /* CSDP value */
Russell King7bedaa52012-04-13 12:10:24 +010068
69 unsigned sglen;
70 struct omap_sg sg[0];
71};
72
Russell King90438262013-11-02 19:57:06 +000073enum {
74 CCR_FS = BIT(5),
75 CCR_READ_PRIORITY = BIT(6),
76 CCR_ENABLE = BIT(7),
77 CCR_AUTO_INIT = BIT(8), /* OMAP1 only */
78 CCR_REPEAT = BIT(9), /* OMAP1 only */
79 CCR_OMAP31_DISABLE = BIT(10), /* OMAP1 only */
80 CCR_SUSPEND_SENSITIVE = BIT(8), /* OMAP2+ only */
81 CCR_RD_ACTIVE = BIT(9), /* OMAP2+ only */
82 CCR_WR_ACTIVE = BIT(10), /* OMAP2+ only */
83 CCR_SRC_AMODE_CONSTANT = 0 << 12,
84 CCR_SRC_AMODE_POSTINC = 1 << 12,
85 CCR_SRC_AMODE_SGLIDX = 2 << 12,
86 CCR_SRC_AMODE_DBLIDX = 3 << 12,
87 CCR_DST_AMODE_CONSTANT = 0 << 14,
88 CCR_DST_AMODE_POSTINC = 1 << 14,
89 CCR_DST_AMODE_SGLIDX = 2 << 14,
90 CCR_DST_AMODE_DBLIDX = 3 << 14,
91 CCR_CONSTANT_FILL = BIT(16),
92 CCR_TRANSPARENT_COPY = BIT(17),
93 CCR_BS = BIT(18),
94 CCR_SUPERVISOR = BIT(22),
95 CCR_PREFETCH = BIT(23),
96 CCR_TRIGGER_SRC = BIT(24),
97 CCR_BUFFERING_DISABLE = BIT(25),
98 CCR_WRITE_PRIORITY = BIT(26),
99 CCR_SYNC_ELEMENT = 0,
100 CCR_SYNC_FRAME = CCR_FS,
101 CCR_SYNC_BLOCK = CCR_BS,
102 CCR_SYNC_PACKET = CCR_BS | CCR_FS,
103
104 CSDP_DATA_TYPE_8 = 0,
105 CSDP_DATA_TYPE_16 = 1,
106 CSDP_DATA_TYPE_32 = 2,
107 CSDP_SRC_PORT_EMIFF = 0 << 2, /* OMAP1 only */
108 CSDP_SRC_PORT_EMIFS = 1 << 2, /* OMAP1 only */
109 CSDP_SRC_PORT_OCP_T1 = 2 << 2, /* OMAP1 only */
110 CSDP_SRC_PORT_TIPB = 3 << 2, /* OMAP1 only */
111 CSDP_SRC_PORT_OCP_T2 = 4 << 2, /* OMAP1 only */
112 CSDP_SRC_PORT_MPUI = 5 << 2, /* OMAP1 only */
113 CSDP_SRC_PACKED = BIT(6),
114 CSDP_SRC_BURST_1 = 0 << 7,
115 CSDP_SRC_BURST_16 = 1 << 7,
116 CSDP_SRC_BURST_32 = 2 << 7,
117 CSDP_SRC_BURST_64 = 3 << 7,
118 CSDP_DST_PORT_EMIFF = 0 << 9, /* OMAP1 only */
119 CSDP_DST_PORT_EMIFS = 1 << 9, /* OMAP1 only */
120 CSDP_DST_PORT_OCP_T1 = 2 << 9, /* OMAP1 only */
121 CSDP_DST_PORT_TIPB = 3 << 9, /* OMAP1 only */
122 CSDP_DST_PORT_OCP_T2 = 4 << 9, /* OMAP1 only */
123 CSDP_DST_PORT_MPUI = 5 << 9, /* OMAP1 only */
124 CSDP_DST_PACKED = BIT(13),
125 CSDP_DST_BURST_1 = 0 << 14,
126 CSDP_DST_BURST_16 = 1 << 14,
127 CSDP_DST_BURST_32 = 2 << 14,
128 CSDP_DST_BURST_64 = 3 << 14,
129
130 CICR_TOUT_IE = BIT(0), /* OMAP1 only */
131 CICR_DROP_IE = BIT(1),
132 CICR_HALF_IE = BIT(2),
133 CICR_FRAME_IE = BIT(3),
134 CICR_LAST_IE = BIT(4),
135 CICR_BLOCK_IE = BIT(5),
136 CICR_PKT_IE = BIT(7), /* OMAP2+ only */
137 CICR_TRANS_ERR_IE = BIT(8), /* OMAP2+ only */
138 CICR_SUPERVISOR_ERR_IE = BIT(10), /* OMAP2+ only */
139 CICR_MISALIGNED_ERR_IE = BIT(11), /* OMAP2+ only */
140 CICR_DRAIN_IE = BIT(12), /* OMAP2+ only */
141 CICR_SUPER_BLOCK_IE = BIT(14), /* OMAP2+ only */
142
143 CLNK_CTRL_ENABLE_LNK = BIT(15),
144};
145
Russell King7bedaa52012-04-13 12:10:24 +0100146static const unsigned es_bytes[] = {
Russell King90438262013-11-02 19:57:06 +0000147 [CSDP_DATA_TYPE_8] = 1,
148 [CSDP_DATA_TYPE_16] = 2,
149 [CSDP_DATA_TYPE_32] = 4,
Russell King7bedaa52012-04-13 12:10:24 +0100150};
151
Jon Hunter8d306622013-02-26 12:27:24 -0600152static struct of_dma_filter_info omap_dma_info = {
153 .filter_fn = omap_dma_filter_fn,
154};
155
Russell King7bedaa52012-04-13 12:10:24 +0100156static inline struct omap_dmadev *to_omap_dma_dev(struct dma_device *d)
157{
158 return container_of(d, struct omap_dmadev, ddev);
159}
160
161static inline struct omap_chan *to_omap_dma_chan(struct dma_chan *c)
162{
163 return container_of(c, struct omap_chan, vc.chan);
164}
165
166static inline struct omap_desc *to_omap_dma_desc(struct dma_async_tx_descriptor *t)
167{
168 return container_of(t, struct omap_desc, vd.tx);
169}
170
171static void omap_dma_desc_free(struct virt_dma_desc *vd)
172{
173 kfree(container_of(vd, struct omap_desc, vd));
174}
175
Russell King596c4712013-12-10 11:08:01 +0000176static void omap_dma_write(uint32_t val, unsigned type, void __iomem *addr)
177{
178 switch (type) {
179 case OMAP_DMA_REG_16BIT:
180 writew_relaxed(val, addr);
181 break;
182 case OMAP_DMA_REG_2X16BIT:
183 writew_relaxed(val, addr);
184 writew_relaxed(val >> 16, addr + 2);
185 break;
186 case OMAP_DMA_REG_32BIT:
187 writel_relaxed(val, addr);
188 break;
189 default:
190 WARN_ON(1);
191 }
192}
193
194static unsigned omap_dma_read(unsigned type, void __iomem *addr)
195{
196 unsigned val;
197
198 switch (type) {
199 case OMAP_DMA_REG_16BIT:
200 val = readw_relaxed(addr);
201 break;
202 case OMAP_DMA_REG_2X16BIT:
203 val = readw_relaxed(addr);
204 val |= readw_relaxed(addr + 2) << 16;
205 break;
206 case OMAP_DMA_REG_32BIT:
207 val = readl_relaxed(addr);
208 break;
209 default:
210 WARN_ON(1);
211 val = 0;
212 }
213
214 return val;
215}
216
Russell Kingc5ed98b2013-11-06 17:33:09 +0000217static void omap_dma_glbl_write(struct omap_dmadev *od, unsigned reg, unsigned val)
218{
Russell King596c4712013-12-10 11:08:01 +0000219 const struct omap_dma_reg *r = od->reg_map + reg;
220
221 WARN_ON(r->stride);
222
223 omap_dma_write(val, r->type, od->base + r->offset);
Russell Kingc5ed98b2013-11-06 17:33:09 +0000224}
225
226static unsigned omap_dma_glbl_read(struct omap_dmadev *od, unsigned reg)
227{
Russell King596c4712013-12-10 11:08:01 +0000228 const struct omap_dma_reg *r = od->reg_map + reg;
229
230 WARN_ON(r->stride);
231
232 return omap_dma_read(r->type, od->base + r->offset);
Russell Kingc5ed98b2013-11-06 17:33:09 +0000233}
234
235static void omap_dma_chan_write(struct omap_chan *c, unsigned reg, unsigned val)
236{
Russell King596c4712013-12-10 11:08:01 +0000237 const struct omap_dma_reg *r = c->reg_map + reg;
238
239 omap_dma_write(val, r->type, c->channel_base + r->offset);
Russell Kingc5ed98b2013-11-06 17:33:09 +0000240}
241
242static unsigned omap_dma_chan_read(struct omap_chan *c, unsigned reg)
243{
Russell King596c4712013-12-10 11:08:01 +0000244 const struct omap_dma_reg *r = c->reg_map + reg;
245
246 return omap_dma_read(r->type, c->channel_base + r->offset);
Russell Kingc5ed98b2013-11-06 17:33:09 +0000247}
248
Russell King470b23f2013-11-02 21:23:06 +0000249static void omap_dma_clear_csr(struct omap_chan *c)
250{
251 if (dma_omap1())
Russell Kingc5ed98b2013-11-06 17:33:09 +0000252 omap_dma_chan_read(c, CSR);
Russell King470b23f2013-11-02 21:23:06 +0000253 else
Russell Kingc5ed98b2013-11-06 17:33:09 +0000254 omap_dma_chan_write(c, CSR, ~0);
Russell King470b23f2013-11-02 21:23:06 +0000255}
256
Russell King596c4712013-12-10 11:08:01 +0000257static void omap_dma_assign(struct omap_dmadev *od, struct omap_chan *c,
258 unsigned lch)
259{
260 c->channel_base = od->base + od->plat->channel_stride * lch;
261}
262
Russell Kingfa3ad862013-11-02 17:07:09 +0000263static void omap_dma_start(struct omap_chan *c, struct omap_desc *d)
264{
265 struct omap_dmadev *od = to_omap_dma_dev(c->vc.chan.device);
Russell Kingfa3ad862013-11-02 17:07:09 +0000266
267 if (__dma_omap15xx(od->plat->dma_attr))
Russell Kingc5ed98b2013-11-06 17:33:09 +0000268 omap_dma_chan_write(c, CPC, 0);
Russell Kingfa3ad862013-11-02 17:07:09 +0000269 else
Russell Kingc5ed98b2013-11-06 17:33:09 +0000270 omap_dma_chan_write(c, CDAC, 0);
Russell Kingfa3ad862013-11-02 17:07:09 +0000271
Russell King470b23f2013-11-02 21:23:06 +0000272 omap_dma_clear_csr(c);
Russell Kingfa3ad862013-11-02 17:07:09 +0000273
274 /* Enable interrupts */
Russell Kingc5ed98b2013-11-06 17:33:09 +0000275 omap_dma_chan_write(c, CICR, d->cicr);
Russell Kingfa3ad862013-11-02 17:07:09 +0000276
Russell King45da7b02013-11-06 17:18:42 +0000277 /* Enable channel */
Russell Kingc5ed98b2013-11-06 17:33:09 +0000278 omap_dma_chan_write(c, CCR, d->ccr | CCR_ENABLE);
Russell Kingfa3ad862013-11-02 17:07:09 +0000279}
280
281static void omap_dma_stop(struct omap_chan *c)
282{
283 struct omap_dmadev *od = to_omap_dma_dev(c->vc.chan.device);
284 uint32_t val;
285
286 /* disable irq */
Russell Kingc5ed98b2013-11-06 17:33:09 +0000287 omap_dma_chan_write(c, CICR, 0);
Russell Kingfa3ad862013-11-02 17:07:09 +0000288
Russell King470b23f2013-11-02 21:23:06 +0000289 omap_dma_clear_csr(c);
Russell Kingfa3ad862013-11-02 17:07:09 +0000290
Russell Kingc5ed98b2013-11-06 17:33:09 +0000291 val = omap_dma_chan_read(c, CCR);
Russell King90438262013-11-02 19:57:06 +0000292 if (od->plat->errata & DMA_ERRATA_i541 && val & CCR_TRIGGER_SRC) {
Russell Kingfa3ad862013-11-02 17:07:09 +0000293 uint32_t sysconfig;
294 unsigned i;
295
Russell Kingc5ed98b2013-11-06 17:33:09 +0000296 sysconfig = omap_dma_glbl_read(od, OCP_SYSCONFIG);
Russell Kingfa3ad862013-11-02 17:07:09 +0000297 val = sysconfig & ~DMA_SYSCONFIG_MIDLEMODE_MASK;
298 val |= DMA_SYSCONFIG_MIDLEMODE(DMA_IDLEMODE_NO_IDLE);
Russell Kingc5ed98b2013-11-06 17:33:09 +0000299 omap_dma_glbl_write(od, OCP_SYSCONFIG, val);
Russell Kingfa3ad862013-11-02 17:07:09 +0000300
Russell Kingc5ed98b2013-11-06 17:33:09 +0000301 val = omap_dma_chan_read(c, CCR);
Russell King90438262013-11-02 19:57:06 +0000302 val &= ~CCR_ENABLE;
Russell Kingc5ed98b2013-11-06 17:33:09 +0000303 omap_dma_chan_write(c, CCR, val);
Russell Kingfa3ad862013-11-02 17:07:09 +0000304
305 /* Wait for sDMA FIFO to drain */
306 for (i = 0; ; i++) {
Russell Kingc5ed98b2013-11-06 17:33:09 +0000307 val = omap_dma_chan_read(c, CCR);
Russell King90438262013-11-02 19:57:06 +0000308 if (!(val & (CCR_RD_ACTIVE | CCR_WR_ACTIVE)))
Russell Kingfa3ad862013-11-02 17:07:09 +0000309 break;
310
311 if (i > 100)
312 break;
313
314 udelay(5);
315 }
316
Russell King90438262013-11-02 19:57:06 +0000317 if (val & (CCR_RD_ACTIVE | CCR_WR_ACTIVE))
Russell Kingfa3ad862013-11-02 17:07:09 +0000318 dev_err(c->vc.chan.device->dev,
319 "DMA drain did not complete on lch %d\n",
320 c->dma_ch);
321
Russell Kingc5ed98b2013-11-06 17:33:09 +0000322 omap_dma_glbl_write(od, OCP_SYSCONFIG, sysconfig);
Russell Kingfa3ad862013-11-02 17:07:09 +0000323 } else {
Russell King90438262013-11-02 19:57:06 +0000324 val &= ~CCR_ENABLE;
Russell Kingc5ed98b2013-11-06 17:33:09 +0000325 omap_dma_chan_write(c, CCR, val);
Russell Kingfa3ad862013-11-02 17:07:09 +0000326 }
327
328 mb();
329
330 if (!__dma_omap15xx(od->plat->dma_attr) && c->cyclic) {
Russell Kingc5ed98b2013-11-06 17:33:09 +0000331 val = omap_dma_chan_read(c, CLNK_CTRL);
Russell Kingfa3ad862013-11-02 17:07:09 +0000332
333 if (dma_omap1())
334 val |= 1 << 14; /* set the STOP_LNK bit */
335 else
Russell King90438262013-11-02 19:57:06 +0000336 val &= ~CLNK_CTRL_ENABLE_LNK;
Russell Kingfa3ad862013-11-02 17:07:09 +0000337
Russell Kingc5ed98b2013-11-06 17:33:09 +0000338 omap_dma_chan_write(c, CLNK_CTRL, val);
Russell Kingfa3ad862013-11-02 17:07:09 +0000339 }
340}
341
Russell King7bedaa52012-04-13 12:10:24 +0100342static void omap_dma_start_sg(struct omap_chan *c, struct omap_desc *d,
343 unsigned idx)
344{
345 struct omap_sg *sg = d->sg + idx;
Russell King893e63e2013-11-03 11:17:11 +0000346 unsigned cxsa, cxei, cxfi;
Russell King7bedaa52012-04-13 12:10:24 +0100347
Russell Kingb9e97822013-11-02 13:26:57 +0000348 if (d->dir == DMA_DEV_TO_MEM) {
Russell King893e63e2013-11-03 11:17:11 +0000349 cxsa = CDSA;
350 cxei = CDEI;
351 cxfi = CDFI;
Russell Kingb9e97822013-11-02 13:26:57 +0000352 } else {
Russell King893e63e2013-11-03 11:17:11 +0000353 cxsa = CSSA;
354 cxei = CSEI;
355 cxfi = CSFI;
Russell Kingb9e97822013-11-02 13:26:57 +0000356 }
357
Russell Kingc5ed98b2013-11-06 17:33:09 +0000358 omap_dma_chan_write(c, cxsa, sg->addr);
359 omap_dma_chan_write(c, cxei, 0);
360 omap_dma_chan_write(c, cxfi, 0);
361 omap_dma_chan_write(c, CEN, sg->en);
362 omap_dma_chan_write(c, CFN, sg->fn);
Russell King7bedaa52012-04-13 12:10:24 +0100363
Russell Kingfa3ad862013-11-02 17:07:09 +0000364 omap_dma_start(c, d);
Russell King7bedaa52012-04-13 12:10:24 +0100365}
366
367static void omap_dma_start_desc(struct omap_chan *c)
368{
369 struct virt_dma_desc *vd = vchan_next_desc(&c->vc);
370 struct omap_desc *d;
Russell King893e63e2013-11-03 11:17:11 +0000371 unsigned cxsa, cxei, cxfi;
Russell King7bedaa52012-04-13 12:10:24 +0100372
373 if (!vd) {
374 c->desc = NULL;
375 return;
376 }
377
378 list_del(&vd->node);
379
380 c->desc = d = to_omap_dma_desc(&vd->tx);
381 c->sgidx = 0;
382
Russell King59871902013-11-06 17:15:16 +0000383 /*
384 * This provides the necessary barrier to ensure data held in
385 * DMA coherent memory is visible to the DMA engine prior to
386 * the transfer starting.
387 */
388 mb();
389
Russell Kingc5ed98b2013-11-06 17:33:09 +0000390 omap_dma_chan_write(c, CCR, d->ccr);
Russell King3ed4d182013-11-02 19:16:09 +0000391 if (dma_omap1())
Russell Kingc5ed98b2013-11-06 17:33:09 +0000392 omap_dma_chan_write(c, CCR2, d->ccr >> 16);
Russell Kingb9e97822013-11-02 13:26:57 +0000393
Russell King3ed4d182013-11-02 19:16:09 +0000394 if (d->dir == DMA_DEV_TO_MEM) {
Russell King893e63e2013-11-03 11:17:11 +0000395 cxsa = CSSA;
396 cxei = CSEI;
397 cxfi = CSFI;
Russell Kingb9e97822013-11-02 13:26:57 +0000398 } else {
Russell King893e63e2013-11-03 11:17:11 +0000399 cxsa = CDSA;
400 cxei = CDEI;
401 cxfi = CDFI;
Russell Kingb9e97822013-11-02 13:26:57 +0000402 }
Russell King7bedaa52012-04-13 12:10:24 +0100403
Russell Kingc5ed98b2013-11-06 17:33:09 +0000404 omap_dma_chan_write(c, cxsa, d->dev_addr);
405 omap_dma_chan_write(c, cxei, 0);
406 omap_dma_chan_write(c, cxfi, d->fi);
407 omap_dma_chan_write(c, CSDP, d->csdp);
408 omap_dma_chan_write(c, CLNK_CTRL, d->clnk_ctrl);
Russell King913a2d02013-11-02 14:41:42 +0000409
Russell King7bedaa52012-04-13 12:10:24 +0100410 omap_dma_start_sg(c, d, 0);
411}
412
413static void omap_dma_callback(int ch, u16 status, void *data)
414{
415 struct omap_chan *c = data;
416 struct omap_desc *d;
417 unsigned long flags;
418
419 spin_lock_irqsave(&c->vc.lock, flags);
420 d = c->desc;
421 if (d) {
Russell King3a774ea2012-06-21 10:40:15 +0100422 if (!c->cyclic) {
423 if (++c->sgidx < d->sglen) {
424 omap_dma_start_sg(c, d, c->sgidx);
425 } else {
426 omap_dma_start_desc(c);
427 vchan_cookie_complete(&d->vd);
428 }
Russell King7bedaa52012-04-13 12:10:24 +0100429 } else {
Russell King3a774ea2012-06-21 10:40:15 +0100430 vchan_cyclic_callback(&d->vd);
Russell King7bedaa52012-04-13 12:10:24 +0100431 }
432 }
433 spin_unlock_irqrestore(&c->vc.lock, flags);
434}
435
436/*
437 * This callback schedules all pending channels. We could be more
438 * clever here by postponing allocation of the real DMA channels to
439 * this point, and freeing them when our virtual channel becomes idle.
440 *
441 * We would then need to deal with 'all channels in-use'
442 */
443static void omap_dma_sched(unsigned long data)
444{
445 struct omap_dmadev *d = (struct omap_dmadev *)data;
446 LIST_HEAD(head);
447
448 spin_lock_irq(&d->lock);
449 list_splice_tail_init(&d->pending, &head);
450 spin_unlock_irq(&d->lock);
451
452 while (!list_empty(&head)) {
453 struct omap_chan *c = list_first_entry(&head,
454 struct omap_chan, node);
455
456 spin_lock_irq(&c->vc.lock);
457 list_del_init(&c->node);
458 omap_dma_start_desc(c);
459 spin_unlock_irq(&c->vc.lock);
460 }
461}
462
463static int omap_dma_alloc_chan_resources(struct dma_chan *chan)
464{
Russell King596c4712013-12-10 11:08:01 +0000465 struct omap_dmadev *od = to_omap_dma_dev(chan->device);
Russell King7bedaa52012-04-13 12:10:24 +0100466 struct omap_chan *c = to_omap_dma_chan(chan);
Russell King596c4712013-12-10 11:08:01 +0000467 int ret;
Russell King7bedaa52012-04-13 12:10:24 +0100468
Russell King596c4712013-12-10 11:08:01 +0000469 dev_dbg(od->ddev.dev, "allocating channel for %u\n", c->dma_sig);
Russell King7bedaa52012-04-13 12:10:24 +0100470
Russell King596c4712013-12-10 11:08:01 +0000471 ret = omap_request_dma(c->dma_sig, "DMA engine", omap_dma_callback,
472 c, &c->dma_ch);
473
474 if (ret >= 0)
475 omap_dma_assign(od, c, c->dma_ch);
476
477 return ret;
Russell King7bedaa52012-04-13 12:10:24 +0100478}
479
480static void omap_dma_free_chan_resources(struct dma_chan *chan)
481{
482 struct omap_chan *c = to_omap_dma_chan(chan);
483
Russell King596c4712013-12-10 11:08:01 +0000484 c->channel_base = NULL;
Russell King7bedaa52012-04-13 12:10:24 +0100485 vchan_free_chan_resources(&c->vc);
486 omap_free_dma(c->dma_ch);
487
Ezequiel Garcia9e2f7d82013-12-19 22:22:29 -0300488 dev_dbg(c->vc.chan.device->dev, "freeing channel for %u\n", c->dma_sig);
Russell King7bedaa52012-04-13 12:10:24 +0100489}
490
Russell King3850e222012-06-21 10:37:35 +0100491static size_t omap_dma_sg_size(struct omap_sg *sg)
492{
493 return sg->en * sg->fn;
494}
495
496static size_t omap_dma_desc_size(struct omap_desc *d)
497{
498 unsigned i;
499 size_t size;
500
501 for (size = i = 0; i < d->sglen; i++)
502 size += omap_dma_sg_size(&d->sg[i]);
503
504 return size * es_bytes[d->es];
505}
506
507static size_t omap_dma_desc_size_pos(struct omap_desc *d, dma_addr_t addr)
508{
509 unsigned i;
510 size_t size, es_size = es_bytes[d->es];
511
512 for (size = i = 0; i < d->sglen; i++) {
513 size_t this_size = omap_dma_sg_size(&d->sg[i]) * es_size;
514
515 if (size)
516 size += this_size;
517 else if (addr >= d->sg[i].addr &&
518 addr < d->sg[i].addr + this_size)
519 size += d->sg[i].addr + this_size - addr;
520 }
521 return size;
522}
523
Russell Kingb07fd622013-11-06 19:26:45 +0000524/*
525 * OMAP 3.2/3.3 erratum: sometimes 0 is returned if CSAC/CDAC is
526 * read before the DMA controller finished disabling the channel.
527 */
528static uint32_t omap_dma_chan_read_3_3(struct omap_chan *c, unsigned reg)
529{
530 struct omap_dmadev *od = to_omap_dma_dev(c->vc.chan.device);
531 uint32_t val;
532
533 val = omap_dma_chan_read(c, reg);
534 if (val == 0 && od->plat->errata & DMA_ERRATA_3_3)
535 val = omap_dma_chan_read(c, reg);
536
537 return val;
538}
539
Russell King3997cab2013-11-02 18:04:17 +0000540static dma_addr_t omap_dma_get_src_pos(struct omap_chan *c)
541{
542 struct omap_dmadev *od = to_omap_dma_dev(c->vc.chan.device);
Russell Kingb07fd622013-11-06 19:26:45 +0000543 dma_addr_t addr, cdac;
Russell King3997cab2013-11-02 18:04:17 +0000544
Russell Kingb07fd622013-11-06 19:26:45 +0000545 if (__dma_omap15xx(od->plat->dma_attr)) {
Russell Kingc5ed98b2013-11-06 17:33:09 +0000546 addr = omap_dma_chan_read(c, CPC);
Russell Kingb07fd622013-11-06 19:26:45 +0000547 } else {
548 addr = omap_dma_chan_read_3_3(c, CSAC);
549 cdac = omap_dma_chan_read_3_3(c, CDAC);
Russell King3997cab2013-11-02 18:04:17 +0000550
Russell King3997cab2013-11-02 18:04:17 +0000551 /*
552 * CDAC == 0 indicates that the DMA transfer on the channel has
553 * not been started (no data has been transferred so far).
554 * Return the programmed source start address in this case.
555 */
Russell Kingb07fd622013-11-06 19:26:45 +0000556 if (cdac == 0)
Russell Kingc5ed98b2013-11-06 17:33:09 +0000557 addr = omap_dma_chan_read(c, CSSA);
Russell King3997cab2013-11-02 18:04:17 +0000558 }
559
560 if (dma_omap1())
Russell Kingc5ed98b2013-11-06 17:33:09 +0000561 addr |= omap_dma_chan_read(c, CSSA) & 0xffff0000;
Russell King3997cab2013-11-02 18:04:17 +0000562
563 return addr;
564}
565
566static dma_addr_t omap_dma_get_dst_pos(struct omap_chan *c)
567{
568 struct omap_dmadev *od = to_omap_dma_dev(c->vc.chan.device);
569 dma_addr_t addr;
570
Russell Kingb07fd622013-11-06 19:26:45 +0000571 if (__dma_omap15xx(od->plat->dma_attr)) {
Russell Kingc5ed98b2013-11-06 17:33:09 +0000572 addr = omap_dma_chan_read(c, CPC);
Russell Kingb07fd622013-11-06 19:26:45 +0000573 } else {
574 addr = omap_dma_chan_read_3_3(c, CDAC);
Russell King3997cab2013-11-02 18:04:17 +0000575
Russell King3997cab2013-11-02 18:04:17 +0000576 /*
Russell Kingb07fd622013-11-06 19:26:45 +0000577 * CDAC == 0 indicates that the DMA transfer on the channel
578 * has not been started (no data has been transferred so
579 * far). Return the programmed destination start address in
580 * this case.
Russell King3997cab2013-11-02 18:04:17 +0000581 */
582 if (addr == 0)
Russell Kingc5ed98b2013-11-06 17:33:09 +0000583 addr = omap_dma_chan_read(c, CDSA);
Russell King3997cab2013-11-02 18:04:17 +0000584 }
585
586 if (dma_omap1())
Russell Kingc5ed98b2013-11-06 17:33:09 +0000587 addr |= omap_dma_chan_read(c, CDSA) & 0xffff0000;
Russell King3997cab2013-11-02 18:04:17 +0000588
589 return addr;
590}
591
Russell King7bedaa52012-04-13 12:10:24 +0100592static enum dma_status omap_dma_tx_status(struct dma_chan *chan,
593 dma_cookie_t cookie, struct dma_tx_state *txstate)
594{
Russell King3850e222012-06-21 10:37:35 +0100595 struct omap_chan *c = to_omap_dma_chan(chan);
596 struct virt_dma_desc *vd;
597 enum dma_status ret;
598 unsigned long flags;
599
600 ret = dma_cookie_status(chan, cookie, txstate);
Vinod Koul7cce5082013-10-16 20:51:54 +0530601 if (ret == DMA_COMPLETE || !txstate)
Russell King3850e222012-06-21 10:37:35 +0100602 return ret;
603
604 spin_lock_irqsave(&c->vc.lock, flags);
605 vd = vchan_find_desc(&c->vc, cookie);
606 if (vd) {
607 txstate->residue = omap_dma_desc_size(to_omap_dma_desc(&vd->tx));
608 } else if (c->desc && c->desc->vd.tx.cookie == cookie) {
609 struct omap_desc *d = c->desc;
610 dma_addr_t pos;
611
612 if (d->dir == DMA_MEM_TO_DEV)
Russell King3997cab2013-11-02 18:04:17 +0000613 pos = omap_dma_get_src_pos(c);
Russell King3850e222012-06-21 10:37:35 +0100614 else if (d->dir == DMA_DEV_TO_MEM)
Russell King3997cab2013-11-02 18:04:17 +0000615 pos = omap_dma_get_dst_pos(c);
Russell King3850e222012-06-21 10:37:35 +0100616 else
617 pos = 0;
618
619 txstate->residue = omap_dma_desc_size_pos(d, pos);
620 } else {
621 txstate->residue = 0;
622 }
623 spin_unlock_irqrestore(&c->vc.lock, flags);
624
625 return ret;
Russell King7bedaa52012-04-13 12:10:24 +0100626}
627
628static void omap_dma_issue_pending(struct dma_chan *chan)
629{
630 struct omap_chan *c = to_omap_dma_chan(chan);
631 unsigned long flags;
632
633 spin_lock_irqsave(&c->vc.lock, flags);
634 if (vchan_issue_pending(&c->vc) && !c->desc) {
Peter Ujfalusi76502462013-04-09 16:33:06 +0200635 /*
636 * c->cyclic is used only by audio and in this case the DMA need
637 * to be started without delay.
638 */
639 if (!c->cyclic) {
640 struct omap_dmadev *d = to_omap_dma_dev(chan->device);
641 spin_lock(&d->lock);
642 if (list_empty(&c->node))
643 list_add_tail(&c->node, &d->pending);
644 spin_unlock(&d->lock);
645 tasklet_schedule(&d->task);
646 } else {
647 omap_dma_start_desc(c);
648 }
Russell King7bedaa52012-04-13 12:10:24 +0100649 }
650 spin_unlock_irqrestore(&c->vc.lock, flags);
651}
652
653static struct dma_async_tx_descriptor *omap_dma_prep_slave_sg(
654 struct dma_chan *chan, struct scatterlist *sgl, unsigned sglen,
655 enum dma_transfer_direction dir, unsigned long tx_flags, void *context)
656{
Russell King49ae0b22013-11-02 21:09:18 +0000657 struct omap_dmadev *od = to_omap_dma_dev(chan->device);
Russell King7bedaa52012-04-13 12:10:24 +0100658 struct omap_chan *c = to_omap_dma_chan(chan);
659 enum dma_slave_buswidth dev_width;
660 struct scatterlist *sgent;
661 struct omap_desc *d;
662 dma_addr_t dev_addr;
Russell King3ed4d182013-11-02 19:16:09 +0000663 unsigned i, j = 0, es, en, frame_bytes;
Russell King7bedaa52012-04-13 12:10:24 +0100664 u32 burst;
665
666 if (dir == DMA_DEV_TO_MEM) {
667 dev_addr = c->cfg.src_addr;
668 dev_width = c->cfg.src_addr_width;
669 burst = c->cfg.src_maxburst;
Russell King7bedaa52012-04-13 12:10:24 +0100670 } else if (dir == DMA_MEM_TO_DEV) {
671 dev_addr = c->cfg.dst_addr;
672 dev_width = c->cfg.dst_addr_width;
673 burst = c->cfg.dst_maxburst;
Russell King7bedaa52012-04-13 12:10:24 +0100674 } else {
675 dev_err(chan->device->dev, "%s: bad direction?\n", __func__);
676 return NULL;
677 }
678
679 /* Bus width translates to the element size (ES) */
680 switch (dev_width) {
681 case DMA_SLAVE_BUSWIDTH_1_BYTE:
Russell King90438262013-11-02 19:57:06 +0000682 es = CSDP_DATA_TYPE_8;
Russell King7bedaa52012-04-13 12:10:24 +0100683 break;
684 case DMA_SLAVE_BUSWIDTH_2_BYTES:
Russell King90438262013-11-02 19:57:06 +0000685 es = CSDP_DATA_TYPE_16;
Russell King7bedaa52012-04-13 12:10:24 +0100686 break;
687 case DMA_SLAVE_BUSWIDTH_4_BYTES:
Russell King90438262013-11-02 19:57:06 +0000688 es = CSDP_DATA_TYPE_32;
Russell King7bedaa52012-04-13 12:10:24 +0100689 break;
690 default: /* not reached */
691 return NULL;
692 }
693
694 /* Now allocate and setup the descriptor. */
695 d = kzalloc(sizeof(*d) + sglen * sizeof(d->sg[0]), GFP_ATOMIC);
696 if (!d)
697 return NULL;
698
699 d->dir = dir;
700 d->dev_addr = dev_addr;
701 d->es = es;
Russell King3ed4d182013-11-02 19:16:09 +0000702
Russell King90438262013-11-02 19:57:06 +0000703 d->ccr = CCR_SYNC_FRAME;
Russell King3ed4d182013-11-02 19:16:09 +0000704 if (dir == DMA_DEV_TO_MEM)
Russell King90438262013-11-02 19:57:06 +0000705 d->ccr |= CCR_DST_AMODE_POSTINC | CCR_SRC_AMODE_CONSTANT;
Russell King3ed4d182013-11-02 19:16:09 +0000706 else
Russell King90438262013-11-02 19:57:06 +0000707 d->ccr |= CCR_DST_AMODE_CONSTANT | CCR_SRC_AMODE_POSTINC;
Russell King3ed4d182013-11-02 19:16:09 +0000708
Russell King90438262013-11-02 19:57:06 +0000709 d->cicr = CICR_DROP_IE | CICR_BLOCK_IE;
Russell King2f0d13b2013-11-02 18:51:53 +0000710 d->csdp = es;
Russell Kingfa3ad862013-11-02 17:07:09 +0000711
Russell King2f0d13b2013-11-02 18:51:53 +0000712 if (dma_omap1()) {
Russell King3ed4d182013-11-02 19:16:09 +0000713 if (__dma_omap16xx(od->plat->dma_attr)) {
Russell King90438262013-11-02 19:57:06 +0000714 d->ccr |= CCR_OMAP31_DISABLE;
Russell King3ed4d182013-11-02 19:16:09 +0000715 /* Duplicate what plat-omap/dma.c does */
716 d->ccr |= c->dma_ch + 1;
717 } else {
718 d->ccr |= c->dma_sig & 0x1f;
719 }
720
Russell King90438262013-11-02 19:57:06 +0000721 d->cicr |= CICR_TOUT_IE;
Russell King2f0d13b2013-11-02 18:51:53 +0000722
723 if (dir == DMA_DEV_TO_MEM)
Russell King90438262013-11-02 19:57:06 +0000724 d->csdp |= CSDP_DST_PORT_EMIFF | CSDP_SRC_PORT_TIPB;
Russell King2f0d13b2013-11-02 18:51:53 +0000725 else
Russell King90438262013-11-02 19:57:06 +0000726 d->csdp |= CSDP_DST_PORT_TIPB | CSDP_SRC_PORT_EMIFF;
Russell King2f0d13b2013-11-02 18:51:53 +0000727 } else {
Russell King3ed4d182013-11-02 19:16:09 +0000728 d->ccr |= (c->dma_sig & ~0x1f) << 14;
729 d->ccr |= c->dma_sig & 0x1f;
Russell King3ed4d182013-11-02 19:16:09 +0000730
731 if (dir == DMA_DEV_TO_MEM)
Russell King90438262013-11-02 19:57:06 +0000732 d->ccr |= CCR_TRIGGER_SRC;
Russell King3ed4d182013-11-02 19:16:09 +0000733
Russell King90438262013-11-02 19:57:06 +0000734 d->cicr |= CICR_MISALIGNED_ERR_IE | CICR_TRANS_ERR_IE;
Russell King2f0d13b2013-11-02 18:51:53 +0000735 }
Russell King49ae0b22013-11-02 21:09:18 +0000736 if (od->plat->errata & DMA_ERRATA_IFRAME_BUFFERING)
737 d->ccr |= CCR_BUFFERING_DISABLE;
Russell King965aeb4d2013-11-06 17:12:30 +0000738 if (od->plat->errata & DMA_ERRATA_PARALLEL_CHANNELS)
739 d->clnk_ctrl = c->dma_ch;
Russell King7bedaa52012-04-13 12:10:24 +0100740
741 /*
742 * Build our scatterlist entries: each contains the address,
743 * the number of elements (EN) in each frame, and the number of
744 * frames (FN). Number of bytes for this entry = ES * EN * FN.
745 *
746 * Burst size translates to number of elements with frame sync.
747 * Note: DMA engine defines burst to be the number of dev-width
748 * transfers.
749 */
750 en = burst;
751 frame_bytes = es_bytes[es] * en;
752 for_each_sg(sgl, sgent, sglen, i) {
753 d->sg[j].addr = sg_dma_address(sgent);
754 d->sg[j].en = en;
755 d->sg[j].fn = sg_dma_len(sgent) / frame_bytes;
756 j++;
757 }
758
759 d->sglen = j;
760
761 return vchan_tx_prep(&c->vc, &d->vd, tx_flags);
762}
763
Russell King3a774ea2012-06-21 10:40:15 +0100764static struct dma_async_tx_descriptor *omap_dma_prep_dma_cyclic(
765 struct dma_chan *chan, dma_addr_t buf_addr, size_t buf_len,
Peter Ujfalusiec8b5e42012-09-14 15:05:47 +0300766 size_t period_len, enum dma_transfer_direction dir, unsigned long flags,
767 void *context)
Russell King3a774ea2012-06-21 10:40:15 +0100768{
Russell Kingfa3ad862013-11-02 17:07:09 +0000769 struct omap_dmadev *od = to_omap_dma_dev(chan->device);
Russell King3a774ea2012-06-21 10:40:15 +0100770 struct omap_chan *c = to_omap_dma_chan(chan);
771 enum dma_slave_buswidth dev_width;
772 struct omap_desc *d;
773 dma_addr_t dev_addr;
Russell King3ed4d182013-11-02 19:16:09 +0000774 unsigned es;
Russell King3a774ea2012-06-21 10:40:15 +0100775 u32 burst;
776
777 if (dir == DMA_DEV_TO_MEM) {
778 dev_addr = c->cfg.src_addr;
779 dev_width = c->cfg.src_addr_width;
780 burst = c->cfg.src_maxburst;
Russell King3a774ea2012-06-21 10:40:15 +0100781 } else if (dir == DMA_MEM_TO_DEV) {
782 dev_addr = c->cfg.dst_addr;
783 dev_width = c->cfg.dst_addr_width;
784 burst = c->cfg.dst_maxburst;
Russell King3a774ea2012-06-21 10:40:15 +0100785 } else {
786 dev_err(chan->device->dev, "%s: bad direction?\n", __func__);
787 return NULL;
788 }
789
790 /* Bus width translates to the element size (ES) */
791 switch (dev_width) {
792 case DMA_SLAVE_BUSWIDTH_1_BYTE:
Russell King90438262013-11-02 19:57:06 +0000793 es = CSDP_DATA_TYPE_8;
Russell King3a774ea2012-06-21 10:40:15 +0100794 break;
795 case DMA_SLAVE_BUSWIDTH_2_BYTES:
Russell King90438262013-11-02 19:57:06 +0000796 es = CSDP_DATA_TYPE_16;
Russell King3a774ea2012-06-21 10:40:15 +0100797 break;
798 case DMA_SLAVE_BUSWIDTH_4_BYTES:
Russell King90438262013-11-02 19:57:06 +0000799 es = CSDP_DATA_TYPE_32;
Russell King3a774ea2012-06-21 10:40:15 +0100800 break;
801 default: /* not reached */
802 return NULL;
803 }
804
805 /* Now allocate and setup the descriptor. */
806 d = kzalloc(sizeof(*d) + sizeof(d->sg[0]), GFP_ATOMIC);
807 if (!d)
808 return NULL;
809
810 d->dir = dir;
811 d->dev_addr = dev_addr;
812 d->fi = burst;
813 d->es = es;
Russell King3a774ea2012-06-21 10:40:15 +0100814 d->sg[0].addr = buf_addr;
815 d->sg[0].en = period_len / es_bytes[es];
816 d->sg[0].fn = buf_len / period_len;
817 d->sglen = 1;
Russell King3ed4d182013-11-02 19:16:09 +0000818
819 d->ccr = 0;
Russell King3ed4d182013-11-02 19:16:09 +0000820 if (dir == DMA_DEV_TO_MEM)
Russell King90438262013-11-02 19:57:06 +0000821 d->ccr |= CCR_DST_AMODE_POSTINC | CCR_SRC_AMODE_CONSTANT;
Russell King3ed4d182013-11-02 19:16:09 +0000822 else
Russell King90438262013-11-02 19:57:06 +0000823 d->ccr |= CCR_DST_AMODE_CONSTANT | CCR_SRC_AMODE_POSTINC;
Russell King3ed4d182013-11-02 19:16:09 +0000824
Russell King90438262013-11-02 19:57:06 +0000825 d->cicr = CICR_DROP_IE;
Russell Kingfa3ad862013-11-02 17:07:09 +0000826 if (flags & DMA_PREP_INTERRUPT)
Russell King90438262013-11-02 19:57:06 +0000827 d->cicr |= CICR_FRAME_IE;
Russell Kingfa3ad862013-11-02 17:07:09 +0000828
Russell King2f0d13b2013-11-02 18:51:53 +0000829 d->csdp = es;
830
831 if (dma_omap1()) {
Russell King3ed4d182013-11-02 19:16:09 +0000832 if (__dma_omap16xx(od->plat->dma_attr)) {
Russell King90438262013-11-02 19:57:06 +0000833 d->ccr |= CCR_OMAP31_DISABLE;
Russell King3ed4d182013-11-02 19:16:09 +0000834 /* Duplicate what plat-omap/dma.c does */
835 d->ccr |= c->dma_ch + 1;
836 } else {
837 d->ccr |= c->dma_sig & 0x1f;
838 }
839
Russell King90438262013-11-02 19:57:06 +0000840 d->cicr |= CICR_TOUT_IE;
Russell King2f0d13b2013-11-02 18:51:53 +0000841
842 if (dir == DMA_DEV_TO_MEM)
Russell King90438262013-11-02 19:57:06 +0000843 d->csdp |= CSDP_DST_PORT_EMIFF | CSDP_SRC_PORT_MPUI;
Russell King2f0d13b2013-11-02 18:51:53 +0000844 else
Russell King90438262013-11-02 19:57:06 +0000845 d->csdp |= CSDP_DST_PORT_MPUI | CSDP_SRC_PORT_EMIFF;
Russell King2f0d13b2013-11-02 18:51:53 +0000846 } else {
Russell King3ed4d182013-11-02 19:16:09 +0000847 d->ccr |= (c->dma_sig & ~0x1f) << 14;
848 d->ccr |= c->dma_sig & 0x1f;
849
850 if (burst)
Russell King90438262013-11-02 19:57:06 +0000851 d->ccr |= CCR_SYNC_PACKET;
852 else
853 d->ccr |= CCR_SYNC_ELEMENT;
Russell King3ed4d182013-11-02 19:16:09 +0000854
855 if (dir == DMA_DEV_TO_MEM)
Russell King90438262013-11-02 19:57:06 +0000856 d->ccr |= CCR_TRIGGER_SRC;
Russell King3ed4d182013-11-02 19:16:09 +0000857
Russell King90438262013-11-02 19:57:06 +0000858 d->cicr |= CICR_MISALIGNED_ERR_IE | CICR_TRANS_ERR_IE;
Russell King3a774ea2012-06-21 10:40:15 +0100859
Russell King90438262013-11-02 19:57:06 +0000860 d->csdp |= CSDP_DST_BURST_64 | CSDP_SRC_BURST_64;
Russell King2f0d13b2013-11-02 18:51:53 +0000861 }
Russell King49ae0b22013-11-02 21:09:18 +0000862 if (od->plat->errata & DMA_ERRATA_IFRAME_BUFFERING)
863 d->ccr |= CCR_BUFFERING_DISABLE;
Russell King2f0d13b2013-11-02 18:51:53 +0000864
Russell King965aeb4d2013-11-06 17:12:30 +0000865 if (__dma_omap15xx(od->plat->dma_attr))
866 d->ccr |= CCR_AUTO_INIT | CCR_REPEAT;
867 else
868 d->clnk_ctrl = c->dma_ch | CLNK_CTRL_ENABLE_LNK;
869
Russell King3ed4d182013-11-02 19:16:09 +0000870 c->cyclic = true;
Russell King3a774ea2012-06-21 10:40:15 +0100871
Peter Ujfalusi2dde5b92012-09-14 15:05:48 +0300872 return vchan_tx_prep(&c->vc, &d->vd, flags);
Russell King3a774ea2012-06-21 10:40:15 +0100873}
874
Russell King7bedaa52012-04-13 12:10:24 +0100875static int omap_dma_slave_config(struct omap_chan *c, struct dma_slave_config *cfg)
876{
877 if (cfg->src_addr_width == DMA_SLAVE_BUSWIDTH_8_BYTES ||
878 cfg->dst_addr_width == DMA_SLAVE_BUSWIDTH_8_BYTES)
879 return -EINVAL;
880
881 memcpy(&c->cfg, cfg, sizeof(c->cfg));
882
883 return 0;
884}
885
886static int omap_dma_terminate_all(struct omap_chan *c)
887{
888 struct omap_dmadev *d = to_omap_dma_dev(c->vc.chan.device);
889 unsigned long flags;
890 LIST_HEAD(head);
891
892 spin_lock_irqsave(&c->vc.lock, flags);
893
894 /* Prevent this channel being scheduled */
895 spin_lock(&d->lock);
896 list_del_init(&c->node);
897 spin_unlock(&d->lock);
898
899 /*
900 * Stop DMA activity: we assume the callback will not be called
Russell Kingfa3ad862013-11-02 17:07:09 +0000901 * after omap_dma_stop() returns (even if it does, it will see
Russell King7bedaa52012-04-13 12:10:24 +0100902 * c->desc is NULL and exit.)
903 */
904 if (c->desc) {
905 c->desc = NULL;
Peter Ujfalusi2dcdf572012-09-14 15:05:45 +0300906 /* Avoid stopping the dma twice */
907 if (!c->paused)
Russell Kingfa3ad862013-11-02 17:07:09 +0000908 omap_dma_stop(c);
Russell King7bedaa52012-04-13 12:10:24 +0100909 }
910
Russell King3a774ea2012-06-21 10:40:15 +0100911 if (c->cyclic) {
912 c->cyclic = false;
Peter Ujfalusi2dcdf572012-09-14 15:05:45 +0300913 c->paused = false;
Russell King3a774ea2012-06-21 10:40:15 +0100914 }
915
Russell King7bedaa52012-04-13 12:10:24 +0100916 vchan_get_all_descriptors(&c->vc, &head);
917 spin_unlock_irqrestore(&c->vc.lock, flags);
918 vchan_dma_desc_free_list(&c->vc, &head);
919
920 return 0;
921}
922
923static int omap_dma_pause(struct omap_chan *c)
924{
Peter Ujfalusi2dcdf572012-09-14 15:05:45 +0300925 /* Pause/Resume only allowed with cyclic mode */
926 if (!c->cyclic)
927 return -EINVAL;
928
929 if (!c->paused) {
Russell Kingfa3ad862013-11-02 17:07:09 +0000930 omap_dma_stop(c);
Peter Ujfalusi2dcdf572012-09-14 15:05:45 +0300931 c->paused = true;
932 }
933
934 return 0;
Russell King7bedaa52012-04-13 12:10:24 +0100935}
936
937static int omap_dma_resume(struct omap_chan *c)
938{
Peter Ujfalusi2dcdf572012-09-14 15:05:45 +0300939 /* Pause/Resume only allowed with cyclic mode */
940 if (!c->cyclic)
941 return -EINVAL;
942
943 if (c->paused) {
Russell Kingfa3ad862013-11-02 17:07:09 +0000944 omap_dma_start(c, c->desc);
Peter Ujfalusi2dcdf572012-09-14 15:05:45 +0300945 c->paused = false;
946 }
947
948 return 0;
Russell King7bedaa52012-04-13 12:10:24 +0100949}
950
951static int omap_dma_control(struct dma_chan *chan, enum dma_ctrl_cmd cmd,
952 unsigned long arg)
953{
954 struct omap_chan *c = to_omap_dma_chan(chan);
955 int ret;
956
957 switch (cmd) {
958 case DMA_SLAVE_CONFIG:
959 ret = omap_dma_slave_config(c, (struct dma_slave_config *)arg);
960 break;
961
962 case DMA_TERMINATE_ALL:
963 ret = omap_dma_terminate_all(c);
964 break;
965
966 case DMA_PAUSE:
967 ret = omap_dma_pause(c);
968 break;
969
970 case DMA_RESUME:
971 ret = omap_dma_resume(c);
972 break;
973
974 default:
975 ret = -ENXIO;
976 break;
977 }
978
979 return ret;
980}
981
982static int omap_dma_chan_init(struct omap_dmadev *od, int dma_sig)
983{
984 struct omap_chan *c;
985
986 c = kzalloc(sizeof(*c), GFP_KERNEL);
987 if (!c)
988 return -ENOMEM;
989
Russell King596c4712013-12-10 11:08:01 +0000990 c->reg_map = od->reg_map;
Russell King7bedaa52012-04-13 12:10:24 +0100991 c->dma_sig = dma_sig;
992 c->vc.desc_free = omap_dma_desc_free;
993 vchan_init(&c->vc, &od->ddev);
994 INIT_LIST_HEAD(&c->node);
995
996 od->ddev.chancnt++;
997
998 return 0;
999}
1000
1001static void omap_dma_free(struct omap_dmadev *od)
1002{
1003 tasklet_kill(&od->task);
1004 while (!list_empty(&od->ddev.channels)) {
1005 struct omap_chan *c = list_first_entry(&od->ddev.channels,
1006 struct omap_chan, vc.chan.device_node);
1007
1008 list_del(&c->vc.chan.device_node);
1009 tasklet_kill(&c->vc.task);
1010 kfree(c);
1011 }
Russell King7bedaa52012-04-13 12:10:24 +01001012}
1013
1014static int omap_dma_probe(struct platform_device *pdev)
1015{
1016 struct omap_dmadev *od;
Russell King596c4712013-12-10 11:08:01 +00001017 struct resource *res;
Russell King7bedaa52012-04-13 12:10:24 +01001018 int rc, i;
1019
Russell King104fce72013-11-02 12:58:29 +00001020 od = devm_kzalloc(&pdev->dev, sizeof(*od), GFP_KERNEL);
Russell King7bedaa52012-04-13 12:10:24 +01001021 if (!od)
1022 return -ENOMEM;
1023
Russell King596c4712013-12-10 11:08:01 +00001024 res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
1025 od->base = devm_ioremap_resource(&pdev->dev, res);
1026 if (IS_ERR(od->base))
1027 return PTR_ERR(od->base);
1028
Russell King1b416c42013-11-02 13:00:03 +00001029 od->plat = omap_get_plat_info();
1030 if (!od->plat)
1031 return -EPROBE_DEFER;
1032
Russell King596c4712013-12-10 11:08:01 +00001033 od->reg_map = od->plat->reg_map;
1034
Russell King7bedaa52012-04-13 12:10:24 +01001035 dma_cap_set(DMA_SLAVE, od->ddev.cap_mask);
Russell King3a774ea2012-06-21 10:40:15 +01001036 dma_cap_set(DMA_CYCLIC, od->ddev.cap_mask);
Russell King7bedaa52012-04-13 12:10:24 +01001037 od->ddev.device_alloc_chan_resources = omap_dma_alloc_chan_resources;
1038 od->ddev.device_free_chan_resources = omap_dma_free_chan_resources;
1039 od->ddev.device_tx_status = omap_dma_tx_status;
1040 od->ddev.device_issue_pending = omap_dma_issue_pending;
1041 od->ddev.device_prep_slave_sg = omap_dma_prep_slave_sg;
Russell King3a774ea2012-06-21 10:40:15 +01001042 od->ddev.device_prep_dma_cyclic = omap_dma_prep_dma_cyclic;
Russell King7bedaa52012-04-13 12:10:24 +01001043 od->ddev.device_control = omap_dma_control;
1044 od->ddev.dev = &pdev->dev;
1045 INIT_LIST_HEAD(&od->ddev.channels);
1046 INIT_LIST_HEAD(&od->pending);
1047 spin_lock_init(&od->lock);
1048
1049 tasklet_init(&od->task, omap_dma_sched, (unsigned long)od);
1050
1051 for (i = 0; i < 127; i++) {
1052 rc = omap_dma_chan_init(od, i);
1053 if (rc) {
1054 omap_dma_free(od);
1055 return rc;
1056 }
1057 }
1058
1059 rc = dma_async_device_register(&od->ddev);
1060 if (rc) {
1061 pr_warn("OMAP-DMA: failed to register slave DMA engine device: %d\n",
1062 rc);
1063 omap_dma_free(od);
Jon Hunter8d306622013-02-26 12:27:24 -06001064 return rc;
1065 }
1066
1067 platform_set_drvdata(pdev, od);
1068
1069 if (pdev->dev.of_node) {
1070 omap_dma_info.dma_cap = od->ddev.cap_mask;
1071
1072 /* Device-tree DMA controller registration */
1073 rc = of_dma_controller_register(pdev->dev.of_node,
1074 of_dma_simple_xlate, &omap_dma_info);
1075 if (rc) {
1076 pr_warn("OMAP-DMA: failed to register DMA controller\n");
1077 dma_async_device_unregister(&od->ddev);
1078 omap_dma_free(od);
1079 }
Russell King7bedaa52012-04-13 12:10:24 +01001080 }
1081
1082 dev_info(&pdev->dev, "OMAP DMA engine driver\n");
1083
1084 return rc;
1085}
1086
1087static int omap_dma_remove(struct platform_device *pdev)
1088{
1089 struct omap_dmadev *od = platform_get_drvdata(pdev);
1090
Jon Hunter8d306622013-02-26 12:27:24 -06001091 if (pdev->dev.of_node)
1092 of_dma_controller_free(pdev->dev.of_node);
1093
Russell King7bedaa52012-04-13 12:10:24 +01001094 dma_async_device_unregister(&od->ddev);
1095 omap_dma_free(od);
1096
1097 return 0;
1098}
1099
Jon Hunter8d306622013-02-26 12:27:24 -06001100static const struct of_device_id omap_dma_match[] = {
1101 { .compatible = "ti,omap2420-sdma", },
1102 { .compatible = "ti,omap2430-sdma", },
1103 { .compatible = "ti,omap3430-sdma", },
1104 { .compatible = "ti,omap3630-sdma", },
1105 { .compatible = "ti,omap4430-sdma", },
1106 {},
1107};
1108MODULE_DEVICE_TABLE(of, omap_dma_match);
1109
Russell King7bedaa52012-04-13 12:10:24 +01001110static struct platform_driver omap_dma_driver = {
1111 .probe = omap_dma_probe,
1112 .remove = omap_dma_remove,
1113 .driver = {
1114 .name = "omap-dma-engine",
1115 .owner = THIS_MODULE,
Jon Hunter8d306622013-02-26 12:27:24 -06001116 .of_match_table = of_match_ptr(omap_dma_match),
Russell King7bedaa52012-04-13 12:10:24 +01001117 },
1118};
1119
1120bool omap_dma_filter_fn(struct dma_chan *chan, void *param)
1121{
1122 if (chan->device->dev->driver == &omap_dma_driver.driver) {
1123 struct omap_chan *c = to_omap_dma_chan(chan);
1124 unsigned req = *(unsigned *)param;
1125
1126 return req == c->dma_sig;
1127 }
1128 return false;
1129}
1130EXPORT_SYMBOL_GPL(omap_dma_filter_fn);
1131
Russell King7bedaa52012-04-13 12:10:24 +01001132static int omap_dma_init(void)
1133{
Tony Lindgrenbe1f9482013-01-11 11:24:19 -08001134 return platform_driver_register(&omap_dma_driver);
Russell King7bedaa52012-04-13 12:10:24 +01001135}
1136subsys_initcall(omap_dma_init);
1137
1138static void __exit omap_dma_exit(void)
1139{
Russell King7bedaa52012-04-13 12:10:24 +01001140 platform_driver_unregister(&omap_dma_driver);
1141}
1142module_exit(omap_dma_exit);
1143
1144MODULE_AUTHOR("Russell King");
1145MODULE_LICENSE("GPL");