内核映像的形成
前面提到了,grub载入内核vmlinuz并解压缩到指定位置。别看这小小一句话,里边的学问可大着哩,要搞懂这里面的内幕,我们还要从内核编译说起。
我们知道,kernel主要是由C语言和汇编语言交叉编译而形成的,其源代码来自网上,网址为www.kernel.org。当你随便下一个内核版本并解压后,就可以看到里面的文件,要么是C代码(.c或.h文件),要么就是汇编代码(.S文件)。这些文件都是内核代码的源文件,如果只有这些源文件的话,你没法把它拿到机器上去运行,因为我们的PC或服务器只认识二进制机器代码,所以,其中的编译过程就很重要的。所以,我们看到,除了这些源文件,内核源文件包里还有些没有后缀名的文件和一些shell脚本文件。这些都是辅助编译的文件,我们称之为内核代码辅助文件。
gcc要把源文件编译成二进制代码,就需要这些辅助文件的帮助。下面我们就来说说整个过程。不过,直接列整个过程的意义并不大,所以我想把我的研究思路列出来,也许我的思路是错误的,也请大家不吝指正。
内核编译的步骤在网上已经出现得很多了,主要是以下四步:
1、make menuconfig:配置编译选项
2、make:编译及链接源代码,主要是些核心程序
3、make modules:编译及链接模块代码,主要是些驱动程序
4、make install:制作vmlinuz
2.1 MakeFile预备知识
我们分析内核的主要任务是学习相关知识,前面我们学习了BIOS即系统启动相关的知识,这里学习一下GUN make的知识。
在shell的提示符号下,若输入“make xxxx”,则它会到目前的目录下找寻Makefile这个文件。然后依照Makefile中所记录的步骤一步一步的来执行。在我们写程序的时候,如果事先就把compiler程式所需要的步骤先写在Makefile中的话,想要compiler程序的时候就只要打入make的指令。
在我们的Linux内核项目文件中,有成百上千个源文件,每次修改其中的一个都需要全部重新编译是不可想象的事情,但通过编辑Makefile文件,利用make命令就可以只针对其中修改的源文件进行编译,而不需要全体编译。这就是make命令在编译项目文件时体现出来的优势。能做到这点,主要是基于Makefile文件的编写,和make命令对Makefile文件的调用。Makefile文件作为make命令的默认参数,使一个基于依赖关系编写的结构文件。
2.1.1 Makefile书写规则
大家经常看到使用make all、make install、make clean等命令,包括我们这里配置编译选项所使用的make menuconfig,而他们处理的目标都是一个Makefile文件,那么all、install、clean这些跟在make后面的参数是如何调用Makefile文件的运行呢?在这里,如果向上面的命令如果能够正确运行的话,那么在Makefile文件里一定有这样的几行,他们的以all、install、clean开始,我们叫做目标(Target):
all:xxxx
xxxxxxxxxxxxx
install:xxxx
xxxxxxxxxxxxx
clean:xxxx
xxxxxxxxxxxxx
Makefile文件的一般以以下方式组成:
target ... : prerequisites ...
command
...
...
target 也就是一个目标文件,可以是Object File,也可以是执行文件。还可以是一个标签(Label),对于标签这种特性,这个就是“伪目标”。
prerequisites 就是,要生成那个target 所需要的文件或是目标。command 也就是make 需要执行的命令。(任意的Shell 命令)
这是一个文件的依赖关系, 也就是说, target 这一个或多个的目标文件依赖于prerequisites 中的文件, 其生成规则定义在command 中。说白一点就是说,prerequisites 中如果有一个以上的文件比target 文件要新的话,command 所定义的命令就会被执行。这就是Makefile 的规则。也就是Makefile 中最核心的内容。有时候一个target可能被包含在一个条件语句分分支中,那么当这个条件不成立时,无论如何这个target都不会被执行。注意,这种情况在我们内核的Makefile中相当普遍。
在默认的方式下,也就是我们只输入 make 命令。那么,
1、 make 会在当前目录下找名字叫“Makefile”或“makefile”的文件。
2、 如果找到,它会找文件中的第一个目标文件(target),并把这个文件作为最终的目标文件。
3、 如果第一个目标文件不存在,或是它所依赖的后面文件的修改时间要比这个文件新,那么,他就会执行后面所定义的命令来生成这个目标文件。
4、 如果第一个目标所依赖的文件也存在,那么make 会在当前文件中找目标为该文件的依赖性,如果找到则再根据那一个规则生成文件。(这有点像一个堆栈的过程)
5、 当然,你的C 文件和H 文件是存在的啦,于是make 会生成 .o 文件,然后再用 .o文件生命make 的终极任务,也就是第一目标所对应的执行文件了。
Makefile书写规则:
targets : prerequisites
command
...
或是这样:
targets : prerequisites ; command
command
...
targets 是文件名,以空格分开,可以使用通配符。一般来说,我们的目标基本上是一个文
件,但也有可能是多个文件。
command 是命令行,如果其不与“target:prerequisites”在一行,那么,必须以[Tab键]开头,如果和prerequisites 在一行,那么可以用分号做为分隔。
prerequisites 也就是目标所依赖的文件(或依赖目标)。如果其中的某个文件要比目标文件要新,那么,目标就被认为是“过时的”,被认为是需要重生成的。这个在前面已经讲过了。
如果命令太长,你可以使用反斜框(‘/’)作为换行符。make 对一行上有多少个字符没有限制。规则告诉make 两件事,文件的依赖关系和如何成成目标文件。一般来说,make 会以UNIX 的标准Shell,也就是/bin/sh 来执行命令。