Linux3.x 之後引入了設備樹,用於描述一個硬件平臺的板級細節。html
設備樹能夠被 bootloader(uboot)傳遞到內核,內核從中獲取設備樹中的硬件信息。node
設備樹的兩個特色:linux
幾個經常使用的縮寫:框架
設備樹是由 一個根節點 和 多個子節點 組成。函數
/dts-v1/; // 表示版本 [memory reservations] // 格式爲: /memreserve/ <address> <length>; / { [property definitions] [child nodes] };
node爲設備樹中的基本單元。格式爲:工具
[label:] node-name[@unit-address] { [properties definitions] [child nodes] };
0-9 a-z A-Z , . _ + -
組成,且開頭只能是大小寫字母。
/
來表示。@
爲分隔符。
node-name@unit-address
總體同級惟一就是 naem = value。佈局
[label:] property-name;
[label:] property-name = value;
clock-frequency = <0x00000001 0x00000000>;
compatible = "lzm-bus";
local-mac-address = [00 00 12 34 56 78]; // 每一個 byte 使用 2 個 16 進制數來表示 local-mac-address = [000012345678]; // 每一個 byte 使用 2 個 16 進制數來表示
example = <0x84218421 23>, "hello world";
通常設備樹都不須要從零開始寫,只須要包含芯片廠商提供的設備樹模板,而後再添加,修改便可。
dts 能夠包含 dtsi 文件,也能夠包含 .h 文件。.h 文件能夠定義一些宏。flex
/dts-v1/; #include <dt-bindings/input/input.h> #include "imx6ull.dtsi" / { …… };
修改、追加設備樹節點均可在文件末尾或新文件修改或追加。編碼
而修改節點,可參考如下兩種方法:標籤法 或 全路徑法:
標籤法:指針
// 方法一:在根節點以外使用標籤引用節點 &red_led { status = "okay"; } // 方法二:使用全路徑引用節點 &{/led@0x020C406C} { status = "okay"; }
全路徑法:
在節點 {} 中包含的內容時節點屬性。這些屬性信息就是板級硬件描述的信息,驅動會經過內核提供的 API 去獲取這些資源信息。
注意:節點屬性可分爲 標準屬性 和 自定義屬性,便是能夠自行添加屬性。
compatible 屬性:
compatible = "A", "B", "C";
。model 屬性:
model = "lzm com,IMX6U-V1";
status 屬性:
value | description |
---|---|
okay | 設備正常 |
disabled | 設備不可操做,但後面可恢復正常 |
fail | 發生嚴重錯誤,須要修復 |
fail-sss | 發生嚴重錯誤,須要修復。sss 表示錯誤信息 |
#address-cells、#size-cells 屬性:
reg 屬性:
/dts-v1/; / { #address-cells = <1>; #size-cells = <1>; memory { reg = <0x80000000 0x20000000>; }; };
ranges 屬性:
ranges=<0x05 0x10 0x20>
name、ldevice_type 屬性:
名稱及內容可自定義,可是名稱不能與標準屬性重名。獲取方式,後述。
根節點:
CPU:
memory:
chosen:
chosen { bootargs = "noinitrd root=/dev/mtdblock4 rw init=/linuxrc console=ttySAC0,115200"; };
aliases:
aliases { can0 = &flexcan1; gpio0 = &gpio1; }
通常的程序猿會修改設備樹便可,沒必要從零開始。
編譯時須要設置一下三個環境變量 ARCH、CROSS_COMPILE、PATH。
在開發環境中進入板子對應的內核源碼目錄,使用內核中的 makefile 便可,執行以下命令來編譯 dtb 文件:
make dtbs V=1
上述命令是單獨編譯設備樹。
會編譯如下設備樹:
在arch/arm/Makefile 或 arch/arm/boot/Makefile 或 arch/arm/boot/dts/Makefile
等相關 Makefile 中找到 dtb-$(xxx)
,該值包含的就是要編譯的 dtb 。
如該文件中宏 dtb-$(CONFIG_SOC_XXX) 包含的 .dtb 就會被編譯出來。
若是想編譯本身的設備樹,添加該值內容,並把本身的設備樹放在 arch/arm/boot/dts
下便可。
(具體查看該 arch/arm/boot/Makefile 內容)
意思是手工使用 dtc 工具直接編譯。
dtc 工具存放於內核目錄 scripts/dtc 下。
若直接使用 dtc 工具手工編譯的話,包含其它文件時不能使用 #include
,而必須使用 /include
。
#include
是由於使用了 交叉編譯鏈。編譯、反編譯的示例命令以下,-I 指定輸入格式,-O 指定輸出格式,-o 指定輸出文件:
./scripts/dtc/dtc -I dts -O dtb -o tmp.dtb arch/arm/boot/dts/xxx.dts // 編譯 dts 爲 dtb ./scripts/dtc/dtc -I dtb -O dts -o tmp.dts arch/arm/boot/dts/xxx.dtb // 反編譯 dtb 爲 dts
通常步驟:
make dtbs
。目錄 /sys/firmware/devicetree 下以目錄結構呈現的 dtb 文件。
目錄 /sys/firmware/fdt 文件,就是 dtb 格式的設備樹文件。
設備樹生命過程:
DTS --(PC)--> DTB --(內核)--> device_node -·(內核)·-> platform_device。
流程:
對於 device_node 和 platform_device,建議去內核源碼看看它們的成員。
simple-bus
;simple-mfd
;isa
;arm,amba-bus
。在驅動程序中,內核加載設備樹後。能夠經過如下函數獲取到設備樹節點中的資源信息。
獲取節點函數及獲取節點內容函數稱爲 of 函數。
device_node 結構體以下:
struct device_node { const char *name; const char *type; phandle phandle; const char *full_name; struct fwnode_handle fwnode; struct property *properties; struct property *deadprops; /* removed properties */ struct device_node *parent; struct device_node *child; struct device_node *sibling; #if defined(CONFIG_OF_KOBJ) struct kobject kobj; #endif unsigned long _flags; void *data; #if defined(CONFIG_SPARC) const char *path_component_name; unsigned int unique_id; struct of_irq_controller *irq_trans; #endif };
of_device_id 結構體以下:
/* Struct used for matching a device */ struct of_device_id { char name[32]; char type[32]; char compatible[128]; const void *data; };
of_find_node_by_path():
struct device_node *of_find_node_by_path(const char *path)
。of_find_node_by_type():
struct device_node *of_find_node_by_type(struct device_node *from, const char *type)
。of_find_compatible_node():
struct device_node *of_find_compatible_node(struct device_node *from, const char *type, const char *compatible)
。of_find_matching_node_and_match():
struct inline device_node *of_find_matching_node_and_match(struct device_node *from, const struct of_device_id *matches, const struct of_device_id **match)
。of_get_parent():
struct device_node *of_get_parent(const struct device_node *node)
。of_get_child():
struct device_node *of_get_child(const struct device_node *node, struct device_node *prev)
。property:
struct property { char *name; int length; void *value; struct property *next; #if defined(CONFIG_OF_DYNAMIC) || defined(CONFIG_SPARC) unsigned long _flags; #endif #if defined(CONFIG_OF_PROMTREE) unsigned int unique_id; #endif #if defined(CONFIG_OF_KOBJ) struct bin_attribute attr; #endif };
resource 結構體:
struct resource { resource_size_t start; resource_size_t end; const char *name; unsigned long flags; unsigned long desc; struct resource *parent, *sibling, *child; };
of_find_property():
函數原型:struct property *of_find_property(const struct device_node *np,const char *name,int *lenp)
源碼路徑:內核源碼/include/linux/of.h。
np:設備節點。
name:屬性名稱。
lenp:實際得到屬性值的長度(函數輸出參數)。
返回值:
能夠了解下 獲取屬性值函數 of_get_property() ,與 of_find_property() 的區別是一個返回屬性值,一個返回屬性結構體。
of_property_read_u8_array:
//8位整數讀取函數 int of_property_read_u8_array(const struct device_node *np, const char *propname, u8 *out_values, size_t sz) //16位整數讀取函數 int of_property_read_u16_array(const struct device_node *np, const char *propname, u16 *out_values, size_t sz) //32位整數讀取函數 int of_property_read_u32_array(const struct device_node *np, const char *propname, u32 *out_values, size_t sz) //64位整數讀取函數 int of_property_read_u64_array(const struct device_node *np, const char *propname, u64 *out_values, size_t sz)
of_property_read_u8:
//8位整數讀取函數 int of_property_read_u8 (const struct device_node *np, const char *propname,u8 *out_values) //16位整數讀取函數 int of_property_read_u16 (const struct device_node *np, const char *propname,u16 *out_values) //32位整數讀取函數 int of_property_read_u32 (const struct device_node *np, const char *propname,u32 *out_values) //64位整數讀取函數 int of_property_read_u64 (const struct device_node *np, const char *propname,u64 *out_values)
of_property_read_string_index:(推薦)
int of_property_read_string_index(const struct device_node *np,const char *propname, int index, const char **out_string)
of_property_read_string:(不推薦)
int of_property_read_string(const struct device_node *np,const char *propname,const char **out_string)
of_property_read_bool():
static inline bool of_property_read_bool(const struct device_node *np, const char *propname)
設備樹提供寄存器的地址段,可是通常狀況下都會使用 ioremap 映射爲虛擬地址使用。
of_address_to_resource 只是獲取 reg 的值,也就是寄存器值。
of_iomap 函數就是獲取 reg 屬性值&指定哪一段內存&映射爲虛擬地址。
of_address_to_resource:
int of_address_to_resource(struct device_node *dev, int index, struct resource *r)
內核源碼/drivers/of/address.c
。of_iomap:
void __iomem *of_iomap(struct device_node *np, int index)