一文搞定 Linux 設(shè)備樹
設(shè)備樹是一種描述硬件的數(shù)據(jù)結(jié)構(gòu),它起源于OpenFirmware(OF)。
在Linux 2.6中, ARM架構(gòu)的板極硬件細節(jié)過多地被硬編碼在arch/arm/plat-xxx和arch/arm/mach-xxx中,采用設(shè)備樹后,許多硬件的細節(jié)可以直接通過它傳遞給Linux,而不再需要在內(nèi)核中進行大量的冗余編碼。
1. linux設(shè)備樹中DTS、 DTC和DTB的關(guān)系
- (1) DTS:.dts文件是設(shè)備樹的源文件。由于一個SoC可能對應(yīng)多個設(shè)備,這些.dst文件可能包含很多共同的部分,共同的部分一般被提煉為一個 .dtsi 文件,這個文件相當于C語言的頭文件。
- (2) DTC:DTC是將.dts編譯為.dtb的工具,相當于gcc。
- (3) DTB:.dtb文件是 .dts 被 DTC 編譯后的二進制格式的設(shè)備樹文件,它可以被linux內(nèi)核解析。
2. DTS語法
2.1 .dtsi 頭文件
和 C 語言一樣,設(shè)備樹也支持頭文件,設(shè)備樹的頭文件擴展名為 .dtsi;同時也可以像C 語言一樣包含 .h頭文件;例如:(代碼來源 linux-4.15/arch/arm/boot/dts/s3c2416.dtsi)
- #include <dt-bindings/clock/s3c2443.h>
- #include "s3c24xx.dtsi"
注:.dtsi 文件一般用于描述 SOC 的內(nèi)部外設(shè)信息,比如 CPU 架構(gòu)、主頻、外設(shè)寄存器地址范圍,比如 UART、 IIC 等等。
2.2 設(shè)備節(jié)點
在設(shè)備樹中節(jié)點命名格式如下:
- node-name@unit-address
node-name:是設(shè)備節(jié)點的名稱,為ASCII字符串,節(jié)點名字應(yīng)該能夠清晰的描述出節(jié)點的功能,比如“uart1”就表示這個節(jié)點是UART1外設(shè);unit-address:一般表示設(shè)備的地址或寄存器首地址,如果某個節(jié)點沒有地址或者寄存器的話 “unit-address” 可以不要;注:根節(jié)點沒有node-name 或者 unit-address,它被定義為 /。
設(shè)備節(jié)點的例子如下圖:
在上圖中:cpu 和 ethernet依靠不同的unit-address 分辨不同的CPU;可見,node-name相同的情況下,可以通過不同的unit-address定義不同的設(shè)備節(jié)點。
2.2.1 設(shè)備節(jié)點的標準屬性
2.2.1.1 compatible 屬性
compatible 屬性也叫做 “兼容性” 屬性,這是非常重要的一個屬性!compatible 屬性的值是一個字符串列表, compatible 屬性用于將設(shè)備和驅(qū)動綁定起來。字符串列表用于選擇設(shè)備所要使用的驅(qū)動程序。compatible 屬性值的推薦格式:
- "manufacturer,model"
- ① manufacturer : 表示廠商;
- ② model : 一般是模塊對應(yīng)的驅(qū)動名字。
例如:
- compatible = "fsl,mpc8641", "ns16550";
上面的compatible有兩個屬性,分別是 "fsl,mpc8641" 和 "ns16550";其中 "fsl,mpc8641" 的廠商是 fsl;設(shè)備首先會使用第一個屬性值在 Linux 內(nèi)核里面查找,看看能不能找到與之匹配的驅(qū)動文件;
如果沒找到,就使用第二個屬性值查找,以此類推,直到查到到對應(yīng)的驅(qū)動程序 或者 查找完整個 Linux 內(nèi)核也沒有對應(yīng)的驅(qū)動程序為止。
注:一般驅(qū)動程序文件都會有一個 OF 匹配表,此 OF 匹配表保存著一些 compatible 值,如果設(shè)備節(jié)點的 compatible 屬性值和 OF 匹配表中的任何一個值相等,那么就表示設(shè)備可以使用這個驅(qū)動。
2.2.1.2 model 屬性
model 屬性值也是一個字符串,一般 model 屬性描述設(shè)備模塊信息,比如名字什么的,例如:
- model = "Samsung S3C2416 SoC";
2.2.1.3 phandle 屬性
phandle屬性為devicetree中唯一的節(jié)點指定一個數(shù)字標識符,節(jié)點中的phandle屬性,它的取值必須是唯一的(不要跟其他的phandle值一樣),例如:
- pic@10000000 {
- phandle = <1>;
- interrupt-controller;
- };
- another-device-node {
- interrupt-parent = <1>; // 使用phandle值為1來引用上述節(jié)點
- };
注:DTS中的大多數(shù)設(shè)備樹將不包含顯式的phandle屬性,當DTS被編譯成二進制DTB格式時,DTC工具會自動插入phandle屬性。
2.2.1.4 status 屬性
status 屬性看名字就知道是和設(shè)備狀態(tài)有關(guān)的, status 屬性值也是字符串,字符串是設(shè)備的狀態(tài)信息,可選的狀態(tài)如下表所示:
status值 | 描述 |
---|---|
“okay” | 表明設(shè)備是可操作的。 |
“disabled” | 表明設(shè)備當前是不可操作的,但是在未來可以變?yōu)榭刹僮鞯模热鐭岵灏卧O(shè)備插入以后。至于 disabled 的具體含義還要看設(shè)備的綁定文檔。 |
“fail” | 表明設(shè)備不可操作,設(shè)備檢測到了一系列的錯誤,而且設(shè)備也不大可能變得可操作。 |
“fail-sss” | 含義和“fail”相同,后面的 sss 部分是檢測到的錯誤內(nèi)容 |
2.2.1.5 #address-cells 和 #size-cells
#address-cells 和 #size-cells的值都是無符號 32 位整型,可以用在任何擁有子節(jié)點的設(shè)備中,用于描述子節(jié)點的地址信息。#address-cells 屬性值決定了子節(jié)點 reg 屬性中地址信息所占用的字長(32 位), #size-cells 屬性值決定了子節(jié)點 reg 屬性中長度信息所占的字長(32 位)。#address-cells 和 #size-cells 表明了子節(jié)點應(yīng)該如何編寫 reg 屬性值,一般 reg 屬性都是和地址有關(guān)的內(nèi)容,和地址相關(guān)的信息有兩種:起始地址和地址長度,reg 屬性的格式一為:
- reg = <address1 length1 address2 length2 address3 length3……>
例如一個64位的處理器:
- soc {
- #address-cells = <2>;
- #size-cells = <1>;
- serial {
- compatible = "xxx";
- reg = <0x4600 0x5000 0x100>; /*地址信息是:0x00004600 00005000,長度信息是:0x100*/
- };
- };
2.2.1.6 reg 屬性
reg 屬性的值一般是 (address, length) 對,reg 屬性一般用于描述設(shè)備地址空間資源信息,一般都是某個外設(shè)的寄存器地址范圍信息。
例如:一個設(shè)備有兩個寄存器塊,一個的地址是0x3000,占據(jù)32字節(jié);另一個的地址是0xFE00,占據(jù)256字節(jié),表示如下:
- reg = <0x3000 0x20 0xFE00 0x100>;
注:上述對應(yīng)#address-cells = <1>; #size-cells = <1>;。
2.2.1.7 ranges 屬性
ranges屬性值可以為空或者按照 (child-bus-address,parent-bus-address,length) 格式編寫的數(shù)字矩陣, ranges 是一個地址映射/轉(zhuǎn)換表, ranges 屬性每個項目由子地址、父地址和地址空間長度這三部分組成:
- child-bus-address:子總線地址空間的物理地址,由父節(jié)點的 #address-cells 確定此物理地址所占用的字長。
- parent-bus-address:父總線地址空間的物理地址,同樣由父節(jié)點的 #address-cells 確定此物理地址所占用的字長。
- length:子地址空間的長度,由父節(jié)點的 #size-cells 確定此地址長度所占用的字長。
- soc {
- compatible = "simple-bus";
- #address-cells = <1>;
- #size-cells = <1>;
- ranges = <0x0 0xe0000000 0x00100000>;
- serial {
- device_type = "serial";
- compatible = "ns16550";
- reg = <0x4600 0x100>;
- clock-frequency = <0>;
- interrupts = <0xA 0x8>;
- interrupt-parent = <&ipic>;
- };
- };
節(jié)點 soc 定義的 ranges 屬性,值為 <0x0 0xe0000000 0x00100000>,此屬性值指定了一個 1024KB(0x00100000) 的地址范圍,子地址空間的物理起始地址為 0x0,父地址空間的物理起始地址為 0xe0000000。
serial 是串口設(shè)備節(jié)點,
reg 屬性定義了 serial 設(shè)備寄存器的起始地址為 0x4600,寄存器長度為 0x100。
經(jīng)過地址轉(zhuǎn)換, serial 設(shè)備可以從 0xe0004600 開始進行讀寫操作,0xe0004600=0x4600+0xe0000000。
2.2.1.8 name 屬性
name 屬性值為字符串, name 屬性用于記錄節(jié)點名字, name 屬性已經(jīng)被棄用,不推薦使用name 屬性,一些老的設(shè)備樹文件可能會使用此屬性。
2.2.1.9 device_type 屬性
device_type 屬性值為字符串, IEEE 1275 會用到此屬性,用于描述設(shè)備的 FCode,但是設(shè)備樹沒有 FCode,所以此屬性也被拋棄了。此屬性只能用于 cpu 節(jié)點或者 memory 節(jié)點。
- memory@30000000 {
- device_type = "memory";
- reg = <0x30000000 0x4000000>;
- };
2.2.2 根節(jié)點
每個設(shè)備樹文件只有一個根節(jié)點,其他所有的設(shè)備節(jié)點都是它的子節(jié)點,它的路徑是 /。根節(jié)點有以下屬性:
屬性 | 屬性值類型 | 描述 |
---|---|---|
#address-cells | < u32 > | 在它的子節(jié)點的reg屬性中, 使用多少個u32整數(shù)來描述地址(address) |
model | < string > | 用于標識系統(tǒng)板卡(例如smdk2440開發(fā)板),推薦的格式是“manufacturer,model-number” |
compatible | < stringlist > | 定義一系列的字符串, 用來指定內(nèi)核中哪個machinedesc可以支持本設(shè)備 |
例如:compatible = "samsung,smdk2440","samsung,s3c24xx" ,內(nèi)核會優(yōu)先尋找支持smdk2440的machinedesc結(jié)構(gòu)體,如果找不到才會繼續(xù)尋找支持s3c24xx的machine_desc結(jié)構(gòu)體(優(yōu)先選擇第一項,然后才是第二項,第三項……)
2.2.3 特殊節(jié)點
2.2.3.1 /aliases 子節(jié)點
aliases 節(jié)點的主要功能就是定義別名,定義別名的目的就是為了方便訪問節(jié)點。
例如:定義 flexcan1 和 flexcan2 的別名是 can0 和 can1。
- aliases {
- can0 = &flexcan1;
- can1 = &flexcan2;
- };
2.2.3.2 /memory 子節(jié)點
所有設(shè)備樹都需要一個memory設(shè)備節(jié)點,它描述了系統(tǒng)的物理內(nèi)存布局。如果系統(tǒng)有多個內(nèi)存塊,可以創(chuàng)建多個memory節(jié)點,或者可以在單個memory節(jié)點的reg屬性中指定這些地址范圍和內(nèi)存空間大小。
例如:一個64位的系統(tǒng)有兩塊內(nèi)存空間:RAM1:起始地址是0x0,地址空間是 0x80000000;RAM2:起始地址是0x10000000,地址空間也是0x80000000;同時根節(jié)點下的 #address-cells = <2>和#size-cells = <2>,這個memory節(jié)點描述為:
- memory@0 {
- device_type = "memory";
- reg = <0x00000000 0x00000000 0x00000000 0x80000000
- 0x00000000 0x10000000 0x00000000 0x80000000>;
- };
或者:
- memory@0 {
- device_type = "memory";
- reg = <0x00000000 0x00000000 0x00000000 0x80000000>;
- };
- memory@10000000 {
- device_type = "memory";
- reg = <0x00000000 0x10000000 0x00000000 0x80000000>;
- };
2.2.3.3 /chosen 子節(jié)點
chosen 并不是一個真實的設(shè)備, chosen 節(jié)點主要是為了 uboot 向 Linux 內(nèi)核傳遞數(shù)據(jù),重點是 bootargs 參數(shù)。例如:
- chosen {
- bootargs = "root=/dev/nfs rw nfsroot=192.168.1.1 console=ttyS0,115200";
- };
2.2.3.4 /cpus 和 /cpus/cpu* 子節(jié)點
cpus節(jié)點下有1個或多個cpu子節(jié)點, cpu子節(jié)點中用reg屬性用來標明自己是哪一個cpu,所以 /cpus 中有以下2個屬性:
- #address-cells // 在它的子節(jié)點的reg屬性中, 使用多少個u32整數(shù)來描述地址(address)
- #size-cells // 在它的子節(jié)點的reg屬性中, 使用多少個u32整數(shù)來描述大小(size)
- // 必須設(shè)置為0
例如:
- cpus {
- #address-cells = <1>;
- #size-cells = <0>;
- cpu@0 {
- device_type = "cpu";
- reg = <0>;
- cache-unified;
- cache-size = <0x8000>; // L1, 32KB
- cache-block-size = <32>;
- timebase-frequency = <82500000>; // 82.5 MHz
- next-level-cache = <&L2_0>; // phandle to L2
- L2_0:l2-cache {
- compatible = "cache";
- cache-unified;
- cache-size = <0x40000>; // 256 KB
- cache-sets = <1024>;
- cache-block-size = <32>;
- cache-level = <2>;
- next-level-cache = <&L3>; // phandle to L3
- L3:l3-cache {
- compatible = "cache";
- cache-unified;
- cache-size = <0x40000>; // 256 KB
- cache-sets = <0x400>; // 1024
- cache-block-size = <32>;
- cache-level = <3>;
- };
- };
- };
- cpu@1 {
- device_type = "cpu";
- reg = <1>;
- cache-unified;
- cache-block-size = <32>;
- cache-size = <0x8000>; // L1, 32KB
- timebase-frequency = <82500000>; // 82.5 MHzclock-frequency = <825000000>; // 825 MHz
- cache-level = <2>;
- next-level-cache = <&L2_1>; // phandle to L2
- L2_1:l2-cache {
- compatible = "cache";
- cache-unified;
- cache-size = <0x40000>; // 256 KB
- cache-sets = <0x400>; // 1024
- cache-line-size = <32>; // 32 bytes
- next-level-cache = <&L3>; // phandle to L3
- };
- };
- };
2.2.3.5 引用其他節(jié)點
2.2.3.5.1 phandle
節(jié)點中的phandle屬性, 它的取值必須是唯一的(不要跟其他的phandle值一樣)
- pic@10000000 {
- phandle = <1>;
- interrupt-controller;
- };
- another-device-node {
- interrupt-parent = <1>; // 使用phandle值為1來引用上述節(jié)點
- };
2.2.3.5.2 label
- PIC: pic@10000000 {
- interrupt-controller;
- };
- another-device-node {
- interrupt-parent = <&PIC>; // 使用label來引用上述節(jié)點,
- // 使用lable時實際上也是使用phandle來引用,
- // 在編譯dts文件為dtb文件時, 編譯器dtc會在dtb中插入phandle屬性
- };
2.2.4 DTB格式
.dtb文件是 .dts 被 DTC 編譯后的二進制格式的設(shè)備樹文件,它的文件布局如下:
從上圖可以看出,DTB文件主要包含四部分內(nèi)容:struct ftdheader、memory reservation block、structure block、strings block;
① struct ftdheader:用來表明各個分部的偏移地址,整個文件的大小,版本號等;
② memory reservation block:在設(shè)備樹中使用/memreserve/ 定義的保留內(nèi)存信息;
③ structure block:保存節(jié)點的信息,節(jié)點的結(jié)構(gòu);
④ strings block:保存屬性的名字,單獨作為字符串保存;
struct ftd_header結(jié)構(gòu)體的定義如下:
- struct fdt_header {
- uint32_t magic; /*它的值為0xd00dfeed,以大端模式保存*/
- uint32_t totalsize; /*整個DTB文件的大小*/
- uint32_t off_dt_struct; /*structure block的偏移地址*/
- uint32_t off_dt_strings; /*strings block的偏移地址*/
- uint32_t off_mem_rsvmap; /*memory reservation block的偏移地址*/
- uint32_t version; /*設(shè)備樹版本信息*/
- uint32_t last_comp_version; /*向后兼容的最低設(shè)備樹版本信息*/
- uint32_t boot_cpuid_phys; /*CPU ID*/
- uint32_t size_dt_strings; /*strings block的大小*/
- uint32_t size_dt_struct; /*structure block的大小*/
- };
fdtreserveentry結(jié)構(gòu)體如下:
- struct fdt_reserve_entry {
- uint64_t address; /*64bit 的地址*/
- uint64_t size; /*保留的內(nèi)存空間的大小*/
- };
該結(jié)構(gòu)體用于表示memreserve的起始地址和內(nèi)存空間的大小,它緊跟在struct ftdheader結(jié)構(gòu)體后面。
例如:/memreserve/ 0x33000000 0x10000,fdtreserve_entry 結(jié)構(gòu)體的成員 address = 0x33000000,size = 0x10000。
structure block是用于描述設(shè)備樹節(jié)點的結(jié)構(gòu),保存著節(jié)點的信息、節(jié)點的結(jié)構(gòu),它有5種標記類型:
① FDTBEGINNODE (0x00000001):表示節(jié)點的開始,它的后面緊跟的是節(jié)點的名字;
② FDTENDNODE (0x00000002):表示節(jié)點的結(jié)束;
③ FDTPROP (0x00000003) :表示開始描述節(jié)點里面的一個屬性,在FDTPROP后面緊跟一個結(jié)構(gòu)體如下所示:
- struct {
- uint32_t len; /*表示屬性值的長度*/
- uint32_t nameoff; /*屬性的名字在string block的偏移*/
- }
注:上面的這個結(jié)構(gòu)體后緊跟著是屬性值,屬性的名字保存在字符串塊(Strings block)中。
④ FDT_END (0x00000009):表示structure block的結(jié)束。
單個節(jié)點在structure block的存儲格式如下圖如所示:(注:子節(jié)點的存儲格式也是一樣)
總結(jié):
(1) DTB文件可以分為四個部分:struct ftdheader、memory reservation block、structure block、strings block;
(2) 最開始的為struct ftdheader,包含其它三個部分的偏移地址;
(3) memory reservation block記錄保留內(nèi)存信息;
(4) structure block保存節(jié)點的信息,節(jié)點的結(jié)構(gòu);
(5) strings block保存屬性的名字,將屬性名字單獨作為字符串保存。
2.2.5 DTB文件分析
編譯生成dtb文件的源設(shè)備樹jz2440.dts文件如下:
- // SPDX-License-Identifier: GPL-2.0
- /*
- * SAMSUNG SMDK2440 board device tree source
- *
- * Copyright (c) 2018 weidongshan@qq.com
- * dtc -I dtb -O dts -o jz2440.dts jz2440.dtb
- */
- #define S3C2410_GPA(_nr) ((0<<16) + (_nr))
- #define S3C2410_GPB(_nr) ((1<<16) + (_nr))
- #define S3C2410_GPC(_nr) ((2<<16) + (_nr))
- #define S3C2410_GPD(_nr) ((3<<16) + (_nr))
- #define S3C2410_GPE(_nr) ((4<<16) + (_nr))
- #define S3C2410_GPF(_nr) ((5<<16) + (_nr))
- #define S3C2410_GPG(_nr) ((6<<16) + (_nr))
- #define S3C2410_GPH(_nr) ((7<<16) + (_nr))
- #define S3C2410_GPJ(_nr) ((8<<16) + (_nr))
- #define S3C2410_GPK(_nr) ((9<<16) + (_nr))
- #define S3C2410_GPL(_nr) ((10<<16) + (_nr))
- #define S3C2410_GPM(_nr) ((11<<16) + (_nr))
- /dts-v1/;
- / {
- model = "SMDK2440";
- compatible = "samsung,smdk2440";
- #address-cells = <1>;
- #size-cells = <1>;
- memory@30000000 {
- device_type = "memory";
- reg = <0x30000000 0x4000000>;
- };
- chosen {
- bootargs = "console=ttySAC0,115200 rw root=/dev/mtdblock4 rootfstype=yaffs2";
- };
- led {
- compatible = "jz2440_led";
- reg = <S3C2410_GPF(5) 1>;
- };
- };
jz2440.dtb 文件的內(nèi)容如下:
接下來我們對應(yīng)上圖的編號逐一分析,其中編號①~⑩表示的是fdtheader 結(jié)構(gòu)體的成員信息:
① 對應(yīng) magic,表示設(shè)備樹魔數(shù),固定為0xd00dfeed;
② 對應(yīng) totalsize,表示整個設(shè)備設(shè)dtb文件的大小,從上圖可知0x000001B9正好是dtb文件的大小441B;
③ 對應(yīng) offdtstruct,表示structure block的偏移地址,為 0x00000038;
④ 對應(yīng)offdtstrings,表示 strings block的偏移地址,為 0x00000174;
⑤ 對應(yīng) offmemrsvmap;,表示memory reservation block的偏移地址,為 0x00000028;
⑥ 對應(yīng) version ,設(shè)備樹版本的版本號為0x11;
⑦ 對應(yīng) lastcompversion,向下兼容版本號0x10;
⑧ 對應(yīng) bootcpuidphys,在多核處理器中用于啟動的主cpu的物理id,為0x0;
⑨ 對應(yīng) sizedtstrings,strings block的大小為 0x45;
⑩ 對應(yīng) sizedtstruct,structure block的大小為 0x0000013C;
⑪~⑫ 對應(yīng)結(jié)構(gòu)體 fdtreserve_entry ,它所在的地址為0x28,jz2440.dts 設(shè)備樹文件沒有設(shè)置 /memreserve/,所以address = 0x0,size = 0x0;
⑬ 所處的地址是0x38,它處在structure block中,0x00000001表示的是設(shè)備節(jié)點的開始;
⑭ 接著緊跟的是設(shè)備節(jié)點的名字,這里是根節(jié)點,所以為0x00000000;
⑮ 0x00000003 表示的是開始描述設(shè)備節(jié)點的一個屬性;
⑯ 表示這個屬性值的長度為0x09;
⑰ 表示這個屬性的名字在strings block的偏移量是0,找到strings block的地址0x0174的地方,可知這個屬性的名字是model;
⑱ 這個model屬性的值是"SMDK2440",加上字符串的結(jié)束符NULL,剛好9個字節(jié)。
2.2.6 DTB文件結(jié)構(gòu)圖
(1) dtb 文件的結(jié)構(gòu)圖如下:
Linux設(shè)備樹語法規(guī)范
(2) 設(shè)備節(jié)點的結(jié)構(gòu)圖如下: