1.Makefile的作用


  (1)决定编译哪些文件

  (2)怎样编译这些文件

  (3)怎样连接这些文件,最重要的是它们的顺序如何

2.Linux内核Makefile分类

*********************************************************************

  顶层Makefile:它是所有Makefile文件的核心,从总体上控制着内核的编译、

               连接

  .config    :配置文件,在配置内核时生成,所有Makefile文件(包括顶层

              目录及各级子目录)都是根据.config来决定使用哪些文件

arch/$(ARCH)/Makefile:对应于体系结构的Makefile,它用来决定哪些体系结

                      构相关的文件参与内核的生成,并提供一些规则来生成

                      特定格式的内核映像

scripts/Makefile.* : Makefile共用的通用规则、脚本等

kbuild Makefile :各级子目录下的Makefile,被上一层Makefile调用来编译

                 当前目录下的文件

*********************************************************************

3.根据Makefile的作用分析这5类文件

(1)决定编译哪些文件

  Linux内核的编译过程从顶层Makefile开始,然后递归地进入各级子目录调用它们的Makefile,分为3个步骤:

   (1)顶层Makefile决定内核根目录下哪些子目录将被编译进内核

       在顶层Makefile中有如下内容:

        init-y       := init/

        drivers-y := drivers/ sound/ firmware/

        net-y     := net/

        libs-y       := lib/

        core-y       := usr/

         ......................

         core-y      += kernel/ mm/ fs/ ipc/ security/ crypto/ block/

      顶层Makefile将这13个子目录分为5类,除去include目录和后面不包

   含内核源码的目录外,还有一个arch目录在arch/$(ARCH)/Makefile中被包

   含进内核,在顶层Makefile中直接包含了这个Makefile,如下所示:

           452   include $(srctree)/arch/$(SRCARCH)/Makefile

   (2)arch/$(ARCH)/Makefile决定arch/$(ARCH)目录下哪些文件、哪些目录将

      被编译进内核

        在arch/arm/Makefile中有如下内容:

       100   head-y      := arch/arm/kernel/head$(MMUEXT).o

                                        arch/arm/kernel/init_task.o

            ...............................................

       196   core-y  += arch/arm/kernel/ arch/arm/mm/ arch/arm/common/

       197   core-y  += $(machdirs) $(platdirs)

       198   core-$(CONFIG_FPE_NWFPE)  += arch/arm/nwfpe/

       199   core-$(CONFIG_FPE_FASTFPE)   += $(FASTFPE_OBJ)

       200   core-$(CONFIG_VFP)    += arch/arm/vfp/

             ...............................................

       204   libs-y    := arch/arm/lib/ $(libs-y)

    

       第100行中MMUEXT在arch/arm/Makefile前面定义,对于没有MMU的处

    理器,使用文件head-nommu.S;对于有MMU的处理器,使用head.S文件

       CONFIG_XXXX在配置内核时定义,它的值有3种:y、m或空

       编译内核时将依次进入init-y、core-y、libs-y、drivers-y和net-y 

    所列出的目录中执行它们的Makefile,每个子目录都会生成一个

    built-in.o(libs-y所列的目录下,有可能生成lib.a文件)。最后,head-y

    所表示的文件将和这些built-in.o、lib.a一起被连接成内核映像文件 

    vmlinux

   (3)各级子目录下的Makefile决定所在目录下哪些文件将被编译进内核,哪

      些文件将被编译成模块(驱动程序),进入哪些子目录继续调用它们的

      Makefile

          在配置内核时,生成配置文件“.config”,根据.config中定义的各

      个变量决定编译哪些文件,Makefile使用如下语句间接包含.config文

      件(因为包含的是"include/config/auto.conf"文件):

               486   -include include/config/auto.conf

          在"include/config/auto.conf"文件中,变量的值为y或m

          1)obj-y用来定义哪些文件被编进(built-in)内核

              obj-y中定义的.o文件由当前目录下的.c或.S文件编译生成,

           它们连同下级子目录的built-in.o文件一起被组合成(使用

          “$(LD)-R”命令)当前目录下的built-in.o文件,这个built-in.o

            文件被它的上一层Makefile使用

          2)obj-m用来定义哪些文件被编译成可加载模块(Loadable module)

              obj-m中定义的.o文件由当前目录下的.c或.S文件编译生成,

           它们被编译成模块,一个模块可以由一个或几个.o文件组成,对于

           有多个源文件的模块,除在obj-m中增加一个.o文件外,还要定义

           一个<module_name>-objs变量来告诉Makefile这个.o文件由哪些  

           文件组成

               例如:

              obj-$(CONFIG_ISDN) += isdn.o

              isdn-objs := isdn_net_lib.o  isdn_v110.o  isdn_common.o

           3)lib-y用来定义哪些文件被编成库文件

               liy-y中定义的.o文件由当前目录下的.c或.S文件编译生成,

            它们被打包成当前目录下的一个库文件:lib.a ;同时出现在

            obj-y和lib-y中的.o文件,不会被包含进lib.a中,要把这个

            lib.a编译进内核中,需要在顶层Makefile中libs-y变量中列出

            当前目录

            4)obj-y、obj-m还可以指定要进入的下一层子目录

                obj-$(CONFIG_JFFS2_FS)   +=  jffs2/

