[funini.com] -> [kei@sodan] -> Kernel Reading

root/sound/soc/at32/at32-ssc.c

/* [<][>][^][v][top][bottom][index][help] */

DEFINITIONS

This source file includes following definitions.
  1. at32_ssc_interrupt
  2. at32_ssc_startup
  3. at32_ssc_shutdown
  4. at32_ssc_set_dai_sysclk
  5. at32_ssc_set_dai_fmt
  6. at32_ssc_set_dai_clkdiv
  7. at32_ssc_hw_params
  8. at32_ssc_prepare
  9. at32_ssc_suspend
  10. at32_ssc_resume

/* 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
 */
static int at32_ssc_set_dai_sysclk(struct snd_soc_dai *cpu_dai,
                                   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()
 */
static int at32_ssc_set_dai_fmt(struct snd_soc_dai *cpu_dai,
                                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()
 */
static int at32_ssc_set_dai_clkdiv(struct snd_soc_dai *cpu_dai,
                                   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,
                            struct snd_soc_dai *cpu_dai)
{
        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,
                           struct snd_soc_dai *cpu_dai)
{
        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)


struct snd_soc_dai at32_ssc_dai[NUM_SSC_DEVICES] = {
        {
         .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");

/* [<][>][^][v][top][bottom][index][help] */

[funini.com] -> [kei@sodan] -> Kernel Reading