2008-06-05 16:49:34 +04:00
|
|
|
/* sound/soc/at32/at32-ssc.c
|
|
|
|
* ASoC platform driver for AT32 using SSC as DAI
|
|
|
|
*
|
|
|
|
* Copyright (C) 2008 Long Range Systems
|
|
|
|
* Geoffrey Wossum <gwossum@acm.org>
|
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License version 2 as
|
|
|
|
* published by the Free Software Foundation.
|
|
|
|
*
|
|
|
|
* Note that this is basically a port of the sound/soc/at91-ssc.c to
|
|
|
|
* the AVR32 kernel. Thanks to Frank Mandarino for that code.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* #define DEBUG */
|
|
|
|
|
|
|
|
#include <linux/init.h>
|
|
|
|
#include <linux/module.h>
|
|
|
|
#include <linux/interrupt.h>
|
|
|
|
#include <linux/device.h>
|
|
|
|
#include <linux/delay.h>
|
|
|
|
#include <linux/clk.h>
|
|
|
|
#include <linux/io.h>
|
|
|
|
#include <linux/atmel_pdc.h>
|
|
|
|
#include <linux/atmel-ssc.h>
|
|
|
|
|
|
|
|
#include <sound/core.h>
|
|
|
|
#include <sound/pcm.h>
|
|
|
|
#include <sound/pcm_params.h>
|
|
|
|
#include <sound/initval.h>
|
|
|
|
#include <sound/soc.h>
|
|
|
|
|
|
|
|
#include "at32-pcm.h"
|
|
|
|
#include "at32-ssc.h"
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/*-------------------------------------------------------------------------*\
|
|
|
|
* Constants
|
|
|
|
\*-------------------------------------------------------------------------*/
|
|
|
|
#define NUM_SSC_DEVICES 3
|
|
|
|
|
|
|
|
/*
|
|
|
|
* SSC direction masks
|
|
|
|
*/
|
|
|
|
#define SSC_DIR_MASK_UNUSED 0
|
|
|
|
#define SSC_DIR_MASK_PLAYBACK 1
|
|
|
|
#define SSC_DIR_MASK_CAPTURE 2
|
|
|
|
|
|
|
|
/*
|
|
|
|
* SSC register values that Atmel left out of <linux/atmel-ssc.h>. These
|
|
|
|
* are expected to be used with SSC_BF
|
|
|
|
*/
|
|
|
|
/* START bit field values */
|
|
|
|
#define SSC_START_CONTINUOUS 0
|
|
|
|
#define SSC_START_TX_RX 1
|
|
|
|
#define SSC_START_LOW_RF 2
|
|
|
|
#define SSC_START_HIGH_RF 3
|
|
|
|
#define SSC_START_FALLING_RF 4
|
|
|
|
#define SSC_START_RISING_RF 5
|
|
|
|
#define SSC_START_LEVEL_RF 6
|
|
|
|
#define SSC_START_EDGE_RF 7
|
|
|
|
#define SSS_START_COMPARE_0 8
|
|
|
|
|
|
|
|
/* CKI bit field values */
|
|
|
|
#define SSC_CKI_FALLING 0
|
|
|
|
#define SSC_CKI_RISING 1
|
|
|
|
|
|
|
|
/* CKO bit field values */
|
|
|
|
#define SSC_CKO_NONE 0
|
|
|
|
#define SSC_CKO_CONTINUOUS 1
|
|
|
|
#define SSC_CKO_TRANSFER 2
|
|
|
|
|
|
|
|
/* CKS bit field values */
|
|
|
|
#define SSC_CKS_DIV 0
|
|
|
|
#define SSC_CKS_CLOCK 1
|
|
|
|
#define SSC_CKS_PIN 2
|
|
|
|
|
|
|
|
/* FSEDGE bit field values */
|
|
|
|
#define SSC_FSEDGE_POSITIVE 0
|
|
|
|
#define SSC_FSEDGE_NEGATIVE 1
|
|
|
|
|
|
|
|
/* FSOS bit field values */
|
|
|
|
#define SSC_FSOS_NONE 0
|
|
|
|
#define SSC_FSOS_NEGATIVE 1
|
|
|
|
#define SSC_FSOS_POSITIVE 2
|
|
|
|
#define SSC_FSOS_LOW 3
|
|
|
|
#define SSC_FSOS_HIGH 4
|
|
|
|
#define SSC_FSOS_TOGGLE 5
|
|
|
|
|
|
|
|
#define START_DELAY 1
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/*-------------------------------------------------------------------------*\
|
|
|
|
* Module data
|
|
|
|
\*-------------------------------------------------------------------------*/
|
|
|
|
/*
|
|
|
|
* SSC PDC registered required by the PCM DMA engine
|
|
|
|
*/
|
|
|
|
static struct at32_pdc_regs pdc_tx_reg = {
|
|
|
|
.xpr = SSC_PDC_TPR,
|
|
|
|
.xcr = SSC_PDC_TCR,
|
|
|
|
.xnpr = SSC_PDC_TNPR,
|
|
|
|
.xncr = SSC_PDC_TNCR,
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static struct at32_pdc_regs pdc_rx_reg = {
|
|
|
|
.xpr = SSC_PDC_RPR,
|
|
|
|
.xcr = SSC_PDC_RCR,
|
|
|
|
.xnpr = SSC_PDC_RNPR,
|
|
|
|
.xncr = SSC_PDC_RNCR,
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* SSC and PDC status bits for transmit and receive
|
|
|
|
*/
|
|
|
|
static struct at32_ssc_mask ssc_tx_mask = {
|
|
|
|
.ssc_enable = SSC_BIT(CR_TXEN),
|
|
|
|
.ssc_disable = SSC_BIT(CR_TXDIS),
|
|
|
|
.ssc_endx = SSC_BIT(SR_ENDTX),
|
|
|
|
.ssc_endbuf = SSC_BIT(SR_TXBUFE),
|
|
|
|
.pdc_enable = SSC_BIT(PDC_PTCR_TXTEN),
|
|
|
|
.pdc_disable = SSC_BIT(PDC_PTCR_TXTDIS),
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static struct at32_ssc_mask ssc_rx_mask = {
|
|
|
|
.ssc_enable = SSC_BIT(CR_RXEN),
|
|
|
|
.ssc_disable = SSC_BIT(CR_RXDIS),
|
|
|
|
.ssc_endx = SSC_BIT(SR_ENDRX),
|
|
|
|
.ssc_endbuf = SSC_BIT(SR_RXBUFF),
|
|
|
|
.pdc_enable = SSC_BIT(PDC_PTCR_RXTEN),
|
|
|
|
.pdc_disable = SSC_BIT(PDC_PTCR_RXTDIS),
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* DMA parameters for each SSC
|
|
|
|
*/
|
|
|
|
static struct at32_pcm_dma_params ssc_dma_params[NUM_SSC_DEVICES][2] = {
|
|
|
|
{
|
|
|
|
{
|
|
|
|
.name = "SSC0 PCM out",
|
|
|
|
.pdc = &pdc_tx_reg,
|
|
|
|
.mask = &ssc_tx_mask,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
.name = "SSC0 PCM in",
|
|
|
|
.pdc = &pdc_rx_reg,
|
|
|
|
.mask = &ssc_rx_mask,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
{
|
|
|
|
.name = "SSC1 PCM out",
|
|
|
|
.pdc = &pdc_tx_reg,
|
|
|
|
.mask = &ssc_tx_mask,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
.name = "SSC1 PCM in",
|
|
|
|
.pdc = &pdc_rx_reg,
|
|
|
|
.mask = &ssc_rx_mask,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
{
|
|
|
|
.name = "SSC2 PCM out",
|
|
|
|
.pdc = &pdc_tx_reg,
|
|
|
|
.mask = &ssc_tx_mask,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
.name = "SSC2 PCM in",
|
|
|
|
.pdc = &pdc_rx_reg,
|
|
|
|
.mask = &ssc_rx_mask,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static struct at32_ssc_info ssc_info[NUM_SSC_DEVICES] = {
|
|
|
|
{
|
|
|
|
.name = "ssc0",
|
|
|
|
.lock = __SPIN_LOCK_UNLOCKED(ssc_info[0].lock),
|
|
|
|
.dir_mask = SSC_DIR_MASK_UNUSED,
|
|
|
|
.initialized = 0,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
.name = "ssc1",
|
|
|
|
.lock = __SPIN_LOCK_UNLOCKED(ssc_info[1].lock),
|
|
|
|
.dir_mask = SSC_DIR_MASK_UNUSED,
|
|
|
|
.initialized = 0,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
.name = "ssc2",
|
|
|
|
.lock = __SPIN_LOCK_UNLOCKED(ssc_info[2].lock),
|
|
|
|
.dir_mask = SSC_DIR_MASK_UNUSED,
|
|
|
|
.initialized = 0,
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/*-------------------------------------------------------------------------*\
|
|
|
|
* ISR
|
|
|
|
\*-------------------------------------------------------------------------*/
|
|
|
|
/*
|
|
|
|
* SSC interrupt handler. Passes PDC interrupts to the DMA interrupt
|
|
|
|
* handler in the PCM driver.
|
|
|
|
*/
|
|
|
|
static irqreturn_t at32_ssc_interrupt(int irq, void *dev_id)
|
|
|
|
{
|
|
|
|
struct at32_ssc_info *ssc_p = dev_id;
|
|
|
|
struct at32_pcm_dma_params *dma_params;
|
|
|
|
u32 ssc_sr;
|
|
|
|
u32 ssc_substream_mask;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
ssc_sr = (ssc_readl(ssc_p->ssc->regs, SR) &
|
|
|
|
ssc_readl(ssc_p->ssc->regs, IMR));
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Loop through substreams attached to this SSC. If a DMA-related
|
|
|
|
* interrupt occured on that substream, call the DMA interrupt
|
|
|
|
* handler function, if one has been registered in the dma_param
|
|
|
|
* structure by the PCM driver.
|
|
|
|
*/
|
|
|
|
for (i = 0; i < ARRAY_SIZE(ssc_p->dma_params); i++) {
|
|
|
|
dma_params = ssc_p->dma_params[i];
|
|
|
|
|
|
|
|
if ((dma_params != NULL) &&
|
|
|
|
(dma_params->dma_intr_handler != NULL)) {
|
|
|
|
ssc_substream_mask = (dma_params->mask->ssc_endx |
|
|
|
|
dma_params->mask->ssc_endbuf);
|
|
|
|
if (ssc_sr & ssc_substream_mask) {
|
|
|
|
dma_params->dma_intr_handler(ssc_sr,
|
|
|
|
dma_params->
|
|
|
|
substream);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
return IRQ_HANDLED;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*-------------------------------------------------------------------------*\
|
|
|
|
* DAI functions
|
|
|
|
\*-------------------------------------------------------------------------*/
|
|
|
|
/*
|
|
|
|
* Startup. Only that one substream allowed in each direction.
|
|
|
|
*/
|
|
|
|
static int at32_ssc_startup(struct snd_pcm_substream *substream)
|
|
|
|
{
|
|
|
|
struct snd_soc_pcm_runtime *rtd = substream->private_data;
|
|
|
|
struct at32_ssc_info *ssc_p = &ssc_info[rtd->dai->cpu_dai->id];
|
|
|
|
int dir_mask;
|
|
|
|
|
|
|
|
dir_mask = ((substream->stream == SNDRV_PCM_STREAM_PLAYBACK) ?
|
|
|
|
SSC_DIR_MASK_PLAYBACK : SSC_DIR_MASK_CAPTURE);
|
|
|
|
|
|
|
|
spin_lock_irq(&ssc_p->lock);
|
|
|
|
if (ssc_p->dir_mask & dir_mask) {
|
|
|
|
spin_unlock_irq(&ssc_p->lock);
|
|
|
|
return -EBUSY;
|
|
|
|
}
|
|
|
|
ssc_p->dir_mask |= dir_mask;
|
|
|
|
spin_unlock_irq(&ssc_p->lock);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Shutdown. Clear DMA parameters and shutdown the SSC if there
|
|
|
|
* are no other substreams open.
|
|
|
|
*/
|
|
|
|
static void at32_ssc_shutdown(struct snd_pcm_substream *substream)
|
|
|
|
{
|
|
|
|
struct snd_soc_pcm_runtime *rtd = substream->private_data;
|
|
|
|
struct at32_ssc_info *ssc_p = &ssc_info[rtd->dai->cpu_dai->id];
|
|
|
|
struct at32_pcm_dma_params *dma_params;
|
|
|
|
int dir_mask;
|
|
|
|
|
|
|
|
dma_params = ssc_p->dma_params[substream->stream];
|
|
|
|
|
|
|
|
if (dma_params != NULL) {
|
|
|
|
ssc_writel(dma_params->ssc->regs, CR,
|
|
|
|
dma_params->mask->ssc_disable);
|
|
|
|
pr_debug("%s disabled SSC_SR=0x%08x\n",
|
|
|
|
(substream->stream ? "receiver" : "transmit"),
|
|
|
|
ssc_readl(ssc_p->ssc->regs, SR));
|
|
|
|
|
|
|
|
dma_params->ssc = NULL;
|
|
|
|
dma_params->substream = NULL;
|
|
|
|
ssc_p->dma_params[substream->stream] = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
dir_mask = 1 << substream->stream;
|
|
|
|
spin_lock_irq(&ssc_p->lock);
|
|
|
|
ssc_p->dir_mask &= ~dir_mask;
|
|
|
|
if (!ssc_p->dir_mask) {
|
|
|
|
/* Shutdown the SSC clock */
|
|
|
|
pr_debug("at32-ssc: Stopping user %d clock\n",
|
|
|
|
ssc_p->ssc->user);
|
|
|
|
clk_disable(ssc_p->ssc->clk);
|
|
|
|
|
|
|
|
if (ssc_p->initialized) {
|
|
|
|
free_irq(ssc_p->ssc->irq, ssc_p);
|
|
|
|
ssc_p->initialized = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Reset the SSC */
|
|
|
|
ssc_writel(ssc_p->ssc->regs, CR, SSC_BIT(CR_SWRST));
|
|
|
|
|
|
|
|
/* clear the SSC dividers */
|
|
|
|
ssc_p->cmr_div = 0;
|
|
|
|
ssc_p->tcmr_period = 0;
|
|
|
|
ssc_p->rcmr_period = 0;
|
|
|
|
}
|
|
|
|
spin_unlock_irq(&ssc_p->lock);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Set the SSC system clock rate
|
|
|
|
*/
|
2008-07-07 19:07:29 +04:00
|
|
|
static int at32_ssc_set_dai_sysclk(struct snd_soc_dai *cpu_dai,
|
2008-06-05 16:49:34 +04:00
|
|
|
int clk_id, unsigned int freq, int dir)
|
|
|
|
{
|
|
|
|
/* TODO: What the heck do I do here? */
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Record DAI format for use by hw_params()
|
|
|
|
*/
|
2008-07-07 19:07:29 +04:00
|
|
|
static int at32_ssc_set_dai_fmt(struct snd_soc_dai *cpu_dai,
|
2008-06-05 16:49:34 +04:00
|
|
|
unsigned int fmt)
|
|
|
|
{
|
|
|
|
struct at32_ssc_info *ssc_p = &ssc_info[cpu_dai->id];
|
|
|
|
|
|
|
|
ssc_p->daifmt = fmt;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Record SSC clock dividers for use in hw_params()
|
|
|
|
*/
|
2008-07-07 19:07:29 +04:00
|
|
|
static int at32_ssc_set_dai_clkdiv(struct snd_soc_dai *cpu_dai,
|
2008-06-05 16:49:34 +04:00
|
|
|
int div_id, int div)
|
|
|
|
{
|
|
|
|
struct at32_ssc_info *ssc_p = &ssc_info[cpu_dai->id];
|
|
|
|
|
|
|
|
switch (div_id) {
|
|
|
|
case AT32_SSC_CMR_DIV:
|
|
|
|
/*
|
|
|
|
* The same master clock divider is used for both
|
|
|
|
* transmit and receive, so if a value has already
|
|
|
|
* been set, it must match this value
|
|
|
|
*/
|
|
|
|
if (ssc_p->cmr_div == 0)
|
|
|
|
ssc_p->cmr_div = div;
|
|
|
|
else if (div != ssc_p->cmr_div)
|
|
|
|
return -EBUSY;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case AT32_SSC_TCMR_PERIOD:
|
|
|
|
ssc_p->tcmr_period = div;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case AT32_SSC_RCMR_PERIOD:
|
|
|
|
ssc_p->rcmr_period = div;
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Configure the SSC
|
|
|
|
*/
|
|
|
|
static int at32_ssc_hw_params(struct snd_pcm_substream *substream,
|
|
|
|
struct snd_pcm_hw_params *params)
|
|
|
|
{
|
|
|
|
struct snd_soc_pcm_runtime *rtd = substream->private_data;
|
|
|
|
int id = rtd->dai->cpu_dai->id;
|
|
|
|
struct at32_ssc_info *ssc_p = &ssc_info[id];
|
|
|
|
struct at32_pcm_dma_params *dma_params;
|
|
|
|
int channels, bits;
|
|
|
|
u32 tfmr, rfmr, tcmr, rcmr;
|
|
|
|
int start_event;
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Currently, there is only one set of dma_params for each direction.
|
|
|
|
* If more are added, this code will have to be changed to select
|
|
|
|
* the proper set
|
|
|
|
*/
|
|
|
|
dma_params = &ssc_dma_params[id][substream->stream];
|
|
|
|
dma_params->ssc = ssc_p->ssc;
|
|
|
|
dma_params->substream = substream;
|
|
|
|
|
|
|
|
ssc_p->dma_params[substream->stream] = dma_params;
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* The cpu_dai->dma_data field is only used to communicate the
|
|
|
|
* appropriate DMA parameters to the PCM driver's hw_params()
|
|
|
|
* function. It should not be used for other purposes as it
|
|
|
|
* is common to all substreams.
|
|
|
|
*/
|
|
|
|
rtd->dai->cpu_dai->dma_data = dma_params;
|
|
|
|
|
|
|
|
channels = params_channels(params);
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Determine sample size in bits and the PDC increment
|
|
|
|
*/
|
|
|
|
switch (params_format(params)) {
|
|
|
|
case SNDRV_PCM_FORMAT_S8:
|
|
|
|
bits = 8;
|
|
|
|
dma_params->pdc_xfer_size = 1;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case SNDRV_PCM_FORMAT_S16:
|
|
|
|
bits = 16;
|
|
|
|
dma_params->pdc_xfer_size = 2;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case SNDRV_PCM_FORMAT_S24:
|
|
|
|
bits = 24;
|
|
|
|
dma_params->pdc_xfer_size = 4;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case SNDRV_PCM_FORMAT_S32:
|
|
|
|
bits = 32;
|
|
|
|
dma_params->pdc_xfer_size = 4;
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
pr_warning("at32-ssc: Unsupported PCM format %d",
|
|
|
|
params_format(params));
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
pr_debug("at32-ssc: bits = %d, pdc_xfer_size = %d, channels = %d\n",
|
|
|
|
bits, dma_params->pdc_xfer_size, channels);
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* The SSC only supports up to 16-bit samples in I2S format, due
|
|
|
|
* to the size of the Frame Mode Register FSLEN field.
|
|
|
|
*/
|
|
|
|
if ((ssc_p->daifmt & SND_SOC_DAIFMT_FORMAT_MASK) == SND_SOC_DAIFMT_I2S)
|
|
|
|
if (bits > 16) {
|
|
|
|
pr_warning("at32-ssc: "
|
|
|
|
"sample size %d is too large for I2S\n",
|
|
|
|
bits);
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Compute the SSC register settings
|
|
|
|
*/
|
|
|
|
switch (ssc_p->daifmt & (SND_SOC_DAIFMT_FORMAT_MASK |
|
|
|
|
SND_SOC_DAIFMT_MASTER_MASK)) {
|
|
|
|
case SND_SOC_DAIFMT_I2S | SND_SOC_DAIFMT_CBS_CFS:
|
|
|
|
/*
|
|
|
|
* I2S format, SSC provides BCLK and LRS clocks.
|
|
|
|
*
|
|
|
|
* The SSC transmit and receive clocks are generated from the
|
|
|
|
* MCK divider, and the BCLK signal is output on the SSC TK line
|
|
|
|
*/
|
|
|
|
pr_debug("at32-ssc: SSC mode is I2S BCLK / FRAME master\n");
|
|
|
|
rcmr = (SSC_BF(RCMR_PERIOD, ssc_p->rcmr_period) |
|
|
|
|
SSC_BF(RCMR_STTDLY, START_DELAY) |
|
|
|
|
SSC_BF(RCMR_START, SSC_START_FALLING_RF) |
|
|
|
|
SSC_BF(RCMR_CKI, SSC_CKI_RISING) |
|
|
|
|
SSC_BF(RCMR_CKO, SSC_CKO_NONE) |
|
|
|
|
SSC_BF(RCMR_CKS, SSC_CKS_DIV));
|
|
|
|
|
|
|
|
rfmr = (SSC_BF(RFMR_FSEDGE, SSC_FSEDGE_POSITIVE) |
|
|
|
|
SSC_BF(RFMR_FSOS, SSC_FSOS_NEGATIVE) |
|
|
|
|
SSC_BF(RFMR_FSLEN, bits - 1) |
|
|
|
|
SSC_BF(RFMR_DATNB, channels - 1) |
|
|
|
|
SSC_BIT(RFMR_MSBF) | SSC_BF(RFMR_DATLEN, bits - 1));
|
|
|
|
|
|
|
|
tcmr = (SSC_BF(TCMR_PERIOD, ssc_p->tcmr_period) |
|
|
|
|
SSC_BF(TCMR_STTDLY, START_DELAY) |
|
|
|
|
SSC_BF(TCMR_START, SSC_START_FALLING_RF) |
|
|
|
|
SSC_BF(TCMR_CKI, SSC_CKI_FALLING) |
|
|
|
|
SSC_BF(TCMR_CKO, SSC_CKO_CONTINUOUS) |
|
|
|
|
SSC_BF(TCMR_CKS, SSC_CKS_DIV));
|
|
|
|
|
|
|
|
tfmr = (SSC_BF(TFMR_FSEDGE, SSC_FSEDGE_POSITIVE) |
|
|
|
|
SSC_BF(TFMR_FSOS, SSC_FSOS_NEGATIVE) |
|
|
|
|
SSC_BF(TFMR_FSLEN, bits - 1) |
|
|
|
|
SSC_BF(TFMR_DATNB, channels - 1) | SSC_BIT(TFMR_MSBF) |
|
|
|
|
SSC_BF(TFMR_DATLEN, bits - 1));
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
|
|
case SND_SOC_DAIFMT_I2S | SND_SOC_DAIFMT_CBM_CFM:
|
|
|
|
/*
|
|
|
|
* I2S format, CODEC supplies BCLK and LRC clock.
|
|
|
|
*
|
|
|
|
* The SSC transmit clock is obtained from the BCLK signal
|
|
|
|
* on the TK line, and the SSC receive clock is generated from
|
|
|
|
* the transmit clock.
|
|
|
|
*
|
|
|
|
* For single channel data, one sample is transferred on the
|
|
|
|
* falling edge of the LRC clock. For two channel data, one
|
|
|
|
* sample is transferred on both edges of the LRC clock.
|
|
|
|
*/
|
|
|
|
pr_debug("at32-ssc: SSC mode is I2S BCLK / FRAME slave\n");
|
|
|
|
start_event = ((channels == 1) ?
|
|
|
|
SSC_START_FALLING_RF : SSC_START_EDGE_RF);
|
|
|
|
|
|
|
|
rcmr = (SSC_BF(RCMR_STTDLY, START_DELAY) |
|
|
|
|
SSC_BF(RCMR_START, start_event) |
|
|
|
|
SSC_BF(RCMR_CKI, SSC_CKI_RISING) |
|
|
|
|
SSC_BF(RCMR_CKO, SSC_CKO_NONE) |
|
|
|
|
SSC_BF(RCMR_CKS, SSC_CKS_CLOCK));
|
|
|
|
|
|
|
|
rfmr = (SSC_BF(RFMR_FSEDGE, SSC_FSEDGE_POSITIVE) |
|
|
|
|
SSC_BF(RFMR_FSOS, SSC_FSOS_NONE) |
|
|
|
|
SSC_BIT(RFMR_MSBF) | SSC_BF(RFMR_DATLEN, bits - 1));
|
|
|
|
|
|
|
|
tcmr = (SSC_BF(TCMR_STTDLY, START_DELAY) |
|
|
|
|
SSC_BF(TCMR_START, start_event) |
|
|
|
|
SSC_BF(TCMR_CKI, SSC_CKI_FALLING) |
|
|
|
|
SSC_BF(TCMR_CKO, SSC_CKO_NONE) |
|
|
|
|
SSC_BF(TCMR_CKS, SSC_CKS_PIN));
|
|
|
|
|
|
|
|
tfmr = (SSC_BF(TFMR_FSEDGE, SSC_FSEDGE_POSITIVE) |
|
|
|
|
SSC_BF(TFMR_FSOS, SSC_FSOS_NONE) |
|
|
|
|
SSC_BIT(TFMR_MSBF) | SSC_BF(TFMR_DATLEN, bits - 1));
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
|
|
case SND_SOC_DAIFMT_DSP_A | SND_SOC_DAIFMT_CBS_CFS:
|
|
|
|
/*
|
|
|
|
* DSP/PCM Mode A format, SSC provides BCLK and LRC clocks.
|
|
|
|
*
|
|
|
|
* The SSC transmit and receive clocks are generated from the
|
|
|
|
* MCK divider, and the BCLK signal is output on the SSC TK line
|
|
|
|
*/
|
|
|
|
pr_debug("at32-ssc: SSC mode is DSP A BCLK / FRAME master\n");
|
|
|
|
rcmr = (SSC_BF(RCMR_PERIOD, ssc_p->rcmr_period) |
|
|
|
|
SSC_BF(RCMR_STTDLY, 1) |
|
|
|
|
SSC_BF(RCMR_START, SSC_START_RISING_RF) |
|
|
|
|
SSC_BF(RCMR_CKI, SSC_CKI_RISING) |
|
|
|
|
SSC_BF(RCMR_CKO, SSC_CKO_NONE) |
|
|
|
|
SSC_BF(RCMR_CKS, SSC_CKS_DIV));
|
|
|
|
|
|
|
|
rfmr = (SSC_BF(RFMR_FSEDGE, SSC_FSEDGE_POSITIVE) |
|
|
|
|
SSC_BF(RFMR_FSOS, SSC_FSOS_POSITIVE) |
|
|
|
|
SSC_BF(RFMR_DATNB, channels - 1) |
|
|
|
|
SSC_BIT(RFMR_MSBF) | SSC_BF(RFMR_DATLEN, bits - 1));
|
|
|
|
|
|
|
|
tcmr = (SSC_BF(TCMR_PERIOD, ssc_p->tcmr_period) |
|
|
|
|
SSC_BF(TCMR_STTDLY, 1) |
|
|
|
|
SSC_BF(TCMR_START, SSC_START_RISING_RF) |
|
|
|
|
SSC_BF(TCMR_CKI, SSC_CKI_RISING) |
|
|
|
|
SSC_BF(TCMR_CKO, SSC_CKO_CONTINUOUS) |
|
|
|
|
SSC_BF(TCMR_CKS, SSC_CKS_DIV));
|
|
|
|
|
|
|
|
tfmr = (SSC_BF(TFMR_FSEDGE, SSC_FSEDGE_POSITIVE) |
|
|
|
|
SSC_BF(TFMR_FSOS, SSC_FSOS_POSITIVE) |
|
|
|
|
SSC_BF(TFMR_DATNB, channels - 1) |
|
|
|
|
SSC_BIT(TFMR_MSBF) | SSC_BF(TFMR_DATLEN, bits - 1));
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
|
|
case SND_SOC_DAIFMT_DSP_A | SND_SOC_DAIFMT_CBM_CFM:
|
|
|
|
default:
|
|
|
|
pr_warning("at32-ssc: unsupported DAI format 0x%x\n",
|
|
|
|
ssc_p->daifmt);
|
|
|
|
return -EINVAL;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
pr_debug("at32-ssc: RCMR=%08x RFMR=%08x TCMR=%08x TFMR=%08x\n",
|
|
|
|
rcmr, rfmr, tcmr, tfmr);
|
|
|
|
|
|
|
|
|
|
|
|
if (!ssc_p->initialized) {
|
|
|
|
/* enable peripheral clock */
|
|
|
|
pr_debug("at32-ssc: Starting clock\n");
|
|
|
|
clk_enable(ssc_p->ssc->clk);
|
|
|
|
|
|
|
|
/* Reset the SSC and its PDC registers */
|
|
|
|
ssc_writel(ssc_p->ssc->regs, CR, SSC_BIT(CR_SWRST));
|
|
|
|
|
|
|
|
ssc_writel(ssc_p->ssc->regs, PDC_RPR, 0);
|
|
|
|
ssc_writel(ssc_p->ssc->regs, PDC_RCR, 0);
|
|
|
|
ssc_writel(ssc_p->ssc->regs, PDC_RNPR, 0);
|
|
|
|
ssc_writel(ssc_p->ssc->regs, PDC_RNCR, 0);
|
|
|
|
|
|
|
|
ssc_writel(ssc_p->ssc->regs, PDC_TPR, 0);
|
|
|
|
ssc_writel(ssc_p->ssc->regs, PDC_TCR, 0);
|
|
|
|
ssc_writel(ssc_p->ssc->regs, PDC_TNPR, 0);
|
|
|
|
ssc_writel(ssc_p->ssc->regs, PDC_TNCR, 0);
|
|
|
|
|
|
|
|
ret = request_irq(ssc_p->ssc->irq, at32_ssc_interrupt, 0,
|
|
|
|
ssc_p->name, ssc_p);
|
|
|
|
if (ret < 0) {
|
|
|
|
pr_warning("at32-ssc: request irq failed (%d)\n", ret);
|
|
|
|
pr_debug("at32-ssc: Stopping clock\n");
|
|
|
|
clk_disable(ssc_p->ssc->clk);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
ssc_p->initialized = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Set SSC clock mode register */
|
|
|
|
ssc_writel(ssc_p->ssc->regs, CMR, ssc_p->cmr_div);
|
|
|
|
|
|
|
|
/* set receive clock mode and format */
|
|
|
|
ssc_writel(ssc_p->ssc->regs, RCMR, rcmr);
|
|
|
|
ssc_writel(ssc_p->ssc->regs, RFMR, rfmr);
|
|
|
|
|
|
|
|
/* set transmit clock mode and format */
|
|
|
|
ssc_writel(ssc_p->ssc->regs, TCMR, tcmr);
|
|
|
|
ssc_writel(ssc_p->ssc->regs, TFMR, tfmr);
|
|
|
|
|
|
|
|
pr_debug("at32-ssc: SSC initialized\n");
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static int at32_ssc_prepare(struct snd_pcm_substream *substream)
|
|
|
|
{
|
|
|
|
struct snd_soc_pcm_runtime *rtd = substream->private_data;
|
|
|
|
struct at32_ssc_info *ssc_p = &ssc_info[rtd->dai->cpu_dai->id];
|
|
|
|
struct at32_pcm_dma_params *dma_params;
|
|
|
|
|
|
|
|
dma_params = ssc_p->dma_params[substream->stream];
|
|
|
|
|
|
|
|
ssc_writel(dma_params->ssc->regs, CR, dma_params->mask->ssc_enable);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef CONFIG_PM
|
|
|
|
static int at32_ssc_suspend(struct platform_device *pdev,
|
2008-07-07 19:07:29 +04:00
|
|
|
struct snd_soc_dai *cpu_dai)
|
2008-06-05 16:49:34 +04:00
|
|
|
{
|
|
|
|
struct at32_ssc_info *ssc_p;
|
|
|
|
|
|
|
|
if (!cpu_dai->active)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
ssc_p = &ssc_info[cpu_dai->id];
|
|
|
|
|
|
|
|
/* Save the status register before disabling transmit and receive */
|
|
|
|
ssc_p->ssc_state.ssc_sr = ssc_readl(ssc_p->ssc->regs, SR);
|
|
|
|
ssc_writel(ssc_p->ssc->regs, CR, SSC_BIT(CR_TXDIS) | SSC_BIT(CR_RXDIS));
|
|
|
|
|
|
|
|
/* Save the current interrupt mask, then disable unmasked interrupts */
|
|
|
|
ssc_p->ssc_state.ssc_imr = ssc_readl(ssc_p->ssc->regs, IMR);
|
|
|
|
ssc_writel(ssc_p->ssc->regs, IDR, ssc_p->ssc_state.ssc_imr);
|
|
|
|
|
|
|
|
ssc_p->ssc_state.ssc_cmr = ssc_readl(ssc_p->ssc->regs, CMR);
|
|
|
|
ssc_p->ssc_state.ssc_rcmr = ssc_readl(ssc_p->ssc->regs, RCMR);
|
|
|
|
ssc_p->ssc_state.ssc_rfmr = ssc_readl(ssc_p->ssc->regs, RFMR);
|
|
|
|
ssc_p->ssc_state.ssc_tcmr = ssc_readl(ssc_p->ssc->regs, TCMR);
|
|
|
|
ssc_p->ssc_state.ssc_tfmr = ssc_readl(ssc_p->ssc->regs, TFMR);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static int at32_ssc_resume(struct platform_device *pdev,
|
2008-07-07 19:07:29 +04:00
|
|
|
struct snd_soc_dai *cpu_dai)
|
2008-06-05 16:49:34 +04:00
|
|
|
{
|
|
|
|
struct at32_ssc_info *ssc_p;
|
|
|
|
u32 cr;
|
|
|
|
|
|
|
|
if (!cpu_dai->active)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
ssc_p = &ssc_info[cpu_dai->id];
|
|
|
|
|
|
|
|
/* restore SSC register settings */
|
|
|
|
ssc_writel(ssc_p->ssc->regs, TFMR, ssc_p->ssc_state.ssc_tfmr);
|
|
|
|
ssc_writel(ssc_p->ssc->regs, TCMR, ssc_p->ssc_state.ssc_tcmr);
|
|
|
|
ssc_writel(ssc_p->ssc->regs, RFMR, ssc_p->ssc_state.ssc_rfmr);
|
|
|
|
ssc_writel(ssc_p->ssc->regs, RCMR, ssc_p->ssc_state.ssc_rcmr);
|
|
|
|
ssc_writel(ssc_p->ssc->regs, CMR, ssc_p->ssc_state.ssc_cmr);
|
|
|
|
|
|
|
|
/* re-enable interrupts */
|
|
|
|
ssc_writel(ssc_p->ssc->regs, IER, ssc_p->ssc_state.ssc_imr);
|
|
|
|
|
|
|
|
/* Re-enable recieve and transmit as appropriate */
|
|
|
|
cr = 0;
|
|
|
|
cr |=
|
|
|
|
(ssc_p->ssc_state.ssc_sr & SSC_BIT(SR_RXEN)) ? SSC_BIT(CR_RXEN) : 0;
|
|
|
|
cr |=
|
|
|
|
(ssc_p->ssc_state.ssc_sr & SSC_BIT(SR_TXEN)) ? SSC_BIT(CR_TXEN) : 0;
|
|
|
|
ssc_writel(ssc_p->ssc->regs, CR, cr);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
#else /* CONFIG_PM */
|
|
|
|
# define at32_ssc_suspend NULL
|
|
|
|
# define at32_ssc_resume NULL
|
|
|
|
#endif /* CONFIG_PM */
|
|
|
|
|
|
|
|
|
|
|
|
#define AT32_SSC_RATES \
|
|
|
|
(SNDRV_PCM_RATE_8000 | SNDRV_PCM_RATE_11025 | SNDRV_PCM_RATE_16000 | \
|
|
|
|
SNDRV_PCM_RATE_22050 | SNDRV_PCM_RATE_32000 | SNDRV_PCM_RATE_44100 | \
|
|
|
|
SNDRV_PCM_RATE_48000 | SNDRV_PCM_RATE_88200 | SNDRV_PCM_RATE_96000)
|
|
|
|
|
|
|
|
|
|
|
|
#define AT32_SSC_FORMATS \
|
|
|
|
(SNDRV_PCM_FMTBIT_S8 | SNDRV_PCM_FMTBIT_S16 | \
|
|
|
|
SNDRV_PCM_FMTBIT_S24 | SNDRV_PCM_FMTBIT_S32)
|
|
|
|
|
|
|
|
|
2008-07-07 19:07:29 +04:00
|
|
|
struct snd_soc_dai at32_ssc_dai[NUM_SSC_DEVICES] = {
|
2008-06-05 16:49:34 +04:00
|
|
|
{
|
|
|
|
.name = "at32-ssc0",
|
|
|
|
.id = 0,
|
|
|
|
.type = SND_SOC_DAI_PCM,
|
|
|
|
.suspend = at32_ssc_suspend,
|
|
|
|
.resume = at32_ssc_resume,
|
|
|
|
.playback = {
|
|
|
|
.channels_min = 1,
|
|
|
|
.channels_max = 2,
|
|
|
|
.rates = AT32_SSC_RATES,
|
|
|
|
.formats = AT32_SSC_FORMATS,
|
|
|
|
},
|
|
|
|
.capture = {
|
|
|
|
.channels_min = 1,
|
|
|
|
.channels_max = 2,
|
|
|
|
.rates = AT32_SSC_RATES,
|
|
|
|
.formats = AT32_SSC_FORMATS,
|
|
|
|
},
|
|
|
|
.ops = {
|
|
|
|
.startup = at32_ssc_startup,
|
|
|
|
.shutdown = at32_ssc_shutdown,
|
|
|
|
.prepare = at32_ssc_prepare,
|
|
|
|
.hw_params = at32_ssc_hw_params,
|
|
|
|
},
|
|
|
|
.dai_ops = {
|
|
|
|
.set_sysclk = at32_ssc_set_dai_sysclk,
|
|
|
|
.set_fmt = at32_ssc_set_dai_fmt,
|
|
|
|
.set_clkdiv = at32_ssc_set_dai_clkdiv,
|
|
|
|
},
|
|
|
|
.private_data = &ssc_info[0],
|
|
|
|
},
|
|
|
|
{
|
|
|
|
.name = "at32-ssc1",
|
|
|
|
.id = 1,
|
|
|
|
.type = SND_SOC_DAI_PCM,
|
|
|
|
.suspend = at32_ssc_suspend,
|
|
|
|
.resume = at32_ssc_resume,
|
|
|
|
.playback = {
|
|
|
|
.channels_min = 1,
|
|
|
|
.channels_max = 2,
|
|
|
|
.rates = AT32_SSC_RATES,
|
|
|
|
.formats = AT32_SSC_FORMATS,
|
|
|
|
},
|
|
|
|
.capture = {
|
|
|
|
.channels_min = 1,
|
|
|
|
.channels_max = 2,
|
|
|
|
.rates = AT32_SSC_RATES,
|
|
|
|
.formats = AT32_SSC_FORMATS,
|
|
|
|
},
|
|
|
|
.ops = {
|
|
|
|
.startup = at32_ssc_startup,
|
|
|
|
.shutdown = at32_ssc_shutdown,
|
|
|
|
.prepare = at32_ssc_prepare,
|
|
|
|
.hw_params = at32_ssc_hw_params,
|
|
|
|
},
|
|
|
|
.dai_ops = {
|
|
|
|
.set_sysclk = at32_ssc_set_dai_sysclk,
|
|
|
|
.set_fmt = at32_ssc_set_dai_fmt,
|
|
|
|
.set_clkdiv = at32_ssc_set_dai_clkdiv,
|
|
|
|
},
|
|
|
|
.private_data = &ssc_info[1],
|
|
|
|
},
|
|
|
|
{
|
|
|
|
.name = "at32-ssc2",
|
|
|
|
.id = 2,
|
|
|
|
.type = SND_SOC_DAI_PCM,
|
|
|
|
.suspend = at32_ssc_suspend,
|
|
|
|
.resume = at32_ssc_resume,
|
|
|
|
.playback = {
|
|
|
|
.channels_min = 1,
|
|
|
|
.channels_max = 2,
|
|
|
|
.rates = AT32_SSC_RATES,
|
|
|
|
.formats = AT32_SSC_FORMATS,
|
|
|
|
},
|
|
|
|
.capture = {
|
|
|
|
.channels_min = 1,
|
|
|
|
.channels_max = 2,
|
|
|
|
.rates = AT32_SSC_RATES,
|
|
|
|
.formats = AT32_SSC_FORMATS,
|
|
|
|
},
|
|
|
|
.ops = {
|
|
|
|
.startup = at32_ssc_startup,
|
|
|
|
.shutdown = at32_ssc_shutdown,
|
|
|
|
.prepare = at32_ssc_prepare,
|
|
|
|
.hw_params = at32_ssc_hw_params,
|
|
|
|
},
|
|
|
|
.dai_ops = {
|
|
|
|
.set_sysclk = at32_ssc_set_dai_sysclk,
|
|
|
|
.set_fmt = at32_ssc_set_dai_fmt,
|
|
|
|
.set_clkdiv = at32_ssc_set_dai_clkdiv,
|
|
|
|
},
|
|
|
|
.private_data = &ssc_info[2],
|
|
|
|
},
|
|
|
|
};
|
|
|
|
EXPORT_SYMBOL_GPL(at32_ssc_dai);
|
|
|
|
|
|
|
|
|
|
|
|
MODULE_AUTHOR("Geoffrey Wossum <gwossum@acm.org>");
|
|
|
|
MODULE_DESCRIPTION("AT32 SSC ASoC Interface");
|
|
|
|
MODULE_LICENSE("GPL");
|