| /* |
| * Copyright (c) 2022 Intel Corporation. |
| * |
| * SPDX-License-Identifier: Apache-2.0 |
| */ |
| |
| #ifndef ZEPHYR_DRIVERS_DMA_DMA_DW_COMMON_H_ |
| #define ZEPHYR_DRIVERS_DMA_DMA_DW_COMMON_H_ |
| |
| #include <zephyr/sys/atomic.h> |
| #include <zephyr/drivers/dma.h> |
| |
| #ifdef __cplusplus |
| extern "C" { |
| #endif |
| |
| #define MASK(b_hi, b_lo) \ |
| (((1ULL << ((b_hi) - (b_lo) + 1ULL)) - 1ULL) << (b_lo)) |
| #define SET_BIT(b, x) (((x) & 1) << (b)) |
| #define SET_BITS(b_hi, b_lo, x) \ |
| (((x) & ((1ULL << ((b_hi) - (b_lo) + 1ULL)) - 1ULL)) << (b_lo)) |
| |
| #define DW_MAX_CHAN 8 |
| #define DW_CH_SIZE 0x58 |
| #define DW_CHAN_OFFSET(chan) (DW_CH_SIZE * chan) |
| |
| #define DW_SAR(chan) \ |
| (0x0000 + DW_CHAN_OFFSET(chan)) |
| #define DW_DAR(chan) \ |
| (0x0008 + DW_CHAN_OFFSET(chan)) |
| #define DW_LLP(chan) \ |
| (0x0010 + DW_CHAN_OFFSET(chan)) |
| #define DW_CTRL_LOW(chan) \ |
| (0x0018 + DW_CHAN_OFFSET(chan)) |
| #define DW_CTRL_HIGH(chan) \ |
| (0x001C + DW_CHAN_OFFSET(chan)) |
| #define DW_CFG_LOW(chan) \ |
| (0x0040 + DW_CHAN_OFFSET(chan)) |
| #define DW_CFG_HIGH(chan) \ |
| (0x0044 + DW_CHAN_OFFSET(chan)) |
| #define DW_DSR(chan) \ |
| (0x0050 + DW_CHAN_OFFSET(chan)) |
| |
| /* registers */ |
| #define DW_RAW_TFR 0x02C0 |
| #define DW_RAW_BLOCK 0x02C8 |
| #define DW_RAW_SRC_TRAN 0x02D0 |
| #define DW_RAW_DST_TRAN 0x02D8 |
| #define DW_RAW_ERR 0x02E0 |
| #define DW_STATUS_TFR 0x02E8 |
| #define DW_STATUS_BLOCK 0x02F0 |
| #define DW_STATUS_SRC_TRAN 0x02F8 |
| #define DW_STATUS_DST_TRAN 0x0300 |
| #define DW_STATUS_ERR 0x0308 |
| #define DW_MASK_TFR 0x0310 |
| #define DW_MASK_BLOCK 0x0318 |
| #define DW_MASK_SRC_TRAN 0x0320 |
| #define DW_MASK_DST_TRAN 0x0328 |
| #define DW_MASK_ERR 0x0330 |
| #define DW_CLEAR_TFR 0x0338 |
| #define DW_CLEAR_BLOCK 0x0340 |
| #define DW_CLEAR_SRC_TRAN 0x0348 |
| #define DW_CLEAR_DST_TRAN 0x0350 |
| #define DW_CLEAR_ERR 0x0358 |
| #define DW_INTR_STATUS 0x0360 |
| #define DW_DMA_CFG 0x0398 |
| #define DW_DMA_CHAN_EN 0x03A0 |
| #define DW_FIFO_PART0_LO 0x400 |
| #define DW_FIFO_PART0_HI 0x404 |
| #define DW_FIFO_PART1_LO 0x408 |
| #define DW_FIFO_PART1_HI 0x40C |
| |
| /* channel bits */ |
| #define DW_CHAN_WRITE_EN_ALL MASK(2 * DW_MAX_CHAN - 1, DW_MAX_CHAN) |
| #define DW_CHAN_WRITE_EN(chan) BIT((chan) + DW_MAX_CHAN) |
| #define DW_CHAN_ALL MASK(DW_MAX_CHAN - 1, 0) |
| #define DW_CHAN(chan) BIT(chan) |
| #define DW_CHAN_MASK_ALL DW_CHAN_WRITE_EN_ALL |
| #define DW_CHAN_MASK(chan) DW_CHAN_WRITE_EN(chan) |
| #define DW_CHAN_UNMASK_ALL (DW_CHAN_WRITE_EN_ALL | DW_CHAN_ALL) |
| #define DW_CHAN_UNMASK(chan) (DW_CHAN_WRITE_EN(chan) | DW_CHAN(chan)) |
| |
| /* CFG_LO */ |
| #define DW_CFGL_RELOAD_DST BIT(31) |
| #define DW_CFGL_RELOAD_SRC BIT(30) |
| #define DW_CFGL_DRAIN BIT(10) |
| #define DW_CFGL_FIFO_EMPTY BIT(9) |
| #define DW_CFGL_SUSPEND BIT(8) |
| #define DW_CFGL_CTL_HI_UPD_EN BIT(5) |
| |
| /* CFG_HI */ |
| #define DW_CFGH_DST_PER_EXT(x) SET_BITS(31, 30, x) |
| #define DW_CFGH_SRC_PER_EXT(x) SET_BITS(29, 28, x) |
| #define DW_CFGH_DST_PER(x) SET_BITS(7, 4, x) |
| #define DW_CFGH_SRC_PER(x) SET_BITS(3, 0, x) |
| #define DW_CFGH_DST(x) \ |
| (DW_CFGH_DST_PER_EXT((x) >> 4) | DW_CFGH_DST_PER(x)) |
| #define DW_CFGH_SRC(x) \ |
| (DW_CFGH_SRC_PER_EXT((x) >> 4) | DW_CFGH_SRC_PER(x)) |
| |
| /* CTL_LO */ |
| #define DW_CTLL_RELOAD_DST BIT(31) |
| #define DW_CTLL_RELOAD_SRC BIT(30) |
| #define DW_CTLL_LLP_S_EN BIT(28) |
| #define DW_CTLL_LLP_D_EN BIT(27) |
| #define DW_CTLL_SMS(x) SET_BIT(25, x) |
| #define DW_CTLL_DMS(x) SET_BIT(23, x) |
| #define DW_CTLL_FC_P2P SET_BITS(21, 20, 3) |
| #define DW_CTLL_FC_P2M SET_BITS(21, 20, 2) |
| #define DW_CTLL_FC_M2P SET_BITS(21, 20, 1) |
| #define DW_CTLL_FC_M2M SET_BITS(21, 20, 0) |
| #define DW_CTLL_D_SCAT_EN BIT(18) |
| #define DW_CTLL_S_GATH_EN BIT(17) |
| #define DW_CTLL_SRC_MSIZE(x) SET_BITS(16, 14, x) |
| #define DW_CTLL_DST_MSIZE(x) SET_BITS(13, 11, x) |
| #define DW_CTLL_SRC_FIX SET_BITS(10, 9, 2) |
| #define DW_CTLL_SRC_DEC SET_BITS(10, 9, 1) |
| #define DW_CTLL_SRC_INC SET_BITS(10, 9, 0) |
| #define DW_CTLL_DST_FIX SET_BITS(8, 7, 2) |
| #define DW_CTLL_DST_DEC SET_BITS(8, 7, 1) |
| #define DW_CTLL_DST_INC SET_BITS(8, 7, 0) |
| #define DW_CTLL_SRC_WIDTH(x) SET_BITS(6, 4, x) |
| #define DW_CTLL_DST_WIDTH(x) SET_BITS(3, 1, x) |
| #define DW_CTLL_INT_EN BIT(0) |
| #define DW_CTLL_SRC_WIDTH_MASK MASK(6, 4) |
| #define DW_CTLL_SRC_WIDTH_SHIFT 4 |
| #define DW_CTLL_DST_WIDTH_MASK MASK(3, 1) |
| #define DW_CTLL_DST_WIDTH_SHIFT 1 |
| |
| /* CTL_HI */ |
| #define DW_CTLH_CLASS(x) SET_BITS(31, 29, x) |
| #define DW_CTLH_WEIGHT(x) SET_BITS(28, 18, x) |
| #define DW_CTLH_DONE(x) SET_BIT(17, x) |
| #define DW_CTLH_BLOCK_TS_MASK MASK(16, 0) |
| |
| /* DSR */ |
| #define DW_DSR_DSC(x) SET_BITS(31, 20, x) |
| #define DW_DSR_DSI(x) SET_BITS(19, 0, x) |
| |
| /* FIFO_PART */ |
| #define DW_FIFO_SIZE 0x80 |
| #define DW_FIFO_UPD BIT(26) |
| #define DW_FIFO_CHx(x) SET_BITS(25, 13, x) |
| #define DW_FIFO_CHy(x) SET_BITS(12, 0, x) |
| |
| /* number of tries to wait for reset */ |
| #define DW_DMA_CFG_TRIES 10000 |
| |
| /* channel drain timeout in microseconds */ |
| #define DW_DMA_TIMEOUT 1333 |
| |
| /* min number of elems for config with irq disabled */ |
| #define DW_DMA_CFG_NO_IRQ_MIN_ELEMS 3 |
| |
| /* linked list item address */ |
| #define DW_DMA_LLI_ADDRESS(lli, dir) \ |
| (((dir) == MEMORY_TO_PERIPHERAL) ? ((lli)->sar) : ((lli)->dar)) |
| |
| /* TODO: add FIFO sizes */ |
| struct dw_chan_arbit_data { |
| uint16_t class; |
| uint16_t weight; |
| }; |
| |
| struct dw_drv_plat_data { |
| struct dw_chan_arbit_data chan[DW_MAX_CHAN]; |
| }; |
| |
| /* DMA descriptor used by HW */ |
| struct dw_lli { |
| uint32_t sar; |
| uint32_t dar; |
| uint32_t llp; |
| uint32_t ctrl_lo; |
| uint32_t ctrl_hi; |
| uint32_t sstat; |
| uint32_t dstat; |
| |
| /* align to 32 bytes to not cross cache line |
| * in case of more than two items |
| */ |
| uint32_t reserved; |
| } __packed; |
| |
| /* pointer data for DW DMA buffer */ |
| struct dw_dma_ptr_data { |
| uint32_t current_ptr; |
| uint32_t start_ptr; |
| uint32_t end_ptr; |
| uint32_t hw_ptr; |
| uint32_t buffer_bytes; |
| }; |
| |
| /* State tracking for each channel */ |
| enum dw_dma_state { |
| DW_DMA_IDLE, |
| DW_DMA_PREPARED, |
| DW_DMA_SUSPENDED, |
| DW_DMA_ACTIVE, |
| }; |
| |
| /* data for each DMA channel */ |
| struct dw_dma_chan_data { |
| uint32_t direction; |
| enum dw_dma_state state; |
| struct dw_lli *lli; /* allocated array of LLI's */ |
| uint32_t lli_count; /* number of lli's in the allocation */ |
| struct dw_lli *lli_current; /* current LLI being used */ |
| uint32_t cfg_lo; |
| uint32_t cfg_hi; |
| struct dw_dma_ptr_data ptr_data; /* pointer data */ |
| dma_callback_t dma_blkcallback; |
| void *blkuser_data; |
| dma_callback_t dma_tfrcallback; |
| void *tfruser_data; |
| }; |
| |
| /* use array to get burst_elems for specific slot number setting. |
| * the relation between msize and burst_elems should be |
| * 2 ^ msize = burst_elems |
| */ |
| static const uint32_t burst_elems[] = {1, 2, 4, 8}; |
| |
| #if CONFIG_DMA_HW_LLI |
| #define DW_DMA_BUFFER_PERIOD_COUNT 4 |
| #else |
| #define DW_DMA_BUFFER_PERIOD_COUNT 2 |
| #endif |
| |
| /* Device run time data */ |
| struct dw_dma_dev_data { |
| struct dma_context dma_ctx; |
| struct dw_drv_plat_data *channel_data; |
| struct dw_dma_chan_data chan[DW_MAX_CHAN]; |
| struct dw_lli lli_pool[DW_MAX_CHAN][CONFIG_DMA_DW_LLI_POOL_SIZE]; |
| |
| ATOMIC_DEFINE(channels_atomic, DW_MAX_CHAN); |
| }; |
| |
| /* Device constant configuration parameters */ |
| struct dw_dma_dev_cfg { |
| uint32_t base; |
| void (*irq_config)(void); |
| }; |
| |
| static ALWAYS_INLINE void dw_write(uint32_t dma_base, uint32_t reg, uint32_t value) |
| { |
| *((volatile uint32_t *)(dma_base + reg)) = value; |
| } |
| |
| static ALWAYS_INLINE uint32_t dw_read(uint32_t dma_base, uint32_t reg) |
| { |
| return *((volatile uint32_t *)(dma_base + reg)); |
| } |
| |
| int dw_dma_setup(const struct device *dev); |
| |
| int dw_dma_config(const struct device *dev, uint32_t channel, |
| struct dma_config *cfg); |
| |
| int dw_dma_reload(const struct device *dev, uint32_t channel, |
| uint32_t src, uint32_t dst, size_t size); |
| |
| int dw_dma_start(const struct device *dev, uint32_t channel); |
| |
| int dw_dma_stop(const struct device *dev, uint32_t channel); |
| |
| int dw_dma_suspend(const struct device *dev, uint32_t channel); |
| |
| int dw_dma_resume(const struct device *dev, uint32_t channel); |
| |
| void dw_dma_isr(const struct device *dev); |
| |
| int dw_dma_get_status(const struct device *dev, uint32_t channel, |
| struct dma_status *stat); |
| |
| #ifdef __cplusplus |
| } |
| #endif |
| |
| #endif /* ZEPHYR_DRIVERS_DMA_DMA_DW_COMMON_H_ */ |