宋牧春: Linux设备树文件结构与解析深度分析(2) 【转】

本文涉及的产品
全局流量管理 GTM,标准版 1个月
云解析 DNS,旗舰版 1个月
公共DNS(含HTTPDNS解析),每月1000万次HTTP解析
简介: 转自:https://mp.weixin.qq.com/s/WPZSElF3OQPMGqdoldm07A             作者简介   宋牧春,linux内核爱好者,喜欢阅读各种开源代码(uboot、linux、ucos、rt-thread等),对于优秀的代码框架及其痴迷。

转自:https://mp.weixin.qq.com/s/WPZSElF3OQPMGqdoldm07A

 

 

 

 

 

 

作者简介

 

宋牧春,linux内核爱好者,喜欢阅读各种开源代码(uboot、linux、ucos、rt-thread等),对于优秀的代码框架及其痴迷。现就职于一家手机研发公司,任职Android BSP开发工程师。

 

正文开始

 

前情提要:

宋牧春: Linux设备树文件结构与解析深度分析(1)

征稿和征稿奖励名单:

Linuxer-"Linux开发者自己的媒体"第二月稿件录取和赠书名单

Linuxer-"Linux开发者自己的媒体"首月稿件录取和赠书名单


 

6. platform_device和device_node绑定

经过以上解析,DeviceTree的数据已经全部解析出具体的struct device_node和struct property结构体,下面需要和具体的device进行绑定。首先讲解platform_device和device_node的绑定过程。在arch/arm/kernel/setup.c文件中,customize_machine()函数负责填充struct platform_device结构体。函数调用过程如图8所示。

 

  图8 platform_device生成流程图

代码分析如下:

 

const struct of_device_id  of_default_bus_match_table[] = {

    {  .compatible = "simple-bus", },

    {  .compatible = "simple-mfd", },

#ifdef CONFIG_ARM_AMBA

    {  .compatible = "arm,amba-bus", },

#endif /* CONFIG_ARM_AMBA */

    {}  /* Empty terminated list */

};

 

int of_platform_populate(struct  device_node *root,

           const  struct of_device_id *matches,

           const  struct of_dev_auxdata *lookup,

           struct  device *parent)

{

    struct  device_node *child;

    int  rc = 0;

 

    /*  获取根节点 */

    root  = root ? of_node_get(root) : of_find_node_by_path("/");

    if  (!root)

       return  -EINVAL;

 

    /*  为根节点下面的每一个节点创建platform_device结构体 */

    for_each_child_of_node(root,  child) {

       rc  = of_platform_bus_create(child, matches, lookup, parent, true);

       if  (rc) {

           of_node_put(child);

           break;

       }

    }

    /*  更新device_node flag标志位 */

    of_node_set_flag(root,  OF_POPULATED_BUS);

 

    of_node_put(root);

    return  rc;

}

 

static int of_platform_bus_create(struct  device_node *bus,

                const struct of_device_id *matches,

                const struct of_dev_auxdata *lookup,

                struct device *parent, bool strict)

{

    const  struct of_dev_auxdata *auxdata;

    struct  device_node *child;

    struct  platform_device *dev;

    const  char *bus_id = NULL;

    void  *platform_data = NULL;

    int  rc = 0;

 

    /*  只有包含"compatible"属性的node节点才会生成相应的platform_device结构体 */

    /*  Make sure it has a compatible property */

    if  (strict && (!of_get_property(bus, "compatible", NULL))) {

       return  0;

    }

    /*  省略部分代码 */

    /*  

     * 针对节点下面得到status = "ok" 或者status = "okay"或者不存在status属性的

     * 节点分配内存并填充platform_device结构体

     */

    dev  = of_platform_device_create_pdata(bus, bus_id, platform_data, parent);

    if  (!dev || !of_match_node(matches, bus))

       return  0;

 

    /*  递归调用节点解析函数,为子节点继续生成platform_device结构体,前提是父节点

     * “compatible” = “simple-bus”,也就是匹配of_default_bus_match_table结构体中的数据

     */

    for_each_child_of_node(bus,  child) {

       rc  = of_platform_bus_create(child, matches, lookup, &dev->dev, strict);

       if  (rc) {

           of_node_put(child);

           break;

       }

    }

    of_node_set_flag(bus,  OF_POPULATED_BUS);

    return  rc;

}

 

总的来说,当of_platform_populate()函数执行完毕,kernel就为DTB中所有包含compatible属性名的第一级node创建platform_device结构体,并向平台设备总线注册设备信息。如果第一级node的compatible属性值等于“simple-bus”、“simple-mfd”或者"arm,amba-bus"的话,kernel会继续为当前node的第二级包含compatible属性的node创建platform_device结构体,并注册设备。Linux系统下的设备大多都是挂载在平台总线下的,因此在平台总线被注册后,会根据of_root节点的树结构,去寻找该总线的子节点,所有的子节点将被作为设备注册到该总线上。

7. i2c_client和device_node绑定

经过customize_machine()函数的初始化,DTB已经转换成platform_device结构体,这其中就包含i2c adapter设备,不同的SoC需要通过平台设备总线的方式自己实现i2c adapter设备的驱动。例如:i2c_adapter驱动的probe函数中会调用i2c_add_numbered_adapter()注册adapter驱动,函数流执行如图9所示。

9 i2c_client绑定流程

