Linux加载DTS设备节点的过程(以高通8974平台为例) .

2024-04-18 14:18

本文主要是介绍Linux加载DTS设备节点的过程(以高通8974平台为例) .,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

DTS是Device Tree Source的缩写,用来描述设备的硬件细节。在过去的ARM Linux中,arch/arm/plat-xxx和arch/arm/mach-xxx中充斥着大量的垃圾代码,相当多数的代码只是在描述板级细节,而这些板级细节对于内核来讲,不过是垃圾,如板上的platform设备、resource、i2c_board_info、spi_board_info以及各种硬件的platform_data。为了去掉这些垃圾代码,Linux采用DTS这种新的数据结构来描述硬件设备。采用Device Tree后,许多硬件的细节可以直接透过它传递给Linux,而不再需要在kernel中进行大量的冗余编码。

       有关DTS的语法格式,网上有很多资料,这里我就不再赘述了。本文主要讲Linux是怎样加载DTS设备节点的流程。下面以高通QCT8974平台(Linux内核)为例进行讲解:

(1)使用DTS注册平台总线的过程

       在讲注册过程之前,我们先看看DTS是怎样描述平台总线结构的,以i2c总线为例:

[c] view plain copy print ?
  1. / {  
  2.     model = "Qualcomm MSM 8974";  
  3.     compatible = "qcom,msm8974";  
  4.     interrupt-parent = <&intc>;  
  5.   
  6.     aliases {  
  7.         spi0 = &spi_0;  
  8.         spi7 = &spi_7;  
  9.         sdhc1 = &sdhc_1; /* SDC1 eMMC slot */  
  10.         sdhc2 = &sdhc_2; /* SDC2 SD card slot */  
  11.         sdhc3 = &sdhc_3; /* SDC3 SDIO slot */  
  12.         sdhc4 = &sdhc_4; /* SDC4 SDIO slot */  
  13.     };  
  14.   
  15.     memory {  
  16.   
  17.         secure_mem: secure_region {  
  18.             linux,contiguous-region;  
  19.             reg = <0 0x7800000="">;  
  20.             label = "secure_mem";  
  21.         };  
  22.   
  23.         adsp_mem: adsp_region {  
  24.             linux,contiguous-region;  
  25.             reg = <0 0x2000000="">;  
  26.             label = "adsp_mem";  
  27.         };  
  28.     };  
  29.   
  30.     intc: interrupt-controller@F9000000 {  
  31.         compatible = "qcom,msm-qgic2";  
  32.         interrupt-controller;  
  33.         #interrupt-cells = <3>;   
  34.         reg = <0xf9000000 0x1000="">,  
  35.               <0xf9002000 0x1000="">;  
  36.     };  
  37.   
  38.     msmgpio: gpio@fd510000 {  
  39.         compatible = "qcom,msm-gpio";  
  40.         gpio-controller;  
  41.         #gpio-cells = <2>;   
  42.         interrupt-controller;  
  43.         #interrupt-cells = <2>;   
  44.         reg = <0xfd510000 0x4000="">;  
  45.         ngpio = <146>;  
  46.         interrupts = <0 208="" 0="">;  
  47.         qcom,direct-connect-irqs = <8>;  
  48.     };  
  49.   
  50.     wcd9xxx_intc: wcd9xxx-irq {  
  51.         compatible = "qcom,wcd9xxx-irq";  
  52.         interrupt-controller;  
  53.         #interrupt-cells = <1>;   
  54.         interrupt-parent = <&msmgpio>;  
  55.         interrupts = <72 0="">;  
  56.         interrupt-names = "cdc-int";  
  57.     };  
  58.   
  59.     timer {  
  60.         compatible = "arm,armv7-timer";  
  61.         interrupts = <1 2="" 0="" 1="" 3="" 0="">;  
  62.         clock-frequency = <19200000>;  
  63.     };  
  64.   
  65.     i2c_0: i2c@f9967000 { /* BLSP#11 */  
  66.         cell-index = <0>;  
  67.         compatible = "qcom,i2c-qup";  
  68.         reg = <0xf9967000 0x1000="">;  
  69.         #address-cells = <1>;   
  70.         #size-cells = <0>;   
  71.         reg-names = "qup_phys_addr";  
  72.         interrupts = <0 105="" 0="">;  
  73.         interrupt-names = "qup_err_intr";  
  74.         qcom,i2c-bus-freq = <100000>;  
  75.         qcom,i2c-src-freq = <50000000>;  
  76.     };  
  77.   
  78.     i2c_2: i2c@f9924000 {  
  79.         cell-index = <2>;  
  80.         compatible = "qcom,i2c-qup";  
  81.         reg = <0xf9924000 0x1000="">;  
  82.         #address-cells = <1>;   
  83.         #size-cells = <0>;   
  84.         reg-names = "qup_phys_addr";  
  85.         interrupts = <0 96="" 0="">;  
  86.         interrupt-names = "qup_err_intr";  
  87.         qcom,i2c-bus-freq = <100000>;  
  88.         qcom,i2c-src-freq = <50000000>;  
  89.     };  
  90.   
  91.     spi_0: spi@f9923000 {  
  92.         compatible = "qcom,spi-qup-v2";  
  93.         reg = <0xf9923000 0x1000="">;  
  94.         interrupts = <0 95="" 0="">;  
  95.         spi-max-frequency = <19200000>;  
  96.         #address-cells = <1>;   
  97.         #size-cells = <0>;   
  98.         gpios = <&msmgpio 3 0>, /* CLK  */  
  99.             <&msmgpio 1 0>, /* MISO */  
  100.             <&msmgpio 0 0>; /* MOSI */  
  101.         cs-gpios = <&msmgpio 9 0>;  
  102.     };  
  103.   
  104. };  

