背景

讲完设备树的有关概念以及语法以后,我们接下来就让 我们的驱动 使用 设备树。

ref : 《内核学习笔记14:内核设备树学习》《u-boot对设备树的支持》

测试代码

本文使用的设备树节点如下:

// 专门用于测试dts的示例,没实例用途
// 名称可以有“,”、“-”,如“ll,i2c-enable”
myfoo {
compatible = "ll,jimkent-foo";
status = "okay"; // string
enable; // bool,无须值
myvalue = <250>; // 默认是32位,如果使用8位读取,结果为0
value = /bits/ 8 <88>; // 8位单独赋值
value16 = /bits/ 16 <166>; // 16位单独赋值
a-cell = <1 2 3 4 5>; // 数组
// 子节点
foo {
label = "foo";
note = "this is foo";
};
bar {
label = "bar";
note = "this is bar";
};
}; // 其中compatible与驱动使用的名称必须一致(这样才能匹配上)。其它内容比较简单,分别是字符串、布尔类型、不同位数的数值、数组、子节点。

驱动实例如下:

/**
* @file foo_drv.c
* @author Late Lee <latelee@163.com>
* @date Wed Jun 7 22:21:19 2019
*
* @brief 测试dts示例
*
* @note 读取dts的值,学习dts。代码有部分警告,不影响
*/ #include <linux/module.h>
#include <linux/kernel.h> /**< printk() */
#include <linux/init.h> #include <linux/types.h> /**< size_t */
#include <linux/errno.h> /**< error codes */
#include <linux/string.h> #include <linux/of.h>
#include <linux/of_device.h> static int foo_remove(struct platform_device *dev)
{
printk(KERN_NOTICE "remove...\n"); return 0;
} static int foo_probe(struct platform_device *dev)
{
int ret = 0;
struct device_node* np = dev->dev.of_node;
struct device_node* child = NULL;
const char* str = NULL;
bool enable = false;
u8 value = 0;
u16 value16 = 0;
u32 value32 = 0; // 测试dts读取API
if(np == NULL)
{
pr_info("of_node is NULL\n");
return 0;
} of_property_read_string(np, "status", &str); // 读字符串 enable = of_property_read_bool(np, "enable"); // bool类型,可判断某字段存在不存在
of_property_read_u32(np, "myvalue", &value32); // 一般地,都使用u32读取数值
of_property_read_u8(np, "value", &value);
of_property_read_u16(np, "value16", &value16); u32 data[3] = {0};
u32 tag = 0;
// a-cell是一个数组,默认读第1个。
of_property_read_u32(np, "a-cell", &tag);
// 也可以读取指定大小的数组(不一定是全部的)
of_property_read_u32_array(np, "a-cell", data, ARRAY_SIZE(data)); printk("of read status: %s enable: %d value: %d %d %d\n", str, enable, value, value16, value32);
printk("of read tag: %d data: %d %d %d\n", tag, data[0], data[1], data[2]); // 获取子节点个数
int count = of_get_available_child_count(np); // 遍历所有子节点,按格式读取属性
int index = 0;
for_each_available_child_of_node(np,child)
{
const char* label = of_get_property(child,"label",NULL) ? : child->name;
const char* note = of_get_property(child,"note",NULL) ? : child->name;
printk("of read: label: %s note: %s\n", label, note);
}
return ret;
} static struct of_device_id foo_of_match[] = {
{ .compatible = "ll,jimkent-foo", },
{ /* sentinel */ }
}; static struct platform_driver foo_driver = {
.driver = {
.name = "foo",
.of_match_table = of_match_ptr(foo_of_match),
},
.probe = foo_probe,
.remove = foo_remove,
}; static int __init foo_drv_init(void)
{
int ret = 0; ret = platform_driver_register(&foo_driver);
if (ret)
{
pr_info("platform_driver_register failed!\n");
return ret;
} pr_info("Init OK!\n"); return ret;
} static void __exit foo_drv_exit(void)
{
platform_driver_unregister(&foo_driver);
} module_init(foo_drv_init);
module_exit(foo_drv_exit); MODULE_AUTHOR("Late Lee");
MODULE_DESCRIPTION("Simple platform driver");
MODULE_LICENSE("GPL");
MODULE_ALIAS("platform:foo");

示例的代码是一个简单的模板,除了学习dts外,没什么用处。但是可以以此展开复杂的、有实际用途的驱动。

与以前的platform驱动不同,platform_driver中指定of_match_tablefoo_of_match结构体的.compatible必须与设备树的compatible一致。

本驱动涉及到的读取设备树节点信息的函数如下,更多函数,参考内核源码的include/linux/of.h头文件:

of_property_read_string // 读取字符串
of_property_read_bool // 判断某个字段是否存在,无须赋值
of_property_read_u8 // 读取8比特
of_property_read_u16 // 读取16比特
of_property_read_u32 // 读取32比特

如果存在多个子节点,用of_get_available_child_count获取个数(可用于开辟内存),然后调用for_each_available_child_of_node遍历所有子节点,注意,of_get_propertyof_property_read_string有相同效果。只是用法不同而已。

