linux-kernel.vger.kernel.org archive mirror
 help / color / mirror / Atom feed
From: Vitaly Wool <vwool@ru.mvista.com>
To: linux-kernel@vger.kernel.org
Cc: david-b@pacbell.net, dpervushin@gmail.com, akpm@osdl.org,
	greg@kroah.com, basicmark@yahoo.com, komal_shah802003@yahoo.com,
	stephen@streetfiresound.com,
	spi-devel-general@lists.sourceforge.net, Joachim_Jaeger@digi.com
Subject: [PATCH 2.6-git 3/4] SPI core refresh: SPI/PNX controller
Date: Mon, 12 Dec 2005 18:26:00 +0300	[thread overview]
Message-ID: <20051212182600.79bb1829.vwool@ru.mvista.com> (raw)
In-Reply-To: <20051212182026.4e393d5a.vwool@ru.mvista.com>

Signed-off-by: Dmitry Pervushin <dpervushinl@gmail.com>
Signed-off-by: Vitaly Wool <vwool@ru.mvista.com>

 Kconfig          |    8
 pnx4008-eeprom.c |  121 +++++++++++++
 spipnx-init.h    |    9 +
 spipnx.c         |  488 +++++++++++++++++++++++++++++++++++++++++++++++++++++++
 spipnx.h         |   58 ++++++
 5 files changed, 684 insertions(+)

Index: linux-2.6.orig/drivers/spi/Kconfig
===================================================================
--- linux-2.6.orig.orig/drivers/spi/Kconfig
+++ linux-2.6.orig/drivers/spi/Kconfig
@@ -42,5 +42,13 @@ config SPI_CHARDEV
 	  Say Y here to use /dev/spiNN device files. They make it possible to have user-space
 	  programs use the SPI bus.
 
+config SPI_PNX
+ 	tristate "PNX SPI bus support"
+ 	depends on ARCH_PNX4008 && SPI
+
+config SPI_PNX4008_EEPROM
+ 	tristate "Dummy EEPROM driver"
+ 	depends on SPI && SPI_PNX && ARCH_PNX4008
+
 endmenu
 
