Linux加載DTS設備節點的過程(以高通8974平台為例)


DTS是Device Tree Source的縮寫,用來描述設備的硬件細節。在過去的ARM Linux中,arch/arm/plat-xxx和arch/arm/mach-xxx中充斥着大量的垃圾代碼,相當多數的代碼只是在描述板級細節,而這些板級細節對於內核來講,不過是垃圾,如板上的platform設備、resource、i2c_board_info、spi_board_info以及各種硬件的platform_data。為了去掉這些垃圾代碼,Linux采用DTS這種新的數據結構來描述硬件設備。采用Device Tree后,許多硬件的細節可以直接透過它傳遞給Linux,而不再需要在kernel中進行大量的冗余編碼。

       有關DTS的語法格式,網上有很多資料,這里我就不再贅述了。本文主要講Linux是怎樣加載DTS設備節點的流程。下面以高通QCT8974平台(Linux內核)為例進行講解:

(1)使用DTS注冊平台總線的過程

       在講注冊過程之前,我們先看看DTS是怎樣描述平台總線結構的,以i2c總線為例:

  1. / {  
  2.     model = "Qualcomm MSM 8974";  
  3.     compatible = "qcom,msm8974";  
  4.     interrupt-parent = <&intc>;  
  5.   
  6.     aliases {  
  7.         spi0 = &spi_0;  
  8.         spi7 = &spi_7;  
  9.         sdhc1 = &sdhc_1; /* SDC1 eMMC slot */  
  10.         sdhc2 = &sdhc_2; /* SDC2 SD card slot */  
  11.         sdhc3 = &sdhc_3; /* SDC3 SDIO slot */  
  12.         sdhc4 = &sdhc_4; /* SDC4 SDIO slot */  
  13.     };  
  14.   
  15.     memory {  
  16.   
  17.         secure_mem: secure_region {  
  18.             linux,contiguous-region;  
  19.             reg = <0 0x7800000="">;  
  20.             label = "secure_mem";  
  21.         };  
  22.   
  23.         adsp_mem: adsp_region {  
  24.             linux,contiguous-region;  
  25.             reg = <0 0x2000000="">;  
  26.             label = "adsp_mem";  
  27.         };  
  28.     };  
  29.   
  30.     intc: interrupt-controller@F9000000 {  
  31.         compatible = "qcom,msm-qgic2";  
  32.         interrupt-controller;  
  33.         #interrupt-cells = <3>;  
  34.         reg = <0xf9000000 0x1000="">,  
  35.               <0xf9002000 0x1000="">;  
  36.     };  
  37.   
  38.     msmgpio: gpio@fd510000 {  
  39.         compatible = "qcom,msm-gpio";  
  40.         gpio-controller;  
  41.         #gpio-cells = <2>;  
  42.         interrupt-controller;  
  43.         #interrupt-cells = <2>;  
  44.         reg = <0xfd510000 0x4000="">;  
  45.         ngpio = <146>;  
  46.         interrupts = <0 208="" 0="">;  
  47.         qcom,direct-connect-irqs = <8>;  
  48.     };  
  49.   
  50.     wcd9xxx_intc: wcd9xxx-irq {  
  51.         compatible = "qcom,wcd9xxx-irq";  
  52.         interrupt-controller;  
  53.         #interrupt-cells = <1>;  
  54.         interrupt-parent = <&msmgpio>;  
  55.         interrupts = <72 0="">;  
  56.         interrupt-names = "cdc-int";  
  57.     };  
  58.   
  59.     timer {  
  60.         compatible = "arm,armv7-timer";  
  61.         interrupts = <1 2="" 0="" 1="" 3="" 0="">;  
  62.         clock-frequency = <19200000>;  
  63.     };  
  64.   
  65.     i2c_0: i2c@f9967000 { /* BLSP#11 */  
  66.         cell-index = <0>;  
  67.         compatible = "qcom,i2c-qup";  
  68.         reg = <0xf9967000 0x1000="">;  
  69.         #address-cells = <1>;  
  70.         #size-cells = <0>;  
  71.         reg-names = "qup_phys_addr";  
  72.         interrupts = <0 105="" 0="">;  
  73.         interrupt-names = "qup_err_intr";  
  74.         qcom,i2c-bus-freq = <100000>;  
  75.         qcom,i2c-src-freq = <50000000>;  
  76.     };  
  77.   
  78.     i2c_2: i2c@f9924000 {  
  79.         cell-index = <2>;  
  80.         compatible = "qcom,i2c-qup";  
  81.         reg = <0xf9924000 0x1000="">;  
  82.         #address-cells = <1>;  
  83.         #size-cells = <0>;  
  84.         reg-names = "qup_phys_addr";  
  85.         interrupts = <0 96="" 0="">;  
  86.         interrupt-names = "qup_err_intr";  
  87.         qcom,i2c-bus-freq = <100000>;  
  88.         qcom,i2c-src-freq = <50000000>;  
  89.     };  
  90.   
  91.     spi_0: spi@f9923000 {  
  92.         compatible = "qcom,spi-qup-v2";  
  93.         reg = <0xf9923000 0x1000="">;  
  94.         interrupts = <0 95="" 0="">;  
  95.         spi-max-frequency = <19200000>;  
  96.         #address-cells = <1>;  
  97.         #size-cells = <0>;  
  98.         gpios = <&msmgpio 3 0>, /* CLK  */  
  99.             <&msmgpio 1 0>, /* MISO */  
  100.             <&msmgpio 0 0>; /* MOSI */  
  101.         cs-gpios = <&msmgpio 9 0>;  
  102.     };  
  103.   
  104. };  

