162306a36Sopenharmony_ci/* SPDX-License-Identifier: GPL-2.0+ */ 262306a36Sopenharmony_ci/* 362306a36Sopenharmony_ci * AMD ALSA SoC PCM Driver 462306a36Sopenharmony_ci * 562306a36Sopenharmony_ci * Copyright 2016 Advanced Micro Devices, Inc. 662306a36Sopenharmony_ci */ 762306a36Sopenharmony_ci 862306a36Sopenharmony_ci#include "chip_offset_byte.h" 962306a36Sopenharmony_ci#include <sound/pcm.h> 1062306a36Sopenharmony_ci#define I2S_SP_INSTANCE 0x01 1162306a36Sopenharmony_ci#define I2S_BT_INSTANCE 0x02 1262306a36Sopenharmony_ci 1362306a36Sopenharmony_ci#define TDM_ENABLE 1 1462306a36Sopenharmony_ci#define TDM_DISABLE 0 1562306a36Sopenharmony_ci 1662306a36Sopenharmony_ci#define ACP3x_DEVS 4 1762306a36Sopenharmony_ci#define ACP3x_PHY_BASE_ADDRESS 0x1240000 1862306a36Sopenharmony_ci#define ACP3x_I2S_MODE 0 1962306a36Sopenharmony_ci#define ACP3x_REG_START 0x1240000 2062306a36Sopenharmony_ci#define ACP3x_REG_END 0x1250200 2162306a36Sopenharmony_ci#define ACP3x_I2STDM_REG_START 0x1242400 2262306a36Sopenharmony_ci#define ACP3x_I2STDM_REG_END 0x1242410 2362306a36Sopenharmony_ci#define ACP3x_BT_TDM_REG_START 0x1242800 2462306a36Sopenharmony_ci#define ACP3x_BT_TDM_REG_END 0x1242810 2562306a36Sopenharmony_ci#define I2S_MODE 0x04 2662306a36Sopenharmony_ci#define I2S_RX_THRESHOLD 27 2762306a36Sopenharmony_ci#define I2S_TX_THRESHOLD 28 2862306a36Sopenharmony_ci#define BT_TX_THRESHOLD 26 2962306a36Sopenharmony_ci#define BT_RX_THRESHOLD 25 3062306a36Sopenharmony_ci#define ACP_ERR_INTR_MASK 29 3162306a36Sopenharmony_ci#define ACP3x_POWER_ON 0x00 3262306a36Sopenharmony_ci#define ACP3x_POWER_ON_IN_PROGRESS 0x01 3362306a36Sopenharmony_ci#define ACP3x_POWER_OFF 0x02 3462306a36Sopenharmony_ci#define ACP3x_POWER_OFF_IN_PROGRESS 0x03 3562306a36Sopenharmony_ci#define ACP3x_SOFT_RESET__SoftResetAudDone_MASK 0x00010001 3662306a36Sopenharmony_ci 3762306a36Sopenharmony_ci#define ACP_SRAM_PTE_OFFSET 0x02050000 3862306a36Sopenharmony_ci#define ACP_SRAM_SP_PB_PTE_OFFSET 0x0 3962306a36Sopenharmony_ci#define ACP_SRAM_SP_CP_PTE_OFFSET 0x100 4062306a36Sopenharmony_ci#define ACP_SRAM_BT_PB_PTE_OFFSET 0x200 4162306a36Sopenharmony_ci#define ACP_SRAM_BT_CP_PTE_OFFSET 0x300 4262306a36Sopenharmony_ci#define PAGE_SIZE_4K_ENABLE 0x2 4362306a36Sopenharmony_ci#define I2S_SP_TX_MEM_WINDOW_START 0x4000000 4462306a36Sopenharmony_ci#define I2S_SP_RX_MEM_WINDOW_START 0x4020000 4562306a36Sopenharmony_ci#define I2S_BT_TX_MEM_WINDOW_START 0x4040000 4662306a36Sopenharmony_ci#define I2S_BT_RX_MEM_WINDOW_START 0x4060000 4762306a36Sopenharmony_ci 4862306a36Sopenharmony_ci#define SP_PB_FIFO_ADDR_OFFSET 0x500 4962306a36Sopenharmony_ci#define SP_CAPT_FIFO_ADDR_OFFSET 0x700 5062306a36Sopenharmony_ci#define BT_PB_FIFO_ADDR_OFFSET 0x900 5162306a36Sopenharmony_ci#define BT_CAPT_FIFO_ADDR_OFFSET 0xB00 5262306a36Sopenharmony_ci#define PLAYBACK_MIN_NUM_PERIODS 2 5362306a36Sopenharmony_ci#define PLAYBACK_MAX_NUM_PERIODS 8 5462306a36Sopenharmony_ci#define PLAYBACK_MAX_PERIOD_SIZE 8192 5562306a36Sopenharmony_ci#define PLAYBACK_MIN_PERIOD_SIZE 1024 5662306a36Sopenharmony_ci#define CAPTURE_MIN_NUM_PERIODS 2 5762306a36Sopenharmony_ci#define CAPTURE_MAX_NUM_PERIODS 8 5862306a36Sopenharmony_ci#define CAPTURE_MAX_PERIOD_SIZE 8192 5962306a36Sopenharmony_ci#define CAPTURE_MIN_PERIOD_SIZE 1024 6062306a36Sopenharmony_ci 6162306a36Sopenharmony_ci#define MAX_BUFFER (PLAYBACK_MAX_PERIOD_SIZE * PLAYBACK_MAX_NUM_PERIODS) 6262306a36Sopenharmony_ci#define MIN_BUFFER MAX_BUFFER 6362306a36Sopenharmony_ci#define FIFO_SIZE 0x100 6462306a36Sopenharmony_ci#define DMA_SIZE 0x40 6562306a36Sopenharmony_ci#define FRM_LEN 0x100 6662306a36Sopenharmony_ci 6762306a36Sopenharmony_ci#define SLOT_WIDTH_8 0x08 6862306a36Sopenharmony_ci#define SLOT_WIDTH_16 0x10 6962306a36Sopenharmony_ci#define SLOT_WIDTH_24 0x18 7062306a36Sopenharmony_ci#define SLOT_WIDTH_32 0x20 7162306a36Sopenharmony_ci#define ACP_PGFSM_CNTL_POWER_ON_MASK 0x01 7262306a36Sopenharmony_ci#define ACP_PGFSM_CNTL_POWER_OFF_MASK 0x00 7362306a36Sopenharmony_ci#define ACP_PGFSM_STATUS_MASK 0x03 7462306a36Sopenharmony_ci#define ACP_POWERED_ON 0x00 7562306a36Sopenharmony_ci#define ACP_POWER_ON_IN_PROGRESS 0x01 7662306a36Sopenharmony_ci#define ACP_POWERED_OFF 0x02 7762306a36Sopenharmony_ci#define ACP_POWER_OFF_IN_PROGRESS 0x03 7862306a36Sopenharmony_ci 7962306a36Sopenharmony_ci#define ACP3x_ITER_IRER_SAMP_LEN_MASK 0x38 8062306a36Sopenharmony_ci#define ACP_EXT_INTR_STAT_CLEAR_MASK 0xFFFFFFFF 8162306a36Sopenharmony_ci 8262306a36Sopenharmony_cistruct acp3x_platform_info { 8362306a36Sopenharmony_ci u16 play_i2s_instance; 8462306a36Sopenharmony_ci u16 cap_i2s_instance; 8562306a36Sopenharmony_ci u16 capture_channel; 8662306a36Sopenharmony_ci}; 8762306a36Sopenharmony_ci 8862306a36Sopenharmony_cistruct i2s_dev_data { 8962306a36Sopenharmony_ci bool tdm_mode; 9062306a36Sopenharmony_ci int i2s_irq; 9162306a36Sopenharmony_ci u16 i2s_instance; 9262306a36Sopenharmony_ci u32 tdm_fmt; 9362306a36Sopenharmony_ci u32 substream_type; 9462306a36Sopenharmony_ci void __iomem *acp3x_base; 9562306a36Sopenharmony_ci struct snd_pcm_substream *play_stream; 9662306a36Sopenharmony_ci struct snd_pcm_substream *capture_stream; 9762306a36Sopenharmony_ci struct snd_pcm_substream *i2ssp_play_stream; 9862306a36Sopenharmony_ci struct snd_pcm_substream *i2ssp_capture_stream; 9962306a36Sopenharmony_ci}; 10062306a36Sopenharmony_ci 10162306a36Sopenharmony_cistruct i2s_stream_instance { 10262306a36Sopenharmony_ci u16 num_pages; 10362306a36Sopenharmony_ci u16 i2s_instance; 10462306a36Sopenharmony_ci u16 capture_channel; 10562306a36Sopenharmony_ci u16 direction; 10662306a36Sopenharmony_ci u16 channels; 10762306a36Sopenharmony_ci u32 xfer_resolution; 10862306a36Sopenharmony_ci u32 val; 10962306a36Sopenharmony_ci dma_addr_t dma_addr; 11062306a36Sopenharmony_ci u64 bytescount; 11162306a36Sopenharmony_ci void __iomem *acp3x_base; 11262306a36Sopenharmony_ci}; 11362306a36Sopenharmony_ci 11462306a36Sopenharmony_cistatic inline u32 rv_readl(void __iomem *base_addr) 11562306a36Sopenharmony_ci{ 11662306a36Sopenharmony_ci return readl(base_addr - ACP3x_PHY_BASE_ADDRESS); 11762306a36Sopenharmony_ci} 11862306a36Sopenharmony_ci 11962306a36Sopenharmony_cistatic inline void rv_writel(u32 val, void __iomem *base_addr) 12062306a36Sopenharmony_ci{ 12162306a36Sopenharmony_ci writel(val, base_addr - ACP3x_PHY_BASE_ADDRESS); 12262306a36Sopenharmony_ci} 12362306a36Sopenharmony_ci 12462306a36Sopenharmony_cistatic inline u64 acp_get_byte_count(struct i2s_stream_instance *rtd, 12562306a36Sopenharmony_ci int direction) 12662306a36Sopenharmony_ci{ 12762306a36Sopenharmony_ci u64 byte_count; 12862306a36Sopenharmony_ci 12962306a36Sopenharmony_ci if (direction == SNDRV_PCM_STREAM_PLAYBACK) { 13062306a36Sopenharmony_ci switch (rtd->i2s_instance) { 13162306a36Sopenharmony_ci case I2S_BT_INSTANCE: 13262306a36Sopenharmony_ci byte_count = rv_readl(rtd->acp3x_base + 13362306a36Sopenharmony_ci mmACP_BT_TX_LINEARPOSITIONCNTR_HIGH); 13462306a36Sopenharmony_ci byte_count |= rv_readl(rtd->acp3x_base + 13562306a36Sopenharmony_ci mmACP_BT_TX_LINEARPOSITIONCNTR_LOW); 13662306a36Sopenharmony_ci break; 13762306a36Sopenharmony_ci case I2S_SP_INSTANCE: 13862306a36Sopenharmony_ci default: 13962306a36Sopenharmony_ci byte_count = rv_readl(rtd->acp3x_base + 14062306a36Sopenharmony_ci mmACP_I2S_TX_LINEARPOSITIONCNTR_HIGH); 14162306a36Sopenharmony_ci byte_count |= rv_readl(rtd->acp3x_base + 14262306a36Sopenharmony_ci mmACP_I2S_TX_LINEARPOSITIONCNTR_LOW); 14362306a36Sopenharmony_ci } 14462306a36Sopenharmony_ci 14562306a36Sopenharmony_ci } else { 14662306a36Sopenharmony_ci switch (rtd->i2s_instance) { 14762306a36Sopenharmony_ci case I2S_BT_INSTANCE: 14862306a36Sopenharmony_ci byte_count = rv_readl(rtd->acp3x_base + 14962306a36Sopenharmony_ci mmACP_BT_RX_LINEARPOSITIONCNTR_HIGH); 15062306a36Sopenharmony_ci byte_count |= rv_readl(rtd->acp3x_base + 15162306a36Sopenharmony_ci mmACP_BT_RX_LINEARPOSITIONCNTR_LOW); 15262306a36Sopenharmony_ci break; 15362306a36Sopenharmony_ci case I2S_SP_INSTANCE: 15462306a36Sopenharmony_ci default: 15562306a36Sopenharmony_ci byte_count = rv_readl(rtd->acp3x_base + 15662306a36Sopenharmony_ci mmACP_I2S_RX_LINEARPOSITIONCNTR_HIGH); 15762306a36Sopenharmony_ci byte_count |= rv_readl(rtd->acp3x_base + 15862306a36Sopenharmony_ci mmACP_I2S_RX_LINEARPOSITIONCNTR_LOW); 15962306a36Sopenharmony_ci } 16062306a36Sopenharmony_ci } 16162306a36Sopenharmony_ci return byte_count; 16262306a36Sopenharmony_ci} 163