Kconfig语法简介

(注:本文节选自正点原子嵌入式Linux开发指南)

    本篇只是大概了解Kconfig原理,关于Kconfig的详细语法介绍,可以参考linux内核源码的文件Documentation/kbuild/kconfig-language.txt。

    打开uboot根目录下的Kconfig,这个Kconfig文件就是顶层Kconfig,我们就以这个文件为例简单学习一下Kconfig语法。

    1、mainmenu

    顾名思义mainmenu就是主菜单,也就是输入“make menuconfig”以后打开的默认界面,在顶层Kconfig中有如下代码:

    

    上述代码就是定义了一个名为“U-Boot  $UBOOTVERSION Configuration”的主菜单,其中UBOOTVERSION = 2016.03,因此主菜单名为“U-Boot 2016.03 Configuration”,如图2所示:

     2、调用其他目录下的Kconfig文件

    和makefile一样,Kconfig也可以调用其他子目录中的Kconfig文件,调用方法如下:

    source    "xxx/Kconfig"    //xxx为具体的目录名,相对路径

    在顶层Kconfig中有如下代码:

 

    从上述代码中可以看出,在顶层Kconfig文件调用了很多其他子目录下的Kconfig文件,这些子目录下的Kconfig文件在主菜单中生成各自的菜单项。

    3、menu/endmenu条目

    menu用于生成菜单,endmenu就是菜单结束标志,这两个一般是成对出现的。在顶层Kconfig中有如下代码:

    

 图3图4展示了一个menu/endmenu代码块的一部分,这个代码块就是一个子菜单,图3中第14行的“menu "General setup"”表示子菜单“General setup”。体现在主菜单界面中就如图5:

 

    在“General setup”菜单上面还有“Architecture select(ARM architecture)”和“ARM architecture”这两个子菜单,但是在顶层Kconfig中并没有看到这两个子菜单对应的menu/endmenu代码块,那这两个子菜单是怎么来的呢?这两个子菜单就是arch/Kconfig文件生成的。包括主界面中的“Boot timing”、“Console recording”等等这些子菜单,都是分别由顶层Kconfig所调用的common/Kconfig、cmd/Kconfig等这些子Kconfig文件来创建的。

    4、config条目

    顶层Kconfig中的“General setup”子菜单内容如下:

     可以看出,在menu/endmenu代码块中有大量的“config  xxxx”的代码块,也就是config条目。config条目就是“General setup”菜单的具体配置项,如图8所示:

    “config  LOCALVERSION”对应着第一个配置项,“config LOCALVERSION_AUTO”对应着第二个配置项,以此类推。我们以“config  LOCALVERSION”和“config LOCALVERSION_AUTO”这两个为例来分析一下config配置项的语法:

     第16行和26行,这两行都以config关键字开头,后面跟着LOCALVERSION和LOCALVERSION_AUTO,这两个就是配置项名字。假如我们使能了LOCALVERSION_AUTO这个功能,那么就会向.config文件中生成CONFIG_LOCALVERSION_AUTO。由此可知,.config文件中的“CONFIG_XXX”(XXX就是具体的配置项名字)就是Kconfig文件中config关键字后面的配置项名字加上"CONFIG_"前缀。

    config关键字下面的这几行是配置项属性,17~24行是LOCALVERSION的属性,27~44行是LOCALVERSION_AUTO属性,属性里面描述了配置项的类型、输入提示、依赖关系、帮助信息和默认值等。

    第17行的string是变量类型,也就是“CONFIG_LOCALVERSION”的变量类型。可以为:bool、tristate、string、hex和int,一共5中。最常用的是bool、tristate和string这三种,bool类型有两种值:y和n,当为y的时候表示使能这个配置项 ,当为n的时候就禁止这个配置项。tristate类型有三种值:y、m和n,其中y和n的含义与bool类型一样,m表示将这个配置项编译为模块。string为字符串类型,所以LOCALVERSION是一个字符串变量,用来存储本地版本号,选中以后即可输入用户定义的本地版本号,如下图:

 

 string后面的“Local version - append to U-Boot release”就是这个配置项在图形界面上显示出来的标题。

    第18行,help表示帮助信息,告诉我们配置项的含义,当我们按下“h”或“?”弹出来的帮助界面就是help的内容。

    第27行,说明“CONFIG_LOCALVERSION_AUTO”是个bool类型,可以通过按下Y或N键来使能或者禁止CONFIG_LOCALVERSION_AUTO.

    第28行,“default  y”表示CONFIG_LOCALVERSION_AUTO的默认值就是y,所以这一行默认会被选中。

    5、depends on 和select

    打开arch/Kconfig文件,在里面有如下代码:

    第9行,“depends on”说明“SYS_GENERIC_BOARD”项依赖于“HAVE_GENERIC_BOARD”,也就是说”“HAVE_GENERIC_BOARD”被选中以后“SYS_GENERIC_BOARD”才能被选中。

    第17~20行,“select”表示方向依赖,当选中“ARC”以后,“HAVE_PRIVATE_LIBGCC”、“HAVE_GENERIC_BOARD”、“SYS_GENERIC_BOARD”和“SUPPORT_OF_CONTROL”这四个也会被选中。

    6、choice/endchoice

    在arch/Kconfig文件中有如下代码:

    choice/endchoice代码段定义了一组可选择项,将多个类似的配置项组合在一起,供用户单选或者多选。图11就是选中处理器架构,可以从ARC、ARM、AVR32等这些架构中选择,这里是单选。在uboot图形配置界面上 选择“Architecture select”,进入以后如图12:

 可以在图12中通过移动光标来选择所使用的CPU架构。第12行的prompt给出这个choice/endchoice段的提示信息为“Architecture  select”。

    7、menuconfig

    menuconfig和menu很类似,但是menuconfig是个带选项的菜单,其一般用法为:

 第1行,定义了一个可选的菜单MODULES,只有选中了MODULES第3~5行的if到endif之间的内容才会显示,在顶层Kconfig中有如下代码:

     第74~99行使用menuconfig实现了一个菜单,路径如下:

    General setup

        ->Configure  standard  U-Boot features(expert  users) --->

    如下图所示:

    从图15可以看出,前面有“[]”说明这个菜单是可选的,当选中这个菜单以后就可以进入到子选项中,也就是图14中的第83~99行所描述的菜单,如图16所示:

     如果不选择“Configure standard U-Boot features(expert users)”,那么图14所示的第83~99行所描述的菜单就不会显示出来,进去以后就是空白的。

    8、comment

    comment用于注释,也就是在图形话界面中显示一行注释,打开文件drivers/mtd/nand/Kconfig,有如下所示代码:

     第81行使用comment标注了一行注释,注释内容为:“Generic NAND options”,这行注释在配置项NAND_ARASAN的下面。在图形化配置界面中按照如下路径打开:

    ->Device  Drivers

        ->NAND  Device  Support

    结果如图18所示:

     从图18可以看出,在配置项“Configure Arasan Nand”下面有一行注释,注释内容为