在of_i2c_register_devices()函数内部便利i2c节点下面的每一个子节点,并为子节点(status = “disable”的除外)创建i2c_client结构体,并与子节点的device_node挂接。其中i2c_client的填充是在i2c_new_device()中进行的,最后device_register()。在构建i2c_client的时候,会对node下面的compatible属性名称的厂商名字去除作为i2c_client的name。例如:compatible = “maxim,ds1338”,则i2c_client->name = “ds1338”。

8. Device_Tree与sysfs

kernel启动流程为start_kernel()→rest_init()→kernel_thread():kernel_init()→do_basic_setup()→driver_init()→of_core_init(),在of_core_init()函数中在sys/firmware/devicetree/base目录下面为设备树展开成sysfs的目录和二进制属性文件,所有的node节点就是一个目录,所有的property属性就是一个二进制属性文件。

【作者】 张昺华
【新浪微博】 张昺华--sky
【twitter】 @sky2030_
【facebook】 张昺华 zhangbinghua
本文版权归作者和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文连接,否则保留追究法律责任的权利.
目录
相关文章
|
20天前
|
人工智能
歌词结构的巧妙安排:写歌词的方法与技巧解析,妙笔生词AI智能写歌词软件
歌词创作是一门艺术,关键在于巧妙的结构安排。开头需迅速吸引听众,主体部分要坚实且富有逻辑,结尾则应留下深刻印象。《妙笔生词智能写歌词软件》提供多种 AI 功能,帮助创作者找到灵感,优化歌词结构,写出打动人心的作品。
|
1月前
|
机器学习/深度学习 数据采集 存储
时间序列预测新突破:深入解析循环神经网络(RNN)在金融数据分析中的应用
【10月更文挑战第7天】时间序列预测是数据科学领域的一个重要课题,特别是在金融行业中。准确的时间序列预测能够帮助投资者做出更明智的决策,比如股票价格预测、汇率变动预测等。近年来,随着深度学习技术的发展,尤其是循环神经网络(Recurrent Neural Networks, RNNs)及其变体如长短期记忆网络(LSTM)和门控循环单元(GRU),在处理时间序列数据方面展现出了巨大的潜力。本文将探讨RNN的基本概念,并通过具体的代码示例展示如何使用这些模型来进行金融数据分析。
210 2
|
11天前
|
算法 Linux 定位技术
Linux内核中的进程调度算法解析####
【10月更文挑战第29天】 本文深入剖析了Linux操作系统的心脏——内核中至关重要的组成部分之一,即进程调度机制。不同于传统的摘要概述,我们将通过一段引人入胜的故事线来揭开进程调度算法的神秘面纱,展现其背后的精妙设计与复杂逻辑,让读者仿佛跟随一位虚拟的“进程侦探”,一步步探索Linux如何高效、公平地管理众多进程,确保系统资源的最优分配与利用。 ####
39 4
|
12天前
|
缓存 负载均衡 算法
Linux内核中的进程调度算法解析####
本文深入探讨了Linux操作系统核心组件之一——进程调度器,着重分析了其采用的CFS(完全公平调度器)算法。不同于传统摘要对研究背景、方法、结果和结论的概述,本文摘要将直接揭示CFS算法的核心优势及其在现代多核处理器环境下如何实现高效、公平的资源分配,同时简要提及该算法如何优化系统响应时间和吞吐量,为读者快速构建对Linux进程调度机制的认知框架。 ####
|
1月前
|
机器学习/深度学习 搜索推荐 大数据
深度解析:如何通过精妙的特征工程与创新模型结构大幅提升推荐系统中的召回率,带你一步步攻克大数据检索难题
【10月更文挑战第2天】在处理大规模数据集的推荐系统项目时,提高检索模型的召回率成为关键挑战。本文分享了通过改进特征工程(如加入用户活跃时段和物品相似度)和优化模型结构(引入注意力机制)来提升召回率的具体策略与实现代码。严格的A/B测试验证了新模型的有效性,为改善用户体验奠定了基础。这次实践加深了对特征工程与模型优化的理解,并为未来的技术探索提供了方向。
87 2
深度解析:如何通过精妙的特征工程与创新模型结构大幅提升推荐系统中的召回率,带你一步步攻克大数据检索难题
|
1月前
|
存储 SQL 分布式计算
湖仓一体架构深度解析:构建企业级数据管理与分析的新基石
【10月更文挑战第7天】湖仓一体架构深度解析:构建企业级数据管理与分析的新基石
55 1
|
13天前
|
机器学习/深度学习 自然语言处理 数据管理
GraphRAG核心组件解析:图结构与检索增强生成
【10月更文挑战第28天】在当今数据科学领域,自然语言处理(NLP)和图数据管理技术的发展日新月异。GraphRAG(Graph Retrieval-Augmented Generation)作为一种结合了图结构和检索增强生成的创新方法,已经在多个应用场景中展现出巨大的潜力。作为一名数据科学家,我对GraphRAG的核心组件进行了深入研究,并在此分享我的理解和实践经验。
36 0
|
1月前
|
弹性计算 网络协议 Ubuntu
如何在阿里云国际版Linux云服务器中自定义配置DNS
如何在阿里云国际版Linux云服务器中自定义配置DNS
|
19天前
光纤电缆(FOC)的结构深度解析
【10月更文挑战第21天】
35 0
|
1月前
|
存储 编译器 C++
【初阶数据结构】掌握二叉树遍历技巧与信息求解:深入解析四种遍历方法及树的结构与统计分析
【初阶数据结构】掌握二叉树遍历技巧与信息求解:深入解析四种遍历方法及树的结构与统计分析