内核如何识别设备树(dtb --> device_node)

内核与设备树的关系:

alt text
但是内核无法直接识别dtb文件,而dtb会每个节点的信息转换成可以供内核识别的
device_node结构体和property结构体
device node结构体定义在include/linux/of.h头文件当中。
property结构体定义在include/linux/of.h头文件当中。

    const char *name;// 节点中的 name 属性
    const char *type;// 节点中的 device_type 属性
    phandle phandle;
    const char *full name;// 节点的名字
    struct fwnode handle fwnode;
    struct property*properties; //指向该设备节点下的第一个属性,其他属性与该属性链表相连

    struct property *deadprops
    struct device node *parent; // 节点的父节点

    struct device node *child;// 节点的子节点

    struct device node *sibling;// 节点的同级节点,也可以叫兄弟节点
......
}


struct property {
    // 属性名字
    char*name;
    // 属性值的长度
    int length;
    // 属性值,
    void *value;
    struct property*next; // 指向该节点的下一个属性
.....
}

实例分析:内核获取设备树的信息:

  • 内核启动时,会从uboot传过来的参数命令行数组中获取设备树的信息,并将其保存在__fdt_pointer变量中。

alt text

  • 启动分析:

alt text

  • 内核的启动函数start_kernel,从汇编部分跳转而来:
    kernel/init/main.c文件里

alt text

  • 查看与设备树相关的代码部分:void __init __no_sanitize_address setup_arch(char **cmdline_p)

alt text

  • cmdline_p = boot_command_line;记录的uboot传给内核的参数命令行数组char __initdata boot_command_line[COMMAND_LINE_SIZE];COMMAND_LINE_SIZE表示数组长度信息,当传入的参数过长时,需要手动修改COMMAND_LINE_SIZE的值

alt text
alt text

  • setup_machine_fdt(__fdt_pointer);记录了dtb在内存中的地址,会提供内核使用的。__fdt_pointer即为在内存中的地址

alt text

  • 实际上dtb位于内存的地址是从x0 传递过来的(汇编部分),x0 里面存放 dtb 的地址是规定,将x0数值给了x21:

alt text

  • 最后又将x21的值给了__fdt_pointer__fdt_pointer的值就是dtb在内存中的地址

alt text

  • 回到setup_arch函数,查看setup_machine_fdt(__fdt_pointer);定义作用:

alt text

  • 当dtb文件超出预期大小,会映射失败

alt text

  • 继续跳转接下来的函数定义:
    kernerl/driver/of/fdt.c
    alt text

alt text

  • bool __init early_init_dt_verify(void *params)为与同级文件下:

/* check device tree validity */
if (fdt_check_header(params))
检验设备树的头部合法性

alt text

  • 继续early_init_dt_scan中的early_init_dt_scan_nodes函数:也在同级目录下
  • early_init_dt_scan_nodes会扫描每一节点的信息,并作初始化
    alt text

路径: drivers/of/fdt.c

  • early_init dt_scan
    • early_init_dt_verify(params);
      • initial_boot_params= params;将 dtb 的虚拟地址保存到initial_boot_params
    • early_init dtscan nodes();
  • 继续回到static void __init setup_machine_fdt中:

alt text

  • fixmap_remap_fdt:将设备树映射的内存的地址设置为只读

alt text

  • 继续下一步:of_flat_dt_get_machine_name()跳转定义,还是在fdt.c文件中:作用是获取model和compatible信息

alt text

  • setup_machine_fdt()算是已经看完了

alt text

  • 回到调用它的setup_arch中,unflatten_device_tree作为dtb展开为device的核心函数,定义任然在ftd.c中

alt text

  • unflatten_device_tree()即为将dtb展开并创建对应的device节点:unflatten_device_tree - create tree of device_nodes from flat blob

alt text

  • 查看其进一步封装的函数:
/*参数:
    dtb的地址
    跟节点
    分配的节点结构体内存大小
    是否是只读
*/

	__unflatten_device_tree(initial_boot_params, NULL, &of_root,early_init_dt_alloc_memory_arch, false);

在__unflatten_device_tree中:

alt text

  • 两次扫描设备树的节点,第一次扫描设备节点,统计设备树需要的内存大小,第二个参数是 NULL,是 NULL就是只统计大小,不做其他事情,一次性为展开的设备上分配内存
  • 第二次扫描的节点:查看unflatten_dt_nodes(blob, mem, dad, mynodes);作用

alt text

  • fdt_next_node用来查找设备树当中的每一个节点,并将其展开,populate_node函数为其创建对应的device节点

alt text

  • 跳转populate_node函数查看作用:

alt text

alt text

  • 第二次扫描的时候,因为第二个参数不是 NULL,所以会执行核心函数unflatten_device_tree()中 populate_node(unflatten_dt_nodes函数中) 和 populate_properties(populate_node函数中),从而构建 device_node 树。

核心函数函数的级联

unflatten_device_tree()
    __unflatten_device_tree()
        // 第一次扫描:分配内存
        unflatten_dt_nodes()
        dt_alloc()
        //第二次扫描:构建 device_node 树
        unflatten_dt_nodes()
            populate_node()
                populate_properties()

整体调用分布图

alt text

  • 19
    点赞
  • 14
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

想和我重名?

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值