array(2) { ["docs"]=> array(10) { [0]=> array(10) { ["id"]=> string(3) "428" ["text"]=> string(77) "Visual Studio 2017 单独启动MSDN帮助(Microsoft Help Viewer)的方法" ["intro"]=> string(288) "目录 ECharts 异步加载 ECharts 数据可视化在过去几年中取得了巨大进展。开发人员对可视化产品的期望不再是简单的图表创建工具,而是在交互、性能、数据处理等方面有更高的要求。 chart.setOption({ color: [ " ["username"]=> string(8) "DonetRen" ["tagsname"]=> string(55) "Visual Studio 2017|MSDN帮助|C#程序|.NET|Help Viewer" ["tagsid"]=> string(23) "[401,402,403,"300",404]" ["catesname"]=> string(0) "" ["catesid"]=> string(2) "[]" ["createtime"]=> string(10) "1511400964" ["_id"]=> string(3) "428" } [1]=> array(10) { ["id"]=> string(3) "427" ["text"]=> string(42) "npm -v;报错 cannot find module "wrapp"" ["intro"]=> string(288) "目录 ECharts 异步加载 ECharts 数据可视化在过去几年中取得了巨大进展。开发人员对可视化产品的期望不再是简单的图表创建工具,而是在交互、性能、数据处理等方面有更高的要求。 chart.setOption({ color: [ " ["username"]=> string(4) "zzty" ["tagsname"]=> string(50) "node.js|npm|cannot find module "wrapp“|node" ["tagsid"]=> string(19) "[398,"239",399,400]" ["catesname"]=> string(0) "" ["catesid"]=> string(2) "[]" ["createtime"]=> string(10) "1511400760" ["_id"]=> string(3) "427" } [2]=> array(10) { ["id"]=> string(3) "426" ["text"]=> string(54) "说说css中pt、px、em、rem都扮演了什么角色" ["intro"]=> string(288) "目录 ECharts 异步加载 ECharts 数据可视化在过去几年中取得了巨大进展。开发人员对可视化产品的期望不再是简单的图表创建工具,而是在交互、性能、数据处理等方面有更高的要求。 chart.setOption({ color: [ " ["username"]=> string(12) "zhengqiaoyin" ["tagsname"]=> string(0) "" ["tagsid"]=> string(2) "[]" ["catesname"]=> string(0) "" ["catesid"]=> string(2) "[]" ["createtime"]=> string(10) "1511400640" ["_id"]=> string(3) "426" } [3]=> array(10) { ["id"]=> string(3) "425" ["text"]=> string(83) "深入学习JS执行--创建执行上下文(变量对象,作用域链,this)" ["intro"]=> string(288) "目录 ECharts 异步加载 ECharts 数据可视化在过去几年中取得了巨大进展。开发人员对可视化产品的期望不再是简单的图表创建工具,而是在交互、性能、数据处理等方面有更高的要求。 chart.setOption({ color: [ " ["username"]=> string(7) "Ry-yuan" ["tagsname"]=> string(33) "Javascript|Javascript执行过程" ["tagsid"]=> string(13) "["169","191"]" ["catesname"]=> string(0) "" ["catesid"]=> string(2) "[]" ["createtime"]=> string(10) "1511399901" ["_id"]=> string(3) "425" } [4]=> array(10) { ["id"]=> string(3) "424" ["text"]=> string(30) "C# 排序技术研究与对比" ["intro"]=> string(288) "目录 ECharts 异步加载 ECharts 数据可视化在过去几年中取得了巨大进展。开发人员对可视化产品的期望不再是简单的图表创建工具,而是在交互、性能、数据处理等方面有更高的要求。 chart.setOption({ color: [ " ["username"]=> string(9) "vveiliang" ["tagsname"]=> string(0) "" ["tagsid"]=> string(2) "[]" ["catesname"]=> string(8) ".Net Dev" ["catesid"]=> string(5) "[199]" ["createtime"]=> string(10) "1511399150" ["_id"]=> string(3) "424" } [5]=> array(10) { ["id"]=> string(3) "423" ["text"]=> string(72) "【算法】小白的算法笔记:快速排序算法的编码和优化" ["intro"]=> string(288) "目录 ECharts 异步加载 ECharts 数据可视化在过去几年中取得了巨大进展。开发人员对可视化产品的期望不再是简单的图表创建工具,而是在交互、性能、数据处理等方面有更高的要求。 chart.setOption({ color: [ " ["username"]=> string(9) "penghuwan" ["tagsname"]=> string(6) "算法" ["tagsid"]=> string(7) "["344"]" ["catesname"]=> string(0) "" ["catesid"]=> string(2) "[]" ["createtime"]=> string(10) "1511398109" ["_id"]=> string(3) "423" } [6]=> array(10) { ["id"]=> string(3) "422" ["text"]=> string(64) "JavaScript数据可视化编程学习(二)Flotr2,雷达图" ["intro"]=> string(288) "目录 ECharts 异步加载 ECharts 数据可视化在过去几年中取得了巨大进展。开发人员对可视化产品的期望不再是简单的图表创建工具,而是在交互、性能、数据处理等方面有更高的要求。 chart.setOption({ color: [ " ["username"]=> string(7) "chengxs" ["tagsname"]=> string(28) "数据可视化|前端学习" ["tagsid"]=> string(9) "[396,397]" ["catesname"]=> string(18) "前端基本知识" ["catesid"]=> string(5) "[198]" ["createtime"]=> string(10) "1511397800" ["_id"]=> string(3) "422" } [7]=> array(10) { ["id"]=> string(3) "421" ["text"]=> string(36) "C#表达式目录树(Expression)" ["intro"]=> string(288) "目录 ECharts 异步加载 ECharts 数据可视化在过去几年中取得了巨大进展。开发人员对可视化产品的期望不再是简单的图表创建工具,而是在交互、性能、数据处理等方面有更高的要求。 chart.setOption({ color: [ " ["username"]=> string(4) "wwym" ["tagsname"]=> string(0) "" ["tagsid"]=> string(2) "[]" ["catesname"]=> string(4) ".NET" ["catesid"]=> string(7) "["119"]" ["createtime"]=> string(10) "1511397474" ["_id"]=> string(3) "421" } [8]=> array(10) { ["id"]=> string(3) "420" ["text"]=> string(47) "数据结构 队列_队列实例:事件处理" ["intro"]=> string(288) "目录 ECharts 异步加载 ECharts 数据可视化在过去几年中取得了巨大进展。开发人员对可视化产品的期望不再是简单的图表创建工具,而是在交互、性能、数据处理等方面有更高的要求。 chart.setOption({ color: [ " ["username"]=> string(7) "idreamo" ["tagsname"]=> string(40) "C语言|数据结构|队列|事件处理" ["tagsid"]=> string(23) "["246","247","248",395]" ["catesname"]=> string(12) "数据结构" ["catesid"]=> string(7) "["133"]" ["createtime"]=> string(10) "1511397279" ["_id"]=> string(3) "420" } [9]=> array(10) { ["id"]=> string(3) "419" ["text"]=> string(47) "久等了,博客园官方Android客户端发布" ["intro"]=> string(288) "目录 ECharts 异步加载 ECharts 数据可视化在过去几年中取得了巨大进展。开发人员对可视化产品的期望不再是简单的图表创建工具,而是在交互、性能、数据处理等方面有更高的要求。 chart.setOption({ color: [ " ["username"]=> string(3) "cmt" ["tagsname"]=> string(0) "" ["tagsid"]=> string(2) "[]" ["catesname"]=> string(0) "" ["catesid"]=> string(2) "[]" ["createtime"]=> string(10) "1511396549" ["_id"]=> string(3) "419" } } ["count"]=> int(200) } 222 ubuntu——Kconfig、.config、Makefile的关系 - 爱码网

原文地址:http://blog.csdn.net/estate66/article/details/5886816 ,本人对此文有改进。

  当我们编写完一个驱动后,我们要把它以模块形式编译或者直接编译进内核时,需要修改相关文件,其中最重要的便是kconfig ,makefile。主要是分析一下三者之间的关系,然后就其语法简要的谈一下。

  当我们在内核源码目录下执行make (或者make menuconfig等命令)命令时,实际上是根据makefile 来进行编译的。我在 mini2440开发板上编写了一个按键控制led灯的驱动,文件名为buttons_leds_zhao.c,属于字符驱动。因此在/driver /char/目录下的makefile部分最后添加一行:

obj-$(CONFIG_BUTTONS_LEDS_ZHAO) += buttons_leds_zhao.o

添加后如下:

  obj-y += mem.o random.o tty_io.o n_tty.o tty_ioctl.o  tty_ldisc.o tty_buffer.o tty_port.o

  obj-$(CONFIG_BFIN_JTAG_COMM) += bfin_jtag_comm.o

  obj-$(CONFIG_CONSOLE_TRANSLATIONS) += consolemap.o consolemap_deftbl.o

  obj-$(CONFIG_HW_CONSOLE) += vt.o defkeymap.o

  obj-$(CONFIG_AUDIT) += tty_audit.o

  obj-$(CONFIG_BUTTONS_LEDS_ZHAO)    += buttons_leds_zhao.o

  其中第一行obj-y  中的y表示编译进内核,而obj-$(CONFIG_LEGACY_PTYS)  中CONFIG_LEGACY_PTYS则表示 一个变量,类似于我们C语言中的变量,用$( )来表示,它一般可以取三种值y ,m ,n(y表示编译进内核,而m则表示以模块的方式进行编译,n表示 不编译进内核)。obj-y+= 等号后面的.o后缀文件则是由该目录下的对应名称的.c文件编译而来。而上面 CONFIG_LEGACY_PTYS变量的取值则是通过.config文件来集中赋值的。.config部分内容如下

  # Character devices 
  # 
  CONFIG_VT=y 
  CONFIG_CONSOLE_TRANSLATIONS=y 
  CONFIG_VT_CONSOLE=y 
  CONFIG_HW_CONSOLE=y 
  # CONFIG_VT_HW_CONSOLE_BINDING is not set 
  # CONFIG_DEVKMEM is not set 
  CONFIG_MINI2440_HELLO_MODULE=m 
  CONFIG_BUTTONS_LEDS_ZHAO=m 
  CONFIG_LEDS_MINI2440=m 
  CONFIG_MINI2440_BUTTONS=m 
  CONFIG_MINI2440_BUZZER=y 
  CONFIG_MINI2440_ADC=y 
  # CONFIG_SERIAL_NONSTANDARD is not set

  从上面几行我们可以看到,在makefile里面的变量都是在.config中赋值的。当我们在源代码目录下输入make命令时,都是默认从.config中读入。

  因此,在输入make之前,用ls  -a 查看一下是否有该文件。对于只包含几个文件的工程来说,手动写.config和makefile并不是一件很难的事情,但如果是一个包含有几百个文件的项目来说,则是一件比较困难的事情。可以用autoconf来自动生成.config,automake来制作 makefile。看起来问题是解决的,但实际上,这种做法缺乏一定的灵活性,不能实现按需定制的要求。如果要添加或删掉某个驱动,将要在.config 文件中找到相应的项进行修改。非常的不方便。因此,便出现了kconfig。

  当我们在内核源码目录下输入make menuconfig时,出现如下内容:

  .config - Linux Kernel v2.6.32.2 Configuration 

──────────────────────────────────────────────────────────────────────────────────────────────────
┌──────────────────────────────── Linux Kernel Configuration ─────────────────────────────────┐
│  Arrow keys navigate the menu.  <Enter> selects submenus --->.  Highlighted letters are     │
│  hotkeys.  Pressing <Y> includes, <N> excludes, <M> modularizes features.  Press <Esc><Esc> │
│  to exit, <?> for Help, </> for Search.  Legend: [*] built-in  [ ] excluded  <M> module     │
│  < > module capable                                                                         │
│ ┌─────────────────────────────────────────────────────────────────────────────────────────┐ │
│ │             General setup  --->                                                         │ │
│ │         [*] Enable loadable module support  --->                                        │ │
│ │         -*- Enable the block layer  --->                                                │ │
│ │             System Type  --->                                                           │ │
│ │             Bus support  --->                                                           │ │
│ │             Kernel Features  --->                                                       │ │
│ │             Boot options  --->                                                          │ │
│ │             CPU Power Management  --->                                                  │ │
│ │             Floating point emulation  --->                                              │ │
│ │             Userspace binary formats  --->                                              │ │
│ │             Power management options  --->                                              │ │
│ │         [*] Networking support  --->                                                    │ │
│ │             Device Drivers  --->                                                        │ │
│ │             File systems  --->                                                          │ │
│ └─────────┴(+)────────────────────────────────────────────────────────────────────────────┘ │
├─────────────────────────────────────────────────────────────────────────────────────────────┤
│                              <Select>    < Exit >    < Help >                               │
└────────────────────────────────────────────────────────────────────────

  它是通过读取在内核源码目录下的Kconfig文件来配置的。在/drivers/char/下的目录kconfig部分内容如下:

  config DEVKMEM 
  bool "/dev/kmem virtual device support" 
  default y 
  help 
  Say Y here if you want to support the /dev/kmem device. The 
  /dev/kmem device is rarely used, but can be used for certain 
  kind of kernel debugging operations. 
  When in doubt, say "N". 

  config MINI2440_HELLO_MODULE 
  tristate "Mini2440 module sample" 
       depends on MACH_MINI2440 
       default m if MACH_MINI2440 
       help 
  Mini2440 module sample. 

  config BUTTONS_LEDS_ZHAO 
  tristate "Mini2440 button  and leds sample" 
       depends on MACH_MINI2440 
       default m if MACH_MINI2440 
       help 
       Mini2440  button and leds  module sample.

  其具体语法格式说明如下:

  config 关键字是一个新的配置选项的入口,其后的选项MINI2440_HELLO_MODULE 省略了CONFIG。完整的表示为 CONFIG_MINI2440_HELLO_MODULE,也即是当我们将该选项设置成y时,它将自动的将 .config的 CONFIG_MINI2440_HELLO_MODULE=m改写成CONFIG_MINI2440_HELLO_MODULE=y。紧接着的是菜单的属性,最主要的有2种tristate ,boolean 。tristate表示三态:编译进内核(y),编译成模块(m),不编译(n)。 boolean 主要有两种y或n。depend则表示依赖项。(depends on MACH_MINI2440)这个例子表明BUTTONS_LEDS_ZHAO这个菜单项只对MACH_MINI2440处理器有效,即只有选择了MACH_MINI2440 , 该菜单才可见(可配置)。default缺省的编译选项m表示默认该文件表示以模块方式编译。后面的help是帮助信息,当我们选中help菜单时就可以看见,它不是必需的。随着操作系统升级,编译选项达到几千个。对于一般人来说,要搞清每一个选项是很困难的,一般很多选项都是默认的。

  总的来说,三者之间的关系如下:当我们在内核源码目录下输入make menuconfig时,在出现的菜单界面中选择一项时,它会自动更新.config相应项的值。如果我们没有选择,则会在.config文件下插入一行注释。类似于# CONFIG_SERIAL_NONSTANDARD is not set,当我们输入make时,根据 makefile文件来编译,makefile文件中的变量值则由.config来进行赋值操作。仅仅只在kconfig中添加选项,只会在菜单界面中显示,即使此时选择y或m,也不会编译文件。还需要在makefile文件中按照规定添加相应行才能进行编译。简单图解如下:

  kconfig------->.config---------->makefile

  最后举个例子:

  假设想把自己写的一个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 加入到内核中。即可达到您的目的。

相关文章: