All of lore.kernel.org
 help / color / mirror / Atom feed
* [PATCH V3] ASoC: SAMSUNG: Add DT support for i2s
@ 2012-11-06  4:41 ` Padmavathi Venna
  0 siblings, 0 replies; 4+ messages in thread
From: Padmavathi Venna @ 2012-11-06  4:41 UTC (permalink / raw)
  To: padma.v, linux-arm-kernel, linux-samsung-soc, alsa-devel,
	devicetree-discuss
  Cc: sbkim73, ben-linux, kgene.kim

Add support for device based discovery.

Signed-off-by: Padmavathi Venna <padma.v@samsung.com>
---

Changes since V2:
	- Rebased on 3.7-rc3
	- Custom DT bindings are prefixed with samsung
	- As generic device tree DMA helpers not yet mainlined
	  I am still using custom dma bindings. So added a
	  priliminary statement regarding the same. I will rework
	  on my patch once generic DMA helpers are mainlined.

Chnages since V1:
	- Rebased on 3.6-rc6
	
 .../devicetree/bindings/sound/samsung-i2s.txt      |   69 ++++++
 sound/soc/samsung/dma.c                            |    1 +
 sound/soc/samsung/dma.h                            |    1 +
 sound/soc/samsung/i2s.c                            |  234 ++++++++++++++++----
 4 files changed, 256 insertions(+), 49 deletions(-)
 create mode 100644 Documentation/devicetree/bindings/sound/samsung-i2s.txt

diff --git a/Documentation/devicetree/bindings/sound/samsung-i2s.txt b/Documentation/devicetree/bindings/sound/samsung-i2s.txt
new file mode 100644
index 0000000..dca61a6
--- /dev/null
+++ b/Documentation/devicetree/bindings/sound/samsung-i2s.txt
@@ -0,0 +1,69 @@
+* Samsung I2S controller
+
+Required SoC Specific Properties:
+
+- compatible : "samsung,samsung-i2s"
+- reg: physical base address of the controller and length of memory mapped
+  region.
+
+[PRELIMINARY: the dma channel allocation will change once there are
+official DMA bindings]
+
+- tx-dma-channel-secondary: The dma channel specifier for secondary tx
+  operations. The format of the dma specifier depends on the dma
+  controller.
+- tx-dma-channel: The dma channel specifier for tx operations. The format of
+  the dma specifier depends on the dma controller.
+- rx-dma-channel: The dma channel specifier for rx operations. The format of
+  the dma specifier depends on the dma controller.
+
+Optional SoC Specific Properties:
+
+- samsung,supports-6ch: If the I2S Primary sound source has 5.1 Channel
+  support, this flag is enabled.
+- samsung,supports-rstclr: This flag should be set if I2S software reset bit
+  control is required. When this flag is set I2S software reset bit will be
+  enabled or disabled based on need.
+- samsung,supports-secdai:If I2S block has a secondary FIFO and internal DMA,
+  then this flag is enabled.
+- samsung,idma-addr: Internal DMA register base address of the audio
+  sub system(used in secondary sound source).
+
+Required Board Specific Properties:
+
+- gpios: The gpio specifier for data out,data in, LRCLK, CDCLK and SCLK
+  interface lines. The format of the gpio specifier depends on the gpio
+  controller.
+
+Aliases:
+
+- All the I2S controller nodes should be represented in the aliases node using
+  the following format 'i2s{n}' where n is a unique number for the alias.
+
+Example:
+
+- SoC Specific Portion:
+
+i2s@03830000 {
+	compatible = "samsung,samsung-i2s";
+	reg = <0x03830000 0x100>;
+	tx-dma-channel-secondary = <&pdma0 8>;
+	tx-dma-channel = <&pdma0 10>;
+	rx-dma-channel = <&pdma0 9>;
+	samsung,supports-6ch;
+	samsung,supports-rstclr;
+	samsung,supports-secdai;
+	samsung,idma-addr = <0x03000000>;
+};
+
+- Board Specific Portion:
+
+i2s_0: i2s@03830000 {
+	gpios = <&gpz 0 2 0 0>,
+		<&gpz 1 2 0 0>,
+		<&gpz 2 2 0 0>,
+		<&gpz 3 2 0 0>,
+		<&gpz 4 2 0 0>,
+		<&gpz 5 2 0 0>,
+		<&gpz 6 2 0 0>;
+};
diff --git a/sound/soc/samsung/dma.c b/sound/soc/samsung/dma.c
index b70964e..359708c 100644
--- a/sound/soc/samsung/dma.c
+++ b/sound/soc/samsung/dma.c
@@ -168,6 +168,7 @@ static int dma_hw_params(struct snd_pcm_substream *substream,
 		req.cap = (samsung_dma_has_circular() ?
 			DMA_CYCLIC : DMA_SLAVE);
 		req.client = prtd->params->client;
+		req.dt_dmach_prop = prtd->params->dma_prop;
 		config.direction =
 			(substream->stream == SNDRV_PCM_STREAM_PLAYBACK
 			? DMA_MEM_TO_DEV : DMA_DEV_TO_MEM);
diff --git a/sound/soc/samsung/dma.h b/sound/soc/samsung/dma.h
index 7d1ead7..2e60415 100644
--- a/sound/soc/samsung/dma.h
+++ b/sound/soc/samsung/dma.h
@@ -19,6 +19,7 @@ struct s3c_dma_params {
 	int dma_size;			/* Size of the DMA transfer */
 	unsigned ch;
 	struct samsung_dma_ops *ops;
+	struct property *dma_prop;
 };
 
 #endif
diff --git a/sound/soc/samsung/i2s.c b/sound/soc/samsung/i2s.c
index 40b00a1..f9d55e5 100644
--- a/sound/soc/samsung/i2s.c
+++ b/sound/soc/samsung/i2s.c
@@ -15,11 +15,15 @@
 #include <linux/clk.h>
 #include <linux/io.h>
 #include <linux/module.h>
+#include <linux/of.h>
+#include <linux/of_gpio.h>
 #include <linux/pm_runtime.h>
 
 #include <sound/soc.h>
 #include <sound/pcm_params.h>
 
+#include <mach/dma.h>
+
 #include <linux/platform_data/asoc-s3c.h>
 
 #include "dma.h"
@@ -49,8 +53,6 @@ struct i2s_dai {
 	struct clk *clk;
 	/* Clock for generating I2S signals */
 	struct clk *op_clk;
-	/* Array of clock names for op_clk */
-	const char **src_clk;
 	/* Pointer to the Primary_Fifo if this is Sec_Fifo, NULL otherwise */
 	struct i2s_dai *pri_dai;
 	/* Pointer to the Secondary_Fifo if it has one, NULL otherwise */
@@ -68,6 +70,8 @@ struct i2s_dai {
 	u32	suspend_i2smod;
 	u32	suspend_i2scon;
 	u32	suspend_i2spsr;
+	unsigned long gpios[7];	/* i2s gpio line numbers */
+	int	dev_id;		/* i2s dev id */
 };
 
 /* Lock for cross i/f checks */
@@ -385,6 +389,7 @@ static int i2s_set_sysclk(struct snd_soc_dai *dai,
 	struct i2s_dai *i2s = to_info(dai);
 	struct i2s_dai *other = i2s->pri_dai ? : i2s->sec_dai;
 	u32 mod = readl(i2s->addr + I2SMOD);
+	char clk_name[16];
 
 	switch (clk_id) {
 	case SAMSUNG_I2S_CDCLK:
@@ -432,8 +437,9 @@ static int i2s_set_sysclk(struct snd_soc_dai *dai,
 				}
 			}
 
+			sprintf(clk_name, "i2s_opclk%d", clk_id);
 			i2s->op_clk = clk_get(&i2s->pdev->dev,
-						i2s->src_clk[clk_id]);
+						clk_name);
 			clk_enable(i2s->op_clk);
 			i2s->rclk_srcrate = clk_get_rate(i2s->op_clk);
 
@@ -980,8 +986,9 @@ struct i2s_dai *i2s_alloc_dai(struct platform_device *pdev, bool sec)
 		i2s->i2s_dai_drv.capture.formats = SAMSUNG_I2S_FMTS;
 	} else {	/* Create a new platform_device for Secondary */
 		i2s->pdev = platform_device_register_resndata(NULL,
-				pdev->name, pdev->id + SAMSUNG_I2S_SECOFF,
-				NULL, 0, NULL, 0);
+				"samsung-i2s",
+				i2s->dev_id + SAMSUNG_I2S_SECOFF, NULL, 0,
+				NULL, 0);
 		if (IS_ERR(i2s->pdev))
 			return NULL;
 	}
@@ -992,49 +999,154 @@ struct i2s_dai *i2s_alloc_dai(struct platform_device *pdev, bool sec)
 	return i2s;
 }
 
+#ifdef CONFIG_OF
+static int samsung_i2s_parse_dt_gpio(struct i2s_dai *i2s)
+{
+	struct device *dev = &i2s->pdev->dev;
+	int index, gpio, ret;
+
+	for (index = 0; index < 7; index++) {
+		gpio = of_get_gpio(dev->of_node, index);
+		if (!gpio_is_valid(gpio)) {
+			dev_err(dev, "invalid gpio[%d]: %d\n", index, gpio);
+			goto free_gpio;
+		}
+
+		ret = gpio_request(gpio, dev_name(dev));
+		if (ret) {
+			dev_err(dev, "gpio [%d] request failed\n", gpio);
+			goto free_gpio;
+		}
+		i2s->gpios[index] = gpio;
+	}
+	return 0;
+
+free_gpio:
+	while (--index >= 0)
+		gpio_free(i2s->gpios[index]);
+	return -EINVAL;
+}
+
+static void samsung_i2s_dt_gpio_free(struct i2s_dai *i2s)
+{
+	unsigned int index;
+	for (index = 0; index < 7; index++)
+		gpio_free(i2s->gpios[index]);
+}
+#else
+static int samsung_i2s_parse_dt_gpio(struct i2s_dai *dai)
+{
+	return -EINVAL;
+}
+
+static void samsung_i2s_dt_gpio_free(struct i2s_dai *dai)
+{
+}
+
+#endif
+
 static __devinit int samsung_i2s_probe(struct platform_device *pdev)
 {
-	u32 dma_pl_chan, dma_cp_chan, dma_pl_sec_chan;
+	u32 dma_pl_chan, dma_cp_chan;
+	u32 dma_pl_sec_chan = 0;
 	struct i2s_dai *pri_dai, *sec_dai = NULL;
-	struct s3c_audio_pdata *i2s_pdata;
-	struct samsung_i2s *i2s_cfg;
+	struct s3c_audio_pdata *i2s_pdata = pdev->dev.platform_data;
+	struct samsung_i2s *i2s_cfg = NULL;
 	struct resource *res;
-	u32 regs_base, quirks;
-	int ret = 0;
+	u32 regs_base, quirks = 0, idma_addr = 0;
+	struct property *prop;
+	struct device_node *np = pdev->dev.of_node;
+	int ret = 0, id;
 
 	/* Call during Seconday interface registration */
-	if (pdev->id >= SAMSUNG_I2S_SECOFF) {
+	if (np) {
+		id = of_alias_get_id(np, "i2s");
+		if (id < 0) {
+			dev_err(&pdev->dev, "failed to get alias id:%d\n", id);
+			return id;
+		}
+	} else {
+		id = pdev->id;
+	}
+
+	if (id >= SAMSUNG_I2S_SECOFF) {
 		sec_dai = dev_get_drvdata(&pdev->dev);
 		snd_soc_register_dai(&sec_dai->pdev->dev,
 			&sec_dai->i2s_dai_drv);
 		return 0;
 	}
 
-	i2s_pdata = pdev->dev.platform_data;
-	if (i2s_pdata == NULL) {
-		dev_err(&pdev->dev, "Can't work without s3c_audio_pdata\n");
-		return -EINVAL;
+	pri_dai = i2s_alloc_dai(pdev, false);
+	if (!pri_dai) {
+		dev_err(&pdev->dev, "Unable to alloc I2S_pri\n");
+		return -ENOMEM;
 	}
 
-	res = platform_get_resource(pdev, IORESOURCE_DMA, 0);
-	if (!res) {
-		dev_err(&pdev->dev, "Unable to get I2S-TX dma resource\n");
-		return -ENXIO;
-	}
-	dma_pl_chan = res->start;
+	if (!np) {
+		res = platform_get_resource(pdev, IORESOURCE_DMA, 0);
+		if (!res) {
+			dev_err(&pdev->dev,
+				"Unable to get I2S-TX dma resource\n");
+			return -ENXIO;
+		}
+		dma_pl_chan = res->start;
 
-	res = platform_get_resource(pdev, IORESOURCE_DMA, 1);
-	if (!res) {
-		dev_err(&pdev->dev, "Unable to get I2S-RX dma resource\n");
-		return -ENXIO;
-	}
-	dma_cp_chan = res->start;
+		res = platform_get_resource(pdev, IORESOURCE_DMA, 1);
+		if (!res) {
+			dev_err(&pdev->dev,
+				"Unable to get I2S-RX dma resource\n");
+			return -ENXIO;
+		}
+		dma_cp_chan = res->start;
 
-	res = platform_get_resource(pdev, IORESOURCE_DMA, 2);
-	if (res)
-		dma_pl_sec_chan = res->start;
-	else
-		dma_pl_sec_chan = 0;
+		res = platform_get_resource(pdev, IORESOURCE_DMA, 2);
+		if (res)
+			dma_pl_sec_chan = res->start;
+
+		if (&i2s_pdata->type)
+			i2s_cfg = &i2s_pdata->type.i2s;
+
+		if (i2s_cfg) {
+			quirks = i2s_cfg->quirks;
+			idma_addr = i2s_cfg->idma_addr;
+		}
+	} else {
+		prop = of_find_property(np, "tx-dma-channel", NULL);
+		if (!prop) {
+			dev_err(&pdev->dev, "tx dma channel property not"\
+					"specified\n");
+			return -ENXIO;
+		}
+		dma_pl_chan = DMACH_DT_PROP;
+		pri_dai->dma_playback.dma_prop = prop;
+
+		prop = of_find_property(np, "rx-dma-channel", NULL);
+		if (!prop) {
+			dev_err(&pdev->dev, "tx dma channel property not"\
+					"specified\n");
+			return -ENXIO;
+		}
+		dma_cp_chan = DMACH_DT_PROP;
+		pri_dai->dma_capture.dma_prop = prop;
+
+		if (of_find_property(np, "samsung,supports-6ch", NULL))
+			quirks |= QUIRK_PRI_6CHAN;
+
+		if (of_find_property(np, "samsung,supports-secdai", NULL))
+			quirks |= QUIRK_SEC_DAI;
+
+		if (of_find_property(np, "samsung,supports-rstclr", NULL))
+			quirks |= QUIRK_NEED_RSTCLR;
+
+		if (of_property_read_u32(np, "samsung,idma-addr",
+					 &idma_addr)) {
+			if (quirks & QUIRK_SEC_DAI) {
+				dev_err(&pdev->dev, "idma address is not"\
+						"specified");
+				return -EINVAL;
+			}
+		}
+	}
 
 	res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
 	if (!res) {
@@ -1049,16 +1161,6 @@ static __devinit int samsung_i2s_probe(struct platform_device *pdev)
 	}
 	regs_base = res->start;
 
-	i2s_cfg = &i2s_pdata->type.i2s;
-	quirks = i2s_cfg->quirks;
-
-	pri_dai = i2s_alloc_dai(pdev, false);
-	if (!pri_dai) {
-		dev_err(&pdev->dev, "Unable to alloc I2S_pri\n");
-		ret = -ENOMEM;
-		goto err;
-	}
-
 	pri_dai->dma_playback.dma_addr = regs_base + I2STXD;
 	pri_dai->dma_capture.dma_addr = regs_base + I2SRXD;
 	pri_dai->dma_playback.client =
@@ -1067,11 +1169,11 @@ static __devinit int samsung_i2s_probe(struct platform_device *pdev)
 		(struct s3c2410_dma_client *)&pri_dai->dma_capture;
 	pri_dai->dma_playback.channel = dma_pl_chan;
 	pri_dai->dma_capture.channel = dma_cp_chan;
-	pri_dai->src_clk = i2s_cfg->src_clk;
 	pri_dai->dma_playback.dma_size = 4;
 	pri_dai->dma_capture.dma_size = 4;
 	pri_dai->base = regs_base;
 	pri_dai->quirks = quirks;
+	pri_dai->dev_id = id;
 
 	if (quirks & QUIRK_PRI_6CHAN)
 		pri_dai->i2s_dai_drv.playback.channels_max = 6;
@@ -1086,21 +1188,42 @@ static __devinit int samsung_i2s_probe(struct platform_device *pdev)
 		sec_dai->dma_playback.dma_addr = regs_base + I2STXDS;
 		sec_dai->dma_playback.client =
 			(struct s3c2410_dma_client *)&sec_dai->dma_playback;
+
+		if (np) {
+			prop = of_find_property(np, "tx-dma-channel-secondary",
+						NULL);
+			if (!prop) {
+				dev_err(&pdev->dev, "tx dma channel property"\
+						"not specified\n");
+				ret = -ENXIO;
+				goto err;
+			}
+			sec_dai->dma_playback.dma_prop = prop;
+		}
+
 		/* Use iDMA always if SysDMA not provided */
 		sec_dai->dma_playback.channel = dma_pl_sec_chan ? : -1;
-		sec_dai->src_clk = i2s_cfg->src_clk;
 		sec_dai->dma_playback.dma_size = 4;
 		sec_dai->base = regs_base;
 		sec_dai->quirks = quirks;
-		sec_dai->idma_playback.dma_addr = i2s_cfg->idma_addr;
+		sec_dai->dev_id = id;
+		sec_dai->idma_playback.dma_addr = idma_addr;
 		sec_dai->pri_dai = pri_dai;
 		pri_dai->sec_dai = sec_dai;
 	}
 
-	if (i2s_pdata->cfg_gpio && i2s_pdata->cfg_gpio(pdev)) {
-		dev_err(&pdev->dev, "Unable to configure gpio\n");
-		ret = -EINVAL;
-		goto err;
+	if (np) {
+		if (samsung_i2s_parse_dt_gpio(pri_dai)) {
+			dev_err(&pdev->dev, "Unable to configure gpio\n");
+			ret = -EINVAL;
+			goto err;
+		}
+	} else {
+		if (i2s_pdata->cfg_gpio && i2s_pdata->cfg_gpio(pdev)) {
+			dev_err(&pdev->dev, "Unable to configure gpio\n");
+			ret = -EINVAL;
+			goto err;
+		}
 	}
 
 	snd_soc_register_dai(&pri_dai->pdev->dev, &pri_dai->i2s_dai_drv);
@@ -1118,10 +1241,14 @@ static __devexit int samsung_i2s_remove(struct platform_device *pdev)
 {
 	struct i2s_dai *i2s, *other;
 	struct resource *res;
+	struct s3c_audio_pdata *i2s_pdata = pdev->dev.platform_data;
 
 	i2s = dev_get_drvdata(&pdev->dev);
 	other = i2s->pri_dai ? : i2s->sec_dai;
 
+	if (!i2s_pdata->cfg_gpio && pdev->dev.of_node)
+		samsung_i2s_dt_gpio_free(i2s->pri_dai);
+
 	if (other) {
 		other->pri_dai = NULL;
 		other->sec_dai = NULL;
@@ -1140,12 +1267,21 @@ static __devexit int samsung_i2s_remove(struct platform_device *pdev)
 	return 0;
 }
 
+#ifdef CONFIG_OF
+static const struct of_device_id exynos_i2s_match[] = {
+	{ .compatible = "samsung,samsung-i2s" },
+	{},
+};
+MODULE_DEVICE_TABLE(of, exynos_i2s_match);
+#endif
+
 static struct platform_driver samsung_i2s_driver = {
 	.probe  = samsung_i2s_probe,
 	.remove = __devexit_p(samsung_i2s_remove),
 	.driver = {
 		.name = "samsung-i2s",
 		.owner = THIS_MODULE,
+		.of_match_table = of_match_ptr(exynos_i2s_match),
 	},
 };
 
-- 
1.7.4.4

^ permalink raw reply related	[flat|nested] 4+ messages in thread

* [PATCH V3] ASoC: SAMSUNG: Add DT support for i2s
@ 2012-11-06  4:41 ` Padmavathi Venna
  0 siblings, 0 replies; 4+ messages in thread