设备树DTS 学习:4-编写实战的更多相关文章

  1. 设备树DTS 学习:2-设备树语法

    背景 通过上一讲了解完设备树DTS有关概念,我们这一讲就来基于设备树例程,学习设备树的语法规则. 参考:设备树详解dts.设备树语法详解.设备树使用总结 设备树框架 1个dts文件 + n个dtsi文 ...

  2. 设备树DTS 学习:1-有关概念

    背景 设备树在Linux驱动开发中是一种比较常用的架构. 参考:<设备树DTS使用总结> .<linux内核设备树及编译> Linux设备树 介绍 在Linux 2.6中,ar ...

  3. 设备树DTS 学习:Linux DTS文件加载过程

    背景 了解机制有利于对内核有更深的认识. wget https://mirrors.aliyun.com/linux-kernel/v3.x/linux-3.2.61.tar.xz 内核 在drive ...

  4. 设备树DTS 学习:3-常用的DTS 函数

    Linux内核中目前DTS相关的函数都是以of_前缀开头的,它们的实现位于内核源码的drivers/of下面 void __iomem*of_iomap(struct device_node *nod ...

  5. 设备树DTS 学习: uboot 传递 dtb 给 内核

    背景 得到 dtb 文件以后,我们需要想办法下载到 板子中,并给 Linux 内核使用. (高级版本的 uboot也有了 自己使用设备树支持,我们这里不讨论 uboot 使用的设备树) Linux 内 ...

  6. Linux dts 设备树详解(二) 动手编写设备树dts

    Linux dts 设备树详解(一) 基础知识 Linux dts 设备树详解(二) 动手编写设备树dts 文章目录 前言 硬件结构 设备树dts文件 前言 在简单了解概念之后,我们可以开始尝试写一个 ...

  7. Linux 设备树 dts

    1. dtb反编译成dts文件命令:./kernel-4.4/scripts/dtc/dtc_overlay -I dtb -O dts out/target/product/m863ur100_p0 ...

  8. Linux dts 设备树详解(一) 基础知识

    Linux dts 设备树详解(一) 基础知识 Linux dts 设备树详解(二) 动手编写设备树dts 文章目录 1 前言 2 概念 2.1 什么是设备树 dts(device tree)? 2. ...

  9. Android驱动之设备树简介

    目录 一.    设备树简介    2 1.    问题一:为什么需要设备树?    2 ①名词解释:    2 ②DT详细介绍:    2 ③DTS是DT的源文件,描述Device Tree中的设备 ...

随机推荐

  1. 【PAT甲级】1046 Shortest Distance (20 分)

    题意: 输入一个正整数N(<=1e5),代表出口的数量,接下来输入N个正整数表示当前出口到下一个出口的距离.接着输入一个正整数M(<=10000),代表询问的次数,每次询问输入两个出口的序 ...

  2. 解决误删libc.so.6过程的参考资料

    说说前因后果:因为之前安装了filezilla,发现安装不了,说是libc.so.6老旧了.差不多过了一个星期,也就是前天升级gcc成功后决定再解决这个问题.gcc升级成功后,决定创建新链接...然后 ...

  3. Android抓包分析-fiddler版

    本文介绍的是如何使用Fiddler工具抓取Android应用的HTTP协议的数据包 工具 Genymotion模拟器 笔记本电脑一台(Win7) Fiddler(v4.6.2),下载地址:http:/ ...

  4. CentOS7.3下载地址

    CentOS 7.3,是CentOS-7系列的第四个发行版本,官方版本号为7.3.1611.该版本的安装映像只有 64 位,具体的安装映像有以下几种: DVD版 推荐(迅雷下载):http://arc ...

  5. nginx的preaccess 阶段的limit_req模块与limit_conn模块

    limit_conn 模块限制并发连接数 [root@python vhast]# vim limit_conn.conf limit_conn_zone $binary_remote_addr zo ...

  6. 820复试算法 快排找第 k 小

    done {20-01-30 12:56} ref: https://blog.csdn.net/fengsigaoju/article/details/50728588 note: void qui ...

  7. Redis 事务在 SpringBoot 中的应用 (io.lettuce.core.RedisCommandExecutionException: ERR EXEC without MULTI)

    我们在 SpringBoot 中使用 Redis 时,会引入如下的 redis starter <dependency> <groupId>org.springframewor ...

  8. linux kill命令以及 信号

    kill命令介绍 命令作用 终止一个进程 语法: kill [-s signal|-p] [-q sigval] [-a] [--] pid... kill -l [signal] 选项 -l 信号, ...

  9. Centos7 nginx 反向代理的配置

    一.正向代理与反向代理 1.正向代理 正向代理往VPN理解 正向代理,也就是传说中的代理,他的工作原理就像一个跳板(VPN),简单的说: 我是一个用户,我访问不了某网站,但是我能访问一个代理服务器,这 ...

  10. mmap 与 munmap

    功能描述 mmap(memory map) 将一个文件或其他对象映射进内存. 文件被映射到多个page上, 若文件的大小不是所有page的大小之和, 最后一个page不被使用的空间将会被清零. mum ...