Commit e2185072 authored by Mark Brown's avatar Mark Brown

spi: sh: Switch to using core message queue

We deprecated open coding of the transfer queue back in 2017 so it's high
time we finished up converting drivers to use the standard message queue
code. The SH driver is fairly straightforward so convert to use
transfer_one_message(), it looks like the driver would be a good fit for
transfer_one() with a little bit of updating but this smaller change seems
safer.

I'm not actually clear how the driver worked robustly previously, it
clears SSA and CR1 when queueing a transfer which looks like it would
interfere with any running transfer. This clearing has been moved to the
start of the message transfer function.

I'm also unclear how exactly the chip select is managed with this driver.
Signed-off-by: default avatarMark Brown <broonie@kernel.org>
Link: https://lore.kernel.org/r/20220610154649.1707851-1-broonie@kernel.orgSigned-off-by: default avatarMark Brown <broonie@kernel.org>
parent 595d68c1
...@@ -73,11 +73,8 @@ struct spi_sh_data { ...@@ -73,11 +73,8 @@ struct spi_sh_data {
void __iomem *addr; void __iomem *addr;
int irq; int irq;
struct spi_master *master; struct spi_master *master;
struct list_head queue;
struct work_struct ws;
unsigned long cr1; unsigned long cr1;
wait_queue_head_t wait; wait_queue_head_t wait;
spinlock_t lock;
int width; int width;
}; };
...@@ -271,47 +268,39 @@ static int spi_sh_receive(struct spi_sh_data *ss, struct spi_message *mesg, ...@@ -271,47 +268,39 @@ static int spi_sh_receive(struct spi_sh_data *ss, struct spi_message *mesg,
return 0; return 0;
} }
static void spi_sh_work(struct work_struct *work) static int spi_sh_transfer_one_message(struct spi_controller *ctlr,
struct spi_message *mesg)
{ {
struct spi_sh_data *ss = container_of(work, struct spi_sh_data, ws); struct spi_sh_data *ss = spi_controller_get_devdata(ctlr);
struct spi_message *mesg;
struct spi_transfer *t; struct spi_transfer *t;
unsigned long flags;
int ret; int ret;
pr_debug("%s: enter\n", __func__); pr_debug("%s: enter\n", __func__);
spin_lock_irqsave(&ss->lock, flags); spi_sh_clear_bit(ss, SPI_SH_SSA, SPI_SH_CR1);
while (!list_empty(&ss->queue)) {
mesg = list_entry(ss->queue.next, struct spi_message, queue);
list_del_init(&mesg->queue);
spin_unlock_irqrestore(&ss->lock, flags);
list_for_each_entry(t, &mesg->transfers, transfer_list) {
pr_debug("tx_buf = %p, rx_buf = %p\n",
t->tx_buf, t->rx_buf);
pr_debug("len = %d, delay.value = %d\n",
t->len, t->delay.value);
if (t->tx_buf) {
ret = spi_sh_send(ss, mesg, t);
if (ret < 0)
goto error;
}
if (t->rx_buf) {
ret = spi_sh_receive(ss, mesg, t);
if (ret < 0)
goto error;
}
mesg->actual_length += t->len;
}
spin_lock_irqsave(&ss->lock, flags);
mesg->status = 0; list_for_each_entry(t, &mesg->transfers, transfer_list) {
if (mesg->complete) pr_debug("tx_buf = %p, rx_buf = %p\n",
mesg->complete(mesg->context); t->tx_buf, t->rx_buf);
pr_debug("len = %d, delay.value = %d\n",
t->len, t->delay.value);
if (t->tx_buf) {
ret = spi_sh_send(ss, mesg, t);
if (ret < 0)
goto error;
}
if (t->rx_buf) {
ret = spi_sh_receive(ss, mesg, t);
if (ret < 0)
goto error;
}
mesg->actual_length += t->len;
} }
mesg->status = 0;
spi_finalize_current_message(ctlr);
clear_fifo(ss); clear_fifo(ss);
spi_sh_set_bit(ss, SPI_SH_SSD, SPI_SH_CR1); spi_sh_set_bit(ss, SPI_SH_SSD, SPI_SH_CR1);
udelay(100); udelay(100);
...@@ -321,12 +310,11 @@ static void spi_sh_work(struct work_struct *work) ...@@ -321,12 +310,11 @@ static void spi_sh_work(struct work_struct *work)
clear_fifo(ss); clear_fifo(ss);
spin_unlock_irqrestore(&ss->lock, flags); return 0;
return;
error: error:
mesg->status = ret; mesg->status = ret;
spi_finalize_current_message(ctlr);
if (mesg->complete) if (mesg->complete)
mesg->complete(mesg->context); mesg->complete(mesg->context);
...@@ -334,6 +322,7 @@ static void spi_sh_work(struct work_struct *work) ...@@ -334,6 +322,7 @@ static void spi_sh_work(struct work_struct *work)
SPI_SH_CR1); SPI_SH_CR1);
clear_fifo(ss); clear_fifo(ss);
return ret;
} }
static int spi_sh_setup(struct spi_device *spi) static int spi_sh_setup(struct spi_device *spi)
...@@ -355,29 +344,6 @@ static int spi_sh_setup(struct spi_device *spi) ...@@ -355,29 +344,6 @@ static int spi_sh_setup(struct spi_device *spi)
return 0; return 0;
} }
static int spi_sh_transfer(struct spi_device *spi, struct spi_message *mesg)
{
struct spi_sh_data *ss = spi_master_get_devdata(spi->master);
unsigned long flags;
pr_debug("%s: enter\n", __func__);
pr_debug("\tmode = %02x\n", spi->mode);
spin_lock_irqsave(&ss->lock, flags);
mesg->actual_length = 0;
mesg->status = -EINPROGRESS;
spi_sh_clear_bit(ss, SPI_SH_SSA, SPI_SH_CR1);
list_add_tail(&mesg->queue, &ss->queue);
schedule_work(&ss->ws);
spin_unlock_irqrestore(&ss->lock, flags);
return 0;
}
static void spi_sh_cleanup(struct spi_device *spi) static void spi_sh_cleanup(struct spi_device *spi)
{ {
struct spi_sh_data *ss = spi_master_get_devdata(spi->master); struct spi_sh_data *ss = spi_master_get_devdata(spi->master);
...@@ -416,7 +382,6 @@ static int spi_sh_remove(struct platform_device *pdev) ...@@ -416,7 +382,6 @@ static int spi_sh_remove(struct platform_device *pdev)
struct spi_sh_data *ss = platform_get_drvdata(pdev); struct spi_sh_data *ss = platform_get_drvdata(pdev);
spi_unregister_master(ss->master); spi_unregister_master(ss->master);
flush_work(&ss->ws);
free_irq(ss->irq, ss); free_irq(ss->irq, ss);
return 0; return 0;
...@@ -467,9 +432,6 @@ static int spi_sh_probe(struct platform_device *pdev) ...@@ -467,9 +432,6 @@ static int spi_sh_probe(struct platform_device *pdev)
dev_err(&pdev->dev, "ioremap error.\n"); dev_err(&pdev->dev, "ioremap error.\n");
return -ENOMEM; return -ENOMEM;
} }
INIT_LIST_HEAD(&ss->queue);
spin_lock_init(&ss->lock);
INIT_WORK(&ss->ws, spi_sh_work);
init_waitqueue_head(&ss->wait); init_waitqueue_head(&ss->wait);
ret = request_irq(irq, spi_sh_irq, 0, "spi_sh", ss); ret = request_irq(irq, spi_sh_irq, 0, "spi_sh", ss);
...@@ -481,7 +443,7 @@ static int spi_sh_probe(struct platform_device *pdev) ...@@ -481,7 +443,7 @@ static int spi_sh_probe(struct platform_device *pdev)
master->num_chipselect = 2; master->num_chipselect = 2;
master->bus_num = pdev->id; master->bus_num = pdev->id;
master->setup = spi_sh_setup; master->setup = spi_sh_setup;
master->transfer = spi_sh_transfer; master->transfer_one_message = spi_sh_transfer_one_message;
master->cleanup = spi_sh_cleanup; master->cleanup = spi_sh_cleanup;
ret = spi_register_master(master); ret = spi_register_master(master);
......
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