OSDN Git Service

ASoC: qcom: Add regmap config support for codec dma driver
authorSrinivasa Rao Mandadapu <quic_srivasam@quicinc.com>
Thu, 24 Feb 2022 15:33:44 +0000 (21:03 +0530)
committerMark Brown <broonie@kernel.org>
Thu, 24 Feb 2022 20:21:44 +0000 (20:21 +0000)
Update regmap configuration for supporting headset playback and
capture and DMIC capture using codec dma interface

Signed-off-by: Srinivasa Rao Mandadapu <quic_srivasam@quicinc.com>
Co-developed-by: Venkata Prasad Potturu <quic_potturu@quicinc.com>
Signed-off-by: Venkata Prasad Potturu <quic_potturu@quicinc.com>
Reviewed-by: Srinivas Kandagatla <srinivas.kandagatla@linaro.org>
Link: https://lore.kernel.org/r/1645716828-15305-6-git-send-email-quic_srivasam@quicinc.com
Signed-off-by: Mark Brown <broonie@kernel.org>
sound/soc/qcom/lpass-cpu.c

index 3bd9eb3..e6846ad 100644 (file)
@@ -28,6 +28,8 @@
 #define LPASS_CPU_I2S_SD2_3_MASK       GENMASK(3, 2)
 #define LPASS_CPU_I2S_SD0_1_2_MASK     GENMASK(2, 0)
 #define LPASS_CPU_I2S_SD0_1_2_3_MASK   GENMASK(3, 0)
