1 /* 2 * Intel Code Loader DMA support 3 * 4 * Copyright (C) 2015, Intel Corporation. 5 * 6 * This program is free software; you can redistribute it and/or modify 7 * it under the terms of the GNU General Public License as version 2, as 8 * published by the Free Software Foundation. 9 * 10 * This program is distributed in the hope that it will be useful, but 11 * WITHOUT ANY WARRANTY; without even the implied warranty of 12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU 13 * General Public License for more details. 14 */ 15 16 #ifndef SKL_SST_CLDMA_H_ 17 #define SKL_SST_CLDMA_H_ 18 19 #define FW_CL_STREAM_NUMBER 0x1 20 21 #define DMA_ADDRESS_128_BITS_ALIGNMENT 7 22 #define BDL_ALIGN(x) (x >> DMA_ADDRESS_128_BITS_ALIGNMENT) 23 24 #define SKL_ADSPIC_CL_DMA 0x2 25 #define SKL_ADSPIS_CL_DMA 0x2 26 #define SKL_CL_DMA_SD_INT_DESC_ERR 0x10 /* Descriptor error interrupt */ 27 #define SKL_CL_DMA_SD_INT_FIFO_ERR 0x08 /* FIFO error interrupt */ 28 #define SKL_CL_DMA_SD_INT_COMPLETE 0x04 /* Buffer completion interrupt */ 29 30 /* Intel HD Audio Code Loader DMA Registers */ 31 32 #define HDA_ADSP_LOADER_BASE 0x80 33 34 /* Stream Registers */ 35 #define SKL_ADSP_REG_CL_SD_CTL (HDA_ADSP_LOADER_BASE + 0x00) 36 #define SKL_ADSP_REG_CL_SD_STS (HDA_ADSP_LOADER_BASE + 0x03) 37 #define SKL_ADSP_REG_CL_SD_LPIB (HDA_ADSP_LOADER_BASE + 0x04) 38 #define SKL_ADSP_REG_CL_SD_CBL (HDA_ADSP_LOADER_BASE + 0x08) 39 #define SKL_ADSP_REG_CL_SD_LVI (HDA_ADSP_LOADER_BASE + 0x0c) 40 #define SKL_ADSP_REG_CL_SD_FIFOW (HDA_ADSP_LOADER_BASE + 0x0e) 41 #define SKL_ADSP_REG_CL_SD_FIFOSIZE (HDA_ADSP_LOADER_BASE + 0x10) 42 #define SKL_ADSP_REG_CL_SD_FORMAT (HDA_ADSP_LOADER_BASE + 0x12) 43 #define SKL_ADSP_REG_CL_SD_FIFOL (HDA_ADSP_LOADER_BASE + 0x14) 44 #define SKL_ADSP_REG_CL_SD_BDLPL (HDA_ADSP_LOADER_BASE + 0x18) 45 #define SKL_ADSP_REG_CL_SD_BDLPU (HDA_ADSP_LOADER_BASE + 0x1c) 46 47 /* CL: Software Position Based FIFO Capability Registers */ 48 #define SKL_ADSP_REG_CL_SPBFIFO (HDA_ADSP_LOADER_BASE + 0x20) 49 #define SKL_ADSP_REG_CL_SPBFIFO_SPBFCH (SKL_ADSP_REG_CL_SPBFIFO + 0x0) 50 #define SKL_ADSP_REG_CL_SPBFIFO_SPBFCCTL (SKL_ADSP_REG_CL_SPBFIFO + 0x4) 51 #define SKL_ADSP_REG_CL_SPBFIFO_SPIB (SKL_ADSP_REG_CL_SPBFIFO + 0x8) 52 #define SKL_ADSP_REG_CL_SPBFIFO_MAXFIFOS (SKL_ADSP_REG_CL_SPBFIFO + 0xc) 53 54 /* CL: Stream Descriptor x Control */ 55 56 /* Stream Reset */ 57 #define CL_SD_CTL_SRST_SHIFT 0 58 #define CL_SD_CTL_SRST_MASK (1 << CL_SD_CTL_SRST_SHIFT) 59 #define CL_SD_CTL_SRST(x) \ 60 ((x << CL_SD_CTL_SRST_SHIFT) & CL_SD_CTL_SRST_MASK) 61 62 /* Stream Run */ 63 #define CL_SD_CTL_RUN_SHIFT 1 64 #define CL_SD_CTL_RUN_MASK (1 << CL_SD_CTL_RUN_SHIFT) 65 #define CL_SD_CTL_RUN(x) \ 66 ((x << CL_SD_CTL_RUN_SHIFT) & CL_SD_CTL_RUN_MASK) 67 68 /* Interrupt On Completion Enable */ 69 #define CL_SD_CTL_IOCE_SHIFT 2 70 #define CL_SD_CTL_IOCE_MASK (1 << CL_SD_CTL_IOCE_SHIFT) 71 #define CL_SD_CTL_IOCE(x) \ 72 ((x << CL_SD_CTL_IOCE_SHIFT) & CL_SD_CTL_IOCE_MASK) 73 74 /* FIFO Error Interrupt Enable */ 75 #define CL_SD_CTL_FEIE_SHIFT 3 76 #define CL_SD_CTL_FEIE_MASK (1 << CL_SD_CTL_FEIE_SHIFT) 77 #define CL_SD_CTL_FEIE(x) \ 78 ((x << CL_SD_CTL_FEIE_SHIFT) & CL_SD_CTL_FEIE_MASK) 79 80 /* Descriptor Error Interrupt Enable */ 81 #define CL_SD_CTL_DEIE_SHIFT 4 82 #define CL_SD_CTL_DEIE_MASK (1 << CL_SD_CTL_DEIE_SHIFT) 83 #define CL_SD_CTL_DEIE(x) \ 84 ((x << CL_SD_CTL_DEIE_SHIFT) & CL_SD_CTL_DEIE_MASK) 85 86 /* FIFO Limit Change */ 87 #define CL_SD_CTL_FIFOLC_SHIFT 5 88 #define CL_SD_CTL_FIFOLC_MASK (1 << CL_SD_CTL_FIFOLC_SHIFT) 89 #define CL_SD_CTL_FIFOLC(x) \ 90 ((x << CL_SD_CTL_FIFOLC_SHIFT) & CL_SD_CTL_FIFOLC_MASK) 91 92 /* Stripe Control */ 93 #define CL_SD_CTL_STRIPE_SHIFT 16 94 #define CL_SD_CTL_STRIPE_MASK (0x3 << CL_SD_CTL_STRIPE_SHIFT) 95 #define CL_SD_CTL_STRIPE(x) \ 96 ((x << CL_SD_CTL_STRIPE_SHIFT) & CL_SD_CTL_STRIPE_MASK) 97 98 /* Traffic Priority */ 99 #define CL_SD_CTL_TP_SHIFT 18 100 #define CL_SD_CTL_TP_MASK (1 << CL_SD_CTL_TP_SHIFT) 101 #define CL_SD_CTL_TP(x) \ 102 ((x << CL_SD_CTL_TP_SHIFT) & CL_SD_CTL_TP_MASK) 103 104 /* Bidirectional Direction Control */ 105 #define CL_SD_CTL_DIR_SHIFT 19 106 #define CL_SD_CTL_DIR_MASK (1 << CL_SD_CTL_DIR_SHIFT) 107 #define CL_SD_CTL_DIR(x) \ 108 ((x << CL_SD_CTL_DIR_SHIFT) & CL_SD_CTL_DIR_MASK) 109 110 /* Stream Number */ 111 #define CL_SD_CTL_STRM_SHIFT 20 112 #define CL_SD_CTL_STRM_MASK (0xf << CL_SD_CTL_STRM_SHIFT) 113 #define CL_SD_CTL_STRM(x) \ 114 ((x << CL_SD_CTL_STRM_SHIFT) & CL_SD_CTL_STRM_MASK) 115 116 /* CL: Stream Descriptor x Status */ 117 118 /* Buffer Completion Interrupt Status */ 119 #define CL_SD_STS_BCIS(x) CL_SD_CTL_IOCE(x) 120 121 /* FIFO Error */ 122 #define CL_SD_STS_FIFOE(x) CL_SD_CTL_FEIE(x) 123 124 /* Descriptor Error */ 125 #define CL_SD_STS_DESE(x) CL_SD_CTL_DEIE(x) 126 127 /* FIFO Ready */ 128 #define CL_SD_STS_FIFORDY(x) CL_SD_CTL_FIFOLC(x) 129 130 131 /* CL: Stream Descriptor x Last Valid Index */ 132 #define CL_SD_LVI_SHIFT 0 133 #define CL_SD_LVI_MASK (0xff << CL_SD_LVI_SHIFT) 134 #define CL_SD_LVI(x) ((x << CL_SD_LVI_SHIFT) & CL_SD_LVI_MASK) 135 136 /* CL: Stream Descriptor x FIFO Eviction Watermark */ 137 #define CL_SD_FIFOW_SHIFT 0 138 #define CL_SD_FIFOW_MASK (0x7 << CL_SD_FIFOW_SHIFT) 139 #define CL_SD_FIFOW(x) \ 140 ((x << CL_SD_FIFOW_SHIFT) & CL_SD_FIFOW_MASK) 141 142 /* CL: Stream Descriptor x Buffer Descriptor List Pointer Lower Base Address */ 143 144 /* Protect Bits */ 145 #define CL_SD_BDLPLBA_PROT_SHIFT 0 146 #define CL_SD_BDLPLBA_PROT_MASK (1 << CL_SD_BDLPLBA_PROT_SHIFT) 147 #define CL_SD_BDLPLBA_PROT(x) \ 148 ((x << CL_SD_BDLPLBA_PROT_SHIFT) & CL_SD_BDLPLBA_PROT_MASK) 149 150 /* Buffer Descriptor List Lower Base Address */ 151 #define CL_SD_BDLPLBA_SHIFT 7 152 #define CL_SD_BDLPLBA_MASK (0x1ffffff << CL_SD_BDLPLBA_SHIFT) 153 #define CL_SD_BDLPLBA(x) \ 154 ((BDL_ALIGN(lower_32_bits(x)) << CL_SD_BDLPLBA_SHIFT) & CL_SD_BDLPLBA_MASK) 155 156 /* Buffer Descriptor List Upper Base Address */ 157 #define CL_SD_BDLPUBA_SHIFT 0 158 #define CL_SD_BDLPUBA_MASK (0xffffffff << CL_SD_BDLPUBA_SHIFT) 159 #define CL_SD_BDLPUBA(x) \ 160 ((upper_32_bits(x) << CL_SD_BDLPUBA_SHIFT) & CL_SD_BDLPUBA_MASK) 161 162 /* 163 * Code Loader - Software Position Based FIFO 164 * Capability Registers x Software Position Based FIFO Header 165 */ 166 167 /* Next Capability Pointer */ 168 #define CL_SPBFIFO_SPBFCH_PTR_SHIFT 0 169 #define CL_SPBFIFO_SPBFCH_PTR_MASK (0xff << CL_SPBFIFO_SPBFCH_PTR_SHIFT) 170 #define CL_SPBFIFO_SPBFCH_PTR(x) \ 171 ((x << CL_SPBFIFO_SPBFCH_PTR_SHIFT) & CL_SPBFIFO_SPBFCH_PTR_MASK) 172 173 /* Capability Identifier */ 174 #define CL_SPBFIFO_SPBFCH_ID_SHIFT 16 175 #define CL_SPBFIFO_SPBFCH_ID_MASK (0xfff << CL_SPBFIFO_SPBFCH_ID_SHIFT) 176 #define CL_SPBFIFO_SPBFCH_ID(x) \ 177 ((x << CL_SPBFIFO_SPBFCH_ID_SHIFT) & CL_SPBFIFO_SPBFCH_ID_MASK) 178 179 /* Capability Version */ 180 #define CL_SPBFIFO_SPBFCH_VER_SHIFT 28 181 #define CL_SPBFIFO_SPBFCH_VER_MASK (0xf << CL_SPBFIFO_SPBFCH_VER_SHIFT) 182 #define CL_SPBFIFO_SPBFCH_VER(x) \ 183 ((x << CL_SPBFIFO_SPBFCH_VER_SHIFT) & CL_SPBFIFO_SPBFCH_VER_MASK) 184 185 /* Software Position in Buffer Enable */ 186 #define CL_SPBFIFO_SPBFCCTL_SPIBE_SHIFT 0 187 #define CL_SPBFIFO_SPBFCCTL_SPIBE_MASK (1 << CL_SPBFIFO_SPBFCCTL_SPIBE_SHIFT) 188 #define CL_SPBFIFO_SPBFCCTL_SPIBE(x) \ 189 ((x << CL_SPBFIFO_SPBFCCTL_SPIBE_SHIFT) & CL_SPBFIFO_SPBFCCTL_SPIBE_MASK) 190 191 /* SST IPC SKL defines */ 192 #define SKL_WAIT_TIMEOUT 500 /* 500 msec */ 193 #define SKL_MAX_BUFFER_SIZE (32 * PAGE_SIZE) 194 195 enum skl_cl_dma_wake_states { 196 SKL_CL_DMA_STATUS_NONE = 0, 197 SKL_CL_DMA_BUF_COMPLETE, 198 SKL_CL_DMA_ERR, /* TODO: Expand the error states */ 199 }; 200 201 struct sst_dsp; 202 203 struct skl_cl_dev_ops { 204 void (*cl_setup_bdle)(struct sst_dsp *ctx, 205 struct snd_dma_buffer *dmab_data, 206 u32 **bdlp, int size, int with_ioc); 207 void (*cl_setup_controller)(struct sst_dsp *ctx, 208 struct snd_dma_buffer *dmab_bdl, 209 unsigned int max_size, u32 page_count); 210 void (*cl_setup_spb)(struct sst_dsp *ctx, 211 unsigned int size, bool enable); 212 void (*cl_cleanup_spb)(struct sst_dsp *ctx); 213 void (*cl_trigger)(struct sst_dsp *ctx, bool enable); 214 void (*cl_cleanup_controller)(struct sst_dsp *ctx); 215 int (*cl_copy_to_dmabuf)(struct sst_dsp *ctx, 216 const void *bin, u32 size, bool wait); 217 void (*cl_stop_dma)(struct sst_dsp *ctx); 218 }; 219 220 /** 221 * skl_cl_dev - holds information for code loader dma transfer 222 * 223 * @dmab_data: buffer pointer 224 * @dmab_bdl: buffer descriptor list 225 * @bufsize: ring buffer size 226 * @frags: Last valid buffer descriptor index in the BDL 227 * @curr_spib_pos: Current position in ring buffer 228 * @dma_buffer_offset: dma buffer offset 229 * @ops: operations supported on CL dma 230 * @wait_queue: wait queue to wake for wake event 231 * @wake_status: DMA wake status 232 * @wait_condition: condition to wait on wait queue 233 * @cl_dma_lock: for synchronized access to cldma 234 */ 235 struct skl_cl_dev { 236 struct snd_dma_buffer dmab_data; 237 struct snd_dma_buffer dmab_bdl; 238 239 unsigned int bufsize; 240 unsigned int frags; 241 242 unsigned int curr_spib_pos; 243 unsigned int dma_buffer_offset; 244 struct skl_cl_dev_ops ops; 245 246 wait_queue_head_t wait_queue; 247 int wake_status; 248 bool wait_condition; 249 }; 250 251 #endif /* SKL_SST_CLDMA_H_ */ 252