从上面可知,系统平台上挂载了很多总线,如i2c、spi、uart等等,每一个总线分别被描述为一个节点。Linux在启动后,到C入口时,会执行以下操作,加载系统平台上的总线和设备:

start_kernel() --> setup_arch() --> unflatten_device_tree() 

unflatten_device_tree()的代码如下:

[c] view plain copy print ?
  1. void __init unflatten_device_tree(void)  
  2. {  
  3.     __unflatten_device_tree(initial_boot_params, &allnodes,  
  4.                 early_init_dt_alloc_memory_arch);  
  5.   
  6.     /* Get pointer to "/chosen" and "/aliasas" nodes for use everywhere */  
  7.     of_alias_scan(early_init_dt_alloc_memory_arch);  
  8. }  

 在执行完unflatten_device_tree()后,DTS节点信息被解析出来,保存到allnodes链表中,allnodes会在后面被用到。

随后,当系统启动到board文件时,会调用.init_machine,高通8974平台对应的是msm8974_init()。接着调用of_platform_populate(....)接口,加载平台总线和平台设备。至此,系统平台上的所有已配置的总线和设备将被注册到系统中。注意:不是dtsi文件中所有的节点都会被注册,在注册总线和设备时,会对dts节点的状态作一个判断,如果节点里面的status属性没有被定义,或者status属性被定义了并且值被设为“ok”或者“okay”,其他情况则不被注册到系统中。


(2)使用DTS注册总线设备的过程

       上面讲了Linux怎样使用DTS注册平台总线和平台设备到系统中,那么其他设备,例如i2c、spi设备是怎样注册到系统中的呢?下面我们就以i2c设备为例,讲解Linux怎样注册i2c设备到系统中。

 以高通8974平台为例,在注册i2c总线时,会调用到qup_i2c_probe()接口,该接口用于申请总线资源和添加i2c适配器。在成功添加i2c适配器后,会调用of_i2c_register_devices()接口。此接口会解析i2c总线节点的子节点(挂载在该总线上的i2c设备节点),获取i2c设备的地址、中断号等硬件信息。然后调用request_module()加载设备对应的驱动文件,调用i2c_new_device(),生成i2c设备。此时设备和驱动都已加载,于是drvier里面的probe方法将被调用。后面流程就和之前一样了。


       简而言之,Linux采用DTS描述设备硬件信息后,省去了大量板文件垃圾信息。Linux在开机启动阶段,会解析DTS文件,保存到全局链表allnodes中,在掉用.init_machine时,会跟据allnodes中的信息注册平台总线和设备。值得注意的是,加载流程并不是按找从树根到树叶的方式递归注册,而是只注册根节点下的第一级子节点,第二级及之后的子节点暂不注册。Linux系统下的设备大多都是挂载在平台总线下的,因此在平台总线被注册后,会根据allnodes节点的树结构,去寻找该总线的子节点,所有的子节点将被作为设备注册到该总线上。


