- 論壇徽章:
- 20
|
本帖最后由 nswcfd 于 2015-09-01 15:37 編輯
對(duì)于EXPORT_SYMBOL導(dǎo)出的符號(hào),比如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
復(fù)制代碼 __kcrctab section的內(nèi)容(就是__crc_變量的地址),在編譯的時(shí)候就有了(不過這時(shí)候是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的過程中被調(dià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
復(fù)制代碼 如果當(dāng)前模塊是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
復(fù)制代碼 關(guān)鍵一步是最后的ld,把weak symbol轉(zhuǎn)換為abs value。 |
|