让我们来看看内核如何将设备树解析成所需的device_node结构。linux内核的底层初始化部分在head.s中,这是汇编代码,暂且不作过多讨论。在head.s完成部分初始化后,就开始调用c语言函数,第一个被调用的c语言函数是start_kernel:
asmlinkage __visible void __init start_kernel(void){
//...
setup_arch(&command_line);
//...
}设备树的处理主要在setup_arch()函数中进行。
void __init __no_sanitize_address setup_arch(char **cmdline_p){
setup_machine_fdt(__fdt_pointer);
......
unflatten_device_tree();
}这两个被调用的函数是主要的设备树处理函数:
setup_machine_fdt:根据传入的设备树dtb的根节点完成一些初始化操作。 unflatten_device_tree:对设备树进行具体解析,将设备树各节点转换成相应的struct device_node结构体。

我们通过代码跟踪仔细分析setup_machine_fdt:
static void __init setup_machine_fdt(phys_addr_t dt_phys){
void *dt_virt = fixmap_remap_fdt(dt_phys, &size, PAGE_KERNEL);
......
early_init_dt_scan(dt_virt)
......
name = of_flat_dt_get_machine_name();
......
}上面的函数作用大致如下:
首先通过fixmap_remap_fdt获取dts的头部地址,然后通过early_init_dt_scan进行下一步的扫描:
bool __init early_init_dt_scan(void *params){
bool status;
status = early_init_dt_verify(params);
if (!status)
return false;
//进行早期扫描
early_init_dt_scan_nodes();
return true;
}void __init early_init_dt_scan_nodes(void){
......
//读取"#address-cells","#size-cells"属性
early_init_dt_scan_root();
......
//查找chosen节点
early_init_dt_scan_chosen(boot_command_line);
......
//查找memory节点
early_init_dt_scan_memory();
......
}其主要包括:
获取root节点的size-cells和address-cells值 解析chosen节点中的initrd和bootargs属性,其中initrd包含其地址和size信息 遍历memory节点的内存region,并将合法的region加入memblock中 这里用一张图简单的总结下是如何获取内核前期初始化所需的bootargs,cmd_line等系统引导参数。

以上就是Linux BSP实战课(设备树篇):设备树的解析的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号