Commit c30c9b15 authored by David Fries's avatar David Fries Committed by Linus Torvalds

W1: fix deadlocks and remove w1_control_thread

w1_control_thread was removed which would wake up every second and process
newly registered family codes and complete some final cleanup for a
removed master.  Those routines were moved to the threads that were
previously requesting those operations.  A new function
w1_reconnect_slaves takes care of reconnecting existing slave devices when
a new family code is registered or removed.  The removal case was missing
and would cause a deadlock waiting for the family code reference count to
decrease, which will now happen.  A problem with registering a family code
was fixed.  A slave device would be unattached if it wasn't yet claimed,
then attached at the end of the list, two unclaimed slaves would cause an
infinite loop.

The struct w1_bus_master.search now takes a pointer to the struct
w1_master device to avoid searching for it, which would have caused a
lock ordering deadlock with the removal of w1_control_thread.
Signed-off-by: default avatarDavid Fries <david@fries.net>
Signed-off-by: default avatarEvgeniy Polyakov <johnpol@2ka.mipt.ru>
Signed-off-by: default avatarAndrew Morton <akpm@linux-foundation.org>
Signed-off-by: default avatarLinus Torvalds <torvalds@linux-foundation.org>
parent dd78c943
...@@ -274,8 +274,8 @@ static u8 ds1wm_reset_bus(void *data) ...@@ -274,8 +274,8 @@ static u8 ds1wm_reset_bus(void *data)
return 0; return 0;
} }
static void ds1wm_search(void *data, u8 search_type, static void ds1wm_search(void *data, struct w1_master *master_dev,
w1_slave_found_callback slave_found) u8 search_type, w1_slave_found_callback slave_found)
{ {
struct ds1wm_data *ds1wm_data = data; struct ds1wm_data *ds1wm_data = data;
int i; int i;
...@@ -313,7 +313,7 @@ static void ds1wm_search(void *data, u8 search_type, ...@@ -313,7 +313,7 @@ static void ds1wm_search(void *data, u8 search_type,
ds1wm_write_register(ds1wm_data, DS1WM_CMD, ~DS1WM_CMD_SRA); ds1wm_write_register(ds1wm_data, DS1WM_CMD, ~DS1WM_CMD_SRA);
ds1wm_reset(ds1wm_data); ds1wm_reset(ds1wm_data);
slave_found(ds1wm_data, rom_id); slave_found(master_dev, rom_id);
} }
/* --------------------------------------------------------------------- */ /* --------------------------------------------------------------------- */
......
...@@ -46,20 +46,16 @@ MODULE_AUTHOR("Evgeniy Polyakov <johnpol@2ka.mipt.ru>"); ...@@ -46,20 +46,16 @@ MODULE_AUTHOR("Evgeniy Polyakov <johnpol@2ka.mipt.ru>");
MODULE_DESCRIPTION("Driver for 1-wire Dallas network protocol."); MODULE_DESCRIPTION("Driver for 1-wire Dallas network protocol.");
static int w1_timeout = 10; static int w1_timeout = 10;
static int w1_control_timeout = 1;
int w1_max_slave_count = 10; int w1_max_slave_count = 10;
int w1_max_slave_ttl = 10; int w1_max_slave_ttl = 10;
module_param_named(timeout, w1_timeout, int, 0); module_param_named(timeout, w1_timeout, int, 0);
module_param_named(control_timeout, w1_control_timeout, int, 0);
module_param_named(max_slave_count, w1_max_slave_count, int, 0); module_param_named(max_slave_count, w1_max_slave_count, int, 0);
module_param_named(slave_ttl, w1_max_slave_ttl, int, 0); module_param_named(slave_ttl, w1_max_slave_ttl, int, 0);
DEFINE_MUTEX(w1_mlock); DEFINE_MUTEX(w1_mlock);
LIST_HEAD(w1_masters); LIST_HEAD(w1_masters);
static struct task_struct *w1_control_thread;
static int w1_master_match(struct device *dev, struct device_driver *drv) static int w1_master_match(struct device *dev, struct device_driver *drv)
{ {
return 1; return 1;
...@@ -390,7 +386,7 @@ int w1_create_master_attributes(struct w1_master *master) ...@@ -390,7 +386,7 @@ int w1_create_master_attributes(struct w1_master *master)
return sysfs_create_group(&master->dev.kobj, &w1_master_defattr_group); return sysfs_create_group(&master->dev.kobj, &w1_master_defattr_group);
} }
static void w1_destroy_master_attributes(struct w1_master *master) void w1_destroy_master_attributes(struct w1_master *master)
{ {
sysfs_remove_group(&master->dev.kobj, &w1_master_defattr_group); sysfs_remove_group(&master->dev.kobj, &w1_master_defattr_group);
} }
...@@ -567,7 +563,7 @@ static int w1_attach_slave_device(struct w1_master *dev, struct w1_reg_num *rn) ...@@ -567,7 +563,7 @@ static int w1_attach_slave_device(struct w1_master *dev, struct w1_reg_num *rn)
return 0; return 0;
} }
static void w1_slave_detach(struct w1_slave *sl) void w1_slave_detach(struct w1_slave *sl)
{ {
struct w1_netlink_msg msg; struct w1_netlink_msg msg;
...@@ -591,24 +587,6 @@ static void w1_slave_detach(struct w1_slave *sl) ...@@ -591,24 +587,6 @@ static void w1_slave_detach(struct w1_slave *sl)
kfree(sl); kfree(sl);
} }
static struct w1_master *w1_search_master(void *data)
{
struct w1_master *dev;
int found = 0;
mutex_lock(&w1_mlock);
list_for_each_entry(dev, &w1_masters, w1_master_entry) {
if (dev->bus_master->data == data) {
found = 1;
atomic_inc(&dev->refcnt);
break;
}
}
mutex_unlock(&w1_mlock);
return (found)?dev:NULL;
}
struct w1_master *w1_search_master_id(u32 id) struct w1_master *w1_search_master_id(u32 id)
{ {
struct w1_master *dev; struct w1_master *dev;
...@@ -656,34 +634,49 @@ struct w1_slave *w1_search_slave(struct w1_reg_num *id) ...@@ -656,34 +634,49 @@ struct w1_slave *w1_search_slave(struct w1_reg_num *id)
return (found)?sl:NULL; return (found)?sl:NULL;
} }
void w1_reconnect_slaves(struct w1_family *f) void w1_reconnect_slaves(struct w1_family *f, int attach)
{ {
struct w1_slave *sl, *sln;
struct w1_master *dev; struct w1_master *dev;
mutex_lock(&w1_mlock); mutex_lock(&w1_mlock);
list_for_each_entry(dev, &w1_masters, w1_master_entry) { list_for_each_entry(dev, &w1_masters, w1_master_entry) {
dev_dbg(&dev->dev, "Reconnecting slaves in %s into new family %02x.\n", dev_dbg(&dev->dev, "Reconnecting slaves in device %s "
dev->name, f->fid); "for family %02x.\n", dev->name, f->fid);
set_bit(W1_MASTER_NEED_RECONNECT, &dev->flags); mutex_lock(&dev->mutex);
list_for_each_entry_safe(sl, sln, &dev->slist, w1_slave_entry) {
/* If it is a new family, slaves with the default
* family driver and are that family will be
* connected. If the family is going away, devices
* matching that family are reconneced.
*/
if ((attach && sl->family->fid == W1_FAMILY_DEFAULT
&& sl->reg_num.family == f->fid) ||
(!attach && sl->family->fid == f->fid)) {
struct w1_reg_num rn;
memcpy(&rn, &sl->reg_num, sizeof(rn));
w1_slave_detach(sl);
w1_attach_slave_device(dev, &rn);
}
}
dev_dbg(&dev->dev, "Reconnecting slaves in device %s "
"has been finished.\n", dev->name);
mutex_unlock(&dev->mutex);
} }
mutex_unlock(&w1_mlock); mutex_unlock(&w1_mlock);
} }
static void w1_slave_found(void *data, u64 rn) static void w1_slave_found(struct w1_master *dev, u64 rn)
{ {
int slave_count; int slave_count;
struct w1_slave *sl; struct w1_slave *sl;
struct list_head *ent; struct list_head *ent;
struct w1_reg_num *tmp; struct w1_reg_num *tmp;
struct w1_master *dev;
u64 rn_le = cpu_to_le64(rn); u64 rn_le = cpu_to_le64(rn);
dev = w1_search_master(data); atomic_inc(&dev->refcnt);
if (!dev) {
printk(KERN_ERR "Failed to find w1 master device for data %p, "
"it is impossible.\n", data);
return;
}
tmp = (struct w1_reg_num *) &rn; tmp = (struct w1_reg_num *) &rn;
...@@ -785,74 +778,9 @@ void w1_search(struct w1_master *dev, u8 search_type, w1_slave_found_callback cb ...@@ -785,74 +778,9 @@ void w1_search(struct w1_master *dev, u8 search_type, w1_slave_found_callback cb
if ( (desc_bit == last_zero) || (last_zero < 0)) if ( (desc_bit == last_zero) || (last_zero < 0))
last_device = 1; last_device = 1;
desc_bit = last_zero; desc_bit = last_zero;
cb(dev->bus_master->data, rn); cb(dev, rn);
}
}
}
static int w1_control(void *data)
{
struct w1_slave *sl, *sln;
struct w1_master *dev, *n;
int have_to_wait = 0;
set_freezable();
while (!kthread_should_stop() || have_to_wait) {
have_to_wait = 0;
try_to_freeze();
msleep_interruptible(w1_control_timeout * 1000);
list_for_each_entry_safe(dev, n, &w1_masters, w1_master_entry) {
if (!kthread_should_stop() && !dev->flags)
continue;
/*
* Little race: we can create thread but not set the flag.
* Get a chance for external process to set flag up.
*/
if (!dev->initialized) {
have_to_wait = 1;
continue;
} }
if (kthread_should_stop() || test_bit(W1_MASTER_NEED_EXIT, &dev->flags)) {
set_bit(W1_MASTER_NEED_EXIT, &dev->flags);
mutex_lock(&w1_mlock);
list_del(&dev->w1_master_entry);
mutex_unlock(&w1_mlock);
mutex_lock(&dev->mutex);
list_for_each_entry_safe(sl, sln, &dev->slist, w1_slave_entry) {
w1_slave_detach(sl);
}
w1_destroy_master_attributes(dev);
mutex_unlock(&dev->mutex);
atomic_dec(&dev->refcnt);
continue;
} }
if (test_bit(W1_MASTER_NEED_RECONNECT, &dev->flags)) {
dev_dbg(&dev->dev, "Reconnecting slaves in device %s.\n", dev->name);
mutex_lock(&dev->mutex);
list_for_each_entry_safe(sl, sln, &dev->slist, w1_slave_entry) {
if (sl->family->fid == W1_FAMILY_DEFAULT) {
struct w1_reg_num rn;
memcpy(&rn, &sl->reg_num, sizeof(rn));
w1_slave_detach(sl);
w1_attach_slave_device(dev, &rn);
}
}
dev_dbg(&dev->dev, "Reconnecting slaves in device %s has been finished.\n", dev->name);
clear_bit(W1_MASTER_NEED_RECONNECT, &dev->flags);
mutex_unlock(&dev->mutex);
}
}
}
return 0;
} }
void w1_search_process(struct w1_master *dev, u8 search_type) void w1_search_process(struct w1_master *dev, u8 search_type)
...@@ -932,18 +860,13 @@ static int w1_init(void) ...@@ -932,18 +860,13 @@ static int w1_init(void)
goto err_out_master_unregister; goto err_out_master_unregister;
} }
w1_control_thread = kthread_run(w1_control, NULL, "w1_control");
if (IS_ERR(w1_control_thread)) {
retval = PTR_ERR(w1_control_thread);
printk(KERN_ERR "Failed to create control thread. err=%d\n",
retval);
goto err_out_slave_unregister;
}
return 0; return 0;
#if 0
/* For undoing the slave register if there was a step after it. */
err_out_slave_unregister: err_out_slave_unregister:
driver_unregister(&w1_slave_driver); driver_unregister(&w1_slave_driver);
#endif
err_out_master_unregister: err_out_master_unregister:
driver_unregister(&w1_master_driver); driver_unregister(&w1_master_driver);
...@@ -959,13 +882,12 @@ static void w1_fini(void) ...@@ -959,13 +882,12 @@ static void w1_fini(void)
{ {
struct w1_master *dev; struct w1_master *dev;
/* Set netlink removal messages and some cleanup */
list_for_each_entry(dev, &w1_masters, w1_master_entry) list_for_each_entry(dev, &w1_masters, w1_master_entry)
__w1_remove_master_device(dev); __w1_remove_master_device(dev);
w1_fini_netlink(); w1_fini_netlink();
kthread_stop(w1_control_thread);
driver_unregister(&w1_slave_driver); driver_unregister(&w1_slave_driver);
driver_unregister(&w1_master_driver); driver_unregister(&w1_master_driver);
bus_unregister(&w1_bus_type); bus_unregister(&w1_bus_type);
......
...@@ -77,7 +77,7 @@ struct w1_slave ...@@ -77,7 +77,7 @@ struct w1_slave
struct completion released; struct completion released;
}; };
typedef void (* w1_slave_found_callback)(void *, u64); typedef void (*w1_slave_found_callback)(struct w1_master *, u64);
/** /**
...@@ -142,12 +142,14 @@ struct w1_bus_master ...@@ -142,12 +142,14 @@ struct w1_bus_master
*/ */
u8 (*reset_bus)(void *); u8 (*reset_bus)(void *);
/** Really nice hardware can handles the different types of ROM search */ /** Really nice hardware can handles the different types of ROM search
void (*search)(void *, u8, w1_slave_found_callback); * w1_master* is passed to the slave found callback.
*/
void (*search)(void *, struct w1_master *,
u8, w1_slave_found_callback);
}; };
#define W1_MASTER_NEED_EXIT 0 #define W1_MASTER_NEED_EXIT 0
#define W1_MASTER_NEED_RECONNECT 1
struct w1_master struct w1_master
{ {
...@@ -181,12 +183,21 @@ struct w1_master ...@@ -181,12 +183,21 @@ struct w1_master
}; };
int w1_create_master_attributes(struct w1_master *); int w1_create_master_attributes(struct w1_master *);
void w1_destroy_master_attributes(struct w1_master *master);
void w1_search(struct w1_master *dev, u8 search_type, w1_slave_found_callback cb); void w1_search(struct w1_master *dev, u8 search_type, w1_slave_found_callback cb);
void w1_search_devices(struct w1_master *dev, u8 search_type, w1_slave_found_callback cb); void w1_search_devices(struct w1_master *dev, u8 search_type, w1_slave_found_callback cb);
struct w1_slave *w1_search_slave(struct w1_reg_num *id); struct w1_slave *w1_search_slave(struct w1_reg_num *id);
void w1_search_process(struct w1_master *dev, u8 search_type); void w1_search_process(struct w1_master *dev, u8 search_type);
struct w1_master *w1_search_master_id(u32 id); struct w1_master *w1_search_master_id(u32 id);
/* Disconnect and reconnect devices in the given family. Used for finding
* unclaimed devices after a family has been registered or releasing devices
* after a family has been unregistered. Set attach to 1 when a new family
* has just been registered, to 0 when it has been unregistered.
*/
void w1_reconnect_slaves(struct w1_family *f, int attach);
void w1_slave_detach(struct w1_slave *sl);
u8 w1_triplet(struct w1_master *dev, int bdir); u8 w1_triplet(struct w1_master *dev, int bdir);
void w1_write_8(struct w1_master *, u8); void w1_write_8(struct w1_master *, u8);
int w1_reset_bus(struct w1_master *); int w1_reset_bus(struct w1_master *);
......
...@@ -53,7 +53,8 @@ int w1_register_family(struct w1_family *newf) ...@@ -53,7 +53,8 @@ int w1_register_family(struct w1_family *newf)
} }
spin_unlock(&w1_flock); spin_unlock(&w1_flock);
w1_reconnect_slaves(newf); /* check default devices against the new set of drivers */
w1_reconnect_slaves(newf, 1);
return ret; return ret;
} }
...@@ -77,6 +78,9 @@ void w1_unregister_family(struct w1_family *fent) ...@@ -77,6 +78,9 @@ void w1_unregister_family(struct w1_family *fent)
spin_unlock(&w1_flock); spin_unlock(&w1_flock);
/* deatch devices using this family code */
w1_reconnect_slaves(fent, 0);
while (atomic_read(&fent->refcnt)) { while (atomic_read(&fent->refcnt)) {
printk(KERN_INFO "Waiting for family %u to become free: refcnt=%d.\n", printk(KERN_INFO "Waiting for family %u to become free: refcnt=%d.\n",
fent->fid, atomic_read(&fent->refcnt)); fent->fid, atomic_read(&fent->refcnt));
......
...@@ -63,6 +63,5 @@ void __w1_family_get(struct w1_family *); ...@@ -63,6 +63,5 @@ void __w1_family_get(struct w1_family *);
struct w1_family * w1_family_registered(u8); struct w1_family * w1_family_registered(u8);
void w1_unregister_family(struct w1_family *); void w1_unregister_family(struct w1_family *);
int w1_register_family(struct w1_family *); int w1_register_family(struct w1_family *);
void w1_reconnect_slaves(struct w1_family *f);
#endif /* __W1_FAMILY_H */ #endif /* __W1_FAMILY_H */
...@@ -148,10 +148,22 @@ int w1_add_master_device(struct w1_bus_master *master) ...@@ -148,10 +148,22 @@ int w1_add_master_device(struct w1_bus_master *master)
void __w1_remove_master_device(struct w1_master *dev) void __w1_remove_master_device(struct w1_master *dev)
{ {
struct w1_netlink_msg msg; struct w1_netlink_msg msg;
struct w1_slave *sl, *sln;
set_bit(W1_MASTER_NEED_EXIT, &dev->flags); set_bit(W1_MASTER_NEED_EXIT, &dev->flags);
kthread_stop(dev->thread); kthread_stop(dev->thread);
mutex_lock(&w1_mlock);
list_del(&dev->w1_master_entry);
mutex_unlock(&w1_mlock);
mutex_lock(&dev->mutex);
list_for_each_entry_safe(sl, sln, &dev->slist, w1_slave_entry)
w1_slave_detach(sl);
w1_destroy_master_attributes(dev);
mutex_unlock(&dev->mutex);
atomic_dec(&dev->refcnt);
while (atomic_read(&dev->refcnt)) { while (atomic_read(&dev->refcnt)) {
dev_info(&dev->dev, "Waiting for %s to become free: refcnt=%d.\n", dev_info(&dev->dev, "Waiting for %s to become free: refcnt=%d.\n",
dev->name, atomic_read(&dev->refcnt)); dev->name, atomic_read(&dev->refcnt));
......
...@@ -277,7 +277,8 @@ void w1_search_devices(struct w1_master *dev, u8 search_type, w1_slave_found_cal ...@@ -277,7 +277,8 @@ void w1_search_devices(struct w1_master *dev, u8 search_type, w1_slave_found_cal
{ {
dev->attempts++; dev->attempts++;
if (dev->bus_master->search) if (dev->bus_master->search)
dev->bus_master->search(dev->bus_master->data, search_type, cb); dev->bus_master->search(dev->bus_master->data, dev,
search_type, cb);
else else
w1_search(dev, search_type, cb); w1_search(dev, search_type, cb);
} }
......
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