(2)怎样编译这些文件----编译选项、连接选项是什么

    这些选项可分为3类:

     1)全局的

        适用于整个内核代码,在顶层Makefile和arch/$(ARCH)/Makefile中

     定义,这些选项的名称为CFLAGS(编译C文件的 选项)、AFLAGS(编译汇编

    文件的选项)、LDFLAGS(连接文件的选项)、 ARFLAGS(制作库文件的选项)

      2)局部的

          在各个子目录中定义,针对当前Makefile中的所有文件,名称分别

      为:EXTRA_CFLAGS、EXTRA_AFLAGS、EXTRA_LDFLAGS、EXTRA_ARFLAGS

      3)个体的

          仅适用于某个文件,如果想针对某个文件定义它的编译选项,可以使

      用CFLAGS_$@, AFLAGS_$@,前者用于编译某个C文件,后者用于编译某

      个汇编文件。$@表示某个目录文件

  (3)怎样连接这些文件,它们的顺序如何

     在顶层Makefile中,目录名的后面直接加上built-in.o或lib.a,表示

   要连接进内核的文件,如下所示:

         init-y      := $(patsubst %/, %/built-in.o, $(init-y))

         core-y      := $(patsubst %/, %/built-in.o, $(core-y))

         drivers-y   := $(patsubst %/, %/built-in.o, $(drivers-y))

         net-y       := $(patsubst %/, %/built-in.o, $(net-y))

         libs-y1     := $(patsubst %/, %/lib.a, $(libs-y))

         libs-y2     := $(patsubst %/, %/built-in.o, $(libs-y))

         libs-y      := $(libs-y1) $(libs-y2)

   patsubst是个字符串处理函数,它的用法如下:    

         $( patsubst  pattern, replacement, text)

      表示寻找“text”中符合格式“pattern”的字,用“replacement”代替

   它们,比如init-y的初值为“init/”,经过变换后变为“init/built-in.o”

       再如下所示:

          vmlinux-init := $(head-y) $(init-y)

          vmlinux-main := $(core-y) $(libs-y) $(drivers-y) $(net-y)

          vmlinux-all  := $(vmlinux-init) $(vmlinux-main)

          vmlinux-lds  := arch/$(SRCARCH)/kernel/vmlinux.lds

       vmlinux-all表示所有构成映象文件vmlinux的目标文件,可知这些目标文件的顺序为head-y、init-y、core-y、libs-y、drivers-y、net-y

4.总结

(1)配置文件.config中定义了一系列的变量,Makefile将结合它们来决定哪些

   文件被编译进内核、哪些文件被编译成模块、涉及哪些子目录

(2)顶层Makefile和arch/$(ARCH)/Makefile决定根目录下哪些子目录、

   arch/$(ARCH)目录下哪些文件和目录将被编译进内核

(3)各级子目录下的Makefile决定所在目录下的哪些文件将被编译进内核,哪些

   文件被编译成模块(即驱动程序),进入哪些子目录继续调用它们的Makefile

(4)顶层Makefile和arch/$(ARCH)/Makefile设置了可以影响所有文件的编译、

   连接选项,即全局选项

(5)各级子目录下的Makefile中可以设置影响当前目录下所有文件的编译、连接

   选项,即局部选项

(6)顶层Makefile按照一定的顺序组织文件,根据连接脚本arch/$(ARCH)

   kernel/vmlinux.lds生成内核映象文件vmlinux