Linux內核配置機制(make menuconfig 、Kconfig、Makefile)講解【轉】

本文轉載自:http://www.codexiu.cn/linux/blog/34801/linux

前面咱們介紹模塊編程的時候介紹了驅動進入內核有兩種方式:模塊和直接編譯進內核,並介紹了模塊的一種編譯方式——在一個獨立的文件夾經過makefile配合內核源碼路徑完成
 

    那麼如何將驅動直接編譯進內核呢?編程

    在咱們實際內核的移植配置過程當中常常據說的內核裁剪又是怎麼麼回事呢?windows

咱們在進行linux內核配置的時候常常會執行make menuconfig這個命令,而後屏幕上會出現如下界面:工具

這個界面是怎麼生成的呢?ui

跟咱們常常說的內核配置與與編譯又有什麼關係呢?spa

下面咱們藉此來說解一下linux內核的配置機制及其編譯過程。debug

1、配置系統的基本結構3d

Linux內核的配置系統由三個部分組成,分別是:code

   一、Makefile:分佈在 Linux 內核源代碼根目錄及各層目錄中,定義 Linux 內核的編譯規則;blog

    二、配置文件(config.in(2.4內核,2.6內核)):給用戶提供配置選擇的功能;

    三、配置工具:包括配置命令解釋器(對配置腳本中使用的配置命令進行解釋)和配置用戶界面(提供基於字符界面、基於 Ncurses 圖形界面以及基於 Xwindows 圖形界面的用戶配置界面,各自對應於 Make config、Make menuconfig 和 make xconfig)。

   這些配置工具都是使用腳本語言,如 Tcl/TK、Perl 編寫的(也包含一些用 C 編寫的代碼)。本文並非對配置系統自己進行分析,而是介紹如何使用配置系統。因此,除非是配置系統的維護者,通常的內核開發者無須瞭解它們的原理,只須要知道如何編寫 Makefile 和配置文件就能夠。

2、makefile menuconfig過程講解

當咱們在執行make menuconfig這個命令時,系統到底幫咱們作了哪些工做呢?

這裏面一共涉及到了一下幾個文件咱們來一一講解

Linux內核根目錄下的scripts文件夾

arch/$ARCH/Kconfig文件、各層目錄下的Kconfig文件

Linux內核根目錄下的makefile文件、各層目錄下的makefile文件

Linux內核根目錄下的的.config文件、arm/$ARCH/下的config文件

Linux內核根目錄下的 include/generated/autoconf.h文件

1)scripts文件夾存放的是跟make menuconfig配置界面的圖形繪製相關的文件,咱們做爲使用者無需關心這個文件夾的內容

2)當咱們執行make menuconfig命令出現上述藍色配置界面之前,系統幫咱們作了如下工做:

    首先系統會讀取arch/$ARCH/目錄下的Kconfig文件生成整個配置界面選項(Kconfig是整個linux配置機制的核心),那麼ARCH環境變量的值等於多少呢?

它是由linux內核根目錄下的makefile文件決定的,在makefile下有此環境變量的定義:

或者經過 make ARCH=arm menuconfig命令來生成配置界面,默認生成的界面是全部參數都是沒有值的

    好比教務處進行考試,考試科數可能有外語、語文、數學等科,這裏至關於咱們選擇了arm科可進行考試,系統就會讀取arm/arm/kconfig文件生成配置選項(選擇了arm科的卷子),系統還提供了x86科、milps科等10幾門功課的考試題

3)假設教務處比較「仁慈」,爲了怕某些同窗作不錯試題,還給咱們準備了一份參考答案(默認配置選項),存放在arch/$ARCH/configs下,對於arm科來講就是arch/arm/configs文件夾:

    此文件夾中有許多選項,系統會讀取哪一個呢?內核默認會讀取linux內核根目錄下.config文件做爲內核的默認選項(試題的參考答案),咱們通常會根據開發板的類型從中選取一個與咱們開發板最接近的系列到Linux內核根目錄下(選擇一個最接近的參考答案)

#cp arch/arm/configs/s3c2410_defconfig .config

4).config

    假設教務處留了一個心眼,他提供的參考答案並不徹底正確(.config文件與咱們的板子並非徹底匹配),這時咱們能夠選擇直接修改.config文件而後執行make menuconfig命令讀取新的選項

    可是通常咱們不採起這個方案,咱們選擇在配置界面中經過空格、esc、回車選擇某些選項選中或者不選中,最後保存退出的時候,Linux內核會把新的選項(正確的參考答案)更新到.config中,此時咱們能夠把.config重命名爲其它文件保存起來(當你執行make distclean時系統會把.config文件刪除),之後咱們再配置內核時就不須要再去arch/arm/configs下考取相應的文件了,省去了從新配置的麻煩,直接將保存的.config文件複製爲.config便可.

5)通過以上兩步,咱們能夠正確的讀取、配置咱們須要的界面了

