Merge tag 'soundwire-5.4-rc1' of git://git.kernel.org/pub/scm/linux/kernel/git/vkoul...
[linux-2.6-block.git] / drivers / soundwire / intel.c
index 317873bc0555deafbf4b2e26a44a1e67a59cfe9c..f1e38a293967052831fceaf440030581ded8d5db 100644 (file)
@@ -6,6 +6,7 @@
  */
 
 #include <linux/acpi.h>
+#include <linux/debugfs.h>
 #include <linux/delay.h>
 #include <linux/module.h>
 #include <linux/interrupt.h>
@@ -16,6 +17,7 @@
 #include <linux/soundwire/sdw.h>
 #include <linux/soundwire/sdw_intel.h>
 #include "cadence_master.h"
+#include "bus.h"
 #include "intel.h"
 
 /* Intel SHIM Registers Definition */
 
 /* Intel ALH Register definitions */
 #define SDW_ALH_STRMZCFG(x)            (0x000 + (0x4 * (x)))
+#define SDW_ALH_NUM_STREAMS            64
 
 #define SDW_ALH_STRMZCFG_DMAT_VAL      0x3
 #define SDW_ALH_STRMZCFG_DMAT          GENMASK(7, 0)
 #define SDW_ALH_STRMZCFG_CHN           GENMASK(19, 16)
 
+#define SDW_INTEL_QUIRK_MASK_BUS_DISABLE       BIT(1)
+
 enum intel_pdi_type {
        INTEL_PDI_IN = 0,
        INTEL_PDI_OUT = 1,
@@ -98,6 +103,9 @@ struct sdw_intel {
        struct sdw_cdns cdns;
        int instance;
        struct sdw_intel_link_res *res;
+#ifdef CONFIG_DEBUG_FS
+       struct dentry *debugfs;
+#endif
 };
 
 #define cdns_to_intel(_cdns) container_of(_cdns, struct sdw_intel, cdns)
@@ -161,6 +169,118 @@ static int intel_set_bit(void __iomem *base, int offset, u32 value, u32 mask)
        return -EAGAIN;
 }
 
+/*
+ * debugfs
+ */
+#ifdef CONFIG_DEBUG_FS
+
+#define RD_BUF (2 * PAGE_SIZE)
+
+static ssize_t intel_sprintf(void __iomem *mem, bool l,
+                            char *buf, size_t pos, unsigned int reg)
+{
+       int value;
+
+       if (l)
+               value = intel_readl(mem, reg);
+       else
+               value = intel_readw(mem, reg);
+
+       return scnprintf(buf + pos, RD_BUF - pos, "%4x\t%4x\n", reg, value);
+}
+
+static int intel_reg_show(struct seq_file *s_file, void *data)
+{
+       struct sdw_intel *sdw = s_file->private;
+       void __iomem *s = sdw->res->shim;
+       void __iomem *a = sdw->res->alh;
+       char *buf;
+       ssize_t ret;
+       int i, j;
+       unsigned int links, reg;
+
+       buf = kzalloc(RD_BUF, GFP_KERNEL);
+       if (!buf)
+               return -ENOMEM;
+
+       links = intel_readl(s, SDW_SHIM_LCAP) & GENMASK(2, 0);
+
+       ret = scnprintf(buf, RD_BUF, "Register  Value\n");
+       ret += scnprintf(buf + ret, RD_BUF - ret, "\nShim\n");
+
+       for (i = 0; i < links; i++) {
+               reg = SDW_SHIM_LCAP + i * 4;
+               ret += intel_sprintf(s, true, buf, ret, reg);
+       }
+
+       for (i = 0; i < links; i++) {
+               ret += scnprintf(buf + ret, RD_BUF - ret, "\nLink%d\n", i);
+               ret += intel_sprintf(s, false, buf, ret, SDW_SHIM_CTLSCAP(i));
+               ret += intel_sprintf(s, false, buf, ret, SDW_SHIM_CTLS0CM(i));
+               ret += intel_sprintf(s, false, buf, ret, SDW_SHIM_CTLS1CM(i));
+               ret += intel_sprintf(s, false, buf, ret, SDW_SHIM_CTLS2CM(i));
+               ret += intel_sprintf(s, false, buf, ret, SDW_SHIM_CTLS3CM(i));
+               ret += intel_sprintf(s, false, buf, ret, SDW_SHIM_PCMSCAP(i));
+
+               ret += scnprintf(buf + ret, RD_BUF - ret, "\n PCMSyCH registers\n");
+
+               /*
+                * the value 10 is the number of PDIs. We will need a
+                * cleanup to remove hard-coded Intel configurations
+                * from cadence_master.c
+                */
+               for (j = 0; j < 10; j++) {
+                       ret += intel_sprintf(s, false, buf, ret,
+                                       SDW_SHIM_PCMSYCHM(i, j));
+                       ret += intel_sprintf(s, false, buf, ret,
+                                       SDW_SHIM_PCMSYCHC(i, j));
+               }
+               ret += scnprintf(buf + ret, RD_BUF - ret, "\n PDMSCAP, IOCTL, CTMCTL\n");
+
+               ret += intel_sprintf(s, false, buf, ret, SDW_SHIM_PDMSCAP(i));
+               ret += intel_sprintf(s, false, buf, ret, SDW_SHIM_IOCTL(i));
+               ret += intel_sprintf(s, false, buf, ret, SDW_SHIM_CTMCTL(i));
+       }
+
+       ret += scnprintf(buf + ret, RD_BUF - ret, "\nWake registers\n");
+       ret += intel_sprintf(s, false, buf, ret, SDW_SHIM_WAKEEN);
+       ret += intel_sprintf(s, false, buf, ret, SDW_SHIM_WAKESTS);
+
+       ret += scnprintf(buf + ret, RD_BUF - ret, "\nALH STRMzCFG\n");
+       for (i = 0; i < SDW_ALH_NUM_STREAMS; i++)
+               ret += intel_sprintf(a, true, buf, ret, SDW_ALH_STRMZCFG(i));
+
+       seq_printf(s_file, "%s", buf);
+       kfree(buf);
+
+       return 0;
+}
+DEFINE_SHOW_ATTRIBUTE(intel_reg);
+
+static void intel_debugfs_init(struct sdw_intel *sdw)
+{
+       struct dentry *root = sdw->cdns.bus.debugfs;
+
+       if (!root)
+               return;
+
+       sdw->debugfs = debugfs_create_dir("intel-sdw", root);
+
+       debugfs_create_file("intel-registers", 0400, sdw->debugfs, sdw,
+                           &intel_reg_fops);
+
+       sdw_cdns_debugfs_init(&sdw->cdns, sdw->debugfs);
+}
+
+static void intel_debugfs_exit(struct sdw_intel *sdw)
+{
+       debugfs_remove_recursive(sdw->debugfs);
+}
+#else
+static void intel_debugfs_init(struct sdw_intel *sdw) {}
+static void intel_debugfs_exit(struct sdw_intel *sdw) {}
+#endif /* CONFIG_DEBUG_FS */
+
 /*
  * shim ops
  */
