properties的value可以是empty或是以下資料型態:
- Text strings (null terminated) are represented with double quotes: string-property = "a string";
- 'Cells' are 32 bit unsigned integers delimited by angle brackets: cell-property = <0xbeef 123 0xabcd1234>;
- Binary data is delimited with square brackets: binary-property = [0x01 0x23 0x45 0x67];
- Data of differing representations can be concatenated together using a comma: mixed-property = "a string", [0x01 0x23 0x45 0x67], <0x12345678>;
- Commas are also used to create lists of strings: string-list = "red fish", "blue fish";
這章節介紹幾個API,用於存取這些常見的資料型態,會以下的DTS的內容進行parse
/ { node1 { compatible = "brook,dts-test"; a-string-property = "A string"; a-string-list-property = "first string", "second string"; // hex is implied in byte arrays. no '0x' prefix is required a-byte-data-property = [01 23 34 56]; child-node1 { first-child-property; second-child-property = <1>; a-string-property = "Hello, world"; }; child-node2 { }; }; node2 { compatible = "brook,dts-test"; an-empty-property; a-cell-property = <1 2 3 4>; /* each number (cell) is a uint32 */ child-node@1 { }; child-node@2 { }; }; };
- of_match_device(): Sanity check for device that device is matching with the node
- of_property_read_string(): To read string property
- of_property_count_strings(): Find and return the number of strings from a multiple strings property.
- of_property_read_string_index(): Read a string with index from string-list property.
- of_find_property(): Find and return the property pointer by giving named
- of_property_for_each_u32(): A macro to iterate the property to get all values
- of_property_read_bool(): Returns true if the property exist false otherwise
- of_find_node_by_name(): Find a node by its "name" property
- for_each_child_of_node(): Traverse all child device node for current device node
#include <linux/module.h> #include <linux/platform_device.h> #include <linux/of.h> #include <linux/of_device.h> static struct of_device_id brook_dt_id[] = { { .compatible = "brook,dts-test", }, {} }; MODULE_DEVICE_TABLE(of, brook_dt_id); static int brook_dts_probe(struct platform_device *pdev) { const struct of_device_id *of_id; struct device_node *node, *child; struct property *pp; int i, ret, of_cnt, len; const char *str; const __be32 *cur; u32 val; // sanity check of_match_device() allows to get the matching entry of_id = of_match_device(brook_dt_id, &pdev->dev); if (!of_id) { pr_err("%s: of_id is NULL\n", __func__); return -1; } node = pdev->dev.of_node; // To read string property ret = of_property_read_string(node, "a-string-property", &str); if (!ret) { printk("a-string-property: %s\n", str); } else { printk("no a-string-property\n"); } // Find and return the number of strings from a multiple strings property. of_cnt = of_property_count_strings(node, "a-string-list-property"); if (of_cnt) { for (i = 0; i < of_cnt; i++) { ret = of_property_read_string_index(node, "a-string-list-property", i, &str); if (!ret) { printk("a-string-list-property[%d]: %s\n", i, str); } } } // to find aproperty named if arg2 pp = of_find_property(node, "a-byte-data-property", &len); if (pp) { u8 *u8p = pp->value; printk("a-byte-data-property: len:%d, pp->len:%d", len, pp->length); for (i = 0; i < len; i++) { printk("a-byte-data-property:[%d] = %02x\n", i, u8p[i]); } } // to find aproperty named if arg2 of_property_for_each_u32(node, "a-cell-property", pp, cur, val) { printk("a-cell-property: %04x/%04x\n", *cur, val); } // Returns true if the property exist false otherwise if (of_property_read_bool(node, "an-empty-property")) { printk("has an-empty-property\n"); } else { printk("no an-empty-property\n"); } // Traverse all child device node for current device node for_each_child_of_node(node, child) { printk("child name: %s\n", child->name); } // Find a node by its "name" property child = of_find_node_by_name(node, "child-node1"); if (child) { printk("%s has a child %s\n", node->name, child->name); } return 0; } static int brook_dts_remove(struct platform_device *pdev) { return 0; } static struct platform_driver brook_dts_platform_driver = { .probe = brook_dts_probe, .remove = brook_dts_remove, .driver = { .owner = THIS_MODULE, .name = "brook-dts", .of_match_table = brook_dt_id, }, }; static int __init brook_init_module(void) { return platform_driver_register(&brook_dts_platform_driver); } static void __exit brook_exit_module(void) { platform_driver_unregister(&brook_dts_platform_driver); } module_init(brook_init_module); module_exit(brook_exit_module); MODULE_LICENSE("GPL");
執行結果如下:
/ # insmod dts.ko a-string-property: A string a-string-list-property[0]: first string a-string-list-property[1]: second string a-byte-data-property: len:4, pp->len:4 a-byte-data-property:[0] = 01 a-byte-data-property:[1] = 23 a-byte-data-property:[2] = 34 a-byte-data-property:[3] = 56 no an-empty-property child name: child-node1 child name: child-node2 node1 has a child child-node1 no a-string-property a-cell-property: 1000000/0001 a-cell-property: 2000000/0002 a-cell-property: 3000000/0003 a-cell-property: 4000000/0004 has an-empty-property child name: child-node child name: child-node
以下摘錄http://www.myexception.cn/linux-unix/1910031.html, linux下devicetree中常用的of函數
linux下devicetree中常用的of函数 从device_node中获取信息: int of_property_read_u8_array(const struct device_node *np, const char *propname,u8 *out_values, size_t sz); int of_property_read_u16_array(const struct device_node *np, const char *propname,u16 *out_values, size_t sz); int of_property_read_u32_array(const struct device_node *np, const char *propname,u32 *out_values, size_t sz); 从设备结点np中读取属性名为propname,类型为8、16、32、位整型数组的属性值,并放入out_values,sz指明了要读取的个数。 static inline int of_property_read_u8(const struct device_node *np,const char *propname,u8 *out_value) static inline int of_property_read_u16(const struct device_node *np,const char *propname,u8 *out_value) static inline int of_property_read_u32(const struct device_node *np,const char *propname,u8 *out_value) 从设备结点np中读取属性名为propname,类型为8、16、32位的属性值,并放入out_values。实际上这里调用的就是sz为1的XXX_array函数。 int of_property_read_u32_index(const struct device_node *np,const char*propname,u32 index, u32 *out_value) 从设备结点np中读取属性名为propname的属性值中第index个u32数值给out_value int of_property_read_u64(conststruct device_node *np, const char *propname,u64 *out_value) 从设备结点np中读取属性名为propname,类型为64位的属性值,并放入out_values int of_property_read_string(struct device_node *np, const char *propname,const char**out_string) 从设备结点np中读取属性名为propname的字符串型属性值 int of_property_read_string_index(struct device_node *np, const char *propname,intindex, const char **output) 从设备结点np中读取属性名为propname的字符串型属性值数组中的第index个字符串 int of_property_count_strings(struct device_node *np, const char *propname) 从设备结点np中读取属性名为propname的字符串型属性值的个数 unsigned int irq_of_parse_and_map(struct device_node *dev, int index) 从设备节点dev中读取第index个irq号 int of_irq_to_resource(struct device_node *dev, int index, struct resource *r) 从设备节点dev中读取第index个irq号,并填充一个irq资源结构体 int of_irq_count(struct device_node *dev) 获取设备节点dev的irq个数 static inline bool of_property_read_bool(const struct device_node *np,const char *propname); 如果设备结点np含有propname属性,则返回true,否则返回false。一般用于检查空属性是否存在。 struct property* of_find_property(const struct device_node *np,const char *name,int *lenp) 根据name参数,在指定的设备结点np中查找匹配的property,并返回这个property const void * of_get_property(const struct device_node *np, const char *name,int *lenp) 根据name参数,在指定的设备结点np中查找匹配的property,并返回这个property的属性值 struct device_node* of_get_parent(const struct device_node *node) 获得node节点的父节点的device node int of_device_is_compatible(const struct device_node *device,const char *compat); 判断设备结点device的compatible属性是否包含compat指定的字符串 从of_allnodes中查找信息: struct device_node* of_find_node_by_path(const char *path) 根据路径参数,在全局链表of_allnodes中,查找匹配的device_node struct device_node* of_find_node_by_name(struct device_node *from,const char *name) 则根据name在全局链表of_allnodes中查找匹配的device_node,若from=NULL表示从头开始查找 struct device_node* of_find_node_by_type(struct device_node *from,const char *type) 根据设备类型在全局链表of_allnodes中查找匹配的device_node struct device_node * of_find_compatible_node(struct device_node *from, const char*type, const char,*compatible); 根据compatible的属性值在全局链表of_allnodes中查找匹配的device_node,大多数情况下,from、type为NULL。 struct device_node* of_find_node_with_property(struct device_node *from,const char *prop_name) 根据节点属性的name在全局链表of_allnodes中查找匹配的device_node struct device_node* of_find_node_by_phandle(phandle handle) 根据phandle在全局链表of_allnodes中查找匹配的device_node 杂: void __iomem* of_iomap(struct device_node *node, int index); 通过设备结点直接进行设备内存区间的 ioremap(),index是内存段的索引。若设备结点的reg属性有多段,可通过index标示要ioremap的是哪一段,只有1段的情况,index为0 unsigned long __init of_get_flat_dt_root(void) 用来查找在dtb中的根节点,好像返回的都是0 int of_alias_get_id(struct device_node *np, const char *stem) 获取节点np对应的aliasid号 struct device_node* of_node_get(struct device_node *node) void of_node_put(struct device_node *node) device node计数增加/减少 const struct of_device_id* of_match_node(const struct of_device_id *matches,const struct device_node*node) 将matches数组中of_device_id结构的name和type与device node的compatible和type匹配,返回匹配度最高的of_device_id结构 platform_device和resource相关: int of_address_to_resource(struct device_node *dev, int index,struct resource *r) 根据设备节点dev的reg属性值,填充资源结构体r。Index参数指明了使用reg属性中第几个属性值,一般设置为0,表示第一个。 struct platform_device* of_device_alloc(struct device_node *np,const char *bus_id,struct device *parent) 根据device node,bus_id以及父节点创建该设备的platform_device结构,同时会初始化它的resource成员。 int of_platform_bus_probe(struct device_node *root,const struct of_device_id *matches,struct device *parent) 遍历of_allnodes中的节点挂接到of_platform_bus_type总线上,由于此时of_platform_bus_type总线上还没有驱动,所以此时不进行匹配 int of_platform_populate(struct device_node *root,const struct of_device_id *matches,const struct of_dev_auxdata *lookup,struct device *parent) 遍历of_allnodes中的所有节点,生成并初始化所以节点的platform_device结构 struct platform_device* of_find_device_by_node(struct device_node *np) 根据device_node查找返回该设备对应的platform_device结构
-
參考資料:
- http://www.myexception.cn/linux-unix/1910031.html, linux下devicetree中常用的of函數
- https://saurabhsengarblog.wordpress.com/2015/11/28/device-tree-tutorial-arm/, Device Tree Tutorial (ARM)
- https://www.cnblogs.com/xiaojiang1025/p/6368260.html, Linux内核 设备树操作常用API