[ALSA] Revised AT32 ASoC Patch
Attached is a revised version of my patch to add AT32 to ASoC. This cleans most of the style issues associated with the previous patch. Also fixes an issue with the playpaq_wm8510.c code depending on a non-released patch to th AT32 portmux support. Patch is against 2.6.24.3.atmel.3 kernel, the latest AVR32 kernel Atmel has released, with the linux-2.6-asoc patches from when v2.6.24 was tagged also applied. [Fixed up minor checkpatch issues and updated for current kernels -- broonie] Signed-off-by: Geoffrey Wossum <gwossum@acm.org> Signed-off-by: Mark Brown <broonie@opensource.wolfsonmicro.com> Signed-off-by: Takashi Iwai <tiwai@suse.de>
This commit is contained in:
		
					parent
					
						
							
								f10485e798
							
						
					
				
			
			
				commit
				
					
						9aaca9683b
					
				
			
		
					 9 changed files with 2050 additions and 1 deletions
				
			
		| 
						 | 
				
			
			@ -22,6 +22,7 @@ config SND_SOC_AC97_BUS
 | 
			
		|||
	bool
 | 
			
		||||
 | 
			
		||||
# All the supported Soc's
 | 
			
		||||
source "sound/soc/at32/Kconfig"
 | 
			
		||||
source "sound/soc/at91/Kconfig"
 | 
			
		||||
source "sound/soc/pxa/Kconfig"
 | 
			
		||||
source "sound/soc/s3c24xx/Kconfig"
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -1,4 +1,5 @@
 | 
			
		|||
snd-soc-core-objs := soc-core.o soc-dapm.o
 | 
			
		||||
 | 
			
		||||
obj-$(CONFIG_SND_SOC)	+= snd-soc-core.o
 | 
			
		||||
obj-$(CONFIG_SND_SOC)	+= codecs/ at91/ pxa/ s3c24xx/ sh/ fsl/ davinci/ omap/
 | 
			
		||||
obj-$(CONFIG_SND_SOC)	+= codecs/ at32/ at91/ pxa/ s3c24xx/ sh/ fsl/ davinci/
 | 
			
		||||
obj-$(CONFIG_SND_SOC)	+= omap/
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
							
								
								
									
										34
									
								
								sound/soc/at32/Kconfig
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										34
									
								
								sound/soc/at32/Kconfig
									
										
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,34 @@
 | 
			
		|||
config SND_AT32_SOC
 | 
			
		||||
        tristate "SoC Audio for the Atmel AT32 System-on-a-Chip"
 | 
			
		||||
        depends on AVR32 && SND_SOC
 | 
			
		||||
        help
 | 
			
		||||
          Say Y or M if you want to add support for codecs attached to 
 | 
			
		||||
          the AT32 SSC interface.  You will also need to
 | 
			
		||||
          to select the audio interfaces to support below.
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
config SND_AT32_SOC_SSC
 | 
			
		||||
        tristate
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
config SND_AT32_SOC_PLAYPAQ
 | 
			
		||||
        tristate "SoC Audio support for PlayPaq with WM8510"
 | 
			
		||||
        depends on SND_AT32_SOC && BOARD_PLAYPAQ
 | 
			
		||||
        select SND_AT32_SOC_SSC
 | 
			
		||||
        select SND_SOC_WM8510
 | 
			
		||||
        help
 | 
			
		||||
          Say Y or M here if you want to add support for SoC audio
 | 
			
		||||
          on the LRS PlayPaq.
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
config SND_AT32_SOC_PLAYPAQ_SLAVE
 | 
			
		||||
        bool "Run CODEC on PlayPaq in slave mode"
 | 
			
		||||
        depends on SND_AT32_SOC_PLAYPAQ
 | 
			
		||||
        default n
 | 
			
		||||
        help
 | 
			
		||||
          Say Y if you want to run with the AT32 SSC generating the BCLK
 | 
			
		||||
          and FRAME signals on the PlayPaq.  Unless you want to play
 | 
			
		||||
          with the AT32 as the SSC master, you probably want to say N here,
 | 
			
		||||
          as this will give you better sound quality.
 | 
			
		||||
							
								
								
									
										11
									
								
								sound/soc/at32/Makefile
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								sound/soc/at32/Makefile
									
										
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,11 @@
 | 
			
		|||
# AT32 Platform Support
 | 
			
		||||
snd-soc-at32-objs := at32-pcm.o
 | 
			
		||||
snd-soc-at32-ssc-objs := at32-ssc.o
 | 
			
		||||
 | 
			
		||||
obj-$(CONFIG_SND_AT32_SOC) += snd-soc-at32.o
 | 
			
		||||
obj-$(CONFIG_SND_AT32_SOC_SSC) += snd-soc-at32-ssc.o
 | 
			
		||||
 | 
			
		||||
# AT32 Machine Support
 | 
			
		||||
snd-soc-playpaq-objs := playpaq_wm8510.o
 | 
			
		||||
 | 
			
		||||
obj-$(CONFIG_SND_AT32_SOC_PLAYPAQ) += snd-soc-playpaq.o
 | 
			
		||||
							
								
								
									
										491
									
								
								sound/soc/at32/at32-pcm.c
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										491
									
								
								sound/soc/at32/at32-pcm.c
									
										
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,491 @@
 | 
			
		|||
/* sound/soc/at32/at32-pcm.c
 | 
			
		||||
 * ASoC PCM interface for Atmel AT32 SoC
 | 
			
		||||
 *
 | 
			
		||||
 * 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-pcm.c to
 | 
			
		||||
 * the AVR32 kernel.  Thanks to Frank Mandarino for that code.
 | 
			
		||||
 */
 | 
			
		||||
 | 
			
		||||
#include <linux/module.h>
 | 
			
		||||
#include <linux/init.h>
 | 
			
		||||
#include <linux/platform_device.h>
 | 
			
		||||
#include <linux/slab.h>
 | 
			
		||||
#include <linux/dma-mapping.h>
 | 
			
		||||
#include <linux/atmel_pdc.h>
 | 
			
		||||
 | 
			
		||||
#include <sound/core.h>
 | 
			
		||||
#include <sound/pcm.h>
 | 
			
		||||
#include <sound/pcm_params.h>
 | 
			
		||||
#include <sound/soc.h>
 | 
			
		||||
 | 
			
		||||
#include "at32-pcm.h"
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
/*--------------------------------------------------------------------------*\
 | 
			
		||||
 * Hardware definition
 | 
			
		||||
\*--------------------------------------------------------------------------*/
 | 
			
		||||
/* TODO: These values were taken from the AT91 platform driver, check
 | 
			
		||||
 *	 them against real values for AT32
 | 
			
		||||
 */
 | 
			
		||||
