Linux 內核:設備樹(1)dtb格式
背景
dtb作為二進制文件被加載到內存中,然后由內核讀取並進行解析,如果對dtb文件的格式不了解,那么在看設備樹解析相關的內核代碼時將會寸步難行,而閱讀源代碼才是了解設備樹最好的方式。
所以,如果需要更透徹的了解設備樹解析的細節,第一步就是需要了解設備樹的格式。
注:本文部分參考:官方文檔
dtb的由來
設備樹的一般操作方式是:開發人員根據開發需求編寫dts文件,然后使用dtc將dts編譯成dtb文件。
dts文件是文本格式的文件,而dtb是二進制文件,在linux啟動時被加載到內存中,接下來我們需要來分析設備樹dtb文件的格式。
dtb格式總覽
dtb的格式是這樣的:
dtb header
但凡涉及到數據的記錄,就一定會有一個總的描述部分,就像磁盤的超級塊,書的目錄,dtb當然也不例外,這個描述頭部就是dtb的header部分,通過這個header部分,用戶可以快速地了解到整個dtb的大致信息。
header可以用這么一個結構體來描述:
struct fdt_header {
fdt32_t magic; /* magic word FDT_MAGIC */
fdt32_t totalsize; /* total size of DT block */
fdt32_t off_dt_struct; /* offset to structure */
fdt32_t off_dt_strings; /* offset to strings */
fdt32_t off_mem_rsvmap; /* offset to memory reserve map */
fdt32_t version; /* format version */
fdt32_t last_comp_version; /* last compatible version */
/* version 2 fields below */
fdt32_t boot_cpuid_phys; /* Which physical CPU id we're
booting on */
/* version 3 fields below */
fdt32_t size_dt_strings; /* size of the strings block */
/* version 17 fields below */
fdt32_t size_dt_struct; /* size of the structure block */
};
magic
設備樹的魔數,魔數其實就是一個用於識別的數字,表示設備樹的開始,linux dtb的魔數為 0xd00dfeed
.
totalsize
這個設備樹的size,也可以理解為所占用的實際內存空間。
off_dt_struct
offset to dt_struct,表示整個dtb中structure部分所在內存相對頭部的偏移地址
off_dt_strings
offset to dt_string,表示整個dtb中string部分所在內存相對頭部的偏移地址
off_mem_rsvmap
offset to memory reserve map,dtb中memory reserve map所在內存相對頭部的偏移地址,
version
設備樹的版本,截至目前的最新版本為17.
last_comp_version
最新的兼容版本
boot_cpuid_phys
這部分僅在版本2中存在,后續版本不再使用。
size_dt_strings
表示整個dtb中string部分的大小
size_dt_struct
表示整個dtb中struct部分的大小
alignment gap
中間的alignment gap部分表示對齊間隙,它並非是必須的,它是否被提供以及大小由具體的平台對數據對齊和的要求以及數據是否已經對齊來決定。
memory reserve map
memory reserve map:描述保留的內存部分,這個map的數據結構是這樣的:
{
uint64_t physical_address;
uint64_t size;
}
這部分存儲了此結構的列表,整個部分的結尾由一個數據為0的結構來表示(即physical_address和size都為0,總共16字節)。
這一部分的數據並非是節點中的memory子節點,而是在設備開始之前(也就是第一個花括號之前)定義的,例如:
/dts-v1/
/memreserve/ 0x10000000 0x100000
/*在結構提中的表示為 physical_address=0x10000000,size=0x100000 */
{
...
}
這一部分的作用是告訴內核哪一些內存空間需要被保留而不應該被系統覆蓋使用,因為在內核啟動時常常需要動態申請大量的內存空間,只有提前進行注冊,用戶需要使用的內存才不會被系統征用而造成數據覆蓋。
值得一提的是,對於設備樹而言,即使不指定保留內存,系統也會默認為設備樹保留相應的內存空間。
同時,這一部分需要64位(8字節)對齊。
device-tree structure
device-tree structure:每個節點都會被描述為一個struct,節點之間可以嵌套,因此也會有嵌套的struct。
structure的的結構是這樣的:
- 一個node開始信號,OF_DT_BEGIN_NODE,內容為:
0x00000001
- 對於版本1-3而言,這一部分是節點的全路徑,以/開頭,而對於版本16及以上,這部分只是unit name(root 除外,它沒有unit name),unit name是以0結尾的字符串
- 可選的對齊字節
- 對於每個屬性字段:
- 由OF_DT_PROP標識,數據為
0x00000003
- 32位的數據,表示屬性的size
- 32位的數據,表示屬性名在string block中的偏移地址
- 屬性中的value data.
- 由OF_DT_PROP標識,數據為
- 如果有子節點,遞歸地對子節點進行描述。
- 節點結束信號,OF_DT_END_NODE ,數據為
0x00000002
.
每個節點的信息都按照上述結構被描述,需要注意的是,所有用於描述一個特定節點的屬性都必須在任何子節點之前定義,雖然設備樹的層次結構不會因此產生二義性,但是linux kernel的解析程序要求這么做。
device-tree strings
device-tree strings:在dtb中有大量的重復字符串,比如"model","compatile"等等,為了節省空間,將這些字符串統一放在某個地址,需要使用的時候直接使用索引來查看。
需要注意的是,屬性部分格式為key = value,key部分被放置在strings部分,而value部分的字符串並不會放在這一部分,而是直接放在structure中。
dtb文件解析示例
光說不練假把式,下面我就使用一個簡單的示例來剖析dtb的文件格式。
下述示例僅僅是一個演示demo,不針對任何平台,為了演示方便,編寫了一個非常簡單的dts文件。
/dts-v1/;
/ {
compatible = "hd,test_dts", "hd,test_xxx";
#address-cells = <0x1>;
#size-cells = <0x1>;
model = "HD test dts";
chosen {
stdout-path = "/ocp/serial@ffff";
};
memory@80000000 {
device_type = "memory";
reg = <0x80000000 0x10000000>;
};
led1:led@2000000 {
compatible = "test_led";
#address-cells = <0x1>;
#size-cells = <0x1>;
reg = <0x200 0x4>;
};
};
編譯當前dts文件,獲取對應的dtb文件。
鑒於dtb文件為二進制文件,普通編輯器打開顯示亂碼,我們使用ultraEdit查看,它將數據以16進制形式顯示:
整個dtb文件還是比較簡單的,圖中的紅色框出的部分為header部分的數據,可以看到:
第1個四字節對應magic,數據為 D00DFEED.
第2四字節對應totalsize,數據為 000001BC,可以由整張圖片看出,這個dtb文件的大小由0x0~0x1bb,大小剛好是0x1bc
第3個四字節對應off_dt_struct,數據為00000038。
第4個四字節對應off_dt_strings,數據為00000174,可以由整張圖片看到,從0x174開始剛好是字符串開始的地方
第5個四字節對應off_mem_rsvmap,數據為00000028
第6個四字節對應version,數據為00000011,十進制為17
第7個四字節對應last_comp_version,數據為00000010,十進制為16,表示兼容版本16
第8個四字節對應boot_cpuid_phys,數據為00000000,僅在版本2中使用,這里為0
第9個四字節對應size_dt_strings,數據為00000048,表示字符串總長。
第10個四字節對應size_dt_struct,數據為0000013c,表示struct部分總長度。
整個頭部為40字節,16進制為0x28,從頭部信息中off_mem_rsvmap部分可以得到,reserve memory起始地址為0x28,上文中提到,這一部分使用一個16字節的struct來描述,以一個全為0的struct結尾。
后16字節全為0,可以看出,這里並沒有設置reserve memory。
structure 部分
上文回顧:每一個屬性都是以 key = value的形式來描述,value部分可選。
偏移地址來到0x00000038(0x28+0x10),接下來8個字節為00000003,根據上述structure中的描述,這是OF_DT_PROP,即標示屬性的開始。
接下來4字節為00000018,表明該屬性的value部分size為24字節。
接下來4字節是當前屬性的key在string 部分的偏移地址,這里是00000000,由頭部信息中off_dt_strings可以得到,string部分的開始為00000174,偏移地址為0,所以對應字符串為"compatible".
之后就是value部分,這部分的數據是字符串,可以直接從圖片右側欄看出,總共24字節的字符串"hd,test_dts", "hd,test_xxx",因為字符串之間以0結尾,所以程序可以識別出這是兩個字符串。
可以看出,到這里,compatible = "hd,test_dts", "hd,test_xxx";這個屬性就被描述完了,對於屬性的描述還是非常簡單的。
按照固有的規律,接下來就是對#address-cells = <0x1>的解析,然后是#size-cells = <0x1>...
然后就是遞歸的子節點chosen,memory@80000000等等都是按照上文中提到的structure解析規則來進行解析,最后以00000002結尾。
與根節點不同的是,子節點有一個unit name,即chosen,memory@80000000這些名稱,並非節點中的.name屬性。
而整個結構的結束由00000009來描述。
一般而言,在32位系統中,dtc在編譯dts文件時會自動考慮對齊問題,所以對於設備樹的對齊字節,我們只需要有所了解即可,並不會常接觸到。
好了,關於linux設備樹dtb文件格式的討論就到此為止啦,如果朋友們對於這個有什么疑問或者發現有文章中有什么錯誤,歡迎留言
關於linux設備樹在內核啟動時的解析可以參考我的另外兩篇博客:
linux設備樹的解析--dtb轉換成device_node
linux設備樹的解析--device_node轉換成platform_deviec
原創博客,轉載請注明出處!
祝各位早日實現項目叢中過,bug不沾身.