那麼他們如何跟makefile文件創建編譯關係呢?

當你保存make menuconfig選項時,系統會除了會自動更新.config外,還會將全部的選項以宏的形式保存在

Linux內核根目錄下的 include/generated/autoconf.h文件下

內核中的源代碼就都會包含以上.h文件,跟宏的定義狀況進行條件編譯。

當咱們須要對一個文件總體選擇如是否編譯時,還須要修改對應的makefile文件,例如:

    咱們選擇是否要編譯s3c2410_ts.c這個文件時,makefile會根據CONFIG_TOUCHSCREEN_S3C2410來決定是編譯此文件,此宏是在Kconfig文件中定義,當咱們配置完成後,會出如今.config及autconf中,至此,咱們就完成了整個linux內核的編譯過程。

    最後咱們會發現,整個linux內核配置過程當中,留給用戶的接口其實只有各層Kconfig、makefile文件以及對應的源文件。

    好比咱們若是想要給內核增長一個功能,而且經過make menuconfig控制其聲稱過程

    首先須要作的工做是:修改對應目錄下的Kconfig文件,按照Kconfig語法增長對應的選項;

    其次執行make menuconfig選擇編譯進內核或者不編譯進內核,或者編譯爲模塊,.config文件和autoconf.h文件會自動生成;

    最後修改對應目錄下的makefile文件完成編譯選項的添加;

    最後的最後執行make zImage命令進行編譯。

3、具體實例

下面咱們之前面作過的模塊實驗爲例,講解如何經過make menuconfig機制將前面單獨編譯的模塊編譯進內核或編譯爲模塊

假設我已經有了這麼一個驅動:

modules.c

#include <linux/module.h>       /*module_init()*/    #include <linux/kernel.h> /* printk() */    #include <linux/init.h>       /* __init __exit */       #define DEBUG   //open debug message       #ifdef DEBUG    #define PRINTK(fmt, arg...)     printk(KERN_WARNING fmt, ##arg)    #else    #define PRINTK(fmt, arg...)     printk(KERN_DEBUG fmt, ##arg)    #endif       /* Module Init & Exit function */   static int __init myModule_init(void)   {       /* Module init code */       PRINTK("myModule_init\n");       return 0;   }      static void __exit myModule_exit(void)   {       /* Module exit code */       PRINTK("myModule_exit\n");       return;   }      module_init(myModule_init);   module_exit(myModule_exit);      MODULE_AUTHOR("dengwei");                          /*模塊做者,可選*/   MODULE_LICENSE("GPL");                             /*模塊許可證實,描述內核模塊的許可權限,必須*/   MODULE_DESCRIPTION("A simple Hello World Module"); /*模塊說明,可選*/
Step1:將modules.c拷到drivers/char/目錄下(這個文件夾通常存放常見的字符驅動)
Step2: vi driver/char/Kconfig,在
    config DEVKMEM後添加如下信息
config MODULES
tristate "modules device support"
default y
help
 Say Y here,the modules will be build in kernel.
 Say M here,the modules willbe build to modules.
 Say N here,there will be nothing to be do. 
Step3:make menuconfig
     Device driver-character devices
           [*]modules device suppor
Step4:vi driver/char/Makefile,在js-rtc後添加
obj-$(CONFIG_MODULES)+= modules.o
CONFIG_MODULES 必須跟上面的Kconfig中保持一致,系統會自動添加CONFIG_前綴

modules.o必須跟你加入的.c文件名一致

最後執行:make zImage modules就會被編譯進內核中

第三步:

Step3:make menuconfig
     Device driver-character devices
           [M]modules device suppor
把星號在配置界面經過空格改成M,最後執行make modules,在driver/char/目錄下會生成一個modules.ko文件

跟咱們前面講的單獨編譯模塊效果同樣,也會生成一個模塊,將它考入開發板執行insmod moudles.ko,便可將生成的模塊插入內核使用。


PS:本身的見解

Linux內核的make menuconfig其實是執行了:

scripts/kconfig/mconf      arch/mips/Kconfig

mconf表示是menuconfig,若是是用基於QT的配置工具,則執行的將會是qconf,arch/mips/Kconfig是要讀取的Kconfig文件,這個會因平臺而異,這裏由於針對的是MIPS平臺,故讀取的是arch/mips/目錄下的Kconfig文件。

若是Linux源碼樹頂層目錄下已有.config文件,make menuconfig則從.config文件取默認參數,若是沒有.config則從各個Kconfig中取默認參數。

mconf會把用戶的選擇保存到Linux源碼樹頂層目錄的.config文件中,而後解析該文件並將解析結果寫入到include/linux/autoconf.h中。include/linux/autoconf.h將會被include/linux/config.h包含,所以,須要關心配置狀況的內核源文件只須要#include <linux/config.h>便可。

mconf解析.config文件時所採用的規則具體要仔細分析mconf.c源代碼。

相關文章
相關標籤/搜索