+#define LPASS_REG_READ 1
+#define LPASS_REG_WRITE 0
 
 /*
  * Channel maps for Quad channel playbacks on MI2S Secondary
@@ -798,6 +800,189 @@ static struct regmap_config lpass_hdmi_regmap_config = {
        .cache_type = REGCACHE_FLAT,
 };
 
+static bool __lpass_rxtx_regmap_accessible(struct device *dev, unsigned int reg, bool rw)
+{
+       struct lpass_data *drvdata = dev_get_drvdata(dev);
+       struct lpass_variant *v = drvdata->variant;
+       int i;
+
+       for (i = 0; i < v->rxtx_irq_ports; ++i) {
+               if (reg == LPAIF_RXTX_IRQCLEAR_REG(v, i))
+                       return true;
+               if (reg == LPAIF_RXTX_IRQEN_REG(v, i))
+                       return true;
+               if (reg == LPAIF_RXTX_IRQSTAT_REG(v, i))
+                       return true;
+       }
+
+       for (i = 0; i < v->rxtx_rdma_channels; ++i) {
+               if (reg == LPAIF_CDC_RXTX_RDMACTL_REG(v, i, LPASS_CDC_DMA_RX0))
+                       return true;
+               if (reg == LPAIF_CDC_RXTX_RDMABASE_REG(v, i, LPASS_CDC_DMA_RX0))
+                       return true;
+               if (reg == LPAIF_CDC_RXTX_RDMABUFF_REG(v, i, LPASS_CDC_DMA_RX0))
+                       return true;
+               if (rw == LPASS_REG_READ) {
+                       if (reg == LPAIF_CDC_RXTX_RDMACURR_REG(v, i, LPASS_CDC_DMA_RX0))
+                               return true;
+               }
+               if (reg == LPAIF_CDC_RXTX_RDMAPER_REG(v, i, LPASS_CDC_DMA_RX0))
+                       return true;
+               if (reg == LPAIF_CDC_RXTX_RDMA_INTF_REG(v, i, LPASS_CDC_DMA_RX0))
+                       return true;
+       }
+
+       for (i = 0; i < v->rxtx_wrdma_channels; ++i) {
+               if (reg == LPAIF_CDC_RXTX_WRDMACTL_REG(v, i + v->rxtx_wrdma_channel_start,
+                                                       LPASS_CDC_DMA_TX3))
+                       return true;
+               if (reg == LPAIF_CDC_RXTX_WRDMABASE_REG(v, i + v->rxtx_wrdma_channel_start,
+                                                       LPASS_CDC_DMA_TX3))
+                       return true;
+               if (reg == LPAIF_CDC_RXTX_WRDMABUFF_REG(v, i + v->rxtx_wrdma_channel_start,
+                                                       LPASS_CDC_DMA_TX3))
+                       return true;
+               if (rw == LPASS_REG_READ) {
+                       if (reg == LPAIF_CDC_RXTX_WRDMACURR_REG(v, i, LPASS_CDC_DMA_RX0))
+                               return true;
+               }
+               if (reg == LPAIF_CDC_RXTX_WRDMAPER_REG(v, i + v->rxtx_wrdma_channel_start,
+                                                       LPASS_CDC_DMA_TX3))
+                       return true;
+               if (reg == LPAIF_CDC_RXTX_WRDMA_INTF_REG(v, i + v->rxtx_wrdma_channel_start,
+                                                       LPASS_CDC_DMA_TX3))
+                       return true;
+       }
+       return false;
+}
+
+static bool lpass_rxtx_regmap_writeable(struct device *dev, unsigned int reg)
+{
+       return __lpass_rxtx_regmap_accessible(dev, reg, LPASS_REG_WRITE);
+}
+
+static bool lpass_rxtx_regmap_readable(struct device *dev, unsigned int reg)
+{
+       return __lpass_rxtx_regmap_accessible(dev, reg, LPASS_REG_READ);
+}
+
+static bool lpass_rxtx_regmap_volatile(struct device *dev, unsigned int reg)
+{
+       struct lpass_data *drvdata = dev_get_drvdata(dev);
+       struct lpass_variant *v = drvdata->variant;
+       int i;
+
+       for (i = 0; i < v->rxtx_irq_ports; ++i) {
+               if (reg == LPAIF_RXTX_IRQCLEAR_REG(v, i))
+                       return true;
+               if (reg == LPAIF_RXTX_IRQSTAT_REG(v, i))
+                       return true;
+       }
+
+       for (i = 0; i < v->rxtx_rdma_channels; ++i)
+               if (reg == LPAIF_CDC_RXTX_RDMACURR_REG(v, i, LPASS_CDC_DMA_RX0))
+                       return true;
+
+       for (i = 0; i < v->rxtx_wrdma_channels; ++i)
+               if (reg == LPAIF_CDC_RXTX_WRDMACURR_REG(v, i + v->rxtx_wrdma_channel_start,
+                                                       LPASS_CDC_DMA_TX3))
+                       return true;
+
+       return false;
+}
+
+static bool __lpass_va_regmap_accessible(struct device *dev, unsigned int reg, bool rw)
+{
+       struct lpass_data *drvdata = dev_get_drvdata(dev);
+       struct lpass_variant *v = drvdata->variant;
+       int i;
+
+       for (i = 0; i < v->va_irq_ports; ++i) {
+               if (reg == LPAIF_VA_IRQCLEAR_REG(v, i))
+                       return true;
+               if (reg == LPAIF_VA_IRQEN_REG(v, i))
+                       return true;
+               if (reg == LPAIF_VA_IRQSTAT_REG(v, i))
+                       return true;
+       }
+
+       for (i = 0; i < v->va_wrdma_channels; ++i) {
+               if (reg == LPAIF_CDC_VA_WRDMACTL_REG(v, i + v->va_wrdma_channel_start,
+                                                       LPASS_CDC_DMA_VA_TX0))
+                       return true;
+               if (reg == LPAIF_CDC_VA_WRDMABASE_REG(v, i + v->va_wrdma_channel_start,
+                                                       LPASS_CDC_DMA_VA_TX0))
+                       return true;
+               if (reg == LPAIF_CDC_VA_WRDMABUFF_REG(v, i + v->va_wrdma_channel_start,
+                                                       LPASS_CDC_DMA_VA_TX0))
+                       return true;
+               if (rw == LPASS_REG_READ) {
+                       if (reg == LPAIF_CDC_VA_WRDMACURR_REG(v, i + v->va_wrdma_channel_start,
+                                                       LPASS_CDC_DMA_VA_TX0))
+                               return true;
+               }
+               if (reg == LPAIF_CDC_VA_WRDMAPER_REG(v, i + v->va_wrdma_channel_start,
+                                                       LPASS_CDC_DMA_VA_TX0))
+                       return true;
+               if (reg == LPAIF_CDC_VA_WRDMA_INTF_REG(v, i + v->va_wrdma_channel_start,
+                                                       LPASS_CDC_DMA_VA_TX0))
+                       return true;
+       }
+       return false;
+}
+
+static bool lpass_va_regmap_writeable(struct device *dev, unsigned int reg)
+{
+       return __lpass_va_regmap_accessible(dev, reg, LPASS_REG_WRITE);
+}
+
+static bool lpass_va_regmap_readable(struct device *dev, unsigned int reg)
+{
+       return __lpass_va_regmap_accessible(dev, reg, LPASS_REG_READ);
+}
+
+static bool lpass_va_regmap_volatile(struct device *dev, unsigned int reg)
+{
+       struct lpass_data *drvdata = dev_get_drvdata(dev);
+       struct lpass_variant *v = drvdata->variant;
+       int i;
+
+       for (i = 0; i < v->va_irq_ports; ++i) {
+               if (reg == LPAIF_VA_IRQCLEAR_REG(v, i))
+                       return true;
+               if (reg == LPAIF_VA_IRQSTAT_REG(v, i))
+                       return true;
+       }
+
+       for (i = 0; i < v->va_wrdma_channels; ++i) {
+               if (reg == LPAIF_CDC_VA_WRDMACURR_REG(v, i + v->va_wrdma_channel_start,
+                                                       LPASS_CDC_DMA_VA_TX0))
+                       return true;
+       }
+
+       return false;
+}
+
+static struct regmap_config lpass_rxtx_regmap_config = {
+       .reg_bits = 32,
+       .reg_stride = 4,
+       .val_bits = 32,
+       .writeable_reg = lpass_rxtx_regmap_writeable,
+       .readable_reg = lpass_rxtx_regmap_readable,
+       .volatile_reg = lpass_rxtx_regmap_volatile,
+       .cache_type = REGCACHE_FLAT,
+};
+
+static struct regmap_config lpass_va_regmap_config = {
+       .reg_bits = 32,
+       .reg_stride = 4,
+       .val_bits = 32,
+       .writeable_reg = lpass_va_regmap_writeable,
+       .readable_reg = lpass_va_regmap_readable,
+       .volatile_reg = lpass_va_regmap_volatile,
+       .cache_type = REGCACHE_FLAT,
+};
+
 static unsigned int of_lpass_cpu_parse_sd_lines(struct device *dev,
                                                struct device_node *node,
                                                const char *name)
@@ -857,6 +1042,8 @@ static void of_lpass_cpu_parse_dai_data(struct device *dev,
                }
                if (id == LPASS_DP_RX) {
                        data->hdmi_port_enable = 1;
+               } else if (is_cdc_dma_port(id)) {
+                       data->codec_dma_enable = 1;
                } else {
                        data->mi2s_playback_sd_mode[id] =
                                of_lpass_cpu_parse_sd_lines(dev, node,
@@ -868,10 +1055,33 @@ static void of_lpass_cpu_parse_dai_data(struct device *dev,
        }
 }
 
+static int of_lpass_cdc_dma_clks_parse(struct device *dev,
+                                       struct lpass_data *data)
+{
+       data->codec_mem0 = devm_clk_get(dev, "audio_cc_codec_mem0");
+       if (IS_ERR(data->codec_mem0))
+               return PTR_ERR(data->codec_mem0);
+
+       data->codec_mem1 = devm_clk_get(dev, "audio_cc_codec_mem1");
+       if (IS_ERR(data->codec_mem1))
+               return PTR_ERR(data->codec_mem1);
+
+       data->codec_mem2 = devm_clk_get(dev, "audio_cc_codec_mem2");
+       if (IS_ERR(data->codec_mem2))
+               return PTR_ERR(data->codec_mem2);
+
+       data->va_mem0 = devm_clk_get(dev, "aon_cc_va_mem0");
+       if (IS_ERR(data->va_mem0))
+               return PTR_ERR(data->va_mem0);
+
+       return 0;
+}
+
 int asoc_qcom_lpass_cpu_platform_probe(struct platform_device *pdev)
 {
        struct lpass_data *drvdata;
        struct device_node *dsp_of_node;
+       struct resource *res;
        struct lpass_variant *variant;
        struct device *dev = &pdev->dev;
        const struct of_device_id *match;
@@ -897,6 +1107,47 @@ int asoc_qcom_lpass_cpu_platform_probe(struct platform_device *pdev)
 
        of_lpass_cpu_parse_dai_data(dev, drvdata);
 
+       if (drvdata->codec_dma_enable) {
+               drvdata->rxtx_lpaif =
+                               devm_platform_ioremap_resource_byname(pdev, "lpass-rxtx-lpaif");
+               if (IS_ERR(drvdata->rxtx_lpaif))
+                       return PTR_ERR(drvdata->rxtx_lpaif);
+
+               drvdata->va_lpaif = devm_platform_ioremap_resource_byname(pdev, "lpass-va-lpaif");
+               if (IS_ERR(drvdata->va_lpaif))
+                       return PTR_ERR(drvdata->va_lpaif);
+
+               lpass_rxtx_regmap_config.max_register = LPAIF_CDC_RXTX_WRDMAPER_REG(variant,
+                                       variant->rxtx_wrdma_channels +
+                                       variant->rxtx_wrdma_channel_start, LPASS_CDC_DMA_TX3);
+
+               drvdata->rxtx_lpaif_map = devm_regmap_init_mmio(dev, drvdata->rxtx_lpaif,
+                                       &lpass_rxtx_regmap_config);
+               if (IS_ERR(drvdata->rxtx_lpaif_map))
+                       return PTR_ERR(drvdata->rxtx_lpaif_map);
+
+               lpass_va_regmap_config.max_register = LPAIF_CDC_VA_WRDMAPER_REG(variant,
+                                       variant->va_wrdma_channels +
+                                       variant->va_wrdma_channel_start, LPASS_CDC_DMA_VA_TX0);
+
+               drvdata->va_lpaif_map = devm_regmap_init_mmio(dev, drvdata->va_lpaif,
+                                       &lpass_va_regmap_config);
+               if (IS_ERR(drvdata->va_lpaif_map))
+                       return PTR_ERR(drvdata->va_lpaif_map);
+
+               ret = of_lpass_cdc_dma_clks_parse(dev, drvdata);
+               if (ret) {
+                       dev_err(dev, "failed to get cdc dma clocks %d\n", ret);
+                       return ret;
+               }
+
+               res = platform_get_resource_byname(pdev, IORESOURCE_MEM, "lpass-rxtx-cdc-dma-lpm");
+               drvdata->rxtx_cdc_dma_lpm_buf = res->start;
+
+               res = platform_get_resource_byname(pdev, IORESOURCE_MEM, "lpass-va-cdc-dma-lpm");
+               drvdata->va_cdc_dma_lpm_buf = res->start;
+       }
+
        drvdata->lpaif = devm_platform_ioremap_resource_byname(pdev, "lpass-lpaif");
        if (IS_ERR(drvdata->lpaif))
                return PTR_ERR(drvdata->lpaif);
@@ -939,7 +1190,7 @@ int asoc_qcom_lpass_cpu_platform_probe(struct platform_device *pdev)
 
        for (i = 0; i < variant->num_dai; i++) {
                dai_id = variant->dai_driver[i].id;
-               if (dai_id == LPASS_DP_RX)
+               if (dai_id == LPASS_DP_RX || is_cdc_dma_port(dai_id))
                        continue;
 
                drvdata->mi2s_osr_clk[dai_id] = devm_clk_get_optional(dev,