The Lyrtech SFFSDR (Small form Factor Software Defined Radio) board
is built around a TI DM6446 and a Xilinx Virtex-4 (SX-35) FPGA. This
patch adds board setup code for the SFFSDR and a device driver for
the Virtex-4 FPGA.
Signed-off-by: Hugo Villeneuve <[EMAIL PROTECTED]>
---
arch/arm/mach-davinci/Kconfig | 23 ++
arch/arm/mach-davinci/Makefile | 2 +
arch/arm/mach-davinci/board-sffsdr.c | 223 +++++++++++++
arch/arm/mach-davinci/sffsdr-fpga.c | 501 +++++++++++++++++++++++++++++
include/asm-arm/plat-sffsdr/sffsdr-fpga.h | 30 ++
5 files changed, 779 insertions(+), 0 deletions(-)
create mode 100644 arch/arm/mach-davinci/board-sffsdr.c
create mode 100644 arch/arm/mach-davinci/sffsdr-fpga.c
create mode 100644 include/asm-arm/plat-sffsdr/sffsdr-fpga.h
diff --git a/arch/arm/mach-davinci/Kconfig b/arch/arm/mach-davinci/Kconfig
index 43b6359..0f46228 100644
--- a/arch/arm/mach-davinci/Kconfig
+++ b/arch/arm/mach-davinci/Kconfig
@@ -42,6 +42,29 @@ config MACH_DAVINCI_DM355_EVM
Configure this option to specify the whether the board used
for development is a DM355 EVM
+config MACH_DAVINCI_SFFSDR
+ bool "Lyrtech SFFSDR"
+ default n
+ depends on ARCH_DAVINCI_DM644x
+ help
+ Say Y here to select the Lyrtech Small Form Factor
+ Software Defined Radio (SFFSDR) board.
+
+config SFFSDR_FPGA
+ tristate "SFFSDR SX-35 FPGA support"
+ default n
+ depends on MACH_DAVINCI_SFFSDR
+ select FPGALOAD
+ select FW_LOADER
+ help
+ This driver supports the SX-35 FPGA on the Lyrtech SFFSDR board.
+ Amongst other things, the FPGA is used to generate the clocks
+ for the audio codec and for transferring data to/from the other
+ stacked boards (using the EMIF or VPSS ports).
+
+ To compile this driver as module, choose M here: the
+ module will be called sffsdr-fpga.
+
config DAVINCI_RESET_CLOCKS
bool "Reset unused clocks during boot"
depends on ARCH_DAVINCI
diff --git a/arch/arm/mach-davinci/Makefile b/arch/arm/mach-davinci/Makefile
index 420a33e..8478220 100644
--- a/arch/arm/mach-davinci/Makefile
+++ b/arch/arm/mach-davinci/Makefile
@@ -11,3 +11,5 @@ obj-y := time.o irq.o clock.o
serial.o io.o id.o psc.o \
obj-$(CONFIG_MACH_DAVINCI_EVM) += board-dm644x-evm.o
obj-$(CONFIG_MACH_DAVINCI_DM646X_EVM) += board-dm646x-evm.o
obj-$(CONFIG_MACH_DAVINCI_DM355_EVM) += board-dm355-evm.o
+obj-$(CONFIG_MACH_DAVINCI_SFFSDR) += board-sffsdr.o
+obj-$(CONFIG_SFFSDR_FPGA) += sffsdr-fpga.o
diff --git a/arch/arm/mach-davinci/board-sffsdr.c
b/arch/arm/mach-davinci/board-sffsdr.c
new file mode 100644
index 0000000..0b03ce5
--- /dev/null
+++ b/arch/arm/mach-davinci/board-sffsdr.c
@@ -0,0 +1,223 @@
+/*
+ * Lyrtech SFFSDR board support.
+ *
+ * Copyright (C) 2008 Philip Balister, OpenSDR <[EMAIL PROTECTED]>
+ * Copyright (C) 2008 Lyrtech <www.lyrtech.com>
+ *
+ * Based on DV-EVM platform, original copyright follows:
+ *
+ * Copyright (C) 2007 MontaVista Software, Inc.
+ *
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation; either version 2 of the License, or
+ * (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program; if not, write to the Free Software
+ * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
+ */
+
+#include <linux/kernel.h>
+#include <linux/module.h>
+#include <linux/init.h>
+#include <linux/dma-mapping.h>
+#include <linux/platform_device.h>
+#include <linux/gpio.h>
+
+#include <linux/i2c.h>
+#include <linux/i2c/at24.h>
+#include <linux/etherdevice.h>
+#include <linux/mtd/mtd.h>
+#include <linux/mtd/nand.h>
+#include <linux/mtd/partitions.h>
+#include <linux/mtd/physmap.h>
+#include <linux/io.h>
+
+#include <asm/setup.h>
+#include <asm/mach-types.h>
+
+#include <asm/mach/arch.h>
+#include <asm/mach/map.h>
+#include <asm/mach/flash.h>
+
+#include <mach/dm644x.h>
+#include <mach/common.h>
+#include <mach/board.h>
+#include <mach/emac.h>
+#include <mach/i2c.h>
+#include <mach/serial.h>
+#include <mach/psc.h>
+
+#define DAVINCI_ASYNC_EMIF_DATA_CE0_BASE 0x02000000
+
+struct mtd_partition davinci_sffsdr_nandflash_partition[] = {
+ /* U-Boot Environment: Block 0
+ * UBL: Block 1
+ * U-Boot: Blocks 6-7 (256 kb)
+ * Integrity Kernel: Blocks 8-31 (3 Mb)
+ * Integrity Data: Blocks 100-END
+ */
+ {
+ .name = "Linux Kernel",
+ .offset = 32 * SZ_128K,
+ .size = 16 * SZ_128K, /* 2 Mb */
+ .mask_flags = MTD_WRITEABLE, /* Force read-only */
+ },
+ {
+ .name = "Linux ROOT",
+ .offset = MTDPART_OFS_APPEND,
+ .size = 256 * SZ_128K, /* 32 Mb */
+ .mask_flags = 0, /* R/W */
+ },
+};
+
+static struct flash_platform_data davinci_sffsdr_nandflash_data = {
+ .parts = davinci_sffsdr_nandflash_partition,
+ .nr_parts = ARRAY_SIZE(davinci_sffsdr_nandflash_partition),
+};
+
+static struct resource davinci_sffsdr_nandflash_resource = {
+ .start = DAVINCI_ASYNC_EMIF_DATA_CE0_BASE,
+ .end = DAVINCI_ASYNC_EMIF_DATA_CE0_BASE + SZ_16M - 1,
+ .flags = IORESOURCE_MEM,
+};
+
+static struct platform_device davinci_sffsdr_nandflash_device = {
+ .name = "davinci_nand", /* Name of driver */
+ .id = 0,
+ .dev = {
+ .platform_data = &davinci_sffsdr_nandflash_data,
+ },
+ .num_resources = 1,
+ .resource = &davinci_sffsdr_nandflash_resource,
+};
+
+/* Get Ethernet address from kernel boot params */
+static u8 davinci_sffsdr_mac_addr[6] = {0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF};
+
+static struct at24_platform_data eeprom_info = {
+ .byte_len = (64*1024) / 8,
+ .page_size = 32,
+ .flags = AT24_FLAG_ADDR16,
+};
+
+static struct i2c_board_info __initdata i2c_info[] = {
+ {
+ I2C_BOARD_INFO("24lc64", 0x50),
+ .platform_data = &eeprom_info,
+ },
+ /* Other I2C devices:
+ * MSP430, addr 0x23 (not used)
+ * PCA9543, addr 0x70 (setup done by U-Boot)
+ * ADS7828, addr 0x48 (ADC for voltage monitoring.)
+ */
+};
+
+static struct davinci_i2c_platform_data i2c_pdata = {
+ .bus_freq = 20 /* kHz */,
+ .bus_delay = 100 /* usec */,
+};
+
+static void __init sffsdr_init_i2c(void)
+{
+ davinci_init_i2c(&i2c_pdata);
+ i2c_register_board_info(1, i2c_info, ARRAY_SIZE(i2c_info));
+}
+
+/* FPGA physical EMIF register resources. */
+static struct resource davinci_sffsdr_fpga_resources[] = {
+ {
+ .name = "selectmap",
+ .start = 0x4000000,
+ .end = 0x4000003, /* 4 bytes */
+ .flags = IORESOURCE_MEM,
+ },
+ {
+ .name = "fpgaregs",
+ .start = 0x4008000,
+ .end = 0x4008FFF, /* 1 kbytes */
+ .flags = IORESOURCE_MEM,
+ },
+};
+
+static struct platform_device davinci_sffsdr_fpga_device = {
+ .name = "sffsdr-fpga", /* Name of driver */
+ .id = 0,
+ .dev = {
+ .platform_data = &davinci_sffsdr_fpga_resources,
+ },
+ .num_resources = ARRAY_SIZE(davinci_sffsdr_fpga_resources),
+ .resource = davinci_sffsdr_fpga_resources,
+};
+
+static struct platform_device *davinci_sffsdr_devices[] __initdata = {
+ &davinci_sffsdr_nandflash_device,
+ &davinci_sffsdr_fpga_device,
+};
+
+static struct davinci_uart_config davinci_sffsdr_uart_config __initdata = {
+ .enabled_uarts = (1 << 0),
+};
+
+static struct davinci_board_config_kernel davinci_sffsdr_config[] __initdata =
{
+ { DAVINCI_TAG_UART, &davinci_sffsdr_uart_config },
+};
+
+static void __init davinci_sffsdr_map_io(void)
+{
+ davinci_map_common_io();
+}
+
+static __init void davinci_sffsdr_init(void)
+{
+ davinci_psc_init();
+ platform_add_devices(davinci_sffsdr_devices,
+ ARRAY_SIZE(davinci_sffsdr_devices));
+ sffsdr_init_i2c();
+ davinci_board_config = davinci_sffsdr_config;
+ davinci_board_config_size = ARRAY_SIZE(davinci_sffsdr_config);
+ davinci_serial_init();
+ davinci_init_emac(davinci_sffsdr_mac_addr);
+ setup_usb(0, 0); /* We support only peripheral mode. */
+}
+
+static int davinci_cpmac_eth_setup(char *str)
+{
+ int i;
+
+ if (str == NULL)
+ return 0;
+
+ /* Conversion of a MAC address from a string (AA:BB:CC:DD:EE:FF)
+ * to a 6 bytes array. */
+ for (i = 0; i < 6; i++)
+ davinci_sffsdr_mac_addr[i] = simple_strtol(&str[i*3],
+ (char **)NULL, 16);
+
+ return 1;
+}
+/* Get MAC address from kernel boot parameter eth=AA:BB:CC:DD:EE:FF */
+__setup("eth=", davinci_cpmac_eth_setup);
+
+static __init void davinci_sffsdr_irq_init(void)
+{
+ davinci_init_common_hw();
+ davinci_irq_init();
+}
+
+MACHINE_START(SFFSDR, "Lyrtech SFFSDR")
+ /* Maintainer: Hugo Villeneuve [EMAIL PROTECTED] */
+ .phys_io = IO_PHYS,
+ .io_pg_offst = (__IO_ADDRESS(IO_PHYS) >> 18) & 0xfffc,
+ .boot_params = (DAVINCI_DDR_BASE + 0x100),
+ .map_io = davinci_sffsdr_map_io,
+ .init_irq = davinci_sffsdr_irq_init,
+ .timer = &davinci_timer,
+ .init_machine = davinci_sffsdr_init,
+MACHINE_END
diff --git a/arch/arm/mach-davinci/sffsdr-fpga.c
b/arch/arm/mach-davinci/sffsdr-fpga.c
new file mode 100644
index 0000000..8b7030f
--- /dev/null
+++ b/arch/arm/mach-davinci/sffsdr-fpga.c
@@ -0,0 +1,501 @@
+/*
+ * SFFSDR-board specific FPGA driver
+ *
+ * Copyright (C) 2008 Lyrtech <www.lyrtech.com>
+ *
+ * The FPGA is loaded using the SelectMAP mode through
+ * the EMIF interface and some dedicated control signals:
+ *
+ * FPGA DM6446
+ * --------------------
+ * PROGRAM_B GPIO37
+ * DONE GPIO39
+ * INIT GPIO40
+ * DOUT_BUSY GPIO42 (Not used)
+ * CS_B EMIF_A13 OR CS3n
+ *
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation; either version 2 of the License, or
+ * (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program; if not, write to the Free Software
+ * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
+ */
+
+#include <linux/kernel.h>
+#include <linux/module.h>
+#include <linux/init.h>
+#include <linux/device.h>
+#include <linux/string.h>
+#include <linux/platform_device.h>
+#include <linux/err.h>
+#include <linux/firmware.h>
+#include <linux/io.h>
+#include <linux/cdev.h>
+#include <linux/fpgaload.h>
+#include <linux/fs.h>
+
+#include <asm/uaccess.h>
+#include <asm/gpio.h>
+
+#define MODULE_NAME "sffsdr-fpga"
+
+/* Define this to have verbose debug messages. */
+#define SFFSDR_FPGA_DEBUG 1
+
+#define XC4VSX35_PAYLOAD_SIZE (1707240)
+
+#ifdef SFFSDR_FPGA_DEBUG
+#define DBGMSG(fmt, args...) \
+ printk(KERN_INFO "%s: "fmt"\n" , MODULE_NAME, ## args)
+#define FAILMSG(fmt, args...) \
+ printk(KERN_ERR "%s: "fmt"\n" , MODULE_NAME, ## args)
+#else
+#define DBGMSG(fmt, args...)
+#define FAILMSG(fmt, args...)
+#endif
+
+/* Lyrtech specific register inside FPGA, accessed through EMIF. */
+#define FPGA_REVISION_REG_OFFSET (0x000 / 4)
+#define FPGA_GLOBAL_CTRL_REG_OFFSET (0x040 / 4)
+#define FPGA_LED_CONTROL_REG_OFFSET (0x300 / 4)
+#define FPGA_PLL_CODEC_REG_OFFSET (0x800 / 4)
+
+#define FPGA_FULL_RESET_VAL 3
+#define FPGA_PARTIAL_RESET_VAL 2
+
+#define FPGA_DS2_ON (1<<0)
+#define FPGA_DS3_ON (1<<1)
+#define FPGA_DS4_ON (1<<2)
+#define FPGA_DS5_ON (1<<3)
+#define FPGA_DS6_ON (1<<4)
+
+/* Sampling frequency divider, bits 5:4 */
+#define FPGA_FS_DIV_BY_1 (0<<4)
+#define FPGA_FS_DIV_BY_2 (1<<4)
+#define FPGA_FS_DIV_BY_4 (2<<4)
+#define FPGA_FS_DIV_RSV (3<<4)
+
+/* Sampling rate selection, bit 2 */
+#define FPGA_SR_STANDARD (0<<2) /* Standard sampling rate, default */
+#define FPGA_SR_DOUBLE (1<<2) /* Double sampling rate */
+
+/* Sampling frequency selection, bits 1:0 */
+#define FPGA_FS_48000 (0<<0) /* 48.0 kHz (PLL 12.288 MHz) */
+#define FPGA_FS_44100 (1<<0) /* 44.1 kHz (PLL 11.2896 MHz) */
+#define FPGA_FS_32000 (2<<0) /* 32.0 kHz (PLL 8.192 MHz) */
+#define FPGA_FS_RSV (3<<0) /* Reserved */
+
+static const char sffsdr_fpga_device_name[] = "sffsdr-fpga";
+/* in /lib/firmware */
+static const char sffsdr_fpga_bitstream_name[] = "sffsdr_sx35.bit";
+
+struct sffsdr_fpga_dev {
+ const char *bitstream_name;
+ u8 *bitstream_data;
+ size_t bitstream_length;
+ size_t bitstream_max_size;
+ dev_t cdevno;
+ struct cdev cdev; /* Char device structure */
+};
+
+static enum {
+ SFFSDR_FPGA_INIT_HAVE_MEM,
+ SFFSDR_FPGA_INIT_HAVE_SELECTMAP_MMIO,
+ SFFSDR_FPGA_INIT_HAVE_FPGAREGS_MMIO,
+ SFFSDR_FPGA_INIT_CHAR_DEV_REGISTERED,
+ SFFSDR_FPGA_INIT_FPGA_DEV_REGISTERED,
+ SFFSDR_FPGA_INIT_FPGA_BITSTREAM_LOADED,
+} sffsdr_fpga_init_state;
+
+static u8 *selectmap;
+static u32 *fpgaregs;
+
+/* Reset the inside logic of the FPGA according to the
+ * bitstream mode. This is done when the bitstream has
+ * been programmed and is Lyrtech SFF-SDR specific. */
+static void sffsdr_fpga_reset(int bitstream_mode)
+{
+ u32 value;
+
+ if (bitstream_mode == BITSTREAM_MODE_FULL)
+ value = FPGA_FULL_RESET_VAL;
+ else
+ value = FPGA_PARTIAL_RESET_VAL;
+
+ fpgaregs[FPGA_GLOBAL_CTRL_REG_OFFSET] = value;
+ fpgaregs[FPGA_GLOBAL_CTRL_REG_OFFSET] = 0;
+}
+
+static int sffsdr_fpga_post_load(int bitstream_mode)
+{
+ DBGMSG("sffsdr_fpga_post_load()");
+
+ sffsdr_fpga_reset(bitstream_mode);
+
+ DBGMSG("FPGA Revision: %d",
+ fpgaregs[FPGA_REVISION_REG_OFFSET] & 0x0000FFFF);
+
+ /* Light LED DS2 to indicate success. */
+ fpgaregs[FPGA_LED_CONTROL_REG_OFFSET] = FPGA_DS2_ON;
+
+ /* Set default CODEC clock values. */
+ fpgaregs[FPGA_PLL_CODEC_REG_OFFSET] =
+ FPGA_FS_DIV_BY_1 | FPGA_FS_44100 | FPGA_SR_STANDARD;
+
+ sffsdr_fpga_init_state = SFFSDR_FPGA_INIT_FPGA_BITSTREAM_LOADED;
+
+ return 0;
+}
+
+/* Swap bits inside a byte. */
+static u8 swapbits(u8 c)
+{
+ int i;
+ u8 result = 0;
+
+ for (i = 0; i < 8; ++i) {
+ result = result << 1;
+ result |= (c & 1);
+ c = c >> 1;
+ }
+
+ return result;
+}
+
+/*
+ * Writes a byte of data to the FPGA using the SelectMAP
+ * interface. The FPGA_SELECT_MAP_REG address is within
+ * the FPGA address space (CS3), and when we write a byte
+ * to that address, the CCLK line will be toggled.
+ */
+static int sffsdr_fpga_write_byte(u8 *data, int size)
+{
+ int k;
+
+ for (k = 0; k < size; k++)
+ selectmap[0] = swapbits(data[k]);
+
+ return 0;
+}
+
+static const struct fpgaload_t fpgaload = {
+ .fpga_family = FPGA_FAMILY_XILINX_XC4V,
+ .payload_full_size = XC4VSX35_PAYLOAD_SIZE,
+ .program_b = GPIO(37),
+ .done = GPIO(39),
+ .init_b = GPIO(40),
+ .write_byte = sffsdr_fpga_write_byte,
+};
+
+static int sffsdr_fpga_bitstream_load(const u8 *data, size_t size)
+{
+ int retval;
+ int bitstream_mode;
+
+ DBGMSG("Programming bitstream");
+ DBGMSG(" File size = %d kb", size / 1024);
+
+ sffsdr_fpga_init_state = SFFSDR_FPGA_INIT_FPGA_DEV_REGISTERED;
+
+ retval = fpgaload_bitstream_load(&fpgaload, data, size,
+ &bitstream_mode);
+ if (retval < 0)
+ FAILMSG("Error, firmware not loaded in FPGA");
+ else
+ retval = sffsdr_fpga_post_load(bitstream_mode);
+
+ return retval;
+}
+
+/* Open method.
+ *
+ * return value:
+ * 0 in case of success
+ * -ENOMEM if unable to allocate memory.
+ */
+int sffsdr_fpga_open(struct inode *inode, struct file *filp)
+{
+ struct sffsdr_fpga_dev *dev;
+
+ DBGMSG("sffsdr_fpga_open()");
+
+ dev = container_of(inode->i_cdev, struct sffsdr_fpga_dev, cdev);
+ filp->private_data = dev; /* for other methods */
+
+ dev->bitstream_length = 0;
+ dev->bitstream_data = kmalloc(dev->bitstream_max_size,
+ GFP_KERNEL);
+ if (!dev->bitstream_data) {
+ FAILMSG("Failed to allocate memory for bitstream");
+ return -ENOMEM;
+ }
+
+ return 0; /* success */
+}
+
+/* Write method. Fill buffer with bitstream data. */
+ssize_t sffsdr_fpga_write(struct file *filp, const char __user *buff,
+ size_t count, loff_t *offp)
+{
+ struct sffsdr_fpga_dev *dev =
+ (struct sffsdr_fpga_dev *) filp->private_data;
+
+ if ((dev->bitstream_length + count) >= dev->bitstream_max_size) {
+ FAILMSG("Bitstream buffer size exceeded");
+ return -EFBIG;
+ }
+
+ if (copy_from_user(dev->bitstream_data + dev->bitstream_length,
+ (void __user *) buff, count))
+ return -EFAULT;
+
+ dev->bitstream_length += count;
+
+ return count;
+}
+
+/* Release method. This will initiate the FPGA programming. */
+int sffsdr_fpga_release(struct inode *inode, struct file *filp)
+{
+ int retval;
+ struct sffsdr_fpga_dev *dev =
+ (struct sffsdr_fpga_dev *) filp->private_data;
+
+ if (!dev->bitstream_data)
+ return -EFAULT;
+
+ retval = sffsdr_fpga_bitstream_load(dev->bitstream_data,
+ dev->bitstream_length);
+ kfree(dev->bitstream_data);
+
+ return retval;
+}
+
+static struct file_operations sffsdr_fpga_fops = {
+ .owner = THIS_MODULE,
+ .open = sffsdr_fpga_open,
+ .write = sffsdr_fpga_write,
+ .release = sffsdr_fpga_release
+};
+
+int sffsdr_fpga_set_codec_fs(int fs)
+{
+ u32 fs_mask;
+
+ if (sffsdr_fpga_init_state != SFFSDR_FPGA_INIT_FPGA_BITSTREAM_LOADED) {
+ FAILMSG("FPGA is not programmed");
+ return -ENODEV;
+ }
+
+ switch (fs) {
+ case 32000:
+ fs_mask = FPGA_FS_32000;
+ break;
+ case 44100:
+ fs_mask = FPGA_FS_44100;
+ break;
+ case 48000:
+ fs_mask = FPGA_FS_48000;
+ break;
+ default:
+ FAILMSG("Unsupported sampling frequency");
+ return -EFAULT;
+ break;
+ }
+
+ fpgaregs[FPGA_PLL_CODEC_REG_OFFSET] =
+ FPGA_FS_DIV_BY_1 | fs_mask | FPGA_SR_STANDARD;
+
+ return 0;
+}
+EXPORT_SYMBOL(sffsdr_fpga_set_codec_fs);
+
+static int sffsdr_fpga_setup_cdev(struct sffsdr_fpga_dev *dev)
+{
+ const int sffsdr_fpga_minor = 0;
+ int result;
+
+ /* Dynamic major */
+ result = alloc_chrdev_region(&dev->cdevno, sffsdr_fpga_minor, 1,
+ sffsdr_fpga_device_name);
+ if (result < 0) {
+ FAILMSG("Can't get major");
+ return result;
+ }
+
+ cdev_init(&dev->cdev, &sffsdr_fpga_fops);
+ dev->cdev.owner = THIS_MODULE;
+ dev->cdev.ops = &sffsdr_fpga_fops;
+
+ result = cdev_add(&dev->cdev, dev->cdevno, 1);
+ if (result)
+ FAILMSG("Error %d registering char driver", result);
+
+ return result;
+}
+
+static void sffsdr_fpga_cleanup(struct sffsdr_fpga_dev *dev)
+{
+ if (!dev)
+ return;
+
+ switch (sffsdr_fpga_init_state) {
+ case SFFSDR_FPGA_INIT_FPGA_BITSTREAM_LOADED:
+ /* Nothing to do. */
+ case SFFSDR_FPGA_INIT_FPGA_DEV_REGISTERED:
+ fpgaload_unregister(&fpgaload);
+ case SFFSDR_FPGA_INIT_CHAR_DEV_REGISTERED:
+ unregister_chrdev_region(dev->cdevno, 1);
+ cdev_del(&dev->cdev);
+ case SFFSDR_FPGA_INIT_HAVE_FPGAREGS_MMIO:
+ iounmap(fpgaregs);
+ case SFFSDR_FPGA_INIT_HAVE_SELECTMAP_MMIO:
+ iounmap(selectmap);
+ case SFFSDR_FPGA_INIT_HAVE_MEM:
+ kfree(dev);
+ }
+}
+
+int __init sffsdr_fpga_probe(struct platform_device *pdev)
+{
+ struct sffsdr_fpga_dev *dev;
+ struct resource *selectmap_res;
+ struct resource *fpgaregs_res;
+ const struct firmware *fw_entry;
+ int len;
+ int result;
+
+ DBGMSG("sffsdr_fpga_probe()");
+
+ dev = kzalloc(sizeof(*dev), GFP_KERNEL);
+ if (!dev) {
+ FAILMSG("Failed to allocate device structure");
+ result = -ENOMEM;
+ goto error;
+ }
+ sffsdr_fpga_init_state = SFFSDR_FPGA_INIT_HAVE_MEM;
+ pdev->dev.driver_data = dev; /* Private driver data */
+ dev->bitstream_max_size = XC4VSX35_PAYLOAD_SIZE + 10240;
+ dev->bitstream_name = sffsdr_fpga_bitstream_name;
+
+ /* Assign virtual addresses to SELECTMAP I/O memory regions. */
+ selectmap_res = platform_get_resource_byname(pdev, IORESOURCE_MEM,
+ "selectmap");
+ if (!selectmap_res) {
+ FAILMSG("Error getting selectmap ressource");
+ result = -ENODEV;
+ goto error;
+ }
+ len = selectmap_res->end - selectmap_res->start;
+ selectmap = ioremap(selectmap_res->start, len);
+ if (!selectmap) {
+ FAILMSG("Can't remap selectmap register");
+ result = -ENXIO;
+ goto error;
+ }
+ sffsdr_fpga_init_state = SFFSDR_FPGA_INIT_HAVE_SELECTMAP_MMIO;
+
+ /* Assign virtual addresses to FPGAREGS I/O memory regions. */
+ fpgaregs_res = platform_get_resource_byname(pdev, IORESOURCE_MEM,
+ "fpgaregs");
+ if (!fpgaregs_res) {
+ FAILMSG("Error getting fpgaregs ressource");
+ result = -ENODEV;
+ goto error;
+ }
+ len = fpgaregs_res->end - fpgaregs_res->start;
+ fpgaregs = ioremap(fpgaregs_res->start, len);
+ if (!fpgaregs) {
+ FAILMSG("Can't remap fpgaregs registers");
+ result = -ENXIO;
+ goto error;
+ }
+ sffsdr_fpga_init_state = SFFSDR_FPGA_INIT_HAVE_FPGAREGS_MMIO;
+
+ result = sffsdr_fpga_setup_cdev(dev);
+ if (result < 0) {
+ FAILMSG("Error registering char driver");
+ goto error;
+ }
+ sffsdr_fpga_init_state = SFFSDR_FPGA_INIT_CHAR_DEV_REGISTERED;
+
+ result = fpgaload_register(&fpgaload);
+ if (result < 0) {
+ FAILMSG("Error registering fpgaload driver");
+ goto error;
+ }
+ sffsdr_fpga_init_state = SFFSDR_FPGA_INIT_FPGA_DEV_REGISTERED;
+
+ /* Try to load firmware through hotplug if available. */
+ result = request_firmware(&fw_entry, dev->bitstream_name,
+ &pdev->dev);
+ if (result < 0) {
+ /* Not an error preventing the driver from being loaded. */
+ result = 0;
+ DBGMSG("Firmware not available");
+ } else {
+ result = sffsdr_fpga_bitstream_load(fw_entry->data,
+ fw_entry->size);
+ release_firmware(fw_entry);
+ }
+
+ return result;
+
+error:
+ sffsdr_fpga_cleanup(dev);
+ return result;
+}
+
+static int __devexit sffsdr_fpga_remove(struct platform_device *pdev)
+{
+ struct sffsdr_fpga_dev *dev = platform_get_drvdata(pdev);
+
+ DBGMSG("sffsdr_fpga_remove()");
+ sffsdr_fpga_cleanup(dev);
+
+ return 0;
+}
+
+static struct platform_driver sffsdr_fpga_platform_driver = {
+ .driver = {
+ .name = MODULE_NAME,
+ .owner = THIS_MODULE,
+ },
+ .remove = sffsdr_fpga_remove,
+};
+
+int __init sffsdr_fpga_init(void)
+{
+ int res;
+
+ DBGMSG("sffsdr_fpga_init()");
+
+ res = platform_driver_probe(&sffsdr_fpga_platform_driver,
+ sffsdr_fpga_probe);
+ if (res) {
+ DBGMSG("platform_driver_probe() failed");
+ return res;
+ }
+
+ return 0;
+}
+module_init(sffsdr_fpga_init);
+
+void __exit sffsdr_fpga_exit(void)
+{
+ DBGMSG("sffsdr_fpga_exit()");
+ platform_driver_unregister(&sffsdr_fpga_platform_driver);
+}
+module_exit(sffsdr_fpga_exit);
+
+MODULE_AUTHOR("Hugo Villeneuve <[EMAIL PROTECTED]>");
+MODULE_DESCRIPTION("Lyrtech SFFSDR SX-35 FPGA driver");
+MODULE_LICENSE("GPL");
diff --git a/include/asm-arm/plat-sffsdr/sffsdr-fpga.h
b/include/asm-arm/plat-sffsdr/sffsdr-fpga.h
new file mode 100644
index 0000000..74fc537
--- /dev/null
+++ b/include/asm-arm/plat-sffsdr/sffsdr-fpga.h
@@ -0,0 +1,30 @@
+/*
+ * sffsdr_fpga0.h
+ *
+ * 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 SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESS OR IMPLIED
+ * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
+ * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN
+ * NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
+ * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
+ * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF
+ * USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON
+ * ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
+ * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ *
+ * 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 __SFFSDR_FPGA0_H
+#define __SFFSDR_FPGA0_H
+
+int sffsdr_fpga_set_codec_fs(int fs);
+
+#endif /* __SFFSDR_FPGA0_H */
--
1.5.5
_______________________________________________
Davinci-linux-open-source mailing list
[email protected]
http://linux.davincidsp.com/mailman/listinfo/davinci-linux-open-source