首页 > 技术文章 > 内核对设备树的操作

zongzi10010 2019-04-29 21:42 原文


title: 内核对设备树的操作
date: 2019/4/28 18:02:18
toc: true

内核对设备树的操作

哪些节点会被转换

以前的程序platform中的driver去匹配dev中的资源文件

struct resource {
	resource_size_t start;
	resource_size_t end;
	const char *name;
	unsigned long flags;
	unsigned long desc;
	struct resource *parent, *sibling, *child;
};

//类型有这么几种
#define IORESOURCE_TYPE_BITS	0x00001f00	/* Resource type */
#define IORESOURCE_IO		0x00000100	/* PCI/ISA I/O ports */
#define IORESOURCE_MEM		0x00000200
#define IORESOURCE_REG		0x00000300	/* Register offsets */
#define IORESOURCE_IRQ		0x00000400
#define IORESOURCE_DMA		0x00000800
#define IORESOURCE_BUS		0x00001000

那么节点中的各种描述怎么转换为resource结构呢?

  • 带有compatible属性的根下的子节点
  • 其他子节点带有compatible属性值为"simple-bus","simple-mfd","isa","arm,amba-bus "

不会转换的节点

  • 根节点 ,虽然带有compatible属性,但这个是为了最开始的机型匹配

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

        / {
              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>;
                      };
              };
          };
    

转换入口

of_platform_default_populate_init 这个函数是有特殊段属性的一个函数,会在kernel初始化的阶段来调用,具体这个段属性在以前的kernel解析中有类似的分析

// drivers/of/platform.c)
of_platform_default_populate_init 

arch_initcall_sync(of_platform_default_populate_init);
#define arch_initcall_sync(fn)		__define_initcall(fn, 3s)
#define __define_initcall(fn, id) ___define_initcall(fn, id, .initcall##id)
#define ___define_initcall(fn, id, __sec) \
	static initcall_t __initcall_##fn##id __used \
		__attribute__((__section__(#__sec ".init"))) = fn;

粗略的流程如下

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函数

节点转换流程

of_platform_default_populate_init

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;
                }
            }

of_platform_bus_create(含有递归)



of_platform_default_populate>of_platform_populate(...of_default_bus_match_table..) //of_default_bus_match_table 表示哪些子节点要被转换
	for_each_child_of_node
	{
-------	of_platform_bus_create(...)
|		{
|			of_platform_device_create_pdata(...)
|			{
|				of_device_alloc(...)
|				{
|					// 分配一个 platform_device ,里面会包含了资源文件的描述
|					dev = platform_device_alloc("", PLATFORM_DEVID_NONE);
|					// 计算内存(包括IORESOURCE_MEM IORESOURCE_IO )  中断 资源
|					of_address_to_resource(..)
|						__of_address_to_resource()	
|					of_irq_count(..)
|					kcalloc(...sizeof(resource)...)
|					// 设置具体的资源描述
|					dev->num_resources = num_reg + num_irq;
|					dev->resource = res;
|					dev->dev.parent = parent ? : &platform_bus;
|					// 设置总线名字
|					dev_set_name
|					of_device_make_bus_id
|					
|				}
|				dev->dev.bus = &platform_bus_type;
|				// 添加到dev 链表
|				of_device_add(...)
|					device_add(...)
|					
|				
|			}
|			
|			for_each_child_of_node(bus, child)
|			{
|-递归子节点---of_platform_bus_create(...)//这里就是递归了
				
			}
		}
		of_node_put(...)
	}

I2C_clinet

在上面的章节中,我们需要知道对于i2cclinet,并没有生成platform dev节点,这需要i2c驱动的具体处理生成clinet,也就是我们在添加adapt的时候,需要根据设备树去创建clinet

   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_clien

of_i2c_register_devices

of_i2c_register_devices
{
	// 寻找到节点名字为 i2c-bus
	of_get_child_by_name(adap->dev.of_node, "i2c-bus");
	of_i2c_register_device(...)
	{
		of_i2c_get_board_info(...)
		{
			of_modalias_node(...)
			{
				// 判断 compatible 属性
				of_get_property(node, "compatible", &cplen);
				
			}
			of_property_read_u32(node, "reg", &addr);
			of_property_read_bool(node, "host-notify")
			of_get_property(node, "wakeup-source", NULL)	
		}
		i2c_new_device(..)	
	}
	
}	

SPI

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);
}

匹配流程

我们先看devdriver的注册流程,可以看到最后的匹配函数是driver_match_device

// drivers/base/platform.c

a. 注册 platform_driver 的过程:
platform_driver_register
    __platform_driver_register
        drv->driver.probe = platform_drv_probe;
        driver_register
            bus_add_driver
                klist_add_tail(&priv->knode_bus, &bus->p->klist_drivers);    // 把 platform_driver 放入 platform_bus_type 的driver链表中
                driver_attach
                    bus_for_each_dev(drv->bus, NULL, drv, __driver_attach);  // 对于plarform_bus_type下的每一个设备, 调用__driver_attach
                        __driver_attach
                            ret = driver_match_device(drv, dev);  // 判断dev和drv是否匹配成功
                                        return drv->bus->match ? drv->bus->match(dev, drv) : 1;  // 调用 platform_bus_type.match
                            driver_probe_device(drv, dev);
                                        really_probe
                                            drv->probe  // platform_drv_probe
                                                platform_drv_probe
                                                    struct platform_driver *drv = to_platform_driver(_dev->driver);
                                                    drv->probe
                            
