diff options
author | Rusty Russell <rusty@rustcorp.com.au> | 2013-04-16 23:50:03 -0400 |
---|---|---|
committer | Rusty Russell <rusty@rustcorp.com.au> | 2013-04-16 23:53:02 -0400 |
commit | 944a1fa01266aa9ace607f29551b73c41e9440e9 (patch) | |
tree | 24ca05b22b795175ceb148d71062ead73f60f250 /kernel/module.c | |
parent | e3f26752f0f8a6aade580115e1e68bcb1a4bc040 (diff) |
module: don't unlink the module until we've removed all exposure.
Otherwise we get a race between unload and reload of the same module:
the new module doesn't see the old one in the list, but then fails because
it can't register over the still-extant entries in sysfs:
[ 103.981925] ------------[ cut here ]------------
[ 103.986902] WARNING: at fs/sysfs/dir.c:536 sysfs_add_one+0xab/0xd0()
[ 103.993606] Hardware name: CrownBay Platform
[ 103.998075] sysfs: cannot create duplicate filename '/module/pch_gbe'
[ 104.004784] Modules linked in: pch_gbe(+) [last unloaded: pch_gbe]
[ 104.011362] Pid: 3021, comm: modprobe Tainted: G W 3.9.0-rc5+ #5
[ 104.018662] Call Trace:
[ 104.021286] [<c103599d>] warn_slowpath_common+0x6d/0xa0
[ 104.026933] [<c1168c8b>] ? sysfs_add_one+0xab/0xd0
[ 104.031986] [<c1168c8b>] ? sysfs_add_one+0xab/0xd0
[ 104.037000] [<c1035a4e>] warn_slowpath_fmt+0x2e/0x30
[ 104.042188] [<c1168c8b>] sysfs_add_one+0xab/0xd0
[ 104.046982] [<c1168dbe>] create_dir+0x5e/0xa0
[ 104.051633] [<c1168e78>] sysfs_create_dir+0x78/0xd0
[ 104.056774] [<c1262bc3>] kobject_add_internal+0x83/0x1f0
[ 104.062351] [<c126daf6>] ? kvasprintf+0x46/0x60
[ 104.067231] [<c1262ebd>] kobject_add_varg+0x2d/0x50
[ 104.072450] [<c1262f07>] kobject_init_and_add+0x27/0x30
[ 104.078075] [<c1089240>] mod_sysfs_setup+0x80/0x540
[ 104.083207] [<c1260851>] ? module_bug_finalize+0x51/0xc0
[ 104.088720] [<c108ab29>] load_module+0x1429/0x18b0
We can teardown sysfs first, then to be sure, put the state in
MODULE_STATE_UNFORMED so it's ignored while we deconstruct it.
Reported-by: Veaceslav Falico <vfalico@redhat.com>
Tested-by: Veaceslav Falico <vfalico@redhat.com>
Signed-off-by: Rusty Russell <rusty@rustcorp.com.au>
Diffstat (limited to 'kernel/module.c')
-rw-r--r-- | kernel/module.c | 13 |
1 files changed, 9 insertions, 4 deletions
diff --git a/kernel/module.c b/kernel/module.c index 3c2c72d3bf84..b049939177f6 100644 --- a/kernel/module.c +++ b/kernel/module.c | |||
@@ -1862,12 +1862,12 @@ static void free_module(struct module *mod) | |||
1862 | { | 1862 | { |
1863 | trace_module_free(mod); | 1863 | trace_module_free(mod); |
1864 | 1864 | ||
1865 | /* Delete from various lists */ | ||
1866 | mutex_lock(&module_mutex); | ||
1867 | stop_machine(__unlink_module, mod, NULL); | ||
1868 | mutex_unlock(&module_mutex); | ||
1869 | mod_sysfs_teardown(mod); | 1865 | mod_sysfs_teardown(mod); |
1870 | 1866 | ||
1867 | /* We leave it in list to prevent duplicate loads, but make sure | ||
1868 | * that noone uses it while it's being deconstructed. */ | ||
1869 | mod->state = MODULE_STATE_UNFORMED; | ||
1870 | |||
1871 | /* Remove dynamic debug info */ | 1871 | /* Remove dynamic debug info */ |
1872 | ddebug_remove_module(mod->name); | 1872 | ddebug_remove_module(mod->name); |
1873 | 1873 | ||
@@ -1880,6 +1880,11 @@ static void free_module(struct module *mod) | |||
1880 | /* Free any allocated parameters. */ | 1880 | /* Free any allocated parameters. */ |
1881 | destroy_params(mod->kp, mod->num_kp); | 1881 | destroy_params(mod->kp, mod->num_kp); |
1882 | 1882 | ||
1883 | /* Now we can delete it from the lists */ | ||
1884 | mutex_lock(&module_mutex); | ||
1885 | stop_machine(__unlink_module, mod, NULL); | ||
1886 | mutex_unlock(&module_mutex); | ||
1887 | |||
1883 | /* This may be NULL, but that's OK */ | 1888 | /* This may be NULL, but that's OK */ |
1884 | unset_module_init_ro_nx(mod); | 1889 | unset_module_init_ro_nx(mod); |
1885 | module_free(mod, mod->module_init); | 1890 | module_free(mod, mod->module_init); |