From: Padmavathi Venna @ 2012-11-06  4:41 UTC (permalink / raw)
  To: linux-arm-kernel

Add support for device based discovery.

Signed-off-by: Padmavathi Venna <padma.v@samsung.com>
---

Changes since V2:
	- Rebased on 3.7-rc3
	- Custom DT bindings are prefixed with samsung
	- As generic device tree DMA helpers not yet mainlined
	  I am still using custom dma bindings. So added a
	  priliminary statement regarding the same. I will rework
	  on my patch once generic DMA helpers are mainlined.

Chnages since V1:
	- Rebased on 3.6-rc6
	
 .../devicetree/bindings/sound/samsung-i2s.txt      |   69 ++++++
 sound/soc/samsung/dma.c                            |    1 +
 sound/soc/samsung/dma.h                            |    1 +
 sound/soc/samsung/i2s.c                            |  234 ++++++++++++++++----
 4 files changed, 256 insertions(+), 49 deletions(-)
 create mode 100644 Documentation/devicetree/bindings/sound/samsung-i2s.txt

diff --git a/Documentation/devicetree/bindings/sound/samsung-i2s.txt b/Documentation/devicetree/bindings/sound/samsung-i2s.txt
new file mode 100644
index 0000000..dca61a6
--- /dev/null
+++ b/Documentation/devicetree/bindings/sound/samsung-i2s.txt
@@ -0,0 +1,69 @@
+* Samsung I2S controller
+
+Required SoC Specific Properties:
+
+- compatible : "samsung,samsung-i2s"
+- reg: physical base address of the controller and length of memory mapped
+  region.
+
+[PRELIMINARY: the dma channel allocation will change once there are
+official DMA bindings]
+
+- tx-dma-channel-secondary: The dma channel specifier for secondary tx
+  operations. The format of the dma specifier depends on the dma
+  controller.
+- tx-dma-channel: The dma channel specifier for tx operations. The format of
+  the dma specifier depends on the dma controller.
+- rx-dma-channel: The dma channel specifier for rx operations. The format of
+  the dma specifier depends on the dma controller.
+
+Optional SoC Specific Properties:
+
+- samsung,supports-6ch: If the I2S Primary sound source has 5.1 Channel
+  support, this flag is enabled.
+- samsung,supports-rstclr: This flag should be set if I2S software reset bit
+  control is required. When this flag is set I2S software reset bit will be
+  enabled or disabled based on need.
+- samsung,supports-secdai:If I2S block has a secondary FIFO and internal DMA,
+  then this flag is enabled.
+- samsung,idma-addr: Internal DMA register base address of the audio
+  sub system(used in secondary sound source).
+
+Required Board Specific Properties:
+
+- gpios: The gpio specifier for data out,data in, LRCLK, CDCLK and SCLK
+  interface lines. The format of the gpio specifier depends on the gpio
+  controller.
+
+Aliases:
+
+- All the I2S controller nodes should be represented in the aliases node using
+  the following format 'i2s{n}' where n is a unique number for the alias.
+
+Example:
+
+- SoC Specific Portion:
+
+i2s at 03830000 {
+	compatible = "samsung,samsung-i2s";
+	reg = <0x03830000 0x100>;
+	tx-dma-channel-secondary = <&pdma0 8>;
+	tx-dma-channel = <&pdma0 10>;
+	rx-dma-channel = <&pdma0 9>;
+	samsung,supports-6ch;
+	samsung,supports-rstclr;
+	samsung,supports-secdai;
+	samsung,idma-addr = <0x03000000>;
+};
+
+- Board Specific Portion:
+
+i2s_0: i2s at 03830000 {
+	gpios = <&gpz 0 2 0 0>,
+		<&gpz 1 2 0 0>,
+		<&gpz 2 2 0 0>,
+		<&gpz 3 2 0 0>,
+		<&gpz 4 2 0 0>,
+		<&gpz 5 2 0 0>,
+		<&gpz 6 2 0 0>;
+};
diff --git a/sound/soc/samsung/dma.c b/sound/soc/samsung/dma.c
index b70964e..359708c 100644
--- a/sound/soc/samsung/dma.c
+++ b/sound/soc/samsung/dma.c
@@ -168,6 +168,7 @@ static int dma_hw_params(struct snd_pcm_substream *substream,
 		req.cap = (samsung_dma_has_circular() ?
 			DMA_CYCLIC : DMA_SLAVE);
 		req.client = prtd->params->client;
+		req.dt_dmach_prop = prtd->params->dma_prop;
 		config.direction =
 			(substream->stream == SNDRV_PCM_STREAM_PLAYBACK
 			? DMA_MEM_TO_DEV : DMA_DEV_TO_MEM);
diff --git a/sound/soc/samsung/dma.h b/sound/soc/samsung/dma.h
index 7d1ead7..2e60415 100644
--- a/sound/soc/samsung/dma.h
+++ b/sound/soc/samsung/dma.h
@@ -19,6 +19,7 @@ struct s3c_dma_params {
 	int dma_size;			/* Size of the DMA transfer */
 	unsigned ch;
 	struct samsung_dma_ops *ops;
+	struct property *dma_prop;
 };
 
 #endif
diff --git a/sound/soc/samsung/i2s.c b/sound/soc/samsung/i2s.c
index 40b00a1..f9d55e5 100644
--- a/sound/soc/samsung/i2s.c
+++ b/sound/soc/samsung/i2s.c
@@ -15,11 +15,15 @@
 #include <linux/clk.h>
 #include <linux/io.h>
 #include <linux/module.h>
+#include <linux/of.h>
+#include <linux/of_gpio.h>
 #include <linux/pm_runtime.h>
 
 #include <sound/soc.h>
 #include <sound/pcm_params.h>
 
+#include <mach/dma.h>
+
 #include <linux/platform_data/asoc-s3c.h>
 
 #include "dma.h"
@@ -49,8 +53,6 @@ struct i2s_dai {
 	struct clk *clk;
 	/* Clock for generating I2S signals */
 	struct clk *op_clk;
-	/* Array of clock names for op_clk */
-	const char **src_clk;
 	/* Pointer to the Primary_Fifo if this is Sec_Fifo, NULL otherwise */
 	struct i2s_dai *pri_dai;
 	/* Pointer to the Secondary_Fifo if it has one, NULL otherwise */
@@ -68,6 +70,8 @@ struct i2s_dai {
 	u32	suspend_i2smod;
 	u32	suspend_i2scon;
 	u32	suspend_i2spsr;
+	unsigned long gpios[7];	/* i2s gpio line numbers */
+	int	dev_id;		/* i2s dev id */
 };
 
 /* Lock for cross i/f checks */
@@ -385,6 +389,7 @@ static int i2s_set_sysclk(struct snd_soc_dai *dai,
 	struct i2s_dai *i2s = to_info(dai);
 	struct i2s_dai *other = i2s->pri_dai ? : i2s->sec_dai;
 	u32 mod = readl(i2s->addr + I2SMOD);
+	char clk_name[16];
 
 	switch (clk_id) {
 	case SAMSUNG_I2S_CDCLK:
@@ -432,8 +437,9 @@ static int i2s_set_sysclk(struct snd_soc_dai *dai,
 				}
 			}
 
+			sprintf(clk_name, "i2s_opclk%d", clk_id);
 			i2s->op_clk = clk_get(&i2s->pdev->dev,
-						i2s->src_clk[clk_id]);
+						clk_name);
 			clk_enable(i2s->op_clk);
 			i2s->rclk_srcrate = clk_get_rate(i2s->op_clk);
 
@@ -980,8 +986,9 @@ struct i2s_dai *i2s_alloc_dai(struct platform_device *pdev, bool sec)
 		i2s->i2s_dai_drv.capture.formats = SAMSUNG_I2S_FMTS;
 	} else {	/* Create a new platform_device for Secondary */
 		i2s->pdev = platform_device_register_resndata(NULL,
-				pdev->name, pdev->id + SAMSUNG_I2S_SECOFF,
-				NULL, 0, NULL, 0);
+				"samsung-i2s",
+				i2s->dev_id + SAMSUNG_I2S_SECOFF, NULL, 0,
+				NULL, 0);
 		if (IS_ERR(i2s->pdev))
 			return NULL;
 	}
