驱动进化之路:设备树的引入及简明教程

驱动进化之路:设备树的引入及简明教程

设备树的基本概念和产生背景

问题1:
以LED为例,当要更换LED所用的GPIO引脚时,需要修改驱动程序源码,重新编译驱动,重新加载驱动。
问题2:
由于芯片种类繁多,里面包含了各种各样的芯片所有包含了许多用不着的C代码。
根据上面两个问题:引入设备树
设备树只是用来给内核里的驱动程序,指定硬件的信息。比如 LED 驱动,在
内核的驱动程序里去操作寄存器,但是操作哪一个引脚?这由设备树指定。

设备树的语法

驱动进化之路:设备树的引入及简明教程_第1张图片
我们需要编写设备树文件(dts: device tree source),它需要编译为
dtb(device tree blob)文件,内核使用的是 dtb 文件。
驱动进化之路:设备树的引入及简明教程_第2张图片

内核对设备树的处理过程

请添加图片描述
1.dts在PC机上被编译为dtb文件
2.u-boot把dtb文件传给内核
3.内核解析dtb文件,把每个节点都转化为device_node结构体
4.对于某些device_node结构体,会被转换为platform_device
dtb中的每一个节点都被转换为device_node结构体
驱动进化之路:设备树的引入及简明教程_第3张图片
根节点被保存在全局变量of_root中,从of_root开始可以访问到任意节点

哪些设备树节点会被转换为platform_device

1.根节点下含有 compatile 属性的子节点
2.含有特定 compatile 属性的节点的子节点
如果一个节点的 compatile 属性,它的值是这 4 者之一:“simplebus”,“simple-mfd”,“isa”,“arm,amba-bus”, 那 么 它 的 子结点 ( 需 含
compatile 属性)也可以转换为 platform_device。
3.总线 I2C、SPI 节点下的子节点:不转换为 platform_device
某个总线下到子节点,应该交给对应的总线驱动程序来处理, 它们不应该被转换为 platform_device。
驱动进化之路:设备树的引入及简明教程_第4张图片
⚫/mytest 会被转换为 platform_device, 因为它兼容"simple-bus";
它的子节点/mytest/mytest@0 也会被转换为 platform_device
⚫ /i2c 节点一般表示 i2c 控制器, 它会被转换为 platform_device, 在内核
中有对应的 platform_driver;
⚫ /i2c/at24c02 节点不会被转换为 platform_device, 它被如何处理完全由
父节点的 platform_driver 决定, 一般是被创建为一个 i2c_client。
⚫ 类似的也有/spi 节点, 它一般也是用来表示 SPI 控制器, 它会被转换为
platform_device, 在内核中有对应的 platform_driver;
⚫ /spi/flash@0 节点不会被转换为 platform_device, 它被如何处理完全由
父节点的 platform_driver 决定, 一般是被创建为一个 spi_device。

怎么转换为 platform_device

内核处理设备树的函数调用过程,这里不去分析;我们只需要得到如下结论:
◼ platform_device 中含有 resource 数组, 它来自 device_node 的
reg, interrupts 属性;
◼ platform_device.dev.of_node 指向 device_node, 可以通过它获
得其他属性

platform_device 如何与 platform_driver 配对

从设备树转换得来的 platform_device 会被注册进内核里,以后当我们每
注册一个 platform_driver 时,它们就会两两确定能否配对,如果能配对成功
就调用 platform_driver 的 probe 函数。
驱动进化之路:设备树的引入及简明教程_第5张图片

1.最先比较:是否强制选择某个driver

platform_device.driver_override 和 platform_driver.driver.name
可以设置 platform_device 的 driver_override,强制选择某个 platform_driver

2.然后比较:设备树信息

platform_device.dev.of_node 和 platform_driver.driver.of_match_table。
由设备树节点转换得来的 platform_device 中,含有一个结构体:of_node
结构如下:
驱动进化之路:设备树的引入及简明教程_第6张图片
如果一个 platform_driver 支 持 设 备 树 , 它 的
platform_driver.driver.of_match_table 是一个数组,类型如下
驱动进化之路:设备树的引入及简明教程_第7张图片
首先,如果 of_match_table 中含有 compatible 值,就跟 dev 的 compatile
属性比较,若一致则成功,否则返回失败;
其次,如果 of_match_table 中含有 type 值,就跟 dev 的 device_type 属性
比较,若一致则成功,否则返回失败;
最后,如果 of_match_table 中含有 name 值,就跟 dev 的 name 属性比
较,若一致则成功,否则返回失败。
而设备树中建议不再使用 devcie_type 和 name 属性,所以基本上只使用设
备节点的 compatible 属性来寻找匹配的 platform_driver。

接下来比较platform_device_id

比较 platform_device. name 和 platform_driver.id_table[i].name,
id_table 中可能有多项。
platform_driver.id_table 是“platform_device_id”指针,表示该 drv
支持若干个 device,它里面列出了各个 device 的{.name, .driver_data},
其中的“name”表示该 drv 支持的设备的名字,driver_data 是些提供给该
device 的私有数据。

最后比较

platform_device.name 和 platform_driver.driver.name
platform_driver.id_table 可能为空,
这 时 可 以 根 据 platform_driver.driver.name 来 寻 找 同 名 的
platform_device。
驱动进化之路:设备树的引入及简明教程_第8张图片

有些节点不会生成 platform_device,怎么访问它们

内核会把 dtb 文件解析出一系列的 device_node 结构体,我们可以直接访
问这些 device_node。

你可能感兴趣的:(Liunx开发板,单片机,物联网,stm32)