static const struct snd_pcm_hardware at32_pcm_hardware = {
 | 
			
		||||
	.info = (SNDRV_PCM_INFO_MMAP |
 | 
			
		||||
		 SNDRV_PCM_INFO_MMAP_VALID |
 | 
			
		||||
		 SNDRV_PCM_INFO_INTERLEAVED |
 | 
			
		||||
		 SNDRV_PCM_INFO_BLOCK_TRANSFER |
 | 
			
		||||
		 SNDRV_PCM_INFO_PAUSE),
 | 
			
		||||
 | 
			
		||||
	.formats = SNDRV_PCM_FMTBIT_S16,
 | 
			
		||||
	.period_bytes_min = 32,
 | 
			
		||||
	.period_bytes_max = 8192,	/* 512 frames * 16 bytes / frame */
 | 
			
		||||
	.periods_min = 2,
 | 
			
		||||
	.periods_max = 1024,
 | 
			
		||||
	.buffer_bytes_max = 32 * 1024,
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
/*--------------------------------------------------------------------------*\
 | 
			
		||||
 * Data types
 | 
			
		||||
\*--------------------------------------------------------------------------*/
 | 
			
		||||
struct at32_runtime_data {
 | 
			
		||||
	struct at32_pcm_dma_params *params;
 | 
			
		||||
	dma_addr_t dma_buffer;	/* physical address of DMA buffer */
 | 
			
		||||
	dma_addr_t dma_buffer_end; /* first address beyond DMA buffer */
 | 
			
		||||
	size_t period_size;
 | 
			
		||||
 | 
			
		||||
	dma_addr_t period_ptr;	/* physical address of next period */
 | 
			
		||||
	int periods;		/* period index of period_ptr */
 | 
			
		||||
 | 
			
		||||
	/* Save PDC registers (for power management) */
 | 
			
		||||
	u32 pdc_xpr_save;
 | 
			
		||||
	u32 pdc_xcr_save;
 | 
			
		||||
	u32 pdc_xnpr_save;
 | 
			
		||||
	u32 pdc_xncr_save;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
/*--------------------------------------------------------------------------*\
 | 
			
		||||
 * Helper functions
 | 
			
		||||
\*--------------------------------------------------------------------------*/
 | 
			
		||||
static int at32_pcm_preallocate_dma_buffer(struct snd_pcm *pcm, int stream)
 | 
			
		||||
{
 | 
			
		||||
	struct snd_pcm_substream *substream = pcm->streams[stream].substream;
 | 
			
		||||
	struct snd_dma_buffer *dmabuf = &substream->dma_buffer;
 | 
			
		||||
	size_t size = at32_pcm_hardware.buffer_bytes_max;
 | 
			
		||||
 | 
			
		||||
	dmabuf->dev.type = SNDRV_DMA_TYPE_DEV;
 | 
			
		||||
	dmabuf->dev.dev = pcm->card->dev;
 | 
			
		||||
	dmabuf->private_data = NULL;
 | 
			
		||||
	dmabuf->area = dma_alloc_coherent(pcm->card->dev, size,
 | 
			
		||||
					  &dmabuf->addr, GFP_KERNEL);
 | 
			
		||||
	pr_debug("at32_pcm: preallocate_dma_buffer: "
 | 
			
		||||
		 "area=%p, addr=%p, size=%ld\n",
 | 
			
		||||
		 (void *)dmabuf->area, (void *)dmabuf->addr, size);
 | 
			
		||||
 | 
			
		||||
	if (!dmabuf->area)
 | 
			
		||||
		return -ENOMEM;
 | 
			
		||||
 | 
			
		||||
	dmabuf->bytes = size;
 | 
			
		||||
	return 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
/*--------------------------------------------------------------------------*\
 | 
			
		||||
 * ISR
 | 
			
		||||
\*--------------------------------------------------------------------------*/
 | 
			
		||||
static void at32_pcm_dma_irq(u32 ssc_sr, struct snd_pcm_substream *substream)
 | 
			
		||||
{
 | 
			
		||||
	struct snd_pcm_runtime *rtd = substream->runtime;
 | 
			
		||||
	struct at32_runtime_data *prtd = rtd->private_data;
 | 
			
		||||
	struct at32_pcm_dma_params *params = prtd->params;
 | 
			
		||||
	static int count;
 | 
			
		||||
 | 
			
		||||
	count++;
 | 
			
		||||
	if (ssc_sr & params->mask->ssc_endbuf) {
 | 
			
		||||
		pr_warning("at32-pcm: buffer %s on %s (SSC_SR=%#x, count=%d)\n",
 | 
			
		||||
			   substream->stream == SNDRV_PCM_STREAM_PLAYBACK ?
 | 
			
		||||
			   "underrun" : "overrun", params->name, ssc_sr, count);
 | 
			
		||||
 | 
			
		||||
		/* re-start the PDC */
 | 
			
		||||
		ssc_writex(params->ssc->regs, ATMEL_PDC_PTCR,
 | 
			
		||||
			   params->mask->pdc_disable);
 | 
			
		||||
		prtd->period_ptr += prtd->period_size;
 | 
			
		||||
		if (prtd->period_ptr >= prtd->dma_buffer_end)
 | 
			
		||||
			prtd->period_ptr = prtd->dma_buffer;
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
		ssc_writex(params->ssc->regs, params->pdc->xpr,
 | 
			
		||||
			   prtd->period_ptr);
 | 
			
		||||
		ssc_writex(params->ssc->regs, params->pdc->xcr,
 | 
			
		||||
			   prtd->period_size / params->pdc_xfer_size);
 | 
			
		||||
		ssc_writex(params->ssc->regs, ATMEL_PDC_PTCR,
 | 
			
		||||
			   params->mask->pdc_enable);
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
	if (ssc_sr & params->mask->ssc_endx) {
 | 
			
		||||
		/* Load the PDC next pointer and counter registers */
 | 
			
		||||
		prtd->period_ptr += prtd->period_size;
 | 
			
		||||
		if (prtd->period_ptr >= prtd->dma_buffer_end)
 | 
			
		||||
			prtd->period_ptr = prtd->dma_buffer;
 | 
			
		||||
		ssc_writex(params->ssc->regs, params->pdc->xnpr,
 | 
			
		||||
			   prtd->period_ptr);
 | 
			
		||||
		ssc_writex(params->ssc->regs, params->pdc->xncr,
 | 
			
		||||
			   prtd->period_size / params->pdc_xfer_size);
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
	snd_pcm_period_elapsed(substream);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
/*--------------------------------------------------------------------------*\
 | 
			
		||||
 * PCM operations
 | 
			
		||||
\*--------------------------------------------------------------------------*/
 | 
			
		||||
static int at32_pcm_hw_params(struct snd_pcm_substream *substream,
 | 
			
		||||
			      struct snd_pcm_hw_params *params)
 | 
			
		||||
{
 | 
			
		||||
	struct snd_pcm_runtime *runtime = substream->runtime;
 | 
			
		||||
	struct at32_runtime_data *prtd = runtime->private_data;
 | 
			
		||||
	struct snd_soc_pcm_runtime *rtd = substream->private_data;
 | 
			
		||||
 | 
			
		||||
	/* this may get called several times by oss emulation
 | 
			
		||||
	 * with different params
 | 
			
		||||
	 */
 | 
			
		||||
	snd_pcm_set_runtime_buffer(substream, &substream->dma_buffer);
 | 
			
		||||
	runtime->dma_bytes = params_buffer_bytes(params);
 | 
			
		||||
 | 
			
		||||
	prtd->params = rtd->dai->cpu_dai->dma_data;
 | 
			
		||||
	prtd->params->dma_intr_handler = at32_pcm_dma_irq;
 | 
			
		||||
 | 
			
		||||
	prtd->dma_buffer = runtime->dma_addr;
 | 
			
		||||
	prtd->dma_buffer_end = runtime->dma_addr + runtime->dma_bytes;
 | 
			
		||||
	prtd->period_size = params_period_bytes(params);
 | 
			
		||||
 | 
			
		||||
	pr_debug("hw_params: DMA for %s initialized "
 | 
			
		||||
		 "(dma_bytes=%ld, period_size=%ld)\n",
 | 
			
		||||
		 prtd->params->name, runtime->dma_bytes, prtd->period_size);
 | 
			
		||||
 | 
			
		||||
	return 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
static int at32_pcm_hw_free(struct snd_pcm_substream *substream)
 | 
			
		||||
{
 | 
			
		||||
	struct at32_runtime_data *prtd = substream->runtime->private_data;
 | 
			
		||||
	struct at32_pcm_dma_params *params = prtd->params;
 | 
			
		||||
 | 
			
		||||
	if (params != NULL) {
 | 
			
		||||
		ssc_writex(params->ssc->regs, SSC_PDC_PTCR,
 | 
			
		||||
			   params->mask->pdc_disable);
 | 
			
		||||
		prtd->params->dma_intr_handler = NULL;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	return 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
static int at32_pcm_prepare(struct snd_pcm_substream *substream)
 | 
			
		||||
{
 | 
			
		||||
	struct at32_runtime_data *prtd = substream->runtime->private_data;
 | 
			
		||||
	struct at32_pcm_dma_params *params = prtd->params;
 | 
			
		||||
 | 
			
		||||
	ssc_writex(params->ssc->regs, SSC_IDR,
 | 
			
		||||
		   params->mask->ssc_endx | params->mask->ssc_endbuf);
 | 
			
		||||
	ssc_writex(params->ssc->regs, ATMEL_PDC_PTCR,
 | 
			
		||||
		   params->mask->pdc_disable);
 | 
			
		||||
 | 
			
		||||
	return 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
static int at32_pcm_trigger(struct snd_pcm_substream *substream, int cmd)
 | 
			
		||||
{
 | 
			
		||||
	struct snd_pcm_runtime *rtd = substream->runtime;
 | 
			
		||||
	struct at32_runtime_data *prtd = rtd->private_data;
 | 
			
		||||
	struct at32_pcm_dma_params *params = prtd->params;
 | 
			
		||||
	int ret = 0;
 | 
			
		||||
 | 
			
		||||
	pr_debug("at32_pcm_trigger: buffer_size = %ld, "
 | 
			
		||||
		 "dma_area = %p, dma_bytes = %ld\n",
 | 
			
		||||
		 rtd->buffer_size, rtd->dma_area, rtd->dma_bytes);
 | 
			
		||||
 | 
			
		||||
	switch (cmd) {
 | 
			
		||||
	case SNDRV_PCM_TRIGGER_START:
 | 
			
		||||
		prtd->period_ptr = prtd->dma_buffer;
 | 
			
		||||
 | 
			
		||||
		ssc_writex(params->ssc->regs, params->pdc->xpr,
 | 
			
		||||
			   prtd->period_ptr);
 | 
			
		||||
		ssc_writex(params->ssc->regs, params->pdc->xcr,
 | 
			
		||||
			   prtd->period_size / params->pdc_xfer_size);
 | 
			
		||||
 | 
			
		||||
		prtd->period_ptr += prtd->period_size;
 | 
			
		||||
		ssc_writex(params->ssc->regs, params->pdc->xnpr,
 | 
			
		||||
			   prtd->period_ptr);
 | 
			
		||||
		ssc_writex(params->ssc->regs, params->pdc->xncr,
 | 
			
		||||
			   prtd->period_size / params->pdc_xfer_size);
 | 
			
		||||
 | 
			
		||||
		pr_debug("trigger: period_ptr=%lx, xpr=%x, "
 | 
			
		||||
			 "xcr=%d, xnpr=%x, xncr=%d\n",
 | 
			
		||||
			 (unsigned long)prtd->period_ptr,
 | 
			
		||||
			 ssc_readx(params->ssc->regs, params->pdc->xpr),
 | 
			
		||||
			 ssc_readx(params->ssc->regs, params->pdc->xcr),
 | 
			
		||||
			 ssc_readx(params->ssc->regs, params->pdc->xnpr),
 | 
			
		||||
			 ssc_readx(params->ssc->regs, params->pdc->xncr));
 | 
			
		||||
 | 
			
		||||
		ssc_writex(params->ssc->regs, SSC_IER,
 | 
			
		||||
			   params->mask->ssc_endx | params->mask->ssc_endbuf);
 | 
			
		||||
		ssc_writex(params->ssc->regs, SSC_PDC_PTCR,
 | 
			
		||||
			   params->mask->pdc_enable);
 | 
			
		||||
 | 
			
		||||
		pr_debug("sr=%x, imr=%x\n",
 | 
			
		||||
			 ssc_readx(params->ssc->regs, SSC_SR),
 | 
			
		||||
			 ssc_readx(params->ssc->regs, SSC_IER));
 | 
			
		||||
		break;		/* SNDRV_PCM_TRIGGER_START */
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
	case SNDRV_PCM_TRIGGER_STOP:
 | 
			
		||||
	case SNDRV_PCM_TRIGGER_SUSPEND:
 | 
			
		||||
	case SNDRV_PCM_TRIGGER_PAUSE_PUSH:
 | 
			
		||||
		ssc_writex(params->ssc->regs, ATMEL_PDC_PTCR,
 | 
			
		||||
			   params->mask->pdc_disable);
 | 
			
		||||
		break;
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
	case SNDRV_PCM_TRIGGER_RESUME:
 | 
			
		||||
	case SNDRV_PCM_TRIGGER_PAUSE_RELEASE:
 | 
			
		||||
		ssc_writex(params->ssc->regs, ATMEL_PDC_PTCR,
 | 
			
		||||
			   params->mask->pdc_enable);
 | 
			
		||||
		break;
 | 
			
		||||
 | 
			
		||||
	default:
 | 
			
		||||
		ret = -EINVAL;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	return ret;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
static snd_pcm_uframes_t at32_pcm_pointer(struct snd_pcm_substream *substream)
 | 
			
		||||
{
 | 
			
		||||
	struct snd_pcm_runtime *runtime = substream->runtime;
 | 
			
		||||
	struct at32_runtime_data *prtd = runtime->private_data;
 | 
			
		||||
	struct at32_pcm_dma_params *params = prtd->params;
 | 
			
		||||
	dma_addr_t ptr;
 | 
			
		||||
	snd_pcm_uframes_t x;
 | 
			
		||||
 | 
			
		||||
	ptr = (dma_addr_t) ssc_readx(params->ssc->regs, params->pdc->xpr);
 | 
			
		||||
	x = bytes_to_frames(runtime, ptr - prtd->dma_buffer);
 | 
			
		||||
 | 
			
		||||
	if (x == runtime->buffer_size)
 | 
			
		||||
		x = 0;
 | 
			
		||||
 | 
			
		||||
	return x;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
static int at32_pcm_open(struct snd_pcm_substream *substream)
 | 
			
		||||
{
 | 
			
		||||
	struct snd_pcm_runtime *runtime = substream->runtime;
 | 
			
		||||
	struct at32_runtime_data *prtd;
 | 
			
		||||
	int ret = 0;
 | 
			
		||||
 | 
			
		||||
	snd_soc_set_runtime_hwparams(substream, &at32_pcm_hardware);
 | 
			
		||||
 | 
			
		||||
	/* ensure that buffer size is a multiple of period size */
 | 
			
		||||
	ret = snd_pcm_hw_constraint_integer(runtime,
 | 
			
		||||
					    SNDRV_PCM_HW_PARAM_PERIODS);
 | 
			
		||||
	if (ret < 0)
 | 
			
		||||
		goto out;
 | 
			
		||||
 | 
			
		||||
	prtd = kzalloc(sizeof(*prtd), GFP_KERNEL);
 | 
			
		||||
	if (prtd == NULL) {
 | 
			
		||||
		ret = -ENOMEM;
 | 
			
		||||
		goto out;
 | 
			
		||||
	}
 | 
			
		||||
	runtime->private_data = prtd;
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
out:
 | 
			
		||||
	return ret;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
static int at32_pcm_close(struct snd_pcm_substream *substream)
 | 
			
		||||
{
 | 
			
		||||
	struct at32_runtime_data *prtd = substream->runtime->private_data;
 | 
			
		||||
 | 
			
		||||
	kfree(prtd);
 | 
			
		||||
	return 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
static int at32_pcm_mmap(struct snd_pcm_substream *substream,
 | 
			
		||||
			 struct vm_area_struct *vma)
 | 
			
		||||
{
 | 
			
		||||
	return remap_pfn_range(vma, vma->vm_start,
 | 
			
		||||
			       substream->dma_buffer.addr >> PAGE_SHIFT,
 | 
			
		||||
			       vma->vm_end - vma->vm_start, vma->vm_page_prot);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
static struct snd_pcm_ops at32_pcm_ops = {
 | 
			
		||||
	.open = at32_pcm_open,
 | 
			
		||||
	.close = at32_pcm_close,
 | 
			
		||||
	.ioctl = snd_pcm_lib_ioctl,
 | 
			
		||||
	.hw_params = at32_pcm_hw_params,
 | 
			
		||||
	.hw_free = at32_pcm_hw_free,
 | 
			
		||||
	.prepare = at32_pcm_prepare,
 | 
			
		||||
	.trigger = at32_pcm_trigger,
 | 
			
		||||
	.pointer = at32_pcm_pointer,
 | 
			
		||||
	.mmap = at32_pcm_mmap,
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
/*--------------------------------------------------------------------------*\
 | 
			
		||||
 * ASoC platform driver
 | 
			
		||||
\*--------------------------------------------------------------------------*/
 | 
			
		||||
static u64 at32_pcm_dmamask = 0xffffffff;
 | 
			
		||||
 | 
			
		||||
static int at32_pcm_new(struct snd_card *card,
 | 
			
		||||
			struct snd_soc_codec_dai *dai,
 | 
			
		||||
			struct snd_pcm *pcm)
 | 
			
		||||
{
 | 
			
		||||
	int ret = 0;
 | 
			
		||||
 | 
			
		||||
	if (!card->dev->dma_mask)
 | 
			
		||||
		card->dev->dma_mask = &at32_pcm_dmamask;
 | 
			
		||||
	if (!card->dev->coherent_dma_mask)
 | 
			
		||||
		card->dev->coherent_dma_mask = 0xffffffff;
 | 
			
		||||
 | 
			
		||||
	if (dai->playback.channels_min) {
 | 
			
		||||
		ret = at32_pcm_preallocate_dma_buffer(
 | 
			
		||||
			  pcm, SNDRV_PCM_STREAM_PLAYBACK);
 | 
			
		||||
		if (ret)
 | 
			
		||||
			goto out;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	if (dai->capture.channels_min) {
 | 
			
		||||
		pr_debug("at32-pcm: Allocating PCM capture DMA buffer\n");
 | 
			
		||||
		ret = at32_pcm_preallocate_dma_buffer(
 | 
			
		||||
			  pcm, SNDRV_PCM_STREAM_CAPTURE);
 | 
			
		||||
		if (ret)
 | 
			
		||||
			goto out;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
out:
 | 
			
		||||
	return ret;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
static void at32_pcm_free_dma_buffers(struct snd_pcm *pcm)
 | 
			
		||||
{
 | 
			
		||||
	struct snd_pcm_substream *substream;
 | 
			
		||||
	struct snd_dma_buffer *buf;
 | 
			
		||||
	int stream;
 | 
			
		||||
 | 
			
		||||
	for (stream = 0; stream < 2; stream++) {
 | 
			
		||||
		substream = pcm->streams[stream].substream;
 | 
			
		||||
		if (substream == NULL)
 | 
			
		||||
			continue;
 | 
			
		||||
 | 
			
		||||
		buf = &substream->dma_buffer;
 | 
			
		||||
		if (!buf->area)
 | 
			
		||||
			continue;
 | 
			
		||||
		dma_free_coherent(pcm->card->dev, buf->bytes,
 | 
			
		||||
				  buf->area, buf->addr);
 | 
			
		||||
		buf->area = NULL;
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
#ifdef CONFIG_PM
 | 
			
		||||
static int at32_pcm_suspend(struct platform_device *pdev,
 | 
			
		||||
			    struct snd_soc_cpu_dai *dai)
 | 
			
		||||
{
 | 
			
		||||
	struct snd_pcm_runtime *runtime = dai->runtime;
 | 
			
		||||
	struct at32_runtime_data *prtd;
 | 
			
		||||
	struct at32_pcm_dma_params *params;
 | 
			
		||||
 | 
			
		||||
	if (runtime == NULL)
 | 
			
		||||
		return 0;
 | 
			
		||||
	prtd = runtime->private_data;
 | 
			
		||||
	params = prtd->params;
 | 
			
		||||
 | 
			
		||||
	/* Disable the PDC and save the PDC registers */
 | 
			
		||||
	ssc_writex(params->ssc->regs, PDC_PTCR, params->mask->pdc_disable);
 | 
			
		||||
 | 
			
		||||
	prtd->pdc_xpr_save = ssc_readx(params->ssc->regs, params->pdc->xpr);
 | 
			
		||||
	prtd->pdc_xcr_save = ssc_readx(params->ssc->regs, params->pdc->xcr);
 | 
			
		||||
	prtd->pdc_xnpr_save = ssc_readx(params->ssc->regs, params->pdc->xnpr);
 | 
			
		||||
	prtd->pdc_xncr_save = ssc_readx(params->ssc->regs, params->pdc->xncr);
 | 
			
		||||
 | 
			
		||||
	return 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
static int at32_pcm_resume(struct platform_device *pdev,
 | 
			
		||||
			   struct snd_soc_cpu_dai *dai)
 | 
			
		||||
{
 | 
			
		||||
	struct snd_pcm_runtime *runtime = dai->runtime;
 | 
			
		||||
	struct at32_runtime_data *prtd;
 | 
			
		||||
	struct at32_pcm_dma_params *params;
 | 
			
		||||
 | 
			
		||||
	if (runtime == NULL)
 | 
			
		||||
		return 0;
 | 
			
		||||
	prtd = runtime->private_data;
 | 
			
		||||
	params = prtd->params;
 | 
			
		||||
 | 
			
		||||
	/* Restore the PDC registers and enable the PDC */
 | 
			
		||||
	ssc_writex(params->ssc->regs, params->pdc->xpr, prtd->pdc_xpr_save);
 | 
			
		||||
	ssc_writex(params->ssc->regs, params->pdc->xcr, prtd->pdc_xcr_save);
 | 
			
		||||
	ssc_writex(params->ssc->regs, params->pdc->xnpr, prtd->pdc_xnpr_save);
 | 
			
		||||
	ssc_writex(params->ssc->regs, params->pdc->xncr, prtd->pdc_xncr_save);
 | 
			
		||||
 | 
			
		||||
	ssc_writex(params->ssc->regs, PDC_PTCR, params->mask->pdc_enable);
 | 
			
		||||
	return 0;
 | 
			
		||||
}
 | 
			
		||||
#else /* CONFIG_PM */
 | 
			
		||||
#  define at32_pcm_suspend	NULL
 | 
			
		||||
#  define at32_pcm_resume	NULL
 | 
			
		||||
#endif /* CONFIG_PM */
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
struct snd_soc_platform at32_soc_platform = {
 | 
			
		||||
	.name = "at32-audio",
 | 
			
		||||
	.pcm_ops = &at32_pcm_ops,
 | 
			
		||||
	.pcm_new = at32_pcm_new,
 | 
			
		||||
	.pcm_free = at32_pcm_free_dma_buffers,
 | 
			
		||||
	.suspend = at32_pcm_suspend,
 | 
			
		||||
	.resume = at32_pcm_resume,
 | 
			
		||||
};
 | 
			
		||||
EXPORT_SYMBOL_GPL(at32_soc_platform);
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
MODULE_AUTHOR("Geoffrey Wossum <gwossum@acm.org>");
 | 
			
		||||
MODULE_DESCRIPTION("Atmel AT32 PCM module");
 | 
			
		||||
MODULE_LICENSE("GPL");
 | 
			
		||||
							
								
								
									
										79
									
								
								sound/soc/at32/at32-pcm.h
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										79
									
								
								sound/soc/at32/at32-pcm.h
									
										
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,79 @@
 | 
			
		|||
/* sound/soc/at32/at32-pcm.h
 | 
			
		||||
 * ASoC PCM interface for Atmel AT32 SoC
 | 
			
		||||
 *
 | 
			
		||||
 * 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.
 | 
			
		||||
 */
 | 
			
		||||
 | 
			
		||||
#ifndef __SOUND_SOC_AT32_AT32_PCM_H
 | 
			
		||||
#define __SOUND_SOC_AT32_AT32_PCM_H __FILE__
 | 
			
		||||
 | 
			
		||||
#include <linux/atmel-ssc.h>
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
/*
 | 
			
		||||
 * Registers and status bits that are required by the PCM driver
 | 
			
		||||
 * TODO: Is ptcr really used?
 | 
			
		||||
 */
 | 
			
		||||
struct at32_pdc_regs {
 | 
			
		||||
	u32 xpr;		/* PDC RX/TX pointer */
 | 
			
		||||
	u32 xcr;		/* PDC RX/TX counter */
 | 
			
		||||
	u32 xnpr;		/* PDC next RX/TX pointer */
 | 
			
		||||
	u32 xncr;		/* PDC next RX/TX counter */
 | 
			
		||||
	u32 ptcr;		/* PDC transfer control */
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
/*
 | 
			
		||||
 * SSC mask info
 | 
			
		||||
 */
 | 
			
		||||
struct at32_ssc_mask {
 | 
			
		||||
	u32 ssc_enable;		/* SSC RX/TX enable */
 | 
			
		||||
	u32 ssc_disable;	/* SSC RX/TX disable */
 | 
			
		||||
	u32 ssc_endx;		/* SSC ENDTX or ENDRX */
 | 
			
		||||
	u32 ssc_endbuf;		/* SSC TXBUFF or RXBUFF */
 | 
			
		||||
	u32 pdc_enable;		/* PDC RX/TX enable */
 | 
			
		||||
	u32 pdc_disable;	/* PDC RX/TX disable */
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
/*
 | 
			
		||||
 * This structure, shared between the PCM driver and the interface,
 | 
			
		||||
 * contains all information required by the PCM driver to perform the
 | 
			
		||||
 * PDC DMA operation.  All fields except dma_intr_handler() are initialized
 | 
			
		||||
 * by the interface.  The dms_intr_handler() pointer is set by the PCM
 | 
			
		||||
 * driver and called by the interface SSC interrupt handler if it is
 | 
			
		||||
 * non-NULL.
 | 
			
		||||
 */
 | 
			
		||||
struct at32_pcm_dma_params {
 | 
			
		||||
	char *name;		/* stream identifier */
 | 
			
		||||
	int pdc_xfer_size;	/* PDC counter increment in bytes */
 | 
			
		||||
	struct ssc_device *ssc;	/* SSC device for stream */
 | 
			
		||||
	struct at32_pdc_regs *pdc;	/* PDC register info */
 | 
			
		||||
	struct at32_ssc_mask *mask;	/* SSC mask info */
 | 
			
		||||
	struct snd_pcm_substream *substream;
 | 
			
		||||
	void (*dma_intr_handler) (u32, struct snd_pcm_substream *);
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
/*
 | 
			
		||||
 * The AT32 ASoC platform driver
 | 
			
		||||
 */
 | 
			
		||||
extern struct snd_soc_platform at32_soc_platform;
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
/*
 | 
			
		||||
 * SSC register access (since ssc_writel() / ssc_readl() require literal name)
 | 
			
		||||
 */
 | 
			
		||||
#define ssc_readx(base, reg)            (__raw_readl((base) + (reg)))
 | 
			
		||||
#define ssc_writex(base, reg, value)    __raw_writel((value), (base) + (reg))
 | 
			
		||||
 | 
			
		||||
#endif /* __SOUND_SOC_AT32_AT32_PCM_H */
 | 
			
		||||
							
								
								
									
										849
									
								
								sound/soc/at32/at32-ssc.c
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										849
									
								
								sound/soc/at32/at32-ssc.c
									
										
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,849 @@
 | 
			
		|||
/* 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_cpu_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_cpu_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_cpu_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_cpu_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_cpu_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_cpu_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");
 | 
			
		||||
							
								
								
									
										59
									
								
								sound/soc/at32/at32-ssc.h
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										59
									
								
								sound/soc/at32/at32-ssc.h
									
										
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,59 @@
 | 
			
		|||
/* sound/soc/at32/at32-ssc.h
 | 
			
		||||
 * ASoC SSC interface for Atmel AT32 SoC
 | 
			
		||||
 *
 | 
			
		||||
 * 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.
 | 
			
		||||
 */
 | 
			
		||||
 | 
			
		||||
#ifndef __SOUND_SOC_AT32_AT32_SSC_H
 | 
			
		||||
#define __SOUND_SOC_AT32_AT32_SSC_H __FILE__
 | 
			
		||||
 | 
			
		||||
#include <linux/types.h>
 | 
			
		||||
#include <linux/atmel-ssc.h>
 | 
			
		||||
 | 
			
		||||
#include "at32-pcm.h"
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
struct at32_ssc_state {
 | 
			
		||||
	u32 ssc_cmr;
 | 
			
		||||
	u32 ssc_rcmr;
 | 
			
		||||
	u32 ssc_rfmr;
 | 
			
		||||
	u32 ssc_tcmr;
 | 
			
		||||
	u32 ssc_tfmr;
 | 
			
		||||
	u32 ssc_sr;
 | 
			
		||||
	u32 ssc_imr;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
struct at32_ssc_info {
 | 
			
		||||
	char *name;
 | 
			
		||||
	struct ssc_device *ssc;
 | 
			
		||||
	spinlock_t lock;	/* lock for dir_mask */
 | 
			
		||||
	unsigned short dir_mask;	/* 0=unused, 1=playback, 2=capture */
 | 
			
		||||
	unsigned short initialized;	/* true if SSC has been initialized */
 | 
			
		||||
	unsigned short daifmt;
 | 
			
		||||
	unsigned short cmr_div;
 | 
			
		||||
	unsigned short tcmr_period;
 | 
			
		||||
	unsigned short rcmr_period;
 | 
			
		||||
	struct at32_pcm_dma_params *dma_params[2];
 | 
			
		||||
	struct at32_ssc_state ssc_state;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
/* SSC divider ids */
 | 
			
		||||
#define AT32_SSC_CMR_DIV        0	/* MCK divider for BCLK */
 | 
			
		||||
#define AT32_SSC_TCMR_PERIOD    1	/* BCLK divider for transmit FS */
 | 
			
		||||
#define AT32_SSC_RCMR_PERIOD    2	/* BCLK divider for receive FS */
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
extern struct snd_soc_cpu_dai at32_ssc_dai[];
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
#endif /* __SOUND_SOC_AT32_AT32_SSC_H */
 | 
			
		||||
							
								
								
									
										524
									
								
								sound/soc/at32/playpaq_wm8510.c
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										524
									
								
								sound/soc/at32/playpaq_wm8510.c
									
										
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,524 @@
 | 
			
		|||
/* sound/soc/at32/playpaq_wm8510.c
 | 
			
		||||
 * ASoC machine driver for PlayPaq using WM8510 codec
 | 
			
		||||
 *
 | 
			
		||||
 * 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.
 | 
			
		||||
 *
 | 
			
		||||
 * This code is largely inspired by sound/soc/at91/eti_b1_wm8731.c
 | 
			
		||||
 *
 | 
			
		||||
 * NOTE: If you don't have the AT32 enhanced portmux configured (which
 | 
			
		||||
 * isn't currently in the mainline or Atmel patched kernel), you will
 | 
			
		||||
 * need to set the MCLK pin (PA30) to peripheral A in your board initialization
 | 
			
		||||
 * code.  Something like:
 | 
			
		||||
 *	at32_select_periph(GPIO_PIN_PA(30), GPIO_PERIPH_A, 0);
 | 
			
		||||
 *
 | 
			
		||||
 */
 | 
			
		||||
 | 
			
		||||
/* #define DEBUG */
 | 
			
		||||
 | 
			
		||||
#include <linux/module.h>
 | 
			
		||||
#include <linux/moduleparam.h>
 | 
			
		||||
#include <linux/version.h>
 | 
			
		||||
#include <linux/kernel.h>
 | 
			
		||||
#include <linux/errno.h>
 | 
			
		||||
#include <linux/clk.h>
 | 
			
		||||
#include <linux/timer.h>
 | 
			
		||||
#include <linux/interrupt.h>
 | 
			
		||||
#include <linux/platform_device.h>
 | 
			
		||||
 | 
			
		||||
#include <sound/core.h>
 | 
			
		||||
#include <sound/pcm.h>
 | 
			
		||||
#include <sound/pcm_params.h>
 | 
			
		||||
#include <sound/soc.h>
 | 
			
		||||
#include <sound/soc-dapm.h>
 | 
			
		||||
 | 
			
		||||
#include <asm/arch/at32ap700x.h>
 | 
			
		||||
#include <asm/arch/portmux.h>
 | 
			
		||||
 | 
			
		||||
#include "../codecs/wm8510.h"
 | 
			
		||||
#include "at32-pcm.h"
 | 
			
		||||
#include "at32-ssc.h"
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
/*-------------------------------------------------------------------------*\
 | 
			
		||||
 * constants
 | 
			
		||||
\*-------------------------------------------------------------------------*/
 | 
			
		||||
#define MCLK_PIN		GPIO_PIN_PA(30)
 | 
			
		||||
#define MCLK_PERIPH		GPIO_PERIPH_A
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
/*-------------------------------------------------------------------------*\
 | 
			
		||||
 * data types
 | 
			
		||||
\*-------------------------------------------------------------------------*/
 | 
			
		||||
/* SSC clocking data */
 | 
			
		||||
struct ssc_clock_data {
 | 
			
		||||
	/* CMR div */
 | 
			
		||||
	unsigned int cmr_div;
 | 
			
		||||
 | 
			
		||||
	/* Frame period (as needed by xCMR.PERIOD) */
 | 
			
		||||
	unsigned int period;
 | 
			
		||||
 | 
			
		||||
	/* The SSC clock rate these settings where calculated for */
 | 
			
		||||
	unsigned long ssc_rate;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
/*-------------------------------------------------------------------------*\
 | 
			
		||||
 * module data
 | 
			
		||||
\*-------------------------------------------------------------------------*/
 | 
			
		||||
static struct clk *_gclk0;
 | 
			
		||||
static struct clk *_pll0;
 | 
			
		||||
 | 
			
		||||
#define CODEC_CLK (_gclk0)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
/*-------------------------------------------------------------------------*\
 | 
			
		||||
 * Sound SOC operations
 | 
			
		||||
\*-------------------------------------------------------------------------*/
 | 
			
		||||
#if defined CONFIG_SND_AT32_SOC_PLAYPAQ_SLAVE
 | 
			
		||||
static struct ssc_clock_data playpaq_wm8510_calc_ssc_clock(
 | 
			
		||||
	struct snd_pcm_hw_params *params,
 | 
			
		||||
	struct snd_soc_cpu_dai *cpu_dai)
 | 
			
		||||
{
 | 
			
		||||
	struct at32_ssc_info *ssc_p = cpu_dai->private_data;
 | 
			
		||||
	struct ssc_device *ssc = ssc_p->ssc;
 | 
			
		||||
	struct ssc_clock_data cd;
 | 
			
		||||
	unsigned int rate, width_bits, channels;
 | 
			
		||||
	unsigned int bitrate, ssc_div;
 | 
			
		||||
	unsigned actual_rate;
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
	/*
 | 
			
		||||
	 * Figure out required bitrate
 | 
			
		||||
	 */
 | 
			
		||||
	rate = params_rate(params);
 | 
			
		||||
	channels = params_channels(params);
 | 
			
		||||
	width_bits = snd_pcm_format_physical_width(params_format(params));
 | 
			
		||||
	bitrate = rate * width_bits * channels;
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
	/*
 | 
			
		||||
	 * Figure out required SSC divider and period for required bitrate
 | 
			
		||||
	 */
 | 
			
		||||
	cd.ssc_rate = clk_get_rate(ssc->clk);
 | 
			
		||||
	ssc_div = cd.ssc_rate / bitrate;
 | 
			
		||||
	cd.cmr_div = ssc_div / 2;
 | 
			
		||||
	if (ssc_div & 1) {
 | 
			
		||||
		/* round cmr_div up */
 | 
			
		||||
		cd.cmr_div++;
 | 
			
		||||
	}
 | 
			
		||||
	cd.period = width_bits - 1;
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
	/*
 | 
			
		||||
	 * Find actual rate, compare to requested rate
 | 
			
		||||
	 */
 | 
			
		||||
	actual_rate = (cd.ssc_rate / (cd.cmr_div * 2)) / (2 * (cd.period + 1));
 | 
			
		||||
	pr_debug("playpaq_wm8510: Request rate = %d, actual rate = %d\n",
 | 
			
		||||
		 rate, actual_rate);
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
	return cd;
 | 
			
		||||
}
 | 
			
		||||
#endif /* CONFIG_SND_AT32_SOC_PLAYPAQ_SLAVE */
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
static int playpaq_wm8510_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_codec_dai *codec_dai = rtd->dai->codec_dai;
 | 
			
		||||
	struct snd_soc_cpu_dai *cpu_dai = rtd->dai->cpu_dai;
 | 
			
		||||
	struct at32_ssc_info *ssc_p = cpu_dai->private_data;
 | 
			
		||||
	struct ssc_device *ssc = ssc_p->ssc;
 | 
			
		||||
	unsigned int pll_out = 0, bclk = 0, mclk_div = 0;
 | 
			
		||||
	int ret;
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
	/* Due to difficulties with getting the correct clocks from the AT32's
 | 
			
		||||
	 * PLL0, we're going to let the CODEC be in charge of all the clocks
 | 
			
		||||
	 */
 | 
			
		||||
#if !defined CONFIG_SND_AT32_SOC_PLAYPAQ_SLAVE
 | 
			
		||||
	const unsigned int fmt = (SND_SOC_DAIFMT_I2S |
 | 
			
		||||
				  SND_SOC_DAIFMT_NB_NF |
 | 
			
		||||
				  SND_SOC_DAIFMT_CBM_CFM);
 | 
			
		||||
#else
 | 
			
		||||
	struct ssc_clock_data cd;
 | 
			
		||||
	const unsigned int fmt = (SND_SOC_DAIFMT_I2S |
 | 
			
		||||
				  SND_SOC_DAIFMT_NB_NF |
 | 
			
		||||
				  SND_SOC_DAIFMT_CBS_CFS);
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
	if (ssc == NULL) {
 | 
			
		||||
		pr_warning("playpaq_wm8510_hw_params: ssc is NULL!\n");
 | 
			
		||||
		return -EINVAL;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
	/*
 | 
			
		||||
	 * Figure out PLL and BCLK dividers for WM8510
 | 
			
		||||
	 */
 | 
			
		||||
	switch (params_rate(params)) {
 | 
			
		||||
	case 48000:
 | 
			
		||||
		pll_out = 12288000;
 | 
			
		||||
		mclk_div = WM8510_MCLKDIV_1;
 | 
			
		||||
		bclk = WM8510_BCLKDIV_8;
 | 
			
		||||
		break;
 | 
			
		||||
 | 
			
		||||
	case 44100:
 | 
			
		||||
		pll_out = 11289600;
 | 
			
		||||
		mclk_div = WM8510_MCLKDIV_1;
 | 
			
		||||
		bclk = WM8510_BCLKDIV_8;
 | 
			
		||||
		break;
 | 
			
		||||
 | 
			
		||||
	case 22050:
 | 
			
		||||
		pll_out = 11289600;
 | 
			
		||||
		mclk_div = WM8510_MCLKDIV_2;
 | 
			
		||||
		bclk = WM8510_BCLKDIV_8;
 | 
			
		||||
		break;
 | 
			
		||||
 | 
			
		||||
	case 16000:
 | 
			
		||||
		pll_out = 12288000;
 | 
			
		||||
		mclk_div = WM8510_MCLKDIV_3;
 | 
			
		||||
		bclk = WM8510_BCLKDIV_8;
 | 
			
		||||
		break;
 | 
			
		||||
 | 
			
		||||
	case 11025:
 | 
			
		||||
		pll_out = 11289600;
 | 
			
		||||
		mclk_div = WM8510_MCLKDIV_4;
 | 
			
		||||
		bclk = WM8510_BCLKDIV_8;
 | 
			
		||||
		break;
 | 
			
		||||
 | 
			
		||||
	case 8000:
 | 
			
		||||
		pll_out = 12288000;
 | 
			
		||||
		mclk_div = WM8510_MCLKDIV_6;
 | 
			
		||||
		bclk = WM8510_BCLKDIV_8;
 | 
			
		||||
		break;
 | 
			
		||||
 | 
			
		||||
	default:
 | 
			
		||||
		pr_warning("playpaq_wm8510: Unsupported sample rate %d\n",
 | 
			
		||||
			   params_rate(params));
 | 
			
		||||
		return -EINVAL;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
	/*
 | 
			
		||||
	 * set CPU and CODEC DAI configuration
 | 
			
		||||
	 */
 | 
			
		||||
	ret = codec_dai->dai_ops.set_fmt(codec_dai, fmt);
 | 
			
		||||
	if (ret < 0) {
 | 
			
		||||
		pr_warning("playpaq_wm8510: "
 | 
			
		||||
			   "Failed to set CODEC DAI format (%d)\n",
 | 
			
		||||
			   ret);
 | 
			
		||||
		return ret;
 | 
			
		||||
	}
 | 
			
		||||
	ret = cpu_dai->dai_ops.set_fmt(cpu_dai, fmt);
 | 
			
		||||
	if (ret < 0) {
 | 
			
		||||
		pr_warning("playpaq_wm8510: "
 | 
			
		||||
			   "Failed to set CPU DAI format (%d)\n",
 | 
			
		||||
			   ret);
 | 
			
		||||
		return ret;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
	/*
 | 
			
		||||
	 * Set CPU clock configuration
 | 
			
		||||
	 */
 | 
			
		||||
#if defined CONFIG_SND_AT32_SOC_PLAYPAQ_SLAVE
 | 
			
		||||
	cd = playpaq_wm8510_calc_ssc_clock(params, cpu_dai);
 | 
			
		||||
	pr_debug("playpaq_wm8510: cmr_div = %d, period = %d\n",
 | 
			
		||||
		 cd.cmr_div, cd.period);
 | 
			
		||||
	ret = cpu_dai->dai_ops.set_clkdiv(cpu_dai,
 | 
			
		||||
					  AT32_SSC_CMR_DIV, cd.cmr_div);
 | 
			
		||||
	if (ret < 0) {
 | 
			
		||||
		pr_warning("playpaq_wm8510: Failed to set CPU CMR_DIV (%d)\n",
 | 
			
		||||
			   ret);
 | 
			
		||||
		return ret;
 | 
			
		||||
	}
 | 
			
		||||
	ret = cpu_dai->dai_ops.set_clkdiv(cpu_dai, AT32_SSC_TCMR_PERIOD,
 | 
			
		||||
					  cd.period);
 | 
			
		||||
	if (ret < 0) {
 | 
			
		||||
		pr_warning("playpaq_wm8510: "
 | 
			
		||||
			   "Failed to set CPU transmit period (%d)\n",
 | 
			
		||||
			   ret);
 | 
			
		||||
		return ret;
 | 
			
		||||
	}
 | 
			
		||||
#endif /* CONFIG_SND_AT32_SOC_PLAYPAQ_SLAVE */
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
	/*
 | 
			
		||||
	 * Set CODEC clock configuration
 | 
			
		||||
	 */
 | 
			
		||||
	pr_debug("playpaq_wm8510: "
 | 
			
		||||
		 "pll_in = %ld, pll_out = %u, bclk = %x, mclk = %x\n",
 | 
			
		||||
		 clk_get_rate(CODEC_CLK), pll_out, bclk, mclk_div);
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
#if !defined CONFIG_SND_AT32_SOC_PLAYPAQ_SLAVE
 | 
			
		||||
	ret = codec_dai->dai_ops.set_clkdiv(codec_dai, WM8510_BCLKDIV, bclk);
 | 
			
		||||
	if (ret < 0) {
 | 
			
		||||
		pr_warning
 | 
			
		||||
		    ("playpaq_wm8510: Failed to set CODEC DAI BCLKDIV (%d)\n",
 | 
			
		||||
		     ret);
 | 
			
		||||
		return ret;
 | 
			
		||||
	}
 | 
			
		||||
#endif /* CONFIG_SND_AT32_SOC_PLAYPAQ_SLAVE */
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
	ret = codec_dai->dai_ops.set_pll(codec_dai, 0,
 | 
			
		||||
					 clk_get_rate(CODEC_CLK), pll_out);
 | 
			
		||||
	if (ret < 0) {
 | 
			
		||||
		pr_warning("playpaq_wm8510: Failed to set CODEC DAI PLL (%d)\n",
 | 
			
		||||
			   ret);
 | 
			
		||||
		return ret;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
	ret = codec_dai->dai_ops.set_clkdiv(codec_dai,
 | 
			
		||||
					    WM8510_MCLKDIV, mclk_div);
 | 
			
		||||
	if (ret < 0) {
 | 
			
		||||
		pr_warning("playpaq_wm8510: Failed to set CODEC MCLKDIV (%d)\n",
 | 
			
		||||
			   ret);
 | 
			
		||||
		return ret;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
	return 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
static struct snd_soc_ops playpaq_wm8510_ops = {
 | 
			
		||||
	.hw_params = playpaq_wm8510_hw_params,
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
static const struct snd_soc_dapm_widget playpaq_dapm_widgets[] = {
 | 
			
		||||
	SND_SOC_DAPM_MIC("Int Mic", NULL),
 | 
			
		||||
	SND_SOC_DAPM_SPK("Ext Spk", NULL),
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
static const char *intercon[][3] = {
 | 
			
		||||
	/* speaker connected to SPKOUT */
 | 
			
		||||
	{"Ext Spk", NULL, "SPKOUTP"},
 | 
			
		||||
	{"Ext Spk", NULL, "SPKOUTN"},
 | 
			
		||||
 | 
			
		||||
	{"Mic Bias", NULL, "Int Mic"},
 | 
			
		||||
	{"MICN", NULL, "Mic Bias"},
 | 
			
		||||
	{"MICP", NULL, "Mic Bias"},
 | 
			
		||||
 | 
			
		||||
	/* Terminator */
 | 
			
		||||
	{NULL, NULL, NULL},
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
static int playpaq_wm8510_init(struct snd_soc_codec *codec)
 | 
			
		||||
{
 | 
			
		||||
	int i;
 | 
			
		||||
 | 
			
		||||
	/*
 | 
			
		||||
	 * Add DAPM widgets
 | 
			
		||||
	 */
 | 
			
		||||
	for (i = 0; i < ARRAY_SIZE(playpaq_dapm_widgets); i++)
 | 
			
		||||
		snd_soc_dapm_new_control(codec, &playpaq_dapm_widgets[i]);
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
	/*
 | 
			
		||||
	 * Setup audio path interconnects
 | 
			
		||||
	 */
 | 
			
		||||
	for (i = 0; intercon[i][0] != NULL; i++) {
 | 
			
		||||
		snd_soc_dapm_connect_input(codec,
 | 
			
		||||
					   intercon[i][0],
 | 
			
		||||
					   intercon[i][1], intercon[i][2]);
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
	/* always connected endpoints */
 | 
			
		||||
	snd_soc_dapm_set_endpoint(codec, "Int Mic", 1);
 | 
			
		||||
	snd_soc_dapm_set_endpoint(codec, "Ext Spk", 1);
 | 
			
		||||
	snd_soc_dapm_sync_endpoints(codec);
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
	/* Make CSB show PLL rate */
 | 
			
		||||
	codec->dai->dai_ops.set_clkdiv(codec->dai, WM8510_OPCLKDIV,
 | 
			
		||||
				       WM8510_OPCLKDIV_1 | 4);
 | 
			
		||||
 | 
			
		||||
	return 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
static struct snd_soc_dai_link playpaq_wm8510_dai = {
 | 
			
		||||
	.name = "WM8510",
 | 
			
		||||
	.stream_name = "WM8510 PCM",
 | 
			
		||||
	.cpu_dai = &at32_ssc_dai[0],
 | 
			
		||||
	.codec_dai = &wm8510_dai,
 | 
			
		||||
	.init = playpaq_wm8510_init,
 | 
			
		||||
	.ops = &playpaq_wm8510_ops,
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
static struct snd_soc_machine snd_soc_machine_playpaq = {
 | 
			
		||||
	.name = "LRS_PlayPaq_WM8510",
 | 
			
		||||
	.dai_link = &playpaq_wm8510_dai,
 | 
			
		||||
	.num_links = 1,
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
static struct wm8510_setup_data playpaq_wm8510_setup = {
 | 
			
		||||
	.i2c_address = 0x1a,
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
static struct snd_soc_device playpaq_wm8510_snd_devdata = {
 | 
			
		||||
	.machine = &snd_soc_machine_playpaq,
 | 
			
		||||
	.platform = &at32_soc_platform,
 | 
			
		||||
	.codec_dev = &soc_codec_dev_wm8510,
 | 
			
		||||
	.codec_data = &playpaq_wm8510_setup,
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
static struct platform_device *playpaq_snd_device;
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
static int __init playpaq_asoc_init(void)
 | 
			
		||||
{
 | 
			
		||||
	int ret = 0;
 | 
			
		||||
	struct at32_ssc_info *ssc_p = playpaq_wm8510_dai.cpu_dai->private_data;
 | 
			
		||||
	struct ssc_device *ssc = NULL;
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
	/*
 | 
			
		||||
	 * Request SSC device
 | 
			
		||||
	 */
 | 
			
		||||
	ssc = ssc_request(0);
 | 
			
		||||
	if (IS_ERR(ssc)) {
 | 
			
		||||
		ret = PTR_ERR(ssc);
 | 
			
		||||
		ssc = NULL;
 | 
			
		||||
		goto err_ssc;
 | 
			
		||||
	}
 | 
			
		||||
	ssc_p->ssc = ssc;
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
	/*
 | 
			
		||||
	 * Configure MCLK for WM8510
 | 
			
		||||
	 */
 | 
			
		||||
	_gclk0 = clk_get(NULL, "gclk0");
 | 
			
		||||
	if (IS_ERR(_gclk0)) {
 | 
			
		||||
		_gclk0 = NULL;
 | 
			
		||||
		goto err_gclk0;
 | 
			
		||||
	}
 | 
			
		||||
	_pll0 = clk_get(NULL, "pll0");
 | 
			
		||||
	if (IS_ERR(_pll0)) {
 | 
			
		||||
		_pll0 = NULL;
 | 
			
		||||
		goto err_pll0;
 | 
			
		||||
	}
 | 
			
		||||
	if (clk_set_parent(_gclk0, _pll0)) {
 | 
			
		||||
		pr_warning("snd-soc-playpaq: "
 | 
			
		||||
			   "Failed to set PLL0 as parent for DAC clock\n");
 | 
			
		||||
		goto err_set_clk;
 | 
			
		||||
	}
 | 
			
		||||
	clk_set_rate(CODEC_CLK, 12000000);
 | 
			
		||||
	clk_enable(CODEC_CLK);
 | 
			
		||||
 | 
			
		||||
#if defined CONFIG_AT32_ENHANCED_PORTMUX
 | 
			
		||||
	at32_select_periph(MCLK_PIN, MCLK_PERIPH, 0);
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
	/*
 | 
			
		||||
	 * Create and register platform device
 | 
			
		||||
	 */
 | 
			
		||||
	playpaq_snd_device = platform_device_alloc("soc-audio", 0);
 | 
			
		||||
	if (playpaq_snd_device == NULL) {
 | 
			
		||||
		ret = -ENOMEM;
 | 
			
		||||
		goto err_device_alloc;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	platform_set_drvdata(playpaq_snd_device, &playpaq_wm8510_snd_devdata);
 | 
			
		||||
	playpaq_wm8510_snd_devdata.dev = &playpaq_snd_device->dev;
 | 
			
		||||
 | 
			
		||||
	ret = platform_device_add(playpaq_snd_device);
 | 
			
		||||
	if (ret) {
 | 
			
		||||
		pr_warning("playpaq_wm8510: platform_device_add failed (%d)\n",
 | 
			
		||||
			   ret);
 | 
			
		||||
		goto err_device_add;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	return 0;
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
err_device_add:
 | 
			
		||||
	if (playpaq_snd_device != NULL) {
 | 
			
		||||
		platform_device_put(playpaq_snd_device);
 | 
			
		||||
		playpaq_snd_device = NULL;
 | 
			
		||||
	}
 | 
			
		||||
err_device_alloc:
 | 
			
		||||
err_set_clk:
 | 
			
		||||
	if (_pll0 != NULL) {
 | 
			
		||||
		clk_put(_pll0);
 | 
			
		||||
		_pll0 = NULL;
 | 
			
		||||
	}
 | 
			
		||||
err_pll0:
 | 
			
		||||
	if (_gclk0 != NULL) {
 | 
			
		||||
		clk_put(_gclk0);
 | 
			
		||||
		_gclk0 = NULL;
 | 
			
		||||
	}
 | 
			
		||||
err_gclk0:
 | 
			
		||||
	if (ssc != NULL) {
 | 
			
		||||
		ssc_free(ssc);
 | 
			
		||||
		ssc = NULL;
 | 
			
		||||
	}
 | 
			
		||||
err_ssc:
 | 
			
		||||
	return ret;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
static void __exit playpaq_asoc_exit(void)
 | 
			
		||||
{
 | 
			
		||||
	struct at32_ssc_info *ssc_p = playpaq_wm8510_dai.cpu_dai->private_data;
 | 
			
		||||
	struct ssc_device *ssc;
 | 
			
		||||
 | 
			
		||||
	if (ssc_p != NULL) {
 | 
			
		||||
		ssc = ssc_p->ssc;
 | 
			
		||||
		if (ssc != NULL)
 | 
			
		||||
			ssc_free(ssc);
 | 
			
		||||
		ssc_p->ssc = NULL;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	if (_gclk0 != NULL) {
 | 
			
		||||
		clk_put(_gclk0);
 | 
			
		||||
		_gclk0 = NULL;
 | 
			
		||||
	}
 | 
			
		||||
	if (_pll0 != NULL) {
 | 
			
		||||
		clk_put(_pll0);
 | 
			
		||||
		_pll0 = NULL;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
#if defined CONFIG_AT32_ENHANCED_PORTMUX
 | 
			
		||||
	at32_free_pin(MCLK_PIN);
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
	platform_device_unregister(playpaq_snd_device);
 | 
			
		||||
	playpaq_snd_device = NULL;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
module_init(playpaq_asoc_init);
 | 
			
		||||
module_exit(playpaq_asoc_exit);
 | 
			
		||||
 | 
			
		||||
MODULE_AUTHOR("Geoffrey Wossum <gwossum@acm.org>");
 | 
			
		||||
MODULE_DESCRIPTION("ASoC machine driver for LRS PlayPaq");
 | 
			
		||||
MODULE_LICENSE("GPL");
 | 
			
		||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue