aboutsummaryrefslogtreecommitdiffstats
path: root/kernel/module.c
diff options
context:
space:
mode:
authorRusty Russell <rusty@rustcorp.com.au>2013-04-16 23:50:03 -0400
committerRusty Russell <rusty@rustcorp.com.au>2013-04-16 23:53:02 -0400
commit944a1fa01266aa9ace607f29551b73c41e9440e9 (patch)
tree24ca05b22b795175ceb148d71062ead73f60f250 /kernel/module.c
parente3f26752f0f8a6aade580115e1e68bcb1a4bc040 (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.c13
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);