Release 4.11 sound/soc/cirrus/ep93xx-i2s.c
  
  
  
/*
 * linux/sound/soc/ep93xx-i2s.c
 * EP93xx I2S driver
 *
 * Copyright (C) 2010 Ryan Mallon
 *
 * Based on the original driver by:
 *   Copyright (C) 2007 Chase Douglas <chasedouglas@gmail>
 *   Copyright (C) 2006 Lennert Buytenhek <buytenh@wantstofly.org>
 *
 * 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
 * published by the Free Software Foundation.
 *
 */
#include <linux/module.h>
#include <linux/init.h>
#include <linux/slab.h>
#include <linux/clk.h>
#include <linux/io.h>
#include <sound/core.h>
#include <sound/dmaengine_pcm.h>
#include <sound/pcm.h>
#include <sound/pcm_params.h>
#include <sound/initval.h>
#include <sound/soc.h>
#include <mach/hardware.h>
#include <mach/ep93xx-regs.h>
#include <linux/platform_data/dma-ep93xx.h>
#include "ep93xx-pcm.h"
#define EP93XX_I2S_TXCLKCFG		0x00
#define EP93XX_I2S_RXCLKCFG		0x04
#define EP93XX_I2S_GLCTRL		0x0C
#define EP93XX_I2S_TXLINCTRLDATA	0x28
#define EP93XX_I2S_TXCTRL		0x2C
#define EP93XX_I2S_TXWRDLEN		0x30
#define EP93XX_I2S_TX0EN		0x34
#define EP93XX_I2S_RXLINCTRLDATA	0x58
#define EP93XX_I2S_RXCTRL		0x5C
#define EP93XX_I2S_RXWRDLEN		0x60
#define EP93XX_I2S_RX0EN		0x64
#define EP93XX_I2S_WRDLEN_16		(0 << 0)
#define EP93XX_I2S_WRDLEN_24		(1 << 0)
#define EP93XX_I2S_WRDLEN_32		(2 << 0)
#define EP93XX_I2S_LINCTRLDATA_R_JUST	(1 << 2) 
/* Right justify */
#define EP93XX_I2S_CLKCFG_LRS		(1 << 0) 
/* lrclk polarity */
#define EP93XX_I2S_CLKCFG_CKP		(1 << 1) 
/* Bit clock polarity */
#define EP93XX_I2S_CLKCFG_REL		(1 << 2) 
/* First bit transition */
#define EP93XX_I2S_CLKCFG_MASTER	(1 << 3) 
/* Master mode */
#define EP93XX_I2S_CLKCFG_NBCG		(1 << 4) 
/* Not bit clock gating */
struct ep93xx_i2s_info {
	
struct clk			*mclk;
	
struct clk			*sclk;
	
struct clk			*lrclk;
	
void __iomem			*regs;
	
struct snd_dmaengine_dai_dma_data dma_params_rx;
	
struct snd_dmaengine_dai_dma_data dma_params_tx;
};
static struct ep93xx_dma_data ep93xx_i2s_dma_data[] = {
	[SNDRV_PCM_STREAM_PLAYBACK] = {
		.name		= "i2s-pcm-out",
		.port		= EP93XX_DMA_I2S1,
		.direction	= DMA_MEM_TO_DEV,
        },
	[SNDRV_PCM_STREAM_CAPTURE] = {
		.name		= "i2s-pcm-in",
		.port		= EP93XX_DMA_I2S1,
		.direction	= DMA_DEV_TO_MEM,
        },
};
static inline void ep93xx_i2s_write_reg(struct ep93xx_i2s_info *info,
					unsigned reg, unsigned val)
{
	__raw_writel(val, info->regs + reg);
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp | 
| Ryan Mallon | 29 | 100.00% | 1 | 100.00% | 
| Total | 29 | 100.00% | 1 | 100.00% | 
static inline unsigned ep93xx_i2s_read_reg(struct ep93xx_i2s_info *info,
					   unsigned reg)
{
	return __raw_readl(info->regs + reg);
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp | 
| Ryan Mallon | 25 | 100.00% | 1 | 100.00% | 
| Total | 25 | 100.00% | 1 | 100.00% | 
static void ep93xx_i2s_enable(struct ep93xx_i2s_info *info, int stream)
{
	unsigned base_reg;
	int i;
	if ((ep93xx_i2s_read_reg(info, EP93XX_I2S_TX0EN) & 0x1) == 0 &&
	    (ep93xx_i2s_read_reg(info, EP93XX_I2S_RX0EN) & 0x1) == 0) {
		/* Enable clocks */
		clk_enable(info->mclk);
		clk_enable(info->sclk);
		clk_enable(info->lrclk);
		/* Enable i2s */
		ep93xx_i2s_write_reg(info, EP93XX_I2S_GLCTRL, 1);
	}
	/* Enable fifos */
	if (stream == SNDRV_PCM_STREAM_PLAYBACK)
		base_reg = EP93XX_I2S_TX0EN;
	else
		base_reg = EP93XX_I2S_RX0EN;
	for (i = 0; i < 3; i++)
		ep93xx_i2s_write_reg(info, base_reg + (i * 4), 1);
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp | 
| Ryan Mallon | 126 | 100.00% | 1 | 100.00% | 
| Total | 126 | 100.00% | 1 | 100.00% | 
static void ep93xx_i2s_disable(struct ep93xx_i2s_info *info, int stream)
{
	unsigned base_reg;
	int i;
	/* Disable fifos */
	if (stream == SNDRV_PCM_STREAM_PLAYBACK)
		base_reg = EP93XX_I2S_TX0EN;
	else
		base_reg = EP93XX_I2S_RX0EN;
	for (i = 0; i < 3; i++)
		ep93xx_i2s_write_reg(info, base_reg + (i * 4), 0);
	if ((ep93xx_i2s_read_reg(info, EP93XX_I2S_TX0EN) & 0x1) == 0 &&
	    (ep93xx_i2s_read_reg(info, EP93XX_I2S_RX0EN) & 0x1) == 0) {
		/* Disable i2s */
		ep93xx_i2s_write_reg(info, EP93XX_I2S_GLCTRL, 0);
		/* Disable clocks */
		clk_disable(info->lrclk);
		clk_disable(info->sclk);
		clk_disable(info->mclk);
	}
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp | 
| Ryan Mallon | 126 | 100.00% | 1 | 100.00% | 
| Total | 126 | 100.00% | 1 | 100.00% | 
static int ep93xx_i2s_dai_probe(struct snd_soc_dai *dai)
{
	struct ep93xx_i2s_info *info = snd_soc_dai_get_drvdata(dai);
	info->dma_params_tx.filter_data =
		&ep93xx_i2s_dma_data[SNDRV_PCM_STREAM_PLAYBACK];
	info->dma_params_rx.filter_data =
		&ep93xx_i2s_dma_data[SNDRV_PCM_STREAM_CAPTURE];
	dai->playback_dma_data = &info->dma_params_tx;
	dai->capture_dma_data = &info->dma_params_rx;
	return 0;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp | 
| Stephen Warren | 35 | 53.03% | 1 | 33.33% | 
| Ryan Mallon | 20 | 30.30% | 1 | 33.33% | 
| Lars-Peter Clausen | 11 | 16.67% | 1 | 33.33% | 
| Total | 66 | 100.00% | 3 | 100.00% | 
static void ep93xx_i2s_shutdown(struct snd_pcm_substream *substream,
				struct snd_soc_dai *dai)
{
	struct ep93xx_i2s_info *info = snd_soc_dai_get_drvdata(dai);
	ep93xx_i2s_disable(info, substream->stream);
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp | 
| Ryan Mallon | 32 | 91.43% | 1 | 50.00% | 
| Liam Girdwood | 3 | 8.57% | 1 | 50.00% | 
| Total | 35 | 100.00% | 2 | 100.00% | 
static int ep93xx_i2s_set_dai_fmt(struct snd_soc_dai *cpu_dai,
				  unsigned int fmt)
{
	struct ep93xx_i2s_info *info = snd_soc_dai_get_drvdata(cpu_dai);
	unsigned int clk_cfg, lin_ctrl;
	clk_cfg  = ep93xx_i2s_read_reg(info, EP93XX_I2S_RXCLKCFG);
	lin_ctrl = ep93xx_i2s_read_reg(info, EP93XX_I2S_RXLINCTRLDATA);
	switch (fmt & SND_SOC_DAIFMT_FORMAT_MASK) {
	case SND_SOC_DAIFMT_I2S:
		clk_cfg |= EP93XX_I2S_CLKCFG_REL;
		lin_ctrl &= ~EP93XX_I2S_LINCTRLDATA_R_JUST;
		break;
	case SND_SOC_DAIFMT_LEFT_J:
		clk_cfg &= ~EP93XX_I2S_CLKCFG_REL;
		lin_ctrl &= ~EP93XX_I2S_LINCTRLDATA_R_JUST;
		break;
	case SND_SOC_DAIFMT_RIGHT_J:
		clk_cfg &= ~EP93XX_I2S_CLKCFG_REL;
		lin_ctrl |= EP93XX_I2S_LINCTRLDATA_R_JUST;
		break;
	default:
		return -EINVAL;
	}
	switch (fmt & SND_SOC_DAIFMT_MASTER_MASK) {
	case SND_SOC_DAIFMT_CBS_CFS:
		/* CPU is master */
		clk_cfg |= EP93XX_I2S_CLKCFG_MASTER;
		break;
	case SND_SOC_DAIFMT_CBM_CFM:
		/* Codec is master */
		clk_cfg &= ~EP93XX_I2S_CLKCFG_MASTER;
		break;
	default:
		return -EINVAL;
	}
	switch (fmt & SND_SOC_DAIFMT_INV_MASK) {
	case SND_SOC_DAIFMT_NB_NF:
		/* Negative bit clock, lrclk low on left word */
		clk_cfg &= ~(EP93XX_I2S_CLKCFG_CKP | EP93XX_I2S_CLKCFG_REL);
		break;
	case SND_SOC_DAIFMT_NB_IF:
		/* Negative bit clock, lrclk low on right word */
		clk_cfg &= ~EP93XX_I2S_CLKCFG_CKP;
		clk_cfg |= EP93XX_I2S_CLKCFG_REL;
		break;
	case SND_SOC_DAIFMT_IB_NF:
		/* Positive bit clock, lrclk low on left word */
		clk_cfg |= EP93XX_I2S_CLKCFG_CKP;
		clk_cfg &= ~EP93XX_I2S_CLKCFG_REL;
		break;
	case SND_SOC_DAIFMT_IB_IF:
		/* Positive bit clock, lrclk low on right word */
		clk_cfg |= EP93XX_I2S_CLKCFG_CKP | EP93XX_I2S_CLKCFG_REL;
		break;
	}
	/* Write new register values */
	ep93xx_i2s_write_reg(info, EP93XX_I2S_RXCLKCFG, clk_cfg);
	ep93xx_i2s_write_reg(info, EP93XX_I2S_TXCLKCFG, clk_cfg);
	ep93xx_i2s_write_reg(info, EP93XX_I2S_RXLINCTRLDATA, lin_ctrl);
	ep93xx_i2s_write_reg(info, EP93XX_I2S_TXLINCTRLDATA, lin_ctrl);
	return 0;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp | 
| Ryan Mallon | 232 | 98.72% | 1 | 50.00% | 
| Liam Girdwood | 3 | 1.28% | 1 | 50.00% | 
| Total | 235 | 100.00% | 2 | 100.00% | 
static int ep93xx_i2s_hw_params(struct snd_pcm_substream *substream,
				struct snd_pcm_hw_params *params,
				struct snd_soc_dai *dai)
{
	struct ep93xx_i2s_info *info = snd_soc_dai_get_drvdata(dai);
	unsigned word_len, div, sdiv, lrdiv;
	int err;
	switch (params_format(params)) {
	case SNDRV_PCM_FORMAT_S16_LE:
		word_len = EP93XX_I2S_WRDLEN_16;
		break;
	case SNDRV_PCM_FORMAT_S24_LE:
		word_len = EP93XX_I2S_WRDLEN_24;
		break;
	case SNDRV_PCM_FORMAT_S32_LE:
		word_len = EP93XX_I2S_WRDLEN_32;
		break;
	default:
		return -EINVAL;
	}
	if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK)
		ep93xx_i2s_write_reg(info, EP93XX_I2S_TXWRDLEN, word_len);
	else
		ep93xx_i2s_write_reg(info, EP93XX_I2S_RXWRDLEN, word_len);
	/*
         * EP93xx I2S module can be setup so SCLK / LRCLK value can be
         * 32, 64, 128. MCLK / SCLK value can be 2 and 4.
         * We set LRCLK equal to `rate' and minimum SCLK / LRCLK 
         * value is 64, because our sample size is 32 bit * 2 channels.
         * I2S standard permits us to transmit more bits than
         * the codec uses.
         */
	div = clk_get_rate(info->mclk) / params_rate(params);
	sdiv = 4;
	if (div > (256 + 512) / 2) {
		lrdiv = 128;
	} else {
		lrdiv = 64;
		if (div < (128 + 256) / 2)
			sdiv = 2;
	}
	err = clk_set_rate(info->sclk, clk_get_rate(info->mclk) / sdiv);
	if (err)
		return err;
	err = clk_set_rate(info->lrclk, clk_get_rate(info->sclk) / lrdiv);
	if (err)
		return err;
	ep93xx_i2s_enable(info, substream->stream);
	return 0;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp | 
| Ryan Mallon | 195 | 84.78% | 1 | 25.00% | 
| Alexander Sverdlin | 31 | 13.48% | 2 | 50.00% | 
| Liam Girdwood | 4 | 1.74% | 1 | 25.00% | 
| Total | 230 | 100.00% | 4 | 100.00% | 
static int ep93xx_i2s_set_sysclk(struct snd_soc_dai *cpu_dai, int clk_id,
				 unsigned int freq, int dir)
{
	struct ep93xx_i2s_info *info = snd_soc_dai_get_drvdata(cpu_dai);
	if (dir == SND_SOC_CLOCK_IN || clk_id != 0)
		return -EINVAL;
	return clk_set_rate(info->mclk, freq);
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp | 
| Ryan Mallon | 52 | 94.55% | 1 | 50.00% | 
| Liam Girdwood | 3 | 5.45% | 1 | 50.00% | 
| Total | 55 | 100.00% | 2 | 100.00% | 
#ifdef CONFIG_PM
static int ep93xx_i2s_suspend(struct snd_soc_dai *dai)
{
	struct ep93xx_i2s_info *info = snd_soc_dai_get_drvdata(dai);
	if (!dai->active)
		return 0;
	ep93xx_i2s_disable(info, SNDRV_PCM_STREAM_PLAYBACK);
	ep93xx_i2s_disable(info, SNDRV_PCM_STREAM_CAPTURE);
	return 0;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp | 
| Ryan Mallon | 39 | 81.25% | 1 | 33.33% | 
| Alexander Sverdlin | 6 | 12.50% | 1 | 33.33% | 
| Liam Girdwood | 3 | 6.25% | 1 | 33.33% | 
| Total | 48 | 100.00% | 3 | 100.00% | 
static int ep93xx_i2s_resume(struct snd_soc_dai *dai)
{
	struct ep93xx_i2s_info *info = snd_soc_dai_get_drvdata(dai);
	if (!dai->active)
		return 0;
	ep93xx_i2s_enable(info, SNDRV_PCM_STREAM_PLAYBACK);
	ep93xx_i2s_enable(info, SNDRV_PCM_STREAM_CAPTURE);
	return 0;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp | 
| Ryan Mallon | 39 | 81.25% | 1 | 33.33% | 
| Alexander Sverdlin | 6 | 12.50% | 1 | 33.33% | 
| Liam Girdwood | 3 | 6.25% | 1 | 33.33% | 
| Total | 48 | 100.00% | 3 | 100.00% | 
#else
#define ep93xx_i2s_suspend	NULL
#define ep93xx_i2s_resume	NULL
#endif
static const struct snd_soc_dai_ops ep93xx_i2s_dai_ops = {
	.shutdown	= ep93xx_i2s_shutdown,
	.hw_params	= ep93xx_i2s_hw_params,
	.set_sysclk	= ep93xx_i2s_set_sysclk,
	.set_fmt	= ep93xx_i2s_set_dai_fmt,
};
#define EP93XX_I2S_FORMATS (SNDRV_PCM_FMTBIT_S32_LE)
static struct snd_soc_dai_driver ep93xx_i2s_dai = {
	.symmetric_rates= 1,
	.probe		= ep93xx_i2s_dai_probe,
	.suspend	= ep93xx_i2s_suspend,
	.resume		= ep93xx_i2s_resume,
	.playback	= {
		.channels_min	= 2,
		.channels_max	= 2,
		.rates		= SNDRV_PCM_RATE_8000_192000,
		.formats	= EP93XX_I2S_FORMATS,
        },
	.capture	= {
		 .channels_min	= 2,
		 .channels_max	= 2,
		 .rates		= SNDRV_PCM_RATE_8000_192000,
		 .formats	= EP93XX_I2S_FORMATS,
        },
	.ops		= &ep93xx_i2s_dai_ops,
};
static const struct snd_soc_component_driver ep93xx_i2s_component = {
	.name		= "ep93xx-i2s",
};
static int ep93xx_i2s_probe(struct platform_device *pdev)
{
	struct ep93xx_i2s_info *info;
	struct resource *res;
	int err;
	info = devm_kzalloc(&pdev->dev, sizeof(*info), GFP_KERNEL);
	if (!info)
		return -ENOMEM;
	res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
	info->regs = devm_ioremap_resource(&pdev->dev, res);
	if (IS_ERR(info->regs))
		return PTR_ERR(info->regs);
	info->mclk = clk_get(&pdev->dev, "mclk");
	if (IS_ERR(info->mclk)) {
		err = PTR_ERR(info->mclk);
		goto fail;
	}
	info->sclk = clk_get(&pdev->dev, "sclk");
	if (IS_ERR(info->sclk)) {
		err = PTR_ERR(info->sclk);
		goto fail_put_mclk;
	}
	info->lrclk = clk_get(&pdev->dev, "lrclk");
	if (IS_ERR(info->lrclk)) {
		err = PTR_ERR(info->lrclk);
		goto fail_put_sclk;
	}
	dev_set_drvdata(&pdev->dev, info);
	err = snd_soc_register_component(&pdev->dev, &ep93xx_i2s_component,
					 &ep93xx_i2s_dai, 1);
	if (err)
		goto fail_put_lrclk;
	err = devm_ep93xx_pcm_platform_register(&pdev->dev);
	if (err)
		goto fail_unregister;
	return 0;
fail_unregister:
	snd_soc_unregister_component(&pdev->dev);
fail_put_lrclk:
	clk_put(info->lrclk);
fail_put_sclk:
	clk_put(info->sclk);
fail_put_mclk:
	clk_put(info->mclk);
fail:
	return err;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp | 
| Ryan Mallon | 226 | 75.08% | 1 | 16.67% | 
| H Hartley Sweeten | 27 | 8.97% | 1 | 16.67% | 
| Stephen Warren | 27 | 8.97% | 1 | 16.67% | 
| Thierry Reding | 10 | 3.32% | 1 | 16.67% | 
| Kuninori Morimoto | 6 | 1.99% | 1 | 16.67% | 
| Liam Girdwood | 5 | 1.66% | 1 | 16.67% | 
| Total | 301 | 100.00% | 6 | 100.00% | 
static int ep93xx_i2s_remove(struct platform_device *pdev)
{
	struct ep93xx_i2s_info *info = dev_get_drvdata(&pdev->dev);
	snd_soc_unregister_component(&pdev->dev);
	clk_put(info->lrclk);
	clk_put(info->sclk);
	clk_put(info->mclk);
	return 0;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp | 
| Ryan Mallon | 41 | 73.21% | 1 | 25.00% | 
| Liam Girdwood | 10 | 17.86% | 1 | 25.00% | 
| H Hartley Sweeten | 4 | 7.14% | 1 | 25.00% | 
| Kuninori Morimoto | 1 | 1.79% | 1 | 25.00% | 
| Total | 56 | 100.00% | 4 | 100.00% | 
static struct platform_driver ep93xx_i2s_driver = {
	.probe	= ep93xx_i2s_probe,
	.remove	= ep93xx_i2s_remove,
	.driver	= {
		.name	= "ep93xx-i2s",
        },
};
module_platform_driver(ep93xx_i2s_driver);
MODULE_ALIAS("platform:ep93xx-i2s");
MODULE_AUTHOR("Ryan Mallon");
MODULE_DESCRIPTION("EP93XX I2S driver");
MODULE_LICENSE("GPL");
Overall Contributors
| Person | Tokens | Prop | Commits | CommitProp | 
| Ryan Mallon | 1534 | 85.70% | 2 | 10.00% | 
| Stephen Warren | 76 | 4.25% | 2 | 10.00% | 
| Alexander Sverdlin | 46 | 2.57% | 4 | 20.00% | 
| Liam Girdwood | 36 | 2.01% | 1 | 5.00% | 
| Lars-Peter Clausen | 32 | 1.79% | 5 | 25.00% | 
| H Hartley Sweeten | 31 | 1.73% | 1 | 5.00% | 
| Kuninori Morimoto | 20 | 1.12% | 1 | 5.00% | 
| Thierry Reding | 10 | 0.56% | 1 | 5.00% | 
| Axel Lin | 2 | 0.11% | 1 | 5.00% | 
| Mika Westerberg | 2 | 0.11% | 1 | 5.00% | 
| Arnd Bergmann | 1 | 0.06% | 1 | 5.00% | 
| Total | 1790 | 100.00% | 20 | 100.00% | 
Information contained on this website is for historical information purposes only and does not indicate or represent copyright ownership.