10.从这里开始
USB Core从USB子系统的初始化开始,我们也需要从那里开始,它们位于文件drivers/usb/core/usb.c中:
938 subsys_initcall(usb_init); 939 module_exit(usb_exit);
我们看到一个subsys_initcall,它也是一个宏,我们可以把它理解为module_init,只不过因为这部分代码比较核心,开发人员们把它看做一个子系统,而不仅仅是一个模块。这也很好理解,usbcore这个模块代表的不是某一个设备,而是所有USB设备赖以生存的模块,在Linux中,像这样一个类别的设备驱动被归结为一个子系统。比如PCI子系统、SCSI子系统,基本上,drivers/目录下面第一层的每个目录都算一个子系统,因为它们代表了一类设备。
subsys_initcall(usb_init)的意思就是告诉我们,usb_init是USB子系统真正的初始化函数,而usb_exit()将是整个USB子系统结束时的清理函数,于是我们就从usb_init开始看起。
863 static int __init usb_init(void) 864 { 865 int retval; 866 if (nousb) { 867 pr_info("%s: USB support disabled\n", usbcore_name); 868 return 0; 869 } 870 871 retval = ksuspend_usb_init(); 872 if (retval) 873 goto out; 874 retval = bus_register(&usb_bus_type); 875 if (retval) 876 goto bus_register_failed; 877 retval = usb_host_init(); 878 if (retval) 879 goto host_init_failed; 880 retval = usb_major_init(); 881 if (retval) 882 goto major_init_failed; 883 retval = usb_register(&usbfs_driver); 884 if (retval) 885 goto driver_register_failed; 886 retval = usb_devio_init(); 887 if (retval) 888 goto usb_devio_init_failed; 889 retval = usbfs_init(); 890 if (retval) 891 goto fs_init_failed; 892 retval = usb_hub_init(); 893 if (retval) 894 goto hub_init_failed; 895 retval = usb_register_device_driver(&usb_generic_driver, THIS_MODULE); 896 if (!retval) 897 goto out; 898 899 usb_hub_cleanup(); 900 hub_init_failed: 901 usbfs_cleanup(); 902 fs_init_failed: 903 usb_devio_cleanup(); 904 usb_devio_init_failed: 905 usb_deregister(&usbfs_driver); 906 driver_register_failed: 907 usb_major_cleanup(); 908 major_init_failed: 909 usb_host_cleanup(); 910 host_init_failed: 911 bus_unregister(&usb_bus_type); 912 bus_register_failed: 913 ksuspend_usb_cleanup(); 914 out: 915 return retval; 916 }
首先看863行的__init标记,写过驱动的人应该不会陌生,它对内核来说就是一种暗示,表明这个函数仅在初始化期间使用,在模块被装载之后,它占用的资源就会释放掉用于它处。它的暗示你懂,可你的暗示它却不懂或者懂装不懂,多么让人感伤。它在自己短暂的一生中一直从事繁重的工作,吃的是草,挤出来的是牛奶,留下的是整个USB子系统的繁荣。
受这种精神所感染,我觉得还是有必要为它说得更多一些。__init的定义在include/linux/init.h文件中:
43 #define __init __attribute__ ((__section__ (".init.text")))
好像这里引出了更多的疑问,__attribute__是什么?Linux内核代码使用了大量的GNU C扩展,以至于GNU C成为能够编译内核的唯一编译器,GNU C的这些扩展对代码优化、目标代码布局、安全检查等方面也提供了很强的支持。而__attribute__就是这些扩展中的一个,它主要被用来声明一些特殊的属性,这些属性主要被用来指示编译器进行特定方面的优化和更仔细的代码检查。GNU C支持十几个属性,section是其中的一个,我们查看GCC的手册可以看到下面的描述:
‘section ("section-name")' Normally, the compiler places the code it generates in the `text' section. Sometimes, however, you need additional sections, or you need certain particular functions to appear in special sections. The `section' attribute specifies that a function lives in a particular section. For example, the declaration: extern void foobar (void) __attribute__ ((section ("bar")));
puts the function ‘foobar' in the ‘bar' section. Some file formats do not support arbitrary sections so the ‘section' attribute is not available on all platfor ms. If you need to map the entire contents of a module to a particular section, consider using the facilities of the linker instead.
通常,编译器将函数放在.text节,变量放在.data节或.bss节,使用section属性,可以让编译器将函数或变量放在指定的节中。那么前面对__init的定义便表示将它修饰的代码放在.init.text节。连接器可以把相同节的代码或数据安排在一起,比如__init修饰的所有代码都会被放在.init.text节里,初始化结束后就可以释放这部分内存。
那内核又是如何调用到这些__init修饰的初始化函数?要回答这个问题,还需要回顾第938行的代码,上面已经提到subsys_initcall也是一个宏,它也在include/linux/init.h中定义:
125 #define subsys_initcall(fn) __define_initcall("4",fn,4)
这里又出现了一个宏__define_initcall,它用于将指定的函数指针fn放到initcall.init节里,而对于具体的subsys_initcall宏,则是把fn放到.initcall.init的子节.initcall4.init里。要弄清楚.initcall.init、.init.text和.initcall4.init,我们还需要了解一点内核可执行文件相关的概念。
内核可执行文件由许多链接在一起的对象文件组成。对象文件有许多节,如文本、数据、init数据、bass等。这些对象文件都是由一个称为链接器脚本的文件链接并装入的。这个链接器脚本的功能是将输入对象文件的各节映射到输出文件中。换句话说,它将所有输入对象的文件都链接到单一的可执行文件中,将该可执行文件的各节装入到指定地址处。vmlinux.lds是存在于arch/<target>/ 目录中的内核链接器脚本,它负责链接内核的各个节并将它们装入内存中特定的偏移量处。
笔者可以负责任地告诉你,要看懂vmlinux.lds这个文件是需要花一番工夫的,不过大家都是聪明人,聪明人做聪明事,所以你需要做的只是搜索initcall.init,然后便会看到似曾相识的内容。
__inicall_start = .; .initcall.init : AT(ADDR(.initcall.init) - 0xC0000000) { *(.initcall1.init) *(.initcall2.init) *(.initcall3.init) *(.initcall4.init) *(.initcall5.init) *(.initcall6.init) *(.initcall7.init) } __initcall_end = .;
这里的__initcall_start指向.initcall.init节的开始,__initcall_end指向它的结尾。而.initcall.init节又被分为了7个子节,分别如下。
.initcall1.init .initcall2.init .initcall3.init .initcall4.init .initcall5.init .initcall6.init .initcall7.init
我们的subsys_initcall宏便是将指定的函数指针放在了.initcall4.init子节。其他的比如core_initcall将函数指针放在.initcall1.init子节,device_initcall将函数指针放在了.initcall6.init子节等,都可以从include/linux/init.h文件找到它们的定义。各个子节的顺序是确定的,即先调用.initcall1.init中的函数指针,再调用.initcall2.init中的函数指针等。__init修饰的初始化函数在内核初始化过程中调用的顺序和.initcall.init节里函数指针的顺序有关,不同的初始化函数被放在不同的子节中,因此,也就决定了它们的调用顺序。
至于实际执行函数调用的地方,就在/init/main.c文件中,内核的初始化不在那里还能在哪里?do_initcalls函数会直接用到这里的__initcall_start、__initcall_end来进行判断。不多说了,还是回到久违的usb_init函数吧。