@@ -992,49 +999,154 @@ struct i2s_dai *i2s_alloc_dai(struct platform_device *pdev, bool sec)
 	return i2s;
 }
 
+#ifdef CONFIG_OF
+static int samsung_i2s_parse_dt_gpio(struct i2s_dai *i2s)
+{
+	struct device *dev = &i2s->pdev->dev;
+	int index, gpio, ret;
+
+	for (index = 0; index < 7; index++) {
+		gpio = of_get_gpio(dev->of_node, index);
+		if (!gpio_is_valid(gpio)) {
+			dev_err(dev, "invalid gpio[%d]: %d\n", index, gpio);
+			goto free_gpio;
+		}
+
+		ret = gpio_request(gpio, dev_name(dev));
+		if (ret) {
+			dev_err(dev, "gpio [%d] request failed\n", gpio);
+			goto free_gpio;
+		}
+		i2s->gpios[index] = gpio;
+	}
+	return 0;
+
+free_gpio:
+	while (--index >= 0)
+		gpio_free(i2s->gpios[index]);
+	return -EINVAL;
+}
+
+static void samsung_i2s_dt_gpio_free(struct i2s_dai *i2s)
+{
+	unsigned int index;
+	for (index = 0; index < 7; index++)
+		gpio_free(i2s->gpios[index]);
+}
+#else
+static int samsung_i2s_parse_dt_gpio(struct i2s_dai *dai)
+{
+	return -EINVAL;
+}
+
+static void samsung_i2s_dt_gpio_free(struct i2s_dai *dai)
+{
+}
+
+#endif
+
 static __devinit int samsung_i2s_probe(struct platform_device *pdev)
 {
-	u32 dma_pl_chan, dma_cp_chan, dma_pl_sec_chan;
+	u32 dma_pl_chan, dma_cp_chan;
+	u32 dma_pl_sec_chan = 0;
 	struct i2s_dai *pri_dai, *sec_dai = NULL;
-	struct s3c_audio_pdata *i2s_pdata;
-	struct samsung_i2s *i2s_cfg;
+	struct s3c_audio_pdata *i2s_pdata = pdev->dev.platform_data;
+	struct samsung_i2s *i2s_cfg = NULL;
 	struct resource *res;
-	u32 regs_base, quirks;
-	int ret = 0;
+	u32 regs_base, quirks = 0, idma_addr = 0;
+	struct property *prop;
+	struct device_node *np = pdev->dev.of_node;
+	int ret = 0, id;
 
 	/* Call during Seconday interface registration */
-	if (pdev->id >= SAMSUNG_I2S_SECOFF) {
+	if (np) {
+		id = of_alias_get_id(np, "i2s");
+		if (id < 0) {
+			dev_err(&pdev->dev, "failed to get alias id:%d\n", id);
+			return id;
+		}
+	} else {
+		id = pdev->id;
+	}
+
+	if (id >= SAMSUNG_I2S_SECOFF) {
 		sec_dai = dev_get_drvdata(&pdev->dev);
 		snd_soc_register_dai(&sec_dai->pdev->dev,
 			&sec_dai->i2s_dai_drv);
 		return 0;
 	}
 
-	i2s_pdata = pdev->dev.platform_data;
-	if (i2s_pdata == NULL) {
-		dev_err(&pdev->dev, "Can't work without s3c_audio_pdata\n");
-		return -EINVAL;
+	pri_dai = i2s_alloc_dai(pdev, false);
+	if (!pri_dai) {
+		dev_err(&pdev->dev, "Unable to alloc I2S_pri\n");
+		return -ENOMEM;
 	}
 
-	res = platform_get_resource(pdev, IORESOURCE_DMA, 0);
-	if (!res) {
-		dev_err(&pdev->dev, "Unable to get I2S-TX dma resource\n");
-		return -ENXIO;
-	}
-	dma_pl_chan = res->start;
+	if (!np) {
+		res = platform_get_resource(pdev, IORESOURCE_DMA, 0);
+		if (!res) {
+			dev_err(&pdev->dev,
+				"Unable to get I2S-TX dma resource\n");
+			return -ENXIO;
+		}
+		dma_pl_chan = res->start;
 
-	res = platform_get_resource(pdev, IORESOURCE_DMA, 1);
-	if (!res) {
-		dev_err(&pdev->dev, "Unable to get I2S-RX dma resource\n");
-		return -ENXIO;
-	}
-	dma_cp_chan = res->start;
+		res = platform_get_resource(pdev, IORESOURCE_DMA, 1);
+		if (!res) {
+			dev_err(&pdev->dev,
+				"Unable to get I2S-RX dma resource\n");
+			return -ENXIO;
+		}
+		dma_cp_chan = res->start;
 
-	res = platform_get_resource(pdev, IORESOURCE_DMA, 2);
-	if (res)
-		dma_pl_sec_chan = res->start;
-	else
-		dma_pl_sec_chan = 0;
+		res = platform_get_resource(pdev, IORESOURCE_DMA, 2);
+		if (res)
+			dma_pl_sec_chan = res->start;
+
+		if (&i2s_pdata->type)
+			i2s_cfg = &i2s_pdata->type.i2s;
+
+		if (i2s_cfg) {
+			quirks = i2s_cfg->quirks;
+			idma_addr = i2s_cfg->idma_addr;
+		}
+	} else {
+		prop = of_find_property(np, "tx-dma-channel", NULL);
+		if (!prop) {
+			dev_err(&pdev->dev, "tx dma channel property not"\
+					"specified\n");
+			return -ENXIO;
+		}
+		dma_pl_chan = DMACH_DT_PROP;
+		pri_dai->dma_playback.dma_prop = prop;
+
+		prop = of_find_property(np, "rx-dma-channel", NULL);
+		if (!prop) {
+			dev_err(&pdev->dev, "tx dma channel property not"\
+					"specified\n");
+			return -ENXIO;
+		}
+		dma_cp_chan = DMACH_DT_PROP;
+		pri_dai->dma_capture.dma_prop = prop;
+
+		if (of_find_property(np, "samsung,supports-6ch", NULL))
+			quirks |= QUIRK_PRI_6CHAN;
+
+		if (of_find_property(np, "samsung,supports-secdai", NULL))
+			quirks |= QUIRK_SEC_DAI;
+
+		if (of_find_property(np, "samsung,supports-rstclr", NULL))
+			quirks |= QUIRK_NEED_RSTCLR;
+
+		if (of_property_read_u32(np, "samsung,idma-addr",
+					 &idma_addr)) {
+			if (quirks & QUIRK_SEC_DAI) {
+				dev_err(&pdev->dev, "idma address is not"\
+						"specified");
+				return -EINVAL;
+			}
+		}
+	}
 
 	res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
 	if (!res) {
@@ -1049,16 +1161,6 @@ static __devinit int samsung_i2s_probe(struct platform_device *pdev)
 	}
 	regs_base = res->start;
 
-	i2s_cfg = &i2s_pdata->type.i2s;
-	quirks = i2s_cfg->quirks;
-
-	pri_dai = i2s_alloc_dai(pdev, false);
-	if (!pri_dai) {
-		dev_err(&pdev->dev, "Unable to alloc I2S_pri\n");
-		ret = -ENOMEM;
-		goto err;
-	}
-
 	pri_dai->dma_playback.dma_addr = regs_base + I2STXD;
 	pri_dai->dma_capture.dma_addr = regs_base + I2SRXD;
 	pri_dai->dma_playback.client =
@@ -1067,11 +1169,11 @@ static __devinit int samsung_i2s_probe(struct platform_device *pdev)
 		(struct s3c2410_dma_client *)&pri_dai->dma_capture;
 	pri_dai->dma_playback.channel = dma_pl_chan;
 	pri_dai->dma_capture.channel = dma_cp_chan;
-	pri_dai->src_clk = i2s_cfg->src_clk;
 	pri_dai->dma_playback.dma_size = 4;
 	pri_dai->dma_capture.dma_size = 4;
 	pri_dai->base = regs_base;
 	pri_dai->quirks = quirks;
+	pri_dai->dev_id = id;
 
 	if (quirks & QUIRK_PRI_6CHAN)
 		pri_dai->i2s_dai_drv.playback.channels_max = 6;
@@ -1086,21 +1188,42 @@ static __devinit int samsung_i2s_probe(struct platform_device *pdev)
 		sec_dai->dma_playback.dma_addr = regs_base + I2STXDS;
 		sec_dai->dma_playback.client =
 			(struct s3c2410_dma_client *)&sec_dai->dma_playback;
+
+		if (np) {
+			prop = of_find_property(np, "tx-dma-channel-secondary",
+						NULL);
+			if (!prop) {
+				dev_err(&pdev->dev, "tx dma channel property"\
+						"not specified\n");
+				ret = -ENXIO;
+				goto err;
+			}
+			sec_dai->dma_playback.dma_prop = prop;
+		}
+
 		/* Use iDMA always if SysDMA not provided */
 		sec_dai->dma_playback.channel = dma_pl_sec_chan ? : -1;
-		sec_dai->src_clk = i2s_cfg->src_clk;
 		sec_dai->dma_playback.dma_size = 4;
 		sec_dai->base = regs_base;
 		sec_dai->quirks = quirks;
-		sec_dai->idma_playback.dma_addr = i2s_cfg->idma_addr;
+		sec_dai->dev_id = id;
+		sec_dai->idma_playback.dma_addr = idma_addr;
 		sec_dai->pri_dai = pri_dai;
 		pri_dai->sec_dai = sec_dai;
 	}
 
-	if (i2s_pdata->cfg_gpio && i2s_pdata->cfg_gpio(pdev)) {
-		dev_err(&pdev->dev, "Unable to configure gpio\n");
-		ret = -EINVAL;
-		goto err;
+	if (np) {
+		if (samsung_i2s_parse_dt_gpio(pri_dai)) {
+			dev_err(&pdev->dev, "Unable to configure gpio\n");
+			ret = -EINVAL;
+			goto err;
+		}
+	} else {
+		if (i2s_pdata->cfg_gpio && i2s_pdata->cfg_gpio(pdev)) {
+			dev_err(&pdev->dev, "Unable to configure gpio\n");
+			ret = -EINVAL;
+			goto err;
+		}
 	}
 
 	snd_soc_register_dai(&pri_dai->pdev->dev, &pri_dai->i2s_dai_drv);
@@ -1118,10 +1241,14 @@ static __devexit int samsung_i2s_remove(struct platform_device *pdev)
 {
 	struct i2s_dai *i2s, *other;
 	struct resource *res;
+	struct s3c_audio_pdata *i2s_pdata = pdev->dev.platform_data;
 
 	i2s = dev_get_drvdata(&pdev->dev);
 	other = i2s->pri_dai ? : i2s->sec_dai;
 
+	if (!i2s_pdata->cfg_gpio && pdev->dev.of_node)
+		samsung_i2s_dt_gpio_free(i2s->pri_dai);
+
 	if (other) {
 		other->pri_dai = NULL;
 		other->sec_dai = NULL;
@@ -1140,12 +1267,21 @@ static __devexit int samsung_i2s_remove(struct platform_device *pdev)
 	return 0;
 }
 
+#ifdef CONFIG_OF
+static const struct of_device_id exynos_i2s_match[] = {
+	{ .compatible = "samsung,samsung-i2s" },
+	{},
+};
+MODULE_DEVICE_TABLE(of, exynos_i2s_match);
+#endif
+
 static struct platform_driver samsung_i2s_driver = {
 	.probe  = samsung_i2s_probe,
 	.remove = __devexit_p(samsung_i2s_remove),
 	.driver = {
 		.name = "samsung-i2s",
 		.owner = THIS_MODULE,
+		.of_match_table = of_match_ptr(exynos_i2s_match),
 	},
 };
 
-- 
1.7.4.4

^ permalink raw reply related	[flat|nested] 4+ messages in thread

* Re: [PATCH V3] ASoC: SAMSUNG: Add DT support for i2s
  2012-11-06  4:41 ` Padmavathi Venna
@ 2012-11-23 11:13   ` Padma Venkat
  -1 siblings, 0 replies; 4+ messages in thread
From: Padma Venkat @ 2012-11-23 11:13 UTC (permalink / raw)
  To: Padmavathi Venna
  Cc: linux-arm-kernel, linux-samsung-soc, alsa-devel,
	devicetree-discuss, sbkim73, ben-linux, kgene.kim, Mark Brown

cc'ing Mark Brown.

On Tue, Nov 6, 2012 at 10:11 AM, Padmavathi Venna <padma.v@samsung.com> wrote:
> Add support for device based discovery.
>
> Signed-off-by: Padmavathi Venna <padma.v@samsung.com>
> ---
>
> Changes since V2:
>         - Rebased on 3.7-rc3
>         - Custom DT bindings are prefixed with samsung
>         - As generic device tree DMA helpers not yet mainlined
>           I am still using custom dma bindings. So added a
>           priliminary statement regarding the same. I will rework
>           on my patch once generic DMA helpers are mainlined.
>
> Chnages since V1:
>         - Rebased on 3.6-rc6
>
>  .../devicetree/bindings/sound/samsung-i2s.txt      |   69 ++++++
>  sound/soc/samsung/dma.c                            |    1 +
>  sound/soc/samsung/dma.h                            |    1 +
>  sound/soc/samsung/i2s.c                            |  234 ++++++++++++++++----
>  4 files changed, 256 insertions(+), 49 deletions(-)
>  create mode 100644 Documentation/devicetree/bindings/sound/samsung-i2s.txt
>
> diff --git a/Documentation/devicetree/bindings/sound/samsung-i2s.txt b/Documentation/devicetree/bindings/sound/samsung-i2s.txt
> new file mode 100644
> index 0000000..dca61a6
> --- /dev/null
> +++ b/Documentation/devicetree/bindings/sound/samsung-i2s.txt
> @@ -0,0 +1,69 @@
> +* Samsung I2S controller
> +
> +Required SoC Specific Properties:
> +
> +- compatible : "samsung,samsung-i2s"
> +- reg: physical base address of the controller and length of memory mapped
> +  region.
> +
> +[PRELIMINARY: the dma channel allocation will change once there are
> +official DMA bindings]
> +
> +- tx-dma-channel-secondary: The dma channel specifier for secondary tx
> +  operations. The format of the dma specifier depends on the dma
> +  controller.
> +- tx-dma-channel: The dma channel specifier for tx operations. The format of
> +  the dma specifier depends on the dma controller.
> +- rx-dma-channel: The dma channel specifier for rx operations. The format of
> +  the dma specifier depends on the dma controller.
> +
> +Optional SoC Specific Properties:
> +
> +- samsung,supports-6ch: If the I2S Primary sound source has 5.1 Channel
> +  support, this flag is enabled.
> +- samsung,supports-rstclr: This flag should be set if I2S software reset bit
> +  control is required. When this flag is set I2S software reset bit will be
> +  enabled or disabled based on need.
> +- samsung,supports-secdai:If I2S block has a secondary FIFO and internal DMA,
> +  then this flag is enabled.
> +- samsung,idma-addr: Internal DMA register base address of the audio
> +  sub system(used in secondary sound source).
> +
> +Required Board Specific Properties:
> +
> +- gpios: The gpio specifier for data out,data in, LRCLK, CDCLK and SCLK
> +  interface lines. The format of the gpio specifier depends on the gpio
> +  controller.
> +
> +Aliases:
> +
> +- All the I2S controller nodes should be represented in the aliases node using
> +  the following format 'i2s{n}' where n is a unique number for the alias.
> +
> +Example:
> +
> +- SoC Specific Portion:
> +
> +i2s@03830000 {
> +       compatible = "samsung,samsung-i2s";
> +       reg = <0x03830000 0x100>;
> +       tx-dma-channel-secondary = <&pdma0 8>;
> +       tx-dma-channel = <&pdma0 10>;
> +       rx-dma-channel = <&pdma0 9>;
> +       samsung,supports-6ch;
> +       samsung,supports-rstclr;
> +       samsung,supports-secdai;
> +       samsung,idma-addr = <0x03000000>;
> +};
> +
> +- Board Specific Portion:
> +
> +i2s_0: i2s@03830000 {
> +       gpios = <&gpz 0 2 0 0>,
> +               <&gpz 1 2 0 0>,
> +               <&gpz 2 2 0 0>,
> +               <&gpz 3 2 0 0>,
> +               <&gpz 4 2 0 0>,
> +               <&gpz 5 2 0 0>,
> +               <&gpz 6 2 0 0>;
> +};
> diff --git a/sound/soc/samsung/dma.c b/sound/soc/samsung/dma.c
> index b70964e..359708c 100644
> --- a/sound/soc/samsung/dma.c
> +++ b/sound/soc/samsung/dma.c
> @@ -168,6 +168,7 @@ static int dma_hw_params(struct snd_pcm_substream *substream,
>                 req.cap = (samsung_dma_has_circular() ?
>                         DMA_CYCLIC : DMA_SLAVE);
>                 req.client = prtd->params->client;
> +               req.dt_dmach_prop = prtd->params->dma_prop;
>                 config.direction =
>                         (substream->stream == SNDRV_PCM_STREAM_PLAYBACK
>                         ? DMA_MEM_TO_DEV : DMA_DEV_TO_MEM);
> diff --git a/sound/soc/samsung/dma.h b/sound/soc/samsung/dma.h
> index 7d1ead7..2e60415 100644
> --- a/sound/soc/samsung/dma.h
> +++ b/sound/soc/samsung/dma.h
> @@ -19,6 +19,7 @@ struct s3c_dma_params {
>         int dma_size;                   /* Size of the DMA transfer */
>         unsigned ch;
>         struct samsung_dma_ops *ops;
> +       struct property *dma_prop;
>  };
>
>  #endif
> diff --git a/sound/soc/samsung/i2s.c b/sound/soc/samsung/i2s.c
> index 40b00a1..f9d55e5 100644
> --- a/sound/soc/samsung/i2s.c
> +++ b/sound/soc/samsung/i2s.c
> @@ -15,11 +15,15 @@
>  #include <linux/clk.h>
>  #include <linux/io.h>
>  #include <linux/module.h>
> +#include <linux/of.h>
> +#include <linux/of_gpio.h>
>  #include <linux/pm_runtime.h>
>
>  #include <sound/soc.h>
>  #include <sound/pcm_params.h>
>
> +#include <mach/dma.h>
> +
>  #include <linux/platform_data/asoc-s3c.h>
>
>  #include "dma.h"
> @@ -49,8 +53,6 @@ struct i2s_dai {
>         struct clk *clk;
>         /* Clock for generating I2S signals */
>         struct clk *op_clk;
> -       /* Array of clock names for op_clk */
> -       const char **src_clk;
>         /* Pointer to the Primary_Fifo if this is Sec_Fifo, NULL otherwise */
>         struct i2s_dai *pri_dai;
>         /* Pointer to the Secondary_Fifo if it has one, NULL otherwise */
> @@ -68,6 +70,8 @@ struct i2s_dai {
>         u32     suspend_i2smod;
>         u32     suspend_i2scon;
>         u32     suspend_i2spsr;
> +       unsigned long gpios[7]; /* i2s gpio line numbers */
> +       int     dev_id;         /* i2s dev id */
>  };
>
>  /* Lock for cross i/f checks */
> @@ -385,6 +389,7 @@ static int i2s_set_sysclk(struct snd_soc_dai *dai,
>         struct i2s_dai *i2s = to_info(dai);
>         struct i2s_dai *other = i2s->pri_dai ? : i2s->sec_dai;
>         u32 mod = readl(i2s->addr + I2SMOD);
> +       char clk_name[16];
>
>         switch (clk_id) {
>         case SAMSUNG_I2S_CDCLK:
> @@ -432,8 +437,9 @@ static int i2s_set_sysclk(struct snd_soc_dai *dai,
>                                 }
>                         }
>
> +                       sprintf(clk_name, "i2s_opclk%d", clk_id);
>                         i2s->op_clk = clk_get(&i2s->pdev->dev,
> -                                               i2s->src_clk[clk_id]);
> +                                               clk_name);
>                         clk_enable(i2s->op_clk);
>                         i2s->rclk_srcrate = clk_get_rate(i2s->op_clk);
>
> @@ -980,8 +986,9 @@ struct i2s_dai *i2s_alloc_dai(struct platform_device *pdev, bool sec)
>                 i2s->i2s_dai_drv.capture.formats = SAMSUNG_I2S_FMTS;
>         } else {        /* Create a new platform_device for Secondary */
>                 i2s->pdev = platform_device_register_resndata(NULL,
> -                               pdev->name, pdev->id + SAMSUNG_I2S_SECOFF,
> -                               NULL, 0, NULL, 0);
> +                               "samsung-i2s",
> +                               i2s->dev_id + SAMSUNG_I2S_SECOFF, NULL, 0,
> +                               NULL, 0);
>                 if (IS_ERR(i2s->pdev))
>                         return NULL;
>         }
> @@ -992,49 +999,154 @@ struct i2s_dai *i2s_alloc_dai(struct platform_device *pdev, bool sec)
>         return i2s;
>  }
>
> +#ifdef CONFIG_OF
> +static int samsung_i2s_parse_dt_gpio(struct i2s_dai *i2s)
> +{
> +       struct device *dev = &i2s->pdev->dev;
> +       int index, gpio, ret;
> +
> +       for (index = 0; index < 7; index++) {
> +               gpio = of_get_gpio(dev->of_node, index);
> +               if (!gpio_is_valid(gpio)) {
> +                       dev_err(dev, "invalid gpio[%d]: %d\n", index, gpio);
> +                       goto free_gpio;
> +               }
> +
> +               ret = gpio_request(gpio, dev_name(dev));
> +               if (ret) {
> +                       dev_err(dev, "gpio [%d] request failed\n", gpio);
> +                       goto free_gpio;
> +               }
> +               i2s->gpios[index] = gpio;
> +       }
> +       return 0;
> +
> +free_gpio:
> +       while (--index >= 0)
> +               gpio_free(i2s->gpios[index]);
> +       return -EINVAL;
> +}
> +
> +static void samsung_i2s_dt_gpio_free(struct i2s_dai *i2s)
> +{
> +       unsigned int index;
> +       for (index = 0; index < 7; index++)
> +               gpio_free(i2s->gpios[index]);
> +}
> +#else
> +static int samsung_i2s_parse_dt_gpio(struct i2s_dai *dai)
> +{
> +       return -EINVAL;
> +}
> +
> +static void samsung_i2s_dt_gpio_free(struct i2s_dai *dai)
> +{
> +}
> +
> +#endif
> +
>  static __devinit int samsung_i2s_probe(struct platform_device *pdev)
>  {
> -       u32 dma_pl_chan, dma_cp_chan, dma_pl_sec_chan;
> +       u32 dma_pl_chan, dma_cp_chan;
> +       u32 dma_pl_sec_chan = 0;
>         struct i2s_dai *pri_dai, *sec_dai = NULL;
> -       struct s3c_audio_pdata *i2s_pdata;
> -       struct samsung_i2s *i2s_cfg;
> +       struct s3c_audio_pdata *i2s_pdata = pdev->dev.platform_data;
> +       struct samsung_i2s *i2s_cfg = NULL;
>         struct resource *res;
> -       u32 regs_base, quirks;
> -       int ret = 0;
> +       u32 regs_base, quirks = 0, idma_addr = 0;
> +       struct property *prop;
> +       struct device_node *np = pdev->dev.of_node;
> +       int ret = 0, id;
>
>         /* Call during Seconday interface registration */
> -       if (pdev->id >= SAMSUNG_I2S_SECOFF) {
> +       if (np) {
> +               id = of_alias_get_id(np, "i2s");
> +               if (id < 0) {
> +                       dev_err(&pdev->dev, "failed to get alias id:%d\n", id);
> +                       return id;
> +               }
> +       } else {
> +               id = pdev->id;
> +       }
> +
> +       if (id >= SAMSUNG_I2S_SECOFF) {
>                 sec_dai = dev_get_drvdata(&pdev->dev);
>                 snd_soc_register_dai(&sec_dai->pdev->dev,
>                         &sec_dai->i2s_dai_drv);
>                 return 0;
>         }
>
> -       i2s_pdata = pdev->dev.platform_data;
> -       if (i2s_pdata == NULL) {
> -               dev_err(&pdev->dev, "Can't work without s3c_audio_pdata\n");
> -               return -EINVAL;
> +       pri_dai = i2s_alloc_dai(pdev, false);
> +       if (!pri_dai) {
> +               dev_err(&pdev->dev, "Unable to alloc I2S_pri\n");
> +               return -ENOMEM;
>         }
>
> -       res = platform_get_resource(pdev, IORESOURCE_DMA, 0);
> -       if (!res) {
> -               dev_err(&pdev->dev, "Unable to get I2S-TX dma resource\n");
> -               return -ENXIO;
> -       }
> -       dma_pl_chan = res->start;
> +       if (!np) {
> +               res = platform_get_resource(pdev, IORESOURCE_DMA, 0);
> +               if (!res) {
> +                       dev_err(&pdev->dev,
> +                               "Unable to get I2S-TX dma resource\n");
> +                       return -ENXIO;
> +               }
> +               dma_pl_chan = res->start;
>
> -       res = platform_get_resource(pdev, IORESOURCE_DMA, 1);
> -       if (!res) {
> -               dev_err(&pdev->dev, "Unable to get I2S-RX dma resource\n");
> -               return -ENXIO;
> -       }
> -       dma_cp_chan = res->start;
> +               res = platform_get_resource(pdev, IORESOURCE_DMA, 1);
> +               if (!res) {
> +                       dev_err(&pdev->dev,
> +                               "Unable to get I2S-RX dma resource\n");
> +                       return -ENXIO;
> +               }
> +               dma_cp_chan = res->start;
>
> -       res = platform_get_resource(pdev, IORESOURCE_DMA, 2);
> -       if (res)
> -               dma_pl_sec_chan = res->start;
> -       else
> -               dma_pl_sec_chan = 0;
> +               res = platform_get_resource(pdev, IORESOURCE_DMA, 2);
> +               if (res)
> +                       dma_pl_sec_chan = res->start;
> +
> +               if (&i2s_pdata->type)
> +                       i2s_cfg = &i2s_pdata->type.i2s;
> +
> +               if (i2s_cfg) {
> +                       quirks = i2s_cfg->quirks;
> +                       idma_addr = i2s_cfg->idma_addr;
> +               }
> +       } else {
> +               prop = of_find_property(np, "tx-dma-channel", NULL);
> +               if (!prop) {
> +                       dev_err(&pdev->dev, "tx dma channel property not"\
> +                                       "specified\n");
> +                       return -ENXIO;
> +               }
> +               dma_pl_chan = DMACH_DT_PROP;
> +               pri_dai->dma_playback.dma_prop = prop;
> +
> +               prop = of_find_property(np, "rx-dma-channel", NULL);
> +               if (!prop) {
> +                       dev_err(&pdev->dev, "tx dma channel property not"\
> +                                       "specified\n");
> +                       return -ENXIO;
> +               }
> +               dma_cp_chan = DMACH_DT_PROP;
> +               pri_dai->dma_capture.dma_prop = prop;
> +
> +               if (of_find_property(np, "samsung,supports-6ch", NULL))
> +                       quirks |= QUIRK_PRI_6CHAN;
> +
> +               if (of_find_property(np, "samsung,supports-secdai", NULL))
> +                       quirks |= QUIRK_SEC_DAI;
> +
> +               if (of_find_property(np, "samsung,supports-rstclr", NULL))
> +                       quirks |= QUIRK_NEED_RSTCLR;
> +
> +               if (of_property_read_u32(np, "samsung,idma-addr",
> +                                        &idma_addr)) {
> +                       if (quirks & QUIRK_SEC_DAI) {
> +                               dev_err(&pdev->dev, "idma address is not"\
> +                                               "specified");
> +                               return -EINVAL;
> +                       }
> +               }
> +       }
>
>         res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
>         if (!res) {
> @@ -1049,16 +1161,6 @@ static __devinit int samsung_i2s_probe(struct platform_device *pdev)
>         }
>         regs_base = res->start;
>
> -       i2s_cfg = &i2s_pdata->type.i2s;
> -       quirks = i2s_cfg->quirks;
> -
> -       pri_dai = i2s_alloc_dai(pdev, false);
> -       if (!pri_dai) {
> -               dev_err(&pdev->dev, "Unable to alloc I2S_pri\n");
> -               ret = -ENOMEM;
> -               goto err;
> -       }
> -
>         pri_dai->dma_playback.dma_addr = regs_base + I2STXD;
>         pri_dai->dma_capture.dma_addr = regs_base + I2SRXD;
>         pri_dai->dma_playback.client =
> @@ -1067,11 +1169,11 @@ static __devinit int samsung_i2s_probe(struct platform_device *pdev)
>                 (struct s3c2410_dma_client *)&pri_dai->dma_capture;
>         pri_dai->dma_playback.channel = dma_pl_chan;
>         pri_dai->dma_capture.channel = dma_cp_chan;
> -       pri_dai->src_clk = i2s_cfg->src_clk;
>         pri_dai->dma_playback.dma_size = 4;
>         pri_dai->dma_capture.dma_size = 4;
>         pri_dai->base = regs_base;
>         pri_dai->quirks = quirks;
> +       pri_dai->dev_id = id;
>
>         if (quirks & QUIRK_PRI_6CHAN)
>                 pri_dai->i2s_dai_drv.playback.channels_max = 6;
> @@ -1086,21 +1188,42 @@ static __devinit int samsung_i2s_probe(struct platform_device *pdev)
>                 sec_dai->dma_playback.dma_addr = regs_base + I2STXDS;
>                 sec_dai->dma_playback.client =
>                         (struct s3c2410_dma_client *)&sec_dai->dma_playback;
> +
> +               if (np) {
> +                       prop = of_find_property(np, "tx-dma-channel-secondary",
> +                                               NULL);
> +                       if (!prop) {
> +                               dev_err(&pdev->dev, "tx dma channel property"\
> +                                               "not specified\n");
> +                               ret = -ENXIO;
> +                               goto err;
> +                       }
> +                       sec_dai->dma_playback.dma_prop = prop;
> +               }
> +
>                 /* Use iDMA always if SysDMA not provided */
>                 sec_dai->dma_playback.channel = dma_pl_sec_chan ? : -1;
> -               sec_dai->src_clk = i2s_cfg->src_clk;
>                 sec_dai->dma_playback.dma_size = 4;
>                 sec_dai->base = regs_base;
>                 sec_dai->quirks = quirks;
> -               sec_dai->idma_playback.dma_addr = i2s_cfg->idma_addr;
> +               sec_dai->dev_id = id;
> +               sec_dai->idma_playback.dma_addr = idma_addr;
>                 sec_dai->pri_dai = pri_dai;
>                 pri_dai->sec_dai = sec_dai;
>         }
>
> -       if (i2s_pdata->cfg_gpio && i2s_pdata->cfg_gpio(pdev)) {
> -               dev_err(&pdev->dev, "Unable to configure gpio\n");
> -               ret = -EINVAL;
> -               goto err;
> +       if (np) {
> +               if (samsung_i2s_parse_dt_gpio(pri_dai)) {
> +                       dev_err(&pdev->dev, "Unable to configure gpio\n");
> +                       ret = -EINVAL;
> +                       goto err;
> +               }
> +       } else {
> +               if (i2s_pdata->cfg_gpio && i2s_pdata->cfg_gpio(pdev)) {
> +                       dev_err(&pdev->dev, "Unable to configure gpio\n");
> +                       ret = -EINVAL;
> +                       goto err;
> +               }
>         }
>
>         snd_soc_register_dai(&pri_dai->pdev->dev, &pri_dai->i2s_dai_drv);
> @@ -1118,10 +1241,14 @@ static __devexit int samsung_i2s_remove(struct platform_device *pdev)
>  {
>         struct i2s_dai *i2s, *other;
>         struct resource *res;
> +       struct s3c_audio_pdata *i2s_pdata = pdev->dev.platform_data;
>
>         i2s = dev_get_drvdata(&pdev->dev);
>         other = i2s->pri_dai ? : i2s->sec_dai;
>
> +       if (!i2s_pdata->cfg_gpio && pdev->dev.of_node)
> +               samsung_i2s_dt_gpio_free(i2s->pri_dai);
> +
>         if (other) {
>                 other->pri_dai = NULL;
>                 other->sec_dai = NULL;
> @@ -1140,12 +1267,21 @@ static __devexit int samsung_i2s_remove(struct platform_device *pdev)
>         return 0;
>  }
>
> +#ifdef CONFIG_OF
> +static const struct of_device_id exynos_i2s_match[] = {
> +       { .compatible = "samsung,samsung-i2s" },
> +       {},
> +};
> +MODULE_DEVICE_TABLE(of, exynos_i2s_match);
> +#endif
> +
>  static struct platform_driver samsung_i2s_driver = {
>         .probe  = samsung_i2s_probe,
>         .remove = __devexit_p(samsung_i2s_remove),
>         .driver = {
>                 .name = "samsung-i2s",
>                 .owner = THIS_MODULE,
> +               .of_match_table = of_match_ptr(exynos_i2s_match),
>         },
>  };
>
> --
> 1.7.4.4
>
> --
> To unsubscribe from this list: send the line "unsubscribe linux-samsung-soc" in
> the body of a message to majordomo@vger.kernel.org
> More majordomo info at  http://vger.kernel.org/majordomo-info.html

^ permalink raw reply	[flat|nested] 4+ messages in thread

* [PATCH V3] ASoC: SAMSUNG: Add DT support for i2s
@ 2012-11-23 11:13   ` Padma Venkat
  0 siblings, 0 replies; 4+ messages in thread
From: Padma Venkat @ 2012-11-23 11:13 UTC (permalink / raw)
  To: linux-arm-kernel

cc'ing Mark Brown.

On Tue, Nov 6, 2012 at 10:11 AM, Padmavathi Venna <padma.v@samsung.com> wrote:
> Add support for device based discovery.
>
> Signed-off-by: Padmavathi Venna <padma.v@samsung.com>
> ---
>
> Changes since V2:
>         - Rebased on 3.7-rc3
>         - Custom DT bindings are prefixed with samsung
>         - As generic device tree DMA helpers not yet mainlined
>           I am still using custom dma bindings. So added a
>           priliminary statement regarding the same. I will rework
>           on my patch once generic DMA helpers are mainlined.
>
> Chnages since V1:
>         - Rebased on 3.6-rc6
>
>  .../devicetree/bindings/sound/samsung-i2s.txt      |   69 ++++++
>  sound/soc/samsung/dma.c                            |    1 +
>  sound/soc/samsung/dma.h                            |    1 +
>  sound/soc/samsung/i2s.c                            |  234 ++++++++++++++++----
>  4 files changed, 256 insertions(+), 49 deletions(-)
>  create mode 100644 Documentation/devicetree/bindings/sound/samsung-i2s.txt
>
> diff --git a/Documentation/devicetree/bindings/sound/samsung-i2s.txt b/Documentation/devicetree/bindings/sound/samsung-i2s.txt
> new file mode 100644
> index 0000000..dca61a6
> --- /dev/null
> +++ b/Documentation/devicetree/bindings/sound/samsung-i2s.txt
> @@ -0,0 +1,69 @@
> +* Samsung I2S controller
> +
> +Required SoC Specific Properties:
> +
> +- compatible : "samsung,samsung-i2s"
> +- reg: physical base address of the controller and length of memory mapped
> +  region.
> +
> +[PRELIMINARY: the dma channel allocation will change once there are
> +official DMA bindings]
> +
> +- tx-dma-channel-secondary: The dma channel specifier for secondary tx
> +  operations. The format of the dma specifier depends on the dma
> +  controller.
> +- tx-dma-channel: The dma channel specifier for tx operations. The format of
> +  the dma specifier depends on the dma controller.
> +- rx-dma-channel: The dma channel specifier for rx operations. The format of
> +  the dma specifier depends on the dma controller.
> +
> +Optional SoC Specific Properties:
> +
> +- samsung,supports-6ch: If the I2S Primary sound source has 5.1 Channel
> +  support, this flag is enabled.
> +- samsung,supports-rstclr: This flag should be set if I2S software reset bit
> +  control is required. When this flag is set I2S software reset bit will be
> +  enabled or disabled based on need.
> +- samsung,supports-secdai:If I2S block has a secondary FIFO and internal DMA,
> +  then this flag is enabled.
> +- samsung,idma-addr: Internal DMA register base address of the audio
> +  sub system(used in secondary sound source).
> +
> +Required Board Specific Properties:
> +
> +- gpios: The gpio specifier for data out,data in, LRCLK, CDCLK and SCLK
> +  interface lines. The format of the gpio specifier depends on the gpio
> +  controller.
> +
> +Aliases:
> +
> +- All the I2S controller nodes should be represented in the aliases node using
> +  the following format 'i2s{n}' where n is a unique number for the alias.
> +
> +Example:
> +
> +- SoC Specific Portion:
> +
> +i2s at 03830000 {
> +       compatible = "samsung,samsung-i2s";
> +       reg = <0x03830000 0x100>;
> +       tx-dma-channel-secondary = <&pdma0 8>;
> +       tx-dma-channel = <&pdma0 10>;
> +       rx-dma-channel = <&pdma0 9>;
> +       samsung,supports-6ch;
> +       samsung,supports-rstclr;
> +       samsung,supports-secdai;
> +       samsung,idma-addr = <0x03000000>;
> +};
> +
> +- Board Specific Portion:
> +
> +i2s_0: i2s at 03830000 {
> +       gpios = <&gpz 0 2 0 0>,
> +               <&gpz 1 2 0 0>,
> +               <&gpz 2 2 0 0>,
> +               <&gpz 3 2 0 0>,
> +               <&gpz 4 2 0 0>,
> +               <&gpz 5 2 0 0>,
> +               <&gpz 6 2 0 0>;
> +};
> diff --git a/sound/soc/samsung/dma.c b/sound/soc/samsung/dma.c
> index b70964e..359708c 100644
> --- a/sound/soc/samsung/dma.c
> +++ b/sound/soc/samsung/dma.c
> @@ -168,6 +168,7 @@ static int dma_hw_params(struct snd_pcm_substream *substream,
>                 req.cap = (samsung_dma_has_circular() ?
>                         DMA_CYCLIC : DMA_SLAVE);
>                 req.client = prtd->params->client;
> +               req.dt_dmach_prop = prtd->params->dma_prop;
>                 config.direction =
>                         (substream->stream == SNDRV_PCM_STREAM_PLAYBACK
>                         ? DMA_MEM_TO_DEV : DMA_DEV_TO_MEM);
> diff --git a/sound/soc/samsung/dma.h b/sound/soc/samsung/dma.h
> index 7d1ead7..2e60415 100644
> --- a/sound/soc/samsung/dma.h
> +++ b/sound/soc/samsung/dma.h
> @@ -19,6 +19,7 @@ struct s3c_dma_params {
>         int dma_size;                   /* Size of the DMA transfer */
>         unsigned ch;
>         struct samsung_dma_ops *ops;
> +       struct property *dma_prop;
>  };
>
>  #endif
> diff --git a/sound/soc/samsung/i2s.c b/sound/soc/samsung/i2s.c
> index 40b00a1..f9d55e5 100644
> --- a/sound/soc/samsung/i2s.c
> +++ b/sound/soc/samsung/i2s.c
> @@ -15,11 +15,15 @@
>  #include <linux/clk.h>
>  #include <linux/io.h>
>  #include <linux/module.h>
> +#include <linux/of.h>
> +#include <linux/of_gpio.h>
>  #include <linux/pm_runtime.h>
>
>  #include <sound/soc.h>
>  #include <sound/pcm_params.h>
>
> +#include <mach/dma.h>
> +
>  #include <linux/platform_data/asoc-s3c.h>
>
>  #include "dma.h"
> @@ -49,8 +53,6 @@ struct i2s_dai {
>         struct clk *clk;
>         /* Clock for generating I2S signals */
>         struct clk *op_clk;
> -       /* Array of clock names for op_clk */
> -       const char **src_clk;
>         /* Pointer to the Primary_Fifo if this is Sec_Fifo, NULL otherwise */
>         struct i2s_dai *pri_dai;
>         /* Pointer to the Secondary_Fifo if it has one, NULL otherwise */
> @@ -68,6 +70,8 @@ struct i2s_dai {
>         u32     suspend_i2smod;
>         u32     suspend_i2scon;
>         u32     suspend_i2spsr;
> +       unsigned long gpios[7]; /* i2s gpio line numbers */
> +       int     dev_id;         /* i2s dev id */
>  };
>
>  /* Lock for cross i/f checks */
> @@ -385,6 +389,7 @@ static int i2s_set_sysclk(struct snd_soc_dai *dai,
>         struct i2s_dai *i2s = to_info(dai);
>         struct i2s_dai *other = i2s->pri_dai ? : i2s->sec_dai;
>         u32 mod = readl(i2s->addr + I2SMOD);
> +       char clk_name[16];
>
>         switch (clk_id) {
>         case SAMSUNG_I2S_CDCLK:
> @@ -432,8 +437,9 @@ static int i2s_set_sysclk(struct snd_soc_dai *dai,
>                                 }
>                         }
>
> +                       sprintf(clk_name, "i2s_opclk%d", clk_id);
>                         i2s->op_clk = clk_get(&i2s->pdev->dev,
> -                                               i2s->src_clk[clk_id]);
> +                                               clk_name);
>                         clk_enable(i2s->op_clk);
>                         i2s->rclk_srcrate = clk_get_rate(i2s->op_clk);
>
> @@ -980,8 +986,9 @@ struct i2s_dai *i2s_alloc_dai(struct platform_device *pdev, bool sec)
>                 i2s->i2s_dai_drv.capture.formats = SAMSUNG_I2S_FMTS;
>         } else {        /* Create a new platform_device for Secondary */
>                 i2s->pdev = platform_device_register_resndata(NULL,
> -                               pdev->name, pdev->id + SAMSUNG_I2S_SECOFF,
> -                               NULL, 0, NULL, 0);
> +                               "samsung-i2s",
> +                               i2s->dev_id + SAMSUNG_I2S_SECOFF, NULL, 0,
> +                               NULL, 0);
>                 if (IS_ERR(i2s->pdev))
>                         return NULL;
>         }
> @@ -992,49 +999,154 @@ struct i2s_dai *i2s_alloc_dai(struct platform_device *pdev, bool sec)
>         return i2s;
>  }
>
> +#ifdef CONFIG_OF
> +static int samsung_i2s_parse_dt_gpio(struct i2s_dai *i2s)
> +{
> +       struct device *dev = &i2s->pdev->dev;
> +       int index, gpio, ret;
> +
> +       for (index = 0; index < 7; index++) {
> +               gpio = of_get_gpio(dev->of_node, index);
> +               if (!gpio_is_valid(gpio)) {
> +                       dev_err(dev, "invalid gpio[%d]: %d\n", index, gpio);
> +                       goto free_gpio;
> +               }
> +
> +               ret = gpio_request(gpio, dev_name(dev));
> +               if (ret) {
> +                       dev_err(dev, "gpio [%d] request failed\n", gpio);
> +                       goto free_gpio;
> +               }
> +               i2s->gpios[index] = gpio;
> +       }
> +       return 0;
> +
> +free_gpio:
> +       while (--index >= 0)
> +               gpio_free(i2s->gpios[index]);
> +       return -EINVAL;
> +}
> +
> +static void samsung_i2s_dt_gpio_free(struct i2s_dai *i2s)
> +{
> +       unsigned int index;
> +       for (index = 0; index < 7; index++)
> +               gpio_free(i2s->gpios[index]);
> +}
> +#else
> +static int samsung_i2s_parse_dt_gpio(struct i2s_dai *dai)
> +{
> +       return -EINVAL;
> +}
> +
> +static void samsung_i2s_dt_gpio_free(struct i2s_dai *dai)
> +{
> +}
> +
> +#endif
> +
>  static __devinit int samsung_i2s_probe(struct platform_device *pdev)
>  {
> -       u32 dma_pl_chan, dma_cp_chan, dma_pl_sec_chan;
> +       u32 dma_pl_chan, dma_cp_chan;
> +       u32 dma_pl_sec_chan = 0;
>         struct i2s_dai *pri_dai, *sec_dai = NULL;
> -       struct s3c_audio_pdata *i2s_pdata;
> -       struct samsung_i2s *i2s_cfg;
> +       struct s3c_audio_pdata *i2s_pdata = pdev->dev.platform_data;
> +       struct samsung_i2s *i2s_cfg = NULL;
>         struct resource *res;
> -       u32 regs_base, quirks;
> -       int ret = 0;
> +       u32 regs_base, quirks = 0, idma_addr = 0;
> +       struct property *prop;
> +       struct device_node *np = pdev->dev.of_node;
> +       int ret = 0, id;
>
>         /* Call during Seconday interface registration */
> -       if (pdev->id >= SAMSUNG_I2S_SECOFF) {
> +       if (np) {
> +               id = of_alias_get_id(np, "i2s");
> +               if (id < 0) {
> +                       dev_err(&pdev->dev, "failed to get alias id:%d\n", id);
> +                       return id;
> +               }
> +       } else {
> +               id = pdev->id;
> +       }
> +
> +       if (id >= SAMSUNG_I2S_SECOFF) {
>                 sec_dai = dev_get_drvdata(&pdev->dev);
>                 snd_soc_register_dai(&sec_dai->pdev->dev,
>                         &sec_dai->i2s_dai_drv);
>                 return 0;
>         }
>
> -       i2s_pdata = pdev->dev.platform_data;
> -       if (i2s_pdata == NULL) {
> -               dev_err(&pdev->dev, "Can't work without s3c_audio_pdata\n");
> -               return -EINVAL;
> +       pri_dai = i2s_alloc_dai(pdev, false);
> +       if (!pri_dai) {
> +               dev_err(&pdev->dev, "Unable to alloc I2S_pri\n");
> +               return -ENOMEM;
>         }
>
> -       res = platform_get_resource(pdev, IORESOURCE_DMA, 0);
> -       if (!res) {
> -               dev_err(&pdev->dev, "Unable to get I2S-TX dma resource\n");
> -               return -ENXIO;
> -       }
> -       dma_pl_chan = res->start;
> +       if (!np) {
> +               res = platform_get_resource(pdev, IORESOURCE_DMA, 0);
> +               if (!res) {
> +                       dev_err(&pdev->dev,
> +                               "Unable to get I2S-TX dma resource\n");
> +                       return -ENXIO;
> +               }
> +               dma_pl_chan = res->start;
>
> -       res = platform_get_resource(pdev, IORESOURCE_DMA, 1);
> -       if (!res) {
> -               dev_err(&pdev->dev, "Unable to get I2S-RX dma resource\n");
> -               return -ENXIO;
> -       }
> -       dma_cp_chan = res->start;
> +               res = platform_get_resource(pdev, IORESOURCE_DMA, 1);
> +               if (!res) {
> +                       dev_err(&pdev->dev,
> +                               "Unable to get I2S-RX dma resource\n");
> +                       return -ENXIO;
> +               }
> +               dma_cp_chan = res->start;
>
> -       res = platform_get_resource(pdev, IORESOURCE_DMA, 2);
> -       if (res)
> -               dma_pl_sec_chan = res->start;
> -       else
> -               dma_pl_sec_chan = 0;
> +               res = platform_get_resource(pdev, IORESOURCE_DMA, 2);
> +               if (res)
> +                       dma_pl_sec_chan = res->start;
> +
> +               if (&i2s_pdata->type)
> +                       i2s_cfg = &i2s_pdata->type.i2s;
> +
> +               if (i2s_cfg) {
> +                       quirks = i2s_cfg->quirks;
> +                       idma_addr = i2s_cfg->idma_addr;
> +               }
> +       } else {
> +               prop = of_find_property(np, "tx-dma-channel", NULL);
> +               if (!prop) {
> +                       dev_err(&pdev->dev, "tx dma channel property not"\
> +                                       "specified\n");
> +                       return -ENXIO;
> +               }
> +               dma_pl_chan = DMACH_DT_PROP;
> +               pri_dai->dma_playback.dma_prop = prop;
> +
> +               prop = of_find_property(np, "rx-dma-channel", NULL);
> +               if (!prop) {
> +                       dev_err(&pdev->dev, "tx dma channel property not"\
> +                                       "specified\n");
> +                       return -ENXIO;
> +               }
> +               dma_cp_chan = DMACH_DT_PROP;
> +               pri_dai->dma_capture.dma_prop = prop;
> +
> +               if (of_find_property(np, "samsung,supports-6ch", NULL))
> +                       quirks |= QUIRK_PRI_6CHAN;
> +
> +               if (of_find_property(np, "samsung,supports-secdai", NULL))
> +                       quirks |= QUIRK_SEC_DAI;
> +
> +               if (of_find_property(np, "samsung,supports-rstclr", NULL))
> +                       quirks |= QUIRK_NEED_RSTCLR;
> +
> +               if (of_property_read_u32(np, "samsung,idma-addr",
> +                                        &idma_addr)) {
> +                       if (quirks & QUIRK_SEC_DAI) {
> +                               dev_err(&pdev->dev, "idma address is not"\
> +                                               "specified");
> +                               return -EINVAL;
> +                       }
> +               }
> +       }
>
>         res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
>         if (!res) {
> @@ -1049,16 +1161,6 @@ static __devinit int samsung_i2s_probe(struct platform_device *pdev)
>         }
>         regs_base = res->start;
>
> -       i2s_cfg = &i2s_pdata->type.i2s;
> -       quirks = i2s_cfg->quirks;
> -
> -       pri_dai = i2s_alloc_dai(pdev, false);
> -       if (!pri_dai) {
> -               dev_err(&pdev->dev, "Unable to alloc I2S_pri\n");
> -               ret = -ENOMEM;
> -               goto err;
> -       }
> -
>         pri_dai->dma_playback.dma_addr = regs_base + I2STXD;
>         pri_dai->dma_capture.dma_addr = regs_base + I2SRXD;
>         pri_dai->dma_playback.client =
> @@ -1067,11 +1169,11 @@ static __devinit int samsung_i2s_probe(struct platform_device *pdev)
>                 (struct s3c2410_dma_client *)&pri_dai->dma_capture;
>         pri_dai->dma_playback.channel = dma_pl_chan;
>         pri_dai->dma_capture.channel = dma_cp_chan;
> -       pri_dai->src_clk = i2s_cfg->src_clk;
>         pri_dai->dma_playback.dma_size = 4;
>         pri_dai->dma_capture.dma_size = 4;
>         pri_dai->base = regs_base;
>         pri_dai->quirks = quirks;
> +       pri_dai->dev_id = id;
>
>         if (quirks & QUIRK_PRI_6CHAN)
>                 pri_dai->i2s_dai_drv.playback.channels_max = 6;
> @@ -1086,21 +1188,42 @@ static __devinit int samsung_i2s_probe(struct platform_device *pdev)
>                 sec_dai->dma_playback.dma_addr = regs_base + I2STXDS;
>                 sec_dai->dma_playback.client =
>                         (struct s3c2410_dma_client *)&sec_dai->dma_playback;
> +
> +               if (np) {
> +                       prop = of_find_property(np, "tx-dma-channel-secondary",
> +                                               NULL);
> +                       if (!prop) {
> +                               dev_err(&pdev->dev, "tx dma channel property"\
> +                                               "not specified\n");
> +                               ret = -ENXIO;
> +                               goto err;
> +                       }
> +                       sec_dai->dma_playback.dma_prop = prop;
> +               }
> +
>                 /* Use iDMA always if SysDMA not provided */
>                 sec_dai->dma_playback.channel = dma_pl_sec_chan ? : -1;
> -               sec_dai->src_clk = i2s_cfg->src_clk;
>                 sec_dai->dma_playback.dma_size = 4;
>                 sec_dai->base = regs_base;
>                 sec_dai->quirks = quirks;
> -               sec_dai->idma_playback.dma_addr = i2s_cfg->idma_addr;
> +               sec_dai->dev_id = id;
> +               sec_dai->idma_playback.dma_addr = idma_addr;
>                 sec_dai->pri_dai = pri_dai;
>                 pri_dai->sec_dai = sec_dai;
>         }
>
> -       if (i2s_pdata->cfg_gpio && i2s_pdata->cfg_gpio(pdev)) {
> -               dev_err(&pdev->dev, "Unable to configure gpio\n");
> -               ret = -EINVAL;
> -               goto err;
> +       if (np) {
> +               if (samsung_i2s_parse_dt_gpio(pri_dai)) {
> +                       dev_err(&pdev->dev, "Unable to configure gpio\n");
> +                       ret = -EINVAL;
> +                       goto err;
> +               }
> +       } else {
> +               if (i2s_pdata->cfg_gpio && i2s_pdata->cfg_gpio(pdev)) {
> +                       dev_err(&pdev->dev, "Unable to configure gpio\n");
> +                       ret = -EINVAL;
> +                       goto err;
> +               }
>         }
>
>         snd_soc_register_dai(&pri_dai->pdev->dev, &pri_dai->i2s_dai_drv);
> @@ -1118,10 +1241,14 @@ static __devexit int samsung_i2s_remove(struct platform_device *pdev)
>  {
>         struct i2s_dai *i2s, *other;
>         struct resource *res;
> +       struct s3c_audio_pdata *i2s_pdata = pdev->dev.platform_data;
>
>         i2s = dev_get_drvdata(&pdev->dev);
>         other = i2s->pri_dai ? : i2s->sec_dai;
>
> +       if (!i2s_pdata->cfg_gpio && pdev->dev.of_node)
> +               samsung_i2s_dt_gpio_free(i2s->pri_dai);
> +
>         if (other) {
>                 other->pri_dai = NULL;
>                 other->sec_dai = NULL;
> @@ -1140,12 +1267,21 @@ static __devexit int samsung_i2s_remove(struct platform_device *pdev)
>         return 0;
>  }
>
> +#ifdef CONFIG_OF
> +static const struct of_device_id exynos_i2s_match[] = {
> +       { .compatible = "samsung,samsung-i2s" },
> +       {},
> +};
> +MODULE_DEVICE_TABLE(of, exynos_i2s_match);
> +#endif
> +
>  static struct platform_driver samsung_i2s_driver = {
>         .probe  = samsung_i2s_probe,
>         .remove = __devexit_p(samsung_i2s_remove),
>         .driver = {
>                 .name = "samsung-i2s",
>                 .owner = THIS_MODULE,
> +               .of_match_table = of_match_ptr(exynos_i2s_match),
>         },
>  };
>
> --
> 1.7.4.4
>
> --
> To unsubscribe from this list: send the line "unsubscribe linux-samsung-soc" in
> the body of a message to majordomo at vger.kernel.org
> More majordomo info at  http://vger.kernel.org/majordomo-info.html

^ permalink raw reply	[flat|nested] 4+ messages in thread

end of thread, other threads:[~2012-11-23 11:13 UTC | newest]

Thread overview: 4+ messages (download: mbox.gz / follow: Atom feed)
-- links below jump to the message on this page --
2012-11-06  4:41 [PATCH V3] ASoC: SAMSUNG: Add DT support for i2s Padmavathi Venna
2012-11-06  4:41 ` Padmavathi Venna
2012-11-23 11:13 ` Padma Venkat
2012-11-23 11:13   ` Padma Venkat

This is an external index of several public inboxes,
see mirroring instructions on how to clone and mirror
all data and code used by this external index.