- 论坛徽章:
- 20
|
本帖最后由 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。 |
|