最近做一個(gè)底層源碼版本管理的工作:
本地調試OK的內核源碼,驅動(dòng)源碼,文件系統等,作為一個(gè)工程上傳至SVN服務(wù)器,以便同事們可以方便共享同步代碼;
1.問(wèn)題:本地編譯的一整套底層代碼down到設備跑都正常,但是由這套代碼上傳SVN服務(wù)器而后checkout出來(lái)的代碼編譯的文件,則出現驅動(dòng)文件加載不上的情況(驅動(dòng)以模塊方式加載),打印如下
log: version magic '3.3.0 preempt mod_unload ARMv5 ' should be '3.3.0-svn87 preempt mod_unload ARMv5 '
2.分析:初步由打印log信息看,是由version magic不匹配造成,找到信息打印點(diǎn),
kernel/module.c
- printk(KERN_ERR"%s: version magic '%s' should be '%s'\n",
- mod->name, modmagic, vermagic);
然后
- static const char vermagic[]= VERMAGIC_STRING;
可知打印語(yǔ)句實(shí)際由宏VERMAGIC_STRING定義,接著(zhù)看include/linux/vermagic.h- #define VERMAGIC_STRING \
- UTS_RELEASE " " \
- MODULE_VERMAGIC_SMP MODULE_VERMAGIC_PREEMPT \
- MODULE_VERMAGIC_MODULE_UNLOAD MODULE_VERMAGIC_MODVERSIONS \
- MODULE_ARCH_VERMAGIC
打印信息的不同點(diǎn)“
'3.3.0”和“
3.3.0-svn87”是由宏
UTS_RELEASE生成,那么重點(diǎn)關(guān)注UTS_RELEASE由名字推測是編譯內核時(shí)版本號給打上了svn相關(guān)標記,但是在內核源碼中沒(méi)看到任何地方定義
UTS_RELEASE,再到Makefile找,搜到這句
- (echo \#define UTS_RELEASE \"$(KERNELRELEASE)\";)
再搜
KERNELRELEASE- # Read KERNELRELEASE from include/config/kernel.release(if it exists)
- 382 KERNELRELEASE = $(shell cat include/config/kernel.release 2>/dev/null)
- 383 KERNELVERSION = $(VERSION)$(if $(PATCHLEVEL),.$(PATCHLEVEL)$(if $(SUBLEVEL),.$(SUBLEVEL)))$(EXTRAVERSION)
可見(jiàn)KERNELRELEASE由include/config/kernel.release文件讀取獲得,但是內核源碼中并無(wú)此文件(為查找差異之前將內核make distclean掉了),再重新編譯
一次,果然生成了include/config/kernel.release文件,查看內容- cat include/config/kernel.release
- 3.3.0-svn87
找到了加載驅動(dòng)時(shí)打印log里的“3.3.0-svn87”
再回到Makefile,搜kernel.release
- # Store (new) KERNELRELASEstring in include/config/kernel.release
- 951 include/config/kernel.release: include/config/auto.conf FORCE
- 952 $(Q)rm-f $@
- 953 $(Q)echo"$(KERNELVERSION)$$($(CONFIG_SHELL) $(srctree)/scripts/setlocalversion $(srctree))"> $@
kernel.release就是在這里生成的,前面版面號忽略,“-svn87”實(shí)際上是執行setlocalversion得到svn版本號加上去的;
查看得知setlocalversion是一個(gè)腳本文件,支持自動(dòng)獲取svn,git等源碼管理工具的版本號;
3.解決:
只需要使version magic一致,驅動(dòng)就可以順利加載;由于廠(chǎng)商提供的部分驅動(dòng)文件沒(méi)提供源碼,并且本地內核也會(huì )不斷升級(svn版本號會(huì )變),所以最簡(jiǎn)單的方法是將
附加的-svnXX去掉,這樣現存的驅動(dòng)和后續編譯的驅動(dòng)和內核就都可以保持一致了:
修改如下,
- include/config/kernel.release: include/config/auto.conf FORCE
- 952 $(Q)rm-f $@
- 953 # $(Q)echo"$(KERNELVERSION)$$($(CONFIG_SHELL) $(srctree)/scripts/setlocalversion $(srctree))"> $@
- 954 $(Q)echo"$(KERNELVERSION)$$($(CONFIG_SHELL) $(srctree))"> $@
重新編譯后,驅動(dòng)可以正常加載工作了。PS:當然,version magic是一個(gè)很好的功能,如果所有驅動(dòng)都有源碼,對于發(fā)布產(chǎn)品的版本,加上svn版本號標記可以有效保證內核驅動(dòng)版本一致性。
本站僅提供存儲服務(wù),所有內容均由用戶(hù)發(fā)布,如發(fā)現有害或侵權內容,請
點(diǎn)擊舉報。