@web-font-path: "roboto-debian.css";
Loading...
Searching...
No Matches
dma.h
Go to the documentation of this file.
1/*
2 * Copyright (c) 2020 Raspberry Pi (Trading) Ltd.
3 *
4 * SPDX-License-Identifier: BSD-3-Clause
5 */
6
7#ifndef _HARDWARE_DMA_H
8#define _HARDWARE_DMA_H
9
10#include "pico.h"
11#include "hardware/structs/dma.h"
12#include "hardware/regs/dreq.h"
13#include "pico/assert.h"
14#include "hardware/regs/intctrl.h"
15
16#ifdef __cplusplus
17extern "C" {
18#endif
19
37
38// PICO_CONFIG: PARAM_ASSERTIONS_ENABLED_HARDWARE_DMA, Enable/disable hardware_dma assertions, type=bool, default=0, group=hardware_dma
39#ifndef PARAM_ASSERTIONS_ENABLED_HARDWARE_DMA
40#ifdef PARAM_ASSERTIONS_ENABLED_DMA // backwards compatibility with SDK < 2.0.0
41#define PARAM_ASSERTIONS_ENABLED_HARDWARE_DMA PARAM_ASSERTIONS_ENABLED_DMA
42#else
43#define PARAM_ASSERTIONS_ENABLED_HARDWARE_DMA 0
44#endif
45#endif
46
55#ifndef DMA_IRQ_NUM
56#define DMA_IRQ_NUM(irq_index) (DMA_IRQ_0 + (irq_index))
57#endif
58
59static inline void check_dma_channel_param(__unused uint channel) {
60#if PARAM_ASSERTIONS_ENABLED(HARDWARE_DMA)
61 // this method is used a lot by inline functions so avoid code bloat by deferring to function
62 extern void check_dma_channel_param_impl(uint channel);
63 check_dma_channel_param_impl(channel);
64#endif
65}
66
67static inline void check_dma_timer_param(__unused uint timer_num) {
68 valid_params_if(HARDWARE_DMA, timer_num < NUM_DMA_TIMERS);
69}
70
71inline static dma_channel_hw_t *dma_channel_hw_addr(uint channel) {
72 check_dma_channel_param(channel);
73 return &dma_hw->ch[channel];
74}
75
85void dma_channel_claim(uint channel);
86
96void dma_claim_mask(uint32_t channel_mask);
97
103void dma_channel_unclaim(uint channel);
104
110void dma_unclaim_mask(uint32_t channel_mask);
111
118int dma_claim_unused_channel(bool required);
119
128bool dma_channel_is_claimed(uint channel);
129
137
148
155#if !PICO_RP2040
156 DMA_ADDRESS_UPDATE_INCREMENT_BY_TWO = 2,
157 DMA_ADDRESS_UPDATE_DECREMENT = 3,
158#endif
160
164typedef struct {
165 uint32_t ctrl;
167
168// backwards compatibility
169typedef dma_channel_config_t dma_channel_config;
170
171#ifndef DMA_ADDRESS_UPDATE_TYPE_TO_DMA_CH_CTRL_READ_BITS
172#if PICO_RP2040
173#define DMA_ADDRESS_UPDATE_TYPE_TO_DMA_CH_CTRL_READ_BITS(u) (((u)&1) << DMA_CH0_CTRL_TRIG_INCR_READ_LSB)
174#else
175#define DMA_ADDRESS_UPDATE_TYPE_TO_DMA_CH_CTRL_READ_BITS(u) \
176 ((((u)&1) << DMA_CH0_CTRL_TRIG_INCR_READ_LSB) | \
177 (((u)&2) << (DMA_CH0_CTRL_TRIG_INCR_READ_REV_LSB - 1)))
178#endif
179#endif
180
181#ifndef DMA_ADDRESS_UPDATE_TYPE_TO_DMA_CH_CTRL_WRITE_BITS
182#if PICO_RP2040
183#define DMA_ADDRESS_UPDATE_TYPE_TO_DMA_CH_CTRL_WRITE_BITS(u) (((u)&1) << DMA_CH0_CTRL_TRIG_INCR_WRITE_LSB)
184#else
185#define DMA_ADDRESS_UPDATE_TYPE_TO_DMA_CH_CTRL_WRITE_BITS(u) \
186((((u)&1) << DMA_CH0_CTRL_TRIG_INCR_WRITE_LSB) | \
187(((u)&2) << (DMA_CH0_CTRL_TRIG_INCR_WRITE_REV_LSB - 1)))
188#endif
189#endif
190
191#ifndef DMA_CH_CTRL_ALL_ADDRESS_UPDATE_READ_BITS
192#if PICO_RP2040
193#define DMA_CH_CTRL_ALL_ADDRESS_UPDATE_READ_BITS DMA_CH0_CTRL_TRIG_INCR_READ_BITS
194#else
195#define DMA_CH_CTRL_ALL_ADDRESS_UPDATE_READ_BITS (DMA_CH0_CTRL_TRIG_INCR_READ_BITS | DMA_CH0_CTRL_TRIG_INCR_READ_REV_BITS)
196#endif
197#endif
198
199#ifndef DMA_CH_CTRL_ALL_ADDRESS_UPDATE_WRITE_BITS
200#if PICO_RP2040
201#define DMA_CH_CTRL_ALL_ADDRESS_UPDATE_WRITE_BITS DMA_CH0_CTRL_TRIG_INCR_WRITE_BITS
202#else
203#define DMA_CH_CTRL_ALL_ADDRESS_UPDATE_WRITE_BITS (DMA_CH0_CTRL_TRIG_INCR_WRITE_BITS | DMA_CH0_CTRL_TRIG_INCR_WRITE_REV_BITS)
204#endif
205#endif
206
216 c->ctrl = (c->ctrl & ~DMA_CH_CTRL_ALL_ADDRESS_UPDATE_READ_BITS) |
217 DMA_ADDRESS_UPDATE_TYPE_TO_DMA_CH_CTRL_READ_BITS(update_type);
218}
219
229 c->ctrl = (c->ctrl & ~DMA_CH_CTRL_ALL_ADDRESS_UPDATE_WRITE_BITS) |
230 DMA_ADDRESS_UPDATE_TYPE_TO_DMA_CH_CTRL_WRITE_BITS(update_type);
231}
232
249
266
282static inline void channel_config_set_dreq(dma_channel_config_t *c, uint dreq) {
283 assert(dreq <= DREQ_FORCE);
284 c->ctrl = (c->ctrl & ~DMA_CH0_CTRL_TRIG_TREQ_SEL_BITS) | (dreq << DMA_CH0_CTRL_TRIG_TREQ_SEL_LSB);
285}
286
296static inline void channel_config_set_chain_to(dma_channel_config_t *c, uint chain_to) {
297 assert(chain_to <= NUM_DMA_CHANNELS);
298 c->ctrl = (c->ctrl & ~DMA_CH0_CTRL_TRIG_CHAIN_TO_BITS) | (chain_to << DMA_CH0_CTRL_TRIG_CHAIN_TO_LSB);
299}
300
311 assert(size == DMA_SIZE_8 || size == DMA_SIZE_16 || size == DMA_SIZE_32);
312 c->ctrl = (c->ctrl & ~DMA_CH0_CTRL_TRIG_DATA_SIZE_BITS) | (((uint)size) << DMA_CH0_CTRL_TRIG_DATA_SIZE_LSB);
313}
314
330static inline void channel_config_set_ring(dma_channel_config_t *c, bool write, uint size_bits) {
331 assert(size_bits < 32);
332 c->ctrl = (c->ctrl & ~(DMA_CH0_CTRL_TRIG_RING_SIZE_BITS | DMA_CH0_CTRL_TRIG_RING_SEL_BITS)) |
333 (size_bits << DMA_CH0_CTRL_TRIG_RING_SIZE_LSB) |
334 (write ? DMA_CH0_CTRL_TRIG_RING_SEL_BITS : 0);
335}
336
346static inline void channel_config_set_bswap(dma_channel_config_t *c, bool bswap) {
347 c->ctrl = bswap ? (c->ctrl | DMA_CH0_CTRL_TRIG_BSWAP_BITS) : (c->ctrl & ~DMA_CH0_CTRL_TRIG_BSWAP_BITS);
348}
349
360static inline void channel_config_set_irq_quiet(dma_channel_config_t *c, bool irq_quiet) {
361 c->ctrl = irq_quiet ? (c->ctrl | DMA_CH0_CTRL_TRIG_IRQ_QUIET_BITS) : (c->ctrl & ~DMA_CH0_CTRL_TRIG_IRQ_QUIET_BITS);
362}
363
378static inline void channel_config_set_high_priority(dma_channel_config_t *c, bool high_priority) {
379 c->ctrl = high_priority ? (c->ctrl | DMA_CH0_CTRL_TRIG_HIGH_PRIORITY_BITS) : (c->ctrl & ~DMA_CH0_CTRL_TRIG_HIGH_PRIORITY_BITS);
380}
381
393static inline void channel_config_set_enable(dma_channel_config_t *c, bool enable) {
394 c->ctrl = enable ? (c->ctrl | DMA_CH0_CTRL_TRIG_EN_BITS) : (c->ctrl & ~DMA_CH0_CTRL_TRIG_EN_BITS);
395}
396
405static inline void channel_config_set_sniff_enable(dma_channel_config_t *c, bool sniff_enable) {
406 c->ctrl = sniff_enable ? (c->ctrl | DMA_CH0_CTRL_TRIG_SNIFF_EN_BITS) : (c->ctrl &
407 ~DMA_CH0_CTRL_TRIG_SNIFF_EN_BITS);
408}
409
445
452static inline dma_channel_config_t dma_get_channel_config(uint channel) {
454 c.ctrl = dma_channel_hw_addr(channel)->ctrl_trig;
455 return c;
456}
457
464static inline uint32_t channel_config_get_ctrl_value(const dma_channel_config_t *config) {
465 return config->ctrl;
466}
467
475static inline void dma_channel_set_config(uint channel, const dma_channel_config_t *config, bool trigger) {
476 // Don't use CTRL_TRIG since we don't want to start a transfer
477 if (!trigger) {
478 dma_channel_hw_addr(channel)->al1_ctrl = channel_config_get_ctrl_value(config);
479 } else {
480 dma_channel_hw_addr(channel)->ctrl_trig = channel_config_get_ctrl_value(config);
481 }
482}
483
491static inline void dma_channel_set_read_addr(uint channel, const volatile void *read_addr, bool trigger) {
492 if (!trigger) {
493 dma_channel_hw_addr(channel)->read_addr = (uintptr_t) read_addr;
494 } else {
495 dma_channel_hw_addr(channel)->al3_read_addr_trig = (uintptr_t) read_addr;
496 }
497}
498
506static inline void dma_channel_set_write_addr(uint channel, volatile void *write_addr, bool trigger) {
507 if (!trigger) {
508 dma_channel_hw_addr(channel)->write_addr = (uintptr_t) write_addr;
509 } else {
510 dma_channel_hw_addr(channel)->al2_write_addr_trig = (uintptr_t) write_addr;
511 }
512}
513
531static inline uint32_t dma_encode_transfer_count(uint transfer_count) {
532#if !PICO_RP2040
533 invalid_params_if(HARDWARE_DMA, transfer_count & DMA_CH0_TRANS_COUNT_MODE_BITS);
534 return transfer_count & DMA_CH0_TRANS_COUNT_COUNT_BITS;
535#else
536 return transfer_count;
537#endif
538}
539
557static inline uint32_t dma_encode_transfer_count_with_self_trigger(uint transfer_count) {
558#if PICO_RP2040
560#else
561 return dma_encode_transfer_count(transfer_count) | (DMA_CH0_TRANS_COUNT_MODE_VALUE_TRIGGER_SELF << DMA_CH0_TRANS_COUNT_MODE_LSB);
562#endif
563}
564
577static inline uint32_t dma_encode_endless_transfer_count(void) {
578#if PICO_RP2040
580#else
581 static_assert(DMA_CH0_TRANS_COUNT_MODE_VALUE_ENDLESS == 0xf);
582 static_assert(DMA_CH0_TRANS_COUNT_MODE_LSB == 28);
583 return 0xffffffffu;
584#endif
585}
586
605static inline void dma_channel_set_transfer_count(uint channel, uint32_t encoded_transfer_count, bool trigger) {
606 if (!trigger) {
607 dma_channel_hw_addr(channel)->transfer_count = encoded_transfer_count;
608 } else {
609 dma_channel_hw_addr(channel)->al1_transfer_count_trig = encoded_transfer_count;
610 }
611}
612
613// backwards compatibility with SDK < 2.2.0
614static inline void dma_channel_set_trans_count(uint channel, uint32_t trans_count, bool trigger) {
615 dma_channel_set_transfer_count(channel, trans_count, trigger);
616}
617
639static inline void dma_channel_configure(uint channel, const dma_channel_config_t *config, volatile void *write_addr,
640 const volatile void *read_addr,
641 uint32_t encoded_transfer_count, bool trigger) {
642 dma_channel_set_read_addr(channel, read_addr, false);
643 dma_channel_set_write_addr(channel, write_addr, false);
644 dma_channel_set_transfer_count(channel, encoded_transfer_count, false);
645 dma_channel_set_config(channel, config, trigger);
646}
647
666inline static void __attribute__((always_inline)) dma_channel_transfer_from_buffer_now(uint channel,
667 const volatile void *read_addr,
668 uint32_t encoded_transfer_count) {
669// check_dma_channel_param(channel);
670 dma_channel_hw_t *hw = dma_channel_hw_addr(channel);
671 hw->read_addr = (uintptr_t) read_addr;
672 hw->al1_transfer_count_trig = encoded_transfer_count;
673}
674
693inline static void dma_channel_transfer_to_buffer_now(uint channel, volatile void *write_addr, uint32_t encoded_transfer_count) {
694 dma_channel_hw_t *hw = dma_channel_hw_addr(channel);
695 hw->write_addr = (uintptr_t) write_addr;
696 hw->al1_transfer_count_trig = encoded_transfer_count;
697}
698
704static inline void dma_start_channel_mask(uint32_t chan_mask) {
705 valid_params_if(HARDWARE_DMA, chan_mask && chan_mask < (1u << NUM_DMA_CHANNELS));
706 dma_hw->multi_channel_trigger = chan_mask;
707}
708
714static inline void dma_channel_start(uint channel) {
715 dma_start_channel_mask(1u << channel);
716}
717
758static inline void dma_channel_abort(uint channel) {
759 check_dma_channel_param(channel);
760 dma_hw->abort = 1u << channel;
761 // Bit will go 0 once channel has reached safe state
762 // (i.e. any in-flight transfers have retired)
763 while (dma_hw->ch[channel].ctrl_trig & DMA_CH0_CTRL_TRIG_BUSY_BITS) tight_loop_contents();
764}
765
772static inline void dma_channel_set_irq0_enabled(uint channel, bool enabled) {
773 check_dma_channel_param(channel);
774 check_hw_layout(dma_hw_t, inte0, DMA_INTE0_OFFSET);
775 if (enabled)
776 hw_set_bits(&dma_hw->inte0, 1u << channel);
777 else
778 hw_clear_bits(&dma_hw->inte0, 1u << channel);
779}
780
787static inline void dma_set_irq0_channel_mask_enabled(uint32_t channel_mask, bool enabled) {
788 if (enabled) {
789 hw_set_bits(&dma_hw->inte0, channel_mask);
790 } else {
791 hw_clear_bits(&dma_hw->inte0, channel_mask);
792 }
793}
794
801static inline void dma_channel_set_irq1_enabled(uint channel, bool enabled) {
802 check_dma_channel_param(channel);
803 check_hw_layout(dma_hw_t, inte1, DMA_INTE1_OFFSET);
804 if (enabled)
805 hw_set_bits(&dma_hw->inte1, 1u << channel);
806 else
807 hw_clear_bits(&dma_hw->inte1, 1u << channel);
808}
809
816static inline void dma_set_irq1_channel_mask_enabled(uint32_t channel_mask, bool enabled) {
817 if (enabled) {
818 hw_set_bits(&dma_hw->inte1, channel_mask);
819 } else {
820 hw_clear_bits(&dma_hw->inte1, channel_mask);
821 }
822}
823
831static inline void dma_irqn_set_channel_enabled(uint irq_index, uint channel, bool enabled) {
832 invalid_params_if(HARDWARE_DMA, irq_index >= NUM_DMA_IRQS);
833
834 if (enabled)
835 hw_set_bits(&dma_hw->irq_ctrl[irq_index].inte, 1u << channel);
836 else
837 hw_clear_bits(&dma_hw->irq_ctrl[irq_index].inte, 1u << channel);
838}
839
847static inline void dma_irqn_set_channel_mask_enabled(uint irq_index, uint32_t channel_mask, bool enabled) {
848 invalid_params_if(HARDWARE_DMA, irq_index >= NUM_DMA_IRQS);
849 if (enabled) {
850 hw_set_bits(&dma_hw->irq_ctrl[irq_index].inte, channel_mask);
851 } else {
852 hw_clear_bits(&dma_hw->irq_ctrl[irq_index].inte, channel_mask);
853 }
854}
855
862static inline bool dma_channel_get_irq0_status(uint channel) {
863 check_dma_channel_param(channel);
864 return dma_hw->ints0 & (1u << channel);
865}
866
873static inline bool dma_channel_get_irq1_status(uint channel) {
874 check_dma_channel_param(channel);
875 return dma_hw->ints1 & (1u << channel);
876}
877
885static inline bool dma_irqn_get_channel_status(uint irq_index, uint channel) {
886 invalid_params_if(HARDWARE_DMA, irq_index >= NUM_DMA_IRQS);
887 check_dma_channel_param(channel);
888 return dma_hw->irq_ctrl[irq_index].ints & (1u << channel);
889}
890
896static inline void dma_channel_acknowledge_irq0(uint channel) {
897 check_dma_channel_param(channel);
898 dma_hw->ints0 = 1u << channel;
899}
900
906static inline void dma_channel_acknowledge_irq1(uint channel) {
907 check_dma_channel_param(channel);
908 dma_hw->ints1 = 1u << channel;
909}
910
917static inline void dma_irqn_acknowledge_channel(uint irq_index, uint channel) {
918 invalid_params_if(HARDWARE_DMA, irq_index >= NUM_DMA_IRQS);
919 check_dma_channel_param(channel);
920 dma_hw->irq_ctrl[irq_index].ints = 1u << channel;
921}
922
929inline static bool dma_channel_is_busy(uint channel) {
930 check_dma_channel_param(channel);
931 return dma_hw->ch[channel].al1_ctrl & DMA_CH0_CTRL_TRIG_BUSY_BITS;
932}
933
939inline static void dma_channel_wait_for_finish_blocking(uint channel) {
940 while (dma_channel_is_busy(channel)) tight_loop_contents();
941 // stop the compiler hoisting a non-volatile buffer access above the DMA completion.
943}
944
965inline static void dma_sniffer_enable(uint channel, uint mode, bool force_channel_enable) {
966 check_dma_channel_param(channel);
967 check_hw_layout(dma_hw_t, sniff_ctrl, DMA_SNIFF_CTRL_OFFSET);
968 if (force_channel_enable) {
969 hw_set_bits(&dma_hw->ch[channel].al1_ctrl, DMA_CH0_CTRL_TRIG_SNIFF_EN_BITS);
970 }
971 hw_write_masked(&dma_hw->sniff_ctrl,
972 (((channel << DMA_SNIFF_CTRL_DMACH_LSB) & DMA_SNIFF_CTRL_DMACH_BITS) |
973 ((mode << DMA_SNIFF_CTRL_CALC_LSB) & DMA_SNIFF_CTRL_CALC_BITS) |
974 DMA_SNIFF_CTRL_EN_BITS),
975 (DMA_SNIFF_CTRL_DMACH_BITS |
976 DMA_SNIFF_CTRL_CALC_BITS |
977 DMA_SNIFF_CTRL_EN_BITS));
978}
979
991inline static void dma_sniffer_set_byte_swap_enabled(bool swap) {
992 if (swap)
993 hw_set_bits(&dma_hw->sniff_ctrl, DMA_SNIFF_CTRL_BSWAP_BITS);
994 else
995 hw_clear_bits(&dma_hw->sniff_ctrl, DMA_SNIFF_CTRL_BSWAP_BITS);
996}
997
1006inline static void dma_sniffer_set_output_invert_enabled(bool invert) {
1007 if (invert)
1008 hw_set_bits(&dma_hw->sniff_ctrl, DMA_SNIFF_CTRL_OUT_INV_BITS);
1009 else
1010 hw_clear_bits(&dma_hw->sniff_ctrl, DMA_SNIFF_CTRL_OUT_INV_BITS);
1011}
1012
1021inline static void dma_sniffer_set_output_reverse_enabled(bool reverse) {
1022 if (reverse)
1023 hw_set_bits(&dma_hw->sniff_ctrl, DMA_SNIFF_CTRL_OUT_REV_BITS);
1024 else
1025 hw_clear_bits(&dma_hw->sniff_ctrl, DMA_SNIFF_CTRL_OUT_REV_BITS);
1026}
1027
1032inline static void dma_sniffer_disable(void) {
1033 dma_hw->sniff_ctrl = 0;
1034}
1035
1044inline static void dma_sniffer_set_data_accumulator(uint32_t seed_value) {
1045 dma_hw->sniff_data = seed_value;
1046}
1047
1053inline static uint32_t dma_sniffer_get_data_accumulator(void) {
1054 return dma_hw->sniff_data;
1055}
1056
1066void dma_timer_claim(uint timer);
1067
1075void dma_timer_unclaim(uint timer);
1076
1083int dma_claim_unused_timer(bool required);
1084
1092bool dma_timer_is_claimed(uint timer);
1093
1105static inline void dma_timer_set_fraction(uint timer, uint16_t numerator, uint16_t denominator) {
1106 check_dma_timer_param(timer);
1107 invalid_params_if(HARDWARE_DMA, numerator > denominator);
1108 dma_hw->timer[timer] = (((uint32_t)numerator) << DMA_TIMER0_X_LSB) | (((uint32_t)denominator) << DMA_TIMER0_Y_LSB);
1109}
1110
1116static inline uint dma_get_timer_dreq(uint timer_num) {
1117 static_assert(DREQ_DMA_TIMER1 == DREQ_DMA_TIMER0 + 1, "");
1118 static_assert(DREQ_DMA_TIMER2 == DREQ_DMA_TIMER0 + 2, "");
1119 static_assert(DREQ_DMA_TIMER3 == DREQ_DMA_TIMER0 + 3, "");
1120 check_dma_timer_param(timer_num);
1121 return DREQ_DMA_TIMER0 + timer_num;
1122}
1123
1130static inline int dma_get_irq_num(uint irq_index) {
1131 valid_params_if(HARDWARE_DMA, irq_index < NUM_DMA_IRQS);
1132 return DMA_IRQ_NUM(irq_index);
1133}
1134
1152void dma_channel_cleanup(uint channel);
1153
1154#ifndef NDEBUG
1155void print_dma_ctrl(dma_channel_hw_t *channel);
1156#endif
1157
1158#ifdef __cplusplus
1159}
1160#endif
1161
1162#endif
static void channel_config_set_enable(dma_channel_config_t *c, bool enable)
Enable/Disable the DMA channel in a channel configuration object.
Definition dma.h:393
static void channel_config_set_transfer_data_size(dma_channel_config_t *c, dma_channel_transfer_size_t size)
Set the size of each DMA bus transfer in a channel configuration object.
Definition dma.h:310
static void channel_config_set_bswap(dma_channel_config_t *c, bool bswap)
Set DMA byte swapping config in a channel configuration object.
Definition dma.h:346
static void channel_config_set_sniff_enable(dma_channel_config_t *c, bool sniff_enable)
Enable access to channel by sniff hardware in a channel configuration object.
Definition dma.h:405
static void channel_config_set_high_priority(dma_channel_config_t *c, bool high_priority)
Set the channel priority in a channel configuration object.
Definition dma.h:378
static void channel_config_set_read_increment(dma_channel_config_t *c, bool incr)
Set DMA channel read increment in a channel configuration object.
Definition dma.h:246
static dma_channel_config_t dma_channel_get_default_config(uint channel)
Get the default channel configuration for a given channel.
Definition dma.h:430
static void channel_config_set_read_address_update_type(dma_channel_config_t *c, dma_address_update_type_t update_type)
Set DMA channel read address update type in a channel configuration object.
Definition dma.h:215
static void channel_config_set_write_address_update_type(dma_channel_config_t *c, dma_address_update_type_t update_type)
Set DMA channel write address update type in a channel configuration object.
Definition dma.h:228
static void channel_config_set_write_increment(dma_channel_config_t *c, bool incr)
Set DMA channel write increment in a channel configuration object.
Definition dma.h:263
static uint32_t channel_config_get_ctrl_value(const dma_channel_config_t *config)
Get the raw configuration register from a channel configuration.
Definition dma.h:464
static void channel_config_set_dreq(dma_channel_config_t *c, uint dreq)
Select a transfer request signal in a channel configuration object.
Definition dma.h:282
static void channel_config_set_chain_to(dma_channel_config_t *c, uint chain_to)
Set DMA channel chain_to channel in a channel configuration object.
Definition dma.h:296
static void channel_config_set_irq_quiet(dma_channel_config_t *c, bool irq_quiet)
Set IRQ quiet mode in a channel configuration object.
Definition dma.h:360
static dma_channel_config_t dma_get_channel_config(uint channel)
Get the current configuration for the specified channel.
Definition dma.h:452
static void channel_config_set_ring(dma_channel_config_t *c, bool write, uint size_bits)
Set address wrapping parameters in a channel configuration object.
Definition dma.h:330
static __force_inline void hw_set_bits(io_rw_32 *addr, uint32_t mask)
Atomically set the specified bits to 1 in a HW register.
Definition address_mapped.h:135
static __force_inline void hw_write_masked(io_rw_32 *addr, uint32_t values, uint32_t write_mask)
Set new values for a sub-set of the bits in a HW register.
Definition address_mapped.h:171
static __force_inline void hw_clear_bits(io_rw_32 *addr, uint32_t mask)
Atomically clear the specified bits to 0 in a HW register.
Definition address_mapped.h:145
void dma_unclaim_mask(uint32_t channel_mask)
Mark multiple dma channels as no longer used.
Definition dma.c:39
static void dma_sniffer_enable(uint channel, uint mode, bool force_channel_enable)
Enable the DMA sniffing targeting the specified channel.
Definition dma.h:965
void dma_timer_claim(uint timer)
Mark a dma timer as used.
Definition dma.c:54
dma_address_update_type
Enumeration of types of updates that can be made to the DMA read or write address after each transfer...
Definition dma.h:152
static void dma_channel_set_transfer_count(uint channel, uint32_t encoded_transfer_count, bool trigger)
Set the number of bus transfers the channel will do.
Definition dma.h:605
static void dma_sniffer_set_output_invert_enabled(bool invert)
Enable the Sniffer output invert function.
Definition dma.h:1006
enum dma_address_update_type dma_address_update_type_t
Enumeration of types of updates that can be made to the DMA read or write address after each transfer...
int dma_claim_unused_timer(bool required)
Claim a free dma timer.
Definition dma.c:64
static uint32_t dma_sniffer_get_data_accumulator(void)
Get the sniffer's data accumulator value.
Definition dma.h:1053
static void dma_channel_start(uint channel)
Start a single DMA channel.
Definition dma.h:714
static void dma_channel_set_config(uint channel, const dma_channel_config_t *config, bool trigger)
Set a channel configuration.
Definition dma.h:475
static bool dma_irqn_get_channel_status(uint irq_index, uint channel)
Determine if a particular channel is a cause of DMA_IRQ_N.
Definition dma.h:885
int dma_claim_unused_channel(bool required)
Claim a free dma channel.
Definition dma.c:45
static void dma_channel_acknowledge_irq1(uint channel)
Acknowledge a channel IRQ, resetting it as the cause of DMA_IRQ_1.
Definition dma.h:906
static void dma_channel_set_irq1_enabled(uint channel, bool enabled)
Enable single DMA channel's interrupt via DMA_IRQ_1.
Definition dma.h:801
static void dma_timer_set_fraction(uint timer, uint16_t numerator, uint16_t denominator)
Set the multiplier for the given DMA timer.
Definition dma.h:1105
static uint32_t dma_encode_transfer_count_with_self_trigger(uint transfer_count)
Encode the specified transfer length, along with a flag to indicate the DMA transfer should be self-t...
Definition dma.h:557
static int dma_get_irq_num(uint irq_index)
Return DMA_IRQ_<irqn>.
Definition dma.h:1130
static void dma_start_channel_mask(uint32_t chan_mask)
Start one or more channels simultaneously.
Definition dma.h:704
void dma_channel_cleanup(uint channel)
Performs DMA channel cleanup after use.
Definition dma.c:73
#define DMA_IRQ_NUM(irq_index)
Returns the irq_num_t for the nth DMA interrupt.
Definition dma.h:56
static void dma_irqn_set_channel_enabled(uint irq_index, uint channel, bool enabled)
Enable single DMA channel interrupt on either DMA_IRQ_0 or DMA_IRQ_1.
Definition dma.h:831
static void dma_irqn_set_channel_mask_enabled(uint irq_index, uint32_t channel_mask, bool enabled)
Enable multiple DMA channels' interrupt via either DMA_IRQ_0 or DMA_IRQ_1.
Definition dma.h:847
static void dma_channel_abort(uint channel)
Stop a DMA transfer.
Definition dma.h:758
static uint dma_get_timer_dreq(uint timer_num)
Return the DREQ number for a given DMA timer.
Definition dma.h:1116
void dma_timer_unclaim(uint timer)
Mark a dma timer as no longer used.
Definition dma.c:59
static uint32_t dma_encode_endless_transfer_count(void)
Return an endless transfer as an "encoded_transfer_length" value suitable for the referenced methods.
Definition dma.h:577
static uint32_t dma_encode_transfer_count(uint transfer_count)
Encode the specified transfer length into an "encoded_transfer_length" value suitable for the referen...
Definition dma.h:531
bool dma_channel_is_claimed(uint channel)
Determine if a dma channel is claimed.
Definition dma.c:49
bool dma_timer_is_claimed(uint timer)
Determine if a dma timer is claimed.
Definition dma.c:68
static void dma_sniffer_disable(void)
Disable the DMA sniffer.
Definition dma.h:1032
static void dma_sniffer_set_output_reverse_enabled(bool reverse)
Enable the Sniffer output bit reversal function.
Definition dma.h:1021
void dma_claim_mask(uint32_t channel_mask)
Mark multiple dma channels as used.
Definition dma.c:28
static bool dma_channel_get_irq0_status(uint channel)
Determine if a particular channel is a cause of DMA_IRQ_0.
Definition dma.h:862
static void dma_set_irq1_channel_mask_enabled(uint32_t channel_mask, bool enabled)
Enable multiple DMA channels' interrupts via DMA_IRQ_1.
Definition dma.h:816
static void dma_channel_wait_for_finish_blocking(uint channel)
Wait for a DMA channel transfer to complete.
Definition dma.h:939
static void dma_channel_set_read_addr(uint channel, const volatile void *read_addr, bool trigger)
Set the DMA initial read address.
Definition dma.h:491
static void dma_channel_transfer_to_buffer_now(uint channel, volatile void *write_addr, uint32_t encoded_transfer_count)
Start a DMA transfer to a buffer immediately.
Definition dma.h:693
static void dma_sniffer_set_byte_swap_enabled(bool swap)
Enable the Sniffer byte swap function.
Definition dma.h:991
static void dma_channel_transfer_from_buffer_now(uint channel, const volatile void *read_addr, uint32_t encoded_transfer_count)
Start a DMA transfer from a buffer immediately.
Definition dma.h:666
void dma_channel_unclaim(uint channel)
Mark a dma channel as no longer used.
Definition dma.c:34
static void dma_sniffer_set_data_accumulator(uint32_t seed_value)
Set the sniffer's data accumulator with initial value.
Definition dma.h:1044
dma_channel_transfer_size
Enumeration of available DMA channel transfer sizes.
Definition dma.h:143
static void dma_irqn_acknowledge_channel(uint irq_index, uint channel)
Acknowledge a channel IRQ, resetting it as the cause of DMA_IRQ_N.
Definition dma.h:917
static void dma_set_irq0_channel_mask_enabled(uint32_t channel_mask, bool enabled)
Enable multiple DMA channels' interrupts via DMA_IRQ_0.
Definition dma.h:787
void dma_channel_claim(uint channel)
Mark a dma channel as used.
Definition dma.c:23
enum dma_channel_transfer_size dma_channel_transfer_size_t
Enumeration of available DMA channel transfer sizes.
static void dma_channel_configure(uint channel, const dma_channel_config_t *config, volatile void *write_addr, const volatile void *read_addr, uint32_t encoded_transfer_count, bool trigger)
Configure all DMA parameters and optionally start transfer.
Definition dma.h:639
static void dma_channel_set_write_addr(uint channel, volatile void *write_addr, bool trigger)
Set the DMA initial write address.
Definition dma.h:506
static void dma_channel_set_irq0_enabled(uint channel, bool enabled)
Enable single DMA channel's interrupt via DMA_IRQ_0.
Definition dma.h:772
static bool dma_channel_get_irq1_status(uint channel)
Determine if a particular channel is a cause of DMA_IRQ_1.
Definition dma.h:873
static bool dma_channel_is_busy(uint channel)
Check if DMA channel is busy.
Definition dma.h:929
static void dma_channel_acknowledge_irq0(uint channel)
Acknowledge a channel IRQ, resetting it as the cause of DMA_IRQ_0.
Definition dma.h:896
@ DMA_ADDRESS_UPDATE_NONE
The address remains the same after each transfer.
Definition dma.h:153
@ DMA_ADDRESS_UPDATE_INCREMENT
The address is incremented by the transfer size after each transfer.
Definition dma.h:154
@ DREQ_DMA_TIMER1
Select DMA_TIMER0 as DREQ.
Definition dreq.h:108
@ DREQ_DMA_TIMER0
Select DMA_TIMER0 as DREQ.
Definition dreq.h:107
@ DREQ_DMA_TIMER3
Select DMA_TIMER3 as DREQ.
Definition dreq.h:110
@ DREQ_DMA_TIMER2
Select DMA_TIMER1 as DREQ.
Definition dreq.h:109
@ DREQ_FORCE
Select FORCE as DREQ.
Definition dreq.h:111
@ DMA_SIZE_16
Half word transfer (16 bits).
Definition dma.h:145
@ DMA_SIZE_8
Byte transfer (8 bits).
Definition dma.h:144
@ DMA_SIZE_32
Word transfer (32 bits).
Definition dma.h:146
void panic_unsupported(void)
Panics with the message "Unsupported".
Definition panic.c:20
static __force_inline void tight_loop_contents(void)
No-op function for the body of tight loops.
Definition common.h:74
static __always_inline void __compiler_memory_barrier(void)
Ensure that the compiler does not move memory access across this method call.
Definition compiler.h:176
Opaque representation of a DMA channel configuration that can be later applied to a hardware DMA chan...
Definition dma.h:164
Definition dma.h:27
Definition dma.h:146