diff options
| author | David Howells <dhowells@redhat.com> | 2010-08-06 12:26:48 -0400 |
|---|---|---|
| committer | Linus Torvalds <torvalds@linux-foundation.org> | 2010-08-07 17:23:37 -0400 |
| commit | df44f9f4f9b5d362b5a2d1c8444fe7e6d4c42653 (patch) | |
| tree | cb27a80199e4730a98ed6fc93411d52df0001d37 | |
| parent | 5df6b8e65ad0f2eaee202ff002ac00d1ac605315 (diff) | |
AFS: Fix the module init error handling
Fix the module init error handling. There are a bunch of goto labels for
aborting the init procedure at different points and just undoing what needs
undoing - they aren't all in the right places, however.
This can lead to an oops like the following:
BUG: unable to handle kernel NULL pointer dereference at 0000000000000020
IP: [<ffffffff81042a31>] destroy_workqueue+0x17/0xc0
...
Modules linked in: kafs(+) dns_resolver rxkad af_rxrpc fscache
Pid: 2171, comm: insmod Not tainted 2.6.35-cachefs+ #319 DG965RY/
...
Process insmod (pid: 2171, threadinfo ffff88003ca6a000, task ffff88003dcc3050)
...
Call Trace:
[<ffffffffa0055994>] afs_callback_update_kill+0x10/0x12 [kafs]
[<ffffffffa007d1c5>] afs_init+0x190/0x1ce [kafs]
[<ffffffffa007d035>] ? afs_init+0x0/0x1ce [kafs]
[<ffffffff810001ef>] do_one_initcall+0x59/0x14e
[<ffffffff8105f7ee>] sys_init_module+0x9c/0x1de
[<ffffffff81001eab>] system_call_fastpath+0x16/0x1b
Signed-off-by: David Howells <dhowells@redhat.com>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
| -rw-r--r-- | fs/afs/main.c | 9 |
1 files changed, 6 insertions, 3 deletions
diff --git a/fs/afs/main.c b/fs/afs/main.c index 66d54d348c55..cfd1cbe25b22 100644 --- a/fs/afs/main.c +++ b/fs/afs/main.c | |||
| @@ -111,6 +111,8 @@ static int __init afs_init(void) | |||
| 111 | 111 | ||
| 112 | /* initialise the callback update process */ | 112 | /* initialise the callback update process */ |
| 113 | ret = afs_callback_update_init(); | 113 | ret = afs_callback_update_init(); |
| 114 | if (ret < 0) | ||
| 115 | goto error_callback_update_init; | ||
| 114 | 116 | ||
| 115 | /* create the RxRPC transport */ | 117 | /* create the RxRPC transport */ |
| 116 | ret = afs_open_socket(); | 118 | ret = afs_open_socket(); |
| @@ -127,15 +129,16 @@ static int __init afs_init(void) | |||
| 127 | error_fs: | 129 | error_fs: |
| 128 | afs_close_socket(); | 130 | afs_close_socket(); |
| 129 | error_open_socket: | 131 | error_open_socket: |
| 132 | afs_callback_update_kill(); | ||
| 133 | error_callback_update_init: | ||
| 134 | afs_vlocation_purge(); | ||
| 130 | error_vl_update_init: | 135 | error_vl_update_init: |
| 136 | afs_cell_purge(); | ||
| 131 | error_cell_init: | 137 | error_cell_init: |
| 132 | #ifdef CONFIG_AFS_FSCACHE | 138 | #ifdef CONFIG_AFS_FSCACHE |
| 133 | fscache_unregister_netfs(&afs_cache_netfs); | 139 | fscache_unregister_netfs(&afs_cache_netfs); |
| 134 | error_cache: | 140 | error_cache: |
| 135 | #endif | 141 | #endif |
| 136 | afs_callback_update_kill(); | ||
| 137 | afs_vlocation_purge(); | ||
| 138 | afs_cell_purge(); | ||
| 139 | afs_proc_cleanup(); | 142 | afs_proc_cleanup(); |
| 140 | rcu_barrier(); | 143 | rcu_barrier(); |
| 141 | printk(KERN_ERR "kAFS: failed to register: %d\n", ret); | 144 | printk(KERN_ERR "kAFS: failed to register: %d\n", ret); |