@@ -289,6 +409,16 @@ intel_pdi_get_ch_cap(struct sdw_intel *sdw, unsigned int pdi_num, bool pcm)
 
        if (pcm) {
                count = intel_readw(shim, SDW_SHIM_PCMSYCHC(link_id, pdi_num));
+
+               /*
+                * WORKAROUND: on all existing Intel controllers, pdi
+                * number 2 reports channel count as 1 even though it
+                * supports 8 channels. Performing hardcoding for pdi
+                * number 2.
+                */
+               if (pdi_num == 2)
+                       count = 7;
+
        } else {
                count = intel_readw(shim, SDW_SHIM_PDMSCAP(link_id));
                count = ((count & SDW_SHIM_PDMSCAP_CPSS) >>
@@ -397,8 +527,10 @@ static int intel_config_stream(struct sdw_intel *sdw,
                               struct snd_soc_dai *dai,
                               struct snd_pcm_hw_params *hw_params, int link_id)
 {
-       if (sdw->res->ops && sdw->res->ops->config_stream)
-               return sdw->res->ops->config_stream(sdw->res->arg,
+       struct sdw_intel_link_res *res = sdw->res;
+
+       if (res->ops && res->ops->config_stream && res->arg)
+               return res->ops->config_stream(res->arg,
                                substream, dai, hw_params, link_id);
 
        return -EIO;
@@ -649,6 +781,19 @@ intel_hw_free(struct snd_pcm_substream *substream, struct snd_soc_dai *dai)
        return ret;
 }
 
+static void intel_shutdown(struct snd_pcm_substream *substream,
+                          struct snd_soc_dai *dai)
+{
+       struct sdw_cdns_dma_data *dma;
+
+       dma = snd_soc_dai_get_dma_data(dai, substream);
+       if (!dma)
+               return;
+
+       snd_soc_dai_set_dma_data(dai, substream, NULL);
+       kfree(dma);
+}
+
 static int intel_pcm_set_sdw_stream(struct snd_soc_dai *dai,
                                    void *stream, int direction)
 {
@@ -664,14 +809,14 @@ static int intel_pdm_set_sdw_stream(struct snd_soc_dai *dai,
 static const struct snd_soc_dai_ops intel_pcm_dai_ops = {
        .hw_params = intel_hw_params,
        .hw_free = intel_hw_free,
-       .shutdown = sdw_cdns_shutdown,
+       .shutdown = intel_shutdown,
        .set_sdw_stream = intel_pcm_set_sdw_stream,
 };
 
 static const struct snd_soc_dai_ops intel_pdm_dai_ops = {
        .hw_params = intel_hw_params,
        .hw_free = intel_hw_free,
-       .shutdown = sdw_cdns_shutdown,
+       .shutdown = intel_shutdown,
        .set_sdw_stream = intel_pdm_set_sdw_stream,
 };
 
@@ -796,21 +941,44 @@ static int intel_register_dai(struct sdw_intel *sdw)
                                          dais, num_dai);
 }
 
+static int sdw_master_read_intel_prop(struct sdw_bus *bus)
+{
+       struct sdw_master_prop *prop = &bus->prop;
+       struct fwnode_handle *link;
+       char name[32];
+       u32 quirk_mask;
+
+       /* Find master handle */
+       snprintf(name, sizeof(name),
+                "mipi-sdw-link-%d-subproperties", bus->link_id);
+
+       link = device_get_named_child_node(bus->dev, name);
+       if (!link) {
+               dev_err(bus->dev, "Master node %s not found\n", name);
+               return -EIO;
+       }
+
+       fwnode_property_read_u32(link,
+                                "intel-sdw-ip-clock",
+                                &prop->mclk_freq);
+
+       fwnode_property_read_u32(link,
+                                "intel-quirk-mask",
+                                &quirk_mask);
+
+       if (quirk_mask & SDW_INTEL_QUIRK_MASK_BUS_DISABLE)
+               prop->hw_disabled = true;
+
+       return 0;
+}
+
 static int intel_prop_read(struct sdw_bus *bus)
 {
        /* Initialize with default handler to read all DisCo properties */
        sdw_master_read_prop(bus);
 
-       /* BIOS is not giving some values correctly. So, lets override them */
-       bus->prop.num_clk_freq = 1;
-       bus->prop.clk_freq = devm_kcalloc(bus->dev, bus->prop.num_clk_freq,
-                                         sizeof(*bus->prop.clk_freq),
-                                         GFP_KERNEL);
-       if (!bus->prop.clk_freq)
-               return -ENOMEM;
-
-       bus->prop.clk_freq[0] = bus->prop.max_clk_freq;
-       bus->prop.err_threshold = 5;
+       /* read Intel-specific properties */
+       sdw_master_read_intel_prop(bus);
 
        return 0;
 }
@@ -861,6 +1029,12 @@ static int intel_probe(struct platform_device *pdev)
                goto err_master_reg;
        }
 
+       if (sdw->cdns.bus.prop.hw_disabled) {
+               dev_info(&pdev->dev, "SoundWire master %d is disabled, ignoring\n",
+                        sdw->cdns.bus.link_id);
+               return 0;
+       }
+
        /* Initialize shim and controller */
        intel_link_power_up(sdw);
        intel_shim_init(sdw);
@@ -896,6 +1070,8 @@ static int intel_probe(struct platform_device *pdev)
                goto err_dai;
        }
 
+       intel_debugfs_init(sdw);
+
        return 0;
 
 err_dai:
@@ -912,8 +1088,11 @@ static int intel_remove(struct platform_device *pdev)
 
        sdw = platform_get_drvdata(pdev);
 
-       free_irq(sdw->res->irq, sdw);
-       snd_soc_unregister_component(sdw->cdns.dev);
+       if (!sdw->cdns.bus.prop.hw_disabled) {
+               intel_debugfs_exit(sdw);
+               free_irq(sdw->res->irq, sdw);
+               snd_soc_unregister_component(sdw->cdns.dev);
+       }
        sdw_delete_bus_master(&sdw->cdns.bus);
 
        return 0;