mirror of
https://github.com/torvalds/linux.git
synced 2024-12-14 15:13:52 +00:00
cdce778344
The .remove() callback for a platform driver returns an int which makes many driver authors wrongly assume it's possible to do error handling by returning an error code. However the value returned is (mostly) ignored and this typically results in resource leaks. To improve here there is a quest to make the remove callback return void. In the first step of this quest all drivers are converted to .remove_new() which already returns void. Trivially convert this driver from always returning zero in the remove callback to the void returning variant. Signed-off-by: Uwe Kleine-König <u.kleine-koenig@pengutronix.de> Acked-by: Takashi Iwai <tiwai@suse.de> Acked-by: Nicolas Ferre <nicolas.ferre@microchip.com> Link: https://lore.kernel.org/r/20230315150745.67084-124-u.kleine-koenig@pengutronix.de Signed-off-by: Mark Brown <broonie@kernel.org>
397 lines
9.3 KiB
C
397 lines
9.3 KiB
C
// SPDX-License-Identifier: GPL-2.0-only
|
|
/* sound/soc/rockchip/rk_spdif.c
|
|
*
|
|
* ALSA SoC Audio Layer - Rockchip I2S Controller driver
|
|
*
|
|
* Copyright (c) 2014 Rockchip Electronics Co. Ltd.
|
|
* Author: Jianqun <jay.xu@rock-chips.com>
|
|
* Copyright (c) 2015 Collabora Ltd.
|
|
* Author: Sjoerd Simons <sjoerd.simons@collabora.co.uk>
|
|
*/
|
|
|
|
#include <linux/module.h>
|
|
#include <linux/delay.h>
|
|
#include <linux/of_gpio.h>
|
|
#include <linux/clk.h>
|
|
#include <linux/pm_runtime.h>
|
|
#include <linux/mfd/syscon.h>
|
|
#include <linux/regmap.h>
|
|
#include <sound/pcm_params.h>
|
|
#include <sound/dmaengine_pcm.h>
|
|
|
|
#include "rockchip_spdif.h"
|
|
|
|
enum rk_spdif_type {
|
|
RK_SPDIF_RK3066,
|
|
RK_SPDIF_RK3188,
|
|
RK_SPDIF_RK3288,
|
|
RK_SPDIF_RK3366,
|
|
};
|
|
|
|
#define RK3288_GRF_SOC_CON2 0x24c
|
|
|
|
struct rk_spdif_dev {
|
|
struct device *dev;
|
|
|
|
struct clk *mclk;
|
|
struct clk *hclk;
|
|
|
|
struct snd_dmaengine_dai_dma_data playback_dma_data;
|
|
|
|
struct regmap *regmap;
|
|
};
|
|
|
|
static const struct of_device_id rk_spdif_match[] __maybe_unused = {
|
|
{ .compatible = "rockchip,rk3066-spdif",
|
|
.data = (void *)RK_SPDIF_RK3066 },
|
|
{ .compatible = "rockchip,rk3188-spdif",
|
|
.data = (void *)RK_SPDIF_RK3188 },
|
|
{ .compatible = "rockchip,rk3228-spdif",
|
|
.data = (void *)RK_SPDIF_RK3366 },
|
|
{ .compatible = "rockchip,rk3288-spdif",
|
|
.data = (void *)RK_SPDIF_RK3288 },
|
|
{ .compatible = "rockchip,rk3328-spdif",
|
|
.data = (void *)RK_SPDIF_RK3366 },
|
|
{ .compatible = "rockchip,rk3366-spdif",
|
|
.data = (void *)RK_SPDIF_RK3366 },
|
|
{ .compatible = "rockchip,rk3368-spdif",
|
|
.data = (void *)RK_SPDIF_RK3366 },
|
|
{ .compatible = "rockchip,rk3399-spdif",
|
|
.data = (void *)RK_SPDIF_RK3366 },
|
|
{ .compatible = "rockchip,rk3568-spdif",
|
|
.data = (void *)RK_SPDIF_RK3366 },
|
|
{},
|
|
};
|
|
MODULE_DEVICE_TABLE(of, rk_spdif_match);
|
|
|
|
static int __maybe_unused rk_spdif_runtime_suspend(struct device *dev)
|
|
{
|
|
struct rk_spdif_dev *spdif = dev_get_drvdata(dev);
|
|
|
|
regcache_cache_only(spdif->regmap, true);
|
|
clk_disable_unprepare(spdif->mclk);
|
|
clk_disable_unprepare(spdif->hclk);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int __maybe_unused rk_spdif_runtime_resume(struct device *dev)
|
|
{
|
|
struct rk_spdif_dev *spdif = dev_get_drvdata(dev);
|
|
int ret;
|
|
|
|
ret = clk_prepare_enable(spdif->mclk);
|
|
if (ret) {
|
|
dev_err(spdif->dev, "mclk clock enable failed %d\n", ret);
|
|
return ret;
|
|
}
|
|
|
|
ret = clk_prepare_enable(spdif->hclk);
|
|
if (ret) {
|
|
clk_disable_unprepare(spdif->mclk);
|
|
dev_err(spdif->dev, "hclk clock enable failed %d\n", ret);
|
|
return ret;
|
|
}
|
|
|
|
regcache_cache_only(spdif->regmap, false);
|
|
regcache_mark_dirty(spdif->regmap);
|
|
|
|
ret = regcache_sync(spdif->regmap);
|
|
if (ret) {
|
|
clk_disable_unprepare(spdif->mclk);
|
|
clk_disable_unprepare(spdif->hclk);
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
static int rk_spdif_hw_params(struct snd_pcm_substream *substream,
|
|
struct snd_pcm_hw_params *params,
|
|
struct snd_soc_dai *dai)
|
|
{
|
|
struct rk_spdif_dev *spdif = snd_soc_dai_get_drvdata(dai);
|
|
unsigned int val = SPDIF_CFGR_HALFWORD_ENABLE;
|
|
int srate, mclk;
|
|
int ret;
|
|
|
|
srate = params_rate(params);
|
|
mclk = srate * 128;
|
|
|
|
switch (params_format(params)) {
|
|
case SNDRV_PCM_FORMAT_S16_LE:
|
|
val |= SPDIF_CFGR_VDW_16;
|
|
break;
|
|
case SNDRV_PCM_FORMAT_S20_3LE:
|
|
val |= SPDIF_CFGR_VDW_20;
|
|
break;
|
|
case SNDRV_PCM_FORMAT_S24_LE:
|
|
val |= SPDIF_CFGR_VDW_24;
|
|
break;
|
|
default:
|
|
return -EINVAL;
|
|
}
|
|
|
|
/* Set clock and calculate divider */
|
|
ret = clk_set_rate(spdif->mclk, mclk);
|
|
if (ret != 0) {
|
|
dev_err(spdif->dev, "Failed to set module clock rate: %d\n",
|
|
ret);
|
|
return ret;
|
|
}
|
|
|
|
ret = regmap_update_bits(spdif->regmap, SPDIF_CFGR,
|
|
SPDIF_CFGR_CLK_DIV_MASK |
|
|
SPDIF_CFGR_HALFWORD_ENABLE |
|
|
SDPIF_CFGR_VDW_MASK, val);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static int rk_spdif_trigger(struct snd_pcm_substream *substream,
|
|
int cmd, struct snd_soc_dai *dai)
|
|
{
|
|
struct rk_spdif_dev *spdif = snd_soc_dai_get_drvdata(dai);
|
|
int ret;
|
|
|
|
switch (cmd) {
|
|
case SNDRV_PCM_TRIGGER_START:
|
|
case SNDRV_PCM_TRIGGER_RESUME:
|
|
case SNDRV_PCM_TRIGGER_PAUSE_RELEASE:
|
|
ret = regmap_update_bits(spdif->regmap, SPDIF_DMACR,
|
|
SPDIF_DMACR_TDE_ENABLE |
|
|
SPDIF_DMACR_TDL_MASK,
|
|
SPDIF_DMACR_TDE_ENABLE |
|
|
SPDIF_DMACR_TDL(16));
|
|
|
|
if (ret != 0)
|
|
return ret;
|
|
|
|
ret = regmap_update_bits(spdif->regmap, SPDIF_XFER,
|
|
SPDIF_XFER_TXS_START,
|
|
SPDIF_XFER_TXS_START);
|
|
break;
|
|
case SNDRV_PCM_TRIGGER_SUSPEND:
|
|
case SNDRV_PCM_TRIGGER_STOP:
|
|
case SNDRV_PCM_TRIGGER_PAUSE_PUSH:
|
|
ret = regmap_update_bits(spdif->regmap, SPDIF_DMACR,
|
|
SPDIF_DMACR_TDE_ENABLE,
|
|
SPDIF_DMACR_TDE_DISABLE);
|
|
|
|
if (ret != 0)
|
|
return ret;
|
|
|
|
ret = regmap_update_bits(spdif->regmap, SPDIF_XFER,
|
|
SPDIF_XFER_TXS_START,
|
|
SPDIF_XFER_TXS_STOP);
|
|
break;
|
|
default:
|
|
ret = -EINVAL;
|
|
break;
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
static int rk_spdif_dai_probe(struct snd_soc_dai *dai)
|
|
{
|
|
struct rk_spdif_dev *spdif = snd_soc_dai_get_drvdata(dai);
|
|
|
|
snd_soc_dai_dma_data_set_playback(dai, &spdif->playback_dma_data);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static const struct snd_soc_dai_ops rk_spdif_dai_ops = {
|
|
.hw_params = rk_spdif_hw_params,
|
|
.trigger = rk_spdif_trigger,
|
|
};
|
|
|
|
static struct snd_soc_dai_driver rk_spdif_dai = {
|
|
.probe = rk_spdif_dai_probe,
|
|
.playback = {
|
|
.stream_name = "Playback",
|
|
.channels_min = 2,
|
|
.channels_max = 2,
|
|
.rates = (SNDRV_PCM_RATE_32000 |
|
|
SNDRV_PCM_RATE_44100 |
|
|
SNDRV_PCM_RATE_48000 |
|
|
SNDRV_PCM_RATE_96000 |
|
|
SNDRV_PCM_RATE_192000),
|
|
.formats = (SNDRV_PCM_FMTBIT_S16_LE |
|
|
SNDRV_PCM_FMTBIT_S20_3LE |
|
|
SNDRV_PCM_FMTBIT_S24_LE),
|
|
},
|
|
.ops = &rk_spdif_dai_ops,
|
|
};
|
|
|
|
static const struct snd_soc_component_driver rk_spdif_component = {
|
|
.name = "rockchip-spdif",
|
|
.legacy_dai_naming = 1,
|
|
};
|
|
|
|
static bool rk_spdif_wr_reg(struct device *dev, unsigned int reg)
|
|
{
|
|
switch (reg) {
|
|
case SPDIF_CFGR:
|
|
case SPDIF_DMACR:
|
|
case SPDIF_INTCR:
|
|
case SPDIF_XFER:
|
|
case SPDIF_SMPDR:
|
|
return true;
|
|
default:
|
|
return false;
|
|
}
|
|
}
|
|
|
|
static bool rk_spdif_rd_reg(struct device *dev, unsigned int reg)
|
|
{
|
|
switch (reg) {
|
|
case SPDIF_CFGR:
|
|
case SPDIF_SDBLR:
|
|
case SPDIF_INTCR:
|
|
case SPDIF_INTSR:
|
|
case SPDIF_XFER:
|
|
case SPDIF_SMPDR:
|
|
return true;
|
|
default:
|
|
return false;
|
|
}
|
|
}
|
|
|
|
static bool rk_spdif_volatile_reg(struct device *dev, unsigned int reg)
|
|
{
|
|
switch (reg) {
|
|
case SPDIF_INTSR:
|
|
case SPDIF_SDBLR:
|
|
case SPDIF_SMPDR:
|
|
return true;
|
|
default:
|
|
return false;
|
|
}
|
|
}
|
|
|
|
static const struct regmap_config rk_spdif_regmap_config = {
|
|
.reg_bits = 32,
|
|
.reg_stride = 4,
|
|
.val_bits = 32,
|
|
.max_register = SPDIF_SMPDR,
|
|
.writeable_reg = rk_spdif_wr_reg,
|
|
.readable_reg = rk_spdif_rd_reg,
|
|
.volatile_reg = rk_spdif_volatile_reg,
|
|
.cache_type = REGCACHE_FLAT,
|
|
};
|
|
|
|
static int rk_spdif_probe(struct platform_device *pdev)
|
|
{
|
|
struct device_node *np = pdev->dev.of_node;
|
|
struct rk_spdif_dev *spdif;
|
|
const struct of_device_id *match;
|
|
struct resource *res;
|
|
void __iomem *regs;
|
|
int ret;
|
|
|
|
match = of_match_node(rk_spdif_match, np);
|
|
if (match->data == (void *)RK_SPDIF_RK3288) {
|
|
struct regmap *grf;
|
|
|
|
grf = syscon_regmap_lookup_by_phandle(np, "rockchip,grf");
|
|
if (IS_ERR(grf)) {
|
|
dev_err(&pdev->dev,
|
|
"rockchip_spdif missing 'rockchip,grf'\n");
|
|
return PTR_ERR(grf);
|
|
}
|
|
|
|
/* Select the 8 channel SPDIF solution on RK3288 as
|
|
* the 2 channel one does not appear to work
|
|
*/
|
|
regmap_write(grf, RK3288_GRF_SOC_CON2, BIT(1) << 16);
|
|
}
|
|
|
|
spdif = devm_kzalloc(&pdev->dev, sizeof(*spdif), GFP_KERNEL);
|
|
if (!spdif)
|
|
return -ENOMEM;
|
|
|
|
spdif->hclk = devm_clk_get(&pdev->dev, "hclk");
|
|
if (IS_ERR(spdif->hclk))
|
|
return PTR_ERR(spdif->hclk);
|
|
|
|
spdif->mclk = devm_clk_get(&pdev->dev, "mclk");
|
|
if (IS_ERR(spdif->mclk))
|
|
return PTR_ERR(spdif->mclk);
|
|
|
|
regs = devm_platform_get_and_ioremap_resource(pdev, 0, &res);
|
|
if (IS_ERR(regs))
|
|
return PTR_ERR(regs);
|
|
|
|
spdif->regmap = devm_regmap_init_mmio_clk(&pdev->dev, "hclk", regs,
|
|
&rk_spdif_regmap_config);
|
|
if (IS_ERR(spdif->regmap))
|
|
return PTR_ERR(spdif->regmap);
|
|
|
|
spdif->playback_dma_data.addr = res->start + SPDIF_SMPDR;
|
|
spdif->playback_dma_data.addr_width = DMA_SLAVE_BUSWIDTH_4_BYTES;
|
|
spdif->playback_dma_data.maxburst = 4;
|
|
|
|
spdif->dev = &pdev->dev;
|
|
dev_set_drvdata(&pdev->dev, spdif);
|
|
|
|
pm_runtime_enable(&pdev->dev);
|
|
if (!pm_runtime_enabled(&pdev->dev)) {
|
|
ret = rk_spdif_runtime_resume(&pdev->dev);
|
|
if (ret)
|
|
goto err_pm_runtime;
|
|
}
|
|
|
|
ret = devm_snd_soc_register_component(&pdev->dev,
|
|
&rk_spdif_component,
|
|
&rk_spdif_dai, 1);
|
|
if (ret) {
|
|
dev_err(&pdev->dev, "Could not register DAI\n");
|
|
goto err_pm_suspend;
|
|
}
|
|
|
|
ret = devm_snd_dmaengine_pcm_register(&pdev->dev, NULL, 0);
|
|
if (ret) {
|
|
dev_err(&pdev->dev, "Could not register PCM\n");
|
|
goto err_pm_suspend;
|
|
}
|
|
|
|
return 0;
|
|
|
|
err_pm_suspend:
|
|
if (!pm_runtime_status_suspended(&pdev->dev))
|
|
rk_spdif_runtime_suspend(&pdev->dev);
|
|
err_pm_runtime:
|
|
pm_runtime_disable(&pdev->dev);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static void rk_spdif_remove(struct platform_device *pdev)
|
|
{
|
|
pm_runtime_disable(&pdev->dev);
|
|
if (!pm_runtime_status_suspended(&pdev->dev))
|
|
rk_spdif_runtime_suspend(&pdev->dev);
|
|
}
|
|
|
|
static const struct dev_pm_ops rk_spdif_pm_ops = {
|
|
SET_RUNTIME_PM_OPS(rk_spdif_runtime_suspend, rk_spdif_runtime_resume,
|
|
NULL)
|
|
};
|
|
|
|
static struct platform_driver rk_spdif_driver = {
|
|
.probe = rk_spdif_probe,
|
|
.remove_new = rk_spdif_remove,
|
|
.driver = {
|
|
.name = "rockchip-spdif",
|
|
.of_match_table = of_match_ptr(rk_spdif_match),
|
|
.pm = &rk_spdif_pm_ops,
|
|
},
|
|
};
|
|
module_platform_driver(rk_spdif_driver);
|
|
|
|
MODULE_ALIAS("platform:rockchip-spdif");
|
|
MODULE_DESCRIPTION("ROCKCHIP SPDIF transceiver Interface");
|
|
MODULE_AUTHOR("Sjoerd Simons <sjoerd.simons@collabora.co.uk>");
|
|
MODULE_LICENSE("GPL v2");
|