1、ARM Linux社区为什么要引入设备树

Linux之父Linus Torvalds闲来无事,在翻看ARM Linux代码的时候,有一天终于忍不住了。他在2011年3月17日的ARM Linux邮件列表中说道:“This whole ARM thing is a f*cking pain in the ass”。这句话迫使ARM Linux社区引入了设备树。

Linus Torvalds为什么会发飙呢?而ARM Linux社区的牛人为什么又乖乖地听话了?你得首先理解Linux设备驱动框架中一个非常好的设计:设备信息和驱动分离。

为了说明设备信息和驱动分离的概念,这里用一个简单的模拟代码来解释:

【例-1】实现一个代码,把要使用的信息简单写死在代码中:

int add() /*模拟驱动代码*/

{

return 3+5; /*模拟设备信息*/

}

优点:简单

缺点:一旦加数和被加数发生变化就得改代码

改进设计如下:

【例-2】实现一个代码,把要使用的信息和操作代码分离开来:

struct dev{

int id;

int x;

int y;

}; /*模拟设备信息结构*/

strcut drv{

int id;

int (*add)(struct dev *info);

}; /*模拟驱动结构*/

int add(struct dev *info) /*模拟驱动代码*/

{

return info->x + info->y; /*模拟设备信息-通过参数传递进来*/

}

struct drv drv = {

.id = 1,

.add = add,

};

/*模拟设备信息*/

struct dev dev = {

.id = 1,

.x = 3,

.y = 5,

};

/*模拟总线初始化匹配设备信息和驱动代码*/

int bus()

{

if(dev.id == drv.id){

return drv.add(&dev);

}

...

}

优点:不管加数和被加数怎么变化,不需要修改代码,仅需要修改信息

缺点:结构比较复杂

那这个设备信息和驱动分离的设计跟驱动有什么关系呢?熟悉硬件编程的同学都知道,硬件一般的构成可以使用下图简单表述:

操作外设的驱动代码逻辑,只要硬件是一样的,就不会变化。但是外设挂到不同的主机上,可能会存在I/O地址的变化,如果有中断也是一样的,中断号也可能不同。这些I/O地址和中断号就是设备信息,使用这些信息来操作控制硬件的代码就是驱动。

如果采用【例-1】的设计方式,那么同一个硬件外设接到不同的主机,或是换了地址线/中断线,设备信息就变化了,得去修改驱动。但是采用【例-2】的方式进行设计,问题就迎刃而解:不管同样的外设硬件接到哪里或是那个平台,其驱动代码逻辑并不需要改动,而仅仅需要改变下设备信息,主要的就是I/O地址和中断号。

说了这么半天,跟引入设备树有什么关系呢?华清教学使用的开发板(A8/A9)都使用DM9000网卡芯片。DM9000驱动是开源的,在主线内核源码中就有。我们每次基于A8/A9板子移植的时候,DM9000驱动并没有修改过,仅仅是选配了下,主要的工作是在板级文件中添加了设备信息。DM9000驱动使用的是platform框架,所以添加了一份DM9000网卡芯片的platform_device信息。问题来了,如果使用C代码的形式来描述设备信息,则在内核源码中,将会有多份DM9000的platform_device设备信息,造成了内核代码冗余。

解决这个问题的办法就是引入设备树,改造【例-2】来说明设备树的作用。

【例-3】实现一个代码,不仅把要使用的信息和操作代码分离开来,而且信息不是C代码编写的,而是文本配置文件保存的:

struct dev{

int id;

int x;

int y;

}; /*模拟设备信息结构*/

strcut drv{

int id;

int (*add)(struct dev *info);

}; /*模拟驱动结构*/

int add(struct dev *info) /*模拟驱动代码*/

{

return info->x + info->y; /*模拟设备信息-通过参数传递进来*/

}

struct drv drv = {

.id = 1,

.add = add,

};

/*模拟设备树-一个特殊的配置文件,xxx.dtbs的文本文件*/

/{

......

Dm9000{

x = 3;

y = 5;

};

......

};

/*模拟总线初始化匹配设备信息和驱动代码*/

int bus()

{

/*模拟设备树初始化处理*/

读文件(xxx.dtbs);

解析文件内容(根据设备树的规则来解析);

生成struct dev设备信息;

if(dev.id == drv.id){

return drv.add(&dev);

}

...

}

如果像【例-3】这样,就可以解决大量设备信息的代码冗余问题。

推而广之,系统的软硬件信息都可以使用设备树来描述。这样的话,ARM Linux社区就不会因为支持板子和驱动越来越多造成内核源码中出现很多冗余代码(主要是板级文件),仅仅需要移植者,把系统的软硬件信息通过设备树提供出来,选配一下内核代码,就可以了。

2、设备树的概述

2.1、参考资料

内核源码目录Documentation\devicetree设备树说明文档

内核源码drivers/of/源码分析

2.2、基本概念

设备树是描述软/硬件信息的,包含节点和属性的一个树形结构。节点用以归类描述了一个硬件信息或是软件信息(好比文件系统的目录)。节点内描述了一个或多个属性,属性是键值对,描述具体的软/硬信息。简单形式如下:

/{

node{

property=value;

...

child_node{

child_property=value;

...

};

...

};

...

};

说明如下:

/:根节点,节点使用“{};”的语法描述作用范围

node:根节点下的一个子节点

child_node:node节点下的一个子节点

property:node节点内描述的属性,value就是属性的值(任意字节数据,可以是整型、字符串、数组、等等)。描述行以“;”结束

2.3、存储形式

在《ARM Linux社区为什么要引入设备树》中,已经讨论过设备树的使用方式。简而言之:内核初始化时,以配置的文件形式读取设备树文件的内容,并解析后生成相应的软/硬件信息,以供相应的内核代码使用。

编写设备树文件是以.dts的文本文件存储的,主要是为了修改、添加编辑方便。

那么问题来了,如果纯文本解析的话,显然比较慢且麻烦。譬如如果属性值是一个I/O地址:0x80000000,如果是字符串的形式存储,那么“0x80000000”就是一个字符串,内核代码解析这个信息的时候还得转换成整型数,不仅仅是慢,无形设备树文件大小还会增加不少,还得增加更多的初始化代码。

所以.dts的设备树文件,在内核使用前需要转换一次,主要是把繁复的语法形式及属性值转换成字节数据(特殊的数据结构),而非符号。.dts文件转换后是.dtb的二进制文件。

3、节点

3.1、命名

节点的命名以字母、数字、_、等等符号构成。常见的命令方式如下:

A、以“设备名”为节点名,范例:

DM9000命名如下:

/{

...

dm9000{

...

};

...

};

B、以“设备@I/O地址”为“节点名@I/O地址”,范例:

DM9000在主机端的I/O地址为0x8000 0000,可以命名如下:

/{

...

dm9000@80000000{

...

};

...

};

C、以“设备类型@I/O地址”为“节点名@I/O地址”,范例:

DM9000在主机端的I/O地址为0x8000 0000,可以命名如下:

/{

...

ethernet@80000000{

...

};

...

};

3.2、节点路径

A、

/{

...

dm9000{

...

};

...

};

节点名:dm9000

节点路径:/dm9000

B、

/{

...

dm9000@80000000{

...

};

...

};

节点名:dm9000

节点路径:/dm9000@80000000

C、

/{

...

ethernet@80000000{

...

};

...

};

节点名:ethernet

节点路径:/ethernet@80000000

3.3、节点引用

/{

aliases {

demo = &demo;

};

...

demo:demo@80000000{

...

};

...

};

节点名:demo

节点路径:/demo@80000000

引用路径:demo(等价/demo@80000000,解决路径名过程的问题)

设备树中引用节点“/demo@80000000”的范例:

&demo{

...

};

3.4、节点查找

有时候,分享内核代码或是编写内核代码的时候,可能会涉及使用查找节点函数。内核提供很多内核函数来查找(解析设备树)一个指定节点:

A、路径查找

/*

* 功能:通过路径查找指定节点

* 参数:

* const char *path - 节点路径,可以是路径,也可以是路径引用

* 返回值:

* 成功:得到节点对象的首地址;失败:NULL

*/

struct device_node *of_find_node_by_path(const char *path);

B、节点名查找

/*

* 功能:通过节点名查找指定节点

* 参数:

* struct device_node *from - 指向开始路径的节点,如果为NULL,则从根节点开始

* const char *name- 节点名

* 返回值:

* 成功:得到节点对象的首地址;失败:NULL

*/

