嵌入式linux内核调用设备树,6. 内核对设备树的处理

一、设备树的描述

对于设备树,其描述的信息可以分成三部分;在内核中,对设备树的处理也会分成三部分:

Linux uses DT data for three major purposes:

1) platform identification, (平台识别信息)

2) runtime configuration, and (运行时配置信息)

3) device population. (设备信息)

二、内核head.S对dtb的简单处理

u-boot把一些参数,把设备树文件传给内核,那么内核如何处理设备树文件呢?从内核的第一个启动文件head.S着手,分析其对dtb的处理。(详细的head.s的分析可参见:嵌入式Linux完全开发手册 - 移植Linux内核 - Linux内核启动概述)

bootloader启动内核时,会设置r0,r1,r2三个寄存器,由这三个寄存器将参数传给内核:

r0一般设置为0;

r1一般设置为machine id (在使用设备树时该参数没有被使用);

r2一般设置ATAGS或DTB的开始地址;

这里额外介绍下machine id的作用。假设一个内核镜像uImage可以支持多种单板:

77665b4de9de

head.S分析如下:

a. [ __lookup_processor_type ] 使用汇编指令读取CPU ID, 根据该ID找到对应的proc_info_list结构体(里面含有这类CPU的初始化函数、信息)

b. [ __vet_atags ] 判断是否存在可用的ATAGS或DTB;

c. [ __create_page_tables ] 创建页表, 即创建虚拟地址和物理地址的映射关系;

d. [ __enable_mmu ] 使能MMU, 以后就要使用虚拟地址了

e. [ __mmap_switched ] 上述函数里将会调用__mmap_switched

f. 把bootloader传入的r2参数, 保存到变量__atags_pointer中

g. 调用C函数start_kernel

head.S/head-common.S :

把bootloader传来的r1值, 赋给了C变量: __machine_arch_type

把bootloader传来的r2值, 赋给了C变量: __atags_pointer // dtb首地址

由此可见,head.S对dtb的处理还是比较简单的,只是将u-boot传来的R2寄存器里的值,也就是DTB的首地址赋给了内核中的C变量:__atags_pointer 。

三、内核对设备树中平台信息的处理(选择machine_desc)

platform identification, (平台识别信息)

上面的介绍中,已经知道在以前u-boot的ATAG传参中,会传入machine id 由此来匹配machine desc;当使用dtb传参时,u-boot并不传递machine id了,内核machine desc 的依赖于dtb 根节点下的compatible属性,内核根据该属性来找到合适的machine desc。

77665b4de9de

作为内核的使用者,我们已经列出清单,期望在内核中找到支持某个单板的machine desc 。

对于内核,每个machine desc需表明能支持哪些单板;

77665b4de9de

compatible属性对machine desc的查找

a. 设备树根节点的compatible属性列出了一系列的字符串,表示它兼容的单板名,从"最兼容"到次之。

b. 内核中有多个machine_desc, 其中有dt_compat成员, 它指向一个字符串数组, 里面表示该machine_desc支持哪些单板

c. 使用compatile属性的值, 跟每一个machine_desc.dt_compat 比较,成绩为"吻合的compatile属性值的位置",成绩越低越匹配, 对应的machine_desc即被选中

77665b4de9de

分析下代码的调用流程:

从上面的分析已经知道,内核在启动后,将dtb文件的首地址保存到变量__atags_pointer中,然后调用start_kernel,接下来从start_kernel开始分析:

函数调用过程:

start_kernel // init/main.c

setup_arch(&command_line); // arch/arm/kernel/setup.c

mdesc = setup_machine_fdt(__atags_pointer); // arch/arm/kernel/devtree.c

// 判断是否有效的dtb, drivers/of/ftd.c

