提交 a836f585 编写于 作者: C Chris Lesiak 提交者: Linus Torvalds

[PATCH] spi: destroy workqueue after spi_unregister_master

Fix a bug in the cleanup of an spi_bitbang bus.

The workqueue associated with the bus was destroyed before the call to
spi_unregister_master.  That meant that spi devices on that bus would be
unable to do IO in their remove method.  The shutdown flag should have been
able to prevent a segfault, but was never getting set.  By waiting to
destroy the workqueue until after the master is unregistered, devices are
able to do IO in their remove methods.  An added benefit is that neither
the shutdown flag nor a wait for the queue of messages to empty is needed.
Signed-off-by: NChris Lesiak <chris.lesiak@licor.com>
Signed-off-by: NDavid Brownell <dbrownell@users.sourceforge.net>
Signed-off-by: NAndrew Morton <akpm@linux-foundation.org>
Signed-off-by: NLinus Torvalds <torvalds@linux-foundation.org>
上级 65b8291c
...@@ -302,10 +302,6 @@ static void bitbang_work(struct work_struct *work) ...@@ -302,10 +302,6 @@ static void bitbang_work(struct work_struct *work)
setup_transfer = NULL; setup_transfer = NULL;
list_for_each_entry (t, &m->transfers, transfer_list) { list_for_each_entry (t, &m->transfers, transfer_list) {
if (bitbang->shutdown) {
status = -ESHUTDOWN;
break;
}
/* override or restore speed and wordsize */ /* override or restore speed and wordsize */
if (t->speed_hz || t->bits_per_word) { if (t->speed_hz || t->bits_per_word) {
...@@ -410,8 +406,6 @@ int spi_bitbang_transfer(struct spi_device *spi, struct spi_message *m) ...@@ -410,8 +406,6 @@ int spi_bitbang_transfer(struct spi_device *spi, struct spi_message *m)
m->status = -EINPROGRESS; m->status = -EINPROGRESS;
bitbang = spi_master_get_devdata(spi->master); bitbang = spi_master_get_devdata(spi->master);
if (bitbang->shutdown)
return -ESHUTDOWN;
spin_lock_irqsave(&bitbang->lock, flags); spin_lock_irqsave(&bitbang->lock, flags);
if (!spi->max_speed_hz) if (!spi->max_speed_hz)
...@@ -507,28 +501,12 @@ EXPORT_SYMBOL_GPL(spi_bitbang_start); ...@@ -507,28 +501,12 @@ EXPORT_SYMBOL_GPL(spi_bitbang_start);
*/ */
int spi_bitbang_stop(struct spi_bitbang *bitbang) int spi_bitbang_stop(struct spi_bitbang *bitbang)
{ {
unsigned limit = 500; spi_unregister_master(bitbang->master);
spin_lock_irq(&bitbang->lock);
bitbang->shutdown = 0;
while (!list_empty(&bitbang->queue) && limit--) {
spin_unlock_irq(&bitbang->lock);
dev_dbg(bitbang->master->cdev.dev, "wait for queue\n"); WARN_ON(!list_empty(&bitbang->queue));
msleep(10);
spin_lock_irq(&bitbang->lock);
}
spin_unlock_irq(&bitbang->lock);
if (!list_empty(&bitbang->queue)) {
dev_err(bitbang->master->cdev.dev, "queue didn't empty\n");
return -EBUSY;
}
destroy_workqueue(bitbang->workqueue); destroy_workqueue(bitbang->workqueue);
spi_unregister_master(bitbang->master);
return 0; return 0;
} }
EXPORT_SYMBOL_GPL(spi_bitbang_stop); EXPORT_SYMBOL_GPL(spi_bitbang_stop);
......
...@@ -25,7 +25,6 @@ struct spi_bitbang { ...@@ -25,7 +25,6 @@ struct spi_bitbang {
spinlock_t lock; spinlock_t lock;
struct list_head queue; struct list_head queue;
u8 busy; u8 busy;
u8 shutdown;
u8 use_dma; u8 use_dma;
struct spi_master *master; struct spi_master *master;
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册