Index: linux-2.6.orig/drivers/spi/pnx4008-eeprom.c
===================================================================
--- /dev/null
+++ linux-2.6.orig/drivers/spi/pnx4008-eeprom.c
@@ -0,0 +1,121 @@
+#include <linux/kernel.h>
+#include <linux/init.h>
+#include <linux/module.h>
+#include <linux/slab.h>
+#include <linux/sched.h>
+#include <linux/device.h>
+#include <linux/spi.h>
+#include <linux/proc_fs.h>
+#include <linux/ctype.h>
+
+#include "spipnx.h"
+
+#define EEPROM_SIZE		256
+#define DRIVER_NAME		"EEPROM"
+#define READ_BUFF_SIZE 160
+
+static int __init spi_eeprom_init(void);
+static void __exit spi_eeprom_cleanup(void);
+
+static int spiee_read_block (struct device *d, void *block)
+{
+	struct spi_device *device = TO_SPI_DEV (d);
+	char cmd[2];
+	struct spi_msg *msg = spimsg_alloc(device,
+					SPI_M_CS|SPI_M_CSREL,
+					NULL,
+					0,
+					NULL);
+	struct spi_msg *msg_cmd = spimsg_chain(msg, SPI_M_CS|SPI_M_WR|SPI_M_DMAUNSAFE, cmd, 2, NULL);
+	spimsg_chain(msg_cmd, SPI_M_RD|SPI_M_CSREL,  block, 256, NULL);
+
+	cmd[ 0 ] = 0x03;
+	cmd[ 1 ] = 0x00;
+
+	spimsg_set_clock(msg, 2000000); /* 2 MHz */
+	spi_transfer(msg, NULL);
+	spimsg_free(msg);
+	return 256;
+}
+static ssize_t blk_show (struct device *d, struct device_attribute *attr, char *text )
+{
+	char *rdbuff = kmalloc (256, SLAB_KERNEL);
+	char line1[80],line2[80];
+	char item1[5], item2[5];
+	int bytes, i, x, blen;
+
+	blen = spiee_read_block (d, rdbuff);
+
+	bytes = 0;
+
+	strcpy(text, "");
+	for (i = 0; i < blen; i += 8) {
+		strcpy(line1, "");
+		strcpy(line2, "" );
+		for (x = i; x < i + 8; x++) {
+			if (x > blen) {
+				sprintf(item1, "   ");
+				sprintf(item2, " " );
+			} else {
+				sprintf(item1, "%02x ", rdbuff[x]);
+				if (isprint(rdbuff[x])) {
+					sprintf(item2, "%c", rdbuff[x]);
+				} else {
+					sprintf(item2, ".");
+				}
+			}
+			strcat(line1, item1);
+			strcat(line2, item2);
+		}
+
+		strcat(text, line1);
+		strcat(text, "|  " );
+		strcat(text, line2);
+		strcat(text, "\n" );
+
+		bytes += (strlen (line1 ) + strlen(line2) + 4);
+	}
+
+	kfree (rdbuff);
+
+	return bytes + 1;
+}
+
+static DEVICE_ATTR(blk, S_IRUGO, blk_show, NULL );
+
+
+static int spiee_probe(struct spi_device *this_dev)
+{
+	device_create_file(&this_dev->dev, &dev_attr_blk);
+	return 0;
+}
+
+static int spiee_remove(struct spi_device *this_dev)
+{
+	device_remove_file(&this_dev->dev, &dev_attr_blk);
+	return 0;
+}
+
+static struct spi_driver eeprom_driver = {
+	.driver = {
+		   .name = DRIVER_NAME,
+	},
+	.probe = spiee_probe,
+	.remove = spiee_remove,
+};
+
+static int __init spi_eeprom_init(void)
+{
+	return spi_driver_add(&eeprom_driver);
+}
+static void __exit spi_eeprom_cleanup(void)
+{
+	spi_driver_del(&eeprom_driver);
+}
+
+module_init(spi_eeprom_init);
+module_exit(spi_eeprom_cleanup);
+
+MODULE_LICENSE("GPL");
+MODULE_AUTHOR("dmitry pervushin <dpervushin@gmail.com>");
+MODULE_DESCRIPTION("SPI EEPROM driver");
Index: linux-2.6.orig/drivers/spi/spipnx-init.h
===================================================================
--- /dev/null
+++ linux-2.6.orig/drivers/spi/spipnx-init.h
@@ -0,0 +1,9 @@
+#include <linux/kernel.h>
+#include <asm/arch/spi.h>
+
+#ifdef CONFIG_ARCH_PNX4008
+struct spipnx_wifi_params spipnx_wifi_params = {
+	.gpio_cs_line = GPIO_03,
+};
+#endif
+
Index: linux-2.6.orig/drivers/spi/spipnx.c
===================================================================
--- /dev/null
+++ linux-2.6.orig/drivers/spi/spipnx.c
@@ -0,0 +1,488 @@
+/*
+ * drivers/spi/spi-pnx.c
+ *
+ * SPI support for PNX 010x/4008 boards.
+ *
+ * Author: dmitry pervushin <dpervushin@ru.mvista.com>
+ * Based on Dennis Kovalev's <dkovalev@ru.mvista.com> bus driver for pnx010x
+ *
+ * 2004 (c) MontaVista Software, Inc. This file is licensed under
+ * the terms of the GNU General Public License version 2. This program
+ * is licensed "as is" without any warranty of any kind, whether express
+ * or implied.
+ */
+
+#include <linux/module.h>
+#include <linux/config.h>
+#include <linux/version.h>
+#include <linux/kernel.h>
+#include <linux/ioport.h>
+#include <linux/pci.h>
+#include <linux/types.h>
+#include <linux/delay.h>
+#include <linux/platform.h>
+#include <linux/spi.h>
+#include <linux/fs.h>
+#include <linux/init.h>
+#include <linux/mm.h>
+#include <linux/miscdevice.h>
+#include <linux/timer.h>
+#include <linux/spinlock.h>
+#include <linux/slab.h>
+#include <linux/completion.h>
+#include <linux/interrupt.h>
+#include <linux/dma-mapping.h>
+#include <asm/io.h>
+#include <asm/hardware.h>
+#include <asm/irq.h>
+#include <asm/uaccess.h>
+#include <asm/dma.h>
+#include <asm/dma-mapping.h>
+#include <asm/hardware/clock.h>
+
+#include "spipnx.h"
+#include "spipnx-init.h"
+
+#define lock_device( dev )	/* down( &dev->sem ); */
+#define unlock_device( dev )	/* up( &dev->sem );   */
+
+#define spipnx_dump_xchg(xxx...) pr_debug( xxx )
+
+
+static int spipnx_spi_init(struct device *bus_device);
+static int spipnx_xfer(struct spi_msg *msg);
+static int spipnx_probe(struct device *bus_device);
+static int spipnx_remove(struct device *bus_device);
+static int spipnx_suspend(struct device *bus_device, pm_message_t);
+static int spipnx_resume(struct device *bus_device);
+static void spipnx_control(struct spi_device *device, int code, u32 ctl);
+static int spipnx_request_hardware(struct spipnx_driver_data *dd,
+				   struct platform_device *dev);
+static void spipnx_free_hardware(struct spipnx_driver_data *dd,
+				 struct platform_device *dev);
+static void spipnx_set_clock(struct device *bus_device, u32 clock);
+static void spipnx_reset (struct device *bus_device, u32 context );
+
+static struct spi_bus_driver spipnx_driver = {
+	.driver = {
+		   .bus = &platform_bus_type,
+		   .name = "spipnx",
+		   .probe = spipnx_probe,
+		   .remove = spipnx_remove,
+		   .suspend = spipnx_suspend,
+		   .resume = spipnx_resume,
+		   },
+	.xfer = spipnx_xfer,
+	.set_clock = spipnx_set_clock,
+ 	.reset = spipnx_reset,
+};
+
+void spipnx_set_mode(struct device *bus_device, int mode)
+{
+	struct spipnx_driver_data *dd = dev_get_drvdata(bus_device);
+
+	lock_device(bus_device);
+	dd->dma_mode = mode;
+	unlock_device(bus_device);
+}
+
+static void spipnx_set_clock(struct device *bus_device, u32 clock)
+{
+	struct spipnx_driver_data *dd = dev_get_drvdata(bus_device);
+
+	dd->saved_clock = clock;
+	dev_dbg (bus_device, "setting clock speed to %d\n", dd->saved_clock );
+}
+
+static void spipnx_reset (struct device *bus_device, u32 context )
+{
+	struct spipnx_driver_data *dd = dev_get_drvdata(bus_device);
+
+	dd->spi_regs->con = context;
+	dev_dbg (bus_device, "CON set to 0x%08X\n", context );
+}
+
+static int spipnx_probe(struct device *device)
+{
+	struct spipnx_driver_data *dd;
+	int err;
+
+	dd = kzalloc(sizeof(struct spipnx_driver_data), GFP_KERNEL);
+	if (!dd) {
+		err = -ENOMEM;
+		goto probe_fail_1;
+	}
+	/*
+	 * these special values are needed because 0 is the valid
+	 * assignment to IRQ# and DMA channel as well as DMA slave#
+	 */
+	dd->dma_channel = -1;
+	dd->slave_nr = -1;
+	dd->irq = NO_IRQ;
+
+	err = spipnx_request_hardware(dd, to_platform_device(device));
+	if (err)
+		goto probe_fail_2;
+
+	spi_bus_driver_init(&spipnx_driver, device);
+
+	dd->state = SPIPNX_STATE_READY;
+
+	dev_set_drvdata(device, dd);
+
+	spipnx_arch_add_devices( device, to_platform_device(device)->id );
+
+	return 0;
+
+probe_fail_2:
+	spipnx_remove(device);
+probe_fail_1:
+	return err;
+}
+
+int spipnx_remove(struct device *device)
+{
+	struct spipnx_driver_data *dd = dev_get_drvdata(device);
+
+	spi_bus_driver_cleanup(TO_SPI_BUS_DRIVER(device->driver), device);
+	spipnx_free_hardware(dd, to_platform_device(device));
+	kfree(dd);
+	return 0;
+}
+
+static void spipnx_free_hardware(struct spipnx_driver_data *dd,
+				 struct platform_device *dev)
+{
+	lock_device(&dev->dev);
+
+	spipnx_clk_stop(dd->clk);
+
+	if (dd->io_flags & IORESOURCE_IRQ) {
+		free_irq(dd->irq, dev->dev.bus_id);
+		dd->io_flags &= ~IORESOURCE_IRQ;
+	}
+	if (dd->io_flags & IORESOURCE_MEM) {
+		/*
+		release_mem_region((unsigned long)dd->iostart,
+				   dd->ioend - dd->iostart + 1);
+		*/
+		dd->io_flags &= ~IORESOURCE_MEM;
+	}
+	if (dd->io_flags & IORESOURCE_DMA) {
+		spipnx_release_dma(&dd->dma_channel);
+		dd->io_flags &= ~IORESOURCE_DMA;
+	}
+
+	spipnx_clk_free(dd->clk);
+
+#ifdef DEBUG
+	if (dd->io_flags)
+		printk(KERN_ERR
+		       "Oops, dd_ioflags for driver data %p is still 0x%x!\n",
+		       dd, dd->io_flags);
+#endif
+	unlock_device(&dev->dev);
+}
+
+void spipnx_dma_handler (int channel, int cause, void* context, struct pt_regs* pt_regs )
+{
+#ifdef CONFIG_ARCH_PNX4008
+	struct spipnx_driver_data *dd = context;
+
+	if (cause & DMA_TC_INT )
+		complete( &dd->threshold );
+#endif
+}
+
+int spipnx_request_hardware(struct spipnx_driver_data *dd,
+			    struct platform_device *dev)
+{
+	int err = 0;
+	struct resource *rsrc;
+
+	lock_device(&dev->dev);
+	rsrc = platform_get_resource(dev, IORESOURCE_MEM, 0);
+	if (rsrc) {
+		dd->iostart = rsrc->start;
+		dd->ioend = rsrc->end;
+		dd->spi_regs = ioremap(dd->iostart, SZ_4K);
+		/*
+		if (!request_mem_region
+		    (dd->iostart, dd->ioend - dd->iostart + 1, dev->dev.bus_id))
+			err = -ENODEV;
+		else
+		*/
+			dd->io_flags |= IORESOURCE_MEM;
+	}
+	dd->clk = spipnx_clk_init(dev);
+	if (IS_ERR(dd->clk)) {
+		err = PTR_ERR(dd->clk);
+		goto out;
+	}
+
+	spipnx_clk_start(dd->clk);
+
+	dd->irq = platform_get_irq(dev, 0);
+	if (dd->irq != NO_IRQ) {
+		dd->spi_regs->ier = 0;
+		dd->spi_regs->stat = 0;
+	}
+
+	rsrc = platform_get_resource(dev, IORESOURCE_DMA, 0);
+	if (rsrc) {
+		dd->slave_nr = rsrc->start;
+		err = spipnx_request_dma(&dd->dma_channel, dev->dev.bus_id, spipnx_dma_handler, dd);
+		if (!err)
+			dd->io_flags |= IORESOURCE_DMA;
+	}
+out:
+	unlock_device(&dev.dev);
+	return err;
+}
+
+int __init spipnx_init(void)
+{
+	printk("PNX/SPI bus driver\n");
+	return spi_bus_driver_register(&spipnx_driver);
+}
+
+void __exit spipnx_cleanup(void)
+{
+	spi_bus_driver_unregister(&spipnx_driver);
+}
+
+static int spipnx_spi_init(struct device *bus_device)
+{
+	struct spipnx_driver_data *dd = dev_get_drvdata(bus_device);
+	int timeout;
+	u32 config;
+
+	config = dd->spi_regs->con;
+
+	dd->spi_regs->global = SPIPNX_GLOBAL_RESET_SPI | SPIPNX_GLOBAL_SPI_ON;
+	udelay(10);
+	dd->spi_regs->global = SPIPNX_GLOBAL_SPI_ON;
+	for (timeout = 10000; timeout >= 0; --timeout)
+		if (dd->spi_regs->global & SPIPNX_GLOBAL_SPI_ON)
+			break;
+	if (timeout <= 0)
+		return -1;
+
+	config |= (7<<9);
+	config &= ~(SPIPNX_CON_RXTX);
+	dd->spi_regs->con = config;
+	spipnx_arch_init(dd->saved_clock / 1000, dd->spi_regs);
+	dd->spi_regs->stat |= 0x100;
+	return 0;
+}
+
+void spipnx_control(struct spi_device *device, int type, u32 ctl)
+{
+	spipnx_arch_control(device, type, ctl);
+}
+
+static inline int
+spipnx_xfer_buggy (struct spi_msg *msg)
+{
+	u8* dat;
+	void *buf;
+	int len;
+	struct spi_msg *pmsg;
+	struct spi_device *spidev;
+	struct device *dev;
+	struct spipnx_driver_data *dd;
+	vhblas_spiregs* regs;
+	int i, rc = 0;
+	u32 stat, fc;
+	struct spipnx_dma_transfer_t params;
+	u32 flags = 0;
+
+	if (!msg) {
+		rc = -EINVAL;
+		goto out;
+	}
+
+	spidev = spimsg_get_spidev(msg);
+	dev = &spidev->dev;
+	dd = dev_get_drvdata(dev->parent);
+	regs = dd->spi_regs;
+
+	for (pmsg = msg; pmsg; pmsg = spimsg_getnext(pmsg)) {
+		flags = spimsg_get_flags(pmsg);
+		len = spimsg_get_buffer(pmsg, &buf);
+		dat = buf;
+
+		spipnx_control(spidev, MESSAGE_START, flags & SPI_M_CS);
+		if (flags & SPI_M_WR) {
+			regs->con |= SPIPNX_CON_RXTX;
+			regs->ier |= SPIPNX_IER_EOT;
+			regs->con &= ~SPIPNX_CON_SHIFT_OFF;
+			regs->frm = len;
+
+			if (dd->dma_mode && len >= FIFO_CHUNK_SIZE) {
+				params.dma_buffer = dma_map_single(dev->parent, dat, len, DMA_TO_DEVICE);
+				params.saved_ll = NULL;
+				params.saved_ll_dma = 0;
+
+				spipnx_setup_dma(dd->iostart, dd->slave_nr, dd->dma_channel,
+					 DMA_MODE_WRITE, dat, len);
+				init_completion (&dd->threshold);
+				spipnx_start_dma(dd->dma_channel);
+				wait_for_completion (&dd->threshold);
+				spipnx_stop_dma(dd->dma_channel, dat);
+				dma_unmap_single(dev->parent, params.dma_buffer, len, DMA_TO_DEVICE);
+			} else {
+				regs->con &= ~SPIPNX_CON_SHIFT_OFF;
+				for (i = 0; i < len ; i ++ ) {
+					while (regs->stat & 0x04)
+			       			continue;
+					regs->dat = *dat;
+					dat ++;
+				}
+			}
+
+			while ((regs->stat & SPIPNX_STAT_EOT) == 0)
+				continue;
+		}
+
+		if (flags & SPI_M_RD) {
+			int stopped;
+			u8 c;
+
+			regs->con &= ~SPIPNX_CON_RXTX;
+			regs->ier |= SPIPNX_IER_EOT;
+			regs->con &= ~SPIPNX_CON_SHIFT_OFF;
+			regs->frm = len;
+
+			regs->dat = 0;  /* kick off read */
+
+			if (dd->dma_mode && len > FIFO_CHUNK_SIZE - 1) {
+				params.dma_buffer = dma_map_single(dev->parent, dat, len - (FIFO_CHUNK_SIZE - 1), DMA_FROM_DEVICE);
+				params.saved_ll = NULL;
+				params.saved_ll_dma = 0;
+
+				spipnx_setup_dma(dd->iostart, dd->slave_nr, dd->dma_channel,
+					 DMA_MODE_READ, dat, len - (FIFO_CHUNK_SIZE - 1));
+				init_completion (&dd->threshold);
+				spipnx_start_dma(dd->dma_channel);
+				wait_for_completion (&dd->threshold);
+				spipnx_stop_dma(dd->dma_channel, dat);
+				dma_unmap_single(dev->parent, params.dma_buffer, len - (FIFO_CHUNK_SIZE - 1), DMA_FROM_DEVICE);
+				dat = dat + len - (FIFO_CHUNK_SIZE - 1);
+				len = FIFO_CHUNK_SIZE - 1;
+			}
+
+                	stopped = 0;
+	                stat = 0;
+        	        for(i = 0; i < len; ) {
+				unsigned long irq_flags;
+				local_irq_save(irq_flags);
+
+				stat = dd->spi_regs->stat;;
+				fc = dd->spi_regs->frm;
+				/* has hardware finished ? */
+				if(fc == 0) {
+					regs->con |= SPIPNX_CON_SHIFT_OFF;
+					stopped = 1;
+				}
+				/* FIFO not empty and hardware not about to finish */
+				if((!(stat & SPI_STAT_SPI_BE)) && ((fc > 4) || (stopped ))) {
+					*dat++= c = dd->spi_regs->dat;
+					i++;
+				}
+				if((stat & SPI_STAT_SPI_BF) && (!stopped)) {
+					*dat++= c = dd->spi_regs->dat;
+					i++;
+				}
+				local_irq_restore(irq_flags);
+			 }
+		}
+		regs->stat |= SPIPNX_STAT_CLRINT;
+		spipnx_control(spidev, MESSAGE_END, flags & SPI_M_CSREL);
+		spimsg_put_buffer(pmsg, buf);
+	}
+out:
+	return rc;
+}
+
+static inline char *spimsg_flags_str(unsigned flags)
+{
+	static char s[4] = "";
+
+	s[0] = 0;
+	if (flags & SPI_M_RD)
+		strcat(s, "R");
+	if (flags & SPI_M_WR)
+		strcat(s, "W");
+	if (flags & ~(SPI_M_WR | SPI_M_RD))
+		strcat(s, "*");
+	return s;
+}
+
+int spipnx_xfer(struct spi_msg *msg)
+{
+	struct spi_device *dev = spimsg_get_spidev(msg);
+	struct spipnx_driver_data *dd = dev_get_drvdata(dev->dev.parent);
+	int status;
+
+	lock_device(dev->parent);
+	if (!SPIPNX_IS_READY(dd)) {
+		status = -EIO;
+		goto unlock_and_out;
+	}
+
+	dev_dbg(&dev->dev, " message (%s)\n",
+		spimsg_flags_str(spimsg_get_flags(msg)));
+
+	if (spipnx_spi_init(dev->dev.parent) < 0) {
+		dev_dbg(&dev->dev, "spi_init failed, skipping the transfer\n");
+		status = -EFAULT;
+		goto unlock_and_out;
+	}
+
+	dd->progress = 1;
+	status = spipnx_xfer_buggy(msg);
+	dd->progress = 0;
+unlock_and_out:
+	unlock_device(dev->parent);
+	return status;
+}
+
+static int spipnx_suspend(struct device *dev, pm_message_t level)
+{
+	int err = 0;
+#ifdef CONFIG_PM
+	struct spipnx_driver_data *dd = dev_get_drvdata(dev);
+
+	lock_device(dev);
+	spipnx_release_dma(&dd->dma_channel);	/* any sanity checks, eg using DMA,
+						   will be done here */
+	spipnx_clk_stop(dd->clk);
+	dev->power.power_state = level;
+	unlock_device(dev);
+#endif
+	return err;
+}
+
+static int spipnx_resume(struct device *dev)
+{
+	int err = 0;
+#ifdef CONFIG_PM
+	struct spipnx_driver_data *dd = dev_get_drvdata(dev);
+
+	lock_device(dev);
+	spipnx_clk_start(dd->clk);
+	if (dd->slave_nr >= 0)
+		spipnx_request_dma(&dd->dma_channel, dev->bus_id, spipnx_dma_handler, dd);
+	dev->power.power_state = PMSG_ON;
+	unlock_device(dev);
+#endif				/* CONFIG_PM */
+	return err;
+}
+
+EXPORT_SYMBOL(spipnx_set_mode);
+MODULE_AUTHOR("dmitry pervushin <dpervushin@ru.mvista.com>");
+MODULE_DESCRIPTION("SPI driver for Philips' PNX boards");
+MODULE_LICENSE("GPL");
+module_init(spipnx_init);
+module_exit(spipnx_cleanup);
Index: linux-2.6.orig/drivers/spi/spipnx.h
===================================================================
--- /dev/null
+++ linux-2.6.orig/drivers/spi/spipnx.h
@@ -0,0 +1,58 @@
+/*
+ * SPI support for pnx010x.
+ *
+ * Author: Dennis Kovalev <dkovalev@ru.mvista.com>
+ *
+ * 2004 (c) MontaVista Software, Inc. This file is licensed under
+ * the terms of the GNU General Public License version 2. This program
+ * is licensed "as is" without any warranty of any kind, whether express
+ * or implied.
+ */
+
+#ifndef _SPI_PNX_BUS_DRIVER
+#define _SPI_PNX_BUS_DRIVER
+
+#include <asm/arch/platform.h>
+#include <asm/arch/dma.h>
+#include <asm/dma.h>
+#include <asm/arch/spi.h>
+
+/* structures */
+struct spipnx_driver_data {
+	unsigned io_flags;
+
+	int irq;
+
+	int dma_mode;
+	int dma_channel;
+	int slave_nr;
+
+	u32 iostart, ioend;
+	vhblas_spiregs *spi_regs;
+
+	struct clk *clk;
+	int clk_id[4];
+
+	int state;
+	u32 saved_clock;
+
+	struct completion op_complete;
+	struct completion threshold;
+
+	int progress;
+
+	void (*control) (struct spi_device * dev, int code, u32 ctl);
+};
+
+#define SPIPNX_STATE_UNINITIALIZED 0
+#define SPIPNX_STATE_READY         1
+#define SPIPNX_STATE_SUSPENDED     2
+
+#define SPIPNX_IS_READY( bus )  ( (bus)->state == SPIPNX_STATE_READY )
+
+#define FIFO_CHUNK_SIZE		56
+
+#define SPI_ENDIAN_SWAP_NO                 0
+#define SPI_ENDIAN_SWAP_YES                1
+
+#endif				// __SPI_PNX_BUS_DRIVER

  parent reply	other threads:[~2005-12-12 15:25 UTC|newest]

