Hi,
the patch inlined is changing the SPI core and its users to have transfers in the SPI message structure as linked list not as an array, for the reasons discussed recently in this list.
We decided to post is as a single patch, it's not that big to split it :)
Vitaly
Signed-off-by: Vitaly Wool <[email protected]>
Signed-off-by: Dmitry Pervushin <[email protected]>
drivers/input/touchscreen/ads7846.c | 12 ++++++++----
drivers/mtd/devices/m25p80.c | 15 +++++++--------
drivers/mtd/devices/mtd_dataflash.c | 27 +++++++++++++--------------
drivers/spi/spi.c | 6 +++---
drivers/spi/spi_bitbang.c | 13 ++++---------
include/linux/spi/spi.h | 27 ++++++++++++++++-----------
6 files changed, 51 insertions(+), 49 deletions(-)
diff -uNr --exclude='patches*' --exclude=.pc linux-2.6.orig/drivers/input/touchscreen/ads7846.c linux-2.6/drivers/input/touchscreen/ads7846.c
--- linux-2.6.orig/drivers/input/touchscreen/ads7846.c 2005-12-23 13:46:16.000000000 +0300
+++ linux-2.6/drivers/input/touchscreen/ads7846.c 2005-12-23 13:34:01.000000000 +0300
@@ -155,10 +155,13 @@
struct ser_req *req = kzalloc(sizeof *req, SLAB_KERNEL);
int status;
int sample;
+ int i;
if (!req)
return -ENOMEM;
+ INIT_LIST_HEAD(&req->msg.transfers);
+
/* activate reference, so it has time to settle; */
req->xfer[0].tx_buf = &ref_on;
req->xfer[0].len = 1;
@@ -192,8 +195,8 @@
/* group all the transfers together, so we can't interfere with
* reading touchscreen state; disable penirq while sampling
*/
- req->msg.transfers = req->xfer;
- req->msg.n_transfer = 6;
+ for (i = 0; i < 6; i++)
+ list_add_tail(&req->xfer[i].link, &req->msg.transfers);
disable_irq(spi->irq);
status = spi_sync(spi, &req->msg);
@@ -398,6 +401,7 @@
struct ads7846 *ts;
struct ads7846_platform_data *pdata = spi->dev.platform_data;
struct spi_transfer *x;
+ int i;
if (!spi->irq) {
dev_dbg(&spi->dev, "no IRQ?\n");
@@ -500,8 +504,8 @@
CS_CHANGE(x[-1]);
- ts->msg.transfers = ts->xfer;
- ts->msg.n_transfer = x - ts->xfer;
+ for (i = 0; i < x - ts->xfer; i++)
+ list_add_tail(&ts->xfer[i].link, &ts->msg.transfers);
ts->msg.complete = ads7846_rx;
ts->msg.context = ts;
diff -uNr --exclude='patches*' --exclude=.pc linux-2.6.orig/drivers/mtd/devices/m25p80.c linux-2.6/drivers/mtd/devices/m25p80.c
--- linux-2.6.orig/drivers/mtd/devices/m25p80.c 2005-12-23 13:46:16.000000000 +0300
+++ linux-2.6/drivers/mtd/devices/m25p80.c 2005-12-23 15:38:19.000000000 +0300
@@ -231,7 +231,7 @@
{
struct m25p *flash = mtd_to_m25p(mtd);
struct spi_transfer t[2];
- struct spi_message m;
+ DECLARE_SPI_MESSAGE(m);
DEBUG(MTD_DEBUG_LEVEL2, "%s: %s %s 0x%08x, len %z\n", spi->dev.bus_id,
__FUNCTION__, "from", (u32) from, len);
@@ -268,12 +268,11 @@
t[0].tx_buf = flash->command;
t[0].len = sizeof(flash->command);
+ list_add_tail(&t[0].link, &m.transfers);
t[1].rx_buf = buf;
t[1].len = len;
-
- m.transfers = t;
- m.n_transfer = 2;
+ list_add_tail(&t[1].link, &m.transfers);
spi_sync(flash->spi, &m);
@@ -295,7 +294,7 @@
struct m25p *flash = mtd_to_m25p(mtd);
u32 page_offset, page_size;
struct spi_transfer t[2];
- struct spi_message m;
+ DECLARE_SPI_MESSAGE(m);
DEBUG(MTD_DEBUG_LEVEL2, "%s: %s %s 0x%08x, len %z\n", spi->dev.bus_id,
__FUNCTION__, "to", (u32) to, len);
@@ -326,12 +325,12 @@
flash->command[2] = to >> 8;
flash->command[3] = to;
+ list_add_tail(&t[0].link, &m.transfers);
+ list_add_tail(&t[1].link, &m.transfers);
+
t[0].tx_buf = flash->command;
t[0].len = sizeof(flash->command);
- m.transfers = t;
- m.n_transfer = 2;
-
/* what page do we start with? */
page_offset = to % FLASH_PAGESIZE;
diff -uNr --exclude='patches*' --exclude=.pc linux-2.6.orig/drivers/mtd/devices/mtd_dataflash.c linux-2.6/drivers/mtd/devices/mtd_dataflash.c
--- linux-2.6.orig/drivers/mtd/devices/mtd_dataflash.c 2005-12-23 13:46:16.000000000 +0300
+++ linux-2.6/drivers/mtd/devices/mtd_dataflash.c 2005-12-23 15:31:14.000000000 +0300
@@ -147,8 +147,8 @@
{
struct dataflash *priv = (struct dataflash *)mtd->priv;
struct spi_device *spi = priv->spi;
- struct spi_transfer x[1] = { { .tx_dma = 0, }, };
- struct spi_message msg;
+ struct spi_transfer x = { .tx_dma = 0, };
+ DECLARE_SPI_MESSAGE(msg);
unsigned blocksize = priv->page_size << 3;
u8 *command;
@@ -162,10 +162,9 @@
|| (instr->addr % priv->page_size) != 0)
return -EINVAL;
- x[0].tx_buf = command = priv->command;
- x[0].len = 4;
- msg.transfers = x;
- msg.n_transfer = 1;
+ x.tx_buf = command = priv->command;
+ x.len = 4;
+ list_add_tail(&x.link, &msg.transfers);
down(&priv->lock);
while (instr->len > 0) {
@@ -231,7 +230,7 @@
{
struct dataflash *priv = (struct dataflash *)mtd->priv;
struct spi_transfer x[2] = { { .tx_dma = 0, }, };
- struct spi_message msg;
+ DECLARE_SPI_MESSAGE(msg);
unsigned int addr;
u8 *command;
int status;
@@ -258,10 +257,10 @@
x[0].tx_buf = command;
x[0].len = 8;
+ list_add_tail(&x[0].link, &msg.transfers);
x[1].rx_buf = buf;
x[1].len = len;
- msg.transfers = x;
- msg.n_transfer = 2;
+ list_add_tail(&x[1].link, &msg.transfers);
down(&priv->lock);
@@ -302,7 +301,7 @@
struct dataflash *priv = (struct dataflash *)mtd->priv;
struct spi_device *spi = priv->spi;
struct spi_transfer x[2] = { { .tx_dma = 0, }, };
- struct spi_message msg;
+ DECLARE_SPI_MESSAGE(msg);
unsigned int pageaddr, addr, offset, writelen;
size_t remaining = len;
u_char *writebuf = (u_char *) buf;
@@ -320,9 +319,10 @@
if ((to + len) > mtd->size)
return -EINVAL;
+ list_add_tail(&x[0].link, &msg.transfers);
+
x[0].tx_buf = command = priv->command;
x[0].len = 4;
- msg.transfers = x;
pageaddr = ((unsigned)to / priv->page_size);
offset = ((unsigned)to % priv->page_size);
@@ -364,7 +364,6 @@
DEBUG(MTD_DEBUG_LEVEL3, "TRANSFER: (%x) %x %x %x\n",
command[0], command[1], command[2], command[3]);
- msg.n_transfer = 1;
status = spi_sync(spi, &msg);
if (status < 0)
DEBUG(MTD_DEBUG_LEVEL1, "%s: xfer %u -> %d \n",
@@ -385,8 +384,9 @@
x[1].tx_buf = writebuf;
x[1].len = writelen;
- msg.n_transfer = 2;
+ list_add_tail(&x[1].link, &msg.transfers);
status = spi_sync(spi, &msg);
+ list_del(&x[1].link);
if (status < 0)
DEBUG(MTD_DEBUG_LEVEL1, "%s: pgm %u/%u -> %d \n",
spi->dev.bus_id, addr, writelen, status);
@@ -405,7 +405,6 @@
DEBUG(MTD_DEBUG_LEVEL3, "COMPARE: (%x) %x %x %x\n",
command[0], command[1], command[2], command[3]);
- msg.n_transfer = 1;
status = spi_sync(spi, &msg);
if (status < 0)
DEBUG(MTD_DEBUG_LEVEL1, "%s: compare %u -> %d \n",
diff -uNr --exclude='patches*' --exclude=.pc linux-2.6.orig/drivers/spi/spi_bitbang.c linux-2.6/drivers/spi/spi_bitbang.c
--- linux-2.6.orig/drivers/spi/spi_bitbang.c 2005-12-23 13:46:16.000000000 +0300
+++ linux-2.6/drivers/spi/spi_bitbang.c 2005-12-23 16:13:35.000000000 +0300
@@ -242,7 +242,7 @@
struct spi_message *m;
struct spi_device *spi;
unsigned nsecs;
- struct spi_transfer *t;
+ struct spi_transfer *t = NULL;
unsigned tmp;
unsigned chipselect;
int status;
@@ -259,12 +259,11 @@
nsecs = 100;
spi = m->spi;
- t = m->transfers;
tmp = 0;
chipselect = 0;
status = 0;
- for (;;t++) {
+ list_for_each_entry (t, &m->transfers, link) {
if (bitbang->shutdown) {
status = -ESHUTDOWN;
break;
@@ -295,10 +294,6 @@
if (t->delay_usecs)
udelay(t->delay_usecs);
- tmp++;
- if (tmp >= m->n_transfer)
- break;
-
chipselect = !t->cs_change;
if (chipselect)
continue;
@@ -308,8 +303,8 @@
msleep(1);
}
- tmp = m->n_transfer - 1;
- tmp = m->transfers[tmp].cs_change;
+ if (t)
+ tmp = t->cs_change;
m->status = status;
m->complete(m->context);
diff -uNr --exclude='patches*' --exclude=.pc linux-2.6.orig/drivers/spi/spi.c linux-2.6/drivers/spi/spi.c
--- linux-2.6.orig/drivers/spi/spi.c 2005-12-23 13:46:16.000000000 +0300
+++ linux-2.6/drivers/spi/spi.c 2005-12-23 15:28:58.000000000 +0300
@@ -546,7 +546,7 @@
static DECLARE_MUTEX(lock);
int status;
- struct spi_message message;
+ DECLARE_SPI_MESSAGE(message);
struct spi_transfer x[2];
u8 *local_buf;
@@ -570,13 +570,13 @@
memcpy(local_buf, txbuf, n_tx);
x[0].tx_buf = local_buf;
x[0].len = n_tx;
+ list_add_tail(&x[0].link, &message.transfers);
x[1].rx_buf = local_buf + n_tx;
x[1].len = n_rx;
+ list_add_tail(&x[1].link, &message.transfers);
/* do the i/o */
- message.transfers = x;
- message.n_transfer = ARRAY_SIZE(x);
status = spi_sync(spi, &message);
if (status == 0) {
memcpy(rxbuf, x[1].rx_buf, n_rx);
diff -uNr --exclude='patches*' --exclude=.pc linux-2.6.orig/include/linux/spi/spi.h linux-2.6/include/linux/spi/spi.h
--- linux-2.6.orig/include/linux/spi/spi.h 2005-12-23 13:46:16.000000000 +0300
+++ linux-2.6/include/linux/spi/spi.h 2005-12-23 15:34:21.000000000 +0300
@@ -316,12 +316,13 @@
unsigned cs_change:1;
u16 delay_usecs;
+
+ struct list_head link;
};
/**
* struct spi_message - one multi-segment SPI transaction
* @transfers: the segements of the transaction
- * @n_transfer: how many segments
* @spi: SPI device to which the transaction is queued
* @is_dma_mapped: if true, the caller provided both dma and cpu virtual
* addresses for each transfer buffer
@@ -339,8 +340,7 @@
* insulate against future API updates.
*/
struct spi_message {
- struct spi_transfer *transfers;
- unsigned n_transfer;
+ struct list_head transfers;
struct spi_device *spi;
@@ -371,6 +371,9 @@
void *state;
};
+#define DECLARE_SPI_MESSAGE(mmm) struct spi_message mmm = \
+ { .transfers = LIST_HEAD_INIT((mmm).transfers), }
+
/* It's fine to embed message and transaction structures in other data
* structures so long as you don't free them while they're in use.
*/
@@ -383,8 +386,12 @@
+ ntrans * sizeof(struct spi_transfer),
flags);
if (m) {
- m->transfers = (void *)(m + 1);
- m->n_transfer = ntrans;
+ int i;
+ struct spi_transfer *t = (struct spi_transfer *)(m + 1);
+
+ INIT_LIST_HEAD(&m->transfers);
+ for (i = 0; i < ntrans; i++)
+ list_add_tail(&t[i].link, &m->transfers);
}
return m;
}
@@ -472,10 +479,9 @@
.len = len,
.cs_change = 0,
};
- struct spi_message m;
+ DECLARE_SPI_MESSAGE(m);
- m.transfers = &t;
- m.n_transfer = 1;
+ list_add_tail(&t.link, &m.transfers);
return spi_sync(spi, &m);
}
@@ -497,10 +503,9 @@
.len = len,
.cs_change = 0,
};
- struct spi_message m;
+ DECLARE_SPI_MESSAGE(m);
- m.transfers = &t;
- m.n_transfer = 1;
+ list_add_tail(&t.link, &m.transfers);
return spi_sync(spi, &m);
}
-
To unsubscribe from this list: send the line "unsubscribe linux-kernel" in
the body of a message to [email protected]
More majordomo info at http://vger.kernel.org/majordomo-info.html
Please read the FAQ at http://www.tux.org/lkml/
[Index of Archives]
[Kernel Newbies]
[Netfilter]
[Bugtraq]
[Photo]
[Stuff]
[Gimp]
[Yosemite News]
[MIPS Linux]
[ARM Linux]
[Linux Security]
[Linux RAID]
[Video 4 Linux]
[Linux for the blind]
[Linux Resources]