of_property_read_string_index(转)
https://biscuitos.github.io/blog/DTS-of_property_read_string_index/
源码分析
of_property_read_string_index
of_property_read_string_index
|
|---of_property_read_string_helper
|
|---of_find_property
|
|---of_find_property
|
|---__of_find_property
|
|---of_prop_cmp
平台: ARM32
linux: 3.10/4.18/5.0
函数作用:从 string-list 属性中,读出第 N 个字符串。
of_property_read_string_index
/**
* of_property_read_string_index() - Find and read a string from a multiple
* strings property.
* @np: device node from which the property value is to be read.
* @propname: name of the property to be searched.
* @index: index of the string in the list of strings
* @out_string: pointer to null terminated return string, modified only if
* return value is 0.
*
* Search for a property in a device tree node and retrieve a null
* terminated string value (pointer to data, not a copy) in the list of strings
* contained in that property.
* Returns 0 on success, -EINVAL if the property does not exist, -ENODATA if
* property does not have a value, and -EILSEQ if the string is not
* null-terminated within the length of the property data.
*
* The out_string pointer is modified only if a valid string can be decoded.
*/
static inline int of_property_read_string_index(struct device_node *np,
const char *propname,
int index, const char **output)
{
int rc = of_property_read_string_helper(np, propname, output, 1, index);
return rc < 0 ? rc : 0;
}
参数 np 指向设备节点;propname 指向属性名字;output 参数用于存储指定的字符 串;index 用于指定字符串在 string-list 中的索引。
函数直接调用 of_property_read_string_helper() 函数获得多个字符串。
of_property_read_string_helper
/**
* of_property_read_string_util() - Utility helper for parsing string properties
* @np: device node from which the property value is to be read.
* @propname: name of the property to be searched.
* @out_strs: output array of string pointers.
* @sz: number of array elements to read.
* @skip: Number of strings to skip over at beginning of list.
*
* Don't call this function directly. It is a utility helper for the
* of_property_read_string*() family of functions.
*/
int of_property_read_string_helper(struct device_node *np, const char *propname,
const char **out_strs, size_t sz, int skip)
{
struct property *prop = of_find_property(np, propname, NULL);
int l = 0, i = 0;
const char *p, *end;
if (!prop)
return -EINVAL;
if (!prop->value)
return -ENODATA;
p = prop->value;
end = p + prop->length;
for (i = 0; p < end && (!out_strs || i < skip + sz); i++, p += l) {
l = strnlen(p, end - p) + 1;
if (p + l > end)
return -EILSEQ;
if (out_strs && i >= skip)
*out_strs++ = p;
}
i -= skip;
return i <= 0 ? -ENODATA : i;
}
EXPORT_SYMBOL_GPL(of_property_read_string_helper);
参数 np 指向设备节点;propname 指向属性名字;out_strs 参数用于存储指定的字符 串;sz 参数指定了读取字符串的数量;skip 参数指定了从第几个字符串开始读取。
函数首先调用 of_find_property() 函数获得 propname 对应的属性,然后对获得的属性 和属性值进行有效性检查,检查不过直接返回错误;如果检查通过,接着计算属性的结束 地址后,使用 for 循环遍历属性的值,并且跳过 skip 对应的地址,然后将字符串都存 储在 out_strs 参数里。
of_find_property
struct property *of_find_property(const struct device_node *np,
const char *name,
int *lenp)
{
struct property *pp;
unsigned long flags;
raw_spin_lock_irqsave(&devtree_lock, flags);
pp = __of_find_property(np, name, lenp);
raw_spin_unlock_irqrestore(&devtree_lock, flags);
return pp;
}
EXPORT_SYMBOL(of_find_property);
参数 np 指向一个 device_node;name 参数表示需要查看属性的名字;lenp 用于存储 属性长度。
函数通过调用 raw_spin_lock_irqsave() 函数加锁之后,就直接调用 __of_find_property() 函数。__of_find_property() 函数用于查找并返回属性名字与 name 一致的属性。调用完 __of_find_property() 函数之后,调用 raw_spin_unlock_irqrestore() 函数解锁。最后返回 property 结构。
__of_find_property
static struct property *__of_find_property(const struct device_node *np,
const char *name, int *lenp)
{
struct property *pp;
if (!np)
return NULL;
for (pp = np->properties; pp; pp = pp->next) {
if (of_prop_cmp(pp->name, name) == 0) {
if (lenp)
*lenp = pp->length;
break;
}
}
return pp;
}
由于每个 device_node 包含一个名为 properties 的成员,properties 是一个单链表的 表头,这个单链表包含了该节点的所有属性,函数通过使用 for 循环遍历这个链表,以 此遍历节点所包含的所有属性。每遍历一个属性就会调用 of_prop_cmp() 函数, of_prop_cmp() 函数用于对比两个字符串是否相等,如果相等返回 0. 因此,当遍历到 的属性的名字与参数 name 一致,那么定义为找到了指定的属性。此时,如果 lenp 不 为 NULL,那么会将属性的长度即 length 存储到 lenp 指向的地址。
实践
实践目的是在 DTS 文件中构建一个私有节点,私有节点包含一个 string-list 属性,属 性里面包含了多个字符串,然后通过of_property_read_string_index() 函数读取第 N 个字符串,函数定义如下:
static inline int of_property_read_string_index(struct device_node *np,
const char *propname,
int index, const char **output)
这个函数经常用用于读取 string-list 属性中的第 N 个字符。
本文实践基于 Linux 4.20.8 arm32 平台,开发者可以参考如下文章快速搭建一个 调试环境:
DTS 文件
由于使用的平台是 ARM32,所以在源码 /arch/arm/boot/dts 目录下创建一个 DTSI 文 件,在 root 节点之下创建一个名为 DTS_demo 的子节点。节点包含名为 BiscutisOS-strings 的属性,属性值为 “uboot”, “kernel”, “rootfs”, “BiscuitOS” 四个字符串。具体内容如下:
/*
* DTS Demo Code
*
* (C) 2019.01.06 <buddy.zhang@aliyun.com>
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License version 2 as
* published by the Free Software Foundation.
*/
/ {
DTS_demo {
compatible = "DTS_demo, BiscuitOS";
status = "okay";
BiscuitOS-strings = "uboot", "kernel", "rootfs", "BiscuitOS";
};
};
创建完毕之后,将其保存并命名为 DTS_demo.dtsi。然后开发者在 Linux 4.20.8 的源 码中,找到 arch/arm/boot/dts/vexpress-v2p-ca9.dts 文件,然后在文件开始地方添 加如下内容:
#include "DTS_demo.dtsi"
编写 DTS 对应驱动
准备好 DTSI 文件之后,开发者编写一个简单的驱动,这个驱动作为 DTS_demo 的设备 驱动,在 DTS 机制遍历时会调用匹配成功的驱动,最终运行驱动里面的代码。在驱动的 probe 函数中,首先获得驱动所对应的节点,通过 platform_device 的 of_node 成员传 递。获得驱动对应的节点之后,通过调用 of_property_read_string_index() 函数获得 “BiscuitOS-strings”属性中第二个字符串,并打印属性值,驱动编写如下:
/*
* DTS: of_property_match_string
* of_property_count_strings
* of_property_read_string_index
* of_property_read_string_array
* of_property_read_string_helper
*
* int of_property_match_string(struct device_node *np, const char *propname,
* const char *string)
*
* static inline int of_property_count_strings(struct device_node *np,
* const char *propname)
*
* static inline int of_property_read_string_index(struct device_node *np,
* const char *propname,
* int index, const char **output)
*
* static inline int of_property_read_string_array(struct device_node *np,
* const char *propname,
* const char **out_strs, size_t sz)
*
* int of_property_read_string_helper(struct device_node *np,
* const char *propname,
* const char *out_strs,
* size_t sz, int skip)
*
* (C) 2019.01.11 BuddyZhang1 <buddy.zhang@aliyun.com>
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License version 2 as
* published by the Free Software Foundation.
*/
/*
* Private DTS file: DTS_demo.dtsi
*
* / {
* DTS_demo {
* compatible = "DTS_demo, BiscuitOS";
* status = "okay";
* BiscuitOS-strings = "uboot", "kernel",
* "rootfs", "BiscuitOS";
* };
* };
*
* On Core dtsi:
*
* include "DTS_demo.dtsi"
*/
#include <linux/init.h>
#include <linux/kernel.h>
#include <linux/of_platform.h>
/* define name for device and driver */
#define DEV_NAME "DTS_demo"
/* probe platform driver */
static int DTS_demo_probe(struct platform_device *pdev)
{
struct device_node *np = pdev->dev.of_node;
const char *string_array[4];
const char *string;
int count, index;
int rc;
printk("DTS demo probe entence.\n");
/* Verify property whether contains special string. */
rc = of_property_match_string(np, "BiscuitOS-strings", "rootfs");
if (rc < 0)
printk("BiscuitOS-strings doesn't contain \"rootfs\"\n");
else
printk("BiscuitOS-strings[%d] is \"rootfs\"\n", rc);
/* Count the string on string-list property */
count = of_property_count_strings(np, "BiscuitOS-strings");
printk("String count: %#x\n", count);
/* Read special string on string-list property with index */
for (index = 0; index < count; index++) {
rc = of_property_read_string_index(np, "BiscuitOS-strings",
index, &string);
if (rc < 0) {
printk("Unable to read BiscuitOS-strings[%d]\n", index);
continue;
}
printk("BiscuitOS-strings[%d]: %s\n", index, string);
}
/* Read number of strings from string-list property */
rc = of_property_read_string_array(np, "BiscuitOS-strings",
string_array, count);
if (rc < 0) {
printk("Faild to invoke of_property_read_string_array()\n");
return -1;
}
for (index = 0; index < count; index++)
printk("String_array[%d]: %s\n", index, string_array[index]);
/* Read a string with index on string-list property (index = 2) */
rc = of_property_read_string_helper(np, "BiscuitOS-strings",
&string, 1, 2);
if (rc < 0)
printk("Faild to invoke of_property_read_string_helper()!\n");
else
printk("BiscuitOS-strings[index=2]: %s\n", string);
return 0;
}
/* remove platform driver */
static int DTS_demo_remove(struct platform_device *pdev)
{
return 0;
}
static const struct of_device_id DTS_demo_of_match[] = {
{ .compatible = "DTS_demo, BiscuitOS", },
{ },
};
MODULE_DEVICE_TABLE(of, DTS_demo_of_match);
/* platform driver information */
static struct platform_driver DTS_demo_driver = {
.probe = DTS_demo_probe,
.remove = DTS_demo_remove,
.driver = {
.owner = THIS_MODULE,
.name = DEV_NAME, /* Same as device name */
.of_match_table = DTS_demo_of_match,
},
};
module_platform_driver(DTS_demo_driver);
编写好驱动之后,将其编译进内核。编译内核和 dts,如下命令:
make ARCH=arm BiscuitOS/output/linux-4.20.8/arm-linux-gnueabi/arm-linux-gnueabi/bin/arm-linux-gnueabi- j8
make ARCH=arm BiscuitOS/output/linux-4.20.8/arm-linux-gnueabi/arm-linux-gnueabi/bin/arm-linux-gnueabi- dtbs
启动内核,在启动阶段就会运行驱动的 probe 函数,并打印如下信息:
[ 3.580217] DTS demo probe entence.
[ 3.580233] BiscuitOS-strings[2] is "rootfs"
[ 3.580247] String count: 0x4
[ 3.580258] BiscuitOS-strings[0]: uboot
[ 3.580271] BiscuitOS-strings[1]: kernel
[ 3.580284] BiscuitOS-strings[2]: rootfs
[ 3.580297] BiscuitOS-strings[3]: BiscuitOS
[ 3.580311] String_array[0]: uboot
[ 3.580322] String_array[1]: kernel
[ 3.580334] String_array[2]: rootfs
[ 3.580346] String_array[3]: BiscuitOS
[ 3.580358] BiscuitOS-strings[index=2]: rootfs
of_property_read_string_index(转)的更多相关文章
- ARM Linux 3.x的设备树(Device Tree)
http://blog.csdn.net/21cnbao/article/details/8457546 宋宝华 Barry Song <21cnbao@gmail.com> 1. ...
- ARM Linux 3.x的设备树(Device Tree)
1. ARM Device Tree起源 Linus Torvalds在2011年3月17日的ARM Linux邮件列表宣称“this whole ARM thing is a f*cking pai ...
- linux内核中的GPIO系统之(2):pin control subsystem
一.前言 在linux2.6内核上工作的嵌入式软件工程师在pin control上都会遇到这样的状况: (1)启动一个新的项目后,需要根据硬件平台的设定进行pin control相关的编码.例如:在b ...
- [dts]Device Tree机制
转自:http://blog.csdn.net/machiner1/article/details/47805069 ------------------Based on linux 3.10.24 ...
- 【转】 ARM Linux 3.x的设备树(Device Tree)
1. ARM Device Tree起源 http://blog.csdn.net/21cnbao/article/details/8457546 Linus Torvalds在2011年3月1 ...
- 【转】ARM Linux 3.x的设备树(Device Tree)
原文网址:http://blog.csdn.net/21cnbao/article/details/8457546 1. ARM Device Tree起源 Linus Torvalds在201 ...
- 基於tiny4412的Linux內核移植 --- 实例学习中断背后的知识(1)
作者:彭东林 邮箱:pengdonglin137@163.com QQ:405728433 平台 tiny4412 ADK Linux-4.9 概述 前面几篇博文列举了在有设备树的时候,gpio中断的 ...
- 高通ASOC中的machine驱动
ASoC被分为Machine.Platform和Codec三大部分,其中的Machine驱动负责Platform和Codec之间的耦合以及部分和设备或板子特定的代码,再次引用上一节的内容:Machin ...
- [dts]Device Tree机制【转】
转自:https://www.cnblogs.com/aaronLinux/p/5496559.html 转自:http://blog.csdn.net/machiner1/article/detai ...
随机推荐
- Android模拟器太慢怎么办?使用微软的VS模拟器
开发过android的人都知道,android模拟器非常的慢,推荐使用微软的VS模拟器. (1)到 https://visualstudio.microsoft.com/zh-hans/vs/msft ...
- jquery validate 动态生成的多个同名input的验证
我的应用场景是,添加和修改入库单的明细,明细是以表格的形式呈现,可以动态添加商品,用jquery.validate插件做数据验证. 由于jquery.validate插件验证同名的input时只验证第 ...
- LearnOpenGL笔记(3)着色器
GLSL向量的一些操作 vec2 someVec; vec4 differentVec = someVec.xyxx; vec3 anotherVec = differentVec.zyw; vec4 ...
- 字节输出流FileOutputStream
#字节流 字节输出流FileOutputStream 创建输出流对象 OutputStream 流对象是一个抽象类,不能实例化.所以,我们要找一个具体的子类 :FileOutputStream. 查看 ...
- English--介词省略句型与总结
English|介词省略句型与总结 本篇文章将会介绍介词的省略与整个语法内容的总结.小板凳都带上,准备开始了! 前言 目前所有的文章思想格式都是:知识+情感. 知识:对于所有的知识点的描述.力求不含任 ...
- java request.getInputStream中文乱码解决方案
请求时要指定为UTF-8,中文码码完美解决 /** * * 得到请求body字符串,一般用于content-type:application/json * */ public static Strin ...
- es严格模式、对象和扩展。
01. 严格模式 1. 理解: * 除了正常运行模式(混杂模式),ES5添加了第二种运行模式:"严格模式"(strict mode). * 顾名思义,这种模式使得Javascrip ...
- Clang交叉编译初识
最近工作中要编译一个第三方的C库用于iOS端使用,我直接在Mac OS的终端中./configure & make & make install常规走下来,却无法在真机iOS上使用,提 ...
- 安恒Red Team 内部红蓝对抗框架
0x00 准备钓鱼攻击(从公开资源) 1.常见的红队攻击向量和技术 2.常见的蓝队侦查和预防控制 0x02 发送钓鱼邮件(到目标组织员工邮箱地址) 1.常见的红队攻击向量和技术 2.常见的蓝 ...
- OL8.0静默安装Oracle 19C
首先在edelivery中下载Oracle Linux 8.0 然后就默认安装系统 环境准备工具目前不支持OL8,所以需要手动安装,首先设置内核参数,在/etc/sysctl.conf追加 [root ...