Commit 8aad7924 authored by Nicolas Ferre's avatar Nicolas Ferre Committed by Mark Brown

spi/spi-atmel: add flag to controller data for lock operations

Will allow to drop the lock during DMA operations.

Replacing non-irqsave versions with irqsave versions of the lock
to make it correct in both pdc and dmaengine transfer mode
Signed-off-by: default avatarNicolas Ferre <nicolas.ferre@atmel.com>
[wenyou.yang@atmel.com: submit the patch]
Signed-off-by: default avatarWenyou Yang <wenyou.yang@atmel.com>
Tested-by: default avatarRichard Genoud <richard.genoud@gmail.com>
Signed-off-by: default avatarMark Brown <broonie@opensource.wolfsonmicro.com>
parent dfab30ee
...@@ -195,6 +195,7 @@ struct atmel_spi_caps { ...@@ -195,6 +195,7 @@ struct atmel_spi_caps {
*/ */
struct atmel_spi { struct atmel_spi {
spinlock_t lock; spinlock_t lock;
unsigned long flags;
phys_addr_t phybase; phys_addr_t phybase;
void __iomem *regs; void __iomem *regs;
...@@ -333,6 +334,16 @@ static void cs_deactivate(struct atmel_spi *as, struct spi_device *spi) ...@@ -333,6 +334,16 @@ static void cs_deactivate(struct atmel_spi *as, struct spi_device *spi)
gpio_set_value(asd->npcs_pin, !active); gpio_set_value(asd->npcs_pin, !active);
} }
static void atmel_spi_lock(struct atmel_spi *as)
{
spin_lock_irqsave(&as->lock, as->flags);
}
static void atmel_spi_unlock(struct atmel_spi *as)
{
spin_unlock_irqrestore(&as->lock, as->flags);
}
static inline int atmel_spi_xfer_is_last(struct spi_message *msg, static inline int atmel_spi_xfer_is_last(struct spi_message *msg,
struct spi_transfer *xfer) struct spi_transfer *xfer)
{ {
...@@ -569,9 +580,9 @@ atmel_spi_msg_done(struct spi_master *master, struct atmel_spi *as, ...@@ -569,9 +580,9 @@ atmel_spi_msg_done(struct spi_master *master, struct atmel_spi *as,
"xfer complete: %u bytes transferred\n", "xfer complete: %u bytes transferred\n",
msg->actual_length); msg->actual_length);
spin_unlock(&as->lock); atmel_spi_unlock(as);
msg->complete(msg->context); msg->complete(msg->context);
spin_lock(&as->lock); atmel_spi_lock(as);
as->current_transfer = NULL; as->current_transfer = NULL;
as->next_transfer = NULL; as->next_transfer = NULL;
...@@ -594,7 +605,7 @@ atmel_spi_interrupt(int irq, void *dev_id) ...@@ -594,7 +605,7 @@ atmel_spi_interrupt(int irq, void *dev_id)
u32 status, pending, imr; u32 status, pending, imr;
int ret = IRQ_NONE; int ret = IRQ_NONE;
spin_lock(&as->lock); atmel_spi_lock(as);
xfer = as->current_transfer; xfer = as->current_transfer;
msg = list_entry(as->queue.next, struct spi_message, queue); msg = list_entry(as->queue.next, struct spi_message, queue);
...@@ -697,7 +708,7 @@ atmel_spi_interrupt(int irq, void *dev_id) ...@@ -697,7 +708,7 @@ atmel_spi_interrupt(int irq, void *dev_id)
} }
} }
spin_unlock(&as->lock); atmel_spi_unlock(as);
return ret; return ret;
} }
...@@ -802,13 +813,11 @@ static int atmel_spi_setup(struct spi_device *spi) ...@@ -802,13 +813,11 @@ static int atmel_spi_setup(struct spi_device *spi)
spi->controller_state = asd; spi->controller_state = asd;
gpio_direction_output(npcs_pin, !(spi->mode & SPI_CS_HIGH)); gpio_direction_output(npcs_pin, !(spi->mode & SPI_CS_HIGH));
} else { } else {
unsigned long flags; atmel_spi_lock(as);
spin_lock_irqsave(&as->lock, flags);
if (as->stay == spi) if (as->stay == spi)
as->stay = NULL; as->stay = NULL;
cs_deactivate(as, spi); cs_deactivate(as, spi);
spin_unlock_irqrestore(&as->lock, flags); atmel_spi_unlock(as);
} }
asd->csr = csr; asd->csr = csr;
...@@ -827,7 +836,6 @@ static int atmel_spi_transfer(struct spi_device *spi, struct spi_message *msg) ...@@ -827,7 +836,6 @@ static int atmel_spi_transfer(struct spi_device *spi, struct spi_message *msg)
{ {
struct atmel_spi *as; struct atmel_spi *as;
struct spi_transfer *xfer; struct spi_transfer *xfer;
unsigned long flags;
struct device *controller = spi->master->dev.parent; struct device *controller = spi->master->dev.parent;
u8 bits; u8 bits;
struct atmel_spi_device *asd; struct atmel_spi_device *asd;
...@@ -892,11 +900,11 @@ static int atmel_spi_transfer(struct spi_device *spi, struct spi_message *msg) ...@@ -892,11 +900,11 @@ static int atmel_spi_transfer(struct spi_device *spi, struct spi_message *msg)
msg->status = -EINPROGRESS; msg->status = -EINPROGRESS;
msg->actual_length = 0; msg->actual_length = 0;
spin_lock_irqsave(&as->lock, flags); atmel_spi_lock(as);
list_add_tail(&msg->queue, &as->queue); list_add_tail(&msg->queue, &as->queue);
if (!as->current_transfer) if (!as->current_transfer)
atmel_spi_next_message(spi->master); atmel_spi_next_message(spi->master);
spin_unlock_irqrestore(&as->lock, flags); atmel_spi_unlock(as);
return 0; return 0;
} }
...@@ -906,17 +914,16 @@ static void atmel_spi_cleanup(struct spi_device *spi) ...@@ -906,17 +914,16 @@ static void atmel_spi_cleanup(struct spi_device *spi)
struct atmel_spi *as = spi_master_get_devdata(spi->master); struct atmel_spi *as = spi_master_get_devdata(spi->master);
struct atmel_spi_device *asd = spi->controller_state; struct atmel_spi_device *asd = spi->controller_state;
unsigned gpio = (unsigned) spi->controller_data; unsigned gpio = (unsigned) spi->controller_data;
unsigned long flags;
if (!asd) if (!asd)
return; return;
spin_lock_irqsave(&as->lock, flags); atmel_spi_lock(as);
if (as->stay == spi) { if (as->stay == spi) {
as->stay = NULL; as->stay = NULL;
cs_deactivate(as, spi); cs_deactivate(as, spi);
} }
spin_unlock_irqrestore(&as->lock, flags); atmel_spi_unlock(as);
spi->controller_state = NULL; spi->controller_state = NULL;
gpio_free(gpio); gpio_free(gpio);
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment