On Mon, Sep 09, 2013 at 10:44:03AM +0930, Rusty Russell wrote: > Kyle McMartin <kyle@xxxxxxxxxxxxx> writes: > > On Fri, Sep 06, 2013 at 07:51:18PM +0200, Frantisek Hrbata wrote: > >> > > v2: - reuse mod->ctors for .init_array section for modules, because gcc uses > >> > > .ctors or .init_array, but not both at the same time > >> > > > >> > > Signed-off-by: Frantisek Hrbata <fhrbata@xxxxxxxxxx> > >> > > >> > Might be nice to document which gcc version changed this, so people can > >> > choose whether to cherry-pick this change? > >> > >> Thank you for pointing this out. As per gcc git this was introduced by commit > >> ef1da80 and released in 4.7 version. > >> > >> $ git describe --contains ef1da80 > >> gcc-4_7_0-release~4358 > >> > >> Do you want me to post v3 with this info included in the descrition? > >> > > > > It actually depends on the combination of binutils/ld and gcc you use, not > > simply which gcc version you use. :/ > > Indeed, and seems it was binutils 20110507 which actually handled it > properly. > > AFAICT it's theoretically possible to have .ctors and .init_array in a > module. Unlikely, but the patch should check for both and refuse to > load the module in that case. Otherwise weird things would happen. I'm not sure if coexistence of .ctors and .init_array sections should result in denial of module, but I for sure know nothing about this :). Could you maybe privide one example of the "weird thing"? Anyway many thanks for taking time to look at this. Below is my attepmt to implement the check you proposed. untested/uncompiled diff --git a/include/asm-generic/vmlinux.lds.h b/include/asm-generic/vmlinux.lds.h index 83e2c31..bc2121f 100644 --- a/include/asm-generic/vmlinux.lds.h +++ b/include/asm-generic/vmlinux.lds.h @@ -473,6 +473,7 @@ #define KERNEL_CTORS() . = ALIGN(8); \ VMLINUX_SYMBOL(__ctors_start) = .; \ *(.ctors) \ + *(.init_array) \ VMLINUX_SYMBOL(__ctors_end) = .; #else #define KERNEL_CTORS() diff --git a/include/linux/module.h b/include/linux/module.h index 05f2447..e775b41 100644 --- a/include/linux/module.h +++ b/include/linux/module.h @@ -380,6 +380,8 @@ struct module /* Constructor functions. */ ctor_fn_t *ctors; unsigned int num_ctors; + ctor_fn_t *init_array; + unsigned int num_init_array; #endif }; #ifndef MODULE_ARCH_INIT diff --git a/kernel/module.c b/kernel/module.c index dc58274..831b92d 100644 --- a/kernel/module.c +++ b/kernel/module.c @@ -2768,6 +2768,9 @@ static void find_module_sections(struct module *mod, struct load_info *info) #ifdef CONFIG_CONSTRUCTORS mod->ctors = section_objs(info, ".ctors", sizeof(*mod->ctors), &mod->num_ctors); + mod->init_array = section_objs(info, ".init_array", + sizeof(*mod->init_array), + &mod->num_init_array); #endif #ifdef CONFIG_TRACEPOINTS @@ -2808,6 +2811,18 @@ static void find_module_sections(struct module *mod, struct load_info *info) sizeof(*info->debug), &info->num_debug); } +static void check_module_sections(struct module *mod) +{ +#ifdef CONFIG_CONSTRUCTORS + if (mod->ctors && mod->init_array) { + pr_err("%s: .ctors and .init_array sections mishmash\n", + mod->name); + return -EINVAL; + } +#endif + return 0; +} + static int move_module(struct module *mod, struct load_info *info) { int i; @@ -3032,6 +3047,9 @@ static void do_mod_ctors(struct module *mod) for (i = 0; i < mod->num_ctors; i++) mod->ctors[i](); + + for (i = 0; i < mod->num_init_array; i++) + mod->init_array[i](); #endif } @@ -3265,6 +3283,10 @@ static int load_module(struct load_info *info, const char __user *uargs, * find optional sections. */ find_module_sections(mod, info); + err = check_module_sections(mod); + if (err) + goto free_unload; + err = check_module_license_and_versions(mod); if (err) goto free_unload; -- 1.8.3.1 > > Cheers, > Rusty. -- Frantisek Hrbata -- To unsubscribe from this list: send the line "unsubscribe linux-arch" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html