Thread overview: 63+ messages / expand[flat|nested]  mbox.gz  Atom feed  top
2005-12-12 15:20 [PATCH 2.6-git 0/4] SPI core refresh Vitaly Wool
2005-12-12 15:22 ` [PATCH 2.6-git 1/4] SPI core refresh: SPI core patch Vitaly Wool
2005-12-12 15:49   ` Russell King
2005-12-12 15:24 ` [PATCH 2.6-git 2/4] SPI core refresh: MTD dataflash driver Vitaly Wool
2005-12-12 15:26 ` Vitaly Wool [this message]
2005-12-12 15:27 ` [PATCH 2.6-git 4/4] SPI core refresh: dumb EEPROM driver Vitaly Wool
2005-12-12 18:01 ` [PATCH 2.6-git 0/4] SPI core refresh Rui Sousa
2005-12-13 12:09   ` [spi-devel-general] " dmitry pervushin
2005-12-13 15:11     ` Rui Sousa
2005-12-13 17:06       ` dmitry pervushin
2005-12-14  6:57       ` Vitaly Wool
2005-12-14 14:28         ` Rui Sousa
2005-12-13 16:35     ` David Brownell
2005-12-13 18:02       ` Rui Sousa
2005-12-13 14:06 ` [PATCH/RFC] SPI: add async message handing library to David Brownell's core Vitaly Wool
2005-12-13 16:53   ` [PATCH/RFC] SPI: add DMAUNSAFE analog " Vitaly Wool
2005-12-13 19:01     ` David Brownell
2005-12-13 19:15       ` Greg KH
2005-12-14 13:50         ` Vitaly Wool
2005-12-14 17:18           ` Greg KH
2005-12-14 17:53             ` Vitaly Wool
2005-12-14 18:50               ` [PATCH/RFC] SPI: add DMAUNSAFE analog David Brownell
2005-12-14 19:29                 ` Vitaly Wool
2005-12-14 19:02               ` [PATCH/RFC] SPI: add DMAUNSAFE analog to David Brownell's core David Brownell
2005-12-14 19:19                 ` Vitaly Wool
2005-12-14 19:33                   ` [spi-devel-general] Re: [PATCH/RFC] SPI: add DMAUNSAFE analog David Brownell
2005-12-14 19:34                 ` [PATCH/RFC] SPI: add DMAUNSAFE analog to David Brownell's core Vitaly Wool
2005-12-15  6:47                 ` Vitaly Wool
2005-12-15 16:44                   ` Greg KH
2005-12-15 22:23                     ` Vitaly Wool
2005-12-15 23:02                       ` Greg KH
2005-12-16  8:37                         ` Vitaly Wool
2005-12-16 17:34                           ` Greg KH
2005-12-16 18:32                             ` [spi-devel-general] Re: [PATCH/RFC] SPI: add DMAUNSAFE analog David Brownell
2005-12-15 20:06                   ` David Brownell
2005-12-15 22:17                     ` Vitaly Wool
2005-12-15 22:33                       ` Greg KH
2005-12-16  3:34                         ` Andy Isaacson
2005-12-16  5:17                           ` Greg KH
2005-12-14 19:16               ` [PATCH/RFC] SPI: add DMAUNSAFE analog to David Brownell's core Greg KH
2005-12-14 19:30                 ` Vitaly Wool
2005-12-15 10:00               ` [spi-devel-general] " dmitry pervushin
2005-12-14 17:22           ` David Brownell
2005-12-14 17:50             ` Vitaly Wool
2005-12-14 19:17               ` [PATCH/RFC] SPI: add DMAUNSAFE analog David Brownell
2005-12-14 20:11                 ` Vitaly Wool
2005-12-13 21:47       ` [PATCH/RFC] SPI: add DMAUNSAFE analog to David Brownell's core Vitaly Wool
2005-12-13 22:15       ` Vitaly Wool
2005-12-14 16:55         ` David Brownell
2005-12-14 17:23           ` Vitaly Wool
2005-12-14 18:48   ` [PATCH/RFC] SPI: add async message handing library " Stephen Street
2005-12-14 19:41     ` Vitaly Wool
2005-12-14 21:19       ` Stephen Street
2005-12-14 19:31   ` [PATCH/RFC] SPI: add async message handing library David Brownell
2005-12-15 12:19   ` [PATCH/RFC] SPI: async message handing library update Vitaly Wool
2005-12-18 18:59     ` David Brownell
2005-12-19 15:40       ` [spi-devel-general] " dmitry pervushin
2005-12-20  7:23         ` David Brownell
2005-12-20 18:02       ` Vitaly Wool
2005-12-22 17:28         ` David Brownell
2005-12-22 22:10           ` Vitaly Wool
2005-12-22 23:55             ` David Brownell
2005-12-21 13:17       ` Vitaly Wool

Reply instructions:

You may reply publicly to this message via plain-text email
using any one of the following methods:

* Save the following mbox file, import it into your mail client,
  and reply-to-all from there: mbox

  Avoid top-posting and favor interleaved quoting:
  https://en.wikipedia.org/wiki/Posting_style#Interleaved_style

* Reply using the --to, --cc, and --in-reply-to
  switches of git-send-email(1):

  git send-email \
    --in-reply-to=20051212182600.79bb1829.vwool@ru.mvista.com \
    --to=vwool@ru.mvista.com \
    --cc=Joachim_Jaeger@digi.com \
    --cc=akpm@osdl.org \
    --cc=basicmark@yahoo.com \
    --cc=david-b@pacbell.net \
    --cc=dpervushin@gmail.com \
    --cc=greg@kroah.com \
    --cc=komal_shah802003@yahoo.com \
    --cc=linux-kernel@vger.kernel.org \
    --cc=spi-devel-general@lists.sourceforge.net \
    --cc=stephen@streetfiresound.com \
    /path/to/YOUR_REPLY

  https://kernel.org/pub/software/scm/git/docs/git-send-email.html

* If your mail client supports setting the In-Reply-To header
  via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line before the message body.
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for NNTP newsgroup(s).