early_init_dt_verify(phys_to_virt(dt_phys)

initial_boot_params = params; //将dtb的地址保存在全局变量里

// 找到最匹配的machine_desc, drivers/of/ftd.c

mdesc = of_flat_dt_match_machine(mdesc_best, arch_get_next_mach);

while ((data = get_next_compat(&compat))) {

score = of_flat_dt_match(dt_root, compat);

if (score > 0 && score < best_score) {

best_data = data;

best_score = score;

}

}

machine_desc = mdesc;

四、内核对设备树中运行时配置信息的处理

runtime configuration, (运行时配置信息)

运行时的三种配置信息总结如下:

a. /chosen节点中bootargs属性的值, 存入全局变量: boot_command_line

b. 确定根节点的这2个属性的值: #address-cells, #size-cells

存入全局变量: dt_root_addr_cells, dt_root_size_cells

c. 解析/memory中的reg属性, 提取出"base, size", 最终调用memblock_add(base, size);

函数调用过程:

start_kernel // init/main.c

setup_arch(&command_line); // arch/arm/kernel/setup.c

mdesc = setup_machine_fdt(__atags_pointer); // arch/arm/kernel/devtree.c

early_init_dt_scan_nodes(); // drivers/of/ftd.c

/* Retrieve various information from the /chosen node */

of_scan_flat_dt(early_init_dt_scan_chosen, boot_command_line);

/* Initialize {size,address}-cells info */

of_scan_flat_dt(early_init_dt_scan_root, NULL);

/* Setup memory, calling early_init_dt_add_memory_arch */

of_scan_flat_dt(early_init_dt_scan_memory, NULL);

5

五、内核对设备树中设备信息的处理

device population. (设备信息)

内核启动后,会将dtb中的所有节点转换为device_node,了解了device_node结构体和properties结构体,就可知道大致情况。

➢每一个节点都转换为一个device_node结构体:

struct device_node {

const char *name; // 来自节点中的name属性, 如果没有该属性, 则设为"NULL"

const char *type; // 来自节点中的device_type属性, 如果没有该属性, 则设为"NULL"

phandle phandle;

const char *full_name; // 节点的名字, node-name[@unit-address]

struct fwnode_handle fwnode;

struct property *properties; // 节点的属性

struct property *deadprops; /* removed properties */

struct device_node *parent; // 节点的父亲

struct device_node *child; // 节点的孩子(子节点)

struct device_node *sibling; // 节点的兄弟(同级节点)

#if defined(CONFIG_OF_KOBJ)

struct kobject kobj;

#endif

unsigned long _flags;

void *data;

#if defined(CONFIG_SPARC)

const char *path_component_name;

unsigned int unique_id;

struct of_irq_controller *irq_trans;

#endif

};

➢device_node结构体中有properties, 用来表示该节点的属性,每一个属性对应一个property结构体:

struct property {

char *name; // 属性名字, 指向dtb文件中的字符串

int length; // 属性值的长度

void *value; // 属性值, 指向dtb文件中value所在位置, 数据仍以big endian存储

struct property *next;

#if defined(CONFIG_OF_DYNAMIC) || defined(CONFIG_SPARC)

unsigned long _flags;

#endif

#if defined(CONFIG_OF_PROMTREE)

unsigned int unique_id;

#endif

#if defined(CONFIG_OF_KOBJ)

struct bin_attribute attr;

#endif

};

a. 在DTB文件中,

每一个节点都以TAG(FDT_BEGIN_NODE, 0x00000001)开始, 节点内部可以嵌套其他节点,

每一个属性都以TAG(FDT_PROP, 0x00000003)开始

b. 这些device_node构成一棵树, 根节点为: of_root

跟踪下代码:

函数调用过程:

start_kernel // init/main.c

setup_arch(&command_line); // arch/arm/kernel/setup.c

arm_memblock_init(mdesc); // arch/arm/kernel/setup.c

early_init_fdt_reserve_self();

/* Reserve the dtb region */

// 把DTB所占区域保留下来, 即调用: memblock_reserve

early_init_dt_reserve_memory_arch(__pa(initial_boot_params),

fdt_totalsize(initial_boot_params),

0);

early_init_fdt_scan_reserved_mem(); // 根据dtb中的memreserve信息, 调用memblock_reserve

unflatten_device_tree(); // arch/arm/kernel/setup.c

__unflatten_device_tree(initial_boot_params, NULL, &of_root,

early_init_dt_alloc_memory_arch, false); // drivers/of/fdt.c

/* First pass, scan for size */

size = unflatten_dt_nodes(blob, NULL, dad, NULL);

/* Allocate memory for the expanded device tree */

mem = dt_alloc(size + 4, __alignof__(struct device_node));

/* Second pass, do actual unflattening */

unflatten_dt_nodes(blob, mem, dad, mynodes);

populate_node

np = unflatten_dt_alloc(mem, sizeof(struct device_node) + allocl,

__alignof__(struct device_node));

np->full_name = fn = ((char *)np) + sizeof(*np);

populate_properties

pp = unflatten_dt_alloc(mem, sizeof(struct property),

__alignof__(struct property));

pp->name = (char *)pname;

pp->length = sz;

pp->value = (__be32 *)val;

对于驱动开发者,设备树中还有一个备受关注的信息:platform_device。

dts -> dtb -> device_node -> platform_device

在上述,我们已经知道dts中的节点信息终将被转换为 device_node ,但并不是所有device_node都会被转换成platform_device,需满足:

根节点下含有compatile属性的子节点

如果一个结点的compatile属性含有这些特殊的值("simple-bus","simple-mfd","isa","arm,amba-bus")之一, 那么它的子结点(需含compatile属性)也可以转换为platform_device

i2c, spi等总线节点下的子节点, 应该交给对应的总线驱动程序来处理, 它们不应该被转换为platform_device

转换规则:

platform_device中含有resource数组, 它来自device_node的reg, interrupts属性;

platform_device.dev.of_node指向device_node, 可以通过它获得其他属性;

总结下:

a. 内核函数of_platform_default_populate_init, 遍历device_node树, 生成platform_device

b. 并非所有的device_node都会转换为platform_device

只有以下的device_node会转换:

b.1 该节点必须含有compatible属性

b.2 根节点的子节点(节点必须含有compatible属性)

b.3 含有特殊compatible属性的节点的子节点(子节点必须含有compatible属性):

这些特殊的compatilbe属性为: "simple-bus","simple-mfd","isa","arm,amba-bus"

示例:

比如以下的节点,

/mytest会被转换为platform_device,

因为它兼容"simple-bus", 它的子节点/mytest/mytest@0 也会被转换为platform_device

/i2c节点一般表示i2c控制器, 它会被转换为platform_device, 在内核中有对应的platform_driver;

/i2c/at24c02节点不会被转换为platform_device, 它被如何处理完全由父节点的platform_driver决定, 一般是被创建为一个i2c_client。

类似的也有/spi节点, 它一般也是用来表示SPI控制器, 它会被转换为platform_device, 在内核中有对应的platform_driver;

/spi/flash@0节点不会被转换为platform_device, 它被如何处理完全由父节点的platform_driver决定, 一般是被创建为一个spi_device。

/ {

mytest {

compatile = "mytest", "simple-bus";

mytest@0 {

compatile = "mytest_0";

};

};

i2c {

compatile = "samsung,i2c";

at24c02 {

compatile = "at24c02";

};

};

spi {

compatile = "samsung,spi";

flash@0 {

compatible = "winbond,w25q32dw";

spi-max-frequency = <25000000>;

reg = <0>;

};

};

};

函数调用过程:

a. of_platform_default_populate_init (drivers/of/platform.c) 被调用到过程:

start_kernel // init/main.c

rest_init();

pid = kernel_thread(kernel_init, NULL, CLONE_FS);

kernel_init

kernel_init_freeable();

do_basic_setup();

do_initcalls();

for (level = 0; level < ARRAY_SIZE(initcall_levels) - 1; level++)

do_initcall_level(level); // 比如 do_initcall_level(3)

for (fn = initcall_levels[3]; fn < initcall_levels[3+1]; fn++)

do_one_initcall(initcall_from_entry(fn)); // 就是调用"arch_initcall_sync(fn)"中定义的fn函数

b. of_platform_default_populate_init (drivers/of/platform.c) 生成platform_device的过程:

of_platform_default_populate_init

of_platform_default_populate(NULL, NULL, NULL);

of_platform_populate(NULL, of_default_bus_match_table, NULL, NULL)

for_each_child_of_node(root, child) {

rc = of_platform_bus_create(child, matches, lookup, parent, true); // 调用过程看下面

dev = of_device_alloc(np, bus_id, parent); // 根据device_node节点的属性设置platform_device的resource

if (rc) {

of_node_put(child);

break;

}

}

c. of_platform_bus_create(bus, matches, ...)的调用过程(处理bus节点生成platform_devie, 并决定是否处理它的子节点):

dev = of_platform_device_create_pdata(bus, bus_id, platform_data, parent); // 生成bus节点的platform_device结构体

if (!dev || !of_match_node(matches, bus)) // 如果bus节点的compatile属性不吻合matches成表, 就不处理它的子节点

return 0;

for_each_child_of_node(bus, child) { // 取出每一个子节点

pr_debug(" create child: %pOF\n", child);

rc = of_platform_bus_create(child, matches, lookup, &dev->dev, strict); // 处理它的子节点, of_platform_bus_create是一个递归调用

if (rc) {

of_node_put(child);

break;

}

}

d. I2C总线节点的处理过程:

/i2c节点一般表示i2c控制器, 它会被转换为platform_device, 在内核中有对应的platform_driver;

platform_driver的probe函数中会调用i2c_add_numbered_adapter:

i2c_add_numbered_adapter // drivers/i2c/i2c-core-base.c

__i2c_add_numbered_adapter

i2c_register_adapter

of_i2c_register_devices(adap); // drivers/i2c/i2c-core-of.c

for_each_available_child_of_node(bus, node) {

client = of_i2c_register_device(adap, node);

client = i2c_new_device(adap, &info); // 设备树中的i2c子节点被转换为i2c_client

}

e. SPI总线节点的处理过程:

/spi节点一般表示spi控制器, 它会被转换为platform_device, 在内核中有对应的platform_driver;

platform_driver的probe函数中会调用spi_register_master, 即spi_register_controller:

spi_register_controller // drivers/spi/spi.c

of_register_spi_devices // drivers/spi/spi.c

for_each_available_child_of_node(ctlr->dev.of_node, nc) {

spi = of_register_spi_device(ctlr, nc); // 设备树中的spi子节点被转换为spi_device

spi = spi_alloc_device(ctlr);

rc = of_spi_parse_dt(ctlr, spi, nc);

rc = spi_add_device(spi);

}

最后,便是platform_device跟platform_driver的匹配过程了,具体的过程分析可移步:

1. 驱动程序分层分离概念-总线设备驱动模型。

2. 字符设备驱动-总线设备驱动模型写法。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值