1 /*
2  * Copyright (c) 2022 TOKITA Hiroshi <tokita.hiroshi@gmail.com>
3  *
4  * SPDX-License-Identifier: Apache-2.0
5  */
6 
7 #include <zephyr/device.h>
8 #include <zephyr/drivers/clock_control.h>
9 #include <zephyr/drivers/clock_control/gd32.h>
10 #include <zephyr/drivers/dma.h>
11 #include <zephyr/drivers/reset.h>
12 #include <zephyr/logging/log.h>
13 
14 #include <gd32_dma.h>
15 #include <zephyr/irq.h>
16 
17 #if DT_HAS_COMPAT_STATUS_OKAY(gd_gd32_dma_v1)
18 #define DT_DRV_COMPAT gd_gd32_dma_v1
19 #elif DT_HAS_COMPAT_STATUS_OKAY(gd_gd32_dma)
20 #define DT_DRV_COMPAT gd_gd32_dma
21 #endif
22 
23 #if DT_HAS_COMPAT_STATUS_OKAY(gd_gd32_dma_v1)
24 #define CHXCTL_PERIEN_OFFSET	  ((uint32_t)25U)
25 #define GD32_DMA_CHXCTL_DIR	  BIT(6)
26 #define GD32_DMA_CHXCTL_M2M	  BIT(7)
27 #define GD32_DMA_INTERRUPT_ERRORS (DMA_CHXCTL_SDEIE | DMA_CHXCTL_TAEIE)
28 #define GD32_DMA_FLAG_ERRORS	  (DMA_FLAG_SDE | DMA_FLAG_TAE)
29 #else
30 #define GD32_DMA_CHXCTL_DIR	  BIT(4)
31 #define GD32_DMA_CHXCTL_M2M	  BIT(14)
32 #define GD32_DMA_INTERRUPT_ERRORS DMA_CHXCTL_ERRIE
33 #define GD32_DMA_FLAG_ERRORS	  DMA_FLAG_ERR
34 #endif
35 
36 #ifdef CONFIG_SOC_SERIES_GD32F3X0
37 #undef DMA_INTF
38 #undef DMA_INTC
39 #undef DMA_CHCTL
40 #undef DMA_CHCNT
41 #undef DMA_CHPADDR
42 #undef DMA_CHMADDR
43 
44 #define DMA_INTF(dma)	     REG32(dma + 0x00UL)
45 #define DMA_INTC(dma)	     REG32(dma + 0x04UL)
46 #define DMA_CHCTL(dma, ch)   REG32((dma + 0x08UL) + 0x14UL * (uint32_t)(ch))
47 #define DMA_CHCNT(dma, ch)   REG32((dma + 0x0CUL) + 0x14UL * (uint32_t)(ch))
48 #define DMA_CHPADDR(dma, ch) REG32((dma + 0x10UL) + 0x14UL * (uint32_t)(ch))
49 #define DMA_CHMADDR(dma, ch) REG32((dma + 0x14UL) + 0x14UL * (uint32_t)(ch))
50 #endif
51 
52 #define GD32_DMA_INTF(dma)	  DMA_INTF(dma)
53 #define GD32_DMA_INTC(dma)	  DMA_INTC(dma)
54 #define GD32_DMA_CHCTL(dma, ch)	  DMA_CHCTL((dma), (ch))
55 #define GD32_DMA_CHCNT(dma, ch)	  DMA_CHCNT((dma), (ch))
56 #define GD32_DMA_CHPADDR(dma, ch) DMA_CHPADDR((dma), (ch))
57 #define GD32_DMA_CHMADDR(dma, ch) DMA_CHMADDR((dma), (ch))
58 
59 LOG_MODULE_REGISTER(dma_gd32, CONFIG_DMA_LOG_LEVEL);
60 
61 struct dma_gd32_config {
62 	uint32_t reg;
63 	uint32_t channels;
64 	uint16_t clkid;
65 	bool mem2mem;
66 #if DT_HAS_COMPAT_STATUS_OKAY(gd_gd32_dma_v1)
67 	struct reset_dt_spec reset;
68 #endif
69 	void (*irq_configure)(void);
70 };
71 
72 struct dma_gd32_channel {
73 	dma_callback_t callback;
74 	void *user_data;
75 	uint32_t direction;
76 	bool busy;
77 };
78 
79 struct dma_gd32_data {
80 	struct dma_context ctx;
81 	struct dma_gd32_channel *channels;
82 };
83 
84 struct dma_gd32_srcdst_config {
85 	uint32_t addr;
86 	uint32_t adj;
87 	uint32_t width;
88 };
89 
90 /*
91  * Register access functions
92  */
93 
94 static inline void
gd32_dma_periph_increase_enable(uint32_t reg,dma_channel_enum ch)95 gd32_dma_periph_increase_enable(uint32_t reg, dma_channel_enum ch)
96 {
97 	GD32_DMA_CHCTL(reg, ch) |= DMA_CHXCTL_PNAGA;
98 }
99 
100 static inline void
gd32_dma_periph_increase_disable(uint32_t reg,dma_channel_enum ch)101 gd32_dma_periph_increase_disable(uint32_t reg, dma_channel_enum ch)
102 {
103 	GD32_DMA_CHCTL(reg, ch) &= ~DMA_CHXCTL_PNAGA;
104 }
105 
106 static inline void
gd32_dma_transfer_set_memory_to_memory(uint32_t reg,dma_channel_enum ch)107 gd32_dma_transfer_set_memory_to_memory(uint32_t reg, dma_channel_enum ch)
108 {
109 	GD32_DMA_CHCTL(reg, ch) |= GD32_DMA_CHXCTL_M2M;
110 	GD32_DMA_CHCTL(reg, ch) &= ~GD32_DMA_CHXCTL_DIR;
111 }
112 
113 static inline void
gd32_dma_transfer_set_memory_to_periph(uint32_t reg,dma_channel_enum ch)114 gd32_dma_transfer_set_memory_to_periph(uint32_t reg, dma_channel_enum ch)
115 {
116 	GD32_DMA_CHCTL(reg, ch) &= ~GD32_DMA_CHXCTL_M2M;
117 	GD32_DMA_CHCTL(reg, ch) |= GD32_DMA_CHXCTL_DIR;
118 }
119 
120 static inline void
gd32_dma_transfer_set_periph_to_memory(uint32_t reg,dma_channel_enum ch)121 gd32_dma_transfer_set_periph_to_memory(uint32_t reg, dma_channel_enum ch)
122 {
123 	GD32_DMA_CHCTL(reg, ch) &= ~GD32_DMA_CHXCTL_M2M;
124 	GD32_DMA_CHCTL(reg, ch) &= ~GD32_DMA_CHXCTL_DIR;
125 }
126 
127 static inline void
gd32_dma_memory_increase_enable(uint32_t reg,dma_channel_enum ch)128 gd32_dma_memory_increase_enable(uint32_t reg, dma_channel_enum ch)
129 {
130 	GD32_DMA_CHCTL(reg, ch) |= DMA_CHXCTL_MNAGA;
131 }
132 
133 static inline void
gd32_dma_memory_increase_disable(uint32_t reg,dma_channel_enum ch)134 gd32_dma_memory_increase_disable(uint32_t reg, dma_channel_enum ch)
135 {
136 	GD32_DMA_CHCTL(reg, ch) &= ~DMA_CHXCTL_MNAGA;
137 }
138 
139 static inline void
gd32_dma_circulation_enable(uint32_t reg,dma_channel_enum ch)140 gd32_dma_circulation_enable(uint32_t reg, dma_channel_enum ch)
141 {
142 	GD32_DMA_CHCTL(reg, ch) |= DMA_CHXCTL_CMEN;
143 }
144 
145 static inline void
gd32_dma_circulation_disable(uint32_t reg,dma_channel_enum ch)146 gd32_dma_circulation_disable(uint32_t reg, dma_channel_enum ch)
147 {
148 	GD32_DMA_CHCTL(reg, ch) &= ~DMA_CHXCTL_CMEN;
149 }
150 
gd32_dma_channel_enable(uint32_t reg,dma_channel_enum ch)151 static inline void gd32_dma_channel_enable(uint32_t reg, dma_channel_enum ch)
152 {
153 	GD32_DMA_CHCTL(reg, ch) |= DMA_CHXCTL_CHEN;
154 }
155 
gd32_dma_channel_disable(uint32_t reg,dma_channel_enum ch)156 static inline void gd32_dma_channel_disable(uint32_t reg, dma_channel_enum ch)
157 {
158 	GD32_DMA_CHCTL(reg, ch) &= ~DMA_CHXCTL_CHEN;
159 }
160 
161 static inline void
gd32_dma_interrupt_enable(uint32_t reg,dma_channel_enum ch,uint32_t source)162 gd32_dma_interrupt_enable(uint32_t reg, dma_channel_enum ch, uint32_t source)
163 {
164 	GD32_DMA_CHCTL(reg, ch) |= source;
165 }
166 
167 static inline void
gd32_dma_interrupt_disable(uint32_t reg,dma_channel_enum ch,uint32_t source)168 gd32_dma_interrupt_disable(uint32_t reg, dma_channel_enum ch, uint32_t source)
169 {
170 	GD32_DMA_CHCTL(reg, ch) &= ~source;
171 }
172 
173 static inline void
gd32_dma_priority_config(uint32_t reg,dma_channel_enum ch,uint32_t priority)174 gd32_dma_priority_config(uint32_t reg, dma_channel_enum ch, uint32_t priority)
175 {
176 	uint32_t ctl = GD32_DMA_CHCTL(reg, ch);
177 
178 	GD32_DMA_CHCTL(reg, ch) = (ctl & (~DMA_CHXCTL_PRIO)) | priority;
179 }
180 
181 static inline void
gd32_dma_memory_width_config(uint32_t reg,dma_channel_enum ch,uint32_t mwidth)182 gd32_dma_memory_width_config(uint32_t reg, dma_channel_enum ch, uint32_t mwidth)
183 {
184 	uint32_t ctl = GD32_DMA_CHCTL(reg, ch);
185 
186 	GD32_DMA_CHCTL(reg, ch) = (ctl & (~DMA_CHXCTL_MWIDTH)) | mwidth;
187 }
188 
189 static inline void
gd32_dma_periph_width_config(uint32_t reg,dma_channel_enum ch,uint32_t pwidth)190 gd32_dma_periph_width_config(uint32_t reg, dma_channel_enum ch, uint32_t pwidth)
191 {
192 	uint32_t ctl = GD32_DMA_CHCTL(reg, ch);
193 
194 	GD32_DMA_CHCTL(reg, ch) = (ctl & (~DMA_CHXCTL_PWIDTH)) | pwidth;
195 }
196 
197 #if DT_HAS_COMPAT_STATUS_OKAY(gd_gd32_dma_v1)
198 static inline void
gd32_dma_channel_subperipheral_select(uint32_t reg,dma_channel_enum ch,dma_subperipheral_enum sub_periph)199 gd32_dma_channel_subperipheral_select(uint32_t reg, dma_channel_enum ch,
200 				      dma_subperipheral_enum sub_periph)
201 {
202 	uint32_t ctl = GD32_DMA_CHCTL(reg, ch);
203 
204 	GD32_DMA_CHCTL(reg, ch) =
205 		(ctl & (~DMA_CHXCTL_PERIEN)) |
206 		((uint32_t)sub_periph << CHXCTL_PERIEN_OFFSET);
207 }
208 #endif
209 
210 static inline void
gd32_dma_periph_address_config(uint32_t reg,dma_channel_enum ch,uint32_t addr)211 gd32_dma_periph_address_config(uint32_t reg, dma_channel_enum ch, uint32_t addr)
212 {
213 	GD32_DMA_CHPADDR(reg, ch) = addr;
214 }
215 
216 static inline void
gd32_dma_memory_address_config(uint32_t reg,dma_channel_enum ch,uint32_t addr)217 gd32_dma_memory_address_config(uint32_t reg, dma_channel_enum ch, uint32_t addr)
218 {
219 #if DT_HAS_COMPAT_STATUS_OKAY(gd_gd32_dma_v1)
220 	DMA_CHM0ADDR(reg, ch) = addr;
221 #else
222 	GD32_DMA_CHMADDR(reg, ch) = addr;
223 #endif
224 }
225 
226 static inline void
gd32_dma_transfer_number_config(uint32_t reg,dma_channel_enum ch,uint32_t num)227 gd32_dma_transfer_number_config(uint32_t reg, dma_channel_enum ch, uint32_t num)
228 {
229 	GD32_DMA_CHCNT(reg, ch) = (num & DMA_CHXCNT_CNT);
230 }
231 
232 static inline uint32_t
gd32_dma_transfer_number_get(uint32_t reg,dma_channel_enum ch)233 gd32_dma_transfer_number_get(uint32_t reg, dma_channel_enum ch)
234 {
235 	return GD32_DMA_CHCNT(reg, ch);
236 }
237 
238 static inline void
gd32_dma_interrupt_flag_clear(uint32_t reg,dma_channel_enum ch,uint32_t flag)239 gd32_dma_interrupt_flag_clear(uint32_t reg, dma_channel_enum ch, uint32_t flag)
240 {
241 #if DT_HAS_COMPAT_STATUS_OKAY(gd_gd32_dma_v1)
242 	if (ch < DMA_CH4) {
243 		DMA_INTC0(reg) |= DMA_FLAG_ADD(flag, ch);
244 	} else {
245 		DMA_INTC1(reg) |= DMA_FLAG_ADD(flag, ch - DMA_CH4);
246 	}
247 #else
248 	GD32_DMA_INTC(reg) |= DMA_FLAG_ADD(flag, ch);
249 #endif
250 }
251 
252 static inline void
gd32_dma_flag_clear(uint32_t reg,dma_channel_enum ch,uint32_t flag)253 gd32_dma_flag_clear(uint32_t reg, dma_channel_enum ch, uint32_t flag)
254 {
255 #if DT_HAS_COMPAT_STATUS_OKAY(gd_gd32_dma_v1)
256 	if (ch < DMA_CH4) {
257 		DMA_INTC0(reg) |= DMA_FLAG_ADD(flag, ch);
258 	} else {
259 		DMA_INTC1(reg) |= DMA_FLAG_ADD(flag, ch - DMA_CH4);
260 	}
261 #else
262 	GD32_DMA_INTC(reg) |= DMA_FLAG_ADD(flag, ch);
263 #endif
264 }
265 
266 static inline uint32_t
gd32_dma_interrupt_flag_get(uint32_t reg,dma_channel_enum ch,uint32_t flag)267 gd32_dma_interrupt_flag_get(uint32_t reg, dma_channel_enum ch, uint32_t flag)
268 {
269 #if DT_HAS_COMPAT_STATUS_OKAY(gd_gd32_dma_v1)
270 	if (ch < DMA_CH4) {
271 		return (DMA_INTF0(reg) & DMA_FLAG_ADD(flag, ch));
272 	} else {
273 		return (DMA_INTF1(reg) & DMA_FLAG_ADD(flag, ch - DMA_CH4));
274 	}
275 #else
276 	return (GD32_DMA_INTF(reg) & DMA_FLAG_ADD(flag, ch));
277 #endif
278 }
279 
gd32_dma_deinit(uint32_t reg,dma_channel_enum ch)280 static inline void gd32_dma_deinit(uint32_t reg, dma_channel_enum ch)
281 {
282 	GD32_DMA_CHCTL(reg, ch) &= ~DMA_CHXCTL_CHEN;
283 
284 	GD32_DMA_CHCTL(reg, ch) = DMA_CHCTL_RESET_VALUE;
285 	GD32_DMA_CHCNT(reg, ch) = DMA_CHCNT_RESET_VALUE;
286 	GD32_DMA_CHPADDR(reg, ch) = DMA_CHPADDR_RESET_VALUE;
287 #if DT_HAS_COMPAT_STATUS_OKAY(gd_gd32_dma_v1)
288 	DMA_CHM0ADDR(reg, ch) = DMA_CHMADDR_RESET_VALUE;
289 	DMA_CHFCTL(reg, ch) = DMA_CHFCTL_RESET_VALUE;
290 	if (ch < DMA_CH4) {
291 		DMA_INTC0(reg) |= DMA_FLAG_ADD(DMA_CHINTF_RESET_VALUE, ch);
292 	} else {
293 		DMA_INTC1(reg) |=
294 			DMA_FLAG_ADD(DMA_CHINTF_RESET_VALUE, ch - DMA_CH4);
295 	}
296 #else
297 	GD32_DMA_CHMADDR(reg, ch) = DMA_CHMADDR_RESET_VALUE;
298 	GD32_DMA_INTC(reg) |= DMA_FLAG_ADD(DMA_CHINTF_RESET_VALUE, ch);
299 #endif
300 }
301 
302 /*
303  * Utility functions
304  */
305 
dma_gd32_priority(uint32_t prio)306 static inline uint32_t dma_gd32_priority(uint32_t prio)
307 {
308 	return CHCTL_PRIO(prio);
309 }
310 
dma_gd32_memory_width(uint32_t width)311 static inline uint32_t dma_gd32_memory_width(uint32_t width)
312 {
313 	switch (width) {
314 	case 4:
315 		return CHCTL_MWIDTH(2);
316 	case 2:
317 		return CHCTL_MWIDTH(1);
318 	default:
319 		return CHCTL_MWIDTH(0);
320 	}
321 }
322 
dma_gd32_periph_width(uint32_t width)323 static inline uint32_t dma_gd32_periph_width(uint32_t width)
324 {
325 	switch (width) {
326 	case 4:
327 		return CHCTL_PWIDTH(2);
328 	case 2:
329 		return CHCTL_PWIDTH(1);
330 	default:
331 		return CHCTL_PWIDTH(0);
332 	}
333 }
334 
335 /*
336  * API functions
337  */
338 
dma_gd32_config(const struct device * dev,uint32_t channel,struct dma_config * dma_cfg)339 static int dma_gd32_config(const struct device *dev, uint32_t channel,
340 			   struct dma_config *dma_cfg)
341 {
342 	const struct dma_gd32_config *cfg = dev->config;
343 	struct dma_gd32_data *data = dev->data;
344 	struct dma_gd32_srcdst_config src_cfg;
345 	struct dma_gd32_srcdst_config dst_cfg;
346 	struct dma_gd32_srcdst_config *memory_cfg = NULL;
347 	struct dma_gd32_srcdst_config *periph_cfg = NULL;
348 
349 	if (channel >= cfg->channels) {
350 		LOG_ERR("channel must be < %" PRIu32 " (%" PRIu32 ")",
351 			cfg->channels, channel);
352 		return -EINVAL;
353 	}
354 
355 	if (dma_cfg->block_count != 1) {
356 		LOG_ERR("chained block transfer not supported.");
357 		return -ENOTSUP;
358 	}
359 
360 	if (dma_cfg->channel_priority > 3) {
361 		LOG_ERR("channel_priority must be < 4 (%" PRIu32 ")",
362 			dma_cfg->channel_priority);
363 		return -EINVAL;
364 	}
365 
366 	if (dma_cfg->head_block->source_addr_adj == DMA_ADDR_ADJ_DECREMENT) {
367 		LOG_ERR("source_addr_adj not supported DMA_ADDR_ADJ_DECREMENT");
368 		return -ENOTSUP;
369 	}
370 
371 	if (dma_cfg->head_block->dest_addr_adj == DMA_ADDR_ADJ_DECREMENT) {
372 		LOG_ERR("dest_addr_adj not supported DMA_ADDR_ADJ_DECREMENT");
373 		return -ENOTSUP;
374 	}
375 
376 	if (dma_cfg->head_block->source_addr_adj != DMA_ADDR_ADJ_INCREMENT &&
377 	    dma_cfg->head_block->source_addr_adj != DMA_ADDR_ADJ_NO_CHANGE) {
378 		LOG_ERR("invalid source_addr_adj %" PRIu16,
379 			dma_cfg->head_block->source_addr_adj);
380 		return -ENOTSUP;
381 	}
382 	if (dma_cfg->head_block->dest_addr_adj != DMA_ADDR_ADJ_INCREMENT &&
383 	    dma_cfg->head_block->dest_addr_adj != DMA_ADDR_ADJ_NO_CHANGE) {
384 		LOG_ERR("invalid dest_addr_adj %" PRIu16,
385 			dma_cfg->head_block->dest_addr_adj);
386 		return -ENOTSUP;
387 	}
388 
389 	if (dma_cfg->source_data_size != 1 && dma_cfg->source_data_size != 2 &&
390 	    dma_cfg->source_data_size != 4) {
391 		LOG_ERR("source_data_size must be 1, 2, or 4 (%" PRIu32 ")",
392 			dma_cfg->source_data_size);
393 		return -EINVAL;
394 	}
395 
396 	if (dma_cfg->dest_data_size != 1 && dma_cfg->dest_data_size != 2 &&
397 	    dma_cfg->dest_data_size != 4) {
398 		LOG_ERR("dest_data_size must be 1, 2, or 4 (%" PRIu32 ")",
399 			dma_cfg->dest_data_size);
400 		return -EINVAL;
401 	}
402 
403 	if (dma_cfg->channel_direction > PERIPHERAL_TO_MEMORY) {
404 		LOG_ERR("channel_direction must be MEMORY_TO_MEMORY, "
405 			"MEMORY_TO_PERIPHERAL or PERIPHERAL_TO_MEMORY (%" PRIu32
406 			")",
407 			dma_cfg->channel_direction);
408 		return -ENOTSUP;
409 	}
410 
411 	if (dma_cfg->channel_direction == MEMORY_TO_MEMORY && !cfg->mem2mem) {
412 		LOG_ERR("not supporting MEMORY_TO_MEMORY");
413 		return -ENOTSUP;
414 	}
415 
416 #if DT_HAS_COMPAT_STATUS_OKAY(gd_gd32_dma_v1)
417 	if (dma_cfg->dma_slot > 0xF) {
418 		LOG_ERR("dma_slot must be <7 (%" PRIu32 ")",
419 			dma_cfg->dma_slot);
420 		return -EINVAL;
421 	}
422 #endif
423 
424 	gd32_dma_deinit(cfg->reg, channel);
425 
426 	src_cfg.addr = dma_cfg->head_block->source_address;
427 	src_cfg.adj = dma_cfg->head_block->source_addr_adj;
428 	src_cfg.width = dma_cfg->source_data_size;
429 
430 	dst_cfg.addr = dma_cfg->head_block->dest_address;
431 	dst_cfg.adj = dma_cfg->head_block->dest_addr_adj;
432 	dst_cfg.width = dma_cfg->dest_data_size;
433 
434 	switch (dma_cfg->channel_direction) {
435 	case MEMORY_TO_MEMORY:
436 		gd32_dma_transfer_set_memory_to_memory(cfg->reg, channel);
437 		memory_cfg = &dst_cfg;
438 		periph_cfg = &src_cfg;
439 		break;
440 	case PERIPHERAL_TO_MEMORY:
441 		gd32_dma_transfer_set_periph_to_memory(cfg->reg, channel);
442 		memory_cfg = &dst_cfg;
443 		periph_cfg = &src_cfg;
444 		break;
445 	case MEMORY_TO_PERIPHERAL:
446 		gd32_dma_transfer_set_memory_to_periph(cfg->reg, channel);
447 		memory_cfg = &src_cfg;
448 		periph_cfg = &dst_cfg;
449 		break;
450 	}
451 
452 	gd32_dma_memory_address_config(cfg->reg, channel, memory_cfg->addr);
453 	if (memory_cfg->adj == DMA_ADDR_ADJ_INCREMENT) {
454 		gd32_dma_memory_increase_enable(cfg->reg, channel);
455 	} else {
456 		gd32_dma_memory_increase_disable(cfg->reg, channel);
457 	}
458 
459 	gd32_dma_periph_address_config(cfg->reg, channel, periph_cfg->addr);
460 	if (periph_cfg->adj == DMA_ADDR_ADJ_INCREMENT) {
461 		gd32_dma_periph_increase_enable(cfg->reg, channel);
462 	} else {
463 		gd32_dma_periph_increase_disable(cfg->reg, channel);
464 	}
465 
466 	gd32_dma_transfer_number_config(cfg->reg, channel,
467 					dma_cfg->head_block->block_size);
468 	gd32_dma_priority_config(cfg->reg, channel,
469 				 dma_gd32_priority(dma_cfg->channel_priority));
470 	gd32_dma_memory_width_config(cfg->reg, channel,
471 				     dma_gd32_memory_width(memory_cfg->width));
472 	gd32_dma_periph_width_config(cfg->reg, channel,
473 				     dma_gd32_periph_width(periph_cfg->width));
474 	gd32_dma_circulation_disable(cfg->reg, channel);
475 #if DT_HAS_COMPAT_STATUS_OKAY(gd_gd32_dma_v1)
476 	if (dma_cfg->channel_direction != MEMORY_TO_MEMORY) {
477 		gd32_dma_channel_subperipheral_select(cfg->reg, channel,
478 						      dma_cfg->dma_slot);
479 	}
480 #endif
481 
482 	data->channels[channel].callback = dma_cfg->dma_callback;
483 	data->channels[channel].user_data = dma_cfg->user_data;
484 	data->channels[channel].direction = dma_cfg->channel_direction;
485 
486 	return 0;
487 }
488 
dma_gd32_reload(const struct device * dev,uint32_t ch,uint32_t src,uint32_t dst,size_t size)489 static int dma_gd32_reload(const struct device *dev, uint32_t ch, uint32_t src,
490 			   uint32_t dst, size_t size)
491 {
492 	const struct dma_gd32_config *cfg = dev->config;
493 	struct dma_gd32_data *data = dev->data;
494 
495 	if (ch >= cfg->channels) {
496 		LOG_ERR("reload channel must be < %" PRIu32 " (%" PRIu32 ")",
497 			cfg->channels, ch);
498 		return -EINVAL;
499 	}
500 
501 	if (data->channels[ch].busy) {
502 		return -EBUSY;
503 	}
504 
505 	gd32_dma_channel_disable(cfg->reg, ch);
506 
507 	gd32_dma_transfer_number_config(cfg->reg, ch, size);
508 
509 	switch (data->channels[ch].direction) {
510 	case MEMORY_TO_MEMORY:
511 	case PERIPHERAL_TO_MEMORY:
512 		gd32_dma_memory_address_config(cfg->reg, ch, dst);
513 		gd32_dma_periph_address_config(cfg->reg, ch, src);
514 		break;
515 	case MEMORY_TO_PERIPHERAL:
516 		gd32_dma_memory_address_config(cfg->reg, ch, src);
517 		gd32_dma_periph_address_config(cfg->reg, ch, dst);
518 		break;
519 	}
520 
521 	gd32_dma_channel_enable(cfg->reg, ch);
522 
523 	return 0;
524 }
525 
dma_gd32_start(const struct device * dev,uint32_t ch)526 static int dma_gd32_start(const struct device *dev, uint32_t ch)
527 {
528 	const struct dma_gd32_config *cfg = dev->config;
529 	struct dma_gd32_data *data = dev->data;
530 
531 	if (ch >= cfg->channels) {
532 		LOG_ERR("start channel must be < %" PRIu32 " (%" PRIu32 ")",
533 			cfg->channels, ch);
534 		return -EINVAL;
535 	}
536 
537 	gd32_dma_interrupt_enable(cfg->reg, ch,
538 				  DMA_CHXCTL_FTFIE | GD32_DMA_INTERRUPT_ERRORS);
539 	gd32_dma_channel_enable(cfg->reg, ch);
540 	data->channels[ch].busy = true;
541 
542 	return 0;
543 }
544 
dma_gd32_stop(const struct device * dev,uint32_t ch)545 static int dma_gd32_stop(const struct device *dev, uint32_t ch)
546 {
547 	const struct dma_gd32_config *cfg = dev->config;
548 	struct dma_gd32_data *data = dev->data;
549 
550 	if (ch >= cfg->channels) {
551 		LOG_ERR("stop channel must be < %" PRIu32 " (%" PRIu32 ")",
552 			cfg->channels, ch);
553 		return -EINVAL;
554 	}
555 
556 	gd32_dma_interrupt_disable(
557 		cfg->reg, ch, DMA_CHXCTL_FTFIE | GD32_DMA_INTERRUPT_ERRORS);
558 	gd32_dma_interrupt_flag_clear(cfg->reg, ch,
559 				      DMA_FLAG_FTF | GD32_DMA_FLAG_ERRORS);
560 	gd32_dma_channel_disable(cfg->reg, ch);
561 	data->channels[ch].busy = false;
562 
563 	return 0;
564 }
565 
dma_gd32_get_status(const struct device * dev,uint32_t ch,struct dma_status * stat)566 static int dma_gd32_get_status(const struct device *dev, uint32_t ch,
567 			       struct dma_status *stat)
568 {
569 	const struct dma_gd32_config *cfg = dev->config;
570 	struct dma_gd32_data *data = dev->data;
571 
572 	if (ch >= cfg->channels) {
573 		LOG_ERR("channel must be < %" PRIu32 " (%" PRIu32 ")",
574 			cfg->channels, ch);
575 		return -EINVAL;
576 	}
577 
578 	stat->pending_length = gd32_dma_transfer_number_get(cfg->reg, ch);
579 	stat->dir = data->channels[ch].direction;
580 	stat->busy = data->channels[ch].busy;
581 
582 	return 0;
583 }
584 
dma_gd32_api_chan_filter(const struct device * dev,int ch,void * filter_param)585 static bool dma_gd32_api_chan_filter(const struct device *dev, int ch,
586 				     void *filter_param)
587 {
588 	uint32_t filter;
589 
590 	if (!filter_param) {
591 		LOG_ERR("filter_param must not be NULL");
592 		return false;
593 	}
594 
595 	filter = *((uint32_t *)filter_param);
596 
597 	return (filter & BIT(ch));
598 }
599 
dma_gd32_init(const struct device * dev)600 static int dma_gd32_init(const struct device *dev)
601 {
602 	const struct dma_gd32_config *cfg = dev->config;
603 
604 	(void)clock_control_on(GD32_CLOCK_CONTROLLER,
605 			       (clock_control_subsys_t)&cfg->clkid);
606 
607 #if DT_HAS_COMPAT_STATUS_OKAY(gd_gd32_dma_v1)
608 	(void)reset_line_toggle_dt(&cfg->reset);
609 #endif
610 
611 	for (uint32_t i = 0; i < cfg->channels; i++) {
612 		gd32_dma_interrupt_disable(cfg->reg, i,
613 			   DMA_CHXCTL_FTFIE | GD32_DMA_INTERRUPT_ERRORS);
614 		gd32_dma_deinit(cfg->reg, i);
615 	}
616 
617 	cfg->irq_configure();
618 
619 	return 0;
620 }
621 
dma_gd32_isr(const struct device * dev)622 static void dma_gd32_isr(const struct device *dev)
623 {
624 	const struct dma_gd32_config *cfg = dev->config;
625 	struct dma_gd32_data *data = dev->data;
626 	uint32_t errflag, ftfflag;
627 	int err = 0;
628 
629 	for (uint32_t i = 0; i < cfg->channels; i++) {
630 		errflag = gd32_dma_interrupt_flag_get(cfg->reg, i,
631 						      GD32_DMA_FLAG_ERRORS);
632 		ftfflag =
633 			gd32_dma_interrupt_flag_get(cfg->reg, i, DMA_FLAG_FTF);
634 
635 		if (errflag == 0 && ftfflag == 0) {
636 			continue;
637 		}
638 
639 		if (errflag) {
640 			err = -EIO;
641 		}
642 
643 		gd32_dma_interrupt_flag_clear(
644 			cfg->reg, i, DMA_FLAG_FTF | GD32_DMA_FLAG_ERRORS);
645 		data->channels[i].busy = false;
646 
647 		if (data->channels[i].callback) {
648 			data->channels[i].callback(
649 				dev, data->channels[i].user_data, i, err);
650 		}
651 	}
652 }
653 
654 static DEVICE_API(dma, dma_gd32_driver_api) = {
655 	.config = dma_gd32_config,
656 	.reload = dma_gd32_reload,
657 	.start = dma_gd32_start,
658 	.stop = dma_gd32_stop,
659 	.get_status = dma_gd32_get_status,
660 	.chan_filter = dma_gd32_api_chan_filter,
661 };
662 
663 #define IRQ_CONFIGURE(n, inst)                                                 \
664 	IRQ_CONNECT(DT_INST_IRQ_BY_IDX(inst, n, irq),                          \
665 		    DT_INST_IRQ_BY_IDX(inst, n, priority), dma_gd32_isr,       \
666 		    DEVICE_DT_INST_GET(inst), 0);                              \
667 	irq_enable(DT_INST_IRQ_BY_IDX(inst, n, irq));
668 
669 #define CONFIGURE_ALL_IRQS(inst, n) LISTIFY(n, IRQ_CONFIGURE, (), inst)
670 
671 #define GD32_DMA_INIT(inst)                                                    \
672 	static void dma_gd32##inst##_irq_configure(void)                       \
673 	{                                                                      \
674 		CONFIGURE_ALL_IRQS(inst, DT_NUM_IRQS(DT_DRV_INST(inst)));      \
675 	}                                                                      \
676 	static const struct dma_gd32_config dma_gd32##inst##_config = {        \
677 		.reg = DT_INST_REG_ADDR(inst),                                 \
678 		.channels = DT_INST_PROP(inst, dma_channels),                  \
679 		.clkid = DT_INST_CLOCKS_CELL(inst, id),                        \
680 		.mem2mem = DT_INST_PROP(inst, gd_mem2mem),                     \
681 		IF_ENABLED(DT_HAS_COMPAT_STATUS_OKAY(gd_gd32_dma_v1),          \
682 			   (.reset = RESET_DT_SPEC_INST_GET(inst),))           \
683 		.irq_configure = dma_gd32##inst##_irq_configure,               \
684 	};                                                                     \
685                                                                                \
686 	static struct dma_gd32_channel                                         \
687 		dma_gd32##inst##_channels[DT_INST_PROP(inst, dma_channels)];   \
688 	ATOMIC_DEFINE(dma_gd32_atomic##inst,                                   \
689 		      DT_INST_PROP(inst, dma_channels));                       \
690 	static struct dma_gd32_data dma_gd32##inst##_data = {                  \
691 		.ctx =  {                                                      \
692 			.magic = DMA_MAGIC,                                    \
693 			.atomic = dma_gd32_atomic##inst,                       \
694 			.dma_channels = DT_INST_PROP(inst, dma_channels),      \
695 		},                                                             \
696 		.channels = dma_gd32##inst##_channels,                         \
697 	};                                                                     \
698                                                                                \
699 	DEVICE_DT_INST_DEFINE(inst, &dma_gd32_init, NULL,                      \
700 			      &dma_gd32##inst##_data,                          \
701 			      &dma_gd32##inst##_config, POST_KERNEL,           \
702 			      CONFIG_DMA_INIT_PRIORITY, &dma_gd32_driver_api);
703 
704 DT_INST_FOREACH_STATUS_OKAY(GD32_DMA_INIT)
705