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
 
 

 
 
 
 
 
 
 