这篇关于Linux加载DTS设备节点的过程(以高通8974平台为例) .的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/915021

相关文章

Linux脚本(shell)的使用方式

《Linux脚本(shell)的使用方式》:本文主要介绍Linux脚本(shell)的使用方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录概述语法详解数学运算表达式Shell变量变量分类环境变量Shell内部变量自定义变量:定义、赋值自定义变量:引用、修改、删

canal实现mysql数据同步的详细过程

《canal实现mysql数据同步的详细过程》:本文主要介绍canal实现mysql数据同步的详细过程,本文通过实例图文相结合给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的... 目录1、canal下载2、mysql同步用户创建和授权3、canal admin安装和启动4、canal

MySQL存储过程之循环遍历查询的结果集详解

《MySQL存储过程之循环遍历查询的结果集详解》:本文主要介绍MySQL存储过程之循环遍历查询的结果集,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录前言1. 表结构2. 存储过程3. 关于存储过程的SQL补充总结前言近来碰到这样一个问题:在生产上导入的数据发现

SpringBoot集成LiteFlow实现轻量级工作流引擎的详细过程

《SpringBoot集成LiteFlow实现轻量级工作流引擎的详细过程》LiteFlow是一款专注于逻辑驱动流程编排的轻量级框架,它以组件化方式快速构建和执行业务流程,有效解耦复杂业务逻辑,下面给大... 目录一、基础概念1.1 组件(Component)1.2 规则(Rule)1.3 上下文(Conte

Linux链表操作方式

《Linux链表操作方式》:本文主要介绍Linux链表操作方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、链表基础概念与内核链表优势二、内核链表结构与宏解析三、内核链表的优点四、用户态链表示例五、双向循环链表在内核中的实现优势六、典型应用场景七、调试技巧与

详解Linux中常见环境变量的特点与设置

《详解Linux中常见环境变量的特点与设置》环境变量是操作系统和用户设置的一些动态键值对,为运行的程序提供配置信息,理解环境变量对于系统管理、软件开发都很重要,下面小编就为大家详细介绍一下吧... 目录前言一、环境变量的概念二、常见的环境变量三、环境变量特点及其相关指令3.1 环境变量的全局性3.2、环境变

Linux系统中的firewall-offline-cmd详解(收藏版)

《Linux系统中的firewall-offline-cmd详解(收藏版)》firewall-offline-cmd是firewalld的一个命令行工具,专门设计用于在没有运行firewalld服务的... 目录主要用途基本语法选项1. 状态管理2. 区域管理3. 服务管理4. 端口管理5. ICMP 阻断

Linux实现线程同步的多种方式汇总

《Linux实现线程同步的多种方式汇总》本文详细介绍了Linux下线程同步的多种方法,包括互斥锁、自旋锁、信号量以及它们的使用示例,通过这些同步机制,可以解决线程安全问题,防止资源竞争导致的错误,示例... 目录什么是线程同步?一、互斥锁(单人洗手间规则)适用场景:特点:二、条件变量(咖啡厅取餐系统)工作流

springboot加载不到nacos配置中心的配置问题处理

《springboot加载不到nacos配置中心的配置问题处理》:本文主要介绍springboot加载不到nacos配置中心的配置问题处理,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑... 目录springboot加载不到nacos配置中心的配置两种可能Spring Boot 版本Nacos

Linux中修改Apache HTTP Server(httpd)默认端口的完整指南

《Linux中修改ApacheHTTPServer(httpd)默认端口的完整指南》ApacheHTTPServer(简称httpd)是Linux系统中最常用的Web服务器之一,本文将详细介绍如何... 目录一、修改 httpd 默认端口的步骤1. 查找 httpd 配置文件路径2. 编辑配置文件3. 保存