struct device_node *of_find_node_by_name(struct device_node *from, const char *name);

C、通过compatible属性查找

/*

* 功能:通过compatible属性查找指定节点

* 参数:

* struct device_node *from - 指向开始路径的节点,如果为NULL,则从根节点开始

* const char *type - 节点类型,可以为NULL

* const char *compat - 指向节点的compatible属性的值(字符串)的首地址

* 返回值:

* 成功:得到节点对象的首地址;失败:NULL

*/

struct device_node *of_find_compatible_node(struct device_node *from,

const char *type, const char *compat);

设备ID表结构

struct of_device_id {

char name[32]; /*设备名*/

char type[32]; /*设备类型*/

char compatible[128]; /*用于与设备树compatible属性值匹配的字符串*/

const void *data; /*私有数据*/

};

/*

* 功能:通过compatible属性查找指定节点

* 参数:

* struct device_node *from - 指向开始路径的节点,如果为NULL,则从根节点开始

* const struct of_device_id *matches - 指向设备ID表

* 注意ID表必须以NULL结束

* 范例: const struct of_device_id mydemo_of_match[] = {

{ .compatible = "fs4412,mydemo", },

{}

};

* 返回值:

* 成功:得到节点对象的首地址;失败:NULL

*/

struct device_node *of_find_matching_node(struct device_node *from,

const struct of_device_id *matches);

D、查找子节点

/*

* 功能:查找指定节点的子节点

* 参数:

* const struct device_node *node - 指向要查找子节点的父节点

* const char *name - 子节点名

* 返回值:

* 成功:得到子节点对象的首地址;失败:NULL

*/

struct device_node *of_get_child_by_name(const struct device_node *node,

const char *name);

3.5、节点内容合并

有时候,一个硬件设备的部分信息不会变化,但是部分信息是可能会变化的,就出现了节点内容合并。即:先编写好节点,仅仅描述部分属性值;使用者后加一部分属性值。

在同级路径下,节点名相同的“两个”节点实际是一个节点。

/*参考板的已经编写好的node节点*/

/{

node{

item1=value;

};

};

/*移植者添加的node节点*/

/{

node{

item2=value;

};

};

等价于:

/{

node{

item1=value;

item2=value;

};

};

3.6、节点内容替换

有时候,一个硬件设备的部分属性信息可能会变化,但是设备树里面已经描述了所有的属性值,使用者可以添加已有的属性值,以替换原有的属性值,就出现了节点内容替换。

另外,节点的内容即使不会变化,但是可能不会使用。

在同级路径下,节点名相同的“两个”节点实际是一个节点。

内容替换的常见形式之一:

/*参考板的已经编写好的node节点*/

/{

node{

item=value1;

};

};

/*移植者添加的node节点*/

/{

node{

item=value2;

};

};

等价于:

/{

node{

item=value2;

};

};

内容替换的常见形式之二:

/*参考板的已经编写好的node节点*/

/{

node{

item=value;

status = "disabled";

};

};

/*移植者添加的node节点*/

/{

node{

status = "okay";

};

};

等价于:

/{

node{

item=value;

status = "okay";

};

};

3.7、节点内容引用

有时候,一个节点需要使用到别的节点的属性值,就需要引用的概念。有时候在设备树编写时,要替换节点属性值,或是合并节点的属性值,也会使用引用。

A、引用节点完成属性值的替换及合并:

/*参考板的已经编写好的node节点*/

/{

node:node@80000000{

item1=value;

status = "disabled";

};

};

/*移植者添加的node节点*/

&node{

item2=value;

status = "okay";

};

等价于:

/{

node : node@80000000{

item1=value;

item2=value;

status = "okay";

};

};

B、节点引用另一个节点:

/*参考板的已经编写好的node节点*/

/{

node:node@80000000{

item=value;

};

};

/*移植者添加的demo节点*/

/{

demo{

item=<&node>;

};

};

说明:

demo节点的属性item引用了节点的node的属性值,具体怎么使用node节点的属性值,在属性章节进行讨论。

