mirror of
https://github.com/AetherDroid/android_kernel_samsung_on5xelte.git
synced 2025-10-29 07:18:51 +01:00
Fixed MTP to work with TWRP
This commit is contained in:
commit
f6dfaef42e
50820 changed files with 20846062 additions and 0 deletions
65
sound/soc/sh/Kconfig
Normal file
65
sound/soc/sh/Kconfig
Normal file
|
|
@ -0,0 +1,65 @@
|
|||
menu "SoC Audio support for SuperH"
|
||||
depends on SUPERH || ARCH_SHMOBILE
|
||||
|
||||
config SND_SOC_PCM_SH7760
|
||||
tristate "SoC Audio support for Renesas SH7760"
|
||||
depends on CPU_SUBTYPE_SH7760 && SH_DMABRG
|
||||
help
|
||||
Enable this option for SH7760 AC97/I2S audio support.
|
||||
|
||||
|
||||
##
|
||||
## Audio unit modules
|
||||
##
|
||||
|
||||
config SND_SOC_SH4_HAC
|
||||
tristate
|
||||
select AC97_BUS
|
||||
select SND_SOC_AC97_BUS
|
||||
|
||||
config SND_SOC_SH4_SSI
|
||||
tristate
|
||||
|
||||
config SND_SOC_SH4_FSI
|
||||
tristate "SH4 FSI support"
|
||||
select SND_SIMPLE_CARD
|
||||
help
|
||||
This option enables FSI sound support
|
||||
|
||||
config SND_SOC_SH4_SIU
|
||||
tristate
|
||||
depends on (SUPERH || ARCH_SHMOBILE) && HAVE_CLK
|
||||
select DMA_ENGINE
|
||||
select DMADEVICES
|
||||
select SH_DMAE
|
||||
select FW_LOADER
|
||||
|
||||
config SND_SOC_RCAR
|
||||
tristate "R-Car series SRU/SCU/SSIU/SSI support"
|
||||
select SND_SIMPLE_CARD
|
||||
select REGMAP_MMIO
|
||||
help
|
||||
This option enables R-Car SUR/SCU/SSIU/SSI sound support
|
||||
|
||||
##
|
||||
## Boards
|
||||
##
|
||||
|
||||
config SND_SH7760_AC97
|
||||
tristate "SH7760 AC97 sound support"
|
||||
depends on CPU_SUBTYPE_SH7760 && SND_SOC_PCM_SH7760
|
||||
select SND_SOC_SH4_HAC
|
||||
select SND_SOC_AC97_CODEC
|
||||
help
|
||||
This option enables generic sound support for the first
|
||||
AC97 unit of the SH7760.
|
||||
|
||||
config SND_SIU_MIGOR
|
||||
tristate "SIU sound support on Migo-R"
|
||||
depends on SH_MIGOR && I2C
|
||||
select SND_SOC_SH4_SIU
|
||||
select SND_SOC_WM8978
|
||||
help
|
||||
This option enables sound support for the SH7722 Migo-R board
|
||||
|
||||
endmenu
|
||||
23
sound/soc/sh/Makefile
Normal file
23
sound/soc/sh/Makefile
Normal file
|
|
@ -0,0 +1,23 @@
|
|||
## DMA engines
|
||||
snd-soc-dma-sh7760-objs := dma-sh7760.o
|
||||
obj-$(CONFIG_SND_SOC_PCM_SH7760) += snd-soc-dma-sh7760.o
|
||||
|
||||
## audio units found on some SH-4
|
||||
snd-soc-hac-objs := hac.o
|
||||
snd-soc-ssi-objs := ssi.o
|
||||
snd-soc-fsi-objs := fsi.o
|
||||
snd-soc-siu-objs := siu_pcm.o siu_dai.o
|
||||
obj-$(CONFIG_SND_SOC_SH4_HAC) += snd-soc-hac.o
|
||||
obj-$(CONFIG_SND_SOC_SH4_SSI) += snd-soc-ssi.o
|
||||
obj-$(CONFIG_SND_SOC_SH4_FSI) += snd-soc-fsi.o
|
||||
obj-$(CONFIG_SND_SOC_SH4_SIU) += snd-soc-siu.o
|
||||
|
||||
## audio units for R-Car
|
||||
obj-$(CONFIG_SND_SOC_RCAR) += rcar/
|
||||
|
||||
## boards
|
||||
snd-soc-sh7760-ac97-objs := sh7760-ac97.o
|
||||
snd-soc-migor-objs := migor.o
|
||||
|
||||
obj-$(CONFIG_SND_SH7760_AC97) += snd-soc-sh7760-ac97.o
|
||||
obj-$(CONFIG_SND_SIU_MIGOR) += snd-soc-migor.o
|
||||
359
sound/soc/sh/dma-sh7760.c
Normal file
359
sound/soc/sh/dma-sh7760.c
Normal file
|
|
@ -0,0 +1,359 @@
|
|||
/*
|
||||
* SH7760 ("camelot") DMABRG audio DMA unit support
|
||||
*
|
||||
* Copyright (C) 2007 Manuel Lauss <mano@roarinelk.homelinux.net>
|
||||
* licensed under the terms outlined in the file COPYING at the root
|
||||
* of the linux kernel sources.
|
||||
*
|
||||
* The SH7760 DMABRG provides 4 dma channels (2x rec, 2x play), which
|
||||
* trigger an interrupt when one half of the programmed transfer size
|
||||
* has been xmitted.
|
||||
*
|
||||
* FIXME: little-endian only for now
|
||||
*/
|
||||
|
||||
#include <linux/module.h>
|
||||
#include <linux/gfp.h>
|
||||
#include <linux/init.h>
|
||||
#include <linux/platform_device.h>
|
||||
#include <linux/dma-mapping.h>
|
||||
#include <sound/core.h>
|
||||
#include <sound/pcm.h>
|
||||
#include <sound/pcm_params.h>
|
||||
#include <sound/soc.h>
|
||||
#include <asm/dmabrg.h>
|
||||
|
||||
|
||||
/* registers and bits */
|
||||
#define BRGATXSAR 0x00
|
||||
#define BRGARXDAR 0x04
|
||||
#define BRGATXTCR 0x08
|
||||
#define BRGARXTCR 0x0C
|
||||
#define BRGACR 0x10
|
||||
#define BRGATXTCNT 0x14
|
||||
#define BRGARXTCNT 0x18
|
||||
|
||||
#define ACR_RAR (1 << 18)
|
||||
#define ACR_RDS (1 << 17)
|
||||
#define ACR_RDE (1 << 16)
|
||||
#define ACR_TAR (1 << 2)
|
||||
#define ACR_TDS (1 << 1)
|
||||
#define ACR_TDE (1 << 0)
|
||||
|
||||
/* receiver/transmitter data alignment */
|
||||
#define ACR_RAM_NONE (0 << 24)
|
||||
#define ACR_RAM_4BYTE (1 << 24)
|
||||
#define ACR_RAM_2WORD (2 << 24)
|
||||
#define ACR_TAM_NONE (0 << 8)
|
||||
#define ACR_TAM_4BYTE (1 << 8)
|
||||
#define ACR_TAM_2WORD (2 << 8)
|
||||
|
||||
|
||||
struct camelot_pcm {
|
||||
unsigned long mmio; /* DMABRG audio channel control reg MMIO */
|
||||
unsigned int txid; /* ID of first DMABRG IRQ for this unit */
|
||||
|
||||
struct snd_pcm_substream *tx_ss;
|
||||
unsigned long tx_period_size;
|
||||
unsigned int tx_period;
|
||||
|
||||
struct snd_pcm_substream *rx_ss;
|
||||
unsigned long rx_period_size;
|
||||
unsigned int rx_period;
|
||||
|
||||
} cam_pcm_data[2] = {
|
||||
{
|
||||
.mmio = 0xFE3C0040,
|
||||
.txid = DMABRGIRQ_A0TXF,
|
||||
},
|
||||
{
|
||||
.mmio = 0xFE3C0060,
|
||||
.txid = DMABRGIRQ_A1TXF,
|
||||
},
|
||||
};
|
||||
|
||||
#define BRGREG(x) (*(unsigned long *)(cam->mmio + (x)))
|
||||
|
||||
/*
|
||||
* set a minimum of 16kb per period, to avoid interrupt-"storm" and
|
||||
* resulting skipping. In general, the bigger the minimum size, the
|
||||
* better for overall system performance. (The SH7760 is a puny CPU
|
||||
* with a slow SDRAM interface and poor internal bus bandwidth,
|
||||
* *especially* when the LCDC is active). The minimum for the DMAC
|
||||
* is 8 bytes; 16kbytes are enough to get skip-free playback of a
|
||||
* 44kHz/16bit/stereo MP3 on a lightly loaded system, and maintain
|
||||
* reasonable responsiveness in MPlayer.
|
||||
*/
|
||||
#define DMABRG_PERIOD_MIN 16 * 1024
|
||||
#define DMABRG_PERIOD_MAX 0x03fffffc
|
||||
#define DMABRG_PREALLOC_BUFFER 32 * 1024
|
||||
#define DMABRG_PREALLOC_BUFFER_MAX 32 * 1024
|
||||
|
||||
static struct snd_pcm_hardware camelot_pcm_hardware = {
|
||||
.info = (SNDRV_PCM_INFO_MMAP |
|
||||
SNDRV_PCM_INFO_INTERLEAVED |
|
||||
SNDRV_PCM_INFO_BLOCK_TRANSFER |
|
||||
SNDRV_PCM_INFO_MMAP_VALID |
|
||||
SNDRV_PCM_INFO_BATCH),
|
||||
.buffer_bytes_max = DMABRG_PERIOD_MAX,
|
||||
.period_bytes_min = DMABRG_PERIOD_MIN,
|
||||
.period_bytes_max = DMABRG_PERIOD_MAX / 2,
|
||||
.periods_min = 2,
|
||||
.periods_max = 2,
|
||||
.fifo_size = 128,
|
||||
};
|
||||
|
||||
static void camelot_txdma(void *data)
|
||||
{
|
||||
struct camelot_pcm *cam = data;
|
||||
cam->tx_period ^= 1;
|
||||
snd_pcm_period_elapsed(cam->tx_ss);
|
||||
}
|
||||
|
||||
static void camelot_rxdma(void *data)
|
||||
{
|
||||
struct camelot_pcm *cam = data;
|
||||
cam->rx_period ^= 1;
|
||||
snd_pcm_period_elapsed(cam->rx_ss);
|
||||
}
|
||||
|
||||
static int camelot_pcm_open(struct snd_pcm_substream *substream)
|
||||
{
|
||||
struct snd_soc_pcm_runtime *rtd = substream->private_data;
|
||||
struct camelot_pcm *cam = &cam_pcm_data[rtd->cpu_dai->id];
|
||||
int recv = substream->stream == SNDRV_PCM_STREAM_PLAYBACK ? 0:1;
|
||||
int ret, dmairq;
|
||||
|
||||
snd_soc_set_runtime_hwparams(substream, &camelot_pcm_hardware);
|
||||
|
||||
/* DMABRG buffer half/full events */
|
||||
dmairq = (recv) ? cam->txid + 2 : cam->txid;
|
||||
if (recv) {
|
||||
cam->rx_ss = substream;
|
||||
ret = dmabrg_request_irq(dmairq, camelot_rxdma, cam);
|
||||
if (unlikely(ret)) {
|
||||
pr_debug("audio unit %d irqs already taken!\n",
|
||||
rtd->cpu_dai->id);
|
||||
return -EBUSY;
|
||||
}
|
||||
(void)dmabrg_request_irq(dmairq + 1,camelot_rxdma, cam);
|
||||
} else {
|
||||
cam->tx_ss = substream;
|
||||
ret = dmabrg_request_irq(dmairq, camelot_txdma, cam);
|
||||
if (unlikely(ret)) {
|
||||
pr_debug("audio unit %d irqs already taken!\n",
|
||||
rtd->cpu_dai->id);
|
||||
return -EBUSY;
|
||||
}
|
||||
(void)dmabrg_request_irq(dmairq + 1, camelot_txdma, cam);
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int camelot_pcm_close(struct snd_pcm_substream *substream)
|
||||
{
|
||||
struct snd_soc_pcm_runtime *rtd = substream->private_data;
|
||||
struct camelot_pcm *cam = &cam_pcm_data[rtd->cpu_dai->id];
|
||||
int recv = substream->stream == SNDRV_PCM_STREAM_PLAYBACK ? 0:1;
|
||||
int dmairq;
|
||||
|
||||
dmairq = (recv) ? cam->txid + 2 : cam->txid;
|
||||
|
||||
if (recv)
|
||||
cam->rx_ss = NULL;
|
||||
else
|
||||
cam->tx_ss = NULL;
|
||||
|
||||
dmabrg_free_irq(dmairq + 1);
|
||||
dmabrg_free_irq(dmairq);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int camelot_hw_params(struct snd_pcm_substream *substream,
|
||||
struct snd_pcm_hw_params *hw_params)
|
||||
{
|
||||
struct snd_soc_pcm_runtime *rtd = substream->private_data;
|
||||
struct camelot_pcm *cam = &cam_pcm_data[rtd->cpu_dai->id];
|
||||
int recv = substream->stream == SNDRV_PCM_STREAM_PLAYBACK ? 0:1;
|
||||
int ret;
|
||||
|
||||
ret = snd_pcm_lib_malloc_pages(substream,
|
||||
params_buffer_bytes(hw_params));
|
||||
if (ret < 0)
|
||||
return ret;
|
||||
|
||||
if (recv) {
|
||||
cam->rx_period_size = params_period_bytes(hw_params);
|
||||
cam->rx_period = 0;
|
||||
} else {
|
||||
cam->tx_period_size = params_period_bytes(hw_params);
|
||||
cam->tx_period = 0;
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int camelot_hw_free(struct snd_pcm_substream *substream)
|
||||
{
|
||||
return snd_pcm_lib_free_pages(substream);
|
||||
}
|
||||
|
||||
static int camelot_prepare(struct snd_pcm_substream *substream)
|
||||
{
|
||||
struct snd_pcm_runtime *runtime = substream->runtime;
|
||||
struct snd_soc_pcm_runtime *rtd = substream->private_data;
|
||||
struct camelot_pcm *cam = &cam_pcm_data[rtd->cpu_dai->id];
|
||||
|
||||
pr_debug("PCM data: addr 0x%08ulx len %d\n",
|
||||
(u32)runtime->dma_addr, runtime->dma_bytes);
|
||||
|
||||
if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) {
|
||||
BRGREG(BRGATXSAR) = (unsigned long)runtime->dma_area;
|
||||
BRGREG(BRGATXTCR) = runtime->dma_bytes;
|
||||
} else {
|
||||
BRGREG(BRGARXDAR) = (unsigned long)runtime->dma_area;
|
||||
BRGREG(BRGARXTCR) = runtime->dma_bytes;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static inline void dmabrg_play_dma_start(struct camelot_pcm *cam)
|
||||
{
|
||||
unsigned long acr = BRGREG(BRGACR) & ~(ACR_TDS | ACR_RDS);
|
||||
/* start DMABRG engine: XFER start, auto-addr-reload */
|
||||
BRGREG(BRGACR) = acr | ACR_TDE | ACR_TAR | ACR_TAM_2WORD;
|
||||
}
|
||||
|
||||
static inline void dmabrg_play_dma_stop(struct camelot_pcm *cam)
|
||||
{
|
||||
unsigned long acr = BRGREG(BRGACR) & ~(ACR_TDS | ACR_RDS);
|
||||
/* forcibly terminate data transmission */
|
||||
BRGREG(BRGACR) = acr | ACR_TDS;
|
||||
}
|
||||
|
||||
static inline void dmabrg_rec_dma_start(struct camelot_pcm *cam)
|
||||
{
|
||||
unsigned long acr = BRGREG(BRGACR) & ~(ACR_TDS | ACR_RDS);
|
||||
/* start DMABRG engine: recv start, auto-reload */
|
||||
BRGREG(BRGACR) = acr | ACR_RDE | ACR_RAR | ACR_RAM_2WORD;
|
||||
}
|
||||
|
||||
static inline void dmabrg_rec_dma_stop(struct camelot_pcm *cam)
|
||||
{
|
||||
unsigned long acr = BRGREG(BRGACR) & ~(ACR_TDS | ACR_RDS);
|
||||
/* forcibly terminate data receiver */
|
||||
BRGREG(BRGACR) = acr | ACR_RDS;
|
||||
}
|
||||
|
||||
static int camelot_trigger(struct snd_pcm_substream *substream, int cmd)
|
||||
{
|
||||
struct snd_soc_pcm_runtime *rtd = substream->private_data;
|
||||
struct camelot_pcm *cam = &cam_pcm_data[rtd->cpu_dai->id];
|
||||
int recv = substream->stream == SNDRV_PCM_STREAM_PLAYBACK ? 0:1;
|
||||
|
||||
switch (cmd) {
|
||||
case SNDRV_PCM_TRIGGER_START:
|
||||
if (recv)
|
||||
dmabrg_rec_dma_start(cam);
|
||||
else
|
||||
dmabrg_play_dma_start(cam);
|
||||
break;
|
||||
case SNDRV_PCM_TRIGGER_STOP:
|
||||
if (recv)
|
||||
dmabrg_rec_dma_stop(cam);
|
||||
else
|
||||
dmabrg_play_dma_stop(cam);
|
||||
break;
|
||||
default:
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static snd_pcm_uframes_t camelot_pos(struct snd_pcm_substream *substream)
|
||||
{
|
||||
struct snd_pcm_runtime *runtime = substream->runtime;
|
||||
struct snd_soc_pcm_runtime *rtd = substream->private_data;
|
||||
struct camelot_pcm *cam = &cam_pcm_data[rtd->cpu_dai->id];
|
||||
int recv = substream->stream == SNDRV_PCM_STREAM_PLAYBACK ? 0:1;
|
||||
unsigned long pos;
|
||||
|
||||
/* cannot use the DMABRG pointer register: under load, by the
|
||||
* time ALSA comes around to read the register, it is already
|
||||
* far ahead (or worse, already done with the fragment) of the
|
||||
* position at the time the IRQ was triggered, which results in
|
||||
* fast-playback sound in my test application (ScummVM)
|
||||
*/
|
||||
if (recv)
|
||||
pos = cam->rx_period ? cam->rx_period_size : 0;
|
||||
else
|
||||
pos = cam->tx_period ? cam->tx_period_size : 0;
|
||||
|
||||
return bytes_to_frames(runtime, pos);
|
||||
}
|
||||
|
||||
static struct snd_pcm_ops camelot_pcm_ops = {
|
||||
.open = camelot_pcm_open,
|
||||
.close = camelot_pcm_close,
|
||||
.ioctl = snd_pcm_lib_ioctl,
|
||||
.hw_params = camelot_hw_params,
|
||||
.hw_free = camelot_hw_free,
|
||||
.prepare = camelot_prepare,
|
||||
.trigger = camelot_trigger,
|
||||
.pointer = camelot_pos,
|
||||
};
|
||||
|
||||
static void camelot_pcm_free(struct snd_pcm *pcm)
|
||||
{
|
||||
snd_pcm_lib_preallocate_free_for_all(pcm);
|
||||
}
|
||||
|
||||
static int camelot_pcm_new(struct snd_soc_pcm_runtime *rtd)
|
||||
{
|
||||
struct snd_pcm *pcm = rtd->pcm;
|
||||
|
||||
/* dont use SNDRV_DMA_TYPE_DEV, since it will oops the SH kernel
|
||||
* in MMAP mode (i.e. aplay -M)
|
||||
*/
|
||||
snd_pcm_lib_preallocate_pages_for_all(pcm,
|
||||
SNDRV_DMA_TYPE_CONTINUOUS,
|
||||
snd_dma_continuous_data(GFP_KERNEL),
|
||||
DMABRG_PREALLOC_BUFFER, DMABRG_PREALLOC_BUFFER_MAX);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static struct snd_soc_platform_driver sh7760_soc_platform = {
|
||||
.ops = &camelot_pcm_ops,
|
||||
.pcm_new = camelot_pcm_new,
|
||||
.pcm_free = camelot_pcm_free,
|
||||
};
|
||||
|
||||
static int sh7760_soc_platform_probe(struct platform_device *pdev)
|
||||
{
|
||||
return snd_soc_register_platform(&pdev->dev, &sh7760_soc_platform);
|
||||
}
|
||||
|
||||
static int sh7760_soc_platform_remove(struct platform_device *pdev)
|
||||
{
|
||||
snd_soc_unregister_platform(&pdev->dev);
|
||||
return 0;
|
||||
}
|
||||
|
||||
static struct platform_driver sh7760_pcm_driver = {
|
||||
.driver = {
|
||||
.name = "sh7760-pcm-audio",
|
||||
.owner = THIS_MODULE,
|
||||
},
|
||||
|
||||
.probe = sh7760_soc_platform_probe,
|
||||
.remove = sh7760_soc_platform_remove,
|
||||
};
|
||||
|
||||
module_platform_driver(sh7760_pcm_driver);
|
||||
|
||||
MODULE_LICENSE("GPL");
|
||||
MODULE_DESCRIPTION("SH7760 Audio DMA (DMABRG) driver");
|
||||
MODULE_AUTHOR("Manuel Lauss <mano@roarinelk.homelinux.net>");
|
||||
2137
sound/soc/sh/fsi.c
Normal file
2137
sound/soc/sh/fsi.c
Normal file
File diff suppressed because it is too large
Load diff
347
sound/soc/sh/hac.c
Normal file
347
sound/soc/sh/hac.c
Normal file
|
|
@ -0,0 +1,347 @@
|
|||
/*
|
||||
* Hitachi Audio Controller (AC97) support for SH7760/SH7780
|
||||
*
|
||||
* Copyright (c) 2007 Manuel Lauss <mano@roarinelk.homelinux.net>
|
||||
* licensed under the terms outlined in the file COPYING at the root
|
||||
* of the linux kernel sources.
|
||||
*
|
||||
* dont forget to set IPSEL/OMSEL register bits (in your board code) to
|
||||
* enable HAC output pins!
|
||||
*/
|
||||
|
||||
/* BIG FAT FIXME: although the SH7760 has 2 independent AC97 units, only
|
||||
* the FIRST can be used since ASoC does not pass any information to the
|
||||
* ac97_read/write() functions regarding WHICH unit to use. You'll have
|
||||
* to edit the code a bit to use the other AC97 unit. --mlau
|
||||
*/
|
||||
|
||||
#include <linux/init.h>
|
||||
#include <linux/module.h>
|
||||
#include <linux/platform_device.h>
|
||||
#include <linux/interrupt.h>
|
||||
#include <linux/wait.h>
|
||||
#include <linux/delay.h>
|
||||
#include <sound/core.h>
|
||||
#include <sound/pcm.h>
|
||||
#include <sound/ac97_codec.h>
|
||||
#include <sound/initval.h>
|
||||
#include <sound/soc.h>
|
||||
|
||||
/* regs and bits */
|
||||
#define HACCR 0x08
|
||||
#define HACCSAR 0x20
|
||||
#define HACCSDR 0x24
|
||||
#define HACPCML 0x28
|
||||
#define HACPCMR 0x2C
|
||||
#define HACTIER 0x50
|
||||
#define HACTSR 0x54
|
||||
#define HACRIER 0x58
|
||||
#define HACRSR 0x5C
|
||||
#define HACACR 0x60
|
||||
|
||||
#define CR_CR (1 << 15) /* "codec-ready" indicator */
|
||||
#define CR_CDRT (1 << 11) /* cold reset */
|
||||
#define CR_WMRT (1 << 10) /* warm reset */
|
||||
#define CR_B9 (1 << 9) /* the mysterious "bit 9" */
|
||||
#define CR_ST (1 << 5) /* AC97 link start bit */
|
||||
|
||||
#define CSAR_RD (1 << 19) /* AC97 data read bit */
|
||||
#define CSAR_WR (0)
|
||||
|
||||
#define TSR_CMDAMT (1 << 31)
|
||||
#define TSR_CMDDMT (1 << 30)
|
||||
|
||||
#define RSR_STARY (1 << 22)
|
||||
#define RSR_STDRY (1 << 21)
|
||||
|
||||
#define ACR_DMARX16 (1 << 30)
|
||||
#define ACR_DMATX16 (1 << 29)
|
||||
#define ACR_TX12ATOM (1 << 26)
|
||||
#define ACR_DMARX20 ((1 << 24) | (1 << 22))
|
||||
#define ACR_DMATX20 ((1 << 23) | (1 << 21))
|
||||
|
||||
#define CSDR_SHIFT 4
|
||||
#define CSDR_MASK (0xffff << CSDR_SHIFT)
|
||||
#define CSAR_SHIFT 12
|
||||
#define CSAR_MASK (0x7f << CSAR_SHIFT)
|
||||
|
||||
#define AC97_WRITE_RETRY 1
|
||||
#define AC97_READ_RETRY 5
|
||||
|
||||
/* manual-suggested AC97 codec access timeouts (us) */
|
||||
#define TMO_E1 500 /* 21 < E1 < 1000 */
|
||||
#define TMO_E2 13 /* 13 < E2 */
|
||||
#define TMO_E3 21 /* 21 < E3 */
|
||||
#define TMO_E4 500 /* 21 < E4 < 1000 */
|
||||
|
||||
struct hac_priv {
|
||||
unsigned long mmio; /* HAC base address */
|
||||
} hac_cpu_data[] = {
|
||||
#if defined(CONFIG_CPU_SUBTYPE_SH7760)
|
||||
{
|
||||
.mmio = 0xFE240000,
|
||||
},
|
||||
{
|
||||
.mmio = 0xFE250000,
|
||||
},
|
||||
#elif defined(CONFIG_CPU_SUBTYPE_SH7780)
|
||||
{
|
||||
.mmio = 0xFFE40000,
|
||||
},
|
||||
#else
|
||||
#error "Unsupported SuperH SoC"
|
||||
#endif
|
||||
};
|
||||
|
||||
#define HACREG(reg) (*(unsigned long *)(hac->mmio + (reg)))
|
||||
|
||||
/*
|
||||
* AC97 read/write flow as outlined in the SH7760 manual (pages 903-906)
|
||||
*/
|
||||
static int hac_get_codec_data(struct hac_priv *hac, unsigned short r,
|
||||
unsigned short *v)
|
||||
{
|
||||
unsigned int to1, to2, i;
|
||||
unsigned short adr;
|
||||
|
||||
for (i = AC97_READ_RETRY; i; i--) {
|
||||
*v = 0;
|
||||
/* wait for HAC to receive something from the codec */
|
||||
for (to1 = TMO_E4;
|
||||
to1 && !(HACREG(HACRSR) & RSR_STARY);
|
||||
--to1)
|
||||
udelay(1);
|
||||
for (to2 = TMO_E4;
|
||||
to2 && !(HACREG(HACRSR) & RSR_STDRY);
|
||||
--to2)
|
||||
udelay(1);
|
||||
|
||||
if (!to1 && !to2)
|
||||
return 0; /* codec comm is down */
|
||||
|
||||
adr = ((HACREG(HACCSAR) & CSAR_MASK) >> CSAR_SHIFT);
|
||||
*v = ((HACREG(HACCSDR) & CSDR_MASK) >> CSDR_SHIFT);
|
||||
|
||||
HACREG(HACRSR) &= ~(RSR_STDRY | RSR_STARY);
|
||||
|
||||
if (r == adr)
|
||||
break;
|
||||
|
||||
/* manual says: wait at least 21 usec before retrying */
|
||||
udelay(21);
|
||||
}
|
||||
HACREG(HACRSR) &= ~(RSR_STDRY | RSR_STARY);
|
||||
return i;
|
||||
}
|
||||
|
||||
static unsigned short hac_read_codec_aux(struct hac_priv *hac,
|
||||
unsigned short reg)
|
||||
{
|
||||
unsigned short val;
|
||||
unsigned int i, to;
|
||||
|
||||
for (i = AC97_READ_RETRY; i; i--) {
|
||||
/* send_read_request */
|
||||
local_irq_disable();
|
||||
HACREG(HACTSR) &= ~(TSR_CMDAMT);
|
||||
HACREG(HACCSAR) = (reg << CSAR_SHIFT) | CSAR_RD;
|
||||
local_irq_enable();
|
||||
|
||||
for (to = TMO_E3;
|
||||
to && !(HACREG(HACTSR) & TSR_CMDAMT);
|
||||
--to)
|
||||
udelay(1);
|
||||
|
||||
HACREG(HACTSR) &= ~TSR_CMDAMT;
|
||||
val = 0;
|
||||
if (hac_get_codec_data(hac, reg, &val) != 0)
|
||||
break;
|
||||
}
|
||||
|
||||
return i ? val : ~0;
|
||||
}
|
||||
|
||||
static void hac_ac97_write(struct snd_ac97 *ac97, unsigned short reg,
|
||||
unsigned short val)
|
||||
{
|
||||
int unit_id = 0 /* ac97->private_data */;
|
||||
struct hac_priv *hac = &hac_cpu_data[unit_id];
|
||||
unsigned int i, to;
|
||||
/* write_codec_aux */
|
||||
for (i = AC97_WRITE_RETRY; i; i--) {
|
||||
/* send_write_request */
|
||||
local_irq_disable();
|
||||
HACREG(HACTSR) &= ~(TSR_CMDDMT | TSR_CMDAMT);
|
||||
HACREG(HACCSDR) = (val << CSDR_SHIFT);
|
||||
HACREG(HACCSAR) = (reg << CSAR_SHIFT) & (~CSAR_RD);
|
||||
local_irq_enable();
|
||||
|
||||
/* poll-wait for CMDAMT and CMDDMT */
|
||||
for (to = TMO_E1;
|
||||
to && !(HACREG(HACTSR) & (TSR_CMDAMT|TSR_CMDDMT));
|
||||
--to)
|
||||
udelay(1);
|
||||
|
||||
HACREG(HACTSR) &= ~(TSR_CMDAMT | TSR_CMDDMT);
|
||||
if (to)
|
||||
break;
|
||||
/* timeout, try again */
|
||||
}
|
||||
}
|
||||
|
||||
static unsigned short hac_ac97_read(struct snd_ac97 *ac97,
|
||||
unsigned short reg)
|
||||
{
|
||||
int unit_id = 0 /* ac97->private_data */;
|
||||
struct hac_priv *hac = &hac_cpu_data[unit_id];
|
||||
return hac_read_codec_aux(hac, reg);
|
||||
}
|
||||
|
||||
static void hac_ac97_warmrst(struct snd_ac97 *ac97)
|
||||
{
|
||||
int unit_id = 0 /* ac97->private_data */;
|
||||
struct hac_priv *hac = &hac_cpu_data[unit_id];
|
||||
unsigned int tmo;
|
||||
|
||||
HACREG(HACCR) = CR_WMRT | CR_ST | CR_B9;
|
||||
msleep(10);
|
||||
HACREG(HACCR) = CR_ST | CR_B9;
|
||||
for (tmo = 1000; (tmo > 0) && !(HACREG(HACCR) & CR_CR); tmo--)
|
||||
udelay(1);
|
||||
|
||||
if (!tmo)
|
||||
printk(KERN_INFO "hac: reset: AC97 link down!\n");
|
||||
/* settings this bit lets us have a conversation with codec */
|
||||
HACREG(HACACR) |= ACR_TX12ATOM;
|
||||
}
|
||||
|
||||
static void hac_ac97_coldrst(struct snd_ac97 *ac97)
|
||||
{
|
||||
int unit_id = 0 /* ac97->private_data */;
|
||||
struct hac_priv *hac;
|
||||
hac = &hac_cpu_data[unit_id];
|
||||
|
||||
HACREG(HACCR) = 0;
|
||||
HACREG(HACCR) = CR_CDRT | CR_ST | CR_B9;
|
||||
msleep(10);
|
||||
hac_ac97_warmrst(ac97);
|
||||
}
|
||||
|
||||
static struct snd_ac97_bus_ops hac_ac97_ops = {
|
||||
.read = hac_ac97_read,
|
||||
.write = hac_ac97_write,
|
||||
.reset = hac_ac97_coldrst,
|
||||
.warm_reset = hac_ac97_warmrst,
|
||||
};
|
||||
|
||||
static int hac_hw_params(struct snd_pcm_substream *substream,
|
||||
struct snd_pcm_hw_params *params,
|
||||
struct snd_soc_dai *dai)
|
||||
{
|
||||
struct hac_priv *hac = &hac_cpu_data[dai->id];
|
||||
int d = substream->stream == SNDRV_PCM_STREAM_PLAYBACK ? 0 : 1;
|
||||
|
||||
switch (params->msbits) {
|
||||
case 16:
|
||||
HACREG(HACACR) |= d ? ACR_DMARX16 : ACR_DMATX16;
|
||||
HACREG(HACACR) &= d ? ~ACR_DMARX20 : ~ACR_DMATX20;
|
||||
break;
|
||||
case 20:
|
||||
HACREG(HACACR) &= d ? ~ACR_DMARX16 : ~ACR_DMATX16;
|
||||
HACREG(HACACR) |= d ? ACR_DMARX20 : ACR_DMATX20;
|
||||
break;
|
||||
default:
|
||||
pr_debug("hac: invalid depth %d bit\n", params->msbits);
|
||||
return -EINVAL;
|
||||
break;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
#define AC97_RATES \
|
||||
SNDRV_PCM_RATE_8000_192000
|
||||
|
||||
#define AC97_FMTS \
|
||||
SNDRV_PCM_FMTBIT_S16_LE
|
||||
|
||||
static const struct snd_soc_dai_ops hac_dai_ops = {
|
||||
.hw_params = hac_hw_params,
|
||||
};
|
||||
|
||||
static struct snd_soc_dai_driver sh4_hac_dai[] = {
|
||||
{
|
||||
.name = "hac-dai.0",
|
||||
.ac97_control = 1,
|
||||
.playback = {
|
||||
.rates = AC97_RATES,
|
||||
.formats = AC97_FMTS,
|
||||
.channels_min = 2,
|
||||
.channels_max = 2,
|
||||
},
|
||||
.capture = {
|
||||
.rates = AC97_RATES,
|
||||
.formats = AC97_FMTS,
|
||||
.channels_min = 2,
|
||||
.channels_max = 2,
|
||||
},
|
||||
.ops = &hac_dai_ops,
|
||||
},
|
||||
#ifdef CONFIG_CPU_SUBTYPE_SH7760
|
||||
{
|
||||
.name = "hac-dai.1",
|
||||
.id = 1,
|
||||
.playback = {
|
||||
.rates = AC97_RATES,
|
||||
.formats = AC97_FMTS,
|
||||
.channels_min = 2,
|
||||
.channels_max = 2,
|
||||
},
|
||||
.capture = {
|
||||
.rates = AC97_RATES,
|
||||
.formats = AC97_FMTS,
|
||||
.channels_min = 2,
|
||||
.channels_max = 2,
|
||||
},
|
||||
.ops = &hac_dai_ops,
|
||||
|
||||
},
|
||||
#endif
|
||||
};
|
||||
|
||||
static const struct snd_soc_component_driver sh4_hac_component = {
|
||||
.name = "sh4-hac",
|
||||
};
|
||||
|
||||
static int hac_soc_platform_probe(struct platform_device *pdev)
|
||||
{
|
||||
ret = snd_soc_set_ac97_ops(&hac_ac97_ops);
|
||||
if (ret != 0)
|
||||
return ret;
|
||||
|
||||
return snd_soc_register_component(&pdev->dev, &sh4_hac_component,
|
||||
sh4_hac_dai, ARRAY_SIZE(sh4_hac_dai));
|
||||
}
|
||||
|
||||
static int hac_soc_platform_remove(struct platform_device *pdev)
|
||||
{
|
||||
snd_soc_unregister_component(&pdev->dev);
|
||||
snd_soc_set_ac97_ops(NULL);
|
||||
return 0;
|
||||
}
|
||||
|
||||
static struct platform_driver hac_pcm_driver = {
|
||||
.driver = {
|
||||
.name = "hac-pcm-audio",
|
||||
.owner = THIS_MODULE,
|
||||
},
|
||||
|
||||
.probe = hac_soc_platform_probe,
|
||||
.remove = hac_soc_platform_remove,
|
||||
};
|
||||
|
||||
module_platform_driver(hac_pcm_driver);
|
||||
|
||||
MODULE_LICENSE("GPL");
|
||||
MODULE_DESCRIPTION("SuperH onchip HAC (AC97) audio driver");
|
||||
MODULE_AUTHOR("Manuel Lauss <mano@roarinelk.homelinux.net>");
|
||||
216
sound/soc/sh/migor.c
Normal file
216
sound/soc/sh/migor.c
Normal file
|
|
@ -0,0 +1,216 @@
|
|||
/*
|
||||
* ALSA SoC driver for Migo-R
|
||||
*
|
||||
* Copyright (C) 2009-2010 Guennadi Liakhovetski <g.liakhovetski@gmx.de>
|
||||
*
|
||||
* 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/clkdev.h>
|
||||
#include <linux/device.h>
|
||||
#include <linux/firmware.h>
|
||||
#include <linux/module.h>
|
||||
|
||||
#include <asm/clock.h>
|
||||
|
||||
#include <cpu/sh7722.h>
|
||||
|
||||
#include <sound/core.h>
|
||||
#include <sound/pcm.h>
|
||||
#include <sound/soc.h>
|
||||
|
||||
#include "../codecs/wm8978.h"
|
||||
#include "siu.h"
|
||||
|
||||
/* Default 8000Hz sampling frequency */
|
||||
static unsigned long codec_freq = 8000 * 512;
|
||||
|
||||
static unsigned int use_count;
|
||||
|
||||
/* External clock, sourced from the codec at the SIUMCKB pin */
|
||||
static unsigned long siumckb_recalc(struct clk *clk)
|
||||
{
|
||||
return codec_freq;
|
||||
}
|
||||
|
||||
static struct sh_clk_ops siumckb_clk_ops = {
|
||||
.recalc = siumckb_recalc,
|
||||
};
|
||||
|
||||
static struct clk siumckb_clk = {
|
||||
.ops = &siumckb_clk_ops,
|
||||
.rate = 0, /* initialised at run-time */
|
||||
};
|
||||
|
||||
static struct clk_lookup *siumckb_lookup;
|
||||
|
||||
static int migor_hw_params(struct snd_pcm_substream *substream,
|
||||
struct snd_pcm_hw_params *params)
|
||||
{
|
||||
struct snd_soc_pcm_runtime *rtd = substream->private_data;
|
||||
struct snd_soc_dai *codec_dai = rtd->codec_dai;
|
||||
int ret;
|
||||
unsigned int rate = params_rate(params);
|
||||
|
||||
ret = snd_soc_dai_set_sysclk(codec_dai, WM8978_PLL, 13000000,
|
||||
SND_SOC_CLOCK_IN);
|
||||
if (ret < 0)
|
||||
return ret;
|
||||
|
||||
ret = snd_soc_dai_set_clkdiv(codec_dai, WM8978_OPCLKRATE, rate * 512);
|
||||
if (ret < 0)
|
||||
return ret;
|
||||
|
||||
ret = snd_soc_dai_set_fmt(codec_dai, SND_SOC_DAIFMT_NB_IF |
|
||||
SND_SOC_DAIFMT_I2S | SND_SOC_DAIFMT_CBS_CFS);
|
||||
if (ret < 0)
|
||||
return ret;
|
||||
|
||||
ret = snd_soc_dai_set_fmt(rtd->cpu_dai, SND_SOC_DAIFMT_NB_IF |
|
||||
SND_SOC_DAIFMT_I2S | SND_SOC_DAIFMT_CBS_CFS);
|
||||
if (ret < 0)
|
||||
return ret;
|
||||
|
||||
codec_freq = rate * 512;
|
||||
/*
|
||||
* This propagates the parent frequency change to children and
|
||||
* recalculates the frequency table
|
||||
*/
|
||||
clk_set_rate(&siumckb_clk, codec_freq);
|
||||
dev_dbg(codec_dai->dev, "%s: configure %luHz\n", __func__, codec_freq);
|
||||
|
||||
ret = snd_soc_dai_set_sysclk(rtd->cpu_dai, SIU_CLKB_EXT,
|
||||
codec_freq / 2, SND_SOC_CLOCK_IN);
|
||||
|
||||
if (!ret)
|
||||
use_count++;
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int migor_hw_free(struct snd_pcm_substream *substream)
|
||||
{
|
||||
struct snd_soc_pcm_runtime *rtd = substream->private_data;
|
||||
struct snd_soc_dai *codec_dai = rtd->codec_dai;
|
||||
|
||||
if (use_count) {
|
||||
use_count--;
|
||||
|
||||
if (!use_count)
|
||||
snd_soc_dai_set_sysclk(codec_dai, WM8978_PLL, 0,
|
||||
SND_SOC_CLOCK_IN);
|
||||
} else {
|
||||
dev_dbg(codec_dai->dev, "Unbalanced hw_free!\n");
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static struct snd_soc_ops migor_dai_ops = {
|
||||
.hw_params = migor_hw_params,
|
||||
.hw_free = migor_hw_free,
|
||||
};
|
||||
|
||||
static const struct snd_soc_dapm_widget migor_dapm_widgets[] = {
|
||||
SND_SOC_DAPM_HP("Headphone", NULL),
|
||||
SND_SOC_DAPM_MIC("Onboard Microphone", NULL),
|
||||
SND_SOC_DAPM_MIC("External Microphone", NULL),
|
||||
};
|
||||
|
||||
static const struct snd_soc_dapm_route audio_map[] = {
|
||||
/* Headphone output connected to LHP/RHP, enable OUT4 for VMID */
|
||||
{ "Headphone", NULL, "OUT4 VMID" },
|
||||
{ "OUT4 VMID", NULL, "LHP" },
|
||||
{ "OUT4 VMID", NULL, "RHP" },
|
||||
|
||||
/* On-board microphone */
|
||||
{ "RMICN", NULL, "Mic Bias" },
|
||||
{ "RMICP", NULL, "Mic Bias" },
|
||||
{ "Mic Bias", NULL, "Onboard Microphone" },
|
||||
|
||||
/* External microphone */
|
||||
{ "LMICN", NULL, "Mic Bias" },
|
||||
{ "LMICP", NULL, "Mic Bias" },
|
||||
{ "Mic Bias", NULL, "External Microphone" },
|
||||
};
|
||||
|
||||
/* migor digital audio interface glue - connects codec <--> CPU */
|
||||
static struct snd_soc_dai_link migor_dai = {
|
||||
.name = "wm8978",
|
||||
.stream_name = "WM8978",
|
||||
.cpu_dai_name = "siu-pcm-audio",
|
||||
.codec_dai_name = "wm8978-hifi",
|
||||
.platform_name = "siu-pcm-audio",
|
||||
.codec_name = "wm8978.0-001a",
|
||||
.ops = &migor_dai_ops,
|
||||
};
|
||||
|
||||
/* migor audio machine driver */
|
||||
static struct snd_soc_card snd_soc_migor = {
|
||||
.name = "Migo-R",
|
||||
.owner = THIS_MODULE,
|
||||
.dai_link = &migor_dai,
|
||||
.num_links = 1,
|
||||
|
||||
.dapm_widgets = migor_dapm_widgets,
|
||||
.num_dapm_widgets = ARRAY_SIZE(migor_dapm_widgets),
|
||||
.dapm_routes = audio_map,
|
||||
.num_dapm_routes = ARRAY_SIZE(audio_map),
|
||||
};
|
||||
|
||||
static struct platform_device *migor_snd_device;
|
||||
|
||||
static int __init migor_init(void)
|
||||
{
|
||||
int ret;
|
||||
|
||||
ret = clk_register(&siumckb_clk);
|
||||
if (ret < 0)
|
||||
return ret;
|
||||
|
||||
siumckb_lookup = clkdev_alloc(&siumckb_clk, "siumckb_clk", NULL);
|
||||
if (!siumckb_lookup) {
|
||||
ret = -ENOMEM;
|
||||
goto eclkdevalloc;
|
||||
}
|
||||
clkdev_add(siumckb_lookup);
|
||||
|
||||
/* Port number used on this machine: port B */
|
||||
migor_snd_device = platform_device_alloc("soc-audio", 1);
|
||||
if (!migor_snd_device) {
|
||||
ret = -ENOMEM;
|
||||
goto epdevalloc;
|
||||
}
|
||||
|
||||
platform_set_drvdata(migor_snd_device, &snd_soc_migor);
|
||||
|
||||
ret = platform_device_add(migor_snd_device);
|
||||
if (ret)
|
||||
goto epdevadd;
|
||||
|
||||
return 0;
|
||||
|
||||
epdevadd:
|
||||
platform_device_put(migor_snd_device);
|
||||
epdevalloc:
|
||||
clkdev_drop(siumckb_lookup);
|
||||
eclkdevalloc:
|
||||
clk_unregister(&siumckb_clk);
|
||||
return ret;
|
||||
}
|
||||
|
||||
static void __exit migor_exit(void)
|
||||
{
|
||||
clkdev_drop(siumckb_lookup);
|
||||
clk_unregister(&siumckb_clk);
|
||||
platform_device_unregister(migor_snd_device);
|
||||
}
|
||||
|
||||
module_init(migor_init);
|
||||
module_exit(migor_exit);
|
||||
|
||||
MODULE_AUTHOR("Guennadi Liakhovetski <g.liakhovetski@gmx.de>");
|
||||
MODULE_DESCRIPTION("ALSA SoC Migor");
|
||||
MODULE_LICENSE("GPL v2");
|
||||
2
sound/soc/sh/rcar/Makefile
Normal file
2
sound/soc/sh/rcar/Makefile
Normal file
|
|
@ -0,0 +1,2 @@
|
|||
snd-soc-rcar-objs := core.o gen.o src.o adg.o ssi.o dvc.o
|
||||
obj-$(CONFIG_SND_SOC_RCAR) += snd-soc-rcar.o
|
||||
442
sound/soc/sh/rcar/adg.c
Normal file
442
sound/soc/sh/rcar/adg.c
Normal file
|
|
@ -0,0 +1,442 @@
|
|||
/*
|
||||
* Helper routines for R-Car sound ADG.
|
||||
*
|
||||
* Copyright (C) 2013 Kuninori Morimoto <kuninori.morimoto.gx@renesas.com>
|
||||
*
|
||||
* This file is subject to the terms and conditions of the GNU General Public
|
||||
* License. See the file "COPYING" in the main directory of this archive
|
||||
* for more details.
|
||||
*/
|
||||
#include <linux/sh_clk.h>
|
||||
#include "rsnd.h"
|
||||
|
||||
#define CLKA 0
|
||||
#define CLKB 1
|
||||
#define CLKC 2
|
||||
#define CLKI 3
|
||||
#define CLKMAX 4
|
||||
|
||||
struct rsnd_adg {
|
||||
struct clk *clk[CLKMAX];
|
||||
|
||||
int rbga_rate_for_441khz_div_6; /* RBGA */
|
||||
int rbgb_rate_for_48khz_div_6; /* RBGB */
|
||||
u32 ckr;
|
||||
};
|
||||
|
||||
#define for_each_rsnd_clk(pos, adg, i) \
|
||||
for (i = 0; \
|
||||
(i < CLKMAX) && \
|
||||
((pos) = adg->clk[i]); \
|
||||
i++)
|
||||
#define rsnd_priv_to_adg(priv) ((struct rsnd_adg *)(priv)->adg)
|
||||
|
||||
|
||||
static u32 rsnd_adg_ssi_ws_timing_gen2(struct rsnd_dai_stream *io)
|
||||
{
|
||||
struct rsnd_mod *mod = rsnd_io_to_mod_ssi(io);
|
||||
struct rsnd_priv *priv = rsnd_mod_to_priv(mod);
|
||||
int id = rsnd_mod_id(mod);
|
||||
int ws = id;
|
||||
|
||||
if (rsnd_ssi_is_pin_sharing(rsnd_ssi_mod_get(priv, id))) {
|
||||
switch (id) {
|
||||
case 1:
|
||||
case 2:
|
||||
ws = 0;
|
||||
break;
|
||||
case 4:
|
||||
ws = 3;
|
||||
break;
|
||||
case 8:
|
||||
ws = 7;
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
return (0x6 + ws) << 8;
|
||||
}
|
||||
|
||||
int rsnd_adg_set_cmd_timsel_gen2(struct rsnd_dai *rdai,
|
||||
struct rsnd_mod *mod,
|
||||
struct rsnd_dai_stream *io)
|
||||
{
|
||||
int id = rsnd_mod_id(mod);
|
||||
int shift = (id % 2) ? 16 : 0;
|
||||
u32 mask, val;
|
||||
|
||||
val = rsnd_adg_ssi_ws_timing_gen2(io);
|
||||
|
||||
val = val << shift;
|
||||
mask = 0xffff << shift;
|
||||
|
||||
rsnd_mod_bset(mod, CMDOUT_TIMSEL, mask, val);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int rsnd_adg_set_src_timsel_gen2(struct rsnd_dai *rdai,
|
||||
struct rsnd_mod *mod,
|
||||
struct rsnd_dai_stream *io,
|
||||
u32 timsel)
|
||||
{
|
||||
int is_play = rsnd_dai_is_play(rdai, io);
|
||||
int id = rsnd_mod_id(mod);
|
||||
int shift = (id % 2) ? 16 : 0;
|
||||
u32 mask, ws;
|
||||
u32 in, out;
|
||||
|
||||
ws = rsnd_adg_ssi_ws_timing_gen2(io);
|
||||
|
||||
in = (is_play) ? timsel : ws;
|
||||
out = (is_play) ? ws : timsel;
|
||||
|
||||
in = in << shift;
|
||||
out = out << shift;
|
||||
mask = 0xffff << shift;
|
||||
|
||||
switch (id / 2) {
|
||||
case 0:
|
||||
rsnd_mod_bset(mod, SRCIN_TIMSEL0, mask, in);
|
||||
rsnd_mod_bset(mod, SRCOUT_TIMSEL0, mask, out);
|
||||
break;
|
||||
case 1:
|
||||
rsnd_mod_bset(mod, SRCIN_TIMSEL1, mask, in);
|
||||
rsnd_mod_bset(mod, SRCOUT_TIMSEL1, mask, out);
|
||||
break;
|
||||
case 2:
|
||||
rsnd_mod_bset(mod, SRCIN_TIMSEL2, mask, in);
|
||||
rsnd_mod_bset(mod, SRCOUT_TIMSEL2, mask, out);
|
||||
break;
|
||||
case 3:
|
||||
rsnd_mod_bset(mod, SRCIN_TIMSEL3, mask, in);
|
||||
rsnd_mod_bset(mod, SRCOUT_TIMSEL3, mask, out);
|
||||
break;
|
||||
case 4:
|
||||
rsnd_mod_bset(mod, SRCIN_TIMSEL4, mask, in);
|
||||
rsnd_mod_bset(mod, SRCOUT_TIMSEL4, mask, out);
|
||||
break;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
int rsnd_adg_set_convert_clk_gen2(struct rsnd_mod *mod,
|
||||
struct rsnd_dai *rdai,
|
||||
struct rsnd_dai_stream *io,
|
||||
unsigned int src_rate,
|
||||
unsigned int dst_rate)
|
||||
{
|
||||
struct rsnd_priv *priv = rsnd_mod_to_priv(mod);
|
||||
struct rsnd_adg *adg = rsnd_priv_to_adg(priv);
|
||||
struct device *dev = rsnd_priv_to_dev(priv);
|
||||
int idx, sel, div, step, ret;
|
||||
u32 val, en;
|
||||
unsigned int min, diff;
|
||||
unsigned int sel_rate [] = {
|
||||
clk_get_rate(adg->clk[CLKA]), /* 0000: CLKA */
|
||||
clk_get_rate(adg->clk[CLKB]), /* 0001: CLKB */
|
||||
clk_get_rate(adg->clk[CLKC]), /* 0010: CLKC */
|
||||
adg->rbga_rate_for_441khz_div_6,/* 0011: RBGA */
|
||||
adg->rbgb_rate_for_48khz_div_6, /* 0100: RBGB */
|
||||
};
|
||||
|
||||
min = ~0;
|
||||
val = 0;
|
||||
en = 0;
|
||||
for (sel = 0; sel < ARRAY_SIZE(sel_rate); sel++) {
|
||||
idx = 0;
|
||||
step = 2;
|
||||
|
||||
if (!sel_rate[sel])
|
||||
continue;
|
||||
|
||||
for (div = 2; div <= 98304; div += step) {
|
||||
diff = abs(src_rate - sel_rate[sel] / div);
|
||||
if (min > diff) {
|
||||
val = (sel << 8) | idx;
|
||||
min = diff;
|
||||
en = 1 << (sel + 1); /* fixme */
|
||||
}
|
||||
|
||||
/*
|
||||
* step of 0_0000 / 0_0001 / 0_1101
|
||||
* are out of order
|
||||
*/
|
||||
if ((idx > 2) && (idx % 2))
|
||||
step *= 2;
|
||||
if (idx == 0x1c) {
|
||||
div += step;
|
||||
step *= 2;
|
||||
}
|
||||
idx++;
|
||||
}
|
||||
}
|
||||
|
||||
if (min == ~0) {
|
||||
dev_err(dev, "no Input clock\n");
|
||||
return -EIO;
|
||||
}
|
||||
|
||||
ret = rsnd_adg_set_src_timsel_gen2(rdai, mod, io, val);
|
||||
if (ret < 0) {
|
||||
dev_err(dev, "timsel error\n");
|
||||
return ret;
|
||||
}
|
||||
|
||||
rsnd_mod_bset(mod, DIV_EN, en, en);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
int rsnd_adg_set_convert_timing_gen2(struct rsnd_mod *mod,
|
||||
struct rsnd_dai *rdai,
|
||||
struct rsnd_dai_stream *io)
|
||||
{
|
||||
u32 val = rsnd_adg_ssi_ws_timing_gen2(io);
|
||||
|
||||
return rsnd_adg_set_src_timsel_gen2(rdai, mod, io, val);
|
||||
}
|
||||
|
||||
int rsnd_adg_set_convert_clk_gen1(struct rsnd_priv *priv,
|
||||
struct rsnd_mod *mod,
|
||||
unsigned int src_rate,
|
||||
unsigned int dst_rate)
|
||||
{
|
||||
struct rsnd_adg *adg = rsnd_priv_to_adg(priv);
|
||||
struct device *dev = rsnd_priv_to_dev(priv);
|
||||
int idx, sel, div, shift;
|
||||
u32 mask, val;
|
||||
int id = rsnd_mod_id(mod);
|
||||
unsigned int sel_rate [] = {
|
||||
clk_get_rate(adg->clk[CLKA]), /* 000: CLKA */
|
||||
clk_get_rate(adg->clk[CLKB]), /* 001: CLKB */
|
||||
clk_get_rate(adg->clk[CLKC]), /* 010: CLKC */
|
||||
0, /* 011: MLBCLK (not used) */
|
||||
adg->rbga_rate_for_441khz_div_6,/* 100: RBGA */
|
||||
adg->rbgb_rate_for_48khz_div_6, /* 101: RBGB */
|
||||
};
|
||||
|
||||
/* find div (= 1/128, 1/256, 1/512, 1/1024, 1/2048 */
|
||||
for (sel = 0; sel < ARRAY_SIZE(sel_rate); sel++) {
|
||||
for (div = 128, idx = 0;
|
||||
div <= 2048;
|
||||
div *= 2, idx++) {
|
||||
if (src_rate == sel_rate[sel] / div) {
|
||||
val = (idx << 4) | sel;
|
||||
goto find_rate;
|
||||
}
|
||||
}
|
||||
}
|
||||
dev_err(dev, "can't find convert src clk\n");
|
||||
return -EINVAL;
|
||||
|
||||
find_rate:
|
||||
shift = (id % 4) * 8;
|
||||
mask = 0xFF << shift;
|
||||
val = val << shift;
|
||||
|
||||
dev_dbg(dev, "adg convert src clk = %02x\n", val);
|
||||
|
||||
switch (id / 4) {
|
||||
case 0:
|
||||
rsnd_mod_bset(mod, AUDIO_CLK_SEL3, mask, val);
|
||||
break;
|
||||
case 1:
|
||||
rsnd_mod_bset(mod, AUDIO_CLK_SEL4, mask, val);
|
||||
break;
|
||||
case 2:
|
||||
rsnd_mod_bset(mod, AUDIO_CLK_SEL5, mask, val);
|
||||
break;
|
||||
}
|
||||
|
||||
/*
|
||||
* Gen1 doesn't need dst_rate settings,
|
||||
* since it uses SSI WS pin.
|
||||
* see also rsnd_src_set_route_if_gen1()
|
||||
*/
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static void rsnd_adg_set_ssi_clk(struct rsnd_mod *mod, u32 val)
|
||||
{
|
||||
int id = rsnd_mod_id(mod);
|
||||
int shift = (id % 4) * 8;
|
||||
u32 mask = 0xFF << shift;
|
||||
|
||||
val = val << shift;
|
||||
|
||||
/*
|
||||
* SSI 8 is not connected to ADG.
|
||||
* it works with SSI 7
|
||||
*/
|
||||
if (id == 8)
|
||||
return;
|
||||
|
||||
switch (id / 4) {
|
||||
case 0:
|
||||
rsnd_mod_bset(mod, AUDIO_CLK_SEL0, mask, val);
|
||||
break;
|
||||
case 1:
|
||||
rsnd_mod_bset(mod, AUDIO_CLK_SEL1, mask, val);
|
||||
break;
|
||||
case 2:
|
||||
rsnd_mod_bset(mod, AUDIO_CLK_SEL2, mask, val);
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
int rsnd_adg_ssi_clk_stop(struct rsnd_mod *mod)
|
||||
{
|
||||
/*
|
||||
* "mod" = "ssi" here.
|
||||
* we can get "ssi id" from mod
|
||||
*/
|
||||
rsnd_adg_set_ssi_clk(mod, 0);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
int rsnd_adg_ssi_clk_try_start(struct rsnd_mod *mod, unsigned int rate)
|
||||
{
|
||||
struct rsnd_priv *priv = rsnd_mod_to_priv(mod);
|
||||
struct rsnd_adg *adg = rsnd_priv_to_adg(priv);
|
||||
struct device *dev = rsnd_priv_to_dev(priv);
|
||||
struct clk *clk;
|
||||
int i;
|
||||
u32 data;
|
||||
int sel_table[] = {
|
||||
[CLKA] = 0x1,
|
||||
[CLKB] = 0x2,
|
||||
[CLKC] = 0x3,
|
||||
[CLKI] = 0x0,
|
||||
};
|
||||
|
||||
dev_dbg(dev, "request clock = %d\n", rate);
|
||||
|
||||
/*
|
||||
* find suitable clock from
|
||||
* AUDIO_CLKA/AUDIO_CLKB/AUDIO_CLKC/AUDIO_CLKI.
|
||||
*/
|
||||
data = 0;
|
||||
for_each_rsnd_clk(clk, adg, i) {
|
||||
if (rate == clk_get_rate(clk)) {
|
||||
data = sel_table[i];
|
||||
goto found_clock;
|
||||
}
|
||||
}
|
||||
|
||||
/*
|
||||
* find 1/6 clock from BRGA/BRGB
|
||||
*/
|
||||
if (rate == adg->rbga_rate_for_441khz_div_6) {
|
||||
data = 0x10;
|
||||
goto found_clock;
|
||||
}
|
||||
|
||||
if (rate == adg->rbgb_rate_for_48khz_div_6) {
|
||||
data = 0x20;
|
||||
goto found_clock;
|
||||
}
|
||||
|
||||
return -EIO;
|
||||
|
||||
found_clock:
|
||||
|
||||
/* see rsnd_adg_ssi_clk_init() */
|
||||
rsnd_mod_bset(mod, SSICKR, 0x00FF0000, adg->ckr);
|
||||
rsnd_mod_write(mod, BRRA, 0x00000002); /* 1/6 */
|
||||
rsnd_mod_write(mod, BRRB, 0x00000002); /* 1/6 */
|
||||
|
||||
/*
|
||||
* This "mod" = "ssi" here.
|
||||
* we can get "ssi id" from mod
|
||||
*/
|
||||
rsnd_adg_set_ssi_clk(mod, data);
|
||||
|
||||
dev_dbg(dev, "ADG: ssi%d selects clk%d = %d",
|
||||
rsnd_mod_id(mod), i, rate);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static void rsnd_adg_ssi_clk_init(struct rsnd_priv *priv, struct rsnd_adg *adg)
|
||||
{
|
||||
struct clk *clk;
|
||||
unsigned long rate;
|
||||
u32 ckr;
|
||||
int i;
|
||||
int brg_table[] = {
|
||||
[CLKA] = 0x0,
|
||||
[CLKB] = 0x1,
|
||||
[CLKC] = 0x4,
|
||||
[CLKI] = 0x2,
|
||||
};
|
||||
|
||||
/*
|
||||
* This driver is assuming that AUDIO_CLKA/AUDIO_CLKB/AUDIO_CLKC
|
||||
* have 44.1kHz or 48kHz base clocks for now.
|
||||
*
|
||||
* SSI itself can divide parent clock by 1/1 - 1/16
|
||||
* So, BRGA outputs 44.1kHz base parent clock 1/32,
|
||||
* and, BRGB outputs 48.0kHz base parent clock 1/32 here.
|
||||
* see
|
||||
* rsnd_adg_ssi_clk_try_start()
|
||||
*/
|
||||
ckr = 0;
|
||||
adg->rbga_rate_for_441khz_div_6 = 0;
|
||||
adg->rbgb_rate_for_48khz_div_6 = 0;
|
||||
for_each_rsnd_clk(clk, adg, i) {
|
||||
rate = clk_get_rate(clk);
|
||||
|
||||
if (0 == rate) /* not used */
|
||||
continue;
|
||||
|
||||
/* RBGA */
|
||||
if (!adg->rbga_rate_for_441khz_div_6 && (0 == rate % 44100)) {
|
||||
adg->rbga_rate_for_441khz_div_6 = rate / 6;
|
||||
ckr |= brg_table[i] << 20;
|
||||
}
|
||||
|
||||
/* RBGB */
|
||||
if (!adg->rbgb_rate_for_48khz_div_6 && (0 == rate % 48000)) {
|
||||
adg->rbgb_rate_for_48khz_div_6 = rate / 6;
|
||||
ckr |= brg_table[i] << 16;
|
||||
}
|
||||
}
|
||||
|
||||
adg->ckr = ckr;
|
||||
}
|
||||
|
||||
int rsnd_adg_probe(struct platform_device *pdev,
|
||||
const struct rsnd_of_data *of_data,
|
||||
struct rsnd_priv *priv)
|
||||
{
|
||||
struct rsnd_adg *adg;
|
||||
struct device *dev = rsnd_priv_to_dev(priv);
|
||||
struct clk *clk;
|
||||
int i;
|
||||
|
||||
adg = devm_kzalloc(dev, sizeof(*adg), GFP_KERNEL);
|
||||
if (!adg) {
|
||||
dev_err(dev, "ADG allocate failed\n");
|
||||
return -ENOMEM;
|
||||
}
|
||||
|
||||
adg->clk[CLKA] = devm_clk_get(dev, "clk_a");
|
||||
adg->clk[CLKB] = devm_clk_get(dev, "clk_b");
|
||||
adg->clk[CLKC] = devm_clk_get(dev, "clk_c");
|
||||
adg->clk[CLKI] = devm_clk_get(dev, "clk_i");
|
||||
|
||||
for_each_rsnd_clk(clk, adg, i)
|
||||
dev_dbg(dev, "clk %d : %p\n", i, clk);
|
||||
|
||||
rsnd_adg_ssi_clk_init(priv, adg);
|
||||
|
||||
priv->adg = adg;
|
||||
|
||||
dev_dbg(dev, "adg probed\n");
|
||||
|
||||
return 0;
|
||||
}
|
||||
1112
sound/soc/sh/rcar/core.c
Normal file
1112
sound/soc/sh/rcar/core.c
Normal file
File diff suppressed because it is too large
Load diff
356
sound/soc/sh/rcar/dvc.c
Normal file
356
sound/soc/sh/rcar/dvc.c
Normal file
|
|
@ -0,0 +1,356 @@
|
|||
/*
|
||||
* Renesas R-Car DVC support
|
||||
*
|
||||
* Copyright (C) 2014 Renesas Solutions Corp.
|
||||
* Kuninori Morimoto <kuninori.morimoto.gx@renesas.com>
|
||||
*
|
||||
* 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 "rsnd.h"
|
||||
|
||||
#define RSND_DVC_NAME_SIZE 16
|
||||
#define RSND_DVC_VOLUME_MAX 100
|
||||
#define RSND_DVC_VOLUME_NUM 2
|
||||
|
||||
#define DVC_NAME "dvc"
|
||||
|
||||
struct rsnd_dvc {
|
||||
struct rsnd_dvc_platform_info *info; /* rcar_snd.h */
|
||||
struct rsnd_mod mod;
|
||||
struct clk *clk;
|
||||
u8 volume[RSND_DVC_VOLUME_NUM];
|
||||
u8 mute[RSND_DVC_VOLUME_NUM];
|
||||
};
|
||||
|
||||
#define rsnd_mod_to_dvc(_mod) \
|
||||
container_of((_mod), struct rsnd_dvc, mod)
|
||||
|
||||
#define for_each_rsnd_dvc(pos, priv, i) \
|
||||
for ((i) = 0; \
|
||||
((i) < rsnd_dvc_nr(priv)) && \
|
||||
((pos) = (struct rsnd_dvc *)(priv)->dvc + i); \
|
||||
i++)
|
||||
|
||||
static void rsnd_dvc_volume_update(struct rsnd_mod *mod)
|
||||
{
|
||||
struct rsnd_dvc *dvc = rsnd_mod_to_dvc(mod);
|
||||
u32 max = (0x00800000 - 1);
|
||||
u32 vol[RSND_DVC_VOLUME_NUM];
|
||||
u32 mute = 0;
|
||||
int i;
|
||||
|
||||
for (i = 0; i < RSND_DVC_VOLUME_NUM; i++) {
|
||||
vol[i] = max / RSND_DVC_VOLUME_MAX * dvc->volume[i];
|
||||
mute |= (!!dvc->mute[i]) << i;
|
||||
}
|
||||
|
||||
rsnd_mod_write(mod, DVC_VOL0R, vol[0]);
|
||||
rsnd_mod_write(mod, DVC_VOL1R, vol[1]);
|
||||
|
||||
rsnd_mod_write(mod, DVC_ZCMCR, mute);
|
||||
}
|
||||
|
||||
static int rsnd_dvc_probe_gen2(struct rsnd_mod *mod,
|
||||
struct rsnd_dai *rdai)
|
||||
{
|
||||
struct rsnd_priv *priv = rsnd_mod_to_priv(mod);
|
||||
struct device *dev = rsnd_priv_to_dev(priv);
|
||||
|
||||
dev_dbg(dev, "%s (Gen2) is probed\n", rsnd_mod_name(mod));
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int rsnd_dvc_init(struct rsnd_mod *dvc_mod,
|
||||
struct rsnd_dai *rdai)
|
||||
{
|
||||
struct rsnd_dvc *dvc = rsnd_mod_to_dvc(dvc_mod);
|
||||
struct rsnd_dai_stream *io = rsnd_mod_to_io(dvc_mod);
|
||||
struct rsnd_priv *priv = rsnd_mod_to_priv(dvc_mod);
|
||||
struct rsnd_mod *src_mod = rsnd_io_to_mod_src(io);
|
||||
struct device *dev = rsnd_priv_to_dev(priv);
|
||||
int dvc_id = rsnd_mod_id(dvc_mod);
|
||||
int src_id = rsnd_mod_id(src_mod);
|
||||
u32 route[] = {
|
||||
[0] = 0x30000,
|
||||
[1] = 0x30001,
|
||||
[2] = 0x40000,
|
||||
[3] = 0x10000,
|
||||
[4] = 0x20000,
|
||||
[5] = 0x40100
|
||||
};
|
||||
|
||||
if (src_id >= ARRAY_SIZE(route)) {
|
||||
dev_err(dev, "DVC%d isn't connected to SRC%d\n", dvc_id, src_id);
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
clk_prepare_enable(dvc->clk);
|
||||
|
||||
/*
|
||||
* fixme
|
||||
* it doesn't support CTU/MIX
|
||||
*/
|
||||
rsnd_mod_write(dvc_mod, CMD_ROUTE_SLCT, route[src_id]);
|
||||
|
||||
rsnd_mod_write(dvc_mod, DVC_SWRSR, 0);
|
||||
rsnd_mod_write(dvc_mod, DVC_SWRSR, 1);
|
||||
|
||||
rsnd_mod_write(dvc_mod, DVC_DVUIR, 1);
|
||||
|
||||
rsnd_mod_write(dvc_mod, DVC_ADINR, rsnd_get_adinr(dvc_mod));
|
||||
|
||||
/* enable Volume / Mute */
|
||||
rsnd_mod_write(dvc_mod, DVC_DVUCR, 0x101);
|
||||
|
||||
/* ch0/ch1 Volume */
|
||||
rsnd_dvc_volume_update(dvc_mod);
|
||||
|
||||
rsnd_mod_write(dvc_mod, DVC_DVUIR, 0);
|
||||
|
||||
rsnd_mod_write(dvc_mod, DVC_DVUER, 1);
|
||||
|
||||
rsnd_adg_set_cmd_timsel_gen2(rdai, dvc_mod, io);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int rsnd_dvc_quit(struct rsnd_mod *mod,
|
||||
struct rsnd_dai *rdai)
|
||||
{
|
||||
struct rsnd_dvc *dvc = rsnd_mod_to_dvc(mod);
|
||||
|
||||
clk_disable_unprepare(dvc->clk);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int rsnd_dvc_start(struct rsnd_mod *mod,
|
||||
struct rsnd_dai *rdai)
|
||||
{
|
||||
rsnd_mod_write(mod, CMD_CTRL, 0x10);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int rsnd_dvc_stop(struct rsnd_mod *mod,
|
||||
struct rsnd_dai *rdai)
|
||||
{
|
||||
rsnd_mod_write(mod, CMD_CTRL, 0);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int rsnd_dvc_volume_info(struct snd_kcontrol *kctrl,
|
||||
struct snd_ctl_elem_info *uinfo)
|
||||
{
|
||||
struct rsnd_mod *mod = snd_kcontrol_chip(kctrl);
|
||||
struct rsnd_dvc *dvc = rsnd_mod_to_dvc(mod);
|
||||
u8 *val = (u8 *)kctrl->private_value;
|
||||
|
||||
uinfo->count = RSND_DVC_VOLUME_NUM;
|
||||
uinfo->value.integer.min = 0;
|
||||
|
||||
if (val == dvc->volume) {
|
||||
uinfo->type = SNDRV_CTL_ELEM_TYPE_INTEGER;
|
||||
uinfo->value.integer.max = RSND_DVC_VOLUME_MAX;
|
||||
} else {
|
||||
uinfo->type = SNDRV_CTL_ELEM_TYPE_BOOLEAN;
|
||||
uinfo->value.integer.max = 1;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int rsnd_dvc_volume_get(struct snd_kcontrol *kctrl,
|
||||
struct snd_ctl_elem_value *ucontrol)
|
||||
{
|
||||
u8 *val = (u8 *)kctrl->private_value;
|
||||
int i;
|
||||
|
||||
for (i = 0; i < RSND_DVC_VOLUME_NUM; i++)
|
||||
ucontrol->value.integer.value[i] = val[i];
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int rsnd_dvc_volume_put(struct snd_kcontrol *kctrl,
|
||||
struct snd_ctl_elem_value *ucontrol)
|
||||
{
|
||||
struct rsnd_mod *mod = snd_kcontrol_chip(kctrl);
|
||||
u8 *val = (u8 *)kctrl->private_value;
|
||||
int i, change = 0;
|
||||
|
||||
for (i = 0; i < RSND_DVC_VOLUME_NUM; i++) {
|
||||
change |= (ucontrol->value.integer.value[i] != val[i]);
|
||||
val[i] = ucontrol->value.integer.value[i];
|
||||
}
|
||||
|
||||
if (change)
|
||||
rsnd_dvc_volume_update(mod);
|
||||
|
||||
return change;
|
||||
}
|
||||
|
||||
static int __rsnd_dvc_pcm_new(struct rsnd_mod *mod,
|
||||
struct rsnd_dai *rdai,
|
||||
struct snd_soc_pcm_runtime *rtd,
|
||||
const unsigned char *name,
|
||||
u8 *private)
|
||||
{
|
||||
struct snd_card *card = rtd->card->snd_card;
|
||||
struct snd_kcontrol *kctrl;
|
||||
struct snd_kcontrol_new knew = {
|
||||
.iface = SNDRV_CTL_ELEM_IFACE_MIXER,
|
||||
.name = name,
|
||||
.info = rsnd_dvc_volume_info,
|
||||
.get = rsnd_dvc_volume_get,
|
||||
.put = rsnd_dvc_volume_put,
|
||||
.private_value = (unsigned long)private,
|
||||
};
|
||||
int ret;
|
||||
|
||||
kctrl = snd_ctl_new1(&knew, mod);
|
||||
if (!kctrl)
|
||||
return -ENOMEM;
|
||||
|
||||
ret = snd_ctl_add(card, kctrl);
|
||||
if (ret < 0)
|
||||
return ret;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int rsnd_dvc_pcm_new(struct rsnd_mod *mod,
|
||||
struct rsnd_dai *rdai,
|
||||
struct snd_soc_pcm_runtime *rtd)
|
||||
{
|
||||
struct rsnd_dai_stream *io = rsnd_mod_to_io(mod);
|
||||
struct rsnd_dvc *dvc = rsnd_mod_to_dvc(mod);
|
||||
int ret;
|
||||
|
||||
/* Volume */
|
||||
ret = __rsnd_dvc_pcm_new(mod, rdai, rtd,
|
||||
rsnd_dai_is_play(rdai, io) ?
|
||||
"DVC Out Playback Volume" : "DVC In Capture Volume",
|
||||
dvc->volume);
|
||||
if (ret < 0)
|
||||
return ret;
|
||||
|
||||
/* Mute */
|
||||
ret = __rsnd_dvc_pcm_new(mod, rdai, rtd,
|
||||
rsnd_dai_is_play(rdai, io) ?
|
||||
"DVC Out Mute Switch" : "DVC In Mute Switch",
|
||||
dvc->mute);
|
||||
if (ret < 0)
|
||||
return ret;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static struct rsnd_mod_ops rsnd_dvc_ops = {
|
||||
.name = DVC_NAME,
|
||||
.probe = rsnd_dvc_probe_gen2,
|
||||
.init = rsnd_dvc_init,
|
||||
.quit = rsnd_dvc_quit,
|
||||
.start = rsnd_dvc_start,
|
||||
.stop = rsnd_dvc_stop,
|
||||
.pcm_new = rsnd_dvc_pcm_new,
|
||||
};
|
||||
|
||||
struct rsnd_mod *rsnd_dvc_mod_get(struct rsnd_priv *priv, int id)
|
||||
{
|
||||
if (WARN_ON(id < 0 || id >= rsnd_dvc_nr(priv)))
|
||||
id = 0;
|
||||
|
||||
return &((struct rsnd_dvc *)(priv->dvc) + id)->mod;
|
||||
}
|
||||
|
||||
static void rsnd_of_parse_dvc(struct platform_device *pdev,
|
||||
const struct rsnd_of_data *of_data,
|
||||
struct rsnd_priv *priv)
|
||||
{
|
||||
struct device_node *node;
|
||||
struct rsnd_dvc_platform_info *dvc_info;
|
||||
struct rcar_snd_info *info = rsnd_priv_to_info(priv);
|
||||
struct device *dev = &pdev->dev;
|
||||
int nr;
|
||||
|
||||
if (!of_data)
|
||||
return;
|
||||
|
||||
node = of_get_child_by_name(dev->of_node, "rcar_sound,dvc");
|
||||
if (!node)
|
||||
return;
|
||||
|
||||
nr = of_get_child_count(node);
|
||||
if (!nr)
|
||||
goto rsnd_of_parse_dvc_end;
|
||||
|
||||
dvc_info = devm_kzalloc(dev,
|
||||
sizeof(struct rsnd_dvc_platform_info) * nr,
|
||||
GFP_KERNEL);
|
||||
if (!dvc_info) {
|
||||
dev_err(dev, "dvc info allocation error\n");
|
||||
goto rsnd_of_parse_dvc_end;
|
||||
}
|
||||
|
||||
info->dvc_info = dvc_info;
|
||||
info->dvc_info_nr = nr;
|
||||
|
||||
rsnd_of_parse_dvc_end:
|
||||
of_node_put(node);
|
||||
}
|
||||
|
||||
int rsnd_dvc_probe(struct platform_device *pdev,
|
||||
const struct rsnd_of_data *of_data,
|
||||
struct rsnd_priv *priv)
|
||||
{
|
||||
struct rcar_snd_info *info = rsnd_priv_to_info(priv);
|
||||
struct device *dev = rsnd_priv_to_dev(priv);
|
||||
struct rsnd_dvc *dvc;
|
||||
struct clk *clk;
|
||||
char name[RSND_DVC_NAME_SIZE];
|
||||
int i, nr;
|
||||
|
||||
rsnd_of_parse_dvc(pdev, of_data, priv);
|
||||
|
||||
nr = info->dvc_info_nr;
|
||||
if (!nr)
|
||||
return 0;
|
||||
|
||||
/* This driver doesn't support Gen1 at this point */
|
||||
if (rsnd_is_gen1(priv)) {
|
||||
dev_warn(dev, "CMD is not supported on Gen1\n");
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
dvc = devm_kzalloc(dev, sizeof(*dvc) * nr, GFP_KERNEL);
|
||||
if (!dvc) {
|
||||
dev_err(dev, "CMD allocate failed\n");
|
||||
return -ENOMEM;
|
||||
}
|
||||
|
||||
priv->dvc_nr = nr;
|
||||
priv->dvc = dvc;
|
||||
|
||||
for_each_rsnd_dvc(dvc, priv, i) {
|
||||
snprintf(name, RSND_DVC_NAME_SIZE, "%s.%d",
|
||||
DVC_NAME, i);
|
||||
|
||||
clk = devm_clk_get(dev, name);
|
||||
if (IS_ERR(clk))
|
||||
return PTR_ERR(clk);
|
||||
|
||||
dvc->info = &info->dvc_info[i];
|
||||
dvc->clk = clk;
|
||||
|
||||
rsnd_mod_init(priv, &dvc->mod, &rsnd_dvc_ops, RSND_MOD_DVC, i);
|
||||
|
||||
dev_dbg(dev, "CMD%d probed\n", i);
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
468
sound/soc/sh/rcar/gen.c
Normal file
468
sound/soc/sh/rcar/gen.c
Normal file
|
|
@ -0,0 +1,468 @@
|
|||
/*
|
||||
* Renesas R-Car Gen1 SRU/SSI support
|
||||
*
|
||||
* Copyright (C) 2013 Renesas Solutions Corp.
|
||||
* Kuninori Morimoto <kuninori.morimoto.gx@renesas.com>
|
||||
*
|
||||
* 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 "rsnd.h"
|
||||
|
||||
struct rsnd_gen {
|
||||
void __iomem *base[RSND_BASE_MAX];
|
||||
|
||||
struct rsnd_gen_ops *ops;
|
||||
|
||||
struct regmap *regmap[RSND_BASE_MAX];
|
||||
struct regmap_field *regs[RSND_REG_MAX];
|
||||
};
|
||||
|
||||
#define rsnd_priv_to_gen(p) ((struct rsnd_gen *)(p)->gen)
|
||||
|
||||
struct rsnd_regmap_field_conf {
|
||||
int idx;
|
||||
unsigned int reg_offset;
|
||||
unsigned int id_offset;
|
||||
};
|
||||
|
||||
#define RSND_REG_SET(id, offset, _id_offset) \
|
||||
{ \
|
||||
.idx = id, \
|
||||
.reg_offset = offset, \
|
||||
.id_offset = _id_offset, \
|
||||
}
|
||||
/* single address mapping */
|
||||
#define RSND_GEN_S_REG(id, offset) \
|
||||
RSND_REG_SET(RSND_REG_##id, offset, 0)
|
||||
|
||||
/* multi address mapping */
|
||||
#define RSND_GEN_M_REG(id, offset, _id_offset) \
|
||||
RSND_REG_SET(RSND_REG_##id, offset, _id_offset)
|
||||
|
||||
/*
|
||||
* basic function
|
||||
*/
|
||||
static int rsnd_is_accessible_reg(struct rsnd_priv *priv,
|
||||
struct rsnd_gen *gen, enum rsnd_reg reg)
|
||||
{
|
||||
if (!gen->regs[reg]) {
|
||||
struct device *dev = rsnd_priv_to_dev(priv);
|
||||
|
||||
dev_err(dev, "unsupported register access %x\n", reg);
|
||||
return 0;
|
||||
}
|
||||
|
||||
return 1;
|
||||
}
|
||||
|
||||
u32 rsnd_read(struct rsnd_priv *priv,
|
||||
struct rsnd_mod *mod, enum rsnd_reg reg)
|
||||
{
|
||||
struct device *dev = rsnd_priv_to_dev(priv);
|
||||
struct rsnd_gen *gen = rsnd_priv_to_gen(priv);
|
||||
u32 val;
|
||||
|
||||
if (!rsnd_is_accessible_reg(priv, gen, reg))
|
||||
return 0;
|
||||
|
||||
regmap_fields_read(gen->regs[reg], rsnd_mod_id(mod), &val);
|
||||
|
||||
dev_dbg(dev, "r %s - 0x%04d : %08x\n", rsnd_mod_name(mod), reg, val);
|
||||
|
||||
return val;
|
||||
}
|
||||
|
||||
void rsnd_write(struct rsnd_priv *priv,
|
||||
struct rsnd_mod *mod,
|
||||
enum rsnd_reg reg, u32 data)
|
||||
{
|
||||
struct device *dev = rsnd_priv_to_dev(priv);
|
||||
struct rsnd_gen *gen = rsnd_priv_to_gen(priv);
|
||||
|
||||
if (!rsnd_is_accessible_reg(priv, gen, reg))
|
||||
return;
|
||||
|
||||
regmap_fields_write(gen->regs[reg], rsnd_mod_id(mod), data);
|
||||
|
||||
dev_dbg(dev, "w %s - 0x%04d : %08x\n", rsnd_mod_name(mod), reg, data);
|
||||
}
|
||||
|
||||
void rsnd_bset(struct rsnd_priv *priv, struct rsnd_mod *mod,
|
||||
enum rsnd_reg reg, u32 mask, u32 data)
|
||||
{
|
||||
struct device *dev = rsnd_priv_to_dev(priv);
|
||||
struct rsnd_gen *gen = rsnd_priv_to_gen(priv);
|
||||
|
||||
if (!rsnd_is_accessible_reg(priv, gen, reg))
|
||||
return;
|
||||
|
||||
regmap_fields_update_bits(gen->regs[reg], rsnd_mod_id(mod),
|
||||
mask, data);
|
||||
|
||||
dev_dbg(dev, "b %s - 0x%04d : %08x/%08x\n",
|
||||
rsnd_mod_name(mod), reg, data, mask);
|
||||
}
|
||||
|
||||
#define rsnd_gen_regmap_init(priv, id_size, reg_id, conf) \
|
||||
_rsnd_gen_regmap_init(priv, id_size, reg_id, conf, ARRAY_SIZE(conf))
|
||||
static int _rsnd_gen_regmap_init(struct rsnd_priv *priv,
|
||||
int id_size,
|
||||
int reg_id,
|
||||
struct rsnd_regmap_field_conf *conf,
|
||||
int conf_size)
|
||||
{
|
||||
struct platform_device *pdev = rsnd_priv_to_pdev(priv);
|
||||
struct rsnd_gen *gen = rsnd_priv_to_gen(priv);
|
||||
struct device *dev = rsnd_priv_to_dev(priv);
|
||||
struct resource *res;
|
||||
struct regmap_config regc;
|
||||
struct regmap_field *regs;
|
||||
struct regmap *regmap;
|
||||
struct reg_field regf;
|
||||
void __iomem *base;
|
||||
int i;
|
||||
|
||||
memset(®c, 0, sizeof(regc));
|
||||
regc.reg_bits = 32;
|
||||
regc.val_bits = 32;
|
||||
regc.reg_stride = 4;
|
||||
|
||||
res = platform_get_resource(pdev, IORESOURCE_MEM, reg_id);
|
||||
if (!res)
|
||||
return -ENODEV;
|
||||
|
||||
base = devm_ioremap_resource(dev, res);
|
||||
if (IS_ERR(base))
|
||||
return PTR_ERR(base);
|
||||
|
||||
regmap = devm_regmap_init_mmio(dev, base, ®c);
|
||||
if (IS_ERR(regmap))
|
||||
return PTR_ERR(regmap);
|
||||
|
||||
gen->base[reg_id] = base;
|
||||
gen->regmap[reg_id] = regmap;
|
||||
|
||||
for (i = 0; i < conf_size; i++) {
|
||||
|
||||
regf.reg = conf[i].reg_offset;
|
||||
regf.id_offset = conf[i].id_offset;
|
||||
regf.lsb = 0;
|
||||
regf.msb = 31;
|
||||
regf.id_size = id_size;
|
||||
|
||||
regs = devm_regmap_field_alloc(dev, regmap, regf);
|
||||
if (IS_ERR(regs))
|
||||
return PTR_ERR(regs);
|
||||
|
||||
gen->regs[conf[i].idx] = regs;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
/*
|
||||
* DMA read/write register offset
|
||||
*
|
||||
* RSND_xxx_I_N for Audio DMAC input
|
||||
* RSND_xxx_O_N for Audio DMAC output
|
||||
* RSND_xxx_I_P for Audio DMAC peri peri input
|
||||
* RSND_xxx_O_P for Audio DMAC peri peri output
|
||||
*
|
||||
* ex) R-Car H2 case
|
||||
* mod / DMAC in / DMAC out / DMAC PP in / DMAC pp out
|
||||
* SSI : 0xec541000 / 0xec241008 / 0xec24100c
|
||||
* SSIU: 0xec541000 / 0xec100000 / 0xec100000 / 0xec400000 / 0xec400000
|
||||
* SCU : 0xec500000 / 0xec000000 / 0xec004000 / 0xec300000 / 0xec304000
|
||||
* CMD : 0xec500000 / / 0xec008000 0xec308000
|
||||
*/
|
||||
#define RDMA_SSI_I_N(addr, i) (addr ##_reg - 0x00300000 + (0x40 * i) + 0x8)
|
||||
#define RDMA_SSI_O_N(addr, i) (addr ##_reg - 0x00300000 + (0x40 * i) + 0xc)
|
||||
|
||||
#define RDMA_SSIU_I_N(addr, i) (addr ##_reg - 0x00441000 + (0x1000 * i))
|
||||
#define RDMA_SSIU_O_N(addr, i) (addr ##_reg - 0x00441000 + (0x1000 * i))
|
||||
|
||||
#define RDMA_SSIU_I_P(addr, i) (addr ##_reg - 0x00141000 + (0x1000 * i))
|
||||
#define RDMA_SSIU_O_P(addr, i) (addr ##_reg - 0x00141000 + (0x1000 * i))
|
||||
|
||||
#define RDMA_SRC_I_N(addr, i) (addr ##_reg - 0x00500000 + (0x400 * i))
|
||||
#define RDMA_SRC_O_N(addr, i) (addr ##_reg - 0x004fc000 + (0x400 * i))
|
||||
|
||||
#define RDMA_SRC_I_P(addr, i) (addr ##_reg - 0x00200000 + (0x400 * i))
|
||||
#define RDMA_SRC_O_P(addr, i) (addr ##_reg - 0x001fc000 + (0x400 * i))
|
||||
|
||||
#define RDMA_CMD_O_N(addr, i) (addr ##_reg - 0x004f8000 + (0x400 * i))
|
||||
#define RDMA_CMD_O_P(addr, i) (addr ##_reg - 0x001f8000 + (0x400 * i))
|
||||
|
||||
static dma_addr_t
|
||||
rsnd_gen2_dma_addr(struct rsnd_priv *priv,
|
||||
struct rsnd_mod *mod,
|
||||
int is_play, int is_from)
|
||||
{
|
||||
struct platform_device *pdev = rsnd_priv_to_pdev(priv);
|
||||
struct device *dev = rsnd_priv_to_dev(priv);
|
||||
struct rsnd_dai_stream *io = rsnd_mod_to_io(mod);
|
||||
dma_addr_t ssi_reg = platform_get_resource(pdev,
|
||||
IORESOURCE_MEM, RSND_GEN2_SSI)->start;
|
||||
dma_addr_t src_reg = platform_get_resource(pdev,
|
||||
IORESOURCE_MEM, RSND_GEN2_SCU)->start;
|
||||
int is_ssi = !!(rsnd_io_to_mod_ssi(io) == mod);
|
||||
int use_src = !!rsnd_io_to_mod_src(io);
|
||||
int use_dvc = !!rsnd_io_to_mod_dvc(io);
|
||||
int id = rsnd_mod_id(mod);
|
||||
struct dma_addr {
|
||||
dma_addr_t out_addr;
|
||||
dma_addr_t in_addr;
|
||||
} dma_addrs[3][2][3] = {
|
||||
/* SRC */
|
||||
{{{ 0, 0 },
|
||||
/* Capture */
|
||||
{ RDMA_SRC_O_N(src, id), RDMA_SRC_I_P(src, id) },
|
||||
{ RDMA_CMD_O_N(src, id), RDMA_SRC_I_P(src, id) } },
|
||||
/* Playback */
|
||||
{{ 0, 0, },
|
||||
{ RDMA_SRC_O_P(src, id), RDMA_SRC_I_N(src, id) },
|
||||
{ RDMA_CMD_O_P(src, id), RDMA_SRC_I_N(src, id) } }
|
||||
},
|
||||
/* SSI */
|
||||
/* Capture */
|
||||
{{{ RDMA_SSI_O_N(ssi, id), 0 },
|
||||
{ RDMA_SSIU_O_P(ssi, id), 0 },
|
||||
{ RDMA_SSIU_O_P(ssi, id), 0 } },
|
||||
/* Playback */
|
||||
{{ 0, RDMA_SSI_I_N(ssi, id) },
|
||||
{ 0, RDMA_SSIU_I_P(ssi, id) },
|
||||
{ 0, RDMA_SSIU_I_P(ssi, id) } }
|
||||
},
|
||||
/* SSIU */
|
||||
/* Capture */
|
||||
{{{ RDMA_SSIU_O_N(ssi, id), 0 },
|
||||
{ RDMA_SSIU_O_P(ssi, id), 0 },
|
||||
{ RDMA_SSIU_O_P(ssi, id), 0 } },
|
||||
/* Playback */
|
||||
{{ 0, RDMA_SSIU_I_N(ssi, id) },
|
||||
{ 0, RDMA_SSIU_I_P(ssi, id) },
|
||||
{ 0, RDMA_SSIU_I_P(ssi, id) } } },
|
||||
};
|
||||
|
||||
/* it shouldn't happen */
|
||||
if (use_dvc && !use_src)
|
||||
dev_err(dev, "DVC is selected without SRC\n");
|
||||
|
||||
/* use SSIU or SSI ? */
|
||||
if (is_ssi && (0 == strcmp(rsnd_mod_dma_name(mod), "ssiu")))
|
||||
is_ssi++;
|
||||
|
||||
return (is_from) ?
|
||||
dma_addrs[is_ssi][is_play][use_src + use_dvc].out_addr :
|
||||
dma_addrs[is_ssi][is_play][use_src + use_dvc].in_addr;
|
||||
}
|
||||
|
||||
dma_addr_t rsnd_gen_dma_addr(struct rsnd_priv *priv,
|
||||
struct rsnd_mod *mod,
|
||||
int is_play, int is_from)
|
||||
{
|
||||
/*
|
||||
* gen1 uses default DMA addr
|
||||
*/
|
||||
if (rsnd_is_gen1(priv))
|
||||
return 0;
|
||||
|
||||
if (!mod)
|
||||
return 0;
|
||||
|
||||
return rsnd_gen2_dma_addr(priv, mod, is_play, is_from);
|
||||
}
|
||||
|
||||
/*
|
||||
* Gen2
|
||||
*/
|
||||
static int rsnd_gen2_probe(struct platform_device *pdev,
|
||||
struct rsnd_priv *priv)
|
||||
{
|
||||
struct device *dev = rsnd_priv_to_dev(priv);
|
||||
struct rsnd_regmap_field_conf conf_ssiu[] = {
|
||||
RSND_GEN_S_REG(SSI_MODE0, 0x800),
|
||||
RSND_GEN_S_REG(SSI_MODE1, 0x804),
|
||||
/* FIXME: it needs SSI_MODE2/3 in the future */
|
||||
RSND_GEN_M_REG(SSI_BUSIF_MODE, 0x0, 0x80),
|
||||
RSND_GEN_M_REG(SSI_BUSIF_ADINR, 0x4, 0x80),
|
||||
RSND_GEN_M_REG(BUSIF_DALIGN, 0x8, 0x80),
|
||||
RSND_GEN_M_REG(SSI_CTRL, 0x10, 0x80),
|
||||
RSND_GEN_M_REG(INT_ENABLE, 0x18, 0x80),
|
||||
};
|
||||
struct rsnd_regmap_field_conf conf_scu[] = {
|
||||
RSND_GEN_M_REG(SRC_BUSIF_MODE, 0x0, 0x20),
|
||||
RSND_GEN_M_REG(SRC_ROUTE_MODE0, 0xc, 0x20),
|
||||
RSND_GEN_M_REG(SRC_CTRL, 0x10, 0x20),
|
||||
RSND_GEN_M_REG(CMD_ROUTE_SLCT, 0x18c, 0x20),
|
||||
RSND_GEN_M_REG(CMD_CTRL, 0x190, 0x20),
|
||||
RSND_GEN_M_REG(SRC_SWRSR, 0x200, 0x40),
|
||||
RSND_GEN_M_REG(SRC_SRCIR, 0x204, 0x40),
|
||||
RSND_GEN_M_REG(SRC_ADINR, 0x214, 0x40),
|
||||
RSND_GEN_M_REG(SRC_IFSCR, 0x21c, 0x40),
|
||||
RSND_GEN_M_REG(SRC_IFSVR, 0x220, 0x40),
|
||||
RSND_GEN_M_REG(SRC_SRCCR, 0x224, 0x40),
|
||||
RSND_GEN_M_REG(SRC_BSDSR, 0x22c, 0x40),
|
||||
RSND_GEN_M_REG(SRC_BSISR, 0x238, 0x40),
|
||||
RSND_GEN_M_REG(DVC_SWRSR, 0xe00, 0x100),
|
||||
RSND_GEN_M_REG(DVC_DVUIR, 0xe04, 0x100),
|
||||
RSND_GEN_M_REG(DVC_ADINR, 0xe08, 0x100),
|
||||
RSND_GEN_M_REG(DVC_DVUCR, 0xe10, 0x100),
|
||||
RSND_GEN_M_REG(DVC_ZCMCR, 0xe14, 0x100),
|
||||
RSND_GEN_M_REG(DVC_VOL0R, 0xe28, 0x100),
|
||||
RSND_GEN_M_REG(DVC_VOL1R, 0xe2c, 0x100),
|
||||
RSND_GEN_M_REG(DVC_DVUER, 0xe48, 0x100),
|
||||
};
|
||||
struct rsnd_regmap_field_conf conf_adg[] = {
|
||||
RSND_GEN_S_REG(BRRA, 0x00),
|
||||
RSND_GEN_S_REG(BRRB, 0x04),
|
||||
RSND_GEN_S_REG(SSICKR, 0x08),
|
||||
RSND_GEN_S_REG(AUDIO_CLK_SEL0, 0x0c),
|
||||
RSND_GEN_S_REG(AUDIO_CLK_SEL1, 0x10),
|
||||
RSND_GEN_S_REG(AUDIO_CLK_SEL2, 0x14),
|
||||
RSND_GEN_S_REG(DIV_EN, 0x30),
|
||||
RSND_GEN_S_REG(SRCIN_TIMSEL0, 0x34),
|
||||
RSND_GEN_S_REG(SRCIN_TIMSEL1, 0x38),
|
||||
RSND_GEN_S_REG(SRCIN_TIMSEL2, 0x3c),
|
||||
RSND_GEN_S_REG(SRCIN_TIMSEL3, 0x40),
|
||||
RSND_GEN_S_REG(SRCIN_TIMSEL4, 0x44),
|
||||
RSND_GEN_S_REG(SRCOUT_TIMSEL0, 0x48),
|
||||
RSND_GEN_S_REG(SRCOUT_TIMSEL1, 0x4c),
|
||||
RSND_GEN_S_REG(SRCOUT_TIMSEL2, 0x50),
|
||||
RSND_GEN_S_REG(SRCOUT_TIMSEL3, 0x54),
|
||||
RSND_GEN_S_REG(SRCOUT_TIMSEL4, 0x58),
|
||||
RSND_GEN_S_REG(CMDOUT_TIMSEL, 0x5c),
|
||||
};
|
||||
struct rsnd_regmap_field_conf conf_ssi[] = {
|
||||
RSND_GEN_M_REG(SSICR, 0x00, 0x40),
|
||||
RSND_GEN_M_REG(SSISR, 0x04, 0x40),
|
||||
RSND_GEN_M_REG(SSITDR, 0x08, 0x40),
|
||||
RSND_GEN_M_REG(SSIRDR, 0x0c, 0x40),
|
||||
RSND_GEN_M_REG(SSIWSR, 0x20, 0x40),
|
||||
};
|
||||
int ret_ssiu;
|
||||
int ret_scu;
|
||||
int ret_adg;
|
||||
int ret_ssi;
|
||||
|
||||
ret_ssiu = rsnd_gen_regmap_init(priv, 10, RSND_GEN2_SSIU, conf_ssiu);
|
||||
ret_scu = rsnd_gen_regmap_init(priv, 10, RSND_GEN2_SCU, conf_scu);
|
||||
ret_adg = rsnd_gen_regmap_init(priv, 10, RSND_GEN2_ADG, conf_adg);
|
||||
ret_ssi = rsnd_gen_regmap_init(priv, 10, RSND_GEN2_SSI, conf_ssi);
|
||||
if (ret_ssiu < 0 ||
|
||||
ret_scu < 0 ||
|
||||
ret_adg < 0 ||
|
||||
ret_ssi < 0)
|
||||
return ret_ssiu | ret_scu | ret_adg | ret_ssi;
|
||||
|
||||
dev_dbg(dev, "Gen2 is probed\n");
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
/*
|
||||
* Gen1
|
||||
*/
|
||||
|
||||
static int rsnd_gen1_probe(struct platform_device *pdev,
|
||||
struct rsnd_priv *priv)
|
||||
{
|
||||
struct device *dev = rsnd_priv_to_dev(priv);
|
||||
struct rsnd_regmap_field_conf conf_sru[] = {
|
||||
RSND_GEN_S_REG(SRC_ROUTE_SEL, 0x00),
|
||||
RSND_GEN_S_REG(SRC_TMG_SEL0, 0x08),
|
||||
RSND_GEN_S_REG(SRC_TMG_SEL1, 0x0c),
|
||||
RSND_GEN_S_REG(SRC_TMG_SEL2, 0x10),
|
||||
RSND_GEN_S_REG(SRC_ROUTE_CTRL, 0xc0),
|
||||
RSND_GEN_S_REG(SSI_MODE0, 0xD0),
|
||||
RSND_GEN_S_REG(SSI_MODE1, 0xD4),
|
||||
RSND_GEN_M_REG(SRC_BUSIF_MODE, 0x20, 0x4),
|
||||
RSND_GEN_M_REG(SRC_ROUTE_MODE0, 0x50, 0x8),
|
||||
RSND_GEN_M_REG(SRC_SWRSR, 0x200, 0x40),
|
||||
RSND_GEN_M_REG(SRC_SRCIR, 0x204, 0x40),
|
||||
RSND_GEN_M_REG(SRC_ADINR, 0x214, 0x40),
|
||||
RSND_GEN_M_REG(SRC_IFSCR, 0x21c, 0x40),
|
||||
RSND_GEN_M_REG(SRC_IFSVR, 0x220, 0x40),
|
||||
RSND_GEN_M_REG(SRC_SRCCR, 0x224, 0x40),
|
||||
RSND_GEN_M_REG(SRC_MNFSR, 0x228, 0x40),
|
||||
};
|
||||
struct rsnd_regmap_field_conf conf_adg[] = {
|
||||
RSND_GEN_S_REG(BRRA, 0x00),
|
||||
RSND_GEN_S_REG(BRRB, 0x04),
|
||||
RSND_GEN_S_REG(SSICKR, 0x08),
|
||||
RSND_GEN_S_REG(AUDIO_CLK_SEL0, 0x0c),
|
||||
RSND_GEN_S_REG(AUDIO_CLK_SEL1, 0x10),
|
||||
RSND_GEN_S_REG(AUDIO_CLK_SEL3, 0x18),
|
||||
RSND_GEN_S_REG(AUDIO_CLK_SEL4, 0x1c),
|
||||
RSND_GEN_S_REG(AUDIO_CLK_SEL5, 0x20),
|
||||
};
|
||||
struct rsnd_regmap_field_conf conf_ssi[] = {
|
||||
RSND_GEN_M_REG(SSICR, 0x00, 0x40),
|
||||
RSND_GEN_M_REG(SSISR, 0x04, 0x40),
|
||||
RSND_GEN_M_REG(SSITDR, 0x08, 0x40),
|
||||
RSND_GEN_M_REG(SSIRDR, 0x0c, 0x40),
|
||||
RSND_GEN_M_REG(SSIWSR, 0x20, 0x40),
|
||||
};
|
||||
int ret_sru;
|
||||
int ret_adg;
|
||||
int ret_ssi;
|
||||
|
||||
ret_sru = rsnd_gen_regmap_init(priv, 9, RSND_GEN1_SRU, conf_sru);
|
||||
ret_adg = rsnd_gen_regmap_init(priv, 9, RSND_GEN1_ADG, conf_adg);
|
||||
ret_ssi = rsnd_gen_regmap_init(priv, 9, RSND_GEN1_SSI, conf_ssi);
|
||||
if (ret_sru < 0 ||
|
||||
ret_adg < 0 ||
|
||||
ret_ssi < 0)
|
||||
return ret_sru | ret_adg | ret_ssi;
|
||||
|
||||
dev_dbg(dev, "Gen1 is probed\n");
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
/*
|
||||
* Gen
|
||||
*/
|
||||
static void rsnd_of_parse_gen(struct platform_device *pdev,
|
||||
const struct rsnd_of_data *of_data,
|
||||
struct rsnd_priv *priv)
|
||||
{
|
||||
struct rcar_snd_info *info = priv->info;
|
||||
|
||||
if (!of_data)
|
||||
return;
|
||||
|
||||
info->flags = of_data->flags;
|
||||
}
|
||||
|
||||
int rsnd_gen_probe(struct platform_device *pdev,
|
||||
const struct rsnd_of_data *of_data,
|
||||
struct rsnd_priv *priv)
|
||||
{
|
||||
struct device *dev = rsnd_priv_to_dev(priv);
|
||||
struct rsnd_gen *gen;
|
||||
int ret;
|
||||
|
||||
rsnd_of_parse_gen(pdev, of_data, priv);
|
||||
|
||||
gen = devm_kzalloc(dev, sizeof(*gen), GFP_KERNEL);
|
||||
if (!gen) {
|
||||
dev_err(dev, "GEN allocate failed\n");
|
||||
return -ENOMEM;
|
||||
}
|
||||
|
||||
priv->gen = gen;
|
||||
|
||||
ret = -ENODEV;
|
||||
if (rsnd_is_gen1(priv))
|
||||
ret = rsnd_gen1_probe(pdev, priv);
|
||||
else if (rsnd_is_gen2(priv))
|
||||
ret = rsnd_gen2_probe(pdev, priv);
|
||||
|
||||
if (ret < 0)
|
||||
dev_err(dev, "unknown generation R-Car sound device\n");
|
||||
|
||||
return ret;
|
||||
}
|
||||
427
sound/soc/sh/rcar/rsnd.h
Normal file
427
sound/soc/sh/rcar/rsnd.h
Normal file
|
|
@ -0,0 +1,427 @@
|
|||
/*
|
||||
* Renesas R-Car
|
||||
*
|
||||
* Copyright (C) 2013 Renesas Solutions Corp.
|
||||
* Kuninori Morimoto <kuninori.morimoto.gx@renesas.com>
|
||||
*
|
||||
* 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.
|
||||
*/
|
||||
#ifndef RSND_H
|
||||
#define RSND_H
|
||||
|
||||
#include <linux/clk.h>
|
||||
#include <linux/device.h>
|
||||
#include <linux/dma-mapping.h>
|
||||
#include <linux/io.h>
|
||||
#include <linux/list.h>
|
||||
#include <linux/module.h>
|
||||
#include <linux/of_device.h>
|
||||
#include <linux/of_irq.h>
|
||||
#include <linux/sh_dma.h>
|
||||
#include <linux/workqueue.h>
|
||||
#include <sound/rcar_snd.h>
|
||||
#include <sound/soc.h>
|
||||
#include <sound/pcm_params.h>
|
||||
|
||||
/*
|
||||
* pseudo register
|
||||
*
|
||||
* The register address offsets SRU/SCU/SSIU on Gen1/Gen2 are very different.
|
||||
* This driver uses pseudo register in order to hide it.
|
||||
* see gen1/gen2 for detail
|
||||
*/
|
||||
enum rsnd_reg {
|
||||
/* SRU/SCU/SSIU */
|
||||
RSND_REG_SSI_MODE0,
|
||||
RSND_REG_SSI_MODE1,
|
||||
RSND_REG_SRC_BUSIF_MODE,
|
||||
RSND_REG_SRC_ROUTE_MODE0,
|
||||
RSND_REG_SRC_SWRSR,
|
||||
RSND_REG_SRC_SRCIR,
|
||||
RSND_REG_SRC_ADINR,
|
||||
RSND_REG_SRC_IFSCR,
|
||||
RSND_REG_SRC_IFSVR,
|
||||
RSND_REG_SRC_SRCCR,
|
||||
RSND_REG_CMD_ROUTE_SLCT,
|
||||
RSND_REG_DVC_SWRSR,
|
||||
RSND_REG_DVC_DVUIR,
|
||||
RSND_REG_DVC_ADINR,
|
||||
RSND_REG_DVC_DVUCR,
|
||||
RSND_REG_DVC_ZCMCR,
|
||||
RSND_REG_DVC_VOL0R,
|
||||
RSND_REG_DVC_VOL1R,
|
||||
RSND_REG_DVC_DVUER,
|
||||
|
||||
/* ADG */
|
||||
RSND_REG_BRRA,
|
||||
RSND_REG_BRRB,
|
||||
RSND_REG_SSICKR,
|
||||
RSND_REG_AUDIO_CLK_SEL0,
|
||||
RSND_REG_AUDIO_CLK_SEL1,
|
||||
|
||||
/* SSI */
|
||||
RSND_REG_SSICR,
|
||||
RSND_REG_SSISR,
|
||||
RSND_REG_SSITDR,
|
||||
RSND_REG_SSIRDR,
|
||||
RSND_REG_SSIWSR,
|
||||
|
||||
/* SHARE see below */
|
||||
RSND_REG_SHARE01,
|
||||
RSND_REG_SHARE02,
|
||||
RSND_REG_SHARE03,
|
||||
RSND_REG_SHARE04,
|
||||
RSND_REG_SHARE05,
|
||||
RSND_REG_SHARE06,
|
||||
RSND_REG_SHARE07,
|
||||
RSND_REG_SHARE08,
|
||||
RSND_REG_SHARE09,
|
||||
RSND_REG_SHARE10,
|
||||
RSND_REG_SHARE11,
|
||||
RSND_REG_SHARE12,
|
||||
RSND_REG_SHARE13,
|
||||
RSND_REG_SHARE14,
|
||||
RSND_REG_SHARE15,
|
||||
RSND_REG_SHARE16,
|
||||
RSND_REG_SHARE17,
|
||||
RSND_REG_SHARE18,
|
||||
RSND_REG_SHARE19,
|
||||
RSND_REG_SHARE20,
|
||||
RSND_REG_SHARE21,
|
||||
RSND_REG_SHARE22,
|
||||
|
||||
RSND_REG_MAX,
|
||||
};
|
||||
|
||||
/* Gen1 only */
|
||||
#define RSND_REG_SRC_ROUTE_SEL RSND_REG_SHARE01
|
||||
#define RSND_REG_SRC_TMG_SEL0 RSND_REG_SHARE02
|
||||
#define RSND_REG_SRC_TMG_SEL1 RSND_REG_SHARE03
|
||||
#define RSND_REG_SRC_TMG_SEL2 RSND_REG_SHARE04
|
||||
#define RSND_REG_SRC_ROUTE_CTRL RSND_REG_SHARE05
|
||||
#define RSND_REG_SRC_MNFSR RSND_REG_SHARE06
|
||||
#define RSND_REG_AUDIO_CLK_SEL3 RSND_REG_SHARE07
|
||||
#define RSND_REG_AUDIO_CLK_SEL4 RSND_REG_SHARE08
|
||||
#define RSND_REG_AUDIO_CLK_SEL5 RSND_REG_SHARE09
|
||||
|
||||
/* Gen2 only */
|
||||
#define RSND_REG_SRC_CTRL RSND_REG_SHARE01
|
||||
#define RSND_REG_SSI_CTRL RSND_REG_SHARE02
|
||||
#define RSND_REG_SSI_BUSIF_MODE RSND_REG_SHARE03
|
||||
#define RSND_REG_SSI_BUSIF_ADINR RSND_REG_SHARE04
|
||||
#define RSND_REG_INT_ENABLE RSND_REG_SHARE05
|
||||
#define RSND_REG_SRC_BSDSR RSND_REG_SHARE06
|
||||
#define RSND_REG_SRC_BSISR RSND_REG_SHARE07
|
||||
#define RSND_REG_DIV_EN RSND_REG_SHARE08
|
||||
#define RSND_REG_SRCIN_TIMSEL0 RSND_REG_SHARE09
|
||||
#define RSND_REG_SRCIN_TIMSEL1 RSND_REG_SHARE10
|
||||
#define RSND_REG_SRCIN_TIMSEL2 RSND_REG_SHARE11
|
||||
#define RSND_REG_SRCIN_TIMSEL3 RSND_REG_SHARE12
|
||||
#define RSND_REG_SRCIN_TIMSEL4 RSND_REG_SHARE13
|
||||
#define RSND_REG_SRCOUT_TIMSEL0 RSND_REG_SHARE14
|
||||
#define RSND_REG_SRCOUT_TIMSEL1 RSND_REG_SHARE15
|
||||
#define RSND_REG_SRCOUT_TIMSEL2 RSND_REG_SHARE16
|
||||
#define RSND_REG_SRCOUT_TIMSEL3 RSND_REG_SHARE17
|
||||
#define RSND_REG_SRCOUT_TIMSEL4 RSND_REG_SHARE18
|
||||
#define RSND_REG_AUDIO_CLK_SEL2 RSND_REG_SHARE19
|
||||
#define RSND_REG_CMD_CTRL RSND_REG_SHARE20
|
||||
#define RSND_REG_CMDOUT_TIMSEL RSND_REG_SHARE21
|
||||
#define RSND_REG_BUSIF_DALIGN RSND_REG_SHARE22
|
||||
|
||||
struct rsnd_of_data;
|
||||
struct rsnd_priv;
|
||||
struct rsnd_mod;
|
||||
struct rsnd_dai;
|
||||
struct rsnd_dai_stream;
|
||||
|
||||
/*
|
||||
* R-Car basic functions
|
||||
*/
|
||||
#define rsnd_mod_read(m, r) \
|
||||
rsnd_read(rsnd_mod_to_priv(m), m, RSND_REG_##r)
|
||||
#define rsnd_mod_write(m, r, d) \
|
||||
rsnd_write(rsnd_mod_to_priv(m), m, RSND_REG_##r, d)
|
||||
#define rsnd_mod_bset(m, r, s, d) \
|
||||
rsnd_bset(rsnd_mod_to_priv(m), m, RSND_REG_##r, s, d)
|
||||
|
||||
u32 rsnd_read(struct rsnd_priv *priv, struct rsnd_mod *mod, enum rsnd_reg reg);
|
||||
void rsnd_write(struct rsnd_priv *priv, struct rsnd_mod *mod,
|
||||
enum rsnd_reg reg, u32 data);
|
||||
void rsnd_bset(struct rsnd_priv *priv, struct rsnd_mod *mod, enum rsnd_reg reg,
|
||||
u32 mask, u32 data);
|
||||
u32 rsnd_get_adinr(struct rsnd_mod *mod);
|
||||
|
||||
/*
|
||||
* R-Car DMA
|
||||
*/
|
||||
struct rsnd_dma {
|
||||
struct sh_dmae_slave slave;
|
||||
struct dma_chan *chan;
|
||||
enum dma_transfer_direction dir;
|
||||
dma_addr_t addr;
|
||||
};
|
||||
|
||||
void rsnd_dma_start(struct rsnd_dma *dma);
|
||||
void rsnd_dma_stop(struct rsnd_dma *dma);
|
||||
int rsnd_dma_available(struct rsnd_dma *dma);
|
||||
int rsnd_dma_init(struct rsnd_priv *priv, struct rsnd_dma *dma,
|
||||
int is_play, int id);
|
||||
void rsnd_dma_quit(struct rsnd_priv *priv,
|
||||
struct rsnd_dma *dma);
|
||||
|
||||
|
||||
/*
|
||||
* R-Car sound mod
|
||||
*/
|
||||
enum rsnd_mod_type {
|
||||
RSND_MOD_SRC = 0,
|
||||
RSND_MOD_SSI,
|
||||
RSND_MOD_DVC,
|
||||
RSND_MOD_MAX,
|
||||
};
|
||||
|
||||
struct rsnd_mod_ops {
|
||||
char *name;
|
||||
char* (*dma_name)(struct rsnd_mod *mod);
|
||||
int (*probe)(struct rsnd_mod *mod,
|
||||
struct rsnd_dai *rdai);
|
||||
int (*remove)(struct rsnd_mod *mod,
|
||||
struct rsnd_dai *rdai);
|
||||
int (*init)(struct rsnd_mod *mod,
|
||||
struct rsnd_dai *rdai);
|
||||
int (*quit)(struct rsnd_mod *mod,
|
||||
struct rsnd_dai *rdai);
|
||||
int (*start)(struct rsnd_mod *mod,
|
||||
struct rsnd_dai *rdai);
|
||||
int (*stop)(struct rsnd_mod *mod,
|
||||
struct rsnd_dai *rdai);
|
||||
int (*pcm_new)(struct rsnd_mod *mod,
|
||||
struct rsnd_dai *rdai,
|
||||
struct snd_soc_pcm_runtime *rtd);
|
||||
};
|
||||
|
||||
struct rsnd_dai_stream;
|
||||
struct rsnd_mod {
|
||||
int id;
|
||||
enum rsnd_mod_type type;
|
||||
struct rsnd_priv *priv;
|
||||
struct rsnd_mod_ops *ops;
|
||||
struct rsnd_dma dma;
|
||||
struct rsnd_dai_stream *io;
|
||||
};
|
||||
|
||||
#define rsnd_mod_to_priv(mod) ((mod)->priv)
|
||||
#define rsnd_mod_to_dma(mod) (&(mod)->dma)
|
||||
#define rsnd_dma_to_mod(_dma) container_of((_dma), struct rsnd_mod, dma)
|
||||
#define rsnd_mod_to_io(mod) ((mod)->io)
|
||||
#define rsnd_mod_id(mod) ((mod)->id)
|
||||
|
||||
void rsnd_mod_init(struct rsnd_priv *priv,
|
||||
struct rsnd_mod *mod,
|
||||
struct rsnd_mod_ops *ops,
|
||||
enum rsnd_mod_type type,
|
||||
int id);
|
||||
char *rsnd_mod_name(struct rsnd_mod *mod);
|
||||
char *rsnd_mod_dma_name(struct rsnd_mod *mod);
|
||||
|
||||
/*
|
||||
* R-Car sound DAI
|
||||
*/
|
||||
#define RSND_DAI_NAME_SIZE 16
|
||||
struct rsnd_dai_stream {
|
||||
struct snd_pcm_substream *substream;
|
||||
struct rsnd_mod *mod[RSND_MOD_MAX];
|
||||
struct rsnd_dai_path_info *info; /* rcar_snd.h */
|
||||
int byte_pos;
|
||||
int period_pos;
|
||||
int byte_per_period;
|
||||
int next_period_byte;
|
||||
};
|
||||
#define rsnd_io_to_mod_ssi(io) ((io)->mod[RSND_MOD_SSI])
|
||||
#define rsnd_io_to_mod_src(io) ((io)->mod[RSND_MOD_SRC])
|
||||
#define rsnd_io_to_mod_dvc(io) ((io)->mod[RSND_MOD_DVC])
|
||||
|
||||
struct rsnd_dai {
|
||||
char name[RSND_DAI_NAME_SIZE];
|
||||
struct rsnd_dai_platform_info *info; /* rcar_snd.h */
|
||||
struct rsnd_dai_stream playback;
|
||||
struct rsnd_dai_stream capture;
|
||||
|
||||
unsigned int clk_master:1;
|
||||
unsigned int bit_clk_inv:1;
|
||||
unsigned int frm_clk_inv:1;
|
||||
unsigned int sys_delay:1;
|
||||
unsigned int data_alignment:1;
|
||||
};
|
||||
|
||||
#define rsnd_rdai_nr(priv) ((priv)->rdai_nr)
|
||||
#define for_each_rsnd_dai(rdai, priv, i) \
|
||||
for (i = 0; \
|
||||
(i < rsnd_rdai_nr(priv)) && \
|
||||
((rdai) = rsnd_dai_get(priv, i)); \
|
||||
i++)
|
||||
|
||||
struct rsnd_dai *rsnd_dai_get(struct rsnd_priv *priv, int id);
|
||||
int rsnd_dai_is_play(struct rsnd_dai *rdai, struct rsnd_dai_stream *io);
|
||||
int rsnd_dai_id(struct rsnd_priv *priv, struct rsnd_dai *rdai);
|
||||
#define rsnd_dai_get_platform_info(rdai) ((rdai)->info)
|
||||
#define rsnd_io_to_runtime(io) ((io)->substream->runtime)
|
||||
|
||||
void rsnd_dai_pointer_update(struct rsnd_dai_stream *io, int cnt);
|
||||
int rsnd_dai_pointer_offset(struct rsnd_dai_stream *io, int additional);
|
||||
#define rsnd_dai_is_clk_master(rdai) ((rdai)->clk_master)
|
||||
|
||||
/*
|
||||
* R-Car Gen1/Gen2
|
||||
*/
|
||||
int rsnd_gen_probe(struct platform_device *pdev,
|
||||
const struct rsnd_of_data *of_data,
|
||||
struct rsnd_priv *priv);
|
||||
void __iomem *rsnd_gen_reg_get(struct rsnd_priv *priv,
|
||||
struct rsnd_mod *mod,
|
||||
enum rsnd_reg reg);
|
||||
dma_addr_t rsnd_gen_dma_addr(struct rsnd_priv *priv,
|
||||
struct rsnd_mod *mod,
|
||||
int is_play, int is_from);
|
||||
|
||||
#define rsnd_is_gen1(s) (((s)->info->flags & RSND_GEN_MASK) == RSND_GEN1)
|
||||
#define rsnd_is_gen2(s) (((s)->info->flags & RSND_GEN_MASK) == RSND_GEN2)
|
||||
|
||||
/*
|
||||
* R-Car ADG
|
||||
*/
|
||||
int rsnd_adg_ssi_clk_stop(struct rsnd_mod *mod);
|
||||
int rsnd_adg_ssi_clk_try_start(struct rsnd_mod *mod, unsigned int rate);
|
||||
int rsnd_adg_probe(struct platform_device *pdev,
|
||||
const struct rsnd_of_data *of_data,
|
||||
struct rsnd_priv *priv);
|
||||
int rsnd_adg_set_convert_clk_gen1(struct rsnd_priv *priv,
|
||||
struct rsnd_mod *mod,
|
||||
unsigned int src_rate,
|
||||
unsigned int dst_rate);
|
||||
int rsnd_adg_set_convert_clk_gen2(struct rsnd_mod *mod,
|
||||
struct rsnd_dai *rdai,
|
||||
struct rsnd_dai_stream *io,
|
||||
unsigned int src_rate,
|
||||
unsigned int dst_rate);
|
||||
int rsnd_adg_set_convert_timing_gen2(struct rsnd_mod *mod,
|
||||
struct rsnd_dai *rdai,
|
||||
struct rsnd_dai_stream *io);
|
||||
int rsnd_adg_set_cmd_timsel_gen2(struct rsnd_dai *rdai,
|
||||
struct rsnd_mod *mod,
|
||||
struct rsnd_dai_stream *io);
|
||||
|
||||
/*
|
||||
* R-Car sound priv
|
||||
*/
|
||||
struct rsnd_of_data {
|
||||
u32 flags;
|
||||
};
|
||||
|
||||
struct rsnd_priv {
|
||||
|
||||
struct platform_device *pdev;
|
||||
struct rcar_snd_info *info;
|
||||
spinlock_t lock;
|
||||
|
||||
/*
|
||||
* below value will be filled on rsnd_gen_probe()
|
||||
*/
|
||||
void *gen;
|
||||
|
||||
/*
|
||||
* below value will be filled on rsnd_src_probe()
|
||||
*/
|
||||
void *src;
|
||||
int src_nr;
|
||||
|
||||
/*
|
||||
* below value will be filled on rsnd_adg_probe()
|
||||
*/
|
||||
void *adg;
|
||||
|
||||
/*
|
||||
* below value will be filled on rsnd_ssi_probe()
|
||||
*/
|
||||
void *ssi;
|
||||
int ssi_nr;
|
||||
|
||||
/*
|
||||
* below value will be filled on rsnd_dvc_probe()
|
||||
*/
|
||||
void *dvc;
|
||||
int dvc_nr;
|
||||
|
||||
/*
|
||||
* below value will be filled on rsnd_dai_probe()
|
||||
*/
|
||||
struct snd_soc_dai_driver *daidrv;
|
||||
struct rsnd_dai *rdai;
|
||||
int rdai_nr;
|
||||
};
|
||||
|
||||
#define rsnd_priv_to_pdev(priv) ((priv)->pdev)
|
||||
#define rsnd_priv_to_dev(priv) (&(rsnd_priv_to_pdev(priv)->dev))
|
||||
#define rsnd_priv_to_info(priv) ((priv)->info)
|
||||
#define rsnd_lock(priv, flags) spin_lock_irqsave(&priv->lock, flags)
|
||||
#define rsnd_unlock(priv, flags) spin_unlock_irqrestore(&priv->lock, flags)
|
||||
|
||||
#define rsnd_info_is_playback(priv, type) \
|
||||
({ \
|
||||
struct rcar_snd_info *info = rsnd_priv_to_info(priv); \
|
||||
int i, is_play = 0; \
|
||||
for (i = 0; i < info->dai_info_nr; i++) { \
|
||||
if (info->dai_info[i].playback.type == (type)->info) { \
|
||||
is_play = 1; \
|
||||
break; \
|
||||
} \
|
||||
} \
|
||||
is_play; \
|
||||
})
|
||||
|
||||
/*
|
||||
* R-Car SRC
|
||||
*/
|
||||
int rsnd_src_probe(struct platform_device *pdev,
|
||||
const struct rsnd_of_data *of_data,
|
||||
struct rsnd_priv *priv);
|
||||
struct rsnd_mod *rsnd_src_mod_get(struct rsnd_priv *priv, int id);
|
||||
unsigned int rsnd_src_get_ssi_rate(struct rsnd_priv *priv,
|
||||
struct rsnd_dai_stream *io,
|
||||
struct snd_pcm_runtime *runtime);
|
||||
int rsnd_src_ssiu_start(struct rsnd_mod *ssi_mod,
|
||||
struct rsnd_dai *rdai,
|
||||
int use_busif);
|
||||
int rsnd_src_ssiu_stop(struct rsnd_mod *ssi_mod,
|
||||
struct rsnd_dai *rdai,
|
||||
int use_busif);
|
||||
int rsnd_src_enable_ssi_irq(struct rsnd_mod *ssi_mod,
|
||||
struct rsnd_dai *rdai);
|
||||
|
||||
#define rsnd_src_nr(priv) ((priv)->src_nr)
|
||||
|
||||
/*
|
||||
* R-Car SSI
|
||||
*/
|
||||
int rsnd_ssi_probe(struct platform_device *pdev,
|
||||
const struct rsnd_of_data *of_data,
|
||||
struct rsnd_priv *priv);
|
||||
struct rsnd_mod *rsnd_ssi_mod_get(struct rsnd_priv *priv, int id);
|
||||
int rsnd_ssi_is_pin_sharing(struct rsnd_mod *mod);
|
||||
|
||||
/*
|
||||
* R-Car DVC
|
||||
*/
|
||||
int rsnd_dvc_probe(struct platform_device *pdev,
|
||||
const struct rsnd_of_data *of_data,
|
||||
struct rsnd_priv *priv);
|
||||
void rsnd_dvc_remove(struct platform_device *pdev,
|
||||
struct rsnd_priv *priv);
|
||||
struct rsnd_mod *rsnd_dvc_mod_get(struct rsnd_priv *priv, int id);
|
||||
|
||||
#define rsnd_dvc_nr(priv) ((priv)->dvc_nr)
|
||||
|
||||
|
||||
#endif
|
||||
753
sound/soc/sh/rcar/src.c
Normal file
753
sound/soc/sh/rcar/src.c
Normal file
|
|
@ -0,0 +1,753 @@
|
|||
/*
|
||||
* Renesas R-Car SRC support
|
||||
*
|
||||
* Copyright (C) 2013 Renesas Solutions Corp.
|
||||
* Kuninori Morimoto <kuninori.morimoto.gx@renesas.com>
|
||||
*
|
||||
* 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 "rsnd.h"
|
||||
|
||||
#define SRC_NAME "src"
|
||||
|
||||
struct rsnd_src {
|
||||
struct rsnd_src_platform_info *info; /* rcar_snd.h */
|
||||
struct rsnd_mod mod;
|
||||
struct clk *clk;
|
||||
};
|
||||
|
||||
#define RSND_SRC_NAME_SIZE 16
|
||||
|
||||
#define rsnd_src_convert_rate(p) ((p)->info->convert_rate)
|
||||
#define rsnd_mod_to_src(_mod) \
|
||||
container_of((_mod), struct rsnd_src, mod)
|
||||
#define rsnd_src_dma_available(src) \
|
||||
rsnd_dma_available(rsnd_mod_to_dma(&(src)->mod))
|
||||
|
||||
#define for_each_rsnd_src(pos, priv, i) \
|
||||
for ((i) = 0; \
|
||||
((i) < rsnd_src_nr(priv)) && \
|
||||
((pos) = (struct rsnd_src *)(priv)->src + i); \
|
||||
i++)
|
||||
|
||||
|
||||
/*
|
||||
* image of SRC (Sampling Rate Converter)
|
||||
*
|
||||
* 96kHz <-> +-----+ 48kHz +-----+ 48kHz +-------+
|
||||
* 48kHz <-> | SRC | <------> | SSI | <-----> | codec |
|
||||
* 44.1kHz <-> +-----+ +-----+ +-------+
|
||||
* ...
|
||||
*
|
||||
*/
|
||||
|
||||
/*
|
||||
* src.c is caring...
|
||||
*
|
||||
* Gen1
|
||||
*
|
||||
* [mem] -> [SRU] -> [SSI]
|
||||
* |--------|
|
||||
*
|
||||
* Gen2
|
||||
*
|
||||
* [mem] -> [SRC] -> [SSIU] -> [SSI]
|
||||
* |-----------------|
|
||||
*/
|
||||
|
||||
/*
|
||||
* How to use SRC bypass mode for debugging
|
||||
*
|
||||
* SRC has bypass mode, and it is useful for debugging.
|
||||
* In Gen2 case,
|
||||
* SRCm_MODE controls whether SRC is used or not
|
||||
* SSI_MODE0 controls whether SSIU which receives SRC data
|
||||
* is used or not.
|
||||
* Both SRCm_MODE/SSI_MODE0 settings are needed if you use SRC,
|
||||
* but SRC bypass mode needs SSI_MODE0 only.
|
||||
*
|
||||
* This driver request
|
||||
* struct rsnd_src_platform_info {
|
||||
* u32 convert_rate;
|
||||
* int dma_id;
|
||||
* }
|
||||
*
|
||||
* rsnd_src_convert_rate() indicates
|
||||
* above convert_rate, and it controls
|
||||
* whether SRC is used or not.
|
||||
*
|
||||
* ex) doesn't use SRC
|
||||
* static struct rsnd_dai_platform_info rsnd_dai = {
|
||||
* .playback = { .ssi = &rsnd_ssi[0], },
|
||||
* };
|
||||
*
|
||||
* ex) uses SRC
|
||||
* static struct rsnd_src_platform_info rsnd_src[] = {
|
||||
* RSND_SCU(48000, 0),
|
||||
* ...
|
||||
* };
|
||||
* static struct rsnd_dai_platform_info rsnd_dai = {
|
||||
* .playback = { .ssi = &rsnd_ssi[0], .src = &rsnd_src[0] },
|
||||
* };
|
||||
*
|
||||
* ex) uses SRC bypass mode
|
||||
* static struct rsnd_src_platform_info rsnd_src[] = {
|
||||
* RSND_SCU(0, 0),
|
||||
* ...
|
||||
* };
|
||||
* static struct rsnd_dai_platform_info rsnd_dai = {
|
||||
* .playback = { .ssi = &rsnd_ssi[0], .src = &rsnd_src[0] },
|
||||
* };
|
||||
*
|
||||
*/
|
||||
|
||||
/*
|
||||
* Gen1/Gen2 common functions
|
||||
*/
|
||||
int rsnd_src_ssiu_start(struct rsnd_mod *ssi_mod,
|
||||
struct rsnd_dai *rdai,
|
||||
int use_busif)
|
||||
{
|
||||
struct rsnd_dai_stream *io = rsnd_mod_to_io(ssi_mod);
|
||||
struct snd_pcm_runtime *runtime = rsnd_io_to_runtime(io);
|
||||
int ssi_id = rsnd_mod_id(ssi_mod);
|
||||
|
||||
/*
|
||||
* SSI_MODE0
|
||||
*/
|
||||
rsnd_mod_bset(ssi_mod, SSI_MODE0, (1 << ssi_id),
|
||||
!use_busif << ssi_id);
|
||||
|
||||
/*
|
||||
* SSI_MODE1
|
||||
*/
|
||||
if (rsnd_ssi_is_pin_sharing(ssi_mod)) {
|
||||
int shift = -1;
|
||||
switch (ssi_id) {
|
||||
case 1:
|
||||
shift = 0;
|
||||
break;
|
||||
case 2:
|
||||
shift = 2;
|
||||
break;
|
||||
case 4:
|
||||
shift = 16;
|
||||
break;
|
||||
}
|
||||
|
||||
if (shift >= 0)
|
||||
rsnd_mod_bset(ssi_mod, SSI_MODE1,
|
||||
0x3 << shift,
|
||||
rsnd_dai_is_clk_master(rdai) ?
|
||||
0x2 << shift : 0x1 << shift);
|
||||
}
|
||||
|
||||
/*
|
||||
* DMA settings for SSIU
|
||||
*/
|
||||
if (use_busif) {
|
||||
u32 val = 0x76543210;
|
||||
u32 mask = ~0;
|
||||
|
||||
rsnd_mod_write(ssi_mod, SSI_BUSIF_ADINR,
|
||||
rsnd_get_adinr(ssi_mod));
|
||||
rsnd_mod_write(ssi_mod, SSI_BUSIF_MODE, 1);
|
||||
rsnd_mod_write(ssi_mod, SSI_CTRL, 0x1);
|
||||
|
||||
mask <<= runtime->channels * 4;
|
||||
val = val & mask;
|
||||
|
||||
switch (runtime->sample_bits) {
|
||||
case 16:
|
||||
val |= 0x67452301 & ~mask;
|
||||
break;
|
||||
case 32:
|
||||
val |= 0x76543210 & ~mask;
|
||||
break;
|
||||
}
|
||||
rsnd_mod_write(ssi_mod, BUSIF_DALIGN, val);
|
||||
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
int rsnd_src_ssiu_stop(struct rsnd_mod *ssi_mod,
|
||||
struct rsnd_dai *rdai,
|
||||
int use_busif)
|
||||
{
|
||||
/*
|
||||
* DMA settings for SSIU
|
||||
*/
|
||||
if (use_busif)
|
||||
rsnd_mod_write(ssi_mod, SSI_CTRL, 0);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
int rsnd_src_enable_ssi_irq(struct rsnd_mod *ssi_mod,
|
||||
struct rsnd_dai *rdai)
|
||||
{
|
||||
struct rsnd_priv *priv = rsnd_mod_to_priv(ssi_mod);
|
||||
|
||||
/* enable PIO interrupt if Gen2 */
|
||||
if (rsnd_is_gen2(priv))
|
||||
rsnd_mod_write(ssi_mod, INT_ENABLE, 0x0f000000);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
unsigned int rsnd_src_get_ssi_rate(struct rsnd_priv *priv,
|
||||
struct rsnd_dai_stream *io,
|
||||
struct snd_pcm_runtime *runtime)
|
||||
{
|
||||
struct rsnd_mod *src_mod = rsnd_io_to_mod_src(io);
|
||||
struct rsnd_src *src;
|
||||
unsigned int rate = 0;
|
||||
|
||||
if (src_mod) {
|
||||
src = rsnd_mod_to_src(src_mod);
|
||||
|
||||
/*
|
||||
* return convert rate if SRC is used,
|
||||
* otherwise, return runtime->rate as usual
|
||||
*/
|
||||
rate = rsnd_src_convert_rate(src);
|
||||
}
|
||||
|
||||
if (!rate)
|
||||
rate = runtime->rate;
|
||||
|
||||
return rate;
|
||||
}
|
||||
|
||||
static int rsnd_src_set_convert_rate(struct rsnd_mod *mod,
|
||||
struct rsnd_dai *rdai)
|
||||
{
|
||||
struct rsnd_dai_stream *io = rsnd_mod_to_io(mod);
|
||||
struct snd_pcm_runtime *runtime = rsnd_io_to_runtime(io);
|
||||
struct rsnd_src *src = rsnd_mod_to_src(mod);
|
||||
u32 convert_rate = rsnd_src_convert_rate(src);
|
||||
u32 fsrate = 0;
|
||||
|
||||
if (convert_rate)
|
||||
fsrate = 0x0400000 / convert_rate * runtime->rate;
|
||||
|
||||
/* set/clear soft reset */
|
||||
rsnd_mod_write(mod, SRC_SWRSR, 0);
|
||||
rsnd_mod_write(mod, SRC_SWRSR, 1);
|
||||
|
||||
/*
|
||||
* Initialize the operation of the SRC internal circuits
|
||||
* see rsnd_src_start()
|
||||
*/
|
||||
rsnd_mod_write(mod, SRC_SRCIR, 1);
|
||||
|
||||
/* Set channel number and output bit length */
|
||||
rsnd_mod_write(mod, SRC_ADINR, rsnd_get_adinr(mod));
|
||||
|
||||
/* Enable the initial value of IFS */
|
||||
if (fsrate) {
|
||||
rsnd_mod_write(mod, SRC_IFSCR, 1);
|
||||
|
||||
/* Set initial value of IFS */
|
||||
rsnd_mod_write(mod, SRC_IFSVR, fsrate);
|
||||
}
|
||||
|
||||
/* use DMA transfer */
|
||||
rsnd_mod_write(mod, SRC_BUSIF_MODE, 1);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int rsnd_src_init(struct rsnd_mod *mod,
|
||||
struct rsnd_dai *rdai)
|
||||
{
|
||||
struct rsnd_src *src = rsnd_mod_to_src(mod);
|
||||
|
||||
clk_prepare_enable(src->clk);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int rsnd_src_quit(struct rsnd_mod *mod,
|
||||
struct rsnd_dai *rdai)
|
||||
{
|
||||
struct rsnd_src *src = rsnd_mod_to_src(mod);
|
||||
|
||||
clk_disable_unprepare(src->clk);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int rsnd_src_start(struct rsnd_mod *mod,
|
||||
struct rsnd_dai *rdai)
|
||||
{
|
||||
struct rsnd_src *src = rsnd_mod_to_src(mod);
|
||||
|
||||
/*
|
||||
* Cancel the initialization and operate the SRC function
|
||||
* see rsnd_src_set_convert_rate()
|
||||
*/
|
||||
rsnd_mod_write(mod, SRC_SRCIR, 0);
|
||||
|
||||
if (rsnd_src_convert_rate(src))
|
||||
rsnd_mod_write(mod, SRC_ROUTE_MODE0, 1);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
|
||||
static int rsnd_src_stop(struct rsnd_mod *mod,
|
||||
struct rsnd_dai *rdai)
|
||||
{
|
||||
struct rsnd_src *src = rsnd_mod_to_src(mod);
|
||||
|
||||
if (rsnd_src_convert_rate(src))
|
||||
rsnd_mod_write(mod, SRC_ROUTE_MODE0, 0);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
/*
|
||||
* Gen1 functions
|
||||
*/
|
||||
static int rsnd_src_set_route_gen1(struct rsnd_mod *mod,
|
||||
struct rsnd_dai *rdai)
|
||||
{
|
||||
struct rsnd_dai_stream *io = rsnd_mod_to_io(mod);
|
||||
struct src_route_config {
|
||||
u32 mask;
|
||||
int shift;
|
||||
} routes[] = {
|
||||
{ 0xF, 0, }, /* 0 */
|
||||
{ 0xF, 4, }, /* 1 */
|
||||
{ 0xF, 8, }, /* 2 */
|
||||
{ 0x7, 12, }, /* 3 */
|
||||
{ 0x7, 16, }, /* 4 */
|
||||
{ 0x7, 20, }, /* 5 */
|
||||
{ 0x7, 24, }, /* 6 */
|
||||
{ 0x3, 28, }, /* 7 */
|
||||
{ 0x3, 30, }, /* 8 */
|
||||
};
|
||||
u32 mask;
|
||||
u32 val;
|
||||
int id;
|
||||
|
||||
id = rsnd_mod_id(mod);
|
||||
if (id < 0 || id >= ARRAY_SIZE(routes))
|
||||
return -EIO;
|
||||
|
||||
/*
|
||||
* SRC_ROUTE_SELECT
|
||||
*/
|
||||
val = rsnd_dai_is_play(rdai, io) ? 0x1 : 0x2;
|
||||
val = val << routes[id].shift;
|
||||
mask = routes[id].mask << routes[id].shift;
|
||||
|
||||
rsnd_mod_bset(mod, SRC_ROUTE_SEL, mask, val);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int rsnd_src_set_convert_timing_gen1(struct rsnd_mod *mod,
|
||||
struct rsnd_dai *rdai)
|
||||
{
|
||||
struct rsnd_dai_stream *io = rsnd_mod_to_io(mod);
|
||||
struct rsnd_priv *priv = rsnd_mod_to_priv(mod);
|
||||
struct rsnd_src *src = rsnd_mod_to_src(mod);
|
||||
struct snd_pcm_runtime *runtime = rsnd_io_to_runtime(io);
|
||||
u32 convert_rate = rsnd_src_convert_rate(src);
|
||||
u32 mask;
|
||||
u32 val;
|
||||
int shift;
|
||||
int id = rsnd_mod_id(mod);
|
||||
int ret;
|
||||
|
||||
/*
|
||||
* SRC_TIMING_SELECT
|
||||
*/
|
||||
shift = (id % 4) * 8;
|
||||
mask = 0x1F << shift;
|
||||
|
||||
/*
|
||||
* ADG is used as source clock if SRC was used,
|
||||
* then, SSI WS is used as destination clock.
|
||||
* SSI WS is used as source clock if SRC is not used
|
||||
* (when playback, source/destination become reverse when capture)
|
||||
*/
|
||||
ret = 0;
|
||||
if (convert_rate) {
|
||||
/* use ADG */
|
||||
val = 0;
|
||||
ret = rsnd_adg_set_convert_clk_gen1(priv, mod,
|
||||
runtime->rate,
|
||||
convert_rate);
|
||||
} else if (8 == id) {
|
||||
/* use SSI WS, but SRU8 is special */
|
||||
val = id << shift;
|
||||
} else {
|
||||
/* use SSI WS */
|
||||
val = (id + 1) << shift;
|
||||
}
|
||||
|
||||
if (ret < 0)
|
||||
return ret;
|
||||
|
||||
switch (id / 4) {
|
||||
case 0:
|
||||
rsnd_mod_bset(mod, SRC_TMG_SEL0, mask, val);
|
||||
break;
|
||||
case 1:
|
||||
rsnd_mod_bset(mod, SRC_TMG_SEL1, mask, val);
|
||||
break;
|
||||
case 2:
|
||||
rsnd_mod_bset(mod, SRC_TMG_SEL2, mask, val);
|
||||
break;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int rsnd_src_set_convert_rate_gen1(struct rsnd_mod *mod,
|
||||
struct rsnd_dai *rdai)
|
||||
{
|
||||
int ret;
|
||||
|
||||
ret = rsnd_src_set_convert_rate(mod, rdai);
|
||||
if (ret < 0)
|
||||
return ret;
|
||||
|
||||
/* Select SRC mode (fixed value) */
|
||||
rsnd_mod_write(mod, SRC_SRCCR, 0x00010110);
|
||||
|
||||
/* Set the restriction value of the FS ratio (98%) */
|
||||
rsnd_mod_write(mod, SRC_MNFSR,
|
||||
rsnd_mod_read(mod, SRC_IFSVR) / 100 * 98);
|
||||
|
||||
/* no SRC_BFSSR settings, since SRC_SRCCR::BUFMD is 0 */
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int rsnd_src_probe_gen1(struct rsnd_mod *mod,
|
||||
struct rsnd_dai *rdai)
|
||||
{
|
||||
struct rsnd_priv *priv = rsnd_mod_to_priv(mod);
|
||||
struct device *dev = rsnd_priv_to_dev(priv);
|
||||
|
||||
dev_dbg(dev, "%s (Gen1) is probed\n", rsnd_mod_name(mod));
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int rsnd_src_init_gen1(struct rsnd_mod *mod,
|
||||
struct rsnd_dai *rdai)
|
||||
{
|
||||
int ret;
|
||||
|
||||
ret = rsnd_src_init(mod, rdai);
|
||||
if (ret < 0)
|
||||
return ret;
|
||||
|
||||
ret = rsnd_src_set_route_gen1(mod, rdai);
|
||||
if (ret < 0)
|
||||
return ret;
|
||||
|
||||
ret = rsnd_src_set_convert_rate_gen1(mod, rdai);
|
||||
if (ret < 0)
|
||||
return ret;
|
||||
|
||||
ret = rsnd_src_set_convert_timing_gen1(mod, rdai);
|
||||
if (ret < 0)
|
||||
return ret;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int rsnd_src_start_gen1(struct rsnd_mod *mod,
|
||||
struct rsnd_dai *rdai)
|
||||
{
|
||||
int id = rsnd_mod_id(mod);
|
||||
|
||||
rsnd_mod_bset(mod, SRC_ROUTE_CTRL, (1 << id), (1 << id));
|
||||
|
||||
return rsnd_src_start(mod, rdai);
|
||||
}
|
||||
|
||||
static int rsnd_src_stop_gen1(struct rsnd_mod *mod,
|
||||
struct rsnd_dai *rdai)
|
||||
{
|
||||
int id = rsnd_mod_id(mod);
|
||||
|
||||
rsnd_mod_bset(mod, SRC_ROUTE_CTRL, (1 << id), 0);
|
||||
|
||||
return rsnd_src_stop(mod, rdai);
|
||||
}
|
||||
|
||||
static struct rsnd_mod_ops rsnd_src_gen1_ops = {
|
||||
.name = SRC_NAME,
|
||||
.probe = rsnd_src_probe_gen1,
|
||||
.init = rsnd_src_init_gen1,
|
||||
.quit = rsnd_src_quit,
|
||||
.start = rsnd_src_start_gen1,
|
||||
.stop = rsnd_src_stop_gen1,
|
||||
};
|
||||
|
||||
/*
|
||||
* Gen2 functions
|
||||
*/
|
||||
static int rsnd_src_set_convert_rate_gen2(struct rsnd_mod *mod,
|
||||
struct rsnd_dai *rdai)
|
||||
{
|
||||
struct rsnd_priv *priv = rsnd_mod_to_priv(mod);
|
||||
struct device *dev = rsnd_priv_to_dev(priv);
|
||||
struct rsnd_dai_stream *io = rsnd_mod_to_io(mod);
|
||||
struct snd_pcm_runtime *runtime = rsnd_io_to_runtime(io);
|
||||
struct rsnd_src *src = rsnd_mod_to_src(mod);
|
||||
uint ratio;
|
||||
int ret;
|
||||
|
||||
/* 6 - 1/6 are very enough ratio for SRC_BSDSR */
|
||||
if (!rsnd_src_convert_rate(src))
|
||||
ratio = 0;
|
||||
else if (rsnd_src_convert_rate(src) > runtime->rate)
|
||||
ratio = 100 * rsnd_src_convert_rate(src) / runtime->rate;
|
||||
else
|
||||
ratio = 100 * runtime->rate / rsnd_src_convert_rate(src);
|
||||
|
||||
if (ratio > 600) {
|
||||
dev_err(dev, "FSO/FSI ratio error\n");
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
ret = rsnd_src_set_convert_rate(mod, rdai);
|
||||
if (ret < 0)
|
||||
return ret;
|
||||
|
||||
rsnd_mod_write(mod, SRC_SRCCR, 0x00011110);
|
||||
|
||||
switch (rsnd_mod_id(mod)) {
|
||||
case 5:
|
||||
case 6:
|
||||
case 7:
|
||||
case 8:
|
||||
rsnd_mod_write(mod, SRC_BSDSR, 0x02400000);
|
||||
break;
|
||||
default:
|
||||
rsnd_mod_write(mod, SRC_BSDSR, 0x01800000);
|
||||
break;
|
||||
}
|
||||
|
||||
rsnd_mod_write(mod, SRC_BSISR, 0x00100060);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int rsnd_src_set_convert_timing_gen2(struct rsnd_mod *mod,
|
||||
struct rsnd_dai *rdai)
|
||||
{
|
||||
struct rsnd_dai_stream *io = rsnd_mod_to_io(mod);
|
||||
struct snd_pcm_runtime *runtime = rsnd_io_to_runtime(io);
|
||||
struct rsnd_src *src = rsnd_mod_to_src(mod);
|
||||
u32 convert_rate = rsnd_src_convert_rate(src);
|
||||
int ret;
|
||||
|
||||
if (convert_rate)
|
||||
ret = rsnd_adg_set_convert_clk_gen2(mod, rdai, io,
|
||||
runtime->rate,
|
||||
convert_rate);
|
||||
else
|
||||
ret = rsnd_adg_set_convert_timing_gen2(mod, rdai, io);
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int rsnd_src_probe_gen2(struct rsnd_mod *mod,
|
||||
struct rsnd_dai *rdai)
|
||||
{
|
||||
struct rsnd_priv *priv = rsnd_mod_to_priv(mod);
|
||||
struct rsnd_src *src = rsnd_mod_to_src(mod);
|
||||
struct device *dev = rsnd_priv_to_dev(priv);
|
||||
int ret;
|
||||
|
||||
ret = rsnd_dma_init(priv,
|
||||
rsnd_mod_to_dma(mod),
|
||||
rsnd_info_is_playback(priv, src),
|
||||
src->info->dma_id);
|
||||
if (ret < 0)
|
||||
dev_err(dev, "SRC DMA failed\n");
|
||||
|
||||
dev_dbg(dev, "%s (Gen2) is probed\n", rsnd_mod_name(mod));
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int rsnd_src_remove_gen2(struct rsnd_mod *mod,
|
||||
struct rsnd_dai *rdai)
|
||||
{
|
||||
rsnd_dma_quit(rsnd_mod_to_priv(mod), rsnd_mod_to_dma(mod));
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int rsnd_src_init_gen2(struct rsnd_mod *mod,
|
||||
struct rsnd_dai *rdai)
|
||||
{
|
||||
int ret;
|
||||
|
||||
ret = rsnd_src_init(mod, rdai);
|
||||
if (ret < 0)
|
||||
return ret;
|
||||
|
||||
ret = rsnd_src_set_convert_rate_gen2(mod, rdai);
|
||||
if (ret < 0)
|
||||
return ret;
|
||||
|
||||
ret = rsnd_src_set_convert_timing_gen2(mod, rdai);
|
||||
if (ret < 0)
|
||||
return ret;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int rsnd_src_start_gen2(struct rsnd_mod *mod,
|
||||
struct rsnd_dai *rdai)
|
||||
{
|
||||
struct rsnd_dai_stream *io = rsnd_mod_to_io(mod);
|
||||
struct rsnd_src *src = rsnd_mod_to_src(mod);
|
||||
u32 val = rsnd_io_to_mod_dvc(io) ? 0x01 : 0x11;
|
||||
|
||||
rsnd_dma_start(rsnd_mod_to_dma(&src->mod));
|
||||
|
||||
rsnd_mod_write(mod, SRC_CTRL, val);
|
||||
|
||||
return rsnd_src_start(mod, rdai);
|
||||
}
|
||||
|
||||
static int rsnd_src_stop_gen2(struct rsnd_mod *mod,
|
||||
struct rsnd_dai *rdai)
|
||||
{
|
||||
struct rsnd_src *src = rsnd_mod_to_src(mod);
|
||||
|
||||
rsnd_mod_write(mod, SRC_CTRL, 0);
|
||||
|
||||
rsnd_dma_stop(rsnd_mod_to_dma(&src->mod));
|
||||
|
||||
return rsnd_src_stop(mod, rdai);
|
||||
}
|
||||
|
||||
static struct rsnd_mod_ops rsnd_src_gen2_ops = {
|
||||
.name = SRC_NAME,
|
||||
.probe = rsnd_src_probe_gen2,
|
||||
.remove = rsnd_src_remove_gen2,
|
||||
.init = rsnd_src_init_gen2,
|
||||
.quit = rsnd_src_quit,
|
||||
.start = rsnd_src_start_gen2,
|
||||
.stop = rsnd_src_stop_gen2,
|
||||
};
|
||||
|
||||
struct rsnd_mod *rsnd_src_mod_get(struct rsnd_priv *priv, int id)
|
||||
{
|
||||
if (WARN_ON(id < 0 || id >= rsnd_src_nr(priv)))
|
||||
id = 0;
|
||||
|
||||
return &((struct rsnd_src *)(priv->src) + id)->mod;
|
||||
}
|
||||
|
||||
static void rsnd_of_parse_src(struct platform_device *pdev,
|
||||
const struct rsnd_of_data *of_data,
|
||||
struct rsnd_priv *priv)
|
||||
{
|
||||
struct device_node *src_node;
|
||||
struct rcar_snd_info *info = rsnd_priv_to_info(priv);
|
||||
struct rsnd_src_platform_info *src_info;
|
||||
struct device *dev = &pdev->dev;
|
||||
int nr;
|
||||
|
||||
if (!of_data)
|
||||
return;
|
||||
|
||||
src_node = of_get_child_by_name(dev->of_node, "rcar_sound,src");
|
||||
if (!src_node)
|
||||
return;
|
||||
|
||||
nr = of_get_child_count(src_node);
|
||||
if (!nr)
|
||||
goto rsnd_of_parse_src_end;
|
||||
|
||||
src_info = devm_kzalloc(dev,
|
||||
sizeof(struct rsnd_src_platform_info) * nr,
|
||||
GFP_KERNEL);
|
||||
if (!src_info) {
|
||||
dev_err(dev, "src info allocation error\n");
|
||||
goto rsnd_of_parse_src_end;
|
||||
}
|
||||
|
||||
info->src_info = src_info;
|
||||
info->src_info_nr = nr;
|
||||
|
||||
rsnd_of_parse_src_end:
|
||||
of_node_put(src_node);
|
||||
}
|
||||
|
||||
int rsnd_src_probe(struct platform_device *pdev,
|
||||
const struct rsnd_of_data *of_data,
|
||||
struct rsnd_priv *priv)
|
||||
{
|
||||
struct rcar_snd_info *info = rsnd_priv_to_info(priv);
|
||||
struct device *dev = rsnd_priv_to_dev(priv);
|
||||
struct rsnd_src *src;
|
||||
struct rsnd_mod_ops *ops;
|
||||
struct clk *clk;
|
||||
char name[RSND_SRC_NAME_SIZE];
|
||||
int i, nr;
|
||||
|
||||
ops = NULL;
|
||||
if (rsnd_is_gen1(priv))
|
||||
ops = &rsnd_src_gen1_ops;
|
||||
if (rsnd_is_gen2(priv))
|
||||
ops = &rsnd_src_gen2_ops;
|
||||
if (!ops) {
|
||||
dev_err(dev, "unknown Generation\n");
|
||||
return -EIO;
|
||||
}
|
||||
|
||||
rsnd_of_parse_src(pdev, of_data, priv);
|
||||
|
||||
/*
|
||||
* init SRC
|
||||
*/
|
||||
nr = info->src_info_nr;
|
||||
if (!nr)
|
||||
return 0;
|
||||
|
||||
src = devm_kzalloc(dev, sizeof(*src) * nr, GFP_KERNEL);
|
||||
if (!src) {
|
||||
dev_err(dev, "SRC allocate failed\n");
|
||||
return -ENOMEM;
|
||||
}
|
||||
|
||||
priv->src_nr = nr;
|
||||
priv->src = src;
|
||||
|
||||
for_each_rsnd_src(src, priv, i) {
|
||||
snprintf(name, RSND_SRC_NAME_SIZE, "%s.%d",
|
||||
SRC_NAME, i);
|
||||
|
||||
clk = devm_clk_get(dev, name);
|
||||
if (IS_ERR(clk))
|
||||
return PTR_ERR(clk);
|
||||
|
||||
src->info = &info->src_info[i];
|
||||
src->clk = clk;
|
||||
|
||||
rsnd_mod_init(priv, &src->mod, ops, RSND_MOD_SRC, i);
|
||||
|
||||
dev_dbg(dev, "SRC%d probed\n", i);
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
686
sound/soc/sh/rcar/ssi.c
Normal file
686
sound/soc/sh/rcar/ssi.c
Normal file
|
|
@ -0,0 +1,686 @@
|
|||
/*
|
||||
* Renesas R-Car SSIU/SSI support
|
||||
*
|
||||
* Copyright (C) 2013 Renesas Solutions Corp.
|
||||
* Kuninori Morimoto <kuninori.morimoto.gx@renesas.com>
|
||||
*
|
||||
* Based on fsi.c
|
||||
* Kuninori Morimoto <morimoto.kuninori@renesas.com>
|
||||
*
|
||||
* 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/delay.h>
|
||||
#include "rsnd.h"
|
||||
#define RSND_SSI_NAME_SIZE 16
|
||||
|
||||
/*
|
||||
* SSICR
|
||||
*/
|
||||
#define FORCE (1 << 31) /* Fixed */
|
||||
#define DMEN (1 << 28) /* DMA Enable */
|
||||
#define UIEN (1 << 27) /* Underflow Interrupt Enable */
|
||||
#define OIEN (1 << 26) /* Overflow Interrupt Enable */
|
||||
#define IIEN (1 << 25) /* Idle Mode Interrupt Enable */
|
||||
#define DIEN (1 << 24) /* Data Interrupt Enable */
|
||||
|
||||
#define DWL_8 (0 << 19) /* Data Word Length */
|
||||
#define DWL_16 (1 << 19) /* Data Word Length */
|
||||
#define DWL_18 (2 << 19) /* Data Word Length */
|
||||
#define DWL_20 (3 << 19) /* Data Word Length */
|
||||
#define DWL_22 (4 << 19) /* Data Word Length */
|
||||
#define DWL_24 (5 << 19) /* Data Word Length */
|
||||
#define DWL_32 (6 << 19) /* Data Word Length */
|
||||
|
||||
#define SWL_32 (3 << 16) /* R/W System Word Length */
|
||||
#define SCKD (1 << 15) /* Serial Bit Clock Direction */
|
||||
#define SWSD (1 << 14) /* Serial WS Direction */
|
||||
#define SCKP (1 << 13) /* Serial Bit Clock Polarity */
|
||||
#define SWSP (1 << 12) /* Serial WS Polarity */
|
||||
#define SDTA (1 << 10) /* Serial Data Alignment */
|
||||
#define DEL (1 << 8) /* Serial Data Delay */
|
||||
#define CKDV(v) (v << 4) /* Serial Clock Division Ratio */
|
||||
#define TRMD (1 << 1) /* Transmit/Receive Mode Select */
|
||||
#define EN (1 << 0) /* SSI Module Enable */
|
||||
|
||||
/*
|
||||
* SSISR
|
||||
*/
|
||||
#define UIRQ (1 << 27) /* Underflow Error Interrupt Status */
|
||||
#define OIRQ (1 << 26) /* Overflow Error Interrupt Status */
|
||||
#define IIRQ (1 << 25) /* Idle Mode Interrupt Status */
|
||||
#define DIRQ (1 << 24) /* Data Interrupt Status Flag */
|
||||
|
||||
/*
|
||||
* SSIWSR
|
||||
*/
|
||||
#define CONT (1 << 8) /* WS Continue Function */
|
||||
|
||||
#define SSI_NAME "ssi"
|
||||
|
||||
struct rsnd_ssi {
|
||||
struct clk *clk;
|
||||
struct rsnd_ssi_platform_info *info; /* rcar_snd.h */
|
||||
struct rsnd_ssi *parent;
|
||||
struct rsnd_mod mod;
|
||||
|
||||
struct rsnd_dai *rdai;
|
||||
u32 cr_own;
|
||||
u32 cr_clk;
|
||||
u32 cr_etc;
|
||||
int err;
|
||||
unsigned int usrcnt;
|
||||
unsigned int rate;
|
||||
};
|
||||
|
||||
#define for_each_rsnd_ssi(pos, priv, i) \
|
||||
for (i = 0; \
|
||||
(i < rsnd_ssi_nr(priv)) && \
|
||||
((pos) = ((struct rsnd_ssi *)(priv)->ssi + i)); \
|
||||
i++)
|
||||
|
||||
#define rsnd_ssi_nr(priv) ((priv)->ssi_nr)
|
||||
#define rsnd_mod_to_ssi(_mod) container_of((_mod), struct rsnd_ssi, mod)
|
||||
#define rsnd_dma_to_ssi(dma) rsnd_mod_to_ssi(rsnd_dma_to_mod(dma))
|
||||
#define rsnd_ssi_pio_available(ssi) ((ssi)->info->pio_irq > 0)
|
||||
#define rsnd_ssi_dma_available(ssi) \
|
||||
rsnd_dma_available(rsnd_mod_to_dma(&(ssi)->mod))
|
||||
#define rsnd_ssi_clk_from_parent(ssi) ((ssi)->parent)
|
||||
#define rsnd_ssi_mode_flags(p) ((p)->info->flags)
|
||||
#define rsnd_ssi_dai_id(ssi) ((ssi)->info->dai_id)
|
||||
|
||||
static int rsnd_ssi_use_busif(struct rsnd_mod *mod)
|
||||
{
|
||||
struct rsnd_ssi *ssi = rsnd_mod_to_ssi(mod);
|
||||
struct rsnd_dai_stream *io = rsnd_mod_to_io(mod);
|
||||
int use_busif = 0;
|
||||
|
||||
if (!(rsnd_ssi_mode_flags(ssi) & RSND_SSI_NO_BUSIF))
|
||||
use_busif = 1;
|
||||
if (rsnd_io_to_mod_src(io))
|
||||
use_busif = 1;
|
||||
|
||||
return use_busif;
|
||||
}
|
||||
|
||||
static void rsnd_ssi_status_check(struct rsnd_mod *mod,
|
||||
u32 bit)
|
||||
{
|
||||
struct rsnd_priv *priv = rsnd_mod_to_priv(mod);
|
||||
struct device *dev = rsnd_priv_to_dev(priv);
|
||||
u32 status;
|
||||
int i;
|
||||
|
||||
for (i = 0; i < 1024; i++) {
|
||||
status = rsnd_mod_read(mod, SSISR);
|
||||
if (status & bit)
|
||||
return;
|
||||
|
||||
udelay(50);
|
||||
}
|
||||
|
||||
dev_warn(dev, "status check failed\n");
|
||||
}
|
||||
|
||||
static int rsnd_ssi_master_clk_start(struct rsnd_ssi *ssi,
|
||||
struct rsnd_dai_stream *io)
|
||||
{
|
||||
struct rsnd_priv *priv = rsnd_mod_to_priv(&ssi->mod);
|
||||
struct snd_pcm_runtime *runtime = rsnd_io_to_runtime(io);
|
||||
struct device *dev = rsnd_priv_to_dev(priv);
|
||||
int i, j, ret;
|
||||
int adg_clk_div_table[] = {
|
||||
1, 6, /* see adg.c */
|
||||
};
|
||||
int ssi_clk_mul_table[] = {
|
||||
1, 2, 4, 8, 16, 6, 12,
|
||||
};
|
||||
unsigned int main_rate;
|
||||
unsigned int rate = rsnd_src_get_ssi_rate(priv, io, runtime);
|
||||
|
||||
/*
|
||||
* Find best clock, and try to start ADG
|
||||
*/
|
||||
for (i = 0; i < ARRAY_SIZE(adg_clk_div_table); i++) {
|
||||
for (j = 0; j < ARRAY_SIZE(ssi_clk_mul_table); j++) {
|
||||
|
||||
/*
|
||||
* this driver is assuming that
|
||||
* system word is 64fs (= 2 x 32bit)
|
||||
* see rsnd_ssi_init()
|
||||
*/
|
||||
main_rate = rate / adg_clk_div_table[i]
|
||||
* 32 * 2 * ssi_clk_mul_table[j];
|
||||
|
||||
ret = rsnd_adg_ssi_clk_try_start(&ssi->mod, main_rate);
|
||||
if (0 == ret) {
|
||||
ssi->rate = rate;
|
||||
ssi->cr_clk = FORCE | SWL_32 |
|
||||
SCKD | SWSD | CKDV(j);
|
||||
|
||||
dev_dbg(dev, "ssi%d outputs %u Hz\n",
|
||||
rsnd_mod_id(&ssi->mod), rate);
|
||||
|
||||
return 0;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
dev_err(dev, "unsupported clock rate\n");
|
||||
return -EIO;
|
||||
}
|
||||
|
||||
static void rsnd_ssi_master_clk_stop(struct rsnd_ssi *ssi)
|
||||
{
|
||||
ssi->rate = 0;
|
||||
ssi->cr_clk = 0;
|
||||
rsnd_adg_ssi_clk_stop(&ssi->mod);
|
||||
}
|
||||
|
||||
static void rsnd_ssi_hw_start(struct rsnd_ssi *ssi,
|
||||
struct rsnd_dai *rdai,
|
||||
struct rsnd_dai_stream *io)
|
||||
{
|
||||
struct rsnd_priv *priv = rsnd_mod_to_priv(&ssi->mod);
|
||||
struct device *dev = rsnd_priv_to_dev(priv);
|
||||
u32 cr;
|
||||
|
||||
if (0 == ssi->usrcnt) {
|
||||
clk_prepare_enable(ssi->clk);
|
||||
|
||||
if (rsnd_dai_is_clk_master(rdai)) {
|
||||
if (rsnd_ssi_clk_from_parent(ssi))
|
||||
rsnd_ssi_hw_start(ssi->parent, rdai, io);
|
||||
else
|
||||
rsnd_ssi_master_clk_start(ssi, io);
|
||||
}
|
||||
}
|
||||
|
||||
cr = ssi->cr_own |
|
||||
ssi->cr_clk |
|
||||
ssi->cr_etc |
|
||||
EN;
|
||||
|
||||
rsnd_mod_write(&ssi->mod, SSICR, cr);
|
||||
|
||||
ssi->usrcnt++;
|
||||
|
||||
dev_dbg(dev, "ssi%d hw started\n", rsnd_mod_id(&ssi->mod));
|
||||
}
|
||||
|
||||
static void rsnd_ssi_hw_stop(struct rsnd_ssi *ssi,
|
||||
struct rsnd_dai *rdai)
|
||||
{
|
||||
struct rsnd_priv *priv = rsnd_mod_to_priv(&ssi->mod);
|
||||
struct device *dev = rsnd_priv_to_dev(priv);
|
||||
u32 cr;
|
||||
|
||||
if (0 == ssi->usrcnt) /* stop might be called without start */
|
||||
return;
|
||||
|
||||
ssi->usrcnt--;
|
||||
|
||||
if (0 == ssi->usrcnt) {
|
||||
/*
|
||||
* disable all IRQ,
|
||||
* and, wait all data was sent
|
||||
*/
|
||||
cr = ssi->cr_own |
|
||||
ssi->cr_clk;
|
||||
|
||||
rsnd_mod_write(&ssi->mod, SSICR, cr | EN);
|
||||
rsnd_ssi_status_check(&ssi->mod, DIRQ);
|
||||
|
||||
/*
|
||||
* disable SSI,
|
||||
* and, wait idle state
|
||||
*/
|
||||
rsnd_mod_write(&ssi->mod, SSICR, cr); /* disabled all */
|
||||
rsnd_ssi_status_check(&ssi->mod, IIRQ);
|
||||
|
||||
if (rsnd_dai_is_clk_master(rdai)) {
|
||||
if (rsnd_ssi_clk_from_parent(ssi))
|
||||
rsnd_ssi_hw_stop(ssi->parent, rdai);
|
||||
else
|
||||
rsnd_ssi_master_clk_stop(ssi);
|
||||
}
|
||||
|
||||
clk_disable_unprepare(ssi->clk);
|
||||
}
|
||||
|
||||
dev_dbg(dev, "ssi%d hw stopped\n", rsnd_mod_id(&ssi->mod));
|
||||
}
|
||||
|
||||
/*
|
||||
* SSI mod common functions
|
||||
*/
|
||||
static int rsnd_ssi_init(struct rsnd_mod *mod,
|
||||
struct rsnd_dai *rdai)
|
||||
{
|
||||
struct rsnd_ssi *ssi = rsnd_mod_to_ssi(mod);
|
||||
struct rsnd_dai_stream *io = rsnd_mod_to_io(mod);
|
||||
struct snd_pcm_runtime *runtime = rsnd_io_to_runtime(io);
|
||||
u32 cr;
|
||||
|
||||
cr = FORCE;
|
||||
|
||||
/*
|
||||
* always use 32bit system word for easy clock calculation.
|
||||
* see also rsnd_ssi_master_clk_enable()
|
||||
*/
|
||||
cr |= SWL_32;
|
||||
|
||||
/*
|
||||
* init clock settings for SSICR
|
||||
*/
|
||||
switch (runtime->sample_bits) {
|
||||
case 16:
|
||||
cr |= DWL_16;
|
||||
break;
|
||||
case 32:
|
||||
cr |= DWL_24;
|
||||
break;
|
||||
default:
|
||||
return -EIO;
|
||||
}
|
||||
|
||||
if (rdai->bit_clk_inv)
|
||||
cr |= SCKP;
|
||||
if (rdai->frm_clk_inv)
|
||||
cr |= SWSP;
|
||||
if (rdai->data_alignment)
|
||||
cr |= SDTA;
|
||||
if (rdai->sys_delay)
|
||||
cr |= DEL;
|
||||
if (rsnd_dai_is_play(rdai, io))
|
||||
cr |= TRMD;
|
||||
|
||||
/*
|
||||
* set ssi parameter
|
||||
*/
|
||||
ssi->rdai = rdai;
|
||||
ssi->cr_own = cr;
|
||||
ssi->err = -1; /* ignore 1st error */
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int rsnd_ssi_quit(struct rsnd_mod *mod,
|
||||
struct rsnd_dai *rdai)
|
||||
{
|
||||
struct rsnd_ssi *ssi = rsnd_mod_to_ssi(mod);
|
||||
struct rsnd_priv *priv = rsnd_mod_to_priv(mod);
|
||||
struct device *dev = rsnd_priv_to_dev(priv);
|
||||
|
||||
if (ssi->err > 0)
|
||||
dev_warn(dev, "ssi under/over flow err = %d\n", ssi->err);
|
||||
|
||||
ssi->rdai = NULL;
|
||||
ssi->cr_own = 0;
|
||||
ssi->err = 0;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static void rsnd_ssi_record_error(struct rsnd_ssi *ssi, u32 status)
|
||||
{
|
||||
/* under/over flow error */
|
||||
if (status & (UIRQ | OIRQ)) {
|
||||
ssi->err++;
|
||||
|
||||
/* clear error status */
|
||||
rsnd_mod_write(&ssi->mod, SSISR, 0);
|
||||
}
|
||||
}
|
||||
|
||||
/*
|
||||
* SSI PIO
|
||||
*/
|
||||
static irqreturn_t rsnd_ssi_pio_interrupt(int irq, void *data)
|
||||
{
|
||||
struct rsnd_ssi *ssi = data;
|
||||
struct rsnd_mod *mod = &ssi->mod;
|
||||
struct rsnd_dai_stream *io = rsnd_mod_to_io(mod);
|
||||
u32 status = rsnd_mod_read(mod, SSISR);
|
||||
irqreturn_t ret = IRQ_NONE;
|
||||
|
||||
if (io && (status & DIRQ)) {
|
||||
struct rsnd_dai *rdai = ssi->rdai;
|
||||
struct snd_pcm_runtime *runtime = rsnd_io_to_runtime(io);
|
||||
u32 *buf = (u32 *)(runtime->dma_area +
|
||||
rsnd_dai_pointer_offset(io, 0));
|
||||
|
||||
rsnd_ssi_record_error(ssi, status);
|
||||
|
||||
/*
|
||||
* 8/16/32 data can be assesse to TDR/RDR register
|
||||
* directly as 32bit data
|
||||
* see rsnd_ssi_init()
|
||||
*/
|
||||
if (rsnd_dai_is_play(rdai, io))
|
||||
rsnd_mod_write(mod, SSITDR, *buf);
|
||||
else
|
||||
*buf = rsnd_mod_read(mod, SSIRDR);
|
||||
|
||||
rsnd_dai_pointer_update(io, sizeof(*buf));
|
||||
|
||||
ret = IRQ_HANDLED;
|
||||
}
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int rsnd_ssi_pio_probe(struct rsnd_mod *mod,
|
||||
struct rsnd_dai *rdai)
|
||||
{
|
||||
struct rsnd_priv *priv = rsnd_mod_to_priv(mod);
|
||||
struct device *dev = rsnd_priv_to_dev(priv);
|
||||
struct rsnd_ssi *ssi = rsnd_mod_to_ssi(mod);
|
||||
int irq = ssi->info->pio_irq;
|
||||
int ret;
|
||||
|
||||
ret = devm_request_irq(dev, irq,
|
||||
rsnd_ssi_pio_interrupt,
|
||||
IRQF_SHARED,
|
||||
dev_name(dev), ssi);
|
||||
if (ret)
|
||||
dev_err(dev, "SSI request interrupt failed\n");
|
||||
|
||||
dev_dbg(dev, "%s (PIO) is probed\n", rsnd_mod_name(mod));
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int rsnd_ssi_pio_start(struct rsnd_mod *mod,
|
||||
struct rsnd_dai *rdai)
|
||||
{
|
||||
struct rsnd_ssi *ssi = rsnd_mod_to_ssi(mod);
|
||||
struct rsnd_dai_stream *io = rsnd_mod_to_io(mod);
|
||||
|
||||
/* enable PIO IRQ */
|
||||
ssi->cr_etc = UIEN | OIEN | DIEN;
|
||||
|
||||
rsnd_src_ssiu_start(mod, rdai, 0);
|
||||
|
||||
rsnd_src_enable_ssi_irq(mod, rdai);
|
||||
|
||||
rsnd_ssi_hw_start(ssi, rdai, io);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int rsnd_ssi_pio_stop(struct rsnd_mod *mod,
|
||||
struct rsnd_dai *rdai)
|
||||
{
|
||||
struct rsnd_ssi *ssi = rsnd_mod_to_ssi(mod);
|
||||
|
||||
ssi->cr_etc = 0;
|
||||
|
||||
rsnd_ssi_hw_stop(ssi, rdai);
|
||||
|
||||
rsnd_src_ssiu_stop(mod, rdai, 0);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static struct rsnd_mod_ops rsnd_ssi_pio_ops = {
|
||||
.name = SSI_NAME,
|
||||
.probe = rsnd_ssi_pio_probe,
|
||||
.init = rsnd_ssi_init,
|
||||
.quit = rsnd_ssi_quit,
|
||||
.start = rsnd_ssi_pio_start,
|
||||
.stop = rsnd_ssi_pio_stop,
|
||||
};
|
||||
|
||||
static int rsnd_ssi_dma_probe(struct rsnd_mod *mod,
|
||||
struct rsnd_dai *rdai)
|
||||
{
|
||||
struct rsnd_priv *priv = rsnd_mod_to_priv(mod);
|
||||
struct rsnd_ssi *ssi = rsnd_mod_to_ssi(mod);
|
||||
struct device *dev = rsnd_priv_to_dev(priv);
|
||||
int dma_id = ssi->info->dma_id;
|
||||
int ret;
|
||||
|
||||
ret = rsnd_dma_init(
|
||||
priv, rsnd_mod_to_dma(mod),
|
||||
rsnd_info_is_playback(priv, ssi),
|
||||
dma_id);
|
||||
|
||||
if (ret < 0)
|
||||
dev_err(dev, "SSI DMA failed\n");
|
||||
|
||||
dev_dbg(dev, "%s (DMA) is probed\n", rsnd_mod_name(mod));
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int rsnd_ssi_dma_remove(struct rsnd_mod *mod,
|
||||
struct rsnd_dai *rdai)
|
||||
{
|
||||
rsnd_dma_quit(rsnd_mod_to_priv(mod), rsnd_mod_to_dma(mod));
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int rsnd_ssi_dma_start(struct rsnd_mod *mod,
|
||||
struct rsnd_dai *rdai)
|
||||
{
|
||||
struct rsnd_ssi *ssi = rsnd_mod_to_ssi(mod);
|
||||
struct rsnd_dma *dma = rsnd_mod_to_dma(&ssi->mod);
|
||||
struct rsnd_dai_stream *io = rsnd_mod_to_io(mod);
|
||||
|
||||
/* enable DMA transfer */
|
||||
ssi->cr_etc = DMEN;
|
||||
|
||||
rsnd_src_ssiu_start(mod, rdai, rsnd_ssi_use_busif(mod));
|
||||
|
||||
rsnd_dma_start(dma);
|
||||
|
||||
rsnd_ssi_hw_start(ssi, ssi->rdai, io);
|
||||
|
||||
/* enable WS continue */
|
||||
if (rsnd_dai_is_clk_master(rdai))
|
||||
rsnd_mod_write(&ssi->mod, SSIWSR, CONT);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int rsnd_ssi_dma_stop(struct rsnd_mod *mod,
|
||||
struct rsnd_dai *rdai)
|
||||
{
|
||||
struct rsnd_ssi *ssi = rsnd_mod_to_ssi(mod);
|
||||
struct rsnd_dma *dma = rsnd_mod_to_dma(&ssi->mod);
|
||||
|
||||
ssi->cr_etc = 0;
|
||||
|
||||
rsnd_ssi_record_error(ssi, rsnd_mod_read(mod, SSISR));
|
||||
|
||||
rsnd_ssi_hw_stop(ssi, rdai);
|
||||
|
||||
rsnd_dma_stop(dma);
|
||||
|
||||
rsnd_src_ssiu_stop(mod, rdai, 1);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static char *rsnd_ssi_dma_name(struct rsnd_mod *mod)
|
||||
{
|
||||
return rsnd_ssi_use_busif(mod) ? "ssiu" : SSI_NAME;
|
||||
}
|
||||
|
||||
static struct rsnd_mod_ops rsnd_ssi_dma_ops = {
|
||||
.name = SSI_NAME,
|
||||
.dma_name = rsnd_ssi_dma_name,
|
||||
.probe = rsnd_ssi_dma_probe,
|
||||
.remove = rsnd_ssi_dma_remove,
|
||||
.init = rsnd_ssi_init,
|
||||
.quit = rsnd_ssi_quit,
|
||||
.start = rsnd_ssi_dma_start,
|
||||
.stop = rsnd_ssi_dma_stop,
|
||||
};
|
||||
|
||||
/*
|
||||
* Non SSI
|
||||
*/
|
||||
static struct rsnd_mod_ops rsnd_ssi_non_ops = {
|
||||
.name = SSI_NAME,
|
||||
};
|
||||
|
||||
/*
|
||||
* ssi mod function
|
||||
*/
|
||||
struct rsnd_mod *rsnd_ssi_mod_get(struct rsnd_priv *priv, int id)
|
||||
{
|
||||
if (WARN_ON(id < 0 || id >= rsnd_ssi_nr(priv)))
|
||||
id = 0;
|
||||
|
||||
return &((struct rsnd_ssi *)(priv->ssi) + id)->mod;
|
||||
}
|
||||
|
||||
int rsnd_ssi_is_pin_sharing(struct rsnd_mod *mod)
|
||||
{
|
||||
struct rsnd_ssi *ssi = rsnd_mod_to_ssi(mod);
|
||||
|
||||
return !!(rsnd_ssi_mode_flags(ssi) & RSND_SSI_CLK_PIN_SHARE);
|
||||
}
|
||||
|
||||
static void rsnd_ssi_parent_clk_setup(struct rsnd_priv *priv, struct rsnd_ssi *ssi)
|
||||
{
|
||||
if (!rsnd_ssi_is_pin_sharing(&ssi->mod))
|
||||
return;
|
||||
|
||||
switch (rsnd_mod_id(&ssi->mod)) {
|
||||
case 1:
|
||||
case 2:
|
||||
ssi->parent = rsnd_mod_to_ssi(rsnd_ssi_mod_get(priv, 0));
|
||||
break;
|
||||
case 4:
|
||||
ssi->parent = rsnd_mod_to_ssi(rsnd_ssi_mod_get(priv, 3));
|
||||
break;
|
||||
case 8:
|
||||
ssi->parent = rsnd_mod_to_ssi(rsnd_ssi_mod_get(priv, 7));
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
static void rsnd_of_parse_ssi(struct platform_device *pdev,
|
||||
const struct rsnd_of_data *of_data,
|
||||
struct rsnd_priv *priv)
|
||||
{
|
||||
struct device_node *node;
|
||||
struct device_node *np;
|
||||
struct rsnd_ssi_platform_info *ssi_info;
|
||||
struct rcar_snd_info *info = rsnd_priv_to_info(priv);
|
||||
struct device *dev = &pdev->dev;
|
||||
int nr, i;
|
||||
|
||||
if (!of_data)
|
||||
return;
|
||||
|
||||
node = of_get_child_by_name(dev->of_node, "rcar_sound,ssi");
|
||||
if (!node)
|
||||
return;
|
||||
|
||||
nr = of_get_child_count(node);
|
||||
if (!nr)
|
||||
goto rsnd_of_parse_ssi_end;
|
||||
|
||||
ssi_info = devm_kzalloc(dev,
|
||||
sizeof(struct rsnd_ssi_platform_info) * nr,
|
||||
GFP_KERNEL);
|
||||
if (!ssi_info) {
|
||||
dev_err(dev, "ssi info allocation error\n");
|
||||
goto rsnd_of_parse_ssi_end;
|
||||
}
|
||||
|
||||
info->ssi_info = ssi_info;
|
||||
info->ssi_info_nr = nr;
|
||||
|
||||
i = -1;
|
||||
for_each_child_of_node(node, np) {
|
||||
i++;
|
||||
|
||||
ssi_info = info->ssi_info + i;
|
||||
|
||||
/*
|
||||
* pin settings
|
||||
*/
|
||||
if (of_get_property(np, "shared-pin", NULL))
|
||||
ssi_info->flags |= RSND_SSI_CLK_PIN_SHARE;
|
||||
|
||||
/*
|
||||
* irq
|
||||
*/
|
||||
ssi_info->pio_irq = irq_of_parse_and_map(np, 0);
|
||||
|
||||
/*
|
||||
* DMA
|
||||
*/
|
||||
ssi_info->dma_id = of_get_property(np, "pio-transfer", NULL) ?
|
||||
0 : 1;
|
||||
|
||||
if (of_get_property(np, "no-busif", NULL))
|
||||
ssi_info->flags |= RSND_SSI_NO_BUSIF;
|
||||
}
|
||||
|
||||
rsnd_of_parse_ssi_end:
|
||||
of_node_put(node);
|
||||
}
|
||||
|
||||
int rsnd_ssi_probe(struct platform_device *pdev,
|
||||
const struct rsnd_of_data *of_data,
|
||||
struct rsnd_priv *priv)
|
||||
{
|
||||
struct rcar_snd_info *info = rsnd_priv_to_info(priv);
|
||||
struct rsnd_ssi_platform_info *pinfo;
|
||||
struct device *dev = rsnd_priv_to_dev(priv);
|
||||
struct rsnd_mod_ops *ops;
|
||||
struct clk *clk;
|
||||
struct rsnd_ssi *ssi;
|
||||
char name[RSND_SSI_NAME_SIZE];
|
||||
int i, nr;
|
||||
|
||||
rsnd_of_parse_ssi(pdev, of_data, priv);
|
||||
|
||||
/*
|
||||
* init SSI
|
||||
*/
|
||||
nr = info->ssi_info_nr;
|
||||
ssi = devm_kzalloc(dev, sizeof(*ssi) * nr, GFP_KERNEL);
|
||||
if (!ssi) {
|
||||
dev_err(dev, "SSI allocate failed\n");
|
||||
return -ENOMEM;
|
||||
}
|
||||
|
||||
priv->ssi = ssi;
|
||||
priv->ssi_nr = nr;
|
||||
|
||||
for_each_rsnd_ssi(ssi, priv, i) {
|
||||
pinfo = &info->ssi_info[i];
|
||||
|
||||
snprintf(name, RSND_SSI_NAME_SIZE, "%s.%d",
|
||||
SSI_NAME, i);
|
||||
|
||||
clk = devm_clk_get(dev, name);
|
||||
if (IS_ERR(clk))
|
||||
return PTR_ERR(clk);
|
||||
|
||||
ssi->info = pinfo;
|
||||
ssi->clk = clk;
|
||||
|
||||
ops = &rsnd_ssi_non_ops;
|
||||
if (pinfo->dma_id > 0)
|
||||
ops = &rsnd_ssi_dma_ops;
|
||||
else if (rsnd_ssi_pio_available(ssi))
|
||||
ops = &rsnd_ssi_pio_ops;
|
||||
|
||||
rsnd_mod_init(priv, &ssi->mod, ops, RSND_MOD_SSI, i);
|
||||
|
||||
rsnd_ssi_parent_clk_setup(priv, ssi);
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
73
sound/soc/sh/sh7760-ac97.c
Normal file
73
sound/soc/sh/sh7760-ac97.c
Normal file
|
|
@ -0,0 +1,73 @@
|
|||
/*
|
||||
* Generic AC97 sound support for SH7760
|
||||
*
|
||||
* (c) 2007 Manuel Lauss
|
||||
*
|
||||
* Licensed under the GPLv2.
|
||||
*/
|
||||
|
||||
#include <linux/module.h>
|
||||
#include <linux/moduleparam.h>
|
||||
#include <linux/platform_device.h>
|
||||
#include <sound/core.h>
|
||||
#include <sound/pcm.h>
|
||||
#include <sound/soc.h>
|
||||
#include <asm/io.h>
|
||||
|
||||
#define IPSEL 0xFE400034
|
||||
|
||||
static struct snd_soc_dai_link sh7760_ac97_dai = {
|
||||
.name = "AC97",
|
||||
.stream_name = "AC97 HiFi",
|
||||
.cpu_dai_name = "hac-dai.0", /* HAC0 */
|
||||
.codec_dai_name = "ac97-hifi",
|
||||
.platform_name = "sh7760-pcm-audio",
|
||||
.codec_name = "ac97-codec",
|
||||
.ops = NULL,
|
||||
};
|
||||
|
||||
static struct snd_soc_card sh7760_ac97_soc_machine = {
|
||||
.name = "SH7760 AC97",
|
||||
.owner = THIS_MODULE,
|
||||
.dai_link = &sh7760_ac97_dai,
|
||||
.num_links = 1,
|
||||
};
|
||||
|
||||
static struct platform_device *sh7760_ac97_snd_device;
|
||||
|
||||
static int __init sh7760_ac97_init(void)
|
||||
{
|
||||
int ret;
|
||||
unsigned short ipsel;
|
||||
|
||||
/* enable both AC97 controllers in pinmux reg */
|
||||
ipsel = __raw_readw(IPSEL);
|
||||
__raw_writew(ipsel | (3 << 10), IPSEL);
|
||||
|
||||
ret = -ENOMEM;
|
||||
sh7760_ac97_snd_device = platform_device_alloc("soc-audio", -1);
|
||||
if (!sh7760_ac97_snd_device)
|
||||
goto out;
|
||||
|
||||
platform_set_drvdata(sh7760_ac97_snd_device,
|
||||
&sh7760_ac97_soc_machine);
|
||||
ret = platform_device_add(sh7760_ac97_snd_device);
|
||||
|
||||
if (ret)
|
||||
platform_device_put(sh7760_ac97_snd_device);
|
||||
|
||||
out:
|
||||
return ret;
|
||||
}
|
||||
|
||||
static void __exit sh7760_ac97_exit(void)
|
||||
{
|
||||
platform_device_unregister(sh7760_ac97_snd_device);
|
||||
}
|
||||
|
||||
module_init(sh7760_ac97_init);
|
||||
module_exit(sh7760_ac97_exit);
|
||||
|
||||
MODULE_LICENSE("GPL");
|
||||
MODULE_DESCRIPTION("Generic SH7760 AC97 sound machine");
|
||||
MODULE_AUTHOR("Manuel Lauss <mano@roarinelk.homelinux.net>");
|
||||
194
sound/soc/sh/siu.h
Normal file
194
sound/soc/sh/siu.h
Normal file
|
|
@ -0,0 +1,194 @@
|
|||
/*
|
||||
* siu.h - ALSA SoC driver for Renesas SH7343, SH7722 SIU peripheral.
|
||||
*
|
||||
* Copyright (C) 2009-2010 Guennadi Liakhovetski <g.liakhovetski@gmx.de>
|
||||
* Copyright (C) 2006 Carlos Munoz <carlos@kenati.com>
|
||||
*
|
||||
* This program is free software; you can redistribute it and/or modify
|
||||
* it under the terms of the GNU General Public License as published by
|
||||
* the Free Software Foundation; either version 2 of the License, or
|
||||
* (at your option) any later version.
|
||||
*
|
||||
* This program is distributed in the hope that it will be useful,
|
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
* GNU General Public License for more details.
|
||||
*
|
||||
* You should have received a copy of the GNU General Public License
|
||||
* along with this program; if not, write to the Free Software
|
||||
* Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
||||
*/
|
||||
|
||||
#ifndef SIU_H
|
||||
#define SIU_H
|
||||
|
||||
/* Common kernel and user-space firmware-building defines and types */
|
||||
|
||||
#define YRAM0_SIZE (0x0040 / 4) /* 16 */
|
||||
#define YRAM1_SIZE (0x0080 / 4) /* 32 */
|
||||
#define YRAM2_SIZE (0x0040 / 4) /* 16 */
|
||||
#define YRAM3_SIZE (0x0080 / 4) /* 32 */
|
||||
#define YRAM4_SIZE (0x0080 / 4) /* 32 */
|
||||
#define YRAM_DEF_SIZE (YRAM0_SIZE + YRAM1_SIZE + YRAM2_SIZE + \
|
||||
YRAM3_SIZE + YRAM4_SIZE)
|
||||
#define YRAM_FIR_SIZE (0x0400 / 4) /* 256 */
|
||||
#define YRAM_IIR_SIZE (0x0200 / 4) /* 128 */
|
||||
|
||||
#define XRAM0_SIZE (0x0400 / 4) /* 256 */
|
||||
#define XRAM1_SIZE (0x0200 / 4) /* 128 */
|
||||
#define XRAM2_SIZE (0x0200 / 4) /* 128 */
|
||||
|
||||
/* PRAM program array size */
|
||||
#define PRAM0_SIZE (0x0100 / 4) /* 64 */
|
||||
#define PRAM1_SIZE ((0x2000 - 0x0100) / 4) /* 1984 */
|
||||
|
||||
#include <linux/types.h>
|
||||
|
||||
struct siu_spb_param {
|
||||
__u32 ab1a; /* input FIFO address */
|
||||
__u32 ab0a; /* output FIFO address */
|
||||
__u32 dir; /* 0=the ather except CPUOUTPUT, 1=CPUINPUT */
|
||||
__u32 event; /* SPB program starting conditions */
|
||||
__u32 stfifo; /* STFIFO register setting value */
|
||||
__u32 trdat; /* TRDAT register setting value */
|
||||
};
|
||||
|
||||
struct siu_firmware {
|
||||
__u32 yram_fir_coeff[YRAM_FIR_SIZE];
|
||||
__u32 pram0[PRAM0_SIZE];
|
||||
__u32 pram1[PRAM1_SIZE];
|
||||
__u32 yram0[YRAM0_SIZE];
|
||||
__u32 yram1[YRAM1_SIZE];
|
||||
__u32 yram2[YRAM2_SIZE];
|
||||
__u32 yram3[YRAM3_SIZE];
|
||||
__u32 yram4[YRAM4_SIZE];
|
||||
__u32 spbpar_num;
|
||||
struct siu_spb_param spbpar[32];
|
||||
};
|
||||
|
||||
#ifdef __KERNEL__
|
||||
|
||||
#include <linux/dmaengine.h>
|
||||
#include <linux/interrupt.h>
|
||||
#include <linux/io.h>
|
||||
#include <linux/sh_dma.h>
|
||||
|
||||
#include <sound/core.h>
|
||||
#include <sound/pcm.h>
|
||||
#include <sound/soc.h>
|
||||
|
||||
#define SIU_PERIOD_BYTES_MAX 8192 /* DMA transfer/period size */
|
||||
#define SIU_PERIOD_BYTES_MIN 256 /* DMA transfer/period size */
|
||||
#define SIU_PERIODS_MAX 64 /* Max periods in buffer */
|
||||
#define SIU_PERIODS_MIN 4 /* Min periods in buffer */
|
||||
#define SIU_BUFFER_BYTES_MAX (SIU_PERIOD_BYTES_MAX * SIU_PERIODS_MAX)
|
||||
|
||||
/* SIU ports: only one can be used at a time */
|
||||
enum {
|
||||
SIU_PORT_A,
|
||||
SIU_PORT_B,
|
||||
SIU_PORT_NUM,
|
||||
};
|
||||
|
||||
/* SIU clock configuration */
|
||||
enum {
|
||||
SIU_CLKA_PLL,
|
||||
SIU_CLKA_EXT,
|
||||
SIU_CLKB_PLL,
|
||||
SIU_CLKB_EXT
|
||||
};
|
||||
|
||||
struct device;
|
||||
struct siu_info {
|
||||
struct device *dev;
|
||||
int port_id;
|
||||
u32 __iomem *pram;
|
||||
u32 __iomem *xram;
|
||||
u32 __iomem *yram;
|
||||
u32 __iomem *reg;
|
||||
struct siu_firmware fw;
|
||||
};
|
||||
|
||||
struct siu_stream {
|
||||
struct tasklet_struct tasklet;
|
||||
struct snd_pcm_substream *substream;
|
||||
snd_pcm_format_t format;
|
||||
size_t buf_bytes;
|
||||
size_t period_bytes;
|
||||
int cur_period; /* Period currently in dma */
|
||||
u32 volume;
|
||||
snd_pcm_sframes_t xfer_cnt; /* Number of frames */
|
||||
u8 rw_flg; /* transfer status */
|
||||
/* DMA status */
|
||||
struct dma_chan *chan; /* DMA channel */
|
||||
struct dma_async_tx_descriptor *tx_desc;
|
||||
dma_cookie_t cookie;
|
||||
struct sh_dmae_slave param;
|
||||
};
|
||||
|
||||
struct siu_port {
|
||||
unsigned long play_cap; /* Used to track full duplex */
|
||||
struct snd_pcm *pcm;
|
||||
struct siu_stream playback;
|
||||
struct siu_stream capture;
|
||||
u32 stfifo; /* STFIFO value from firmware */
|
||||
u32 trdat; /* TRDAT value from firmware */
|
||||
};
|
||||
|
||||
extern struct siu_port *siu_ports[SIU_PORT_NUM];
|
||||
|
||||
static inline struct siu_port *siu_port_info(struct snd_pcm_substream *substream)
|
||||
{
|
||||
struct platform_device *pdev =
|
||||
to_platform_device(substream->pcm->card->dev);
|
||||
return siu_ports[pdev->id];
|
||||
}
|
||||
|
||||
/* Register access */
|
||||
static inline void siu_write32(u32 __iomem *addr, u32 val)
|
||||
{
|
||||
__raw_writel(val, addr);
|
||||
}
|
||||
|
||||
static inline u32 siu_read32(u32 __iomem *addr)
|
||||
{
|
||||
return __raw_readl(addr);
|
||||
}
|
||||
|
||||
/* SIU registers */
|
||||
#define SIU_IFCTL (0x000 / sizeof(u32))
|
||||
#define SIU_SRCTL (0x004 / sizeof(u32))
|
||||
#define SIU_SFORM (0x008 / sizeof(u32))
|
||||
#define SIU_CKCTL (0x00c / sizeof(u32))
|
||||
#define SIU_TRDAT (0x010 / sizeof(u32))
|
||||
#define SIU_STFIFO (0x014 / sizeof(u32))
|
||||
#define SIU_DPAK (0x01c / sizeof(u32))
|
||||
#define SIU_CKREV (0x020 / sizeof(u32))
|
||||
#define SIU_EVNTC (0x028 / sizeof(u32))
|
||||
#define SIU_SBCTL (0x040 / sizeof(u32))
|
||||
#define SIU_SBPSET (0x044 / sizeof(u32))
|
||||
#define SIU_SBFSTS (0x068 / sizeof(u32))
|
||||
#define SIU_SBDVCA (0x06c / sizeof(u32))
|
||||
#define SIU_SBDVCB (0x070 / sizeof(u32))
|
||||
#define SIU_SBACTIV (0x074 / sizeof(u32))
|
||||
#define SIU_DMAIA (0x090 / sizeof(u32))
|
||||
#define SIU_DMAIB (0x094 / sizeof(u32))
|
||||
#define SIU_DMAOA (0x098 / sizeof(u32))
|
||||
#define SIU_DMAOB (0x09c / sizeof(u32))
|
||||
#define SIU_DMAML (0x0a0 / sizeof(u32))
|
||||
#define SIU_SPSTS (0x0cc / sizeof(u32))
|
||||
#define SIU_SPCTL (0x0d0 / sizeof(u32))
|
||||
#define SIU_BRGASEL (0x100 / sizeof(u32))
|
||||
#define SIU_BRRA (0x104 / sizeof(u32))
|
||||
#define SIU_BRGBSEL (0x108 / sizeof(u32))
|
||||
#define SIU_BRRB (0x10c / sizeof(u32))
|
||||
|
||||
extern struct snd_soc_platform_driver siu_platform;
|
||||
extern struct siu_info *siu_i2s_data;
|
||||
|
||||
int siu_init_port(int port, struct siu_port **port_info, struct snd_card *card);
|
||||
void siu_free_port(struct siu_port *port_info);
|
||||
|
||||
#endif
|
||||
|
||||
#endif /* SIU_H */
|
||||
859
sound/soc/sh/siu_dai.c
Normal file
859
sound/soc/sh/siu_dai.c
Normal file
|
|
@ -0,0 +1,859 @@
|
|||
/*
|
||||
* siu_dai.c - ALSA SoC driver for Renesas SH7343, SH7722 SIU peripheral.
|
||||
*
|
||||
* Copyright (C) 2009-2010 Guennadi Liakhovetski <g.liakhovetski@gmx.de>
|
||||
* Copyright (C) 2006 Carlos Munoz <carlos@kenati.com>
|
||||
*
|
||||
* This program is free software; you can redistribute it and/or modify
|
||||
* it under the terms of the GNU General Public License as published by
|
||||
* the Free Software Foundation; either version 2 of the License, or
|
||||
* (at your option) any later version.
|
||||
*
|
||||
* This program is distributed in the hope that it will be useful,
|
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
* GNU General Public License for more details.
|
||||
*
|
||||
* You should have received a copy of the GNU General Public License
|
||||
* along with this program; if not, write to the Free Software
|
||||
* Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
||||
*/
|
||||
|
||||
#include <linux/delay.h>
|
||||
#include <linux/firmware.h>
|
||||
#include <linux/pm_runtime.h>
|
||||
#include <linux/slab.h>
|
||||
#include <linux/module.h>
|
||||
|
||||
#include <asm/clock.h>
|
||||
#include <asm/siu.h>
|
||||
|
||||
#include <sound/control.h>
|
||||
#include <sound/soc.h>
|
||||
|
||||
#include "siu.h"
|
||||
|
||||
/* Board specifics */
|
||||
#if defined(CONFIG_CPU_SUBTYPE_SH7722)
|
||||
# define SIU_MAX_VOLUME 0x1000
|
||||
#else
|
||||
# define SIU_MAX_VOLUME 0x7fff
|
||||
#endif
|
||||
|
||||
#define PRAM_SIZE 0x2000
|
||||
#define XRAM_SIZE 0x800
|
||||
#define YRAM_SIZE 0x800
|
||||
|
||||
#define XRAM_OFFSET 0x4000
|
||||
#define YRAM_OFFSET 0x6000
|
||||
#define REG_OFFSET 0xc000
|
||||
|
||||
#define PLAYBACK_ENABLED 1
|
||||
#define CAPTURE_ENABLED 2
|
||||
|
||||
#define VOLUME_CAPTURE 0
|
||||
#define VOLUME_PLAYBACK 1
|
||||
#define DFLT_VOLUME_LEVEL 0x08000800
|
||||
|
||||
/*
|
||||
* SPDIF is only available on port A and on some SIU implementations it is only
|
||||
* available for input. Due to the lack of hardware to test it, SPDIF is left
|
||||
* disabled in this driver version
|
||||
*/
|
||||
struct format_flag {
|
||||
u32 i2s;
|
||||
u32 pcm;
|
||||
u32 spdif;
|
||||
u32 mask;
|
||||
};
|
||||
|
||||
struct port_flag {
|
||||
struct format_flag playback;
|
||||
struct format_flag capture;
|
||||
};
|
||||
|
||||
struct siu_info *siu_i2s_data;
|
||||
|
||||
static struct port_flag siu_flags[SIU_PORT_NUM] = {
|
||||
[SIU_PORT_A] = {
|
||||
.playback = {
|
||||
.i2s = 0x50000000,
|
||||
.pcm = 0x40000000,
|
||||
.spdif = 0x80000000, /* not on all SIU versions */
|
||||
.mask = 0xd0000000,
|
||||
},
|
||||
.capture = {
|
||||
.i2s = 0x05000000,
|
||||
.pcm = 0x04000000,
|
||||
.spdif = 0x08000000,
|
||||
.mask = 0x0d000000,
|
||||
},
|
||||
},
|
||||
[SIU_PORT_B] = {
|
||||
.playback = {
|
||||
.i2s = 0x00500000,
|
||||
.pcm = 0x00400000,
|
||||
.spdif = 0, /* impossible - turn off */
|
||||
.mask = 0x00500000,
|
||||
},
|
||||
.capture = {
|
||||
.i2s = 0x00050000,
|
||||
.pcm = 0x00040000,
|
||||
.spdif = 0, /* impossible - turn off */
|
||||
.mask = 0x00050000,
|
||||
},
|
||||
},
|
||||
};
|
||||
|
||||
static void siu_dai_start(struct siu_port *port_info)
|
||||
{
|
||||
struct siu_info *info = siu_i2s_data;
|
||||
u32 __iomem *base = info->reg;
|
||||
|
||||
dev_dbg(port_info->pcm->card->dev, "%s\n", __func__);
|
||||
|
||||
/* Issue software reset to siu */
|
||||
siu_write32(base + SIU_SRCTL, 0);
|
||||
|
||||
/* Wait for the reset to take effect */
|
||||
udelay(1);
|
||||
|
||||
port_info->stfifo = 0;
|
||||
port_info->trdat = 0;
|
||||
|
||||
/* portA, portB, SIU operate */
|
||||
siu_write32(base + SIU_SRCTL, 0x301);
|
||||
|
||||
/* portA=256fs, portB=256fs */
|
||||
siu_write32(base + SIU_CKCTL, 0x40400000);
|
||||
|
||||
/* portA's BRG does not divide SIUCKA */
|
||||
siu_write32(base + SIU_BRGASEL, 0);
|
||||
siu_write32(base + SIU_BRRA, 0);
|
||||
|
||||
/* portB's BRG divides SIUCKB by half */
|
||||
siu_write32(base + SIU_BRGBSEL, 1);
|
||||
siu_write32(base + SIU_BRRB, 0);
|
||||
|
||||
siu_write32(base + SIU_IFCTL, 0x44440000);
|
||||
|
||||
/* portA: 32 bit/fs, master; portB: 32 bit/fs, master */
|
||||
siu_write32(base + SIU_SFORM, 0x0c0c0000);
|
||||
|
||||
/*
|
||||
* Volume levels: looks like the DSP firmware implements volume controls
|
||||
* differently from what's described in the datasheet
|
||||
*/
|
||||
siu_write32(base + SIU_SBDVCA, port_info->playback.volume);
|
||||
siu_write32(base + SIU_SBDVCB, port_info->capture.volume);
|
||||
}
|
||||
|
||||
static void siu_dai_stop(struct siu_port *port_info)
|
||||
{
|
||||
struct siu_info *info = siu_i2s_data;
|
||||
u32 __iomem *base = info->reg;
|
||||
|
||||
/* SIU software reset */
|
||||
siu_write32(base + SIU_SRCTL, 0);
|
||||
}
|
||||
|
||||
static void siu_dai_spbAselect(struct siu_port *port_info)
|
||||
{
|
||||
struct siu_info *info = siu_i2s_data;
|
||||
struct siu_firmware *fw = &info->fw;
|
||||
u32 *ydef = fw->yram0;
|
||||
u32 idx;
|
||||
|
||||
/* path A use */
|
||||
if (!info->port_id)
|
||||
idx = 1; /* portA */
|
||||
else
|
||||
idx = 2; /* portB */
|
||||
|
||||
ydef[0] = (fw->spbpar[idx].ab1a << 16) |
|
||||
(fw->spbpar[idx].ab0a << 8) |
|
||||
(fw->spbpar[idx].dir << 7) | 3;
|
||||
ydef[1] = fw->yram0[1]; /* 0x03000300 */
|
||||
ydef[2] = (16 / 2) << 24;
|
||||
ydef[3] = fw->yram0[3]; /* 0 */
|
||||
ydef[4] = fw->yram0[4]; /* 0 */
|
||||
ydef[7] = fw->spbpar[idx].event;
|
||||
port_info->stfifo |= fw->spbpar[idx].stfifo;
|
||||
port_info->trdat |= fw->spbpar[idx].trdat;
|
||||
}
|
||||
|
||||
static void siu_dai_spbBselect(struct siu_port *port_info)
|
||||
{
|
||||
struct siu_info *info = siu_i2s_data;
|
||||
struct siu_firmware *fw = &info->fw;
|
||||
u32 *ydef = fw->yram0;
|
||||
u32 idx;
|
||||
|
||||
/* path B use */
|
||||
if (!info->port_id)
|
||||
idx = 7; /* portA */
|
||||
else
|
||||
idx = 8; /* portB */
|
||||
|
||||
ydef[5] = (fw->spbpar[idx].ab1a << 16) |
|
||||
(fw->spbpar[idx].ab0a << 8) | 1;
|
||||
ydef[6] = fw->spbpar[idx].event;
|
||||
port_info->stfifo |= fw->spbpar[idx].stfifo;
|
||||
port_info->trdat |= fw->spbpar[idx].trdat;
|
||||
}
|
||||
|
||||
static void siu_dai_open(struct siu_stream *siu_stream)
|
||||
{
|
||||
struct siu_info *info = siu_i2s_data;
|
||||
u32 __iomem *base = info->reg;
|
||||
u32 srctl, ifctl;
|
||||
|
||||
srctl = siu_read32(base + SIU_SRCTL);
|
||||
ifctl = siu_read32(base + SIU_IFCTL);
|
||||
|
||||
switch (info->port_id) {
|
||||
case SIU_PORT_A:
|
||||
/* portA operates */
|
||||
srctl |= 0x200;
|
||||
ifctl &= ~0xc2;
|
||||
break;
|
||||
case SIU_PORT_B:
|
||||
/* portB operates */
|
||||
srctl |= 0x100;
|
||||
ifctl &= ~0x31;
|
||||
break;
|
||||
}
|
||||
|
||||
siu_write32(base + SIU_SRCTL, srctl);
|
||||
/* Unmute and configure portA */
|
||||
siu_write32(base + SIU_IFCTL, ifctl);
|
||||
}
|
||||
|
||||
/*
|
||||
* At the moment only fixed Left-upper, Left-lower, Right-upper, Right-lower
|
||||
* packing is supported
|
||||
*/
|
||||
static void siu_dai_pcmdatapack(struct siu_stream *siu_stream)
|
||||
{
|
||||
struct siu_info *info = siu_i2s_data;
|
||||
u32 __iomem *base = info->reg;
|
||||
u32 dpak;
|
||||
|
||||
dpak = siu_read32(base + SIU_DPAK);
|
||||
|
||||
switch (info->port_id) {
|
||||
case SIU_PORT_A:
|
||||
dpak &= ~0xc0000000;
|
||||
break;
|
||||
case SIU_PORT_B:
|
||||
dpak &= ~0x00c00000;
|
||||
break;
|
||||
}
|
||||
|
||||
siu_write32(base + SIU_DPAK, dpak);
|
||||
}
|
||||
|
||||
static int siu_dai_spbstart(struct siu_port *port_info)
|
||||
{
|
||||
struct siu_info *info = siu_i2s_data;
|
||||
u32 __iomem *base = info->reg;
|
||||
struct siu_firmware *fw = &info->fw;
|
||||
u32 *ydef = fw->yram0;
|
||||
int cnt;
|
||||
u32 __iomem *add;
|
||||
u32 *ptr;
|
||||
|
||||
/* Load SPB Program in PRAM */
|
||||
ptr = fw->pram0;
|
||||
add = info->pram;
|
||||
for (cnt = 0; cnt < PRAM0_SIZE; cnt++, add++, ptr++)
|
||||
siu_write32(add, *ptr);
|
||||
|
||||
ptr = fw->pram1;
|
||||
add = info->pram + (0x0100 / sizeof(u32));
|
||||
for (cnt = 0; cnt < PRAM1_SIZE; cnt++, add++, ptr++)
|
||||
siu_write32(add, *ptr);
|
||||
|
||||
/* XRAM initialization */
|
||||
add = info->xram;
|
||||
for (cnt = 0; cnt < XRAM0_SIZE + XRAM1_SIZE + XRAM2_SIZE; cnt++, add++)
|
||||
siu_write32(add, 0);
|
||||
|
||||
/* YRAM variable area initialization */
|
||||
add = info->yram;
|
||||
for (cnt = 0; cnt < YRAM_DEF_SIZE; cnt++, add++)
|
||||
siu_write32(add, ydef[cnt]);
|
||||
|
||||
/* YRAM FIR coefficient area initialization */
|
||||
add = info->yram + (0x0200 / sizeof(u32));
|
||||
for (cnt = 0; cnt < YRAM_FIR_SIZE; cnt++, add++)
|
||||
siu_write32(add, fw->yram_fir_coeff[cnt]);
|
||||
|
||||
/* YRAM IIR coefficient area initialization */
|
||||
add = info->yram + (0x0600 / sizeof(u32));
|
||||
for (cnt = 0; cnt < YRAM_IIR_SIZE; cnt++, add++)
|
||||
siu_write32(add, 0);
|
||||
|
||||
siu_write32(base + SIU_TRDAT, port_info->trdat);
|
||||
port_info->trdat = 0x0;
|
||||
|
||||
|
||||
/* SPB start condition: software */
|
||||
siu_write32(base + SIU_SBACTIV, 0);
|
||||
/* Start SPB */
|
||||
siu_write32(base + SIU_SBCTL, 0xc0000000);
|
||||
/* Wait for program to halt */
|
||||
cnt = 0x10000;
|
||||
while (--cnt && siu_read32(base + SIU_SBCTL) != 0x80000000)
|
||||
cpu_relax();
|
||||
|
||||
if (!cnt)
|
||||
return -EBUSY;
|
||||
|
||||
/* SPB program start address setting */
|
||||
siu_write32(base + SIU_SBPSET, 0x00400000);
|
||||
/* SPB hardware start(FIFOCTL source) */
|
||||
siu_write32(base + SIU_SBACTIV, 0xc0000000);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static void siu_dai_spbstop(struct siu_port *port_info)
|
||||
{
|
||||
struct siu_info *info = siu_i2s_data;
|
||||
u32 __iomem *base = info->reg;
|
||||
|
||||
siu_write32(base + SIU_SBACTIV, 0);
|
||||
/* SPB stop */
|
||||
siu_write32(base + SIU_SBCTL, 0);
|
||||
|
||||
port_info->stfifo = 0;
|
||||
}
|
||||
|
||||
/* API functions */
|
||||
|
||||
/* Playback and capture hardware properties are identical */
|
||||
static struct snd_pcm_hardware siu_dai_pcm_hw = {
|
||||
.info = SNDRV_PCM_INFO_INTERLEAVED,
|
||||
.formats = SNDRV_PCM_FMTBIT_S16,
|
||||
.rates = SNDRV_PCM_RATE_8000_48000,
|
||||
.rate_min = 8000,
|
||||
.rate_max = 48000,
|
||||
.channels_min = 2,
|
||||
.channels_max = 2,
|
||||
.buffer_bytes_max = SIU_BUFFER_BYTES_MAX,
|
||||
.period_bytes_min = SIU_PERIOD_BYTES_MIN,
|
||||
.period_bytes_max = SIU_PERIOD_BYTES_MAX,
|
||||
.periods_min = SIU_PERIODS_MIN,
|
||||
.periods_max = SIU_PERIODS_MAX,
|
||||
};
|
||||
|
||||
static int siu_dai_info_volume(struct snd_kcontrol *kctrl,
|
||||
struct snd_ctl_elem_info *uinfo)
|
||||
{
|
||||
struct siu_port *port_info = snd_kcontrol_chip(kctrl);
|
||||
|
||||
dev_dbg(port_info->pcm->card->dev, "%s\n", __func__);
|
||||
|
||||
uinfo->type = SNDRV_CTL_ELEM_TYPE_INTEGER;
|
||||
uinfo->count = 2;
|
||||
uinfo->value.integer.min = 0;
|
||||
uinfo->value.integer.max = SIU_MAX_VOLUME;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int siu_dai_get_volume(struct snd_kcontrol *kctrl,
|
||||
struct snd_ctl_elem_value *ucontrol)
|
||||
{
|
||||
struct siu_port *port_info = snd_kcontrol_chip(kctrl);
|
||||
struct device *dev = port_info->pcm->card->dev;
|
||||
u32 vol;
|
||||
|
||||
dev_dbg(dev, "%s\n", __func__);
|
||||
|
||||
switch (kctrl->private_value) {
|
||||
case VOLUME_PLAYBACK:
|
||||
/* Playback is always on port 0 */
|
||||
vol = port_info->playback.volume;
|
||||
ucontrol->value.integer.value[0] = vol & 0xffff;
|
||||
ucontrol->value.integer.value[1] = vol >> 16 & 0xffff;
|
||||
break;
|
||||
case VOLUME_CAPTURE:
|
||||
/* Capture is always on port 1 */
|
||||
vol = port_info->capture.volume;
|
||||
ucontrol->value.integer.value[0] = vol & 0xffff;
|
||||
ucontrol->value.integer.value[1] = vol >> 16 & 0xffff;
|
||||
break;
|
||||
default:
|
||||
dev_err(dev, "%s() invalid private_value=%ld\n",
|
||||
__func__, kctrl->private_value);
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int siu_dai_put_volume(struct snd_kcontrol *kctrl,
|
||||
struct snd_ctl_elem_value *ucontrol)
|
||||
{
|
||||
struct siu_port *port_info = snd_kcontrol_chip(kctrl);
|
||||
struct device *dev = port_info->pcm->card->dev;
|
||||
struct siu_info *info = siu_i2s_data;
|
||||
u32 __iomem *base = info->reg;
|
||||
u32 new_vol;
|
||||
u32 cur_vol;
|
||||
|
||||
dev_dbg(dev, "%s\n", __func__);
|
||||
|
||||
if (ucontrol->value.integer.value[0] < 0 ||
|
||||
ucontrol->value.integer.value[0] > SIU_MAX_VOLUME ||
|
||||
ucontrol->value.integer.value[1] < 0 ||
|
||||
ucontrol->value.integer.value[1] > SIU_MAX_VOLUME)
|
||||
return -EINVAL;
|
||||
|
||||
new_vol = ucontrol->value.integer.value[0] |
|
||||
ucontrol->value.integer.value[1] << 16;
|
||||
|
||||
/* See comment above - DSP firmware implementation */
|
||||
switch (kctrl->private_value) {
|
||||
case VOLUME_PLAYBACK:
|
||||
/* Playback is always on port 0 */
|
||||
cur_vol = port_info->playback.volume;
|
||||
siu_write32(base + SIU_SBDVCA, new_vol);
|
||||
port_info->playback.volume = new_vol;
|
||||
break;
|
||||
case VOLUME_CAPTURE:
|
||||
/* Capture is always on port 1 */
|
||||
cur_vol = port_info->capture.volume;
|
||||
siu_write32(base + SIU_SBDVCB, new_vol);
|
||||
port_info->capture.volume = new_vol;
|
||||
break;
|
||||
default:
|
||||
dev_err(dev, "%s() invalid private_value=%ld\n",
|
||||
__func__, kctrl->private_value);
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
if (cur_vol != new_vol)
|
||||
return 1;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static struct snd_kcontrol_new playback_controls = {
|
||||
.iface = SNDRV_CTL_ELEM_IFACE_MIXER,
|
||||
.name = "PCM Playback Volume",
|
||||
.index = 0,
|
||||
.info = siu_dai_info_volume,
|
||||
.get = siu_dai_get_volume,
|
||||
.put = siu_dai_put_volume,
|
||||
.private_value = VOLUME_PLAYBACK,
|
||||
};
|
||||
|
||||
static struct snd_kcontrol_new capture_controls = {
|
||||
.iface = SNDRV_CTL_ELEM_IFACE_MIXER,
|
||||
.name = "PCM Capture Volume",
|
||||
.index = 0,
|
||||
.info = siu_dai_info_volume,
|
||||
.get = siu_dai_get_volume,
|
||||
.put = siu_dai_put_volume,
|
||||
.private_value = VOLUME_CAPTURE,
|
||||
};
|
||||
|
||||
int siu_init_port(int port, struct siu_port **port_info, struct snd_card *card)
|
||||
{
|
||||
struct device *dev = card->dev;
|
||||
struct snd_kcontrol *kctrl;
|
||||
int ret;
|
||||
|
||||
*port_info = kzalloc(sizeof(**port_info), GFP_KERNEL);
|
||||
if (!*port_info)
|
||||
return -ENOMEM;
|
||||
|
||||
dev_dbg(dev, "%s: port #%d@%p\n", __func__, port, *port_info);
|
||||
|
||||
(*port_info)->playback.volume = DFLT_VOLUME_LEVEL;
|
||||
(*port_info)->capture.volume = DFLT_VOLUME_LEVEL;
|
||||
|
||||
/*
|
||||
* Add mixer support. The SPB is used to change the volume. Both
|
||||
* ports use the same SPB. Therefore, we only register one
|
||||
* control instance since it will be used by both channels.
|
||||
* In error case we continue without controls.
|
||||
*/
|
||||
kctrl = snd_ctl_new1(&playback_controls, *port_info);
|
||||
ret = snd_ctl_add(card, kctrl);
|
||||
if (ret < 0)
|
||||
dev_err(dev,
|
||||
"failed to add playback controls %p port=%d err=%d\n",
|
||||
kctrl, port, ret);
|
||||
|
||||
kctrl = snd_ctl_new1(&capture_controls, *port_info);
|
||||
ret = snd_ctl_add(card, kctrl);
|
||||
if (ret < 0)
|
||||
dev_err(dev,
|
||||
"failed to add capture controls %p port=%d err=%d\n",
|
||||
kctrl, port, ret);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
void siu_free_port(struct siu_port *port_info)
|
||||
{
|
||||
kfree(port_info);
|
||||
}
|
||||
|
||||
static int siu_dai_startup(struct snd_pcm_substream *substream,
|
||||
struct snd_soc_dai *dai)
|
||||
{
|
||||
struct siu_info *info = snd_soc_dai_get_drvdata(dai);
|
||||
struct snd_pcm_runtime *rt = substream->runtime;
|
||||
struct siu_port *port_info = siu_port_info(substream);
|
||||
int ret;
|
||||
|
||||
dev_dbg(substream->pcm->card->dev, "%s: port=%d@%p\n", __func__,
|
||||
info->port_id, port_info);
|
||||
|
||||
snd_soc_set_runtime_hwparams(substream, &siu_dai_pcm_hw);
|
||||
|
||||
ret = snd_pcm_hw_constraint_integer(rt, SNDRV_PCM_HW_PARAM_PERIODS);
|
||||
if (unlikely(ret < 0))
|
||||
return ret;
|
||||
|
||||
siu_dai_start(port_info);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static void siu_dai_shutdown(struct snd_pcm_substream *substream,
|
||||
struct snd_soc_dai *dai)
|
||||
{
|
||||
struct siu_info *info = snd_soc_dai_get_drvdata(dai);
|
||||
struct siu_port *port_info = siu_port_info(substream);
|
||||
|
||||
dev_dbg(substream->pcm->card->dev, "%s: port=%d@%p\n", __func__,
|
||||
info->port_id, port_info);
|
||||
|
||||
if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK)
|
||||
port_info->play_cap &= ~PLAYBACK_ENABLED;
|
||||
else
|
||||
port_info->play_cap &= ~CAPTURE_ENABLED;
|
||||
|
||||
/* Stop the siu if the other stream is not using it */
|
||||
if (!port_info->play_cap) {
|
||||
/* during stmread or stmwrite ? */
|
||||
if (WARN_ON(port_info->playback.rw_flg || port_info->capture.rw_flg))
|
||||
return;
|
||||
siu_dai_spbstop(port_info);
|
||||
siu_dai_stop(port_info);
|
||||
}
|
||||
}
|
||||
|
||||
/* PCM part of siu_dai_playback_prepare() / siu_dai_capture_prepare() */
|
||||
static int siu_dai_prepare(struct snd_pcm_substream *substream,
|
||||
struct snd_soc_dai *dai)
|
||||
{
|
||||
struct siu_info *info = snd_soc_dai_get_drvdata(dai);
|
||||
struct snd_pcm_runtime *rt = substream->runtime;
|
||||
struct siu_port *port_info = siu_port_info(substream);
|
||||
struct siu_stream *siu_stream;
|
||||
int self, ret;
|
||||
|
||||
dev_dbg(substream->pcm->card->dev,
|
||||
"%s: port %d, active streams %lx, %d channels\n",
|
||||
__func__, info->port_id, port_info->play_cap, rt->channels);
|
||||
|
||||
if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) {
|
||||
self = PLAYBACK_ENABLED;
|
||||
siu_stream = &port_info->playback;
|
||||
} else {
|
||||
self = CAPTURE_ENABLED;
|
||||
siu_stream = &port_info->capture;
|
||||
}
|
||||
|
||||
/* Set up the siu if not already done */
|
||||
if (!port_info->play_cap) {
|
||||
siu_stream->rw_flg = 0; /* stream-data transfer flag */
|
||||
|
||||
siu_dai_spbAselect(port_info);
|
||||
siu_dai_spbBselect(port_info);
|
||||
|
||||
siu_dai_open(siu_stream);
|
||||
|
||||
siu_dai_pcmdatapack(siu_stream);
|
||||
|
||||
ret = siu_dai_spbstart(port_info);
|
||||
if (ret < 0)
|
||||
goto fail;
|
||||
} else {
|
||||
ret = 0;
|
||||
}
|
||||
|
||||
port_info->play_cap |= self;
|
||||
|
||||
fail:
|
||||
return ret;
|
||||
}
|
||||
|
||||
/*
|
||||
* SIU can set bus format to I2S / PCM / SPDIF independently for playback and
|
||||
* capture, however, the current API sets the bus format globally for a DAI.
|
||||
*/
|
||||
static int siu_dai_set_fmt(struct snd_soc_dai *dai,
|
||||
unsigned int fmt)
|
||||
{
|
||||
struct siu_info *info = snd_soc_dai_get_drvdata(dai);
|
||||
u32 __iomem *base = info->reg;
|
||||
u32 ifctl;
|
||||
|
||||
dev_dbg(dai->dev, "%s: fmt 0x%x on port %d\n",
|
||||
__func__, fmt, info->port_id);
|
||||
|
||||
if (info->port_id < 0)
|
||||
return -ENODEV;
|
||||
|
||||
/* Here select between I2S / PCM / SPDIF */
|
||||
switch (fmt & SND_SOC_DAIFMT_FORMAT_MASK) {
|
||||
case SND_SOC_DAIFMT_I2S:
|
||||
ifctl = siu_flags[info->port_id].playback.i2s |
|
||||
siu_flags[info->port_id].capture.i2s;
|
||||
break;
|
||||
case SND_SOC_DAIFMT_LEFT_J:
|
||||
ifctl = siu_flags[info->port_id].playback.pcm |
|
||||
siu_flags[info->port_id].capture.pcm;
|
||||
break;
|
||||
/* SPDIF disabled - see comment at the top */
|
||||
default:
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
ifctl |= ~(siu_flags[info->port_id].playback.mask |
|
||||
siu_flags[info->port_id].capture.mask) &
|
||||
siu_read32(base + SIU_IFCTL);
|
||||
siu_write32(base + SIU_IFCTL, ifctl);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int siu_dai_set_sysclk(struct snd_soc_dai *dai, int clk_id,
|
||||
unsigned int freq, int dir)
|
||||
{
|
||||
struct clk *siu_clk, *parent_clk;
|
||||
char *siu_name, *parent_name;
|
||||
int ret;
|
||||
|
||||
if (dir != SND_SOC_CLOCK_IN)
|
||||
return -EINVAL;
|
||||
|
||||
dev_dbg(dai->dev, "%s: using clock %d\n", __func__, clk_id);
|
||||
|
||||
switch (clk_id) {
|
||||
case SIU_CLKA_PLL:
|
||||
siu_name = "siua_clk";
|
||||
parent_name = "pll_clk";
|
||||
break;
|
||||
case SIU_CLKA_EXT:
|
||||
siu_name = "siua_clk";
|
||||
parent_name = "siumcka_clk";
|
||||
break;
|
||||
case SIU_CLKB_PLL:
|
||||
siu_name = "siub_clk";
|
||||
parent_name = "pll_clk";
|
||||
break;
|
||||
case SIU_CLKB_EXT:
|
||||
siu_name = "siub_clk";
|
||||
parent_name = "siumckb_clk";
|
||||
break;
|
||||
default:
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
siu_clk = clk_get(dai->dev, siu_name);
|
||||
if (IS_ERR(siu_clk)) {
|
||||
dev_err(dai->dev, "%s: cannot get a SIU clock: %ld\n", __func__,
|
||||
PTR_ERR(siu_clk));
|
||||
return PTR_ERR(siu_clk);
|
||||
}
|
||||
|
||||
parent_clk = clk_get(dai->dev, parent_name);
|
||||
if (IS_ERR(parent_clk)) {
|
||||
ret = PTR_ERR(parent_clk);
|
||||
dev_err(dai->dev, "cannot get a SIU clock parent: %d\n", ret);
|
||||
goto epclkget;
|
||||
}
|
||||
|
||||
ret = clk_set_parent(siu_clk, parent_clk);
|
||||
if (ret < 0) {
|
||||
dev_err(dai->dev, "cannot reparent the SIU clock: %d\n", ret);
|
||||
goto eclksetp;
|
||||
}
|
||||
|
||||
ret = clk_set_rate(siu_clk, freq);
|
||||
if (ret < 0)
|
||||
dev_err(dai->dev, "cannot set SIU clock rate: %d\n", ret);
|
||||
|
||||
/* TODO: when clkdev gets reference counting we'll move these to siu_dai_shutdown() */
|
||||
eclksetp:
|
||||
clk_put(parent_clk);
|
||||
epclkget:
|
||||
clk_put(siu_clk);
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
static const struct snd_soc_dai_ops siu_dai_ops = {
|
||||
.startup = siu_dai_startup,
|
||||
.shutdown = siu_dai_shutdown,
|
||||
.prepare = siu_dai_prepare,
|
||||
.set_sysclk = siu_dai_set_sysclk,
|
||||
.set_fmt = siu_dai_set_fmt,
|
||||
};
|
||||
|
||||
static struct snd_soc_dai_driver siu_i2s_dai = {
|
||||
.name = "siu-i2s-dai",
|
||||
.playback = {
|
||||
.channels_min = 2,
|
||||
.channels_max = 2,
|
||||
.formats = SNDRV_PCM_FMTBIT_S16,
|
||||
.rates = SNDRV_PCM_RATE_8000_48000,
|
||||
},
|
||||
.capture = {
|
||||
.channels_min = 2,
|
||||
.channels_max = 2,
|
||||
.formats = SNDRV_PCM_FMTBIT_S16,
|
||||
.rates = SNDRV_PCM_RATE_8000_48000,
|
||||
},
|
||||
.ops = &siu_dai_ops,
|
||||
};
|
||||
|
||||
static const struct snd_soc_component_driver siu_i2s_component = {
|
||||
.name = "siu-i2s",
|
||||
};
|
||||
|
||||
static int siu_probe(struct platform_device *pdev)
|
||||
{
|
||||
const struct firmware *fw_entry;
|
||||
struct resource *res, *region;
|
||||
struct siu_info *info;
|
||||
int ret;
|
||||
|
||||
info = kmalloc(sizeof(*info), GFP_KERNEL);
|
||||
if (!info)
|
||||
return -ENOMEM;
|
||||
siu_i2s_data = info;
|
||||
info->dev = &pdev->dev;
|
||||
|
||||
ret = request_firmware(&fw_entry, "siu_spb.bin", &pdev->dev);
|
||||
if (ret)
|
||||
goto ereqfw;
|
||||
|
||||
/*
|
||||
* Loaded firmware is "const" - read only, but we have to modify it in
|
||||
* snd_siu_sh7343_spbAselect() and snd_siu_sh7343_spbBselect()
|
||||
*/
|
||||
memcpy(&info->fw, fw_entry->data, fw_entry->size);
|
||||
|
||||
release_firmware(fw_entry);
|
||||
|
||||
res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
|
||||
if (!res) {
|
||||
ret = -ENODEV;
|
||||
goto egetres;
|
||||
}
|
||||
|
||||
region = request_mem_region(res->start, resource_size(res),
|
||||
pdev->name);
|
||||
if (!region) {
|
||||
dev_err(&pdev->dev, "SIU region already claimed\n");
|
||||
ret = -EBUSY;
|
||||
goto ereqmemreg;
|
||||
}
|
||||
|
||||
ret = -ENOMEM;
|
||||
info->pram = ioremap(res->start, PRAM_SIZE);
|
||||
if (!info->pram)
|
||||
goto emappram;
|
||||
info->xram = ioremap(res->start + XRAM_OFFSET, XRAM_SIZE);
|
||||
if (!info->xram)
|
||||
goto emapxram;
|
||||
info->yram = ioremap(res->start + YRAM_OFFSET, YRAM_SIZE);
|
||||
if (!info->yram)
|
||||
goto emapyram;
|
||||
info->reg = ioremap(res->start + REG_OFFSET, resource_size(res) -
|
||||
REG_OFFSET);
|
||||
if (!info->reg)
|
||||
goto emapreg;
|
||||
|
||||
dev_set_drvdata(&pdev->dev, info);
|
||||
|
||||
/* register using ARRAY version so we can keep dai name */
|
||||
ret = snd_soc_register_component(&pdev->dev, &siu_i2s_component,
|
||||
&siu_i2s_dai, 1);
|
||||
if (ret < 0)
|
||||
goto edaiinit;
|
||||
|
||||
ret = snd_soc_register_platform(&pdev->dev, &siu_platform);
|
||||
if (ret < 0)
|
||||
goto esocregp;
|
||||
|
||||
pm_runtime_enable(&pdev->dev);
|
||||
|
||||
return ret;
|
||||
|
||||
esocregp:
|
||||
snd_soc_unregister_component(&pdev->dev);
|
||||
edaiinit:
|
||||
iounmap(info->reg);
|
||||
emapreg:
|
||||
iounmap(info->yram);
|
||||
emapyram:
|
||||
iounmap(info->xram);
|
||||
emapxram:
|
||||
iounmap(info->pram);
|
||||
emappram:
|
||||
release_mem_region(res->start, resource_size(res));
|
||||
ereqmemreg:
|
||||
egetres:
|
||||
ereqfw:
|
||||
kfree(info);
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int siu_remove(struct platform_device *pdev)
|
||||
{
|
||||
struct siu_info *info = dev_get_drvdata(&pdev->dev);
|
||||
struct resource *res;
|
||||
|
||||
pm_runtime_disable(&pdev->dev);
|
||||
|
||||
snd_soc_unregister_platform(&pdev->dev);
|
||||
snd_soc_unregister_component(&pdev->dev);
|
||||
|
||||
iounmap(info->reg);
|
||||
iounmap(info->yram);
|
||||
iounmap(info->xram);
|
||||
iounmap(info->pram);
|
||||
res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
|
||||
if (res)
|
||||
release_mem_region(res->start, resource_size(res));
|
||||
kfree(info);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static struct platform_driver siu_driver = {
|
||||
.driver = {
|
||||
.owner = THIS_MODULE,
|
||||
.name = "siu-pcm-audio",
|
||||
},
|
||||
.probe = siu_probe,
|
||||
.remove = siu_remove,
|
||||
};
|
||||
|
||||
module_platform_driver(siu_driver);
|
||||
|
||||
MODULE_AUTHOR("Carlos Munoz <carlos@kenati.com>");
|
||||
MODULE_DESCRIPTION("ALSA SoC SH7722 SIU driver");
|
||||
MODULE_LICENSE("GPL");
|
||||
613
sound/soc/sh/siu_pcm.c
Normal file
613
sound/soc/sh/siu_pcm.c
Normal file
|
|
@ -0,0 +1,613 @@
|
|||
/*
|
||||
* siu_pcm.c - ALSA driver for Renesas SH7343, SH7722 SIU peripheral.
|
||||
*
|
||||
* Copyright (C) 2009-2010 Guennadi Liakhovetski <g.liakhovetski@gmx.de>
|
||||
* Copyright (C) 2006 Carlos Munoz <carlos@kenati.com>
|
||||
*
|
||||
* This program is free software; you can redistribute it and/or modify
|
||||
* it under the terms of the GNU General Public License as published by
|
||||
* the Free Software Foundation; either version 2 of the License, or
|
||||
* (at your option) any later version.
|
||||
*
|
||||
* This program is distributed in the hope that it will be useful,
|
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
* GNU General Public License for more details.
|
||||
*
|
||||
* You should have received a copy of the GNU General Public License
|
||||
* along with this program; if not, write to the Free Software
|
||||
* Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
||||
*/
|
||||
#include <linux/delay.h>
|
||||
#include <linux/dma-mapping.h>
|
||||
#include <linux/dmaengine.h>
|
||||
#include <linux/interrupt.h>
|
||||
#include <linux/module.h>
|
||||
#include <linux/platform_device.h>
|
||||
|
||||
#include <sound/control.h>
|
||||
#include <sound/core.h>
|
||||
#include <sound/pcm.h>
|
||||
#include <sound/pcm_params.h>
|
||||
#include <sound/soc.h>
|
||||
|
||||
#include <asm/siu.h>
|
||||
|
||||
#include "siu.h"
|
||||
|
||||
#define GET_MAX_PERIODS(buf_bytes, period_bytes) \
|
||||
((buf_bytes) / (period_bytes))
|
||||
#define PERIOD_OFFSET(buf_addr, period_num, period_bytes) \
|
||||
((buf_addr) + ((period_num) * (period_bytes)))
|
||||
|
||||
#define RWF_STM_RD 0x01 /* Read in progress */
|
||||
#define RWF_STM_WT 0x02 /* Write in progress */
|
||||
|
||||
struct siu_port *siu_ports[SIU_PORT_NUM];
|
||||
|
||||
/* transfersize is number of u32 dma transfers per period */
|
||||
static int siu_pcm_stmwrite_stop(struct siu_port *port_info)
|
||||
{
|
||||
struct siu_info *info = siu_i2s_data;
|
||||
u32 __iomem *base = info->reg;
|
||||
struct siu_stream *siu_stream = &port_info->playback;
|
||||
u32 stfifo;
|
||||
|
||||
if (!siu_stream->rw_flg)
|
||||
return -EPERM;
|
||||
|
||||
/* output FIFO disable */
|
||||
stfifo = siu_read32(base + SIU_STFIFO);
|
||||
siu_write32(base + SIU_STFIFO, stfifo & ~0x0c180c18);
|
||||
pr_debug("%s: STFIFO %x -> %x\n", __func__,
|
||||
stfifo, stfifo & ~0x0c180c18);
|
||||
|
||||
/* during stmwrite clear */
|
||||
siu_stream->rw_flg = 0;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int siu_pcm_stmwrite_start(struct siu_port *port_info)
|
||||
{
|
||||
struct siu_stream *siu_stream = &port_info->playback;
|
||||
|
||||
if (siu_stream->rw_flg)
|
||||
return -EPERM;
|
||||
|
||||
/* Current period in buffer */
|
||||
port_info->playback.cur_period = 0;
|
||||
|
||||
/* during stmwrite flag set */
|
||||
siu_stream->rw_flg = RWF_STM_WT;
|
||||
|
||||
/* DMA transfer start */
|
||||
tasklet_schedule(&siu_stream->tasklet);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static void siu_dma_tx_complete(void *arg)
|
||||
{
|
||||
struct siu_stream *siu_stream = arg;
|
||||
|
||||
if (!siu_stream->rw_flg)
|
||||
return;
|
||||
|
||||
/* Update completed period count */
|
||||
if (++siu_stream->cur_period >=
|
||||
GET_MAX_PERIODS(siu_stream->buf_bytes,
|
||||
siu_stream->period_bytes))
|
||||
siu_stream->cur_period = 0;
|
||||
|
||||
pr_debug("%s: done period #%d (%u/%u bytes), cookie %d\n",
|
||||
__func__, siu_stream->cur_period,
|
||||
siu_stream->cur_period * siu_stream->period_bytes,
|
||||
siu_stream->buf_bytes, siu_stream->cookie);
|
||||
|
||||
tasklet_schedule(&siu_stream->tasklet);
|
||||
|
||||
/* Notify alsa: a period is done */
|
||||
snd_pcm_period_elapsed(siu_stream->substream);
|
||||
}
|
||||
|
||||
static int siu_pcm_wr_set(struct siu_port *port_info,
|
||||
dma_addr_t buff, u32 size)
|
||||
{
|
||||
struct siu_info *info = siu_i2s_data;
|
||||
u32 __iomem *base = info->reg;
|
||||
struct siu_stream *siu_stream = &port_info->playback;
|
||||
struct snd_pcm_substream *substream = siu_stream->substream;
|
||||
struct device *dev = substream->pcm->card->dev;
|
||||
struct dma_async_tx_descriptor *desc;
|
||||
dma_cookie_t cookie;
|
||||
struct scatterlist sg;
|
||||
u32 stfifo;
|
||||
|
||||
sg_init_table(&sg, 1);
|
||||
sg_set_page(&sg, pfn_to_page(PFN_DOWN(buff)),
|
||||
size, offset_in_page(buff));
|
||||
sg_dma_len(&sg) = size;
|
||||
sg_dma_address(&sg) = buff;
|
||||
|
||||
desc = dmaengine_prep_slave_sg(siu_stream->chan,
|
||||
&sg, 1, DMA_MEM_TO_DEV, DMA_PREP_INTERRUPT | DMA_CTRL_ACK);
|
||||
if (!desc) {
|
||||
dev_err(dev, "Failed to allocate a dma descriptor\n");
|
||||
return -ENOMEM;
|
||||
}
|
||||
|
||||
desc->callback = siu_dma_tx_complete;
|
||||
desc->callback_param = siu_stream;
|
||||
cookie = dmaengine_submit(desc);
|
||||
if (cookie < 0) {
|
||||
dev_err(dev, "Failed to submit a dma transfer\n");
|
||||
return cookie;
|
||||
}
|
||||
|
||||
siu_stream->tx_desc = desc;
|
||||
siu_stream->cookie = cookie;
|
||||
|
||||
dma_async_issue_pending(siu_stream->chan);
|
||||
|
||||
/* only output FIFO enable */
|
||||
stfifo = siu_read32(base + SIU_STFIFO);
|
||||
siu_write32(base + SIU_STFIFO, stfifo | (port_info->stfifo & 0x0c180c18));
|
||||
dev_dbg(dev, "%s: STFIFO %x -> %x\n", __func__,
|
||||
stfifo, stfifo | (port_info->stfifo & 0x0c180c18));
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int siu_pcm_rd_set(struct siu_port *port_info,
|
||||
dma_addr_t buff, size_t size)
|
||||
{
|
||||
struct siu_info *info = siu_i2s_data;
|
||||
u32 __iomem *base = info->reg;
|
||||
struct siu_stream *siu_stream = &port_info->capture;
|
||||
struct snd_pcm_substream *substream = siu_stream->substream;
|
||||
struct device *dev = substream->pcm->card->dev;
|
||||
struct dma_async_tx_descriptor *desc;
|
||||
dma_cookie_t cookie;
|
||||
struct scatterlist sg;
|
||||
u32 stfifo;
|
||||
|
||||
dev_dbg(dev, "%s: %u@%llx\n", __func__, size, (unsigned long long)buff);
|
||||
|
||||
sg_init_table(&sg, 1);
|
||||
sg_set_page(&sg, pfn_to_page(PFN_DOWN(buff)),
|
||||
size, offset_in_page(buff));
|
||||
sg_dma_len(&sg) = size;
|
||||
sg_dma_address(&sg) = buff;
|
||||
|
||||
desc = dmaengine_prep_slave_sg(siu_stream->chan,
|
||||
&sg, 1, DMA_DEV_TO_MEM, DMA_PREP_INTERRUPT | DMA_CTRL_ACK);
|
||||
if (!desc) {
|
||||
dev_err(dev, "Failed to allocate dma descriptor\n");
|
||||
return -ENOMEM;
|
||||
}
|
||||
|
||||
desc->callback = siu_dma_tx_complete;
|
||||
desc->callback_param = siu_stream;
|
||||
cookie = dmaengine_submit(desc);
|
||||
if (cookie < 0) {
|
||||
dev_err(dev, "Failed to submit dma descriptor\n");
|
||||
return cookie;
|
||||
}
|
||||
|
||||
siu_stream->tx_desc = desc;
|
||||
siu_stream->cookie = cookie;
|
||||
|
||||
dma_async_issue_pending(siu_stream->chan);
|
||||
|
||||
/* only input FIFO enable */
|
||||
stfifo = siu_read32(base + SIU_STFIFO);
|
||||
siu_write32(base + SIU_STFIFO, siu_read32(base + SIU_STFIFO) |
|
||||
(port_info->stfifo & 0x13071307));
|
||||
dev_dbg(dev, "%s: STFIFO %x -> %x\n", __func__,
|
||||
stfifo, stfifo | (port_info->stfifo & 0x13071307));
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static void siu_io_tasklet(unsigned long data)
|
||||
{
|
||||
struct siu_stream *siu_stream = (struct siu_stream *)data;
|
||||
struct snd_pcm_substream *substream = siu_stream->substream;
|
||||
struct device *dev = substream->pcm->card->dev;
|
||||
struct snd_pcm_runtime *rt = substream->runtime;
|
||||
struct siu_port *port_info = siu_port_info(substream);
|
||||
|
||||
dev_dbg(dev, "%s: flags %x\n", __func__, siu_stream->rw_flg);
|
||||
|
||||
if (!siu_stream->rw_flg) {
|
||||
dev_dbg(dev, "%s: stream inactive\n", __func__);
|
||||
return;
|
||||
}
|
||||
|
||||
if (substream->stream == SNDRV_PCM_STREAM_CAPTURE) {
|
||||
dma_addr_t buff;
|
||||
size_t count;
|
||||
u8 *virt;
|
||||
|
||||
buff = (dma_addr_t)PERIOD_OFFSET(rt->dma_addr,
|
||||
siu_stream->cur_period,
|
||||
siu_stream->period_bytes);
|
||||
virt = PERIOD_OFFSET(rt->dma_area,
|
||||
siu_stream->cur_period,
|
||||
siu_stream->period_bytes);
|
||||
count = siu_stream->period_bytes;
|
||||
|
||||
/* DMA transfer start */
|
||||
siu_pcm_rd_set(port_info, buff, count);
|
||||
} else {
|
||||
siu_pcm_wr_set(port_info,
|
||||
(dma_addr_t)PERIOD_OFFSET(rt->dma_addr,
|
||||
siu_stream->cur_period,
|
||||
siu_stream->period_bytes),
|
||||
siu_stream->period_bytes);
|
||||
}
|
||||
}
|
||||
|
||||
/* Capture */
|
||||
static int siu_pcm_stmread_start(struct siu_port *port_info)
|
||||
{
|
||||
struct siu_stream *siu_stream = &port_info->capture;
|
||||
|
||||
if (siu_stream->xfer_cnt > 0x1000000)
|
||||
return -EINVAL;
|
||||
if (siu_stream->rw_flg)
|
||||
return -EPERM;
|
||||
|
||||
/* Current period in buffer */
|
||||
siu_stream->cur_period = 0;
|
||||
|
||||
/* during stmread flag set */
|
||||
siu_stream->rw_flg = RWF_STM_RD;
|
||||
|
||||
tasklet_schedule(&siu_stream->tasklet);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int siu_pcm_stmread_stop(struct siu_port *port_info)
|
||||
{
|
||||
struct siu_info *info = siu_i2s_data;
|
||||
u32 __iomem *base = info->reg;
|
||||
struct siu_stream *siu_stream = &port_info->capture;
|
||||
struct device *dev = siu_stream->substream->pcm->card->dev;
|
||||
u32 stfifo;
|
||||
|
||||
if (!siu_stream->rw_flg)
|
||||
return -EPERM;
|
||||
|
||||
/* input FIFO disable */
|
||||
stfifo = siu_read32(base + SIU_STFIFO);
|
||||
siu_write32(base + SIU_STFIFO, stfifo & ~0x13071307);
|
||||
dev_dbg(dev, "%s: STFIFO %x -> %x\n", __func__,
|
||||
stfifo, stfifo & ~0x13071307);
|
||||
|
||||
/* during stmread flag clear */
|
||||
siu_stream->rw_flg = 0;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int siu_pcm_hw_params(struct snd_pcm_substream *ss,
|
||||
struct snd_pcm_hw_params *hw_params)
|
||||
{
|
||||
struct siu_info *info = siu_i2s_data;
|
||||
struct device *dev = ss->pcm->card->dev;
|
||||
int ret;
|
||||
|
||||
dev_dbg(dev, "%s: port=%d\n", __func__, info->port_id);
|
||||
|
||||
ret = snd_pcm_lib_malloc_pages(ss, params_buffer_bytes(hw_params));
|
||||
if (ret < 0)
|
||||
dev_err(dev, "snd_pcm_lib_malloc_pages() failed\n");
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int siu_pcm_hw_free(struct snd_pcm_substream *ss)
|
||||
{
|
||||
struct siu_info *info = siu_i2s_data;
|
||||
struct siu_port *port_info = siu_port_info(ss);
|
||||
struct device *dev = ss->pcm->card->dev;
|
||||
struct siu_stream *siu_stream;
|
||||
|
||||
if (ss->stream == SNDRV_PCM_STREAM_PLAYBACK)
|
||||
siu_stream = &port_info->playback;
|
||||
else
|
||||
siu_stream = &port_info->capture;
|
||||
|
||||
dev_dbg(dev, "%s: port=%d\n", __func__, info->port_id);
|
||||
|
||||
return snd_pcm_lib_free_pages(ss);
|
||||
}
|
||||
|
||||
static bool filter(struct dma_chan *chan, void *slave)
|
||||
{
|
||||
struct sh_dmae_slave *param = slave;
|
||||
|
||||
pr_debug("%s: slave ID %d\n", __func__, param->shdma_slave.slave_id);
|
||||
|
||||
chan->private = ¶m->shdma_slave;
|
||||
return true;
|
||||
}
|
||||
|
||||
static int siu_pcm_open(struct snd_pcm_substream *ss)
|
||||
{
|
||||
/* Playback / Capture */
|
||||
struct snd_soc_pcm_runtime *rtd = ss->private_data;
|
||||
struct siu_platform *pdata = rtd->platform->dev->platform_data;
|
||||
struct siu_info *info = siu_i2s_data;
|
||||
struct siu_port *port_info = siu_port_info(ss);
|
||||
struct siu_stream *siu_stream;
|
||||
u32 port = info->port_id;
|
||||
struct device *dev = ss->pcm->card->dev;
|
||||
dma_cap_mask_t mask;
|
||||
struct sh_dmae_slave *param;
|
||||
|
||||
dma_cap_zero(mask);
|
||||
dma_cap_set(DMA_SLAVE, mask);
|
||||
|
||||
dev_dbg(dev, "%s, port=%d@%p\n", __func__, port, port_info);
|
||||
|
||||
if (ss->stream == SNDRV_PCM_STREAM_PLAYBACK) {
|
||||
siu_stream = &port_info->playback;
|
||||
param = &siu_stream->param;
|
||||
param->shdma_slave.slave_id = port ? pdata->dma_slave_tx_b :
|
||||
pdata->dma_slave_tx_a;
|
||||
} else {
|
||||
siu_stream = &port_info->capture;
|
||||
param = &siu_stream->param;
|
||||
param->shdma_slave.slave_id = port ? pdata->dma_slave_rx_b :
|
||||
pdata->dma_slave_rx_a;
|
||||
}
|
||||
|
||||
/* Get DMA channel */
|
||||
siu_stream->chan = dma_request_channel(mask, filter, param);
|
||||
if (!siu_stream->chan) {
|
||||
dev_err(dev, "DMA channel allocation failed!\n");
|
||||
return -EBUSY;
|
||||
}
|
||||
|
||||
siu_stream->substream = ss;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int siu_pcm_close(struct snd_pcm_substream *ss)
|
||||
{
|
||||
struct siu_info *info = siu_i2s_data;
|
||||
struct device *dev = ss->pcm->card->dev;
|
||||
struct siu_port *port_info = siu_port_info(ss);
|
||||
struct siu_stream *siu_stream;
|
||||
|
||||
dev_dbg(dev, "%s: port=%d\n", __func__, info->port_id);
|
||||
|
||||
if (ss->stream == SNDRV_PCM_STREAM_PLAYBACK)
|
||||
siu_stream = &port_info->playback;
|
||||
else
|
||||
siu_stream = &port_info->capture;
|
||||
|
||||
dma_release_channel(siu_stream->chan);
|
||||
siu_stream->chan = NULL;
|
||||
|
||||
siu_stream->substream = NULL;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int siu_pcm_prepare(struct snd_pcm_substream *ss)
|
||||
{
|
||||
struct siu_info *info = siu_i2s_data;
|
||||
struct siu_port *port_info = siu_port_info(ss);
|
||||
struct device *dev = ss->pcm->card->dev;
|
||||
struct snd_pcm_runtime *rt = ss->runtime;
|
||||
struct siu_stream *siu_stream;
|
||||
snd_pcm_sframes_t xfer_cnt;
|
||||
|
||||
if (ss->stream == SNDRV_PCM_STREAM_PLAYBACK)
|
||||
siu_stream = &port_info->playback;
|
||||
else
|
||||
siu_stream = &port_info->capture;
|
||||
|
||||
rt = siu_stream->substream->runtime;
|
||||
|
||||
siu_stream->buf_bytes = snd_pcm_lib_buffer_bytes(ss);
|
||||
siu_stream->period_bytes = snd_pcm_lib_period_bytes(ss);
|
||||
|
||||
dev_dbg(dev, "%s: port=%d, %d channels, period=%u bytes\n", __func__,
|
||||
info->port_id, rt->channels, siu_stream->period_bytes);
|
||||
|
||||
/* We only support buffers that are multiples of the period */
|
||||
if (siu_stream->buf_bytes % siu_stream->period_bytes) {
|
||||
dev_err(dev, "%s() - buffer=%d not multiple of period=%d\n",
|
||||
__func__, siu_stream->buf_bytes,
|
||||
siu_stream->period_bytes);
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
xfer_cnt = bytes_to_frames(rt, siu_stream->period_bytes);
|
||||
if (!xfer_cnt || xfer_cnt > 0x1000000)
|
||||
return -EINVAL;
|
||||
|
||||
siu_stream->format = rt->format;
|
||||
siu_stream->xfer_cnt = xfer_cnt;
|
||||
|
||||
dev_dbg(dev, "port=%d buf=%lx buf_bytes=%d period_bytes=%d "
|
||||
"format=%d channels=%d xfer_cnt=%d\n", info->port_id,
|
||||
(unsigned long)rt->dma_addr, siu_stream->buf_bytes,
|
||||
siu_stream->period_bytes,
|
||||
siu_stream->format, rt->channels, (int)xfer_cnt);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int siu_pcm_trigger(struct snd_pcm_substream *ss, int cmd)
|
||||
{
|
||||
struct siu_info *info = siu_i2s_data;
|
||||
struct device *dev = ss->pcm->card->dev;
|
||||
struct siu_port *port_info = siu_port_info(ss);
|
||||
int ret;
|
||||
|
||||
dev_dbg(dev, "%s: port=%d@%p, cmd=%d\n", __func__,
|
||||
info->port_id, port_info, cmd);
|
||||
|
||||
switch (cmd) {
|
||||
case SNDRV_PCM_TRIGGER_START:
|
||||
if (ss->stream == SNDRV_PCM_STREAM_PLAYBACK)
|
||||
ret = siu_pcm_stmwrite_start(port_info);
|
||||
else
|
||||
ret = siu_pcm_stmread_start(port_info);
|
||||
|
||||
if (ret < 0)
|
||||
dev_warn(dev, "%s: start failed on port=%d\n",
|
||||
__func__, info->port_id);
|
||||
|
||||
break;
|
||||
case SNDRV_PCM_TRIGGER_STOP:
|
||||
if (ss->stream == SNDRV_PCM_STREAM_PLAYBACK)
|
||||
siu_pcm_stmwrite_stop(port_info);
|
||||
else
|
||||
siu_pcm_stmread_stop(port_info);
|
||||
ret = 0;
|
||||
|
||||
break;
|
||||
default:
|
||||
dev_err(dev, "%s() unsupported cmd=%d\n", __func__, cmd);
|
||||
ret = -EINVAL;
|
||||
}
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
/*
|
||||
* So far only resolution of one period is supported, subject to extending the
|
||||
* dmangine API
|
||||
*/
|
||||
static snd_pcm_uframes_t siu_pcm_pointer_dma(struct snd_pcm_substream *ss)
|
||||
{
|
||||
struct device *dev = ss->pcm->card->dev;
|
||||
struct siu_info *info = siu_i2s_data;
|
||||
u32 __iomem *base = info->reg;
|
||||
struct siu_port *port_info = siu_port_info(ss);
|
||||
struct snd_pcm_runtime *rt = ss->runtime;
|
||||
size_t ptr;
|
||||
struct siu_stream *siu_stream;
|
||||
|
||||
if (ss->stream == SNDRV_PCM_STREAM_PLAYBACK)
|
||||
siu_stream = &port_info->playback;
|
||||
else
|
||||
siu_stream = &port_info->capture;
|
||||
|
||||
/*
|
||||
* ptr is the offset into the buffer where the dma is currently at. We
|
||||
* check if the dma buffer has just wrapped.
|
||||
*/
|
||||
ptr = PERIOD_OFFSET(rt->dma_addr,
|
||||
siu_stream->cur_period,
|
||||
siu_stream->period_bytes) - rt->dma_addr;
|
||||
|
||||
dev_dbg(dev,
|
||||
"%s: port=%d, events %x, FSTS %x, xferred %u/%u, cookie %d\n",
|
||||
__func__, info->port_id, siu_read32(base + SIU_EVNTC),
|
||||
siu_read32(base + SIU_SBFSTS), ptr, siu_stream->buf_bytes,
|
||||
siu_stream->cookie);
|
||||
|
||||
if (ptr >= siu_stream->buf_bytes)
|
||||
ptr = 0;
|
||||
|
||||
return bytes_to_frames(ss->runtime, ptr);
|
||||
}
|
||||
|
||||
static int siu_pcm_new(struct snd_soc_pcm_runtime *rtd)
|
||||
{
|
||||
/* card->dev == socdev->dev, see snd_soc_new_pcms() */
|
||||
struct snd_card *card = rtd->card->snd_card;
|
||||
struct snd_pcm *pcm = rtd->pcm;
|
||||
struct siu_info *info = siu_i2s_data;
|
||||
struct platform_device *pdev = to_platform_device(card->dev);
|
||||
int ret;
|
||||
int i;
|
||||
|
||||
/* pdev->id selects between SIUA and SIUB */
|
||||
if (pdev->id < 0 || pdev->id >= SIU_PORT_NUM)
|
||||
return -EINVAL;
|
||||
|
||||
info->port_id = pdev->id;
|
||||
|
||||
/*
|
||||
* While the siu has 2 ports, only one port can be on at a time (only 1
|
||||
* SPB). So far all the boards using the siu had only one of the ports
|
||||
* wired to a codec. To simplify things, we only register one port with
|
||||
* alsa. In case both ports are needed, it should be changed here
|
||||
*/
|
||||
for (i = pdev->id; i < pdev->id + 1; i++) {
|
||||
struct siu_port **port_info = &siu_ports[i];
|
||||
|
||||
ret = siu_init_port(i, port_info, card);
|
||||
if (ret < 0)
|
||||
return ret;
|
||||
|
||||
ret = snd_pcm_lib_preallocate_pages_for_all(pcm,
|
||||
SNDRV_DMA_TYPE_DEV, NULL,
|
||||
SIU_BUFFER_BYTES_MAX, SIU_BUFFER_BYTES_MAX);
|
||||
if (ret < 0) {
|
||||
dev_err(card->dev,
|
||||
"snd_pcm_lib_preallocate_pages_for_all() err=%d",
|
||||
ret);
|
||||
goto fail;
|
||||
}
|
||||
|
||||
(*port_info)->pcm = pcm;
|
||||
|
||||
/* IO tasklets */
|
||||
tasklet_init(&(*port_info)->playback.tasklet, siu_io_tasklet,
|
||||
(unsigned long)&(*port_info)->playback);
|
||||
tasklet_init(&(*port_info)->capture.tasklet, siu_io_tasklet,
|
||||
(unsigned long)&(*port_info)->capture);
|
||||
}
|
||||
|
||||
dev_info(card->dev, "SuperH SIU driver initialized.\n");
|
||||
return 0;
|
||||
|
||||
fail:
|
||||
siu_free_port(siu_ports[pdev->id]);
|
||||
dev_err(card->dev, "SIU: failed to initialize.\n");
|
||||
return ret;
|
||||
}
|
||||
|
||||
static void siu_pcm_free(struct snd_pcm *pcm)
|
||||
{
|
||||
struct platform_device *pdev = to_platform_device(pcm->card->dev);
|
||||
struct siu_port *port_info = siu_ports[pdev->id];
|
||||
|
||||
tasklet_kill(&port_info->capture.tasklet);
|
||||
tasklet_kill(&port_info->playback.tasklet);
|
||||
|
||||
siu_free_port(port_info);
|
||||
snd_pcm_lib_preallocate_free_for_all(pcm);
|
||||
|
||||
dev_dbg(pcm->card->dev, "%s\n", __func__);
|
||||
}
|
||||
|
||||
static struct snd_pcm_ops siu_pcm_ops = {
|
||||
.open = siu_pcm_open,
|
||||
.close = siu_pcm_close,
|
||||
.ioctl = snd_pcm_lib_ioctl,
|
||||
.hw_params = siu_pcm_hw_params,
|
||||
.hw_free = siu_pcm_hw_free,
|
||||
.prepare = siu_pcm_prepare,
|
||||
.trigger = siu_pcm_trigger,
|
||||
.pointer = siu_pcm_pointer_dma,
|
||||
};
|
||||
|
||||
struct snd_soc_platform_driver siu_platform = {
|
||||
.ops = &siu_pcm_ops,
|
||||
.pcm_new = siu_pcm_new,
|
||||
.pcm_free = siu_pcm_free,
|
||||
};
|
||||
EXPORT_SYMBOL_GPL(siu_platform);
|
||||
412
sound/soc/sh/ssi.c
Normal file
412
sound/soc/sh/ssi.c
Normal file
|
|
@ -0,0 +1,412 @@
|
|||
/*
|
||||
* Serial Sound Interface (I2S) support for SH7760/SH7780
|
||||
*
|
||||
* Copyright (c) 2007 Manuel Lauss <mano@roarinelk.homelinux.net>
|
||||
*
|
||||
* licensed under the terms outlined in the file COPYING at the root
|
||||
* of the linux kernel sources.
|
||||
*
|
||||
* dont forget to set IPSEL/OMSEL register bits (in your board code) to
|
||||
* enable SSI output pins!
|
||||
*/
|
||||
|
||||
/*
|
||||
* LIMITATIONS:
|
||||
* The SSI unit has only one physical data line, so full duplex is
|
||||
* impossible. This can be remedied on the SH7760 by using the
|
||||
* other SSI unit for recording; however the SH7780 has only 1 SSI
|
||||
* unit, and its pins are shared with the AC97 unit, among others.
|
||||
*
|
||||
* FEATURES:
|
||||
* The SSI features "compressed mode": in this mode it continuously
|
||||
* streams PCM data over the I2S lines and uses LRCK as a handshake
|
||||
* signal. Can be used to send compressed data (AC3/DTS) to a DSP.
|
||||
* The number of bits sent over the wire in a frame can be adjusted
|
||||
* and can be independent from the actual sample bit depth. This is
|
||||
* useful to support TDM mode codecs like the AD1939 which have a
|
||||
* fixed TDM slot size, regardless of sample resolution.
|
||||
*/
|
||||
|
||||
#include <linux/init.h>
|
||||
#include <linux/module.h>
|
||||
#include <linux/platform_device.h>
|
||||
#include <sound/core.h>
|
||||
#include <sound/pcm.h>
|
||||
#include <sound/initval.h>
|
||||
#include <sound/soc.h>
|
||||
#include <asm/io.h>
|
||||
|
||||
#define SSICR 0x00
|
||||
#define SSISR 0x04
|
||||
|
||||
#define CR_DMAEN (1 << 28)
|
||||
#define CR_CHNL_SHIFT 22
|
||||
#define CR_CHNL_MASK (3 << CR_CHNL_SHIFT)
|
||||
#define CR_DWL_SHIFT 19
|
||||
#define CR_DWL_MASK (7 << CR_DWL_SHIFT)
|
||||
#define CR_SWL_SHIFT 16
|
||||
#define CR_SWL_MASK (7 << CR_SWL_SHIFT)
|
||||
#define CR_SCK_MASTER (1 << 15) /* bitclock master bit */
|
||||
#define CR_SWS_MASTER (1 << 14) /* wordselect master bit */
|
||||
#define CR_SCKP (1 << 13) /* I2Sclock polarity */
|
||||
#define CR_SWSP (1 << 12) /* LRCK polarity */
|
||||
#define CR_SPDP (1 << 11)
|
||||
#define CR_SDTA (1 << 10) /* i2s alignment (msb/lsb) */
|
||||
#define CR_PDTA (1 << 9) /* fifo data alignment */
|
||||
#define CR_DEL (1 << 8) /* delay data by 1 i2sclk */
|
||||
#define CR_BREN (1 << 7) /* clock gating in burst mode */
|
||||
#define CR_CKDIV_SHIFT 4
|
||||
#define CR_CKDIV_MASK (7 << CR_CKDIV_SHIFT) /* bitclock divider */
|
||||
#define CR_MUTE (1 << 3) /* SSI mute */
|
||||
#define CR_CPEN (1 << 2) /* compressed mode */
|
||||
#define CR_TRMD (1 << 1) /* transmit/receive select */
|
||||
#define CR_EN (1 << 0) /* enable SSI */
|
||||
|
||||
#define SSIREG(reg) (*(unsigned long *)(ssi->mmio + (reg)))
|
||||
|
||||
struct ssi_priv {
|
||||
unsigned long mmio;
|
||||
unsigned long sysclk;
|
||||
int inuse;
|
||||
} ssi_cpu_data[] = {
|
||||
#if defined(CONFIG_CPU_SUBTYPE_SH7760)
|
||||
{
|
||||
.mmio = 0xFE680000,
|
||||
},
|
||||
{
|
||||
.mmio = 0xFE690000,
|
||||
},
|
||||
#elif defined(CONFIG_CPU_SUBTYPE_SH7780)
|
||||
{
|
||||
.mmio = 0xFFE70000,
|
||||
},
|
||||
#else
|
||||
#error "Unsupported SuperH SoC"
|
||||
#endif
|
||||
};
|
||||
|
||||
/*
|
||||
* track usage of the SSI; it is simplex-only so prevent attempts of
|
||||
* concurrent playback + capture. FIXME: any locking required?
|
||||
*/
|
||||
static int ssi_startup(struct snd_pcm_substream *substream,
|
||||
struct snd_soc_dai *dai)
|
||||
{
|
||||
struct ssi_priv *ssi = &ssi_cpu_data[dai->id];
|
||||
if (ssi->inuse) {
|
||||
pr_debug("ssi: already in use!\n");
|
||||
return -EBUSY;
|
||||
} else
|
||||
ssi->inuse = 1;
|
||||
return 0;
|
||||
}
|
||||
|
||||
static void ssi_shutdown(struct snd_pcm_substream *substream,
|
||||
struct snd_soc_dai *dai)
|
||||
{
|
||||
struct ssi_priv *ssi = &ssi_cpu_data[dai->id];
|
||||
|
||||
ssi->inuse = 0;
|
||||
}
|
||||
|
||||
static int ssi_trigger(struct snd_pcm_substream *substream, int cmd,
|
||||
struct snd_soc_dai *dai)
|
||||
{
|
||||
struct ssi_priv *ssi = &ssi_cpu_data[dai->id];
|
||||
|
||||
switch (cmd) {
|
||||
case SNDRV_PCM_TRIGGER_START:
|
||||
SSIREG(SSICR) |= CR_DMAEN | CR_EN;
|
||||
break;
|
||||
case SNDRV_PCM_TRIGGER_STOP:
|
||||
SSIREG(SSICR) &= ~(CR_DMAEN | CR_EN);
|
||||
break;
|
||||
default:
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int ssi_hw_params(struct snd_pcm_substream *substream,
|
||||
struct snd_pcm_hw_params *params,
|
||||
struct snd_soc_dai *dai)
|
||||
{
|
||||
struct ssi_priv *ssi = &ssi_cpu_data[dai->id];
|
||||
unsigned long ssicr = SSIREG(SSICR);
|
||||
unsigned int bits, channels, swl, recv, i;
|
||||
|
||||
channels = params_channels(params);
|
||||
bits = params->msbits;
|
||||
recv = (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) ? 0 : 1;
|
||||
|
||||
pr_debug("ssi_hw_params() enter\nssicr was %08lx\n", ssicr);
|
||||
pr_debug("bits: %u channels: %u\n", bits, channels);
|
||||
|
||||
ssicr &= ~(CR_TRMD | CR_CHNL_MASK | CR_DWL_MASK | CR_PDTA |
|
||||
CR_SWL_MASK);
|
||||
|
||||
/* direction (send/receive) */
|
||||
if (!recv)
|
||||
ssicr |= CR_TRMD; /* transmit */
|
||||
|
||||
/* channels */
|
||||
if ((channels < 2) || (channels > 8) || (channels & 1)) {
|
||||
pr_debug("ssi: invalid number of channels\n");
|
||||
return -EINVAL;
|
||||
}
|
||||
ssicr |= ((channels >> 1) - 1) << CR_CHNL_SHIFT;
|
||||
|
||||
/* DATA WORD LENGTH (DWL): databits in audio sample */
|
||||
i = 0;
|
||||
switch (bits) {
|
||||
case 32: ++i;
|
||||
case 24: ++i;
|
||||
case 22: ++i;
|
||||
case 20: ++i;
|
||||
case 18: ++i;
|
||||
case 16: ++i;
|
||||
ssicr |= i << CR_DWL_SHIFT;
|
||||
case 8: break;
|
||||
default:
|
||||
pr_debug("ssi: invalid sample width\n");
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
/*
|
||||
* SYSTEM WORD LENGTH: size in bits of half a frame over the I2S
|
||||
* wires. This is usually bits_per_sample x channels/2; i.e. in
|
||||
* Stereo mode the SWL equals DWL. SWL can be bigger than the
|
||||
* product of (channels_per_slot x samplebits), e.g. for codecs
|
||||
* like the AD1939 which only accept 32bit wide TDM slots. For
|
||||
* "standard" I2S operation we set SWL = chans / 2 * DWL here.
|
||||
* Waiting for ASoC to get TDM support ;-)
|
||||
*/
|
||||
if ((bits > 16) && (bits <= 24)) {
|
||||
bits = 24; /* these are padded by the SSI */
|
||||
/*ssicr |= CR_PDTA;*/ /* cpu/data endianness ? */
|
||||
}
|
||||
i = 0;
|
||||
swl = (bits * channels) / 2;
|
||||
switch (swl) {
|
||||
case 256: ++i;
|
||||
case 128: ++i;
|
||||
case 64: ++i;
|
||||
case 48: ++i;
|
||||
case 32: ++i;
|
||||
case 16: ++i;
|
||||
ssicr |= i << CR_SWL_SHIFT;
|
||||
case 8: break;
|
||||
default:
|
||||
pr_debug("ssi: invalid system word length computed\n");
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
SSIREG(SSICR) = ssicr;
|
||||
|
||||
pr_debug("ssi_hw_params() leave\nssicr is now %08lx\n", ssicr);
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int ssi_set_sysclk(struct snd_soc_dai *cpu_dai, int clk_id,
|
||||
unsigned int freq, int dir)
|
||||
{
|
||||
struct ssi_priv *ssi = &ssi_cpu_data[cpu_dai->id];
|
||||
|
||||
ssi->sysclk = freq;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
/*
|
||||
* This divider is used to generate the SSI_SCK (I2S bitclock) from the
|
||||
* clock at the HAC_BIT_CLK ("oversampling clock") pin.
|
||||
*/
|
||||
static int ssi_set_clkdiv(struct snd_soc_dai *dai, int did, int div)
|
||||
{
|
||||
struct ssi_priv *ssi = &ssi_cpu_data[dai->id];
|
||||
unsigned long ssicr;
|
||||
int i;
|
||||
|
||||
i = 0;
|
||||
ssicr = SSIREG(SSICR) & ~CR_CKDIV_MASK;
|
||||
switch (div) {
|
||||
case 16: ++i;
|
||||
case 8: ++i;
|
||||
case 4: ++i;
|
||||
case 2: ++i;
|
||||
SSIREG(SSICR) = ssicr | (i << CR_CKDIV_SHIFT);
|
||||
case 1: break;
|
||||
default:
|
||||
pr_debug("ssi: invalid sck divider %d\n", div);
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int ssi_set_fmt(struct snd_soc_dai *dai, unsigned int fmt)
|
||||
{
|
||||
struct ssi_priv *ssi = &ssi_cpu_data[dai->id];
|
||||
unsigned long ssicr = SSIREG(SSICR);
|
||||
|
||||
pr_debug("ssi_set_fmt()\nssicr was 0x%08lx\n", ssicr);
|
||||
|
||||
ssicr &= ~(CR_DEL | CR_PDTA | CR_BREN | CR_SWSP | CR_SCKP |
|
||||
CR_SWS_MASTER | CR_SCK_MASTER);
|
||||
|
||||
switch (fmt & SND_SOC_DAIFMT_FORMAT_MASK) {
|
||||
case SND_SOC_DAIFMT_I2S:
|
||||
break;
|
||||
case SND_SOC_DAIFMT_RIGHT_J:
|
||||
ssicr |= CR_DEL | CR_PDTA;
|
||||
break;
|
||||
case SND_SOC_DAIFMT_LEFT_J:
|
||||
ssicr |= CR_DEL;
|
||||
break;
|
||||
default:
|
||||
pr_debug("ssi: unsupported format\n");
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
switch (fmt & SND_SOC_DAIFMT_CLOCK_MASK) {
|
||||
case SND_SOC_DAIFMT_CONT:
|
||||
break;
|
||||
case SND_SOC_DAIFMT_GATED:
|
||||
ssicr |= CR_BREN;
|
||||
break;
|
||||
}
|
||||
|
||||
switch (fmt & SND_SOC_DAIFMT_INV_MASK) {
|
||||
case SND_SOC_DAIFMT_NB_NF:
|
||||
ssicr |= CR_SCKP; /* sample data at low clkedge */
|
||||
break;
|
||||
case SND_SOC_DAIFMT_NB_IF:
|
||||
ssicr |= CR_SCKP | CR_SWSP;
|
||||
break;
|
||||
case SND_SOC_DAIFMT_IB_NF:
|
||||
break;
|
||||
case SND_SOC_DAIFMT_IB_IF:
|
||||
ssicr |= CR_SWSP; /* word select starts low */
|
||||
break;
|
||||
default:
|
||||
pr_debug("ssi: invalid inversion\n");
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
switch (fmt & SND_SOC_DAIFMT_MASTER_MASK) {
|
||||
case SND_SOC_DAIFMT_CBM_CFM:
|
||||
break;
|
||||
case SND_SOC_DAIFMT_CBS_CFM:
|
||||
ssicr |= CR_SCK_MASTER;
|
||||
break;
|
||||
case SND_SOC_DAIFMT_CBM_CFS:
|
||||
ssicr |= CR_SWS_MASTER;
|
||||
break;
|
||||
case SND_SOC_DAIFMT_CBS_CFS:
|
||||
ssicr |= CR_SWS_MASTER | CR_SCK_MASTER;
|
||||
break;
|
||||
default:
|
||||
pr_debug("ssi: invalid master/slave configuration\n");
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
SSIREG(SSICR) = ssicr;
|
||||
pr_debug("ssi_set_fmt() leave\nssicr is now 0x%08lx\n", ssicr);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
/* the SSI depends on an external clocksource (at HAC_BIT_CLK) even in
|
||||
* Master mode, so really this is board specific; the SSI can do any
|
||||
* rate with the right bitclk and divider settings.
|
||||
*/
|
||||
#define SSI_RATES \
|
||||
SNDRV_PCM_RATE_8000_192000
|
||||
|
||||
/* the SSI can do 8-32 bit samples, with 8 possible channels */
|
||||
#define SSI_FMTS \
|
||||
(SNDRV_PCM_FMTBIT_S8 | SNDRV_PCM_FMTBIT_U8 | \
|
||||
SNDRV_PCM_FMTBIT_S16_LE | SNDRV_PCM_FMTBIT_U16_LE | \
|
||||
SNDRV_PCM_FMTBIT_S20_3LE | SNDRV_PCM_FMTBIT_U20_3LE | \
|
||||
SNDRV_PCM_FMTBIT_S24_3LE | SNDRV_PCM_FMTBIT_U24_3LE | \
|
||||
SNDRV_PCM_FMTBIT_S32_LE | SNDRV_PCM_FMTBIT_U32_LE)
|
||||
|
||||
static const struct snd_soc_dai_ops ssi_dai_ops = {
|
||||
.startup = ssi_startup,
|
||||
.shutdown = ssi_shutdown,
|
||||
.trigger = ssi_trigger,
|
||||
.hw_params = ssi_hw_params,
|
||||
.set_sysclk = ssi_set_sysclk,
|
||||
.set_clkdiv = ssi_set_clkdiv,
|
||||
.set_fmt = ssi_set_fmt,
|
||||
};
|
||||
|
||||
static struct snd_soc_dai_driver sh4_ssi_dai[] = {
|
||||
{
|
||||
.name = "ssi-dai.0",
|
||||
.playback = {
|
||||
.rates = SSI_RATES,
|
||||
.formats = SSI_FMTS,
|
||||
.channels_min = 2,
|
||||
.channels_max = 8,
|
||||
},
|
||||
.capture = {
|
||||
.rates = SSI_RATES,
|
||||
.formats = SSI_FMTS,
|
||||
.channels_min = 2,
|
||||
.channels_max = 8,
|
||||
},
|
||||
.ops = &ssi_dai_ops,
|
||||
},
|
||||
#ifdef CONFIG_CPU_SUBTYPE_SH7760
|
||||
{
|
||||
.name = "ssi-dai.1",
|
||||
.playback = {
|
||||
.rates = SSI_RATES,
|
||||
.formats = SSI_FMTS,
|
||||
.channels_min = 2,
|
||||
.channels_max = 8,
|
||||
},
|
||||
.capture = {
|
||||
.rates = SSI_RATES,
|
||||
.formats = SSI_FMTS,
|
||||
.channels_min = 2,
|
||||
.channels_max = 8,
|
||||
},
|
||||
.ops = &ssi_dai_ops,
|
||||
},
|
||||
#endif
|
||||
};
|
||||
|
||||
static const struct snd_soc_component_driver sh4_ssi_component = {
|
||||
.name = "sh4-ssi",
|
||||
};
|
||||
|
||||
static int sh4_soc_dai_probe(struct platform_device *pdev)
|
||||
{
|
||||
return snd_soc_register_component(&pdev->dev, &sh4_ssi_component,
|
||||
sh4_ssi_dai, ARRAY_SIZE(sh4_ssi_dai));
|
||||
}
|
||||
|
||||
static int sh4_soc_dai_remove(struct platform_device *pdev)
|
||||
{
|
||||
snd_soc_unregister_component(&pdev->dev);
|
||||
return 0;
|
||||
}
|
||||
|
||||
static struct platform_driver sh4_ssi_driver = {
|
||||
.driver = {
|
||||
.name = "sh4-ssi-dai",
|
||||
.owner = THIS_MODULE,
|
||||
},
|
||||
|
||||
.probe = sh4_soc_dai_probe,
|
||||
.remove = sh4_soc_dai_remove,
|
||||
};
|
||||
|
||||
module_platform_driver(sh4_ssi_driver);
|
||||
|
||||
MODULE_LICENSE("GPL");
|
||||
MODULE_DESCRIPTION("SuperH onchip SSI (I2S) audio driver");
|
||||
MODULE_AUTHOR("Manuel Lauss <mano@roarinelk.homelinux.net>");
|
||||
Loading…
Add table
Add a link
Reference in a new issue