大家好,又见面了,我是你们的朋友全栈君。
众所周知操作系统一直在不断的更新和发展,而在Linux驱动的架构上面也是不断的进步和完善。在早期的Linux内核和ARM架构中并没有采用设备树。在没有设备树的时候Linux是通过大量的arch/arm/mach-xxx 和arch/arm/plat-xxx文件夹来描述对应平台的板机信息。而随着智能终端设备,智能手机的发展,每年新出的ARM架构芯片都有数百款,从而导致Linux内核中的板机信息文件过多,使得Linux内核虚胖。 当 Linux之父 linus看到 ARM社区向 社区向 Linux内核添加了大量“无用”、冗余的板级信息文件,不禁发出了一句“ This whole ARM thing is a f*cking pain in the ass”。从此以后 ARM社区就引入了PowerPC等架构已经采用的设备树(Flattened Device Tree),将这些描述板机硬件信息的内容都从Linux中分离出来,用一个专属的文件格式来描述,这个专属的文件就叫做设备树。 这个用这个通用的文件就是.dtsi文件,类似于C语言中的头文件。一般用.dts描述板机信息(也就是开发板上有多少个IIC设备、SPI设备等),dtsi描述SOC级信息(也就是SOC有几个CPU、主频是多少、多少个外设控制寄存器信息等)。
设备树(Device Tree),将这个词分开就是“设备”和“树”,描述设备设备树的文件叫做DTS(Device Tree Source),这个DTS文件采用了树形结构来描述板机设备,也就是开发板信息,比如CPU数量、内存基地址、IIC接口上接了那些设备、SPI接口上接了那些设备等。如最开始的图片所示! 在图片中,树的主干就是系统总线,IIC控制器、SPI控制器等都是接到系统主线的分支上的。通过DTS这个文件描述设备信息是有相关的语法规则的,并且在Linux内核中只有3.x版本以后的才支持设备树。
设备树源文件扩展名为.dts, 之前我跟着正点原子的教程时一直使用的是.dtb文件,这两个文件的关系是什么呢?其实DTS是设备树源码文件,DTB是将DTS编译以后得到的一个二进制文件。在Linux中将.c文件编译成.o文件需要用到gcc编译器,那么将 ** .dts编译为.dtb需要用到的工具就是DTC工具**!而这个.dtb文件就是UBOOT通过bootz或者bootm命令向Linux内核中传递的二进制设备树文件(.dtb))。
虽然在平时工作中我们基本上不会从头到尾写一个.dts文件,但是我作为学生来学习这么一个知识可以尝试着学习一下.dts的基本语法,同时也为以后找工作可能修改SOC原厂提供的.dts文件上进行修改。DTS其实是一种ASCII文本文件,不论是阅读还是修改都相对比较方便。
和C语言一样,设备树也支持头文件,设备树的头文件扩展名为.dtsi。在imx6ull-alientek-emmc.dts中有以下内容:
12 #include <dt-bindings/input/input.h> //引用了“input.h”这个.h头文件
13 #include "imx6ull.dtsi" //引用.dtsi头文件
通过以上代码可以看出在.dtsi
文件中可以直接通过include来引用.h
、.dtsi
、.dts
。一般的.dtsi用于描述SOC的内部外设信息,比如CPU架构、主频、外设寄存器地址范围,比如UART、IIC等等。
设备树采用树形结构来描述板子上的设备信息的文件,每一个设备都是一个节点,叫做设备节点,每个节点都是通过一些属性信息来描述节点信息,属性就是键值对。以下是借鉴正点原子驱动开发手册的从imx6ull.dtsi文件中缩减出来的设备树文件内容:
/ {
aliases {
can0 = &flexcan1;
};
cpus {
#address-cells = <1>;
#size-cells = <0>;
cpu0: cpu@0 {
compatible = "arm,cortex-a7";
device_type = "cpu";
reg = <0>;
};
};
intc: interrupt-controller@00a01000 {
compatible = "arm,cortex-a7-gic";
#interrupt-cells = <3>;
interrupt-controller;
reg = <0x00a01000 0x1000>,
<0x00a02000 0x100>;
};
}
node-name@unit-address
node-name
是节点的名字,为ASCII字符串,节点名字应该能够清晰的辨别出节点的功能,比如uart1
就表示这个节点是UART1
外设。unit-address一般表示设备的地址或寄存器首地址,如果某个节点没有地址或者寄存器的话unit-address
可以不要,比如cpu@0
、interrupt-controller@00a01000
cpu0:cpu
@0并不是node-name@unit-address
这样的格式,而是用:
隔开成了两部分,:
前面是节点标签,:
后面是节点名字,格式为lable:node-name@unit-address
,引入label的目的就是为了方便访问节点,可以直接通过&label来访问这个节点,比如通过&cpu0就可以访问cpu@0这个节点,而不需要输入完整的节点名字。数据形式 | 实现方式 | 详细描述 |
---|---|---|
字符串 | compatible = "arm,cortex-a7; | 设置compatible属性的值为字符串arm,cortex-a7 |
32位无符号整数 | reg=<0> | 设置reg的值也可以设置为一组值reg=<0 0x123456 100>; |
字符串列表 | compatible = "fsl,imx6ull-gpmi-nand", "fsl, imx6ul-gpmi-nand"; | 字符串与字符串之间用,隔开 |
节点是由一堆属性组成,节点都是具体的设备,不同的设备需要的属性不同,用户可以自定义属性。除了用户自定义的属性,有很多属性是标准属性,Linux下的很多外设驱动都会使用这些标准属性。
compatible属性也叫做“兼容性”属性,这是一个非常重要的属性!compatible属性的值是一个字符串列表,compatibel属性用于将设备和驱动绑定起来。字符串列表用于选择设备所要使用的驱动程序,compatible属性格式如下:
"manufacturer,model"
其中的manufacturer表示厂商,model一般是模块对应驱动的名字。I.MX6U-ALPHA开发板上的音频芯片采用的欧圣出品的WM8960,sound节点的compatible属性如下:
compatible = "fsl,imx6ul-evk-wm8960","fsl,imx-audio-wm8960";
可以看出属性值有两个,分别是”fsl,imx6ul-evk-wm8960″和”fsl,imx-audio-wm8960″,其中的fsl表示厂商是飞思卡尔,imx6ul-evk-wm8960
和imx-audio-wm8960
表示驱动模块的名字。sound这个设备首先使用第一个兼容值再Linux内核中查找,查看能否找到对应的驱动文件,如果没有找到的话就使用第二个兼容值查找,直到找到或者查找玩整个Linux内核也没有找到对应的驱动。
一般程序驱动程序文件都会有一个OF匹配表,此OF匹配表保存着一些compatible值,如果设备节点的compatible属性值和OF匹配表中的任何一个值相等,那么就表示设备可以使用这个驱动。
model=”wm8960-audio";
值 | 描述 |
---|---|
okay | 表明设备是可操作的 |
disabled | 表明设备当前是不可操作的,但是在未来可以变为可操作的,比如热插拔设备插入以后。至于disabled的具体含义还要看设备绑定文档 |
fail | 表明设备不可操作,设备检测到了一系列的错误,而且设备也大可能变得可操作 |
fail-sss | 含义和fail相同,后面的sss部分是检测到的错误内容 |
reg = <address1 length1 address2 length2 address3 length3…… >
每个address length组合表示一个地址范围,其中address是起始地址,length是地址长度,#address-cells表明address这个数据所占用的字长,#size-cells表明length这个数据所占用的字长,比如:spi4 {
compatible = "spi-gpio";
#address-cells = <1>; #说明了spi4的子节点reg属性中起始地址所占用的字长为1
#size-cells = <0>; #地址长度所占用的字长为0
gpio_spi: gpio_spi@0 {
#reg属性值为0,因为父节点中设置了相关的值,继承父节点起始地址,没有设置地址长度
compatible = "fairchild,74hc595";
reg = <0>;
};
};
aips3: aips-bus@02200000 {
compatible = "fsl,aips-bus", "simple-bus";
#address-cells = <1>;
#size-cells = <1>;
dcp: dcp@02280000 {
compatible = "fsl,imx6sl-dcp";
reg = <0x02280000 0x4000>; #address=0X02280000,length=0X4000
};
};
如果ranges属性值为空值,说明子地址空间和父地址空间完全相同,不需要进行地址转换。
cpu0: cpu@0 {
compatible = "arm,cortex-a7";
device_type = "cpu";
reg = <0>;
......
};
每个节点都有compatible属性,根节点/
也不例外,imx6ull-alientek-emmc.dts文件中根节点的compatible属性内容如下:
/{
model = "Freescale i.MX6 ULL 14x14 EVK Board";
compatible = "fsl,imx6ull-14x14-evk", "fsl,imx6ull";
......
}
可以看出根节点的compatible属性可以知道我们所使用的设备,一般第一个值描述了所使用的硬件设备名字,使用某个设备第二个值就是描述设个设备所使用的SOC,如果使用的是imx6ull
这颗SOC。Linux内核会通过根节点的compoatible属性查看是否支持此设备,如果支持这个设备的话设备就会启动Linux内核。
在没有使用设备树之前,uboot会向Linux内核传递一个叫machine id的值,machine id也就是设备ID,告诉Linux内核自己是一个什么设备,看看Linux内核是否支持。具体实现就是判断machine id这个参数是否与代码中的宏MACH_TYPE_XXX进行对比,看有没有相等的,如果相等的话就表示Linux内核支持这个设备,如果不支持的话那么这个设备就没法启动Linux内核。
当Linux内核引入设备树以后就不在使用MACHINE_START了,而是换为了DT_MACHINE_START。说明引入了设备树以后就不会根据machine id来检查Linux 内核是否支持这个设备。在Linux内核中通过start_kernel函数启动内核,然后start_kernel函数会调用setup_arch函数来匹配machine_desc,然后再调用setup_machine_fdt函数进一步获取匹配的machine_desc,这个函数的参数就是atags的首地址(也就是uboot传递给Linux内核的dtb文件首地址),setup_machine_fdt函数返回值就是找到的最匹配的machine_desc。然后通过of_get_flat_dt_root获取设备树的根节点。
发布者:全栈程序员栈长,转载请注明出处:https://javaforall.cn/149413.html原文链接:https://javaforall.cn