diff options
Diffstat (limited to 'kernel/module.c')
| -rw-r--r-- | kernel/module.c | 154 |
1 files changed, 108 insertions, 46 deletions
diff --git a/kernel/module.c b/kernel/module.c index b10b048367e1..eab08274ec9b 100644 --- a/kernel/module.c +++ b/kernel/module.c | |||
| @@ -188,6 +188,7 @@ struct load_info { | |||
| 188 | ongoing or failed initialization etc. */ | 188 | ongoing or failed initialization etc. */ |
| 189 | static inline int strong_try_module_get(struct module *mod) | 189 | static inline int strong_try_module_get(struct module *mod) |
| 190 | { | 190 | { |
| 191 | BUG_ON(mod && mod->state == MODULE_STATE_UNFORMED); | ||
| 191 | if (mod && mod->state == MODULE_STATE_COMING) | 192 | if (mod && mod->state == MODULE_STATE_COMING) |
| 192 | return -EBUSY; | 193 | return -EBUSY; |
| 193 | if (try_module_get(mod)) | 194 | if (try_module_get(mod)) |
| @@ -343,6 +344,9 @@ bool each_symbol_section(bool (*fn)(const struct symsearch *arr, | |||
| 343 | #endif | 344 | #endif |
| 344 | }; | 345 | }; |
| 345 | 346 | ||
| 347 | if (mod->state == MODULE_STATE_UNFORMED) | ||
| 348 | continue; | ||
| 349 | |||
| 346 | if (each_symbol_in_section(arr, ARRAY_SIZE(arr), mod, fn, data)) | 350 | if (each_symbol_in_section(arr, ARRAY_SIZE(arr), mod, fn, data)) |
| 347 | return true; | 351 | return true; |
| 348 | } | 352 | } |
| @@ -450,16 +454,24 @@ const struct kernel_symbol *find_symbol(const char *name, | |||
| 450 | EXPORT_SYMBOL_GPL(find_symbol); | 454 | EXPORT_SYMBOL_GPL(find_symbol); |
| 451 | 455 | ||
| 452 | /* Search for module by name: must hold module_mutex. */ | 456 | /* Search for module by name: must hold module_mutex. */ |
| 453 | struct module *find_module(const char *name) | 457 | static struct module *find_module_all(const char *name, |
| 458 | bool even_unformed) | ||
| 454 | { | 459 | { |
| 455 | struct module *mod; | 460 | struct module *mod; |
| 456 | 461 | ||
| 457 | list_for_each_entry(mod, &modules, list) { | 462 | list_for_each_entry(mod, &modules, list) { |
| 463 | if (!even_unformed && mod->state == MODULE_STATE_UNFORMED) | ||
| 464 | continue; | ||
| 458 | if (strcmp(mod->name, name) == 0) | 465 | if (strcmp(mod->name, name) == 0) |
| 459 | return mod; | 466 | return mod; |
| 460 | } | 467 | } |
| 461 | return NULL; | 468 | return NULL; |
| 462 | } | 469 | } |
| 470 | |||
| 471 | struct module *find_module(const char *name) | ||
| 472 | { | ||
| 473 | return find_module_all(name, false); | ||
| 474 | } | ||
| 463 | EXPORT_SYMBOL_GPL(find_module); | 475 | EXPORT_SYMBOL_GPL(find_module); |
| 464 | 476 | ||
| 465 | #ifdef CONFIG_SMP | 477 | #ifdef CONFIG_SMP |
| @@ -525,6 +537,8 @@ bool is_module_percpu_address(unsigned long addr) | |||
| 525 | preempt_disable(); | 537 | preempt_disable(); |
| 526 | 538 | ||
| 527 | list_for_each_entry_rcu(mod, &modules, list) { | 539 | list_for_each_entry_rcu(mod, &modules, list) { |
| 540 | if (mod->state == MODULE_STATE_UNFORMED) | ||
| 541 | continue; | ||
| 528 | if (!mod->percpu_size) | 542 | if (!mod->percpu_size) |
| 529 | continue; | 543 | continue; |
| 530 | for_each_possible_cpu(cpu) { | 544 | for_each_possible_cpu(cpu) { |
| @@ -1048,6 +1062,8 @@ static ssize_t show_initstate(struct module_attribute *mattr, | |||
| 1048 | case MODULE_STATE_GOING: | 1062 | case MODULE_STATE_GOING: |
| 1049 | state = "going"; | 1063 | state = "going"; |
| 1050 | break; | 1064 | break; |
| 1065 | default: | ||
| 1066 | BUG(); | ||
| 1051 | } | 1067 | } |
| 1052 | return sprintf(buffer, "%s\n", state); | 1068 | return sprintf(buffer, "%s\n", state); |
| 1053 | } | 1069 | } |
| @@ -1786,6 +1802,8 @@ void set_all_modules_text_rw(void) | |||
| 1786 | 1802 | ||
| 1787 | mutex_lock(&module_mutex); | 1803 | mutex_lock(&module_mutex); |
| 1788 | list_for_each_entry_rcu(mod, &modules, list) { | 1804 | list_for_each_entry_rcu(mod, &modules, list) { |
| 1805 | if (mod->state == MODULE_STATE_UNFORMED) | ||
| 1806 | continue; | ||
| 1789 | if ((mod->module_core) && (mod->core_text_size)) { | 1807 | if ((mod->module_core) && (mod->core_text_size)) { |
| 1790 | set_page_attributes(mod->module_core, | 1808 | set_page_attributes(mod->module_core, |
| 1791 | mod->module_core + mod->core_text_size, | 1809 | mod->module_core + mod->core_text_size, |
| @@ -1807,6 +1825,8 @@ void set_all_modules_text_ro(void) | |||
| 1807 | 1825 | ||
| 1808 | mutex_lock(&module_mutex); | 1826 | mutex_lock(&module_mutex); |
| 1809 | list_for_each_entry_rcu(mod, &modules, list) { | 1827 | list_for_each_entry_rcu(mod, &modules, list) { |
| 1828 | if (mod->state == MODULE_STATE_UNFORMED) | ||
| 1829 | continue; | ||
| 1810 | if ((mod->module_core) && (mod->core_text_size)) { | 1830 | if ((mod->module_core) && (mod->core_text_size)) { |
| 1811 | set_page_attributes(mod->module_core, | 1831 | set_page_attributes(mod->module_core, |
| 1812 | mod->module_core + mod->core_text_size, | 1832 | mod->module_core + mod->core_text_size, |
| @@ -2527,6 +2547,13 @@ static int copy_module_from_fd(int fd, struct load_info *info) | |||
| 2527 | err = -EFBIG; | 2547 | err = -EFBIG; |
| 2528 | goto out; | 2548 | goto out; |
| 2529 | } | 2549 | } |
| 2550 | |||
| 2551 | /* Don't hand 0 to vmalloc, it whines. */ | ||
| 2552 | if (stat.size == 0) { | ||
| 2553 | err = -EINVAL; | ||
| 2554 | goto out; | ||
| 2555 | } | ||
| 2556 | |||
| 2530 | info->hdr = vmalloc(stat.size); | 2557 | info->hdr = vmalloc(stat.size); |
| 2531 | if (!info->hdr) { | 2558 | if (!info->hdr) { |
| 2532 | err = -ENOMEM; | 2559 | err = -ENOMEM; |
| @@ -2990,8 +3017,9 @@ static bool finished_loading(const char *name) | |||
| 2990 | bool ret; | 3017 | bool ret; |
| 2991 | 3018 | ||
| 2992 | mutex_lock(&module_mutex); | 3019 | mutex_lock(&module_mutex); |
| 2993 | mod = find_module(name); | 3020 | mod = find_module_all(name, true); |
| 2994 | ret = !mod || mod->state != MODULE_STATE_COMING; | 3021 | ret = !mod || mod->state == MODULE_STATE_LIVE |
| 3022 | || mod->state == MODULE_STATE_GOING; | ||
| 2995 | mutex_unlock(&module_mutex); | 3023 | mutex_unlock(&module_mutex); |
| 2996 | 3024 | ||
| 2997 | return ret; | 3025 | return ret; |
| @@ -3136,6 +3164,32 @@ static int load_module(struct load_info *info, const char __user *uargs, | |||
| 3136 | goto free_copy; | 3164 | goto free_copy; |
| 3137 | } | 3165 | } |
| 3138 | 3166 | ||
| 3167 | /* | ||
| 3168 | * We try to place it in the list now to make sure it's unique | ||
| 3169 | * before we dedicate too many resources. In particular, | ||
| 3170 | * temporary percpu memory exhaustion. | ||
| 3171 | */ | ||
| 3172 | mod->state = MODULE_STATE_UNFORMED; | ||
| 3173 | again: | ||
| 3174 | mutex_lock(&module_mutex); | ||
| 3175 | if ((old = find_module_all(mod->name, true)) != NULL) { | ||
| 3176 | if (old->state == MODULE_STATE_COMING | ||
| 3177 | || old->state == MODULE_STATE_UNFORMED) { | ||
| 3178 | /* Wait in case it fails to load. */ | ||
| 3179 | mutex_unlock(&module_mutex); | ||
| 3180 | err = wait_event_interruptible(module_wq, | ||
| 3181 | finished_loading(mod->name)); | ||
| 3182 | if (err) | ||
| 3183 | goto free_module; | ||
| 3184 | goto again; | ||
| 3185 | } | ||
| 3186 | err = -EEXIST; | ||
| 3187 | mutex_unlock(&module_mutex); | ||
| 3188 | goto free_module; | ||
| 3189 | } | ||
| 3190 | list_add_rcu(&mod->list, &modules); | ||
| 3191 | mutex_unlock(&module_mutex); | ||
| 3192 | |||
| 3139 | #ifdef CONFIG_MODULE_SIG | 3193 | #ifdef CONFIG_MODULE_SIG |
| 3140 | mod->sig_ok = info->sig_ok; | 3194 | mod->sig_ok = info->sig_ok; |
| 3141 | if (!mod->sig_ok) | 3195 | if (!mod->sig_ok) |
| @@ -3145,7 +3199,7 @@ static int load_module(struct load_info *info, const char __user *uargs, | |||
| 3145 | /* Now module is in final location, initialize linked lists, etc. */ | 3199 | /* Now module is in final location, initialize linked lists, etc. */ |
| 3146 | err = module_unload_init(mod); | 3200 | err = module_unload_init(mod); |
| 3147 | if (err) | 3201 | if (err) |
| 3148 | goto free_module; | 3202 | goto unlink_mod; |
| 3149 | 3203 | ||
| 3150 | /* Now we've got everything in the final locations, we can | 3204 | /* Now we've got everything in the final locations, we can |
| 3151 | * find optional sections. */ | 3205 | * find optional sections. */ |
| @@ -3180,54 +3234,33 @@ static int load_module(struct load_info *info, const char __user *uargs, | |||
| 3180 | goto free_arch_cleanup; | 3234 | goto free_arch_cleanup; |
| 3181 | } | 3235 | } |
| 3182 | 3236 | ||
| 3183 | /* Mark state as coming so strong_try_module_get() ignores us. */ | ||
| 3184 | mod->state = MODULE_STATE_COMING; | ||
| 3185 | |||
| 3186 | /* Now sew it into the lists so we can get lockdep and oops | ||
| 3187 | * info during argument parsing. No one should access us, since | ||
| 3188 | * strong_try_module_get() will fail. | ||
| 3189 | * lockdep/oops can run asynchronous, so use the RCU list insertion | ||
| 3190 | * function to insert in a way safe to concurrent readers. | ||
| 3191 | * The mutex protects against concurrent writers. | ||
| 3192 | */ | ||
| 3193 | again: | ||
| 3194 | mutex_lock(&module_mutex); | ||
| 3195 | if ((old = find_module(mod->name)) != NULL) { | ||
| 3196 | if (old->state == MODULE_STATE_COMING) { | ||
| 3197 | /* Wait in case it fails to load. */ | ||
| 3198 | mutex_unlock(&module_mutex); | ||
| 3199 | err = wait_event_interruptible(module_wq, | ||
| 3200 | finished_loading(mod->name)); | ||
| 3201 | if (err) | ||
| 3202 | goto free_arch_cleanup; | ||
| 3203 | goto again; | ||
| 3204 | } | ||
| 3205 | err = -EEXIST; | ||
| 3206 | goto unlock; | ||
| 3207 | } | ||
| 3208 | |||
| 3209 | /* This has to be done once we're sure module name is unique. */ | ||
| 3210 | dynamic_debug_setup(info->debug, info->num_debug); | 3237 | dynamic_debug_setup(info->debug, info->num_debug); |
| 3211 | 3238 | ||
| 3212 | /* Find duplicate symbols */ | 3239 | mutex_lock(&module_mutex); |
| 3240 | /* Find duplicate symbols (must be called under lock). */ | ||
| 3213 | err = verify_export_symbols(mod); | 3241 | err = verify_export_symbols(mod); |
| 3214 | if (err < 0) | 3242 | if (err < 0) |
| 3215 | goto ddebug; | 3243 | goto ddebug_cleanup; |
| 3216 | 3244 | ||
| 3245 | /* This relies on module_mutex for list integrity. */ | ||
| 3217 | module_bug_finalize(info->hdr, info->sechdrs, mod); | 3246 | module_bug_finalize(info->hdr, info->sechdrs, mod); |
| 3218 | list_add_rcu(&mod->list, &modules); | 3247 | |
| 3248 | /* Mark state as coming so strong_try_module_get() ignores us, | ||
| 3249 | * but kallsyms etc. can see us. */ | ||
| 3250 | mod->state = MODULE_STATE_COMING; | ||
| 3251 | |||
| 3219 | mutex_unlock(&module_mutex); | 3252 | mutex_unlock(&module_mutex); |
| 3220 | 3253 | ||
| 3221 | /* Module is ready to execute: parsing args may do that. */ | 3254 | /* Module is ready to execute: parsing args may do that. */ |
| 3222 | err = parse_args(mod->name, mod->args, mod->kp, mod->num_kp, | 3255 | err = parse_args(mod->name, mod->args, mod->kp, mod->num_kp, |
| 3223 | -32768, 32767, &ddebug_dyndbg_module_param_cb); | 3256 | -32768, 32767, &ddebug_dyndbg_module_param_cb); |
| 3224 | if (err < 0) | 3257 | if (err < 0) |
| 3225 | goto unlink; | 3258 | goto bug_cleanup; |
| 3226 | 3259 | ||
| 3227 | /* Link in to syfs. */ | 3260 | /* Link in to syfs. */ |
| 3228 | err = mod_sysfs_setup(mod, info, mod->kp, mod->num_kp); | 3261 | err = mod_sysfs_setup(mod, info, mod->kp, mod->num_kp); |
| 3229 | if (err < 0) | 3262 | if (err < 0) |
| 3230 | goto unlink; | 3263 | goto bug_cleanup; |
| 3231 | 3264 | ||
| 3232 | /* Get rid of temporary copy. */ | 3265 | /* Get rid of temporary copy. */ |
| 3233 | free_copy(info); | 3266 | free_copy(info); |
| @@ -3237,16 +3270,13 @@ again: | |||
| 3237 | 3270 | ||
| 3238 | return do_init_module(mod); | 3271 | return do_init_module(mod); |
| 3239 | 3272 | ||
| 3240 | unlink: | 3273 | bug_cleanup: |
| 3274 | /* module_bug_cleanup needs module_mutex protection */ | ||
| 3241 | mutex_lock(&module_mutex); | 3275 | mutex_lock(&module_mutex); |
| 3242 | /* Unlink carefully: kallsyms could be walking list. */ | ||
| 3243 | list_del_rcu(&mod->list); | ||
| 3244 | module_bug_cleanup(mod); | 3276 | module_bug_cleanup(mod); |
| 3245 | wake_up_all(&module_wq); | 3277 | ddebug_cleanup: |
| 3246 | ddebug: | ||
| 3247 | dynamic_debug_remove(info->debug); | ||
| 3248 | unlock: | ||
| 3249 | mutex_unlock(&module_mutex); | 3278 | mutex_unlock(&module_mutex); |
| 3279 | dynamic_debug_remove(info->debug); | ||
| 3250 | synchronize_sched(); | 3280 | synchronize_sched(); |
| 3251 | kfree(mod->args); | 3281 | kfree(mod->args); |
| 3252 | free_arch_cleanup: | 3282 | free_arch_cleanup: |
| @@ -3255,6 +3285,12 @@ again: | |||
| 3255 | free_modinfo(mod); | 3285 | free_modinfo(mod); |
| 3256 | free_unload: | 3286 | free_unload: |
| 3257 | module_unload_free(mod); | 3287 | module_unload_free(mod); |
| 3288 | unlink_mod: | ||
| 3289 | mutex_lock(&module_mutex); | ||
| 3290 | /* Unlink carefully: kallsyms could be walking list. */ | ||
| 3291 | list_del_rcu(&mod->list); | ||
| 3292 | wake_up_all(&module_wq); | ||
| 3293 | mutex_unlock(&module_mutex); | ||
| 3258 | free_module: | 3294 | free_module: |
| 3259 | module_deallocate(mod, info); | 3295 | module_deallocate(mod, info); |
| 3260 | free_copy: | 3296 | free_copy: |
| @@ -3377,6 +3413,8 @@ const char *module_address_lookup(unsigned long addr, | |||
| 3377 | 3413 | ||
| 3378 | preempt_disable(); | 3414 | preempt_disable(); |
| 3379 | list_for_each_entry_rcu(mod, &modules, list) { | 3415 | list_for_each_entry_rcu(mod, &modules, list) { |
| 3416 | if (mod->state == MODULE_STATE_UNFORMED) | ||
| 3417 | continue; | ||
| 3380 | if (within_module_init(addr, mod) || | 3418 | if (within_module_init(addr, mod) || |
| 3381 | within_module_core(addr, mod)) { | 3419 | within_module_core(addr, mod)) { |
| 3382 | if (modname) | 3420 | if (modname) |
| @@ -3400,6 +3438,8 @@ int lookup_module_symbol_name(unsigned long addr, char *symname) | |||
| 3400 | 3438 | ||
| 3401 | preempt_disable(); | 3439 | preempt_disable(); |
| 3402 | list_for_each_entry_rcu(mod, &modules, list) { | 3440 | list_for_each_entry_rcu(mod, &modules, list) { |
| 3441 | if (mod->state == MODULE_STATE_UNFORMED) | ||
| 3442 | continue; | ||
| 3403 | if (within_module_init(addr, mod) || | 3443 | if (within_module_init(addr, mod) || |
| 3404 | within_module_core(addr, mod)) { | 3444 | within_module_core(addr, mod)) { |
| 3405 | const char *sym; | 3445 | const char *sym; |
| @@ -3424,6 +3464,8 @@ int lookup_module_symbol_attrs(unsigned long addr, unsigned long *size, | |||
| 3424 | 3464 | ||
| 3425 | preempt_disable(); | 3465 | preempt_disable(); |
| 3426 | list_for_each_entry_rcu(mod, &modules, list) { | 3466 | list_for_each_entry_rcu(mod, &modules, list) { |
| 3467 | if (mod->state == MODULE_STATE_UNFORMED) | ||
| 3468 | continue; | ||
| 3427 | if (within_module_init(addr, mod) || | 3469 | if (within_module_init(addr, mod) || |
| 3428 | within_module_core(addr, mod)) { | 3470 | within_module_core(addr, mod)) { |
| 3429 | const char *sym; | 3471 | const char *sym; |
| @@ -3451,6 +3493,8 @@ int module_get_kallsym(unsigned int symnum, unsigned long *value, char *type, | |||
| 3451 | 3493 | ||
| 3452 | preempt_disable(); | 3494 | preempt_disable(); |
| 3453 | list_for_each_entry_rcu(mod, &modules, list) { | 3495 | list_for_each_entry_rcu(mod, &modules, list) { |
| 3496 | if (mod->state == MODULE_STATE_UNFORMED) | ||
| 3497 | continue; | ||
| 3454 | if (symnum < mod->num_symtab) { | 3498 | if (symnum < mod->num_symtab) { |
| 3455 | *value = mod->symtab[symnum].st_value; | 3499 | *value = mod->symtab[symnum].st_value; |
| 3456 | *type = mod->symtab[symnum].st_info; | 3500 | *type = mod->symtab[symnum].st_info; |
| @@ -3493,9 +3537,12 @@ unsigned long module_kallsyms_lookup_name(const char *name) | |||
| 3493 | ret = mod_find_symname(mod, colon+1); | 3537 | ret = mod_find_symname(mod, colon+1); |
| 3494 | *colon = ':'; | 3538 | *colon = ':'; |
| 3495 | } else { | 3539 | } else { |
| 3496 | list_for_each_entry_rcu(mod, &modules, list) | 3540 | list_for_each_entry_rcu(mod, &modules, list) { |
| 3541 | if (mod->state == MODULE_STATE_UNFORMED) | ||
| 3542 | continue; | ||
| 3497 | if ((ret = mod_find_symname(mod, name)) != 0) | 3543 | if ((ret = mod_find_symname(mod, name)) != 0) |
| 3498 | break; | 3544 | break; |
| 3545 | } | ||
| 3499 | } | 3546 | } |
| 3500 | preempt_enable(); | 3547 | preempt_enable(); |
| 3501 | return ret; | 3548 | return ret; |
| @@ -3510,6 +3557,8 @@ int module_kallsyms_on_each_symbol(int (*fn)(void *, const char *, | |||
| 3510 | int ret; | 3557 | int ret; |
| 3511 | 3558 | ||
| 3512 | list_for_each_entry(mod, &modules, list) { | 3559 | list_for_each_entry(mod, &modules, list) { |
| 3560 | if (mod->state == MODULE_STATE_UNFORMED) | ||
| 3561 | continue; | ||
| 3513 | for (i = 0; i < mod->num_symtab; i++) { | 3562 | for (i = 0; i < mod->num_symtab; i++) { |
| 3514 | ret = fn(data, mod->strtab + mod->symtab[i].st_name, | 3563 | ret = fn(data, mod->strtab + mod->symtab[i].st_name, |
| 3515 | mod, mod->symtab[i].st_value); | 3564 | mod, mod->symtab[i].st_value); |
| @@ -3525,6 +3574,7 @@ static char *module_flags(struct module *mod, char *buf) | |||
| 3525 | { | 3574 | { |
| 3526 | int bx = 0; | 3575 | int bx = 0; |
| 3527 | 3576 | ||
| 3577 | BUG_ON(mod->state == MODULE_STATE_UNFORMED); | ||
| 3528 | if (mod->taints || | 3578 | if (mod->taints || |
| 3529 | mod->state == MODULE_STATE_GOING || | 3579 | mod->state == MODULE_STATE_GOING || |
| 3530 | mod->state == MODULE_STATE_COMING) { | 3580 | mod->state == MODULE_STATE_COMING) { |
| @@ -3566,6 +3616,10 @@ static int m_show(struct seq_file *m, void *p) | |||
| 3566 | struct module *mod = list_entry(p, struct module, list); | 3616 | struct module *mod = list_entry(p, struct module, list); |
| 3567 | char buf[8]; | 3617 | char buf[8]; |
| 3568 | 3618 | ||
| 3619 | /* We always ignore unformed modules. */ | ||
| 3620 | if (mod->state == MODULE_STATE_UNFORMED) | ||
| 3621 | return 0; | ||
| 3622 | |||
| 3569 | seq_printf(m, "%s %u", | 3623 | seq_printf(m, "%s %u", |
| 3570 | mod->name, mod->init_size + mod->core_size); | 3624 | mod->name, mod->init_size + mod->core_size); |
| 3571 | print_unload_info(m, mod); | 3625 | print_unload_info(m, mod); |
| @@ -3626,6 +3680,8 @@ const struct exception_table_entry *search_module_extables(unsigned long addr) | |||
| 3626 | 3680 | ||
| 3627 | preempt_disable(); | 3681 | preempt_disable(); |
| 3628 | list_for_each_entry_rcu(mod, &modules, list) { | 3682 | list_for_each_entry_rcu(mod, &modules, list) { |
| 3683 | if (mod->state == MODULE_STATE_UNFORMED) | ||
| 3684 | continue; | ||
| 3629 | if (mod->num_exentries == 0) | 3685 | if (mod->num_exentries == 0) |
| 3630 | continue; | 3686 | continue; |
| 3631 | 3687 | ||
| @@ -3674,10 +3730,13 @@ struct module *__module_address(unsigned long addr) | |||
| 3674 | if (addr < module_addr_min || addr > module_addr_max) | 3730 | if (addr < module_addr_min || addr > module_addr_max) |
| 3675 | return NULL; | 3731 | return NULL; |
| 3676 | 3732 | ||
| 3677 | list_for_each_entry_rcu(mod, &modules, list) | 3733 | list_for_each_entry_rcu(mod, &modules, list) { |
| 3734 | if (mod->state == MODULE_STATE_UNFORMED) | ||
| 3735 | continue; | ||
| 3678 | if (within_module_core(addr, mod) | 3736 | if (within_module_core(addr, mod) |
| 3679 | || within_module_init(addr, mod)) | 3737 | || within_module_init(addr, mod)) |
| 3680 | return mod; | 3738 | return mod; |
| 3739 | } | ||
| 3681 | return NULL; | 3740 | return NULL; |
| 3682 | } | 3741 | } |
| 3683 | EXPORT_SYMBOL_GPL(__module_address); | 3742 | EXPORT_SYMBOL_GPL(__module_address); |
| @@ -3730,8 +3789,11 @@ void print_modules(void) | |||
| 3730 | printk(KERN_DEFAULT "Modules linked in:"); | 3789 | printk(KERN_DEFAULT "Modules linked in:"); |
| 3731 | /* Most callers should already have preempt disabled, but make sure */ | 3790 | /* Most callers should already have preempt disabled, but make sure */ |
| 3732 | preempt_disable(); | 3791 | preempt_disable(); |
| 3733 | list_for_each_entry_rcu(mod, &modules, list) | 3792 | list_for_each_entry_rcu(mod, &modules, list) { |
| 3793 | if (mod->state == MODULE_STATE_UNFORMED) | ||
| 3794 | continue; | ||
| 3734 | printk(" %s%s", mod->name, module_flags(mod, buf)); | 3795 | printk(" %s%s", mod->name, module_flags(mod, buf)); |
| 3796 | } | ||
| 3735 | preempt_enable(); | 3797 | preempt_enable(); |
| 3736 | if (last_unloaded_module[0]) | 3798 | if (last_unloaded_module[0]) |
| 3737 | printk(" [last unloaded: %s]", last_unloaded_module); | 3799 | printk(" [last unloaded: %s]", last_unloaded_module); |
