内核源码之Kconfig和Makefile

转自:http://www.cnblogs.com/image-eye/archive/2011/08/28/2156005.html

内核源码之Kconfig和Makefile

Linux内核源码树的每个目录下都有两个文档KconfigMakefile。分布到各目录的Kconfig构成了一个分布式的内核配置数据库,每个Kconfig分别描述了所属目录源文档相关的内核配置菜单。在执行内核配置make menuconfig时,从Kconfig中读出菜单,用户选择后保存到.config的内核配置文档中。在内核编译时,主Makefile调用这个.config,就知道了用户的选择。这个内容说明了,Kconfig就是对应着内核的每级配置菜单。
假如要想添加新的驱动到内核的源码中,要修改Kconfig,这样就能够选择这个驱动,假如想使这个驱动被编译,则要修改Makefile。添加新的驱动时需要修改的文档有两种(如果添加的只是文件,则只需修改当前层Kconfig和Makefile文件;如果添加的是目录,则需修改当前层和目录下的共一对Kconfig和Makefile)Kconfig和Makefile。要想知道怎么修改这两种文档,就要知道两种文档的语法结构。

Kconfig

Kconfig:每个菜单都有一个关键字标识,最常见的就是config。语法:config symbol,是个新的标记的菜单项,options是在这个新的菜单项下的属性和选项。

关于kconfig的语法,在内核的开发文档中有详细的介绍,具体可以参Documentationkbuildkconfig-language.txt

1,每个config菜单项都要有类型定义,bool布尔类型、 tristate三态:内建、模块、移除。bool类型的只能选中或不选中,tristate类型的菜单项多了编译成内核模块的选项,假如选择编译成内核模块,则会在.config中生成一个CONFIG_HELLO_MODULE=m的配置;假如选择内建,就是直接编译成内核映像,就会在.config中生成一个CONFIG_HELLO_MODULE=y的配置。

2,依赖型定义depends on或requires,指此菜单的出现和否依赖于另一个定义

config HELLO_MODULE

bool "hello test module"

depends on ARCH_PXA

这个例子表明HELLO_MODULE这个菜单项只对XScale处理器有效。

此外可通过select,反向依赖,即选中此项时,同时选中select后面定义的那一项。

3,帮助性定义

只是增加帮助用关键字help或---help---。

举一个完整实例如下,例如添加一个I2C芯片:

config QL_VEE

tristate "QL Visual Enhancement Engine (VEE)"

default y

depends on I2C && EXPERIMENTAL


help

QL Visual Enhancement Engine (VEE) v1.0 with I2C-Compatible Interface and 24-Bit RGB Support Rev.

Makefile

Makefile:内核的Makefile分为5个组成部分:

(1)Makefile     最顶层的Makefile

(2).config        内核的当前配置文档,编译时成为顶层Makefile的一部分

(3)arch/$(ARCH)/Makefile    和体系结构相关的Makefile

(4)Makefile.*      一些特定Makefile的规则

(5)kbuild级别Makefile     

各级目录下的大概约500个文档,编译时根据上层Makefile传下来的宏定义和其他编译规则,将源代码编译成模块或编入内核。顶层的Makefile文档读取.config文档的内容,并总体上负责build内核和模块。Arch Makefile则提供补充体系结构相关的信息。其中.config的内容是在make menuconfig的时候,通过Kconfig文档配置的结果。

各目录下Makefile根据配置选项编译成模块或编入内核,编译方式可分为:单模块单文件,单模块多文件,多模块多文件。

下以编译成模块为例说明:

单模块单文件:obj-m := hello.o

单模块多文件:obj-m := myhello.o     myhello-objs := hello.o goo.o

多模块多文件:obj-m := myhello1.o myhello2.o  

               myhello1-objs := hello1.o goo1.o

               myhello2-objs := hello2.o goo2.o

具体可参考:driver makefile 文档

obj-y += foo.o: 把由foo.c或者foo.s文件编译得到的foo.o,并连接进内核
obj-m += foo.o: 表示该文件作为模块编译
除了y,m以外的obj-x形式的目标则不会被编译.

foo-objs表示多文件编译。

添加示例
假设想把自己写的一个flash的驱动程式加载到工程中,而且能够通过menuconfig配置内核时选择该驱动该怎么办呢?如下:

第一:将您写的flashtest.c 文档添加到/driver/mtd/maps/ 目录下。

第二:修改/driver/mtd/maps目录下的kconfig文档:

config MTD_flashtest

tristate “ap71 flash"
这样当make menuconfig时 ,将会出现 ap71 flash选项。

第三:修改该目录下makefile文档。添加如下内容:obj-$(CONFIG_MTD_flashtest)       += flashtest.o

这样,当您运行make menucofnig时,您将发现ap71 flash选项,假如您选择了此项。该选择就会保存在.config文档中。当您编译内核时,将会读取.config文档,当发现ap71 flash 选项为yes,系统在调用/driver/mtd/maps/下的makefile 时,将会把 flashtest.o 加入到内核中。

参考:

    1. http://www.cnblogs.com/image-eye/archive/2011/08/28/2156005.html
    2. driver makefile
原文地址:https://www.cnblogs.com/embedded-linux/p/5805923.html