從上面可知,系統平台上掛載了很多總線,如i2c、spi、uart等等,每一個總線分別被描述為一個節點。Linux在啟動后,到C入口時,會執行以下操作,加載系統平台上的總線和設備:

start_kernel() --> setup_arch() --> unflatten_device_tree() 

unflatten_device_tree()的代碼如下:

  1. void __init unflatten_device_tree(void)  
  2. {  
  3.     __unflatten_device_tree(initial_boot_params, &allnodes,  
  4.                 early_init_dt_alloc_memory_arch);  
  5.   
  6.     /* Get pointer to "/chosen" and "/aliasas" nodes for use everywhere */  
  7.     of_alias_scan(early_init_dt_alloc_memory_arch);  
  8. }  

 在執行完unflatten_device_tree()后,DTS節點信息被解析出來,保存到allnodes鏈表中,allnodes會在后面被用到。

隨后,當系統啟動到board文件時,會調用.init_machine,高通8974平台對應的是msm8974_init()。接着調用of_platform_populate(....)接口,加載平台總線和平台設備。至此,系統平台上的所有已配置的總線和設備將被注冊到系統中。注意:不是dtsi文件中所有的節點都會被注冊,在注冊總線和設備時,會對dts節點的狀態作一個判斷,如果節點里面的status屬性沒有被定義,或者status屬性被定義了並且值被設為“ok”或者“okay”,其他情況則不被注冊到系統中。


(2)使用DTS注冊總線設備的過程

       上面講了Linux怎樣使用DTS注冊平台總線和平台設備到系統中,那么其他設備,例如i2c、spi設備是怎樣注冊到系統中的呢?下面我們就以i2c設備為例,講解Linux怎樣注冊i2c設備到系統中。

 以高通8974平台為例,在注冊i2c總線時,會調用到qup_i2c_probe()接口,該接口用於申請總線資源和添加i2c適配器。在成功添加i2c適配器后,會調用of_i2c_register_devices()接口。此接口會解析i2c總線節點的子節點(掛載在該總線上的i2c設備節點),獲取i2c設備的地址、中斷號等硬件信息。然后調用request_module()加載設備對應的驅動文件,調用i2c_new_device(),生成i2c設備。此時設備和驅動都已加載,於是drvier里面的probe方法將被調用。后面流程就和之前一樣了。


       簡而言之,Linux采用DTS描述設備硬件信息后,省去了大量板文件垃圾信息。Linux在開機啟動階段,會解析DTS文件,保存到全局鏈表allnodes中,在掉用.init_machine時,會跟據allnodes中的信息注冊平台總線和設備。值得注意的是,加載流程並不是按找從樹根到樹葉的方式遞歸注冊,而是只注冊根節點下的第一級子節點,第二級及之后的子節點暫不注冊。Linux系統下的設備大多都是掛載在平台總線下的,因此在平台總線被注冊后,會根據allnodes節點的樹結構,去尋找該總線的子節點,所有的子節點將被作為設備注冊到該總線上。


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2026 CODEPRJ.COM