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