Linux DTS(Device Tree Source)是一種描述硬件信息的數據結構,主要用于描述嵌入式系統中各個硬件設備的信息,包括設備的地址、中斷、寄存器配置以及設備驅動等。在Linux內核中,DTS文件通常被用來描述板載設備的硬件信息,以便操作系統能夠正確地識別和使用硬件設備。
DTS文件結構
一個典型的DTS文件如下所示:
/dts-v1/;
#include <imx6qdl-pico.dtsi>
#include <imx6qdl-pico-m4.dtsi>
/ {
compatible = "fsl,imx6q-pico", "fsl,imx6q";
model = "Boundary Devices i.MX6 Quad SABRE Lite";
memory {
device_type = "memory";
reg = <0x10000000 0x40000000>;
};
chosen {
compatible = "brcm,bcm2835";
uart_boot = <&uart1>;
};
aliases {
serial0 = &uart1;
};
soc {
compatible = "simple-bus";
#address-cells = <1>;
#size-cells = <1>;
ranges;
gpio: gpio@0209c000 {
compatible = "fsl,imx6ul-gpio";
reg = <0x0209c000 0x1000>;
interrupts = <GPIOn IRQn>;
gpio-controller;
#gpio-cells = <2>;
};
};
uart1: serial@02020000 {
compatible = "fsl,imx6q-uart", "fsl,imx21-uart";
reg = <0x02020000 0x40000>;
interrupts = <78>;
clocks = <&clks 82>;
clock-names = "ipg", "per";
status = "okay";
};
sound {
compatible = "fsl,imx6-sai";
model = "imx6-sai";
status = "okay";
/* SSI1 */
ssi@021d8000 {
compatible = "fsl,imx6-sai";
reg = <0x021d8000 0x4000>;
interrupts = <0 125 0>;
clocks = <&clks 2>;
dmas = <&sdma 9 11 0>, <&sdma 10 11 0>, <&sdma 11 11 0>;
dma-names = "tx", "rx", "mclk";
status = "okay";
};
};
};
登錄后復制
DTS文件內容說明
/dts-v1/: 指定版本為DTS版本1,描述DTS文件的版本信息。#include : 包含其他DTS文件,可復用其定義。/: 根節點,描述整個設備樹結構。compatible: 指定設備兼容性信息。model: 設備型號信息。memory: 描述內存信息。chosen: 描述一些選項信息。aliases: 定義設備別名。soc: 描述SoC相關信息。gpio: 描述GPIO控制器。uart1: 描述UART1硬件信息。sound: 描述聲音設備信息。
如何使用Linux DTS
- 編輯DTS文件:在Linux內核源碼中的
arch/arm/boot/dts/目錄下找到對應平臺的DTS文件(如imx6qdl-pico.dtsi),根據實際硬件信息編輯DTS文件。
編譯DTS文件:在Linux內核源碼根目錄下執行以下命令編譯DTS文件:
make ARCH=arm CROSS_COMPILE=arm-linux-gnueabihf- dtbs
登錄后復制替換設備樹二進制文件:將生成的.dtb文件(設備樹二進制文件)替換到目標設備的引導分區。使用設備樹:在Linux內核啟動時,會加載設備樹文件來描述硬件信息,從而正確識別和配置硬件設備。
代碼示例
#include <linux/module.h>
#include <linux/of_device.h>
#include <linux/platform_device.h>
static int my_driver_probe(struct platform_device *pdev)
{
struct device_node *np = pdev->dev.of_node;
if (!np) {
dev_err(&pdev->dev, "No device tree node found
");
return -ENODEV;
}
// 解析設備樹節點信息
u32 reg;
if (of_property_read_u32(np, "reg", ®)) {
dev_err(&pdev->dev, "Failed to read 'reg' property
");
return -EINVAL;
}
dev_info(&pdev->dev, "Got 'reg' property: %u
", reg);
return 0;
}
static const struct of_device_id my_driver_of_match[] = {
{ .compatible = "my_driver", },
{ },
};
MODULE_DEVICE_TABLE(of, my_driver_of_match);
static struct platform_driver my_driver = {
.probe = my_driver_probe,
.driver = {
.name = "my_driver",
.of_match_table = my_driver_of_match,
.owner = THIS_MODULE,
}
};
module_platform_driver(my_driver);
MODULE_LICENSE("GPL");
MODULE_AUTHOR("Author Name");
MODULE_DESCRIPTION("Sample driver using Device Tree");
登錄后復制
以上是一個簡單的Linux設備驅動程序示例,通過解析設備樹節點中的屬性來配置硬件設備。在probe函數中,首先獲取設備樹節點,然后讀取其中的reg屬性并輸出。在of_device_id中聲明了需要匹配的設備樹節點的兼容性信息,以便驅動程序正確匹配設備。






