我们来看看内核是如何把设备树解析成所需的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();
}
这两个被调用的函数就是主要的设备树处理函数:
下面我们再来通过代码跟踪仔细分析。
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();
......
}
上面的函数作用大致如下:
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();
......
}
其主要包括:
这里用一张图简单的总结下是如何获取内核前期初始化所需的bootargs,cmd_line等系统引导参数。