aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorRusty Russell <rusty@rustcorp.com.au>2010-08-05 14:59:08 -0400
committerRusty Russell <rusty@rustcorp.com.au>2010-08-04 23:29:09 -0400
commitd913188c75191114051cf0bac75dad444c6080fa (patch)
tree1c32e48b3524f775ef9998f71b016d325205697b
parent511ca6ae43fbe0a7c9e0b50ad275f7ef24ef3b58 (diff)
module: layout_and_allocate
layout_and_allocate() does everything up to and including the final struct module placement inside the allocated module memory. We have to store the symbol layout information in our struct load_info though. This avoids the nasty code we had before where 'mod' pointed first to the version inside the temporary allocation containing the entire file, then later was moved to point to the real struct module: now the main code only ever sees the final module address. (Includes fix for the Tony Luck-found Linus-diagnosed failure path error). Signed-off-by: Rusty Russell <rusty@rustcorp.com.au>
-rw-r--r--kernel/module.c224
1 files changed, 125 insertions, 99 deletions
diff --git a/kernel/module.c b/kernel/module.c
index 60cdd0459eac..fb11e2a88233 100644
--- a/kernel/module.c
+++ b/kernel/module.c
@@ -115,6 +115,8 @@ struct load_info {
115 unsigned long len; 115 unsigned long len;
116 Elf_Shdr *sechdrs; 116 Elf_Shdr *sechdrs;
117 char *secstrings, *args, *strtab; 117 char *secstrings, *args, *strtab;
118 unsigned long *strmap;
119 unsigned long symoffs, stroffs;
118 struct { 120 struct {
119 unsigned int sym, str, mod, vers, info, pcpu; 121 unsigned int sym, str, mod, vers, info, pcpu;
120 } index; 122 } index;
@@ -402,7 +404,8 @@ static int percpu_modalloc(struct module *mod,
402 mod->percpu = __alloc_reserved_percpu(size, align); 404 mod->percpu = __alloc_reserved_percpu(size, align);
403 if (!mod->percpu) { 405 if (!mod->percpu) {
404 printk(KERN_WARNING 406 printk(KERN_WARNING
405 "Could not allocate %lu bytes percpu data\n", size); 407 "%s: Could not allocate %lu bytes percpu data\n",
408 mod->name, size);
406 return -ENOMEM; 409 return -ENOMEM;
407 } 410 }
408 mod->percpu_size = size; 411 mod->percpu_size = size;
@@ -2032,10 +2035,7 @@ static unsigned long layout_symtab(struct module *mod,
2032 return symoffs; 2035 return symoffs;
2033} 2036}
2034 2037
2035static void add_kallsyms(struct module *mod, struct load_info *info, 2038static void add_kallsyms(struct module *mod, struct load_info *info)
2036 unsigned long symoffs,
2037 unsigned long stroffs,
2038 unsigned long *strmap)
2039{ 2039{
2040 unsigned int i, ndst; 2040 unsigned int i, ndst;
2041 const Elf_Sym *src; 2041 const Elf_Sym *src;
@@ -2052,21 +2052,22 @@ static void add_kallsyms(struct module *mod, struct load_info *info,
2052 for (i = 0; i < mod->num_symtab; i++) 2052 for (i = 0; i < mod->num_symtab; i++)
2053 mod->symtab[i].st_info = elf_type(&mod->symtab[i], info); 2053 mod->symtab[i].st_info = elf_type(&mod->symtab[i], info);
2054 2054
2055 mod->core_symtab = dst = mod->module_core + symoffs; 2055 mod->core_symtab = dst = mod->module_core + info->symoffs;
2056 src = mod->symtab; 2056 src = mod->symtab;
2057 *dst = *src; 2057 *dst = *src;
2058 for (ndst = i = 1; i < mod->num_symtab; ++i, ++src) { 2058 for (ndst = i = 1; i < mod->num_symtab; ++i, ++src) {
2059 if (!is_core_symbol(src, info->sechdrs, info->hdr->e_shnum)) 2059 if (!is_core_symbol(src, info->sechdrs, info->hdr->e_shnum))
2060 continue; 2060 continue;
2061 dst[ndst] = *src; 2061 dst[ndst] = *src;
2062 dst[ndst].st_name = bitmap_weight(strmap, dst[ndst].st_name); 2062 dst[ndst].st_name = bitmap_weight(info->strmap,
2063 dst[ndst].st_name);
2063 ++ndst; 2064 ++ndst;
2064 } 2065 }
2065 mod->core_num_syms = ndst; 2066 mod->core_num_syms = ndst;
2066 2067
2067 mod->core_strtab = s = mod->module_core + stroffs; 2068 mod->core_strtab = s = mod->module_core + info->stroffs;
2068 for (*s = 0, i = 1; i < info->sechdrs[info->index.str].sh_size; ++i) 2069 for (*s = 0, i = 1; i < info->sechdrs[info->index.str].sh_size; ++i)
2069 if (test_bit(i, strmap)) 2070 if (test_bit(i, info->strmap))
2070 *++s = mod->strtab[i]; 2071 *++s = mod->strtab[i];
2071} 2072}
2072#else 2073#else
@@ -2082,10 +2083,7 @@ static inline unsigned long layout_symtab(struct module *mod,
2082 return 0; 2083 return 0;
2083} 2084}
2084 2085
2085static void add_kallsyms(struct module *mod, struct load_info *info, 2086static void add_kallsyms(struct module *mod, struct load_info *info)
2086 unsigned long symoffs,
2087 unsigned long stroffs,
2088 unsigned long *strmap)
2089{ 2087{
2090} 2088}
2091#endif /* CONFIG_KALLSYMS */ 2089#endif /* CONFIG_KALLSYMS */
@@ -2150,8 +2148,10 @@ static inline void kmemleak_load_module(struct module *mod, Elf_Ehdr *hdr,
2150} 2148}
2151#endif 2149#endif
2152 2150
2153/* Sets info->hdr and info->len. */ 2151/* Sets info->hdr, info->len and info->args. */
2154static int copy_and_check(struct load_info *info, const void __user *umod, unsigned long len) 2152static int copy_and_check(struct load_info *info,
2153 const void __user *umod, unsigned long len,
2154 const char __user *uargs)
2155{ 2155{
2156 int err; 2156 int err;
2157 Elf_Ehdr *hdr; 2157 Elf_Ehdr *hdr;
@@ -2183,6 +2183,14 @@ static int copy_and_check(struct load_info *info, const void __user *umod, unsig
2183 err = -ENOEXEC; 2183 err = -ENOEXEC;
2184 goto free_hdr; 2184 goto free_hdr;
2185 } 2185 }
2186
2187 /* Now copy in args */
2188 info->args = strndup_user(uargs, ~0UL >> 1);
2189 if (IS_ERR(info->args)) {
2190 err = PTR_ERR(info->args);
2191 goto free_hdr;
2192 }
2193
2186 info->hdr = hdr; 2194 info->hdr = hdr;
2187 info->len = len; 2195 info->len = len;
2188 return 0; 2196 return 0;
@@ -2192,6 +2200,12 @@ free_hdr:
2192 return err; 2200 return err;
2193} 2201}
2194 2202
2203static void free_copy(struct load_info *info)
2204{
2205 kfree(info->args);
2206 vfree(info->hdr);
2207}
2208
2195static int rewrite_section_headers(struct load_info *info) 2209static int rewrite_section_headers(struct load_info *info)
2196{ 2210{
2197 unsigned int i; 2211 unsigned int i;
@@ -2385,9 +2399,9 @@ static void find_module_sections(struct module *mod, Elf_Ehdr *hdr,
2385 mod->name); 2399 mod->name);
2386} 2400}
2387 2401
2388static struct module *move_module(struct module *mod, 2402static int move_module(struct module *mod,
2389 Elf_Ehdr *hdr, Elf_Shdr *sechdrs, 2403 Elf_Ehdr *hdr, Elf_Shdr *sechdrs,
2390 const char *secstrings, unsigned modindex) 2404 const char *secstrings, unsigned modindex)
2391{ 2405{
2392 int i; 2406 int i;
2393 void *ptr; 2407 void *ptr;
@@ -2401,7 +2415,7 @@ static struct module *move_module(struct module *mod,
2401 */ 2415 */
2402 kmemleak_not_leak(ptr); 2416 kmemleak_not_leak(ptr);
2403 if (!ptr) 2417 if (!ptr)
2404 return ERR_PTR(-ENOMEM); 2418 return -ENOMEM;
2405 2419
2406 memset(ptr, 0, mod->core_size); 2420 memset(ptr, 0, mod->core_size);
2407 mod->module_core = ptr; 2421 mod->module_core = ptr;
@@ -2416,7 +2430,7 @@ static struct module *move_module(struct module *mod,
2416 kmemleak_ignore(ptr); 2430 kmemleak_ignore(ptr);
2417 if (!ptr && mod->init_size) { 2431 if (!ptr && mod->init_size) {
2418 module_free(mod, mod->module_core); 2432 module_free(mod, mod->module_core);
2419 return ERR_PTR(-ENOMEM); 2433 return -ENOMEM;
2420 } 2434 }
2421 memset(ptr, 0, mod->init_size); 2435 memset(ptr, 0, mod->init_size);
2422 mod->module_init = ptr; 2436 mod->module_init = ptr;
@@ -2443,10 +2457,8 @@ static struct module *move_module(struct module *mod,
2443 DEBUGP("\t0x%lx %s\n", 2457 DEBUGP("\t0x%lx %s\n",
2444 sechdrs[i].sh_addr, secstrings + sechdrs[i].sh_name); 2458 sechdrs[i].sh_addr, secstrings + sechdrs[i].sh_name);
2445 } 2459 }
2446 /* Module has been moved. */ 2460
2447 mod = (void *)sechdrs[modindex].sh_addr; 2461 return 0;
2448 kmemleak_load_module(mod, hdr, sechdrs, secstrings);
2449 return mod;
2450} 2462}
2451 2463
2452static int check_module_license_and_versions(struct module *mod, 2464static int check_module_license_and_versions(struct module *mod,
@@ -2503,87 +2515,107 @@ static void flush_module_icache(const struct module *mod)
2503 set_fs(old_fs); 2515 set_fs(old_fs);
2504} 2516}
2505 2517
2506/* Allocate and load the module: note that size of section 0 is always 2518static struct module *layout_and_allocate(struct load_info *info)
2507 zero, and we rely on this for optional sections. */
2508static noinline struct module *load_module(void __user *umod,
2509 unsigned long len,
2510 const char __user *uargs)
2511{ 2519{
2512 struct load_info info = { NULL, }; 2520 /* Module within temporary copy. */
2513 struct module *mod; 2521 struct module *mod;
2514 long err; 2522 int err;
2515 unsigned long symoffs, stroffs, *strmap;
2516 void __percpu *percpu;
2517 struct _ddebug *debug = NULL;
2518 unsigned int num_debug = 0;
2519 2523
2520 DEBUGP("load_module: umod=%p, len=%lu, uargs=%p\n", 2524 mod = setup_load_info(info);
2521 umod, len, uargs); 2525 if (IS_ERR(mod))
2526 return mod;
2522 2527
2523 err = copy_and_check(&info, umod, len); 2528 err = check_modinfo(mod, info->sechdrs, info->index.info, info->index.vers);
2524 if (err) 2529 if (err)
2525 return ERR_PTR(err); 2530 return ERR_PTR(err);
2526 2531
2527 mod = setup_load_info(&info);
2528 if (IS_ERR(mod)) {
2529 err = PTR_ERR(mod);
2530 goto free_hdr;
2531 }
2532
2533 err = check_modinfo(mod, info.sechdrs, info.index.info, info.index.vers);
2534 if (err)
2535 goto free_hdr;
2536
2537 /* Now copy in args */
2538 info.args = strndup_user(uargs, ~0UL >> 1);
2539 if (IS_ERR(info.args)) {
2540 err = PTR_ERR(info.args);
2541 goto free_hdr;
2542 }
2543
2544 strmap = kzalloc(BITS_TO_LONGS(info.sechdrs[info.index.str].sh_size)
2545 * sizeof(long), GFP_KERNEL);
2546 if (!strmap) {
2547 err = -ENOMEM;
2548 goto free_mod;
2549 }
2550
2551 mod->state = MODULE_STATE_COMING;
2552
2553 /* Allow arches to frob section contents and sizes. */ 2532 /* Allow arches to frob section contents and sizes. */
2554 err = module_frob_arch_sections(info.hdr, info.sechdrs, info.secstrings, mod); 2533 err = module_frob_arch_sections(info->hdr, info->sechdrs, info->secstrings, mod);
2555 if (err < 0) 2534 if (err < 0)
2556 goto free_mod; 2535 goto free_args;
2557 2536
2558 if (info.index.pcpu) { 2537 if (info->index.pcpu) {
2559 /* We have a special allocation for this section. */ 2538 /* We have a special allocation for this section. */
2560 err = percpu_modalloc(mod, info.sechdrs[info.index.pcpu].sh_size, 2539 err = percpu_modalloc(mod, info->sechdrs[info->index.pcpu].sh_size,
2561 info.sechdrs[info.index.pcpu].sh_addralign); 2540 info->sechdrs[info->index.pcpu].sh_addralign);
2562 if (err) 2541 if (err)
2563 goto free_mod; 2542 goto free_args;
2564 info.sechdrs[info.index.pcpu].sh_flags &= ~(unsigned long)SHF_ALLOC; 2543 info->sechdrs[info->index.pcpu].sh_flags &= ~(unsigned long)SHF_ALLOC;
2565 } 2544 }
2566 /* Keep this around for failure path. */
2567 percpu = mod_percpu(mod);
2568 2545
2569 /* Determine total sizes, and put offsets in sh_entsize. For now 2546 /* Determine total sizes, and put offsets in sh_entsize. For now
2570 this is done generically; there doesn't appear to be any 2547 this is done generically; there doesn't appear to be any
2571 special cases for the architectures. */ 2548 special cases for the architectures. */
2572 layout_sections(mod, info.hdr, info.sechdrs, info.secstrings); 2549 layout_sections(mod, info->hdr, info->sechdrs, info->secstrings);
2573 symoffs = layout_symtab(mod, info.sechdrs, info.index.sym, info.index.str, info.hdr, 2550
2574 info.secstrings, &stroffs, strmap); 2551 info->strmap = kzalloc(BITS_TO_LONGS(info->sechdrs[info->index.str].sh_size)
2552 * sizeof(long), GFP_KERNEL);
2553 if (!info->strmap) {
2554 err = -ENOMEM;
2555 goto free_percpu;
2556 }
2557 info->symoffs = layout_symtab(mod, info->sechdrs, info->index.sym, info->index.str, info->hdr,
2558 info->secstrings, &info->stroffs, info->strmap);
2575 2559
2576 /* Allocate and move to the final place */ 2560 /* Allocate and move to the final place */
2577 mod = move_module(mod, info.hdr, info.sechdrs, info.secstrings, info.index.mod); 2561 err = move_module(mod, info->hdr, info->sechdrs, info->secstrings, info->index.mod);
2562 if (err)
2563 goto free_strmap;
2564
2565 /* Module has been copied to its final place now: return it. */
2566 mod = (void *)info->sechdrs[info->index.mod].sh_addr;
2567 kmemleak_load_module(mod, info->hdr, info->sechdrs, info->secstrings);
2568 return mod;
2569
2570free_strmap:
2571 kfree(info->strmap);
2572free_percpu:
2573 percpu_modfree(mod);
2574free_args:
2575 kfree(info->args);
2576 return ERR_PTR(err);
2577}
2578
2579/* mod is no longer valid after this! */
2580static void module_deallocate(struct module *mod, struct load_info *info)
2581{
2582 kfree(info->strmap);
2583 percpu_modfree(mod);
2584 module_free(mod, mod->module_init);
2585 module_free(mod, mod->module_core);
2586}
2587
2588/* Allocate and load the module: note that size of section 0 is always
2589 zero, and we rely on this for optional sections. */
2590static noinline struct module *load_module(void __user *umod,
2591 unsigned long len,
2592 const char __user *uargs)
2593{
2594 struct load_info info = { NULL, };
2595 struct module *mod;
2596 long err;
2597 struct _ddebug *debug = NULL;
2598 unsigned int num_debug = 0;
2599
2600 DEBUGP("load_module: umod=%p, len=%lu, uargs=%p\n",
2601 umod, len, uargs);
2602
2603 /* Copy in the blobs from userspace, check they are vaguely sane. */
2604 err = copy_and_check(&info, umod, len, uargs);
2605 if (err)
2606 return ERR_PTR(err);
2607
2608 /* Figure out module layout, and allocate all the memory. */
2609 mod = layout_and_allocate(&info);
2578 if (IS_ERR(mod)) { 2610 if (IS_ERR(mod)) {
2579 err = PTR_ERR(mod); 2611 err = PTR_ERR(mod);
2580 goto free_percpu; 2612 goto free_copy;
2581 } 2613 }
2582 2614
2583 /* Now we've moved module, initialize linked lists, etc. */ 2615 /* Now we've moved module, initialize linked lists, etc. */
2584 err = module_unload_init(mod); 2616 err = module_unload_init(mod);
2585 if (err) 2617 if (err)
2586 goto free_init; 2618 goto free_module;
2587 2619
2588 /* Now we've got everything in the final locations, we can 2620 /* Now we've got everything in the final locations, we can
2589 * find optional sections. */ 2621 * find optional sections. */
@@ -2600,11 +2632,11 @@ static noinline struct module *load_module(void __user *umod,
2600 err = simplify_symbols(info.sechdrs, info.index.sym, info.strtab, info.index.vers, info.index.pcpu, 2632 err = simplify_symbols(info.sechdrs, info.index.sym, info.strtab, info.index.vers, info.index.pcpu,
2601 mod); 2633 mod);
2602 if (err < 0) 2634 if (err < 0)
2603 goto cleanup; 2635 goto free_modinfo;
2604 2636
2605 err = apply_relocations(mod, info.hdr, info.sechdrs, info.index.sym, info.index.str); 2637 err = apply_relocations(mod, info.hdr, info.sechdrs, info.index.sym, info.index.str);
2606 if (err < 0) 2638 if (err < 0)
2607 goto cleanup; 2639 goto free_modinfo;
2608 2640
2609 /* Set up and sort exception table */ 2641 /* Set up and sort exception table */
2610 mod->extable = section_objs(info.hdr, info.sechdrs, info.secstrings, "__ex_table", 2642 mod->extable = section_objs(info.hdr, info.sechdrs, info.secstrings, "__ex_table",
@@ -2615,9 +2647,7 @@ static noinline struct module *load_module(void __user *umod,
2615 percpu_modcopy(mod, (void *)info.sechdrs[info.index.pcpu].sh_addr, 2647 percpu_modcopy(mod, (void *)info.sechdrs[info.index.pcpu].sh_addr,
2616 info.sechdrs[info.index.pcpu].sh_size); 2648 info.sechdrs[info.index.pcpu].sh_size);
2617 2649
2618 add_kallsyms(mod, &info, symoffs, stroffs, strmap); 2650 add_kallsyms(mod, &info);
2619 kfree(strmap);
2620 strmap = NULL;
2621 2651
2622 if (!mod->taints) 2652 if (!mod->taints)
2623 debug = section_objs(info.hdr, info.sechdrs, info.secstrings, "__verbose", 2653 debug = section_objs(info.hdr, info.sechdrs, info.secstrings, "__verbose",
@@ -2625,12 +2655,14 @@ static noinline struct module *load_module(void __user *umod,
2625 2655
2626 err = module_finalize(info.hdr, info.sechdrs, mod); 2656 err = module_finalize(info.hdr, info.sechdrs, mod);
2627 if (err < 0) 2657 if (err < 0)
2628 goto cleanup; 2658 goto free_modinfo;
2629 2659
2630 flush_module_icache(mod); 2660 flush_module_icache(mod);
2631 2661
2632 mod->args = info.args; 2662 mod->args = info.args;
2633 2663
2664 mod->state = MODULE_STATE_COMING;
2665
2634 /* Now sew it into the lists so we can get lockdep and oops 2666 /* Now sew it into the lists so we can get lockdep and oops
2635 * info during argument parsing. Noone should access us, since 2667 * info during argument parsing. Noone should access us, since
2636 * strong_try_module_get() will fail. 2668 * strong_try_module_get() will fail.
@@ -2666,8 +2698,9 @@ static noinline struct module *load_module(void __user *umod,
2666 add_sect_attrs(mod, info.hdr->e_shnum, info.secstrings, info.sechdrs); 2698 add_sect_attrs(mod, info.hdr->e_shnum, info.secstrings, info.sechdrs);
2667 add_notes_attrs(mod, info.hdr->e_shnum, info.secstrings, info.sechdrs); 2699 add_notes_attrs(mod, info.hdr->e_shnum, info.secstrings, info.sechdrs);
2668 2700
2669 /* Get rid of temporary copy */ 2701 /* Get rid of temporary copy and strmap. */
2670 vfree(info.hdr); 2702 kfree(info.strmap);
2703 free_copy(&info);
2671 2704
2672 trace_module_load(mod); 2705 trace_module_load(mod);
2673 2706
@@ -2684,21 +2717,14 @@ static noinline struct module *load_module(void __user *umod,
2684 mutex_unlock(&module_mutex); 2717 mutex_unlock(&module_mutex);
2685 synchronize_sched(); 2718 synchronize_sched();
2686 module_arch_cleanup(mod); 2719 module_arch_cleanup(mod);
2687 cleanup: 2720 free_modinfo:
2688 free_modinfo(mod); 2721 free_modinfo(mod);
2689 free_unload: 2722 free_unload:
2690 module_unload_free(mod); 2723 module_unload_free(mod);
2691 free_init: 2724 free_module:
2692 module_free(mod, mod->module_init); 2725 module_deallocate(mod, &info);
2693 module_free(mod, mod->module_core); 2726 free_copy:
2694 /* mod will be freed with core. Don't access it beyond this line! */ 2727 free_copy(&info);
2695 free_percpu:
2696 free_percpu(percpu);
2697 free_mod:
2698 kfree(info.args);
2699 kfree(strmap);
2700 free_hdr:
2701 vfree(info.hdr);
2702 return ERR_PTR(err); 2728 return ERR_PTR(err);
2703} 2729}
2704 2730