/* arch/arm/mach-lh7a40x/ssp-cpld.c * * Copyright (C) 2004,2005 Marc Singer * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU General Public License * version 2 as published by the Free Software Foundation. * * SSP/SPI driver for the CardEngine CPLD. * */ /* NOTES ----- o *** This driver is cribbed from the 7952x implementation. Some comments may not apply. o This driver contains sufficient logic to control either the serial EEPROMs or the audio codec. It is included in the kernel to support the codec. The EEPROMs are really the responsibility of the boot loader and should probably be left alone. o The code must be augmented to cope with multiple, simultaneous clients. o The audio codec writes to the codec chip whenever playback starts. o The touchscreen driver writes to the ads chip every time it samples. o The audio codec must write 16 bits, but the touch chip writes are 8 bits long. o We need to be able to keep these configurations separate while simultaneously active. */ #include #include //#include #include #include //#include #include #include #include #include #include #include #include //#define TALK #if defined (TALK) #define PRINTK(f...) printk (f) #else #define PRINTK(f...) do {} while (0) #endif #if defined (CONFIG_ARCH_LH7A400) # define CPLD_SPID __REGP16(CPLD06_VIRT) /* SPI data */ # define CPLD_SPIC __REGP16(CPLD08_VIRT) /* SPI control */ # define CPLD_SPIC_CS_CODEC (1<<0) # define CPLD_SPIC_CS_TOUCH (1<<1) # define CPLD_SPIC_WRITE (0<<2) # define CPLD_SPIC_READ (1<<2) # define CPLD_SPIC_DONE (1<<3) /* r/o */ # define CPLD_SPIC_LOAD (1<<4) # define CPLD_SPIC_START (1<<4) # define CPLD_SPIC_LOADED (1<<5) /* r/o */ #endif #define CPLD_SPI __REGP16(CPLD0A_VIRT) /* SPI operation */ #define CPLD_SPI_CS_EEPROM (1<<3) #define CPLD_SPI_SCLK (1<<2) #define CPLD_SPI_TX_SHIFT (1) #define CPLD_SPI_TX (1< %2d", v & 0x1ff, (v >> 9) & 0x7f); #endif PRINTK ("\n"); if (ssp_configuration.device == DEVICE_CODEC) select = CPLD_SPIC_CS_CODEC; if (ssp_configuration.device == DEVICE_TOUCH) select = CPLD_SPIC_CS_TOUCH; if (cwrite) { for (cwrite = (cwrite + 7)/8; cwrite-- > 0; ) { CPLD_SPID = (v >> (8*cwrite)) & 0xff; CPLD_SPIC = select | CPLD_SPIC_LOAD; while (!(CPLD_SPIC & CPLD_SPIC_LOADED)) ; CPLD_SPIC = select; while (!(CPLD_SPIC & CPLD_SPIC_DONE)) ; } v = 0; } if (cread) { mdelay (2); /* *** FIXME: required by ads7843? */ v = 0; for (cread = (cread + 7)/8; cread-- > 0;) { CPLD_SPID = 0; CPLD_SPIC = select | CPLD_SPIC_READ | CPLD_SPIC_START; while (!(CPLD_SPIC & CPLD_SPIC_LOADED)) ; CPLD_SPIC = select | CPLD_SPIC_READ; while (!(CPLD_SPIC & CPLD_SPIC_DONE)) ; v = (v << 8) | CPLD_SPID; } } return v; } #endif PRINTK ("spi(%d) 0x%04x -> 0x%x\r\n", ssp_configuration.device, v & 0x1ff, (v >> 9) & 0x7f); enable_cs (); v <<= CPLD_SPI_TX_SHIFT; /* Correction for position of SPI_TX bit */ while (cwrite--) { CPLD_SPI = (CPLD_SPI & ~CPLD_SPI_TX) | ((v >> cwrite) & CPLD_SPI_TX); udelay (T_DIS); pulse_clock (); } if (cread < 0) { int delay = 10; disable_cs (); udelay (1); enable_cs (); l = -1; do { if (CPLD_SPI & CPLD_SPI_RX) { l = 0; break; } } while (udelay (1), --delay); } else /* We pulse the clock before the data to skip the leading zero. */ while (cread-- > 0) { pulse_clock (); l = (l<<1) | (((CPLD_SPI & CPLD_SPI_RX) >> CPLD_SPI_RX_SHIFT) & 0x1); } disable_cs (); return l; } static int ssp_init (void) { spin_lock_init (&ssp_lock); memset (&ssp_configuration, 0, sizeof (ssp_configuration)); return 0; } /* ssp_chip_select drops the chip select line for the CPLD shift-register controlled devices. It doesn't enable chip */ static void ssp_chip_select (int enable) { #if defined (CONFIG_MACH_LPD7A400) int select; if (ssp_configuration.device == DEVICE_CODEC) select = CPLD_SPIC_CS_CODEC; else if (ssp_configuration.device == DEVICE_TOUCH) select = CPLD_SPIC_CS_TOUCH; else return; if (enable) CPLD_SPIC = select; else CPLD_SPIC = 0; #endif } static void ssp_acquire (void) { spin_lock (&ssp_lock); } static void ssp_release (void) { ssp_chip_select (0); /* just in case */ spin_unlock (&ssp_lock); } static int ssp_configure (int device, int mode, int speed, int frame_size_write, int frame_size_read) { ssp_configuration.device = device; ssp_configuration.mode = mode; ssp_configuration.speed = speed; ssp_configuration.frame_size_write = frame_size_write; ssp_configuration.frame_size_read = frame_size_read; return 0; } static int ssp_read (void) { return execute_spi_command (0, 0, ssp_configuration.frame_size_read); } static int ssp_write (u16 data) { execute_spi_command (data, ssp_configuration.frame_size_write, 0); return 0; } static int ssp_write_read (u16 data) { return execute_spi_command (data, ssp_configuration.frame_size_write, ssp_configuration.frame_size_read); } struct ssp_driver lh7a40x_cpld_ssp_driver = { .init = ssp_init, .acquire = ssp_acquire, .release = ssp_release, .configure = ssp_configure, .chip_select = ssp_chip_select, .read = ssp_read, .write = ssp_write, .write_read = ssp_write_read, }; MODULE_AUTHOR("Marc Singer"); MODULE_DESCRIPTION("LPD7A40X CPLD SPI driver"); MODULE_LICENSE("GPL");