b. 注册 platform_device 的过程:
platform_device_register
    platform_device_add
        device_add
            bus_add_device
                klist_add_tail(&dev->p->knode_bus, &bus->p->klist_devices); // 把 platform_device 放入 platform_bus_type的device链表中
            bus_probe_device(dev);
                device_initial_probe
                    __device_attach
                        ret = bus_for_each_drv(dev->bus, NULL, &data, __device_attach_driver); // // 对于plarform_bus_type下的每一个driver, 调用 __device_attach_driver
                                    __device_attach_driver
                                        ret = driver_match_device(drv, dev);
                                                    return drv->bus->match ? drv->bus->match(dev, drv) : 1;  // 调用platform_bus_type.match
                                        driver_probe_device
                                            

driver_match_device

这个匹配函数也是属于总线匹配的一种,可以看到这个结构,具体为什么是这个结构,需要去看以前的知识点了

struct bus_type platform_bus_type = {
	.name		= "platform",
	.dev_groups	= platform_dev_groups,
	.match		= platform_match,
	.uevent		= platform_uevent,
	.dma_configure	= platform_dma_configure,
	.pm		= &platform_dev_pm_ops,
};

所以最终就是platform_match

platform_match

// 1. 匹配	pdev->driver_override 和  drv->name
/* When driver_override is set, only bind to the matching driver */
if (pdev->driver_override)
	return !strcmp(pdev->driver_override, drv->name);

// 2. 匹配 dev->of_node->properties中的compatible属性  drv->of_match_table
/* Attempt an OF style match first */
if (of_driver_match_device(dev, drv))
	of_match_device(drv->of_match_table, dev)
		of_match_node(matches, dev->of_node)
			__of_match_node(matches, node)
				for (; matches->name[0] || matches->type[0] || matches->compatible[0]; matches++)
					__of_device_is_compatible //寻找匹配的
						__of_find_property(device, "compatible", NULL)
					if (score > best_score) { //寻找到最匹配的
						best_match = matches;
						best_score = score;
					}
					
// 这个比较复杂,在arm基本不用
/* Then try ACPI style match */
if (acpi_driver_match_device(dev, drv))
	return 1;

// 4. 匹配 pdrv->id_table 和 pdev-name
if (pdrv->id_table)
	return platform_match_id(pdrv->id_table, pdev) != NULL;

// 5. 匹配 pdev->name 和 drv->name
return (strcmp(pdev->name, drv->name) == 0);

总结一下

  1. 比较 platform_dev.driver_override 和 platform_driver.drv->name
  2. 比较 platform_dev.dev.of_node的compatible属性 和 platform_driver.drv->of_match_table
  3. 比较 platform_dev.name 和 platform_driver.id_table
  4. 比较 platform_dev.name 和 platform_driver.drv->name

下面老师的这个图很形象具体了

mark

内核操作设备树函数

内核中设备树存在3个形式dtb -> device_node -> platform_device

处理DTB

of_fdt.h           // dtb文件的相关操作函数, 我们一般用不到, 因为dtb文件在内核中已经被转换为device_node树(它更易于使用)

处理device_node

of.h               // 提供设备树的一般处理函数, 比如 of_property_read_u32(读取某个属性的u32值), *of_get_child_count(获取某个device_node的子节点数)
of_address.h       // 地址相关的函数, 比如 of_get_address(获得reg属性中的addr, size值)
of_match_device(从matches数组中取出与当前设备最匹配的一项)
of_dma.h           // 设备树中DMA相关属性的函数
of_gpio.h          // GPIO相关的函数
of_graph.h         // GPU相关驱动中用到的函数, 从设备树中获得GPU信息
of_iommu.h         // 很少用到
of_irq.h           // 中断相关的函数
of_mdio.h          // MDIO (Ethernet PHY) API
of_net.h           // OF helpers for network devices. 
of_pci.h           // PCI相关函数
of_pdt.h           // 很少用到
of_reserved_mem.h  // reserved_mem的相关函数

example

// 官方设备树规格书里面的设备示例
soc {
#address-cells = <1>;
#size-cells = <1>;
serial {
compatible = "ns16550";
reg = <0x4600 0x100>;
clock-frequency = <0>;
interrupts = <0xA 0x8>;
interrupt-parent = <&ipic>;
};
};

//解析方法
int of_irq_parse_one(struct device_node *device, int index, struct of_phandle_args *out_irq);
//或者使用
int of_irq_parse_raw(const __be32 *addr, struct of_phandle_args *out_irq);

处理 platform_device

of_platform.h      // 把device_node转换为platform_device时用到的函数, 

example

/* Platform drivers register/unregister */
extern struct platform_device *of_device_alloc(struct device_node *np,
					 const char *bus_id,
					 struct device *parent);
//这个函数转换节点为platform_device 中大量使用
// 比如of_device_alloc(根据device_node分配设置platform_device), 
//     of_find_device_by_node (根据device_node查找到platform_device),
//     of_platform_bus_probe (处理device_node及它的子节点)
of_device.h        // 设备相关的函数, 比如 of_match_device

根文件系统查看设备树

  1. dtb文件

    hexdump -C /sys/firmware/fdt
    
  2. 目录形式展示

    /sys/firmware/devicetree/base
    
  3. 具体的dev文件被创建,如果是通过设备树创建的,则在具体的dev下有of_node这个文件夹,这个of_node是指向2中/sys/firmware/devicetree中具体的设备目录

    /sys/devices/platform
    
    /sys/devices/platform/<设备名>/of_node > /sys/firmware/devicetree/base
    

参考链接

http://wiki.100ask.org

推荐阅读