“*** Generic  NAND  options***”。

    9、source

    source用于读取另一个Kconfig,比如:

    source  "arch/Kconfig"

    这个前面已经讲过了。

    

 

linux2.6.x的配置文件kconfig语法linux在2.6版本以后将配置文件由原来的config.in 改为kconfig,对于kconfig语法在/Documentation/kbuild/kconfig-language.txt中做了详细的说 明,在这里给出kconfig-language.txt的中文版。 介绍 ---- 在配置数据库的配置选项是以树的形式组织的: +- Code maturity level options | +- Prompt for development and/or incomplete code/drivers +- General setup | +- Networking support | +- System V IPC | +- BSD Process Accounting | +- Sysctl support +- Loadable module support | +- Enable loadable module support | +- Set version information on all module symbols | +- Kernel module loader +- ... 每个选项都有其自己的依赖关系。这些依赖关系决定了选项是否是可见的。父选项可见,子选项才能可见。 菜单选项 -------- 大多数的选项都定义了一个配置选项,其它选项则有助于对它们进行组织。(原文:Most entries define a config option, all other entries help to organize them.)一个配置选项定义可以是下面 的形式: config MODVERSIONS bool "Set version information on all module symbols" depends MODULES help Usually, modules have to be recompiled whenever you switch to a new kernel. ... 每行都是以关键字开始,并可以接多个参数。"config" 为定义了一新的配置选项。下面的几行定义了该配置 选项的属性。属性可以是该配置选项的类型,输入提示(input prompt),依赖关系,帮助信息和默认值。一 配置选项可以用相同的名字定义多次,但每个定义只能有一个输入提示并且类型还不能冲突。 菜单属性 -------- 一菜单选项可以有多个属性。并不要求这些属性可以用在任何地方(见语法)。 - 类型定义:"bool"/"tristate"/"string"/"hex"/"int" 每个配置选项都必须指定类型。有两个基本类型:tristate 和 string,其他类型都是基于这两个基本 类型。类型定义可以用输入提示,所以下面的两个例子是等价的: bool "Networking support" 和 bool prompt "Networking support" - 输入提示: "prompt" ["if" ] 每个菜单选项最多只能有一个显示给用户的输入提示。可以用 "if" 来表示该提示的依赖关系,当然这是 可选的。 - 默认值:"default" ["if" ] 一个配置选项可以有任意多个默认值。如果有多个默认值,那么只有第一个被定义的值是可用的。默认值并 不是只限于应用在定义他们的菜单选项。这就意味着默认值可以定义在任何地方或被更早的定义覆盖。 如果用户没有设置(通过上面的输入提示),配置选项的值就是默认值。如果可以显示输入提示的话,就会把 默认值显示给用户,并可以让用户进行修改。 默认值的依赖关系可以用 "if" 添加。(可选项) - 依赖关系:"depends on"/"requires" 为一菜单选项定义依赖关系。如果定义了多个依赖关系,它们之间用 '&&' 间隔。依赖关系也可以应用到 该菜单中所有的其它选项(同样接受一if表达式),所以下面的两个例子是等价的: bool "foo" if BAR default y if BAR and depends on BAR bool "foo" default y - 反向依赖关系:"select" ["if" ] 尽管普通的依赖关系可以降低一选项的上限,反向依赖能将这一限制降的更低。当前菜单选项的值是symbol 的最小值。如果symbol被选择了多次,上限就是其中的最大值。 反向依赖只能用在 boolean 或 tristate 选项上。 - 数据范围:"range" ["if" ] 为int和hex类型的选项设置可以接受输入值范围。用户只能输入大于等于第一个symbol,小于等于第二个 symbol的值。 - 帮助信息: "help" or "---help---" 定义一帮助信息。帮助信息的结束就由缩进的水平决定的,这也就意味着信息是在第一个比帮助信息开始行 的缩进小的行结束。 "---help---" 和 "help" 在实现的作用上没有区别,"---help---" 有助于将文件中的配置逻辑与 给开发人员的提示分开。 菜单依赖关系 ------------ 依赖关系决定了菜单选项是否可见,也可以减少tristate的输入范围。tristate逻辑比boolean逻辑在表 达式中用更多的状态(state)来表示模块的状态。依赖关系表达式的语法如下: ::= (1) '=' (2) '!=' (3) '(' ')' (4) '!' (5) '&&' (6) '||' (7) 表达式是以优先级的降序列出的。 (1) 将symbol赋给表达式。boolean和tristate类型的symbol直接赋给表达式。所有其它类型的symbol 都赋 'n'。 (2) 如果两个symbol相等,返回'y',否则为'n'。 (3) 如果两个symbol相等,返回'n',否则为'y'。 (4) 返回表达式的值。用于改变优先级。 (5) 返回 (2-/expr/) 的结果。 (6) 返回 min(/expr/,/expr/) 的结果。 (7) 返回 max(/expr/,/expr/) 的结果。 一个表达式的值可以是'n','m'或'y'(或者是计算的结果 0,1,2)。当表达式的值为'm'或'y'的时候,菜 单项才是可见的。 symbol有两种类型:不可变的和可变的。不可变的symbol是最普通的,由'config'语句定义,完全由数字 、字母和下划线组成(alphanumeric characters or underscores)。 不可变的symbol只是表达式的一部分。经常用单引号或双引号括起来。在引号中,可以使用任何字符,使用引 号要用转义字符'\'。 菜单结构 -------- 菜单在树中的位置可由两种方法决定。第一种可以是这样: menu "Network device support" depends NET config NETDEVICES ... endmenu 所有的在"menu" ... "endmenu" 之间都是"Network device support"的子菜单。所有的子菜单选项 都继承了父菜单的依赖关系,比如,"NET"的依赖关系就被加到了配置选项NETDEVICES的依赖列表中。 还有就是通过分析依赖关系生成菜单的结构。如果菜单选项在一定程度上依赖于前面的选项,它就能成为该选 项的子菜单。首先,前面的(父)选项必须是依赖列表中的一部分并且它们中必须有满足下面两个条件的选项: - 如果父选项为'n',子选项必须不可见。 - 如果父选项可见,子选项才能可见。 config MODULES bool "Enable loadable module support" config MODVERSIONS bool "Set version information on all module symbols" depends MODULES comment "module support disabled" depends !MODULES MODVERSIONS 直接依赖 MODULES,这就意味着如果MODULES不为'n',该选项才可见。换句话说,当 MODULES可见时,选项才可见(MODULES的(空)依赖关系也是选项依赖关系的一部分)。 Kconfig 语法 ------------ 配置文件描述了菜单选项,每行都是以一关键字开头(除了帮助信息)。下面的关键字结束一菜单选项: - config - menuconfig - choice/endchoice - comment - menu/endmenu - if/endif - source 前5个同样可以用在菜单选项定义的开始。 config: "config" 定义了一配置选项 并且可以接受任何前面介绍的属性。 menuconfig: "menuconfig" 此关键字和前面的关键字很相似,但它在前面的基础上要求所有的子选项作为独立的行显示。(This is similar to the simple config entry above, but it also gives a hint to front ends, that all suboptions should be displayed as a separate list of options.) choices: "choice" "endchoice" 该关键字定义了一组选择项,并且选项可以是前面描述的任何属性。尽管boolean只允许选择一个配置选项, tristate可以抒多个配置选项设为'm',但选项只能是boolean或tristate类型。这可以在一个硬件有多 个驱动的情况下使用,最终只有一个驱动被编译进/加载到内核,,但所有的驱动都可以编译成模块。 选项可以接受的另一个选项是"optional",这样选项就被设置为'n',没有被选中的。 comment: "comment" 这里定义了在配置过程中显示给用户的注释,该注释还将写进输出文件中。唯一可用的可选项是依赖关系。 menu: "menu" "endmenu" 这里定义了一个菜单,详细信息请看前面的"菜单结构"。唯一可用的可选项是依赖关系。 if: "if" "endif" 这里定义了if结构。依赖关系被加到所有在if ... endif 中的菜单选项中。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值