chishanmingshen 发表于 2015-08-27 13:55

关于__kcrctab section的疑问。。。

本帖最后由 chishanmingshen 于 2015-08-27 14:09 编辑

这个section是在哪里填充的?貌似是modpost.c?
谢谢!

nswcfd 发表于 2015-08-31 20:53

应该是的。

参考scripts/mod/modpost.c::handle_modversions

chishanmingshen 发表于 2015-09-01 09:44

回复 2# nswcfd


    那个函数是填充__versions section的,不是__kcrctab section啊。

nswcfd 发表于 2015-09-01 15:34

本帖最后由 nswcfd 于 2015-09-01 15:37 编辑

对于EXPORT_SYMBOL导出的符号,比如int x, EXPORT_SYMBOL生成以下信息extern int x;
extern void *__crc_x;                                                   // weak symbol
static const unsigned long __kcrctab_x = &__crc_x;                      // @ __kcrctab
static const char __kstrtab_x[] = "x";                                  // @ __ksymtab_strings
static const struct kernel_symbol __ksymtab_x = { &x, __kstrtab_x };    // @ __ksymtab__kcrctab section的内容(就是__crc_变量的地址),在编译的时候就有了(不过这时候是weak symbol)。
只是__crc_x这些变量还没有被定义,它们是通过scripts/genksyms/genksyms程序生成的。
例如:
$ printf "int x; void y(int i) { }; EXPORT_SYMBOL(x); EXPORT_SYMBOL(y); " | ./genksyms/genksyms
__crc_x = 0xb8a39400 ;
__crc_y = 0xcad00b48 ;

genksyms是作为rule_cc_o_c的一部分,在生成.o的过程中被调用的(scripts/Makefile.build)#define rule_cc_o_c
        ...
        $(cmd_modversions)
        ..
#enddef
cmd_mod_versions = if objdump -h .tmp_$(@F) | grep -q __ksymtab; \
        then \
                #see cmd_gensymtypes \
                cpp -D__GENKSYMS $< | genksyms -a $(ARCH) -r /dev/null > .tmp_$(@F:.o=.ver) \
                ld -r -o $@ .tmp_$(@F) -T .tmp_$(@F:.o=.ver) \
        fi如果当前模块是xyz.ko,那么$@=xyz.o $<=xyz.c,上述命令就是#gcc -c -o .tmp_xyz.o xyz.c
if objdump -h .tmp_xyz.o | grep -q __ksysmtab; then
        cpp -D__GENKEYSYMS xyz.c | genksyms -a x86_64 -r /dev/null > .tmp_xyz.ver
        ld -r -o xyz.o .tmp_xyz.o -T .tmp_xyz.ver
fi关键一步是最后的ld,把weak symbol转换为abs value。

chishanmingshen 发表于 2015-09-05 22:40

回复 4# nswcfd

谢谢!确实如此!:lol:

看来还是得好好搞清楚kernel makefile。。。
   
页: [1]
查看完整版本: 关于__kcrctab section的疑问。。。