linux 设备树及节点引用相关推荐

  1. Linux设备树特殊节点(aliases、chosen)介绍

    引用一个特定的节点通常使用全路径,aliases 节点可以用于指定一个设备全路径的别名.例如: 1 2 3 4 aliases {         ethernet0 = ð0;         se ...

  2. 我眼中的Linux设备树(二 节点)

    二 节点(node)的表示 首先说节点的表示方法,除了根节点只用一个斜杠"/"表示外,其他节点的表示形式如"node-name@unit-address".@前 ...

  3. Linux设备树翻译计划

    本文翻译自Device Tree Usage主页: http://devicetree.org/Device_Tree_Usage 此译文为本人原创,若要转载请注明! Linux device tre ...

  4. Linux设备树语法详解【转】

    转自:http://www.cnblogs.com/xiaojiang1025/p/6131381.html 概念 Linux内核从3.x开始引入设备树的概念,用于实现驱动代码与设备信息相分离.在设备 ...

  5. linux spidev 应用_嵌入式Linux设备树语法总结

    1 设备树的说明 在写完嵌入式驱动总结后,对于设备树相关的语法和使用一直都想进行系统的描述,但是因为最近比较忙碌,所以一直拖到现在才完成初版,对于整个嵌入式Linux驱动开发中,设备树语法和构建是其中 ...

  6. Linux 设备树的使用技巧

    Linux内核从3.x开始引入设备树的概念,用于实现驱动代码与设备信息相分离.在设备树出现以前,所有关于设备的具体信息都要写在驱动里,一旦外围设备变化,驱动代码就要重写.引入了设备树之后,驱动代码只负 ...

  7. 【正点原子MP157连载】第二十三章 Linux设备树-摘自【正点原子】STM32MP1嵌入式Linux驱动开发指南V1.7

    1)实验平台:正点原子STM32MP157开发板 2)购买链接:https://item.taobao.com/item.htm?&id=629270721801 3)全套实验源码+手册+视频 ...

  8. linux 设备树详解

    2 Linux 设备树 2.1 什么是设备树? 设备树(Device Tree),将这个词分开就是"设备"和"树",描述设备树的文件叫做DTS(Device T ...

  9. linux 设备树dts基础

    ​ ​ 活动地址:CSDN21天学习挑战赛 学习的最大理由是想摆脱平庸,早一天就多一份人生的精彩:迟一天就多一天平庸的困扰.各位小伙伴,如果您: 想系统/深入学习某技术知识点- 一个人摸索学习很难坚持 ...

最新文章

  1. 光测力学-栅线投影/面结构光技术
  2. 每日一皮:程序员距离人生目标只有一个沟
  3. 推荐系统笔记:基于潜在因子模型的协同过滤(latent factor model)
  4. 7收不到邮件 contact form_如何解决Contact Form 7提示的配置错误
  5. 8张图带你轻松温习Java知识
  6. mysql java 获取周_Java中获取Mysql中datetime类型的数据
  7. 车辆动力性经济性 matlab计算
  8. asp.net 将bmp格式图片怎么转换为jpg_PNG图片怎么转换成JPG?原来还可以这么转换...
  9. 记录表类型 oracle,[转]关于oracle的记录类型
  10. 细说CSS的transform
  11. 计算机多余自动启动项,去掉多余的开机启动项
  12. matlab图像画轮毂,轮毂设计及三维造型(全套图纸三维).doc
  13. LoadRunner压力测试:测试报告
  14. 【STM32H7教程】第2章 STM32H7的开发环境搭建
  15. 【董天一】什么是IPFS(三)
  16. STM32小车篇之超声波测距
  17. zbox mysql_centos6.5搭建禅道
  18. 黑客攻击-木马程序(2)
  19. Puppeteer + Nodejs 通用全屏网页截图方案(二)常用参数实现
  20. Nat. Commun. | 从单细胞转录组数据中学习可解释的细胞和基因签名嵌入

热门文章

  1. 每日新闻 | 2019年软件业务收入71768亿元增长15.4%;芯片行业去年收入下滑12%
  2. 2022制冷与空调设备运行操作复训题库及答案
  3. 软件工程 用例建模 习题
  4. 检测设备是否支持Google Play服务
  5. 为了让“对口型”更有意思,我们做了个网页小游戏……
  6. matlab中stms和taylor,基于Matlab的电力系统故障分析与仿真V2.1(网络分享版)
  7. 怎样用Java制作一个选择题
  8. PCB模块化设计10——PCI-E高速PCB布局布线设计规范
  9. char varchar nchar nvarchar区别
  10. Batch Normalization 的实战使用