diff options
author | Linus Walleij <linus.walleij@stericsson.com> | 2010-03-30 09:33:42 -0400 |
---|---|---|
committer | Dan Williams <dan.j.williams@intel.com> | 2010-04-14 17:49:20 -0400 |
commit | 8d318a50b3d72e3daf94131f91e1ab799a8d5ad4 (patch) | |
tree | ae36452931d2e836f725b3f91eebd7f4d9e27589 /arch/arm | |
parent | 6a3cd3ea48584d14f60dce0b3c4e9e4428beb0fe (diff) |
DMAENGINE: Support for ST-Ericssons DMA40 block v3
This is a straightforward driver for the ST-Ericsson DMA40 DMA
controller found in U8500, implemented akin to the existing
COH 901 318 driver.
Signed-off-by: Linus Walleij <linus.walleij@stericsson.com>
Acked-by: Srinidh Kasagar <srinidhi.kasagar@stericsson.com>
Cc: STEricsson_nomadik_linux@list.st.com
Cc: Alessandro Rubini <rubini@unipv.it>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Dan Williams <dan.j.williams@intel.com>
Diffstat (limited to 'arch/arm')
-rw-r--r-- | arch/arm/plat-nomadik/include/plat/ste_dma40.h | 239 |
1 files changed, 239 insertions, 0 deletions
diff --git a/arch/arm/plat-nomadik/include/plat/ste_dma40.h b/arch/arm/plat-nomadik/include/plat/ste_dma40.h new file mode 100644 index 000000000000..4d12ea4ca361 --- /dev/null +++ b/arch/arm/plat-nomadik/include/plat/ste_dma40.h | |||
@@ -0,0 +1,239 @@ | |||
1 | /* | ||
2 | * arch/arm/plat-nomadik/include/plat/ste_dma40.h | ||
3 | * | ||
4 | * Copyright (C) ST-Ericsson 2007-2010 | ||
5 | * License terms: GNU General Public License (GPL) version 2 | ||
6 | * Author: Per Friden <per.friden@stericsson.com> | ||
7 | * Author: Jonas Aaberg <jonas.aberg@stericsson.com> | ||
8 | */ | ||
9 | |||
10 | |||
11 | #ifndef STE_DMA40_H | ||
12 | #define STE_DMA40_H | ||
13 | |||
14 | #include <linux/dmaengine.h> | ||
15 | #include <linux/workqueue.h> | ||
16 | #include <linux/interrupt.h> | ||
17 | #include <linux/dmaengine.h> | ||
18 | |||
19 | /* dev types for memcpy */ | ||
20 | #define STEDMA40_DEV_DST_MEMORY (-1) | ||
21 | #define STEDMA40_DEV_SRC_MEMORY (-1) | ||
22 | |||
23 | /* | ||
24 | * Description of bitfields of channel_type variable is available in | ||
25 | * the info structure. | ||
26 | */ | ||
27 | |||
28 | /* Priority */ | ||
29 | #define STEDMA40_INFO_PRIO_TYPE_POS 2 | ||
30 | #define STEDMA40_HIGH_PRIORITY_CHANNEL (0x1 << STEDMA40_INFO_PRIO_TYPE_POS) | ||
31 | #define STEDMA40_LOW_PRIORITY_CHANNEL (0x2 << STEDMA40_INFO_PRIO_TYPE_POS) | ||
32 | |||
33 | /* Mode */ | ||
34 | #define STEDMA40_INFO_CH_MODE_TYPE_POS 6 | ||
35 | #define STEDMA40_CHANNEL_IN_PHY_MODE (0x1 << STEDMA40_INFO_CH_MODE_TYPE_POS) | ||
36 | #define STEDMA40_CHANNEL_IN_LOG_MODE (0x2 << STEDMA40_INFO_CH_MODE_TYPE_POS) | ||
37 | #define STEDMA40_CHANNEL_IN_OPER_MODE (0x3 << STEDMA40_INFO_CH_MODE_TYPE_POS) | ||
38 | |||
39 | /* Mode options */ | ||
40 | #define STEDMA40_INFO_CH_MODE_OPT_POS 8 | ||
41 | #define STEDMA40_PCHAN_BASIC_MODE (0x1 << STEDMA40_INFO_CH_MODE_OPT_POS) | ||
42 | #define STEDMA40_PCHAN_MODULO_MODE (0x2 << STEDMA40_INFO_CH_MODE_OPT_POS) | ||
43 | #define STEDMA40_PCHAN_DOUBLE_DST_MODE (0x3 << STEDMA40_INFO_CH_MODE_OPT_POS) | ||
44 | #define STEDMA40_LCHAN_SRC_PHY_DST_LOG (0x1 << STEDMA40_INFO_CH_MODE_OPT_POS) | ||
45 | #define STEDMA40_LCHAN_SRC_LOG_DST_PHS (0x2 << STEDMA40_INFO_CH_MODE_OPT_POS) | ||
46 | #define STEDMA40_LCHAN_SRC_LOG_DST_LOG (0x3 << STEDMA40_INFO_CH_MODE_OPT_POS) | ||
47 | |||
48 | /* Interrupt */ | ||
49 | #define STEDMA40_INFO_TIM_POS 10 | ||
50 | #define STEDMA40_NO_TIM_FOR_LINK (0x0 << STEDMA40_INFO_TIM_POS) | ||
51 | #define STEDMA40_TIM_FOR_LINK (0x1 << STEDMA40_INFO_TIM_POS) | ||
52 | |||
53 | /* End of channel_type configuration */ | ||
54 | |||
55 | #define STEDMA40_ESIZE_8_BIT 0x0 | ||
56 | #define STEDMA40_ESIZE_16_BIT 0x1 | ||
57 | #define STEDMA40_ESIZE_32_BIT 0x2 | ||
58 | #define STEDMA40_ESIZE_64_BIT 0x3 | ||
59 | |||
60 | /* The value 4 indicates that PEN-reg shall be set to 0 */ | ||
61 | #define STEDMA40_PSIZE_PHY_1 0x4 | ||
62 | #define STEDMA40_PSIZE_PHY_2 0x0 | ||
63 | #define STEDMA40_PSIZE_PHY_4 0x1 | ||
64 | #define STEDMA40_PSIZE_PHY_8 0x2 | ||
65 | #define STEDMA40_PSIZE_PHY_16 0x3 | ||
66 | |||
67 | /* | ||
68 | * The number of elements differ in logical and | ||
69 | * physical mode | ||
70 | */ | ||
71 | #define STEDMA40_PSIZE_LOG_1 STEDMA40_PSIZE_PHY_2 | ||
72 | #define STEDMA40_PSIZE_LOG_4 STEDMA40_PSIZE_PHY_4 | ||
73 | #define STEDMA40_PSIZE_LOG_8 STEDMA40_PSIZE_PHY_8 | ||
74 | #define STEDMA40_PSIZE_LOG_16 STEDMA40_PSIZE_PHY_16 | ||
75 | |||
76 | enum stedma40_flow_ctrl { | ||
77 | STEDMA40_NO_FLOW_CTRL, | ||
78 | STEDMA40_FLOW_CTRL, | ||
79 | }; | ||
80 | |||
81 | enum stedma40_endianess { | ||
82 | STEDMA40_LITTLE_ENDIAN, | ||
83 | STEDMA40_BIG_ENDIAN | ||
84 | }; | ||
85 | |||
86 | enum stedma40_periph_data_width { | ||
87 | STEDMA40_BYTE_WIDTH = STEDMA40_ESIZE_8_BIT, | ||
88 | STEDMA40_HALFWORD_WIDTH = STEDMA40_ESIZE_16_BIT, | ||
89 | STEDMA40_WORD_WIDTH = STEDMA40_ESIZE_32_BIT, | ||
90 | STEDMA40_DOUBLEWORD_WIDTH = STEDMA40_ESIZE_64_BIT | ||
91 | }; | ||
92 | |||
93 | struct stedma40_half_channel_info { | ||
94 | enum stedma40_endianess endianess; | ||
95 | enum stedma40_periph_data_width data_width; | ||
96 | int psize; | ||
97 | enum stedma40_flow_ctrl flow_ctrl; | ||
98 | }; | ||
99 | |||
100 | enum stedma40_xfer_dir { | ||
101 | STEDMA40_MEM_TO_MEM, | ||
102 | STEDMA40_MEM_TO_PERIPH, | ||
103 | STEDMA40_PERIPH_TO_MEM, | ||
104 | STEDMA40_PERIPH_TO_PERIPH | ||
105 | }; | ||
106 | |||
107 | |||
108 | /** | ||
109 | * struct stedma40_chan_cfg - Structure to be filled by client drivers. | ||
110 | * | ||
111 | * @dir: MEM 2 MEM, PERIPH 2 MEM , MEM 2 PERIPH, PERIPH 2 PERIPH | ||
112 | * @channel_type: priority, mode, mode options and interrupt configuration. | ||
113 | * @src_dev_type: Src device type | ||
114 | * @dst_dev_type: Dst device type | ||
115 | * @src_info: Parameters for dst half channel | ||
116 | * @dst_info: Parameters for dst half channel | ||
117 | * @pre_transfer_data: Data to be passed on to the pre_transfer() function. | ||
118 | * @pre_transfer: Callback used if needed before preparation of transfer. | ||
119 | * Only called if device is set. size of bytes to transfer | ||
120 | * (in case of multiple element transfer size is size of the first element). | ||
121 | * | ||
122 | * | ||
123 | * This structure has to be filled by the client drivers. | ||
124 | * It is recommended to do all dma configurations for clients in the machine. | ||
125 | * | ||
126 | */ | ||
127 | struct stedma40_chan_cfg { | ||
128 | enum stedma40_xfer_dir dir; | ||
129 | unsigned int channel_type; | ||
130 | int src_dev_type; | ||
131 | int dst_dev_type; | ||
132 | struct stedma40_half_channel_info src_info; | ||
133 | struct stedma40_half_channel_info dst_info; | ||
134 | void *pre_transfer_data; | ||
135 | int (*pre_transfer) (struct dma_chan *chan, | ||
136 | void *data, | ||
137 | int size); | ||
138 | }; | ||
139 | |||
140 | /** | ||
141 | * struct stedma40_platform_data - Configuration struct for the dma device. | ||
142 | * | ||
143 | * @dev_len: length of dev_tx and dev_rx | ||
144 | * @dev_tx: mapping between destination event line and io address | ||
145 | * @dev_rx: mapping between source event line and io address | ||
146 | * @memcpy: list of memcpy event lines | ||
147 | * @memcpy_len: length of memcpy | ||
148 | * @memcpy_conf_phy: default configuration of physical channel memcpy | ||
149 | * @memcpy_conf_log: default configuration of logical channel memcpy | ||
150 | * @llis_per_log: number of max linked list items per logical channel | ||
151 | * | ||
152 | */ | ||
153 | struct stedma40_platform_data { | ||
154 | u32 dev_len; | ||
155 | const dma_addr_t *dev_tx; | ||
156 | const dma_addr_t *dev_rx; | ||
157 | int *memcpy; | ||
158 | u32 memcpy_len; | ||
159 | struct stedma40_chan_cfg *memcpy_conf_phy; | ||
160 | struct stedma40_chan_cfg *memcpy_conf_log; | ||
161 | unsigned int llis_per_log; | ||
162 | }; | ||
163 | |||
164 | /** | ||
165 | * setdma40_set_psize() - Used for changing the package size of an | ||
166 | * already configured dma channel. | ||
167 | * | ||
168 | * @chan: dmaengine handle | ||
169 | * @src_psize: new package side for src. (STEDMA40_PSIZE*) | ||
170 | * @src_psize: new package side for dst. (STEDMA40_PSIZE*) | ||
171 | * | ||
172 | * returns 0 on ok, otherwise negative error number. | ||
173 | */ | ||
174 | int stedma40_set_psize(struct dma_chan *chan, | ||
175 | int src_psize, | ||
176 | int dst_psize); | ||
177 | |||
178 | /** | ||
179 | * stedma40_filter() - Provides stedma40_chan_cfg to the | ||
180 | * ste_dma40 dma driver via the dmaengine framework. | ||
181 | * does some checking of what's provided. | ||
182 | * | ||
183 | * Never directly called by client. It used by dmaengine. | ||
184 | * @chan: dmaengine handle. | ||
185 | * @data: Must be of type: struct stedma40_chan_cfg and is | ||
186 | * the configuration of the framework. | ||
187 | * | ||
188 | * | ||
189 | */ | ||
190 | |||
191 | bool stedma40_filter(struct dma_chan *chan, void *data); | ||
192 | |||
193 | /** | ||
194 | * stedma40_memcpy_sg() - extension of the dma framework, memcpy to/from | ||
195 | * scattergatter lists. | ||
196 | * | ||
197 | * @chan: dmaengine handle | ||
198 | * @sgl_dst: Destination scatter list | ||
199 | * @sgl_src: Source scatter list | ||
200 | * @sgl_len: The length of each scatterlist. Both lists must be of equal length | ||
201 | * and each element must match the corresponding element in the other scatter | ||
202 | * list. | ||
203 | * @flags: is actually enum dma_ctrl_flags. See dmaengine.h | ||
204 | */ | ||
205 | |||
206 | struct dma_async_tx_descriptor *stedma40_memcpy_sg(struct dma_chan *chan, | ||
207 | struct scatterlist *sgl_dst, | ||
208 | struct scatterlist *sgl_src, | ||
209 | unsigned int sgl_len, | ||
210 | unsigned long flags); | ||
211 | |||
212 | /** | ||
213 | * stedma40_slave_mem() - Transfers a raw data buffer to or from a slave | ||
214 | * (=device) | ||
215 | * | ||
216 | * @chan: dmaengine handle | ||
217 | * @addr: source or destination physicall address. | ||
218 | * @size: bytes to transfer | ||
219 | * @direction: direction of transfer | ||
220 | * @flags: is actually enum dma_ctrl_flags. See dmaengine.h | ||
221 | */ | ||
222 | |||
223 | static inline struct | ||
224 | dma_async_tx_descriptor *stedma40_slave_mem(struct dma_chan *chan, | ||
225 | dma_addr_t addr, | ||
226 | unsigned int size, | ||
227 | enum dma_data_direction direction, | ||
228 | unsigned long flags) | ||
229 | { | ||
230 | struct scatterlist sg; | ||
231 | sg_init_table(&sg, 1); | ||
232 | sg.dma_address = addr; | ||
233 | sg.length = size; | ||
234 | |||
235 | return chan->device->device_prep_slave_sg(chan, &sg, 1, | ||
236 | direction, flags); | ||
237 | } | ||
238 | |||
239 | #endif | ||