Commit 5f9a50c3 authored by Petr Kulhavy's avatar Petr Kulhavy Committed by Mark Brown

ASoC: Davinci: McBSP: add device tree support for McBSP

This adds DT support for the TI DA8xx/OMAP-L1x/AM17xx/AM18xx McBSP driver.
Signed-off-by: default avatarPetr Kulhavy <petr@barix.com>
Reviewed-by: default avatarPeter Ujfalusi <peter.ujfalusi@ti.com>
Signed-off-by: default avatarMark Brown <broonie@kernel.org>
parent 22225835
...@@ -16,7 +16,11 @@ config SND_EDMA_SOC ...@@ -16,7 +16,11 @@ config SND_EDMA_SOC
- DRA7xx family - DRA7xx family
config SND_DAVINCI_SOC_I2S config SND_DAVINCI_SOC_I2S
tristate tristate "DaVinci Multichannel Buffered Serial Port (McBSP) support"
depends on SND_EDMA_SOC
help
Say Y or M here if you want to have support for McBSP IP found in
Texas Instruments DaVinci DA850 SoCs.
config SND_DAVINCI_SOC_MCASP config SND_DAVINCI_SOC_MCASP
tristate "Multichannel Audio Serial Port (McASP) support" tristate "Multichannel Audio Serial Port (McASP) support"
......
...@@ -4,9 +4,15 @@ ...@@ -4,9 +4,15 @@
* Author: Vladimir Barinov, <vbarinov@embeddedalley.com> * Author: Vladimir Barinov, <vbarinov@embeddedalley.com>
* Copyright: (C) 2007 MontaVista Software, Inc., <source@mvista.com> * Copyright: (C) 2007 MontaVista Software, Inc., <source@mvista.com>
* *
* DT support (c) 2016 Petr Kulhavy, Barix AG <petr@barix.com>
* based on davinci-mcasp.c DT support
*
* This program is free software; you can redistribute it and/or modify * This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License version 2 as * it under the terms of the GNU General Public License version 2 as
* published by the Free Software Foundation. * published by the Free Software Foundation.
*
* TODO:
* on DA850 implement HW FIFOs instead of DMA into DXR and DRR registers
*/ */
#include <linux/init.h> #include <linux/init.h>
...@@ -650,13 +656,24 @@ static const struct snd_soc_component_driver davinci_i2s_component = { ...@@ -650,13 +656,24 @@ static const struct snd_soc_component_driver davinci_i2s_component = {
static int davinci_i2s_probe(struct platform_device *pdev) static int davinci_i2s_probe(struct platform_device *pdev)
{ {
struct snd_dmaengine_dai_dma_data *dma_data;
struct davinci_mcbsp_dev *dev; struct davinci_mcbsp_dev *dev;
struct resource *mem, *res; struct resource *mem, *res;
void __iomem *io_base; void __iomem *io_base;
int *dma; int *dma;
int ret; int ret;
mem = platform_get_resource(pdev, IORESOURCE_MEM, 0); mem = platform_get_resource_byname(pdev, IORESOURCE_MEM, "mpu");
if (!mem) {
dev_warn(&pdev->dev,
"\"mpu\" mem resource not found, using index 0\n");
mem = platform_get_resource(pdev, IORESOURCE_MEM, 0);
if (!mem) {
dev_err(&pdev->dev, "no mem resource?\n");
return -ENODEV;
}
}
io_base = devm_ioremap_resource(&pdev->dev, mem); io_base = devm_ioremap_resource(&pdev->dev, mem);
if (IS_ERR(io_base)) if (IS_ERR(io_base))
return PTR_ERR(io_base); return PTR_ERR(io_base);
...@@ -666,39 +683,43 @@ static int davinci_i2s_probe(struct platform_device *pdev) ...@@ -666,39 +683,43 @@ static int davinci_i2s_probe(struct platform_device *pdev)
if (!dev) if (!dev)
return -ENOMEM; return -ENOMEM;
dev->clk = clk_get(&pdev->dev, NULL);
if (IS_ERR(dev->clk))
return -ENODEV;
clk_enable(dev->clk);
dev->base = io_base; dev->base = io_base;
dev->dma_data[SNDRV_PCM_STREAM_PLAYBACK].addr = /* setup DMA, first TX, then RX */
(dma_addr_t)(mem->start + DAVINCI_MCBSP_DXR_REG); dma_data = &dev->dma_data[SNDRV_PCM_STREAM_PLAYBACK];
dma_data->addr = (dma_addr_t)(mem->start + DAVINCI_MCBSP_DXR_REG);
dev->dma_data[SNDRV_PCM_STREAM_CAPTURE].addr =
(dma_addr_t)(mem->start + DAVINCI_MCBSP_DRR_REG);
/* first TX, then RX */
res = platform_get_resource(pdev, IORESOURCE_DMA, 0); res = platform_get_resource(pdev, IORESOURCE_DMA, 0);
if (!res) { if (res) {
dev_err(&pdev->dev, "no DMA resource\n"); dma = &dev->dma_request[SNDRV_PCM_STREAM_PLAYBACK];
ret = -ENXIO; *dma = res->start;
goto err_release_clk; dma_data->filter_data = dma;
} else if (IS_ENABLED(CONFIG_OF) && pdev->dev.of_node) {
dma_data->filter_data = "tx";
} else {
dev_err(&pdev->dev, "Missing DMA tx resource\n");
return -ENODEV;
} }
dma = &dev->dma_request[SNDRV_PCM_STREAM_PLAYBACK];
*dma = res->start; dma_data = &dev->dma_data[SNDRV_PCM_STREAM_CAPTURE];
dev->dma_data[SNDRV_PCM_STREAM_PLAYBACK].filter_data = dma; dma_data->addr = (dma_addr_t)(mem->start + DAVINCI_MCBSP_DRR_REG);
res = platform_get_resource(pdev, IORESOURCE_DMA, 1); res = platform_get_resource(pdev, IORESOURCE_DMA, 1);
if (!res) { if (res) {
dev_err(&pdev->dev, "no DMA resource\n"); dma = &dev->dma_request[SNDRV_PCM_STREAM_CAPTURE];
ret = -ENXIO; *dma = res->start;
goto err_release_clk; dma_data->filter_data = dma;
} else if (IS_ENABLED(CONFIG_OF) && pdev->dev.of_node) {
dma_data->filter_data = "rx";
} else {
dev_err(&pdev->dev, "Missing DMA rx resource\n");
return -ENODEV;
} }
dma = &dev->dma_request[SNDRV_PCM_STREAM_CAPTURE];
*dma = res->start; dev->clk = clk_get(&pdev->dev, NULL);
dev->dma_data[SNDRV_PCM_STREAM_CAPTURE].filter_data = dma; if (IS_ERR(dev->clk))
return -ENODEV;
clk_enable(dev->clk);
dev->dev = &pdev->dev; dev->dev = &pdev->dev;
dev_set_drvdata(&pdev->dev, dev); dev_set_drvdata(&pdev->dev, dev);
...@@ -737,11 +758,18 @@ static int davinci_i2s_remove(struct platform_device *pdev) ...@@ -737,11 +758,18 @@ static int davinci_i2s_remove(struct platform_device *pdev)
return 0; return 0;
} }
static const struct of_device_id davinci_i2s_match[] = {
{ .compatible = "ti,da850-mcbsp" },
{},
};
MODULE_DEVICE_TABLE(of, davinci_i2s_match);
static struct platform_driver davinci_mcbsp_driver = { static struct platform_driver davinci_mcbsp_driver = {
.probe = davinci_i2s_probe, .probe = davinci_i2s_probe,
.remove = davinci_i2s_remove, .remove = davinci_i2s_remove,
.driver = { .driver = {
.name = "davinci-mcbsp", .name = "davinci-mcbsp",
.of_match_table = of_match_ptr(davinci_i2s_match),
}, },
}; };
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment