最近在调试电视盒子的时候遇到了这样一种情况:
盒子在开机的时候出现开机LOGO已经过去了,HDMI才打开,因此每次开机都看不到LOGO,于是就想怎样能调整Linux内核驱动的初始化顺序,仔细阅读代码发现:
Linux内核为不同驱动的加载顺序对应不同的优先级,定义了一些宏:
include/linux/init.h
#define pure_initcall(fn) __define_initcall("0",fn,0)
#define core_initcall(fn) __define_initcall("1",fn,1)
#define core_initcall_sync(fn) __define_initcall("1s",fn,1s)
#define postcore_initcall(fn) __define_initcall("2",fn,2)
#define postcore_initcall_sync(fn) __define_initcall("2s",fn,2s)
#define arch_initcall(fn) __define_initcall("3",fn,3)
#define arch_initcall_sync(fn) __define_initcall("3s",fn,3s)
#define subsys_initcall(fn) __define_initcall("4",fn,4)
#define subsys_initcall_sync(fn) __define_initcall("4s",fn,4s)
#define fs_initcall(fn) __define_initcall("5",fn,5)
#define fs_initcall_sync(fn) __define_initcall("5s",fn,5s)
#define rootfs_initcall(fn) __define_initcall("rootfs",fn,rootfs)
#define device_initcall(fn) __define_initcall("6",fn,6)
#define device_initcall_sync(fn) __define_initcall("6s",fn,6s)
#define late_initcall(fn) __define_initcall("7",fn,7)
#define late_initcall_sync(fn) __define_initcall("7s",fn,7s)
#define __initcall(fn) device_initcall(fn)
其实,我们写驱动的时候用到的module_init对应的是:
#define module_init(x) __initcall(x),而
#define __initcall(fn) device_initcall(fn)
所以,驱动对应的加载的优先级为6。在上面的不同的优先级中,数字越小,优先级越高。同一等级的优先级驱动,加载顺序是由链接过程决定的,结果是不确定的,我们无法去手动设置谁先谁后。但不同等级的驱动加载的顺序是先优先级高,后优先级低,这是可以确定的。
因此,我把本来优先级为module_init(x)的HDMI驱动改为fs_initcall(fn),发现HDMI在开机的时候很早就打开了,可能是因为HDMI显示必须的其他驱动还没初始给完成吧,虽然HDMI打开了,但不能正常显示,于是又试着把优先级设为fs_initcall_sync(fn),发现这下竟然可以了!
虽然问题基本解决,也知道Linux的驱动可以这样操作,但对其中的原理还是有待深入理解、分析!