From bc755a3b8859e7307a8b10f39ca4cb6401c51987 Mon Sep 17 00:00:00 2001 From: Kurt Mahan <kmahan@freescale.com> Date: Tue, 27 Nov 2007 14:39:37 -0700 Subject: [PATCH] Add M5445x SPI support. LTIBName: m5445x-spi Signed-off-by: Kurt Mahan <kmahan@freescale.com> --- arch/m68k/configs/m54455evb_defconfig | 24 +- drivers/spi/Kconfig | 36 + drivers/spi/Makefile | 4 + drivers/spi/coldfire_edma.c | 358 ++++++++ drivers/spi/spi-m5445x.c | 156 ++++ drivers/spi/spi_coldfire.c | 1552 +++++++++++++++++++++++++++++++++ drivers/spi/ssi_audio.c | 906 +++++++++++++++++++ include/asm-m68k/coldfire_edma.h | 101 ++- include/linux/spi/mcfqspi.h | 80 ++ 9 files changed, 3196 insertions(+), 21 deletions(-) create mode 100644 drivers/spi/coldfire_edma.c create mode 100644 drivers/spi/spi-m5445x.c create mode 100644 drivers/spi/spi_coldfire.c create mode 100644 drivers/spi/ssi_audio.c create mode 100644 include/linux/spi/mcfqspi.h --- a/arch/m68k/configs/m54455evb_defconfig +++ b/arch/m68k/configs/m54455evb_defconfig @@ -321,6 +321,8 @@ CONFIG_MTD_PHYSMAP_BANKWIDTH=1 # # Self-contained MTD device drivers # +# CONFIG_MTD_DATAFLASH is not set +# CONFIG_MTD_M25P80 is not set # CONFIG_MTD_SLRAM is not set # CONFIG_MTD_PHRAM is not set # CONFIG_MTD_MTDRAM is not set @@ -497,8 +499,26 @@ CONFIG_UNIX98_PTYS=y # # SPI support # -# CONFIG_SPI is not set -# CONFIG_SPI_MASTER is not set +CONFIG_SPI=y +# CONFIG_SPI_DEBUG is not set +CONFIG_COLDFIRE_EDMA=y +CONFIG_SPI_MASTER=y + +# +# SPI Master Controller Drivers +# +# CONFIG_SPI_BITBANG is not set +CONFIG_SPI_COLDFIRE=y +CONFIG_SPI_COLDFIRE_DSPI_EDMA=y + +# +# SPI Protocol Masters +# +# CONFIG_SPI_AT25 is not set +# CONFIG_SPI_SPIDEV is not set +# CONFIG_SPI_TLE62X0 is not set +CONFIG_SPI_COLDFIRE_SSI_AUDIO=y +# CONFIG_SSIAUDIO_USE_EDMA is not set # CONFIG_W1 is not set # CONFIG_POWER_SUPPLY is not set # CONFIG_HWMON is not set --- a/drivers/spi/Kconfig +++ b/drivers/spi/Kconfig @@ -35,6 +35,15 @@ config SPI_DEBUG Say "yes" to enable debug messaging (like dev_dbg and pr_debug), sysfs, and debugfs support in SPI controller and protocol drivers. +config COLDFIRE_EDMA + tristate "Coldfire eDMA" + depends on COLDFIRE && EXPERIMENTAL + help + Support for Coldfire eDMA controller. Required for example + by SSI audio device driver. + + + # # MASTER side ... talking to discrete SPI slave chips including microcontrollers # @@ -113,6 +122,21 @@ config SPI_GPIO If unsure, say N. +config SPI_COLDFIRE + tristate "Coldfire QSPI/DSPI SPI Master" + depends on SPI_MASTER && COLDFIRE && EXPERIMENTAL + help + SPI driver for Freescale Coldfire QSPI module in master mode. + Tested with the 5282 processor, but should also work with other + Coldfire variants. + +config SPI_COLDFIRE_DSPI_EDMA + boolean "Coldfire DSPI master driver uses eDMA" + depends on SPI_MASTER && COLDFIRE && SPI_COLDFIRE && EXPERIMENTAL && COLDFIRE_EDMA + default n + help + Say "yes" if you want DSPI master driver to use eDMA for transfers. + config SPI_IMX tristate "Freescale iMX SPI controller" depends on SPI_MASTER && ARCH_IMX && EXPERIMENTAL @@ -255,6 +279,18 @@ config SPI_TLE62X0 # # Add new SPI protocol masters in alphabetical order above this line # +config SPI_COLDFIRE_SSI_AUDIO + tristate "Coldfire SSI AUDIO" + depends on SPI_MASTER && SPI_COLDFIRE && EXPERIMENTAL + help + SSI audio device driver + +config SSIAUDIO_USE_EDMA + boolean "Coldfire DSPI master driver uses eDMA" + default y + depends on EXPERIMENTAL && COLDFIRE_EDMA && SPI_COLDFIRE_SSI_AUDIO + help + Say "yes" if you want SSI audio driver to use eDMA for SSI transfers. # (slave support would go here) --- a/drivers/spi/Makefile +++ b/drivers/spi/Makefile @@ -6,6 +6,8 @@ ifeq ($(CONFIG_SPI_DEBUG),y) EXTRA_CFLAGS += -DDEBUG endif +obj-$(CONFIG_COLDFIRE_EDMA) += coldfire_edma.o + # small core, mostly translating board-specific # config declarations into driver model code obj-$(CONFIG_SPI_MASTER) += spi.o @@ -16,6 +18,7 @@ obj-$(CONFIG_SPI_BFIN) += spi_bfin5xx. obj-$(CONFIG_SPI_BITBANG) += spi_bitbang.o obj-$(CONFIG_SPI_AU1550) += au1550_spi.o obj-$(CONFIG_SPI_BUTTERFLY) += spi_butterfly.o +obj-$(CONFIG_SPI_COLDFIRE) += spi_coldfire.o spi-m5445x.o obj-$(CONFIG_SPI_GPIO) += spi_gpio.o obj-$(CONFIG_SPI_IMX) += spi_imx.o obj-$(CONFIG_SPI_LM70_LLP) += spi_lm70llp.o @@ -35,6 +38,7 @@ obj-$(CONFIG_SPI_SH_SCI) += spi_sh_sci. obj-$(CONFIG_SPI_AT25) += at25.o obj-$(CONFIG_SPI_SPIDEV) += spidev.o obj-$(CONFIG_SPI_TLE62X0) += tle62x0.o +obj-$(CONFIG_SPI_COLDFIRE_SSI_AUDIO) += ssi_audio.o # ... add above this line ... # SPI slave controller drivers (upstream link) --- /dev/null +++ b/drivers/spi/coldfire_edma.c @@ -0,0 +1,358 @@ +/* + * + * coldfire_edma.c - eDMA driver for Coldfire MCF5445x + * + * Yaroslav Vinogradov yaroslav.vinogradov@freescale.com + * + * Copyright Freescale Semiconductor, Inc. 2007 + * + * This program is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License as published by the + * Free Software Foundation; either version 2 of the License, or (at your + * option) any later version. + */ + +#include <linux/init.h> +#include <linux/module.h> +#include <asm/virtconvert.h> +#include <asm/coldfire.h> +#include <linux/fs.h> +#include <linux/cdev.h> +#include <linux/seq_file.h> +#include <linux/proc_fs.h> +#include <asm/mcf5445x_edma.h> +#include <asm/mcf5445x_intc.h> +#include <asm/coldfire_edma.h> + + +/* callback handler data for each TCD */ +struct edma_isr_record { + edma_irq_handler irq_handler; /* interrupt handler */ + edma_error_handler error_handler; /* error interrupt handler */ + void* dev; /* device used for the channel */ + int allocated; /* busy flag */ + spinlock_t *lock; /* spin lock (if needs to be locked in interrupt) */ + const char* device_id; /* device id string, used in proc file system */ +}; + +/* device structure */ +struct coldfire_edma_dev { + struct cdev cdev; /* character device */ + struct edma_isr_record dma_interrupt_handlers[EDMA_CHANNELS]; /* channel handlers */ +}; + +/* allocated major device number */ +static int coldfire_dma_major; +/* device driver structure */ +static struct coldfire_edma_dev* devp = NULL; + +/* device driver file operations */ +struct file_operations coldfire_edma_fops = { + .owner = THIS_MODULE, +}; + +/* eDMA channel interrupt handler */ +static int dmaisr(int irq, void *dev_id) +{ + int channel = irq - EDMA_INT_CONTROLLER_BASE - EDMA_INT_CHANNEL_BASE; + int result = IRQ_HANDLED; + + if (devp!=NULL && devp->dma_interrupt_handlers[channel].lock) { + spin_lock(devp->dma_interrupt_handlers[channel].lock); + } + + if (devp!=NULL && devp->dma_interrupt_handlers[channel].irq_handler) { + result = devp->dma_interrupt_handlers[channel].irq_handler(channel, + devp->dma_interrupt_handlers[channel].dev); + } else { + confirm_edma_interrupt_handled(channel); + printk(EDMA_DRIVER_NAME ": No handler for DMA channel %d\n", channel); + } + + if (devp!=NULL && devp->dma_interrupt_handlers[channel].lock) { + spin_unlock(devp->dma_interrupt_handlers[channel].lock); + } + + return result; +} + +/* eDMA error interrupt handler */ +static int dma_error_isr(int irq, void* dev_id) +{ + u16 err; + int i; + + err = MCF_EDMA_ERR; + for (i=0;i<EDMA_CHANNELS;i++) { + if (err & (1<<i)) { + if (devp!=NULL && devp->dma_interrupt_handlers[i].error_handler) { + devp->dma_interrupt_handlers[i].error_handler(i, devp->dma_interrupt_handlers[i].dev); + } else { + printk(KERN_WARNING EDMA_DRIVER_NAME ": DMA error on channel %d\n", i); + } + } + } + + MCF_EDMA_CERR = MCF_EDMA_CERR_CAER; + return IRQ_HANDLED; +} + +/* sets channel parameters */ +void set_edma_params(int channel, u32 source, u32 dest, + u32 attr, u32 soff, u32 nbytes, u32 slast, + u32 citer, u32 biter, u32 doff, u32 dlast_sga, + int major_int, int disable_req) +{ + + if (channel<0 || channel>EDMA_CHANNELS) + return; + + MCF_EDMA_TCD_SADDR(channel) = source; + MCF_EDMA_TCD_DADDR(channel) = dest; + MCF_EDMA_TCD_ATTR(channel) = attr; + MCF_EDMA_TCD_SOFF(channel) = MCF_EDMA_TCD_SOFF_SOFF(soff); + MCF_EDMA_TCD_NBYTES(channel) = MCF_EDMA_TCD_NBYTES_NBYTES(nbytes); + MCF_EDMA_TCD_SLAST(channel) = MCF_EDMA_TCD_SLAST_SLAST(slast); + MCF_EDMA_TCD_CITER(channel) = MCF_EDMA_TCD_CITER_CITER(citer); + MCF_EDMA_TCD_BITER(channel)=MCF_EDMA_TCD_BITER_BITER(biter); + MCF_EDMA_TCD_DOFF(channel) = MCF_EDMA_TCD_DOFF_DOFF(doff); + MCF_EDMA_TCD_DLAST_SGA(channel) = MCF_EDMA_TCD_DLAST_SGA_DLAST_SGA(dlast_sga); + /* interrupt at the end of major loop */ + if (major_int) { + MCF_EDMA_TCD_CSR(channel) |= MCF_EDMA_TCD_CSR_INT_MAJOR; + } else { + MCF_EDMA_TCD_CSR(channel) &= ~MCF_EDMA_TCD_CSR_INT_MAJOR; + } + /* disable request at the end of major loop of transfer or not*/ + if (disable_req) { + MCF_EDMA_TCD_CSR(channel) |= MCF_EDMA_TCD_CSR_D_REQ; + } else { + MCF_EDMA_TCD_CSR(channel) &= ~MCF_EDMA_TCD_CSR_D_REQ; + } + +} +EXPORT_SYMBOL(set_edma_params); + +/* init eDMA controller */ +void init_edma(void) +{ + MCF_EDMA_CR = 0; +} +EXPORT_SYMBOL(init_edma); + +/* request eDMA channel */ +int request_edma_channel(int channel, + edma_irq_handler handler, + edma_error_handler error_handler, + void* dev, + spinlock_t *lock, + const char* device_id ) +{ + if (devp!=NULL && channel>=0 && channel<=EDMA_CHANNELS) { + if (devp->dma_interrupt_handlers[channel].allocated) { + return -EBUSY; + } + devp->dma_interrupt_handlers[channel].allocated = 1; + devp->dma_interrupt_handlers[channel].irq_handler = handler; + devp->dma_interrupt_handlers[channel].error_handler = error_handler; + devp->dma_interrupt_handlers[channel].dev = dev; + devp->dma_interrupt_handlers[channel].lock = lock; + devp->dma_interrupt_handlers[channel].device_id = device_id; + return 0; + } + return -EINVAL; +} +EXPORT_SYMBOL(request_edma_channel); + +/* free eDMA channel */ +int free_edma_channel(int channel, void* dev) +{ + if (devp!=NULL && channel>=0 && channel<=EDMA_CHANNELS) { + if (devp->dma_interrupt_handlers[channel].allocated) { + if (devp->dma_interrupt_handlers[channel].dev != dev) { + return -EBUSY; + } + devp->dma_interrupt_handlers[channel].allocated = 0; + devp->dma_interrupt_handlers[channel].dev = NULL; + devp->dma_interrupt_handlers[channel].irq_handler = NULL; + devp->dma_interrupt_handlers[channel].error_handler = NULL; + devp->dma_interrupt_handlers[channel].lock = NULL; + } + return 0; + } + return -EINVAL; +} +EXPORT_SYMBOL(free_edma_channel); + +/* clean-up device driver allocated resources */ +static void coldfire_edma_cleanup(void) +{ + dev_t devno; + int i; + + /* free interrupts/memory */ + if (devp) { + for (i=0;i<EDMA_CHANNELS;i++) + { + MCF_INTC0_SIMR = EDMA_INT_CHANNEL_BASE+i; + free_irq(EDMA_INT_CHANNEL_BASE+EDMA_INT_CONTROLLER_BASE+i, devp); + } + MCF_INTC0_SIMR = EDMA_INT_CHANNEL_BASE+EDMA_CHANNELS; + free_irq(EDMA_INT_CHANNEL_BASE+EDMA_INT_CONTROLLER_BASE+EDMA_CHANNELS, devp); + cdev_del(&devp->cdev); + kfree(devp); + } + + /* unregister character device */ + devno = MKDEV(coldfire_dma_major, 0); + unregister_chrdev_region(devno, 1); +} + +#ifdef CONFIG_PROC_FS +/* proc file system support */ + +#define FREE_CHANNEL "free" +#define DEVICE_UNKNOWN "device unknown" + +static int proc_edma_show(struct seq_file *m, void *v) +{ + int i; + + if (devp==NULL) return 0; + + for (i = 0 ; i < EDMA_CHANNELS ; i++) { + if (devp->dma_interrupt_handlers[i].allocated) { + if (devp->dma_interrupt_handlers[i].device_id) + seq_printf(m, "%2d: %s\n", i, devp->dma_interrupt_handlers[i].device_id); + else + seq_printf(m, "%2d: %s\n", i, DEVICE_UNKNOWN); + } else { + seq_printf(m, "%2d: %s\n", i, FREE_CHANNEL); + } + } + return 0; +} + +static int proc_edma_open(struct inode *inode, struct file *file) +{ + return single_open(file, proc_edma_show, NULL); +} + +static const struct file_operations proc_edma_operations = { + .open = proc_edma_open, + .read = seq_read, + .llseek = seq_lseek, + .release = single_release, +}; + +static int __init proc_edma_init(void) +{ + struct proc_dir_entry *e; + + e = create_proc_entry("edma", 0, NULL); + if (e) + e->proc_fops = &proc_edma_operations; + + return 0; +} + +#endif + +/* initializes device driver */ +static int __init coldfire_edma_init(void) +{ + dev_t dev; + int result; + int i; + + /* allocate free major number */ + result = alloc_chrdev_region(&dev, DMA_DEV_MINOR, 1, EDMA_DRIVER_NAME); + if (result<0) { + printk(KERN_WARNING EDMA_DRIVER_NAME": can't get major %d\n", result); + return result; + } + coldfire_dma_major = MAJOR(dev); + + /* allocate device driver structure */ + devp = kmalloc(sizeof(struct coldfire_edma_dev), GFP_KERNEL); + if (!devp) { + result = -ENOMEM; + goto fail; + } + + /* init handlers (no handlers for beggining) */ + for (i=0;i<EDMA_CHANNELS;i++) { + devp->dma_interrupt_handlers[i].irq_handler = NULL; + devp->dma_interrupt_handlers[i].error_handler = NULL; + devp->dma_interrupt_handlers[i].dev = NULL; + devp->dma_interrupt_handlers[i].allocated = 0; + devp->dma_interrupt_handlers[i].lock = NULL; + devp->dma_interrupt_handlers[i].device_id = NULL; + } + + /* register char device */ + cdev_init(&devp->cdev, &coldfire_edma_fops); + devp->cdev.owner = THIS_MODULE; + devp->cdev.ops = &coldfire_edma_fops; + result = cdev_add(&devp->cdev, dev, 1); + if (result) { + printk(KERN_NOTICE EDMA_DRIVER_NAME": Error %d adding coldfire-dma device\n", result); + result = -ENODEV; + goto fail; + } + + /* request/enable irq for each eDMA channel */ + for (i=0;i<EDMA_CHANNELS;i++) + { + result = request_irq(EDMA_INT_CHANNEL_BASE+EDMA_INT_CONTROLLER_BASE+i, + dmaisr, SA_INTERRUPT, EDMA_DRIVER_NAME, devp); + if (result) { + printk(KERN_WARNING EDMA_DRIVER_NAME": Cannot request irq %d\n", + EDMA_INT_CHANNEL_BASE+EDMA_INT_CONTROLLER_BASE+i); + result = -EBUSY; + goto fail; + } + + MCF_INTC0_ICR(EDMA_INT_CHANNEL_BASE+i) = EDMA_IRQ_LEVEL; + MCF_INTC0_CIMR = EDMA_INT_CHANNEL_BASE+i; + + } + + /* request error interrupt */ + result = request_irq(EDMA_INT_CHANNEL_BASE + EDMA_INT_CONTROLLER_BASE + EDMA_CHANNELS, + dma_error_isr, SA_INTERRUPT, EDMA_DRIVER_NAME, devp); + if (result) { + printk(KERN_WARNING EDMA_DRIVER_NAME": Cannot request irq %d\n", + EDMA_INT_CHANNEL_BASE+EDMA_INT_CONTROLLER_BASE+EDMA_CHANNELS); + result = -EBUSY; + goto fail; + } + + /* enable error interrupt in interrupt controller */ + MCF_INTC0_ICR(EDMA_INT_CHANNEL_BASE+EDMA_CHANNELS) = EDMA_IRQ_LEVEL; + MCF_INTC0_CIMR = EDMA_INT_CHANNEL_BASE+EDMA_CHANNELS; + +#ifdef CONFIG_PROC_FS + proc_edma_init(); +#endif + + printk(EDMA_DRIVER_NAME ": initialized successfully\n"); + + return 0; +fail: + coldfire_edma_cleanup(); + return result; + +} + +static void __exit coldfire_edma_exit(void) +{ + coldfire_edma_cleanup(); +} + +module_init(coldfire_edma_init); +module_exit(coldfire_edma_exit); + +MODULE_LICENSE("GPL"); +MODULE_AUTHOR("Yaroslav Vinogradov, Freescale Inc."); +MODULE_DESCRIPTION("eDMA library for Coldfire 5445x"); --- /dev/null +++ b/drivers/spi/spi-m5445x.c @@ -0,0 +1,156 @@ +/***************************************************************************/ +/* + * linux/arch/m68k/coldfire/spi-m5445x.c + * + * Sub-architcture dependant initialization code for the Freescale + * 5445x SPI module + * + * Yaroslav Vinogradov yaroslav.vinogradov@freescale.com + * Copyright Freescale Semiconductor, Inc 2007 + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 2 of the License, or + * (at your option) any later version. + */ +/***************************************************************************/ + + +#include <linux/kernel.h> +#include <linux/sched.h> +#include <linux/param.h> +#include <linux/init.h> +#include <linux/interrupt.h> +#include <linux/device.h> +#include <linux/platform_device.h> +#include <linux/spi/spi.h> + +#include <asm/dma.h> +#include <asm/traps.h> +#include <asm/machdep.h> +#include <asm/coldfire.h> +#include <asm/mcfsim.h> +#include <asm/mcfqspi.h> +#include <asm/mcf5445x_gpio.h> + +#define SPI_NUM_CHIPSELECTS 0x10 +#define SPI_PAR_VAL (0 | MCF_GPIO_PAR_DSPI_PCS5_PCS5 | MCF_GPIO_PAR_DSPI_PCS2_PCS2 \ + | MCF_GPIO_PAR_DSPI_PCS1_PCS1 | MCF_GPIO_PAR_DSPI_PCS0_PCS0 | MCF_GPIO_PAR_DSPI_SIN_SIN \ + | MCF_GPIO_PAR_DSPI_SOUT_SOUT | MCF_GPIO_PAR_DSPI_SCK_SCK) + +#define MCF5445x_DSPI_IRQ_SOURCE (31) +#define MCF5445x_DSPI_IRQ_VECTOR (64 + MCF5445x_DSPI_IRQ_SOURCE) + +#define MCF5445x_DSPI_PAR (0xFC0A4063) +#define MCF5445x_DSPI_MCR (0xFC05C000) +#define MCF5445x_INTC0_ICR (0xFC048040) +#define MCF5445x_INTC0_IMRL (0xFC04800C) + + +#define M5445x_AUDIO_IRQ_SOURCE 49 +#define M5445x_AUDIO_IRQ_VECTOR (128+M5445x_AUDIO_IRQ_SOURCE) +#define M5445x_AUDIO_IRQ_LEVEL 4 + +void coldfire_qspi_cs_control(u8 cs, u8 command) +{ +} + +#if defined(CONFIG_SPI_COLDFIRE_SSI_AUDIO) +static struct coldfire_spi_chip ssi_audio_chip_info = { + .mode = SPI_MODE_0, + .bits_per_word = 16, + .del_cs_to_clk = 16, + .del_after_trans = 16, + .void_write_data = 0 +}; + +#endif + +static struct spi_board_info spi_board_info[] = { + +#if defined(CONFIG_SPI_COLDFIRE_SSI_AUDIO) + { + .modalias = "ssi_audio", + .max_speed_hz = 300000, + .bus_num = 1, + .chip_select = 5, + .irq = M5445x_AUDIO_IRQ_VECTOR, + .platform_data = NULL, + .controller_data = &ssi_audio_chip_info + } +#endif + +}; + +static struct coldfire_spi_master coldfire_master_info = { + .bus_num = 1, + .num_chipselect = SPI_NUM_CHIPSELECTS, + .irq_source = MCF5445x_DSPI_IRQ_SOURCE, + .irq_vector = MCF5445x_DSPI_IRQ_VECTOR, + .irq_mask = (0x01 << MCF5445x_DSPI_IRQ_SOURCE), + .irq_lp = 0x2, /* Level */ + .par_val = SPI_PAR_VAL, +// .par_val16 = SPI_PAR_VAL, + .cs_control = coldfire_qspi_cs_control, +}; + +static struct resource coldfire_spi_resources[] = { + [0] = { + .name = "qspi-par", + .start = MCF5445x_DSPI_PAR, + .end = MCF5445x_DSPI_PAR, + .flags = IORESOURCE_MEM + }, + + [1] = { + .name = "qspi-module", + .start = MCF5445x_DSPI_MCR, + .end = MCF5445x_DSPI_MCR + 0xB8, + .flags = IORESOURCE_MEM + }, + + [2] = { + .name = "qspi-int-level", + .start = MCF5445x_INTC0_ICR + MCF5445x_DSPI_IRQ_SOURCE, + .end = MCF5445x_INTC0_ICR + MCF5445x_DSPI_IRQ_SOURCE, + .flags = IORESOURCE_MEM + }, + + [3] = { + .name = "qspi-int-mask", + .start = MCF5445x_INTC0_IMRL, + .end = MCF5445x_INTC0_IMRL, + .flags = IORESOURCE_MEM + } +}; + +static struct platform_device coldfire_spi = { + .name = "spi_coldfire", //"coldfire-qspi", + .id = -1, + .resource = coldfire_spi_resources, + .num_resources = ARRAY_SIZE(coldfire_spi_resources), + .dev = { + .platform_data = &coldfire_master_info, + } +}; + +static int __init spi_dev_init(void) +{ + int retval = 0; + + retval = platform_device_register(&coldfire_spi); + + if (retval < 0) { + printk(KERN_ERR "SPI-m5445x: platform_device_register failed with code=%d\n", retval); + goto out; + } + + if (ARRAY_SIZE(spi_board_info)) + retval = spi_register_board_info(spi_board_info, ARRAY_SIZE(spi_board_info)); + + +out: + return retval; +} + +arch_initcall(spi_dev_init); --- /dev/null +++ b/drivers/spi/spi_coldfire.c @@ -0,0 +1,1552 @@ +/****************************************************************************/ + +/* + * spi_coldfire.c - Master QSPI/DSPI controller for the ColdFire processors + * + * (C) Copyright 2005, Intec Automation, + * Mike Lavender (mike@steroidmicros) + * + * (C) Copyright 2007, Freescale Inc, + * Yaroslav Vinogradov (yaroslav.vinogradov@freescale.com) + * + + This program is free software; you can redistribute it and/or modify + it under the terms of the GNU General Public License as published by + the Free Software Foundation; either version 2 of the License, or + (at your option) any later version. + + This program is distributed in the hope that it will be useful, + but WITHOUT ANY WARRANTY; without even the implied warranty of + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + GNU General Public License for more details. + + You should have received a copy of the GNU General Public License + along with this program; if not, write to the Free Software + Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. */ +/* ------------------------------------------------------------------------- */ + + +/****************************************************************************/ + +/* + * Includes + */ + +#include <linux/autoconf.h> +#include <linux/init.h> +#include <linux/module.h> +#include <linux/device.h> +#include <linux/interrupt.h> +#include <linux/platform_device.h> +#include <linux/spi/spi.h> +#include <linux/workqueue.h> +#include <linux/delay.h> + +#include <asm/delay.h> +#include <asm/mcfsim.h> +#include <asm/mcfqspi.h> +#include <asm/coldfire.h> +#include <asm/virtconvert.h> + +#if defined(CONFIG_M54455) + #define SPI_DSPI + #if defined(CONFIG_SPI_COLDFIRE_DSPI_EDMA) + #define SPI_DSPI_EDMA + #ifdef CONFIG_MMU + #define SPI_USE_MMU + #endif + #endif +#endif + +#ifdef SPI_DSPI +#include <asm/mcf5445x_dspi.h> + + +#endif + +#if defined(SPI_DSPI_EDMA) + +/* edma buffer size in transfer units (32bits) */ +#define EDMA_BUFFER_SIZE (PAGE_SIZE/4) +#define EDMA_BUFSIZE_KMALLOC (EDMA_BUFFER_SIZE*4) + +#define DSPI_DMA_RX_TCD 12 +#define DSPI_DMA_TX_TCD 13 + + +#include <asm/coldfire_edma.h> +#include <asm/mcf5445x_edma.h> +#endif + + +MODULE_AUTHOR("Mike Lavender"); +MODULE_DESCRIPTION("ColdFire QSPI Contoller"); +MODULE_LICENSE("GPL"); + +#define DRIVER_NAME "Coldfire QSPI/DSPI" + +/****************************************************************************/ + +/* + * Local constants and macros + */ + +#define QSPI_RAM_SIZE 0x10 /* 16 word table */ + +#define QSPI_TRANSMIT_RAM 0x00 +#define QSPI_RECEIVE_RAM 0x10 +#define QSPI_COMMAND_RAM 0x20 + +#define QSPI_COMMAND 0x7000 /* 15: X = Continuous CS + * 14: 1 = Get BITSE from QMR[BITS] + * 13: 1 = Get DT from QDLYR[DTL] + * 12: 1 = Get DSK from QDLYR[QCD] + * 8-11: XXXX = next 4 bytes for CS + * 0-7: 0000 0000 Reserved + */ + +#define QIR_WCEF 0x0008 /* write collison */ +#define QIR_ABRT 0x0004 /* abort */ +#define QIR_SPIF 0x0001 /* finished */ + +#define QIR_WCEFE 0x0800 +#define QIR_ABRTE 0x0400 +#define QIR_SPIFE 0x0100 + +#define QIR_WCEFB 0x8000 +#define QIR_ABRTB 0x4000 +#define QIR_ABRTL 0x1000 + +#define QMR_BITS 0x3C00 +#define QMR_BITS_8 0x2000 + +#define QCR_CONT 0x8000 + +#define QDLYR_SPE 0x8000 + +#define QWR_ENDQP_MASK 0x0F00 +#define QWR_CSIV 0x1000 /* 1 = active low chip selects */ + + +#define START_STATE ((void*)0) +#define RUNNING_STATE ((void*)1) +#define DONE_STATE ((void*)2) +#define ERROR_STATE ((void*)-1) + +#define QUEUE_RUNNING 0 +#define QUEUE_STOPPED 1 + +/****************************************************************************/ + +/* + * Local Data Structures + */ + +struct transfer_state { + u32 index; + u32 len; + void *tx; + void *tx_end; + void *rx; + void *rx_end; + char flags; +#define TRAN_STATE_RX_VOID 0x01 +#define TRAN_STATE_TX_VOID 0x02 +#define TRAN_STATE_WORD_ODD_NUM 0x04 + u8 cs; + u16 void_write_data; + unsigned cs_change:1; +}; + +typedef struct { + unsigned master:1; + unsigned dohie:1; + unsigned bits:4; + unsigned cpol:1; + unsigned cpha:1; + unsigned baud:8; +} QMR; + +typedef struct { + unsigned spe:1; + unsigned qcd:7; + unsigned dtl:8; +} QDLYR; + +typedef struct { + unsigned halt:1; + unsigned wren:1; + unsigned wrto:1; + unsigned csiv:1; + unsigned endqp:4; + unsigned cptqp:4; + unsigned newqp:4; +} QWR; + + +typedef struct { + unsigned master:1; + unsigned cont_scke:1; + unsigned dconf:2; + unsigned frz:1; + unsigned mtfe:1; + unsigned pcsse:1; + unsigned rooe:1; + unsigned pcsis:8; + unsigned reserved15:1; + unsigned mdis:1; + unsigned dis_tx:1; + unsigned dis_rxf:1; + unsigned clr_tx:1; + unsigned clr_rxf:1; + unsigned smpl_pt:2; + unsigned reserved71:7; + unsigned halt:1; +} DSPI_MCR; + +typedef struct { + unsigned dbr:1; + unsigned fmsz:4; + unsigned cpol:1; + unsigned cpha:1; + unsigned lsbfe:1; + unsigned pcssck:2; + unsigned pasc:2; + unsigned pdt:2; + unsigned pbr:2; + unsigned cssck:4; + unsigned asc:4; + unsigned dt:4; + unsigned br:4; +} DSPI_CTAR; + +struct chip_data { +#if defined(SPI_DSPI) + /* dspi data */ + union { + u32 mcr_val; + DSPI_MCR mcr; + }; + union { + u32 ctar_val; + DSPI_CTAR ctar; + }; +#else + union { + u16 qmr_val; + QMR qmr; + }; + union { + u16 qdlyr_val; + QDLYR qdlyr; + }; + union { + u16 qwr_val; + QWR qwr; + }; +#endif + + u16 void_write_data; +}; + + +struct driver_data { + /* Driver model hookup */ + struct platform_device *pdev; + + /* SPI framework hookup */ + struct spi_master *master; + + /* Driver message queue */ + struct workqueue_struct *workqueue; + struct work_struct pump_messages; + spinlock_t lock; + struct list_head queue; + int busy; + int run; + + /* Message Transfer pump */ + struct tasklet_struct pump_transfers; + + /* Current message transfer state info */ + struct spi_message* cur_msg; + struct spi_transfer* cur_transfer; + struct chip_data *cur_chip; + size_t len; + void *tx; + void *tx_end; + void *rx; + void *rx_end; + char flags; +#define TRAN_STATE_RX_VOID 0x01 +#define TRAN_STATE_TX_VOID 0x02 +#define TRAN_STATE_WORD_ODD_NUM 0x04 + u8 cs; + u16 void_write_data; + unsigned cs_change:1; + + u32 trans_cnt; + u32 wce_cnt; + u32 abrt_cnt; +#if defined(SPI_DSPI) + u32 *mcr; /* DSPI MCR register */ + u32 *ctar; /* DSPI CTAR register */ + u32 *dspi_dtfr; /* DSPI DTFR register */ + u32 *dspi_drfr; /* DSPI DRFR register */ + u32 *dspi_rser; /* DSPI RSER register */ + u32 *dspi_sr; /* DSPI status register */ + u8 dspi_ctas; /* DSPI CTAS value*/ + +#if defined(SPI_DSPI_EDMA) + void* edma_tx_buf; + void* edma_rx_buf; +#endif + + +#else + u16 *qmr; /* QSPI mode register */ + u16 *qdlyr; /* QSPI delay register */ + u16 *qwr; /* QSPI wrap register */ + u16 *qir; /* QSPI interrupt register */ + u16 *qar; /* QSPI address register */ + u16 *qdr; /* QSPI data register */ + u16 *qcr; /* QSPI command register */ +#endif + u8 *par; /* Pin assignment register */ + u8 *int_icr; /* Interrupt level and priority register */ + u32 *int_mr; /* Interrupt mask register */ + void (*cs_control)(u8 cs, u8 command); +}; + +#define DSPI_CS(cs) ((1<<(cs))<<16) + + +/****************************************************************************/ + +/* + * SPI local functions + */ + +//#define SPI_COLDFIRE_DEBUG + +static void *next_transfer(struct driver_data *drv_data) +{ + struct spi_message *msg = drv_data->cur_msg; + struct spi_transfer *trans = drv_data->cur_transfer; + + /* Move to next transfer */ + if (trans->transfer_list.next != &msg->transfers) { + drv_data->cur_transfer = + list_entry(trans->transfer_list.next, + struct spi_transfer, + transfer_list); + return RUNNING_STATE; + } else + return DONE_STATE; +} + + +#define DSPI_BITS MCF_DSPI_DCTAR_FMSZ(15) +#define DSPI_BITS_16 MCF_DSPI_DCTAR_FMSZ(15) +#define DSPI_BITS_8 MCF_DSPI_DCTAR_FMSZ(7) +#define DSPI_FIFO_SIZE 16 + +static inline int is_word_transfer(struct driver_data *drv_data) +{ +#if defined(SPI_DSPI) + return ((*drv_data->ctar & DSPI_BITS_16) == DSPI_BITS_8) ? 0 : 1; +#else + return ((*drv_data->qmr & QMR_BITS) == QMR_BITS_8) ? 0 : 1; +#endif +} + +static void inline set_8bit_transfer_mode(struct driver_data *drv_data) +{ +#if defined(SPI_DSPI) + *drv_data->ctar |= (*drv_data->ctar & ~DSPI_BITS) | DSPI_BITS_8; +#else + *drv_data->qmr |= (*drv_data->qmr & ~QMR_BITS) | QMR_BITS_8; +#endif +} + +static void inline set_16bit_transfer_mode(struct driver_data *drv_data) +{ +#if defined(SPI_DSPI) + *drv_data->ctar |= (*drv_data->ctar & ~DSPI_BITS) | DSPI_BITS_16; +#else + *drv_data->qmr |= (*drv_data->qmr & ~QMR_BITS); +#endif +} + +static int write(struct driver_data *drv_data) +{ + int tx_count = 0; +#ifndef SPI_DSPI + int cmd_count = 0; +#endif + int tx_word; + +#if defined(SPI_DSPI) + +#if defined(SPI_DSPI_EDMA) + u32* edma_wr; +#endif + + u16 d16; + u8 d8; + u32 dspi_pushr; + int first = 1; +#endif + + tx_word = is_word_transfer(drv_data); + + // If we are in word mode, but only have a single byte to transfer + // then switch to byte mode temporarily. Will switch back at the + // end of the transfer. + if (tx_word && ((drv_data->tx_end - drv_data->tx) == 1)) { + drv_data->flags |= TRAN_STATE_WORD_ODD_NUM; + set_8bit_transfer_mode(drv_data); + tx_word = 0; + } + + +#if defined(SPI_DSPI) + +#if defined(SPI_DSPI_EDMA) + edma_wr = (u32*)(drv_data->edma_tx_buf); +#endif + + +#if defined(SPI_DSPI_EDMA) + while ((drv_data->tx < drv_data->tx_end) && (tx_count < EDMA_BUFFER_SIZE)) { +#else + while ((drv_data->tx < drv_data->tx_end) && (tx_count < DSPI_FIFO_SIZE)) { +#endif + if (tx_word) { + if ((drv_data->tx_end - drv_data->tx) == 1) + break; + if (!(drv_data->flags & TRAN_STATE_TX_VOID)) { + d16 = *(u16 *)drv_data->tx; + } else { + d16 = drv_data->void_write_data; + } + + dspi_pushr = MCF_DSPI_DTFR_TXDATA(d16) + | DSPI_CS(drv_data->cs) + | MCF_DSPI_DTFR_CTAS(drv_data->dspi_ctas) + //| MCF_DSPI_DTFR_CONT + ; + + drv_data->tx += 2; + +#if defined(SPI_DSPI_EDMA) + if (drv_data->tx == drv_data->tx_end || tx_count==EDMA_BUFFER_SIZE-1) { +#else + if (drv_data->tx == drv_data->tx_end || tx_count==DSPI_FIFO_SIZE-1) { +#endif + // last transfer in queue + dspi_pushr |= MCF_DSPI_DTFR_EOQ; + if (drv_data->cs_change) { + dspi_pushr &= ~MCF_DSPI_DTFR_CONT; + } + } + + if (first) { + first = 0; + dspi_pushr |= MCF_DSPI_DTFR_CTCNT; // clear counter + } +#if defined(SPI_DSPI_EDMA) + *edma_wr = dspi_pushr; + edma_wr++; +#else + *drv_data->dspi_dtfr = dspi_pushr; + //MCF_DSPI_DTFR = dspi_pushr; +#endif + + + } else { + if (!(drv_data->flags & TRAN_STATE_TX_VOID)) { + d8 = *(u8 *)drv_data->tx; + } else { + d8 = *(u8 *)&drv_data->void_write_data; + } + + dspi_pushr = MCF_DSPI_DTFR_TXDATA(d8) + | DSPI_CS(drv_data->cs) + /* | MCF_DSPI_DTFR_PCS5 | */ + | MCF_DSPI_DTFR_CTAS(drv_data->dspi_ctas) + | MCF_DSPI_DTFR_CONT; + + drv_data->tx++; + + if (drv_data->tx == drv_data->tx_end || tx_count==DSPI_FIFO_SIZE-1) { + // last transfer in queue + dspi_pushr |= MCF_DSPI_DTFR_EOQ; + if (drv_data->cs_change) { + dspi_pushr &= ~MCF_DSPI_DTFR_CONT; + } + } + + if (first) { + first = 0; + dspi_pushr |= MCF_DSPI_DTFR_CTCNT; // clear counter + } + +#if defined(SPI_DSPI_EDMA) + *edma_wr = dspi_pushr; + edma_wr++; +#else + *drv_data->dspi_dtfr = dspi_pushr; + //MCF_DSPI_DTFR = dspi_pushr; +#endif + + } + tx_count++; + } + +#if defined(SPI_DSPI_EDMA) + + if (tx_count>0) { + + // TODO: initiate eDMA transfer + set_edma_params(DSPI_DMA_TX_TCD, +#ifdef SPI_USE_MMU + virt_to_phys(drv_data->edma_tx_buf), +#else + drv_data->edma_tx_buf, +#endif + (u32)drv_data->dspi_dtfr, + MCF_EDMA_TCD_ATTR_SSIZE_32BIT | MCF_EDMA_TCD_ATTR_DSIZE_32BIT, + 4, // soff + 4, // nbytes + 0, // slast + tx_count, // citer + tx_count, // biter + 0, // doff + 0, // dlastsga + 0, // major_int + 1 // disable_req + ); + + set_edma_params(DSPI_DMA_RX_TCD, + (u32)drv_data->dspi_drfr, +#ifdef SPI_USE_MMU + virt_to_phys(drv_data->edma_rx_buf), +#else + drv_data->edma_rx_buf, +#endif + MCF_EDMA_TCD_ATTR_SSIZE_32BIT | MCF_EDMA_TCD_ATTR_DSIZE_32BIT, + 0, // soff + 4, // nbytes + 0, // slast + tx_count, // citer + tx_count, // biter + 4, // doff + 0, // dlastsga + 0, // major_int + 1 // disable_req + ); + + + start_edma_transfer(DSPI_DMA_TX_TCD); // transmit SPI data + start_edma_transfer(DSPI_DMA_RX_TCD); // receive SPI data + } +#endif + +#else + + *drv_data->qar = QSPI_TRANSMIT_RAM; + while ((drv_data->tx < drv_data->tx_end) && (tx_count < QSPI_RAM_SIZE)) { + if (tx_word) { + if ((drv_data->tx_end - drv_data->tx) == 1) + break; + + if (!(drv_data->flags & TRAN_STATE_TX_VOID)) + *drv_data->qdr = *(u16 *)drv_data->tx; + else + *drv_data->qdr = drv_data->void_write_data; + drv_data->tx += 2; + } else { + if (!(drv_data->flags & TRAN_STATE_TX_VOID)) + *drv_data->qdr = *(u8 *)drv_data->tx; + else + *drv_data->qdr = *(u8 *)&drv_data->void_write_data; + drv_data->tx++; + } + tx_count++; + } + + + *drv_data->qar = QSPI_COMMAND_RAM; + while (cmd_count < tx_count) { + u16 qcr = QSPI_COMMAND + | QCR_CONT + | (~((0x01 << drv_data->cs) << 8) & 0x0F00); + + if ( (cmd_count == tx_count - 1) + && (drv_data->tx == drv_data->tx_end) + && (drv_data->cs_change) ) { + qcr &= ~QCR_CONT; + } + *drv_data->qcr = qcr; + cmd_count++; + } + + *drv_data->qwr = (*drv_data->qwr & ~QWR_ENDQP_MASK) | ((cmd_count - 1) << 8); + + /* Fire it up! */ + *drv_data->qdlyr |= QDLYR_SPE; +#endif + + return tx_count; +} + + +static int read(struct driver_data *drv_data) +{ + int rx_count = 0; + int rx_word; +#if defined(SPI_DSPI_EDMA) + u32* rx_edma; +#endif + u16 d; + rx_word = is_word_transfer(drv_data); + +#if defined(SPI_DSPI) + +#if defined(SPI_DSPI_EDMA) + rx_edma = (u32*) drv_data->edma_tx_buf; + while ((drv_data->rx < drv_data->rx_end) && (rx_count < EDMA_BUFFER_SIZE)) { +#else + while ((drv_data->rx < drv_data->rx_end) && (rx_count < DSPI_FIFO_SIZE)) { +#endif + if (rx_word) { + if ((drv_data->rx_end - drv_data->rx) == 1) + break; +#if defined(SPI_DSPI_EDMA) + d = MCF_DSPI_DRFR_RXDATA(*rx_edma); + rx_edma++; +#else + d = MCF_DSPI_DRFR_RXDATA(*drv_data->dspi_drfr); +#endif + + if (!(drv_data->flags & TRAN_STATE_RX_VOID)) + *(u16 *)drv_data->rx = d; + drv_data->rx += 2; + } else { +#if defined(SPI_DSPI_EDMA) + d = MCF_DSPI_DRFR_RXDATA(*rx_edma); + rx_edma++; +#else + d = MCF_DSPI_DRFR_RXDATA(*drv_data->dspi_drfr); +#endif + if (!(drv_data->flags & TRAN_STATE_RX_VOID)) + *(u8 *)drv_data->rx = d; + drv_data->rx++; + } + rx_count++; + } + + +#else + + *drv_data->qar = QSPI_RECEIVE_RAM; + while ((drv_data->rx < drv_data->rx_end) && (rx_count < QSPI_RAM_SIZE)) { + if (rx_word) { + if ((drv_data->rx_end - drv_data->rx) == 1) + break; + + if (!(drv_data->flags & TRAN_STATE_RX_VOID)) + *(u16 *)drv_data->rx = *drv_data->qdr; + drv_data->rx += 2; + } else { + if (!(drv_data->flags & TRAN_STATE_RX_VOID)) + *(u8 *)drv_data->rx = *drv_data->qdr; + drv_data->rx++; + } + rx_count++; + } +#endif + + return rx_count; +} + + +static inline void qspi_setup_chip(struct driver_data *drv_data) +{ + struct chip_data *chip = drv_data->cur_chip; + +#if defined(SPI_DSPI) + + *drv_data->mcr = chip->mcr_val; + + // TODO: remove later + chip->ctar_val = 0x78560118; + + *drv_data->ctar = chip->ctar_val; + *drv_data->dspi_rser = 0 + | MCF_DSPI_DRSER_EOQFE +#if defined(SPI_DSPI_EDMA) + | MCF_DSPI_DRSER_TFFFE + | MCF_DSPI_DRSER_TFFFS +#endif + ; + + +#else + *drv_data->qmr = chip->qmr_val; + *drv_data->qdlyr = chip->qdlyr_val; + *drv_data->qwr = chip->qwr_val; + + /* + * Enable all the interrupts and clear all the flags + */ + *drv_data->qir = (QIR_SPIFE | QIR_ABRTE | QIR_WCEFE) + | (QIR_WCEFB | QIR_ABRTB | QIR_ABRTL) + | (QIR_SPIF | QIR_ABRT | QIR_WCEF); +#endif +} + +#if defined(SPI_DSPI_EDMA) +static int edma_tx_handler(int channel, void* dev) +{ + if (channel == DSPI_DMA_TX_TCD) { + stop_edma_transfer(DSPI_DMA_TX_TCD); + } + return IRQ_HANDLED; +} + +static int edma_rx_handler(int channel, void* dev) +{ + if (channel == DSPI_DMA_RX_TCD) { + stop_edma_transfer(DSPI_DMA_RX_TCD); + } + + return IRQ_HANDLED; +} +#endif + +static irqreturn_t qspi_interrupt(int irq, void *dev_id) +{ + struct driver_data *drv_data = (struct driver_data *)dev_id; + struct spi_message *msg = drv_data->cur_msg; +#if defined(SPI_DSPI) +#if !defined(SPI_DSPI_EDMA) + u32 irq_status = *drv_data->dspi_sr; +#endif +#else + u16 irq_status = *drv_data->qir; +#endif + + /* Clear all flags immediately */ +#if defined(SPI_DSPI) + *drv_data->dspi_sr = MCF_DSPI_DSR_EOQF; +#else + *drv_data->qir |= (QIR_SPIF | QIR_ABRT | QIR_WCEF); +#endif + + if (!drv_data->cur_msg || !drv_data->cur_msg->state) { +#if !defined(SPI_DSPI_EDMA) + /* if eDMA is used it happens some time (at least once)*/ + printk(KERN_ERR "coldfire-qspi: bad message or transfer " + "state in interrupt handler. IRQ status=%x\n", irq_status); +#endif + return IRQ_NONE; + } + +#if !defined(SPI_DSPI) + if (irq_status & QIR_SPIF) { +#endif + /* + * Read the data into the buffer and reload and start + * queue with new data if not finished. If finished + * then setup the next transfer + */ + read(drv_data); + + if (drv_data->rx == drv_data->rx_end) { + /* + * Finished now - fall through and schedule next + * transfer tasklet + */ + if (drv_data->flags & TRAN_STATE_WORD_ODD_NUM) { + //*drv_data->qmr &= ~QMR_BITS; + set_16bit_transfer_mode(drv_data); + } + + msg->state = next_transfer(drv_data); + msg->actual_length += drv_data->len; + } else { + /* not finished yet - keep going */ + write(drv_data); + return IRQ_HANDLED; + } +#if !defined(SPI_DSPI) + } else { + if (irq_status & QIR_WCEF) + drv_data->wce_cnt++; + + if (irq_status & QIR_ABRT) + drv_data->abrt_cnt++; + + msg->state = ERROR_STATE; + } +#endif + + tasklet_schedule(&drv_data->pump_transfers); + + return IRQ_HANDLED; +} + +/* caller already set message->status; dma and pio irqs are blocked */ +static void giveback(struct driver_data *drv_data) +{ + struct spi_transfer* last_transfer; + unsigned long flags; + struct spi_message *msg; + + spin_lock_irqsave(&drv_data->lock, flags); + msg = drv_data->cur_msg; + drv_data->cur_msg = NULL; + drv_data->cur_transfer = NULL; + drv_data->cur_chip = NULL; + queue_work(drv_data->workqueue, &drv_data->pump_messages); + spin_unlock_irqrestore(&drv_data->lock, flags); + + last_transfer = list_entry(msg->transfers.prev, + struct spi_transfer, + transfer_list); + + if (!last_transfer->cs_change) + drv_data->cs_control(drv_data->cs, QSPI_CS_DROP); + + msg->state = NULL; + if (msg->complete) + msg->complete(msg->context); +} + + +static void pump_transfers(unsigned long data) +{ + struct driver_data *drv_data = (struct driver_data *)data; + struct spi_message *message = NULL; + struct spi_transfer *transfer = NULL; + struct spi_transfer *previous = NULL; + struct chip_data *chip = NULL; + unsigned long flags; + + /* Get current state information */ + message = drv_data->cur_msg; + transfer = drv_data->cur_transfer; + chip = drv_data->cur_chip; + + /* Handle for abort */ + if (message->state == ERROR_STATE) { + message->status = -EIO; + giveback(drv_data); + return; + } + + /* Handle end of message */ + if (message->state == DONE_STATE) { + message->status = 0; + giveback(drv_data); + return; + } + + if (message->state == START_STATE) { + qspi_setup_chip(drv_data); + + if (drv_data->cs_control) { + //printk( "m s\n" ); + drv_data->cs_control(message->spi->chip_select, QSPI_CS_ASSERT); + } + } + + /* Delay if requested at end of transfer*/ + if (message->state == RUNNING_STATE) { + previous = list_entry(transfer->transfer_list.prev, + struct spi_transfer, + transfer_list); + + if (drv_data->cs_control && transfer->cs_change) + drv_data->cs_control(message->spi->chip_select, QSPI_CS_DROP); + + if (previous->delay_usecs) + udelay(previous->delay_usecs); + + if (drv_data->cs_control && transfer->cs_change) + drv_data->cs_control(message->spi->chip_select, QSPI_CS_ASSERT); + } + + drv_data->flags = 0; + drv_data->tx = (void *)transfer->tx_buf; + drv_data->tx_end = drv_data->tx + transfer->len; + drv_data->rx = transfer->rx_buf; + drv_data->rx_end = drv_data->rx + transfer->len; + drv_data->len = transfer->len; + if (!drv_data->rx) + drv_data->flags |= TRAN_STATE_RX_VOID; + if (!drv_data->tx) + drv_data->flags |= TRAN_STATE_TX_VOID; + drv_data->cs = message->spi->chip_select; + drv_data->cs_change = transfer->cs_change; + drv_data->void_write_data = chip->void_write_data; + + message->state = RUNNING_STATE; + + /* Go baby, go */ + local_irq_save(flags); + write(drv_data); + local_irq_restore(flags); +} + + +static void pump_messages(struct work_struct * work) +{ + struct driver_data *drv_data; + unsigned long flags; + + drv_data = container_of(work, struct driver_data, pump_messages); + + /* Lock queue and check for queue work */ + spin_lock_irqsave(&drv_data->lock, flags); + if (list_empty(&drv_data->queue) || drv_data->run == QUEUE_STOPPED) { + drv_data->busy = 0; + spin_unlock_irqrestore(&drv_data->lock, flags); + return; + } + + /* Make sure we are not already running a message */ + if (drv_data->cur_msg) { + spin_unlock_irqrestore(&drv_data->lock, flags); + return; + } + + /* Extract head of queue */ + drv_data->cur_msg = list_entry(drv_data->queue.next, + struct spi_message, queue); + list_del_init(&drv_data->cur_msg->queue); + + /* Initial message state*/ + drv_data->cur_msg->state = START_STATE; + drv_data->cur_transfer = list_entry(drv_data->cur_msg->transfers.next, + struct spi_transfer, + transfer_list); + + /* Setup the SPI Registers using the per chip configuration */ + drv_data->cur_chip = spi_get_ctldata(drv_data->cur_msg->spi); + + /* Mark as busy and launch transfers */ + tasklet_schedule(&drv_data->pump_transfers); + + drv_data->busy = 1; + spin_unlock_irqrestore(&drv_data->lock, flags); +} + +/****************************************************************************/ + +/* + * SPI master implementation + */ + +static int transfer(struct spi_device *spi, struct spi_message *msg) +{ + struct driver_data *drv_data = spi_master_get_devdata(spi->master); + unsigned long flags; + + spin_lock_irqsave(&drv_data->lock, flags); + + if (drv_data->run == QUEUE_STOPPED) { + spin_unlock_irqrestore(&drv_data->lock, flags); + return -ESHUTDOWN; + } + + msg->actual_length = 0; + msg->status = -EINPROGRESS; + msg->state = START_STATE; + + list_add_tail(&msg->queue, &drv_data->queue); + + if (drv_data->run == QUEUE_RUNNING && !drv_data->busy) + queue_work(drv_data->workqueue, &drv_data->pump_messages); + + spin_unlock_irqrestore(&drv_data->lock, flags); + + return 0; +} + + +static int setup(struct spi_device *spi) +{ + struct coldfire_spi_chip *chip_info; + struct chip_data *chip; +#ifndef SPI_DSPI + u32 baud_divisor = 255; +#endif + + chip_info = (struct coldfire_spi_chip *)spi->controller_data; + + /* Only alloc on first setup */ + chip = spi_get_ctldata(spi); + if (chip == NULL) { + chip = kcalloc(1, sizeof(struct chip_data), GFP_KERNEL); + if (!chip) + return -ENOMEM; + spi->mode = chip_info->mode; + spi->bits_per_word = chip_info->bits_per_word; + } + +#if defined(SPI_DSPI) + chip->mcr.master = 1; + chip->mcr.cont_scke = 0; + chip->mcr.dconf = 0; + chip->mcr.frz = 0; + chip->mcr.mtfe = 1; + chip->mcr.pcsse = 0; + chip->mcr.rooe = 0; + chip->mcr.pcsis = 0xFF; + chip->mcr.reserved15 = 0; + chip->mcr.mdis = 0; + chip->mcr.dis_tx = 0; + chip->mcr.dis_rxf = 0; + chip->mcr.clr_tx = 1; + chip->mcr.clr_rxf = 1; + chip->mcr.smpl_pt = 0; + chip->mcr.reserved71 = 0; + chip->mcr.halt = 0; + + if ((spi->bits_per_word >= 4) && (spi->bits_per_word <= 16)) { + chip->ctar.fmsz = spi->bits_per_word-1; + } else { + printk(KERN_ERR "coldfire-qspi: invalid wordsize\n"); + kfree(chip); + return -ENODEV; + } + + if (spi->mode & SPI_CPHA) + chip->ctar.cpha = 1; + else + chip->ctar.cpha = 0; + + if (spi->mode & SPI_CPOL) + chip->ctar.cpol = 1; + else + chip->ctar.cpol = 0; + + if (spi->mode & SPI_LSB_FIRST) + chip->ctar.lsbfe = 1; + else + chip->ctar.lsbfe = 0; + + /* This values are default for audio device */ + chip->ctar.dbr = 0; + chip->ctar.pbr = 2; + chip->ctar.br = 8; + + /* This values are default for audio device */ + chip->ctar.pcssck = 1; + chip->ctar.pasc = 1; + chip->ctar.pdt = 1; + chip->ctar.cssck = 0; + chip->ctar.asc = 1; + chip->ctar.dt = 1; + + chip->void_write_data = chip_info->void_write_data; + +#else + + chip->qwr.csiv = 1; // Chip selects are active low + chip->qmr.master = 1; // Must set to master mode + chip->qmr.dohie = 1; // Data output high impediance enabled + chip->void_write_data = chip_info->void_write_data; + + chip->qdlyr.qcd = chip_info->del_cs_to_clk; + chip->qdlyr.dtl = chip_info->del_after_trans; + + if (spi->max_speed_hz != 0) + baud_divisor = (MCF_CLK/(2*spi->max_speed_hz)); + + if (baud_divisor < 2) + baud_divisor = 2; + + if (baud_divisor > 255) + baud_divisor = 255; + + chip->qmr.baud = baud_divisor; + + //printk( "QSPI: spi->max_speed_hz %d\n", spi->max_speed_hz ); + //printk( "QSPI: Baud set to %d\n", chip->qmr.baud ); + + if (spi->mode & SPI_CPHA) + chip->qmr.cpha = 1; + + if (spi->mode & SPI_CPOL) + chip->qmr.cpol = 1; + + if (spi->bits_per_word == 16) { + chip->qmr.bits = 0; + } else if ((spi->bits_per_word >= 8) && (spi->bits_per_word <= 15)) { + chip->qmr.bits = spi->bits_per_word; + } else { + printk(KERN_ERR "coldfire-qspi: invalid wordsize\n"); + kfree(chip); + return -ENODEV; + } + +#endif + + spi_set_ctldata(spi, chip); + + return 0; +} + +static int init_queue(struct driver_data *drv_data) +{ + INIT_LIST_HEAD(&drv_data->queue); + spin_lock_init(&drv_data->lock); + + drv_data->run = QUEUE_STOPPED; + drv_data->busy = 0; + + tasklet_init(&drv_data->pump_transfers, + pump_transfers, (unsigned long)drv_data); + + INIT_WORK(&drv_data->pump_messages, pump_messages/*, drv_data*/); + + drv_data->workqueue = create_singlethread_workqueue( + drv_data->master->cdev.dev->bus_id); + if (drv_data->workqueue == NULL) + return -EBUSY; + + return 0; +} + +static int start_queue(struct driver_data *drv_data) +{ + unsigned long flags; + + spin_lock_irqsave(&drv_data->lock, flags); + + if (drv_data->run == QUEUE_RUNNING || drv_data->busy) { + spin_unlock_irqrestore(&drv_data->lock, flags); + return -EBUSY; + } + + drv_data->run = QUEUE_RUNNING; + drv_data->cur_msg = NULL; + drv_data->cur_transfer = NULL; + drv_data->cur_chip = NULL; + spin_unlock_irqrestore(&drv_data->lock, flags); + + queue_work(drv_data->workqueue, &drv_data->pump_messages); + + return 0; +} + +static int stop_queue(struct driver_data *drv_data) +{ + unsigned long flags; + unsigned limit = 500; + int status = 0; + + spin_lock_irqsave(&drv_data->lock, flags); + + /* This is a bit lame, but is optimized for the common execution path. + * A wait_queue on the drv_data->busy could be used, but then the common + * execution path (pump_messages) would be required to call wake_up or + * friends on every SPI message. Do this instead */ + drv_data->run = QUEUE_STOPPED; + while (!list_empty(&drv_data->queue) && drv_data->busy && limit--) { + spin_unlock_irqrestore(&drv_data->lock, flags); + msleep(10); + spin_lock_irqsave(&drv_data->lock, flags); + } + + if (!list_empty(&drv_data->queue) || drv_data->busy) + status = -EBUSY; + + spin_unlock_irqrestore(&drv_data->lock, flags); + + return status; +} + +static int destroy_queue(struct driver_data *drv_data) +{ + int status; + + status = stop_queue(drv_data); + if (status != 0) + return status; + + destroy_workqueue(drv_data->workqueue); + + return 0; +} + + +static void cleanup(const struct spi_device *spi) +{ + struct chip_data *chip = spi_get_ctldata((struct spi_device *)spi); + + dev_dbg(&spi->dev, "spi_device %u.%u cleanup\n", + spi->master->bus_num, spi->chip_select); + + kfree(chip); +} + + +/****************************************************************************/ + +/* + * Generic Device driver routines and interface implementation + */ + +static int coldfire_spi_probe(struct platform_device *pdev) +{ + struct device *dev = &pdev->dev; + struct coldfire_spi_master *platform_info; + struct spi_master *master; + struct driver_data *drv_data = 0; + struct resource *memory_resource; + int irq; + int status = 0; + int i; + +#if defined(SPI_DSPI_EDMA) + init_edma(); +#endif + + platform_info = (struct coldfire_spi_master *)pdev->dev.platform_data; + + master = spi_alloc_master(dev, sizeof(struct driver_data)); + if (!master) + return -ENOMEM; + + drv_data = class_get_devdata(&master->cdev); + drv_data->master = master; + + INIT_LIST_HEAD(&drv_data->queue); + spin_lock_init(&drv_data->lock); + + master->bus_num = platform_info->bus_num; + master->num_chipselect = platform_info->num_chipselect; + master->cleanup = cleanup; + master->setup = setup; + master->transfer = transfer; + + drv_data->cs_control = platform_info->cs_control; + if (drv_data->cs_control) + for(i = 0; i < master->num_chipselect; i++) + drv_data->cs_control(i, QSPI_CS_INIT | QSPI_CS_DROP); + + /* Setup register addresses */ + memory_resource = platform_get_resource_byname(pdev, IORESOURCE_MEM, "qspi-module"); + if (!memory_resource) { + dev_dbg(dev, "can not find platform module memory\n"); + goto out_error_master_alloc; + } + +#if defined(SPI_DSPI_EDMA) + drv_data->edma_tx_buf = kmalloc(EDMA_BUFSIZE_KMALLOC, GFP_DMA); + if (!drv_data->edma_tx_buf) { + dev_dbg(dev, "cannot allocate eDMA TX memory\n"); + goto out_error_master_alloc; + } + drv_data->edma_rx_buf = kmalloc(EDMA_BUFSIZE_KMALLOC, GFP_DMA); + if (!drv_data->edma_rx_buf) { + kfree(drv_data->edma_tx_buf); + dev_dbg(dev, "cannot allocate eDMA RX memory\n"); + goto out_error_master_alloc; + } +#endif + +#if defined(SPI_DSPI) + + drv_data->mcr = (void *)(memory_resource->start + 0x00000000); + drv_data->ctar = (void *)(memory_resource->start + 0x0000000C); + drv_data->dspi_sr = (void *)(memory_resource->start + 0x0000002C); + drv_data->dspi_rser = (void *)(memory_resource->start + 0x00000030); + drv_data->dspi_dtfr = (void *)(memory_resource->start + 0x00000034); + drv_data->dspi_drfr = (void *)(memory_resource->start + 0x00000038); + +#else + + drv_data->qmr = (void *)(memory_resource->start + 0x00000000); + drv_data->qdlyr = (void *)(memory_resource->start + 0x00000004); + drv_data->qwr = (void *)(memory_resource->start + 0x00000008); + drv_data->qir = (void *)(memory_resource->start + 0x0000000c); + drv_data->qar = (void *)(memory_resource->start + 0x00000010); + drv_data->qdr = (void *)(memory_resource->start + 0x00000014); + drv_data->qcr = (void *)(memory_resource->start + 0x00000014); + +#endif + + /* Setup register addresses */ + memory_resource = platform_get_resource_byname(pdev, IORESOURCE_MEM, "qspi-par"); + if (!memory_resource) { + dev_dbg(dev, "can not find platform par memory\n"); + goto out_error_master_alloc; + } + + drv_data->par = (void *)memory_resource->start; + + /* Setup register addresses */ + memory_resource = platform_get_resource_byname(pdev, IORESOURCE_MEM, "qspi-int-level"); + if (!memory_resource) { + dev_dbg(dev, "can not find platform par memory\n"); + goto out_error_master_alloc; + } + + drv_data->int_icr = (void *)memory_resource->start; + + /* Setup register addresses */ + memory_resource = platform_get_resource_byname(pdev, IORESOURCE_MEM, "qspi-int-mask"); + if (!memory_resource) { + dev_dbg(dev, "can not find platform par memory\n"); + goto out_error_master_alloc; + } + + drv_data->int_mr = (void *)memory_resource->start; + + irq = platform_info->irq_vector; + + status = request_irq(platform_info->irq_vector, qspi_interrupt, SA_INTERRUPT, dev->bus_id, drv_data); + if (status < 0) { + dev_err(&pdev->dev, "unable to attach ColdFire QSPI interrupt\n"); + goto out_error_master_alloc; + } + + /* Now that we have all the addresses etc. Let's set it up */ + // TODO: + //*drv_data->par = platform_info->par_val; + + MCF_GPIO_PAR_DSPI = 0 + | MCF_GPIO_PAR_DSPI_PCS5_PCS5 + | MCF_GPIO_PAR_DSPI_PCS2_PCS2 + | MCF_GPIO_PAR_DSPI_PCS1_PCS1 + | MCF_GPIO_PAR_DSPI_PCS0_PCS0 + | MCF_GPIO_PAR_DSPI_SIN_SIN + | MCF_GPIO_PAR_DSPI_SOUT_SOUT + | MCF_GPIO_PAR_DSPI_SCK_SCK; + + *drv_data->int_icr = platform_info->irq_lp; + *drv_data->int_mr &= ~platform_info->irq_mask; + +#ifdef SPI_DSPI + drv_data->dspi_ctas = 0; // TODO: change later +#endif + + /* Initial and start queue */ + status = init_queue(drv_data); + if (status != 0) { + dev_err(&pdev->dev, "problem initializing queue\n"); + goto out_error_irq_alloc; + } + status = start_queue(drv_data); + if (status != 0) { + dev_err(&pdev->dev, "problem starting queue\n"); + goto out_error_irq_alloc; + } + + /* Register with the SPI framework */ + platform_set_drvdata(pdev, drv_data); + status = spi_register_master(master); + if (status != 0) { + dev_err(&pdev->dev, "problem registering spi master\n"); + status = -EINVAL; + goto out_error_queue_alloc; + } + +#if defined(SPI_DSPI_EDMA) + if (request_edma_channel(DSPI_DMA_TX_TCD, + edma_tx_handler, + NULL, + pdev, + NULL, /* spinlock */ + DRIVER_NAME + )!=0) + { + dev_err(&pdev->dev, "problem requesting edma transmit channel\n"); + status = -EINVAL; + goto out_error_queue_alloc; + } + + if (request_edma_channel(DSPI_DMA_RX_TCD, + edma_rx_handler, + NULL, + pdev, + NULL, /* spinlock */ + DRIVER_NAME + )!=0) + { + dev_err(&pdev->dev, "problem requesting edma receive channel\n"); + status = -EINVAL; + goto out_edma_transmit; + } +#endif + + printk( "SPI: Coldfire master initialized\n" ); + //dev_info(&pdev->dev, "driver initialized\n"); + return status; + +#if defined(SPI_DSPI_EDMA) +out_edma_transmit: + free_edma_channel(DSPI_DMA_TX_TCD, pdev); +#endif + +out_error_queue_alloc: + destroy_queue(drv_data); + +out_error_irq_alloc: + free_irq(irq, drv_data); + +out_error_master_alloc: + spi_master_put(master); + return status; + +} + +static int coldfire_spi_remove(struct platform_device *pdev) +{ + struct driver_data *drv_data = platform_get_drvdata(pdev); + int irq; + int status = 0; + + if (!drv_data) + return 0; + +#if defined(SPI_DSPI_EDMA) + free_edma_channel(DSPI_DMA_TX_TCD, pdev); + free_edma_channel(DSPI_DMA_RX_TCD, pdev); +#endif + + /* Remove the queue */ + status = destroy_queue(drv_data); + if (status != 0) + return status; + + /* Disable the SSP at the peripheral and SOC level */ + /*write_SSCR0(0, drv_data->ioaddr); + pxa_set_cken(drv_data->master_info->clock_enable, 0);*/ + + /* Release DMA */ + /*if (drv_data->master_info->enable_dma) { + if (drv_data->ioaddr == SSP1_VIRT) { + DRCMRRXSSDR = 0; + DRCMRTXSSDR = 0; + } else if (drv_data->ioaddr == SSP2_VIRT) { + DRCMRRXSS2DR = 0; + DRCMRTXSS2DR = 0; + } else if (drv_data->ioaddr == SSP3_VIRT) { + DRCMRRXSS3DR = 0; + DRCMRTXSS3DR = 0; + } + pxa_free_dma(drv_data->tx_channel); + pxa_free_dma(drv_data->rx_channel); + }*/ + + /* Release IRQ */ + irq = platform_get_irq(pdev, 0); + if (irq >= 0) + free_irq(irq, drv_data); + + /* Disconnect from the SPI framework */ + spi_unregister_master(drv_data->master); + + /* Prevent double remove */ + platform_set_drvdata(pdev, NULL); + + return 0; +} + +static void coldfire_spi_shutdown(struct platform_device *pdev) +{ + int status = 0; + + if ((status = coldfire_spi_remove(pdev)) != 0) + dev_err(&pdev->dev, "shutdown failed with %d\n", status); +} + + +#ifdef CONFIG_PM +static int suspend_devices(struct device *dev, void *pm_message) +{ + pm_message_t *state = pm_message; + + if (dev->power.power_state.event != state->event) { + dev_warn(dev, "pm state does not match request\n"); + return -1; + } + + return 0; +} + +static int coldfire_spi_suspend(struct platform_device *pdev, pm_message_t state) +{ + struct driver_data *drv_data = platform_get_drvdata(pdev); + int status = 0; + + /* Check all childern for current power state */ + if (device_for_each_child(&pdev->dev, &state, suspend_devices) != 0) { + dev_warn(&pdev->dev, "suspend aborted\n"); + return -1; + } + + status = stop_queue(drv_data); + if (status != 0) + return status; + /*write_SSCR0(0, drv_data->ioaddr); + pxa_set_cken(drv_data->master_info->clock_enable, 0);*/ + + return 0; +} + +static int coldfire_spi_resume(struct platform_device *pdev) +{ + struct driver_data *drv_data = platform_get_drvdata(pdev); + int status = 0; + + /* Enable the SSP clock */ + /*pxa_set_cken(drv_data->master_info->clock_enable, 1);*/ + + /* Start the queue running */ + status = start_queue(drv_data); + if (status != 0) { + dev_err(&pdev->dev, "problem starting queue (%d)\n", status); + return status; + } + + return 0; +} +#else +#define coldfire_spi_suspend NULL +#define coldfire_spi_resume NULL +#endif /* CONFIG_PM */ + +static struct platform_driver driver = { + .driver = { + .name = "spi_coldfire", + .bus = &platform_bus_type, + .owner = THIS_MODULE, + }, + .probe = coldfire_spi_probe, + .remove = __devexit_p(coldfire_spi_remove), + .shutdown = coldfire_spi_shutdown, + .suspend = coldfire_spi_suspend, + .resume = coldfire_spi_resume, +}; + +static int __init coldfire_spi_init(void) +{ + platform_driver_register(&driver); + + return 0; +} +module_init(coldfire_spi_init); + +static void __exit coldfire_spi_exit(void) +{ + platform_driver_unregister(&driver); +} +module_exit(coldfire_spi_exit); --- /dev/null +++ b/drivers/spi/ssi_audio.c @@ -0,0 +1,906 @@ +/* + * MCF5445x audio driver. + * + * Yaroslav Vinogradov yaroslav.vinogradov@freescale.com + * Copyright Freescale Semiconductor, Inc. 2006 + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 2 of the License, or + * (at your option) any later version. + */ + +#include <linux/device.h> +#include <linux/init.h> +#include <linux/delay.h> +#include <linux/spi/spi.h> +#include <linux/fs.h> +#include <linux/kernel.h> +#include <linux/major.h> +#include <asm/mcfsim.h> +#include <linux/interrupt.h> +#include <linux/soundcard.h> +#include <asm/uaccess.h> +#include <asm/virtconvert.h> + +#include <asm/coldfire.h> +#include <asm/coldfire_edma.h> +#include <asm/mcf5445x_ssi.h> +#include <asm/mcf5445x_ccm.h> +#include <asm/mcf5445x_gpio.h> + +#define SOUND_DEVICE_NAME "sound" +#define DRIVER_NAME "ssi_audio" + + +/* #define AUDIO_DEBUG */ + +#ifdef CONFIG_MMU +#define USE_MMU +#endif + +#define MAX_SPEED_HZ 12000000 + +#define M5445x_AUDIO_IRQ_SOURCE 49 +#define M5445x_AUDIO_IRQ_VECTOR (128+M5445x_AUDIO_IRQ_SOURCE) +#define M5445x_AUDIO_IRQ_LEVEL 5 + +/* TLV320DAC23 audio chip registers */ + +#define CODEC_LEFT_IN_REG (0x00) +#define CODEC_RIGHT_IN_REG (0x01) +#define CODEC_LEFT_HP_VOL_REG (0x02) +#define CODEC_RIGHT_HP_VOL_REG (0x03) +#define CODEC_ANALOG_APATH_REG (0x04) +#define CODEC_DIGITAL_APATH_REG (0x05) +#define CODEC_POWER_DOWN_REG (0x06) +#define CODEC_DIGITAL_IF_FMT_REG (0x07) +#define CODEC_SAMPLE_RATE_REG (0x08) +#define CODEC_DIGITAL_IF_ACT_REG (0x09) +#define CODEC_RESET_REG (0x0f) + +#define CODEC_SAMPLE_8KHZ (0x0C) +#define CODEC_SAMPLE_16KHZ (0x58) +#define CODEC_SAMPLE_22KHZ (0x62) +#define CODEC_SAMPLE_32KHZ (0x18) +#define CODEC_SAMPLE_44KHZ (0x22) +#define CODEC_SAMPLE_48KHZ (0x00) + +/* Audio buffer data size */ +#define BUFSIZE (64*1024) +/* DMA transfer size */ +#define DMASIZE (16*1024) + +/* transmit eDMA channel for SSI channel 0 */ +#define DMA_TCD 10 +/* transmit eDMA channel for SSI channel 1 */ +#define DMA_TCD2 11 + +struct ssi_audio { + struct spi_device *spi; + u32 speed; + u32 stereo; + u32 bits; + u32 format; + u8 isopen; + u8 dmaing; + u8 ssi_enabled; + u8 channel; + spinlock_t lock; + u8* audio_buf; +}; + +static struct ssi_audio* audio_device = NULL; +volatile u32 audio_start; +volatile u32 audio_count; +volatile u32 audio_append; +volatile u32 audio_appstart; +volatile u32 audio_txbusy; + +struct ssi_audio_format { + unsigned int format; + unsigned int bits; +} ssi_audio_formattable[] = { + { AFMT_MU_LAW, 8 }, + { AFMT_A_LAW, 8 }, + { AFMT_IMA_ADPCM, 8 }, + { AFMT_U8, 8 }, + { AFMT_S16_LE, 16 }, + { AFMT_S16_BE, 16 }, + { AFMT_S8, 8 }, + { AFMT_U16_LE, 16 }, + { AFMT_U16_BE, 16 }, +}; + +#define FORMATSIZE (sizeof(ssi_audio_formattable) / sizeof(struct ssi_audio_format)) + +static void ssi_audio_setsamplesize(int val) +{ + int i; + + if (audio_device == NULL) return; + + for (i = 0; (i < FORMATSIZE); i++) { + if (ssi_audio_formattable[i].format == val) { + audio_device->format = ssi_audio_formattable[i].format; + audio_device->bits = ssi_audio_formattable[i].bits; + break; + } + } + +#ifdef AUDIO_DEBUG + printk(DRIVER_NAME ":ssi_audio_setsamplesize %d %d\n", audio_device->format, audio_device->bits); +#endif +} + +static void ssi_audio_txdrain(void) +{ +#ifdef AUDIO_DEBUG + printk(DRIVER_NAME ":ssi_audio_txdrain()\n"); +#endif + + if (audio_device == NULL) return; + + while (!signal_pending(current)) { + if (audio_txbusy == 0) + break; + current->state = TASK_INTERRUPTIBLE; + schedule_timeout(1); + } +} + +#ifdef CONFIG_SSIAUDIO_USE_EDMA +/* + * Configure and start DMA engine. + */ +void __inline__ ssi_audio_dmarun(void) +{ + set_edma_params(DMA_TCD, +#ifdef USE_MMU + virt_to_phys(&(audio_device->audio_buf[audio_start])), +#else + (u32)&(audio_device->audio_buf[audio_start]), +#endif + (u32)&MCF_SSI_TX0, + MCF_EDMA_TCD_ATTR_SSIZE_32BIT | MCF_EDMA_TCD_ATTR_DSIZE_32BIT, + 8, + 4, + 0, + audio_count/8, + audio_count/8, + 0, + 0, + 0, // major_int + 0 // disable_req + ); + + set_edma_params(DMA_TCD2, +#ifdef USE_MMU + virt_to_phys(&(audio_device->audio_buf[audio_start+4])), +#else + (u32)&(audio_device->audio_buf[audio_start+4]), +#endif + (u32)&MCF_SSI_TX1, + MCF_EDMA_TCD_ATTR_SSIZE_32BIT | MCF_EDMA_TCD_ATTR_DSIZE_32BIT, + 8, + 4, + 0, + audio_count/8, + audio_count/8, + 0, + 0, + 1, // major_int + 0 // disable_req + ); + + audio_device->dmaing = 1; + audio_txbusy = 1; + + start_edma_transfer(DMA_TCD); + start_edma_transfer(DMA_TCD2); +#if 0 + MCF_EDMA_ERQ |= (1<<DMA_TCD) | (1<<DMA_TCD2); + MCF_EDMA_SSRT = DMA_TCD; + MCF_EDMA_SSRT = DMA_TCD2; +#endif + +} + +/* + * Start DMA'ing a new buffer of data if any available. + */ +static void ssi_audio_dmabuf(void) +{ +#ifdef AUDIO_DEBUG + printk(DRIVER_NAME ":ssi_audio_dmabuf(): append=%x start=%x\n", audio_append, audio_appstart); +#endif + + /* If already running then nothing to do... */ + if (audio_device->dmaing) + return; + + /* Set DMA buffer size */ + audio_count = (audio_append >= audio_appstart) ? + (audio_append - audio_appstart) : + (BUFSIZE - audio_appstart); + if (audio_count > DMASIZE) + audio_count = DMASIZE; + + /* Adjust pointers and counters accordingly */ + audio_appstart += audio_count; + if (audio_appstart >= BUFSIZE) + audio_appstart = 0; + + if (audio_count > 0) + ssi_audio_dmarun(); + else { + audio_txbusy = 0; +#ifdef AUDIO_DEBUG + printk(DRIVER_NAME ":DMA buffer is empty!\n"); +#endif + } +} + +void __inline__ stop_dma(void) { + stop_edma_transfer(DMA_TCD); + stop_edma_transfer(DMA_TCD2); +} + +static int ssi_audio_dma_handler_empty(int channel, void *dev_id) +{ + return IRQ_HANDLED; +} + +static int ssi_audio_dma_handler(int channel, void *dev_id) +{ +#ifdef AUDIO_DEBUG + printk(DRIVER_NAME ":ssi_audio_dma_handler(channel=%d)\n", channel); +#endif + + /* Clear DMA interrupt */ + stop_dma(); + + audio_device->dmaing = 0; + + /* Update data pointers and counts */ + audio_start += audio_count; + if (audio_start >= BUFSIZE) + audio_start = 0; + audio_count = 0; + + /* Start new DMA buffer if we can */ + ssi_audio_dmabuf(); + + return IRQ_HANDLED; +} + +static void init_dma(void) +{ + /* SSI DMA Signals mapped to DMA request */ + MCF_CCM_MISCCR &= ~MCF_CCM_MISCCR_TIMDMA; + init_edma(); +} + +#endif /* CONFIG_SSIAUDIO_USE_EDMA */ + + +/* Write CODEC register using SPI + * address - CODEC register address + * data - data to be written into register + */ +static int codec_write(u8 addr, u16 data) +{ + u16 spi_word; + + if (audio_device==NULL || audio_device->spi==NULL) + return -ENODEV; + + spi_word = ((addr & 0x7F)<<9)|(data & 0x1FF); + return spi_write(audio_device->spi, (const u8*)&spi_word, sizeof(spi_word)); +} + +static inline void enable_ssi(void) +{ + if (audio_device==NULL || audio_device->ssi_enabled) return; + audio_device->ssi_enabled = 1; + MCF_SSI_CR |= MCF_SSI_CR_SSI_EN; /* enable SSI module */ + MCF_SSI_CR |= MCF_SSI_CR_TE; /* enable tranmitter */ +} + +static inline void disable_ssi(void) +{ + if (audio_device==NULL || audio_device->ssi_enabled==0) return; + MCF_SSI_CR &= ~MCF_SSI_CR_TE; /* disable transmitter */ + MCF_SSI_CR &= ~MCF_SSI_CR_SSI_EN; /* disable SSI module */ + audio_device->ssi_enabled = 0; +} + +/* Audio CODEC initialization */ +/* TODO: also the SSI frequency/dividers must be adjusted */ +static void adjust_codec_speed(void) { +#ifdef AUDIO_DEBUG + printk(DRIVER_NAME ":adjust_codec_speed: %d\n", audio_device->speed); +#endif + + if (audio_device->speed == 8000) { + codec_write(CODEC_SAMPLE_RATE_REG,CODEC_SAMPLE_8KHZ); + } else if (audio_device->speed == 16000) { + codec_write(CODEC_SAMPLE_RATE_REG,CODEC_SAMPLE_16KHZ); + } else if (audio_device->speed == 22000) { + codec_write(CODEC_SAMPLE_RATE_REG,CODEC_SAMPLE_22KHZ); + } else if (audio_device->speed == 44000 || audio_device->speed == 44100) { + codec_write(CODEC_SAMPLE_RATE_REG,CODEC_SAMPLE_44KHZ); + } else if (audio_device->speed == 48000) { + codec_write(CODEC_SAMPLE_RATE_REG,CODEC_SAMPLE_48KHZ); + } else { + /* default 44KHz */ + codec_write(CODEC_SAMPLE_RATE_REG,CODEC_SAMPLE_44KHZ); + } +} + +static void codec_reset(void) +{ + codec_write(CODEC_RESET_REG, 0); /* reset the audio chip */ + udelay(1500); /* wait for reset */ +} + +static void init_audio_codec(void) +{ +#ifdef AUDIO_DEBUG + printk(DRIVER_NAME ":init_audio_codec()\n"); +#endif + codec_reset(); + + codec_write(CODEC_LEFT_IN_REG, 0x017); + codec_write(CODEC_RIGHT_IN_REG, 0x017); + codec_write(CODEC_POWER_DOWN_REG, 0x000); /* Turn off line input */ + codec_write(CODEC_DIGITAL_IF_FMT_REG, 0x00A); /* I2S slave mode */ + /* codec_write(CODEC_DIGITAL_IF_FMT_REG, 0x042); // I2S master mode */ + codec_write(CODEC_DIGITAL_APATH_REG, 0x007); /* Set A path */ + + /* set sample rate */ + adjust_codec_speed(); + + codec_write(CODEC_LEFT_HP_VOL_REG, 0x075); /* set volume */ + codec_write(CODEC_RIGHT_HP_VOL_REG, 0x075); /* set volume */ + codec_write(CODEC_DIGITAL_IF_ACT_REG, 1); /* Activate digital interface */ + codec_write(CODEC_ANALOG_APATH_REG, 0x0F2); +} + + +static void chip_init(void) +{ +#ifdef CONFIG_SSIAUDIO_USE_EDMA + init_dma(); +#endif + + /* Enable the SSI pins */ + MCF_GPIO_PAR_SSI = ( 0 + | MCF_GPIO_PAR_SSI_MCLK + | MCF_GPIO_PAR_SSI_STXD(3) + | MCF_GPIO_PAR_SSI_SRXD(3) + | MCF_GPIO_PAR_SSI_FS(3) + | MCF_GPIO_PAR_SSI_BCLK(3) ); + +} + +static void init_ssi(void) +{ +#ifdef AUDIO_DEBUG + printk(DRIVER_NAME ":init_ssi()\n"); +#endif + + /* Dividers are for MCF54445 on 266Mhz, the output is 44.1Khz*/ + /* Enable SSI clock in CCM */ + MCF_CCM_CDR = MCF_CCM_CDR_SSIDIV(47); + + /* Issue a SSI reset */ + MCF_SSI_CR &= ~MCF_SSI_CR_SSI_EN; /* disable SSI module */ + + /* SSI module uses internal CPU clock */ + MCF_CCM_MISCCR |= MCF_CCM_MISCCR_SSISRC; + + MCF_CCM_MISCCR |= MCF_CCM_MISCCR_SSIPUE; + MCF_CCM_MISCCR |= MCF_CCM_MISCCR_SSIPUS_UP; + + MCF_SSI_CR = 0 + | MCF_SSI_CR_CIS + | MCF_SSI_CR_TCH /* Enable two channel mode */ + | MCF_SSI_CR_MCE /* Set clock out on SSI_MCLK pin */ + | MCF_SSI_CR_I2S_MASTER /* Set I2S master mode */ + | MCF_SSI_CR_SYN /* Enable synchronous mode */ + | MCF_SSI_CR_NET + ; + + MCF_SSI_TCR = 0 + | MCF_SSI_TCR_TXDIR /* internally generated bit clock */ + | MCF_SSI_TCR_TFDIR /* internally generated frame sync */ + | MCF_SSI_TCR_TSCKP /* Clock data on falling edge of bit clock */ + | MCF_SSI_TCR_TFSI /* Frame sync active low */ + | MCF_SSI_TCR_TEFS /* TX frame sync 1 bit before data */ + | MCF_SSI_TCR_TFEN0 /* TX FIFO 0 enabled */ + | MCF_SSI_TCR_TFEN1 /* TX FIFO 1 enabled */ + | MCF_SSI_TCR_TXBIT0 + ; + + MCF_SSI_CCR = MCF_SSI_CCR_WL(7) /* 16 bit word length */ + | MCF_SSI_CCR_DC(1) /* Frame rate divider */ + | MCF_SSI_CCR_PM(0) + | MCF_SSI_CCR_DIV2 + ; + + MCF_SSI_FCSR = 0 + | MCF_SSI_FCSR_TFWM0(0) + | MCF_SSI_FCSR_TFWM1(0) + ; + + MCF_SSI_IER = 0 // interrupts +#ifndef CONFIG_SSIAUDIO_USE_EDMA + | MCF_SSI_IER_TIE /* transmit interrupts */ + | MCF_SSI_IER_TFE0 /* transmit FIFO 0 empty */ + | MCF_SSI_IER_TFE1 /* transmit FIFO 1 empty */ +#else + | MCF_SSI_IER_TDMAE /* DMA request enabled */ + | MCF_SSI_IER_TFE0 /* transmit FIFO 0 empty */ + | MCF_SSI_IER_TFE1 /* transmit FIFO 1 empty */ +#endif + ; + +#ifndef CONFIG_SSIAUDIO_USE_EDMA + /* enable IRQ: SSI interrupt */ + MCF_INTC1_ICR(M5445x_AUDIO_IRQ_SOURCE) = M5445x_AUDIO_IRQ_LEVEL; + MCF_INTC1_CIMR = M5445x_AUDIO_IRQ_SOURCE; +#endif +} + +#ifndef CONFIG_SSIAUDIO_USE_EDMA +/* interrupt for SSI */ +static int ssi_audio_isr(int irq, void *dev_id) +{ + unsigned long *bp; + + if (audio_txbusy==0) { + return IRQ_HANDLED; + } + + spin_lock(&(audio_device->lock)); + + if (audio_start == audio_append) { + disable_ssi(); + audio_txbusy = 0; + } else { + if (MCF_SSI_ISR & (MCF_SSI_ISR_TFE0|MCF_SSI_ISR_TFE1)) { + bp = (unsigned long *) &audio_device->audio_buf[audio_start]; + if (audio_device->channel) { + MCF_SSI_TX1 = *bp; + audio_device->channel = 0; + } else { + MCF_SSI_TX0 = *bp; + audio_device->channel = 1; + } + audio_start += 4; + if (audio_start >= BUFSIZE) + audio_start = 0; + } + } + + spin_unlock(&(audio_device->lock)); + + return IRQ_HANDLED; +} +#endif + +/* Set initial driver playback defaults. */ +static void init_driver_variables(void) +{ + audio_device->speed = 44100; + audio_device->format = AFMT_S16_LE; + audio_device->bits = 16; + audio_device->stereo = 1; + audio_device->ssi_enabled = 0; + + audio_start = 0; + audio_count = 0; + audio_append = 0; + audio_appstart = 0; + audio_txbusy = 0; + audio_device->dmaing = 0; +} + +/* open audio device */ +static int ssi_audio_open(struct inode *inode, struct file *filp) +{ +#ifdef AUDIO_DEBUG + printk(DRIVER_NAME ":ssi_audio_open()\n"); +#endif + + if (audio_device==NULL) return (-ENODEV); + + if (audio_device->isopen) + return(-EBUSY); + + spin_lock(&(audio_device->lock)); + + audio_device->isopen = 1; + + init_driver_variables(); + init_ssi(); + init_audio_codec(); + + spin_unlock(&(audio_device->lock)); + + udelay(100); + + return 0; +} + +/* close audio device */ +static int ssi_audio_close(struct inode *inode, struct file *filp) +{ +#ifdef AUDIO_DEBUG + printk(DRIVER_NAME ":ssi_audio_close()\n"); +#endif + + if (audio_device==NULL) return (-ENODEV); + + ssi_audio_txdrain(); + + spin_lock(&(audio_device->lock)); + +#ifdef CONFIG_SSIAUDIO_USE_EDMA + stop_dma(); +#endif + disable_ssi(); + codec_reset(); + init_driver_variables(); + audio_device->isopen = 0; + + spin_unlock(&(audio_device->lock)); + return 0; +} + +/* write to audio device */ +static ssize_t ssi_audio_write(struct file *filp, const char *buf, size_t count, loff_t *ppos) +{ + unsigned long *dp, *buflp; + unsigned short *bufwp; + unsigned char *bufbp; + unsigned int slen, bufcnt, i, s, e; + +#ifdef AUDIO_DEBUG + printk(DRIVER_NAME ":ssi_audio_write(buf=%x,count=%d)\n", (int) buf, count); +#endif + + if (audio_device==NULL) return (-ENODEV); + + if (count <= 0) + return 0; + + spin_lock(&(audio_device->lock)); + + buflp = (unsigned long *) buf; + bufwp = (unsigned short *) buf; + bufbp = (unsigned char *) buf; + + bufcnt = count & ~0x3; + + bufcnt <<= 1; + if (audio_device->stereo == 0) + bufcnt <<= 1; + if (audio_device->bits == 8) + bufcnt <<= 1; + +tryagain: + /* + * Get a snapshot of buffer, so we can figure out how + * much data we can fit in... + */ + s = audio_start; + e = audio_append; + dp = (unsigned long *) &(audio_device->audio_buf[e]); + + slen = ((s > e) ? (s - e) : (BUFSIZE - (e - s))) - 4; + if (slen > bufcnt) + slen = bufcnt; + if ((BUFSIZE - e) < slen) + slen = BUFSIZE - e; + + if (slen == 0) { + if (signal_pending(current)) + return(-ERESTARTSYS); + set_current_state(TASK_INTERRUPTIBLE); + schedule_timeout(1); + goto tryagain; + } + + /* For DMA we need to have data as 32 bit + values (since SSI TX register is 32 bit). + So, the incomming 16 bit data must be put to buffer as 32 bit values. + Also, the endianess is converted if needed + */ + if (audio_device->stereo) { + if (audio_device->bits == 16) { + if (audio_device->format==AFMT_S16_LE) { + /*- convert endianess, probably could be done by SSI also */ + for (i = 0; (i < slen); i += 4) { + unsigned short val = le16_to_cpu((*bufwp++)); + *dp++ = val; + } + } else { + for (i = 0; (i < slen); i += 4) { + *dp++ = *bufwp++; + } + } + } else { + for (i = 0; (i < slen); i += 4) { + *dp = (((unsigned long) *bufbp++) << 24); + *dp++ |= (((unsigned long) *bufbp++) << 8); + } + } + } else { + if (audio_device->bits == 16) { + for (i = 0; (i < slen); i += 4) { + *dp++ = (((unsigned long)*bufwp)<<16) | *bufwp; + bufwp++; + } + } else { + for (i = 0; (i < slen); i += 4) { + *dp++ = (((unsigned long) *bufbp) << 24) | + (((unsigned long) *bufbp) << 8); + bufbp++; + } + } + } + + e += slen; + if (e >= BUFSIZE) + e = 0; + audio_append = e; + + /* If not outputing audio, then start now */ + if (audio_txbusy == 0) { + audio_txbusy++; + audio_device->channel = 0; + enable_ssi(); +#ifdef CONFIG_SSIAUDIO_USE_EDMA + ssi_audio_dmabuf(); /* start first DMA transfer */ +#endif + } + + bufcnt -= slen; + + if (bufcnt > 0) + goto tryagain; + + spin_unlock(&(audio_device->lock)); + + return count; +} + +/* ioctl: control the driver */ +static int ssi_audio_ioctl(struct inode *inode, struct file *filp, unsigned int cmd, unsigned long arg) +{ + long val; + int rc = 0; + +#ifdef AUDIO_DEBUG + printk(DRIVER_NAME ":ssi_audio_ioctl(cmd=%x,arg=%x)\n", (int) cmd, (int) arg); +#endif + + if (audio_device==NULL) return (-ENODEV); + + switch (cmd) { + + case SNDCTL_DSP_SPEED: + if (access_ok(VERIFY_READ, (void *) arg, sizeof(val))) { + get_user(val, (unsigned long *) arg); +#ifdef AUDIO_DEBUG + printk(DRIVER_NAME ":ssi_audio_ioctl: SNDCTL_DSP_SPEED: %ld\n", val); +#endif + ssi_audio_txdrain(); + audio_device->speed = val; + init_audio_codec(); + } else { + rc = -EINVAL; + } + break; + + case SNDCTL_DSP_SAMPLESIZE: + if (access_ok(VERIFY_READ, (void *) arg, sizeof(val))) { + get_user(val, (unsigned long *) arg); + ssi_audio_txdrain(); + ssi_audio_setsamplesize(val); + } else { + rc = -EINVAL; + } + break; + + case SNDCTL_DSP_STEREO: + if (access_ok(VERIFY_READ, (void *) arg, sizeof(val))) { + get_user(val, (unsigned long *) arg); + ssi_audio_txdrain(); + audio_device->stereo = val; + } else { + rc = -EINVAL; + } + break; + + case SNDCTL_DSP_GETBLKSIZE: + if (access_ok(VERIFY_WRITE, (void *) arg, sizeof(long))) + put_user(BUFSIZE, (long *) arg); + else + rc = -EINVAL; + break; + + case SNDCTL_DSP_SYNC: + ssi_audio_txdrain(); + break; + + default: + rc = -EINVAL; + break; + } + + return rc; +} + +/****************************************************************************/ + +struct file_operations ssi_audio_fops = { + open: ssi_audio_open, /* open */ + release: ssi_audio_close, /* close */ + write: ssi_audio_write, /* write */ + ioctl: ssi_audio_ioctl, /* ioctl */ +}; + +/* initialize audio driver */ +static int __devinit ssi_audio_probe(struct spi_device *spi) +{ + struct ssi_audio *audio; + int err; + +#ifdef AUDIO_DEBUG + printk(DRIVER_NAME": probe\n"); +#endif + + if (!spi->irq) { + dev_dbg(&spi->dev, "no IRQ?\n"); + return -ENODEV; + } + + /* don't exceed max specified sample rate */ + if (spi->max_speed_hz > MAX_SPEED_HZ) { + dev_dbg(&spi->dev, "f(sample) %d KHz?\n", + (spi->max_speed_hz)/1000); + return -EINVAL; + } + + /* register charcter device */ + if (register_chrdev(SOUND_MAJOR, SOUND_DEVICE_NAME, &ssi_audio_fops) < 0) { + printk(KERN_WARNING DRIVER_NAME ": failed to register major %d\n", SOUND_MAJOR); + dev_dbg(&spi->dev, DRIVER_NAME ": failed to register major %d\n", SOUND_MAJOR); + return -ENODEV; + } + + audio = kzalloc(sizeof(struct ssi_audio), GFP_KERNEL); + if (!audio) { + err = -ENOMEM; + goto err_out; + } + + /* DMA buffer must be from GFP_DMA zone, so it will not be cached */ + audio->audio_buf = kmalloc(BUFSIZE, GFP_DMA); + if (audio->audio_buf == NULL) { + dev_dbg(&spi->dev, DRIVER_NAME ": failed to allocate DMA[%d] buffer\n", BUFSIZE); + err = -ENOMEM; + goto err_free_mem; + } + + audio_device = audio; + + dev_set_drvdata(&spi->dev, audio); + spi->dev.power.power_state = PMSG_ON; + + audio->spi = spi; + +#ifndef CONFIG_SSIAUDIO_USE_EDMA + if (request_irq(spi->irq, ssi_audio_isr, SA_INTERRUPT, spi->dev.bus_id, audio)) { + dev_dbg(&spi->dev, "irq %d busy?\n", spi->irq); + err = -EBUSY; + goto err_free_mem; + } + +#else + /* request 2 eDMA channels since two channel output mode is used */ + if (request_edma_channel(DMA_TCD, + ssi_audio_dma_handler_empty, + NULL, + audio, + &(audio_device->lock), + DRIVER_NAME + )!=0) + { + dev_dbg(&spi->dev, "DMA channel %d busy?\n", DMA_TCD); + err = -EBUSY; + goto err_free_mem; + } + if (request_edma_channel(DMA_TCD2, + ssi_audio_dma_handler, + NULL, + audio, + &(audio_device->lock), + DRIVER_NAME + )!=0) + { + dev_dbg(&spi->dev, "DMA channel %d busy?\n", DMA_TCD2); + err = -EBUSY; + goto err_free_mem; + } + +#endif + chip_init(); + printk(DRIVER_NAME ": Probed successfully\n"); + + return 0; + + err_free_mem: + kfree(audio); + audio_device = NULL; + err_out: + unregister_chrdev(SOUND_MAJOR, SOUND_DEVICE_NAME); + return err; +} + +static int __devexit ssi_audio_remove(struct spi_device *spi) +{ + struct ssi_audio *audio = dev_get_drvdata(&spi->dev); + + ssi_audio_txdrain(); +#ifndef CONFIG_SSIAUDIO_USE_EDMA + free_irq(spi->irq, audio); +#else + free_edma_channel(DMA_TCD, audio); + free_edma_channel(DMA_TCD2, audio); +#endif + kfree(audio->audio_buf); + kfree(audio); + audio_device = NULL; + unregister_chrdev(SOUND_MAJOR, SOUND_DEVICE_NAME); + dev_dbg(&spi->dev, "unregistered audio\n"); + return 0; +} + +static int ssi_audio_suspend(struct spi_device *spi, pm_message_t message) { + return 0; +} + +static int ssi_audio_resume(struct spi_device *spi) { + return 0; +} + +static struct spi_driver ssi_audio_driver = { + .driver = { + .name = DRIVER_NAME, + .bus = &spi_bus_type, + .owner = THIS_MODULE, + }, + .probe = ssi_audio_probe, + .remove = __devexit_p(ssi_audio_remove), + .suspend = ssi_audio_suspend, + .resume = ssi_audio_resume, +}; + +static int __init ssi_audio_init(void) +{ + return spi_register_driver(&ssi_audio_driver); +} +module_init(ssi_audio_init); + +static void __exit ssi_audio_exit(void) +{ + spi_unregister_driver(&ssi_audio_driver); +} +module_exit(ssi_audio_exit); + +MODULE_DESCRIPTION("SSI/I2S Audio Driver"); +MODULE_LICENSE("GPL"); --- a/include/asm-m68k/coldfire_edma.h +++ b/include/asm-m68k/coldfire_edma.h @@ -1,39 +1,102 @@ +/* + * coldfire_edma.h - eDMA driver for Coldfire MCF5445x + * + * Yaroslav Vinogradov yaroslav.vinogradov@freescale.com + * + * Copyright Freescale Semiconductor, Inc. 2007 + * + * This program is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License as published by the + * Free Software Foundation; either version 2 of the License, or (at your + * option) any later version. + */ + #ifndef _LINUX_COLDFIRE_DMA_H #define _LINUX_COLDFIRE_DMA_H #include <linux/interrupt.h> +#include <asm/mcf5445x_edma.h> -#define EDMA_DRIVER_NAME "ColdFire-eDMA" -#define DMA_DEV_MINOR 1 +#define EDMA_DRIVER_NAME "ColdFire-eDMA" +#define DMA_DEV_MINOR 1 #define EDMA_INT_CHANNEL_BASE 8 #define EDMA_INT_CONTROLLER_BASE 64 #define EDMA_CHANNELS 16 - + #define EDMA_IRQ_LEVEL 5 - + typedef irqreturn_t (*edma_irq_handler)(int, void *); typedef void (*edma_error_handler)(int, void *); - + +/* Setup transfer control descriptor (TCD) + * channel - descriptor number + * source - source address + * dest - destination address + * attr - attributes + * soff - source offset + * nbytes - number of bytes to be transfered in minor loop + * slast - last source address adjustment + * citer - major loop count + * biter - beggining minor loop count + * doff - destination offset + * dlast_sga - last destination address adjustment + * major_int - generate interrupt after each major loop + * disable_req - disable DMA request after major loop + */ void set_edma_params(int channel, u32 source, u32 dest, - u32 attr, u32 soff, u32 nbytes, u32 slast, - u32 citer, u32 biter, u32 doff, u32 dlast_sga); - -void start_edma_transfer(int channel, int major_int); - -void stop_edma_transfer(int channel); - -void confirm_edma_interrupt_handled(int channel); - + u32 attr, u32 soff, u32 nbytes, u32 slast, + u32 citer, u32 biter, u32 doff, u32 dlast_sga, + int major_int, int disable_req); + +/* Starts eDMA transfer on specified channel + * channel - eDMA TCD number + */ +static inline void start_edma_transfer(int channel) +{ + MCF_EDMA_SERQ = channel; + MCF_EDMA_SSRT = channel; +} + +/* Stops eDMA transfer + * channel - eDMA TCD number + */ +static inline void stop_edma_transfer(int channel) +{ + MCF_EDMA_CINT = channel; + MCF_EDMA_CERQ = channel; +} + + +/* Confirm that interrupt has been handled + * channel - eDMA TCD number + */ +static inline void confirm_edma_interrupt_handled(int channel) +{ + MCF_EDMA_CINT = channel; +} + +/* Initialize eDMA controller */ void init_edma(void); - -int request_edma_channel(int channel, - edma_irq_handler handler, - edma_error_handler error_handler, - void *dev, - spinlock_t *lock, - const char *device_id); - + +/* Request eDMA channel: + * channel - eDMA TCD number + * handler - channel IRQ callback + * error_handler - error interrupt handler callback for channel + * dev - device + * lock - spinlock to be locked (can be NULL) + * device_id - device driver name for proc file system output + */ +int request_edma_channel(int channel, + edma_irq_handler handler, + edma_error_handler error_handler, + void *dev, + spinlock_t *lock, + const char *device_id); + +/* Free eDMA channel + * channel - eDMA TCD number + * dev - device + */ int free_edma_channel(int channel, void *dev); - #endif --- /dev/null +++ b/include/linux/spi/mcfqspi.h @@ -0,0 +1,80 @@ +/****************************************************************************/ + +/* + * mcfqspi.c - Master QSPI controller for the ColdFire processors + * + * (C) Copyright 2005, Intec Automation, + * Mike Lavender (mike@steroidmicros) + * + + This program is free software; you can redistribute it and/or modify + it under the terms of the GNU General Public License as published by + the Free Software Foundation; either version 2 of the License, or + (at your option) any later version. + + This program is distributed in the hope that it will be useful, + but WITHOUT ANY WARRANTY; without even the implied warranty of + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + GNU General Public License for more details. + + You should have received a copy of the GNU General Public License + along with this program; if not, write to the Free Software + Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. */ +/* ------------------------------------------------------------------------- */ + +#ifndef MCFQSPI_H_ +#define MCFQSPI_H_ + +#define QSPI_CS_INIT 0x01 +#define QSPI_CS_ASSERT 0x02 +#define QSPI_CS_DROP 0x04 + +#define QSPIIOCS_DOUT_HIZ 1 /* QMR[DOHIE] set hi-z dout between transfers */ +#define QSPIIOCS_BITS 2 /* QMR[BITS] set transfer size */ +#define QSPIIOCG_BITS 3 /* QMR[BITS] get transfer size */ +#define QSPIIOCS_CPOL 4 /* QMR[CPOL] set SCK inactive state */ +#define QSPIIOCS_CPHA 5 /* QMR[CPHA] set SCK phase, 1=rising edge */ +#define QSPIIOCS_BAUD 6 /* QMR[BAUD] set SCK baud rate divider */ +#define QSPIIOCS_QCD 7 /* QDLYR[QCD] set start delay */ +#define QSPIIOCS_DTL 8 /* QDLYR[DTL] set after delay */ +#define QSPIIOCS_CONT 9 /* continuous CS asserted during transfer */ +#define QSPIIOCS_READDATA 10 /* set data send during read */ +#define QSPIIOCS_ODD_MOD 11 /* if length of buffer is a odd number, 16-bit transfers */ + /* are finalized with a 8-bit transfer */ +#define QSPIIOCS_DSP_MOD 12 /* transfers are bounded to 15/30 bytes (a multiple of 3 bytes = 1 DSPword) */ +#define QSPIIOCS_POLL_MOD 13 /* driver uses polling instead of interrupts */ + +#define QSPIIOCS_SET_CSIV 14 /* sets CSIV flag (cs inactive level) */ + +#ifdef CONFIG_M520x +#undef MCF_GPIO_PAR_QSPI +#define MCF_GPIO_PAR_QSPI (0xA4034) +#endif + +struct coldfire_spi_master { + u16 bus_num; + u16 num_chipselect; + u8 irq_source; + u32 irq_vector; + u32 irq_mask; + u8 irq_lp; + u8 par_val; + u16 par_val16; + void (*cs_control)(u8 cs, u8 command); +}; + + +struct coldfire_spi_chip { + u8 mode; + u8 bits_per_word; + u8 del_cs_to_clk; + u8 del_after_trans; + u16 void_write_data; +}; + +typedef struct qspi_read_data { + __u32 length; + __u8 *buf; /* data to send during read */ + unsigned int loop : 1; +} qspi_read_data; +#endif /*MCFQSPI_H_*/