Zephyr 入门-设备树与设备驱动模型

学习链接:https://www.bilibili.com/video/BV1L94y1F7qS/?spm_id_from=333.337.search-card.all.click&vd_source=031c58084cf824f3b16987292f60ed3c
讲解清晰,逻辑清楚。

1. 设备树概述(语法,如何配置硬件,c代码如何访问) driver的实现

讲解zephyr设备树的意义:单片机开发过程中,一般使用宏定义来表示硬件引脚,但是为了达成高内聚低耦合的软件架构,又设计了把初始化函数和中断放在一个board.c文件,其他按照功能分开存放。虽然已经很好了,但是切换硬件平台后还是不能很好的移植。(比如我们更换平台后,需要更换调用的GPIO library 库) 而且不同的人又有不同的代码风格,但是zephyr横空出世。规范了这一切,都得按照我的规矩来。这样导致平台移植非常方便,这简直是提高了人类的效率啊。
思考: zephyr把硬件都规定好了,他能不能自己生成board.c文件呢?zephyr的设备树简单了解:芯片厂商会自己适配zephyr,客户可以根据需要修改,同时zephyr会调驱动把有设备树的外设初始化好。同时客户还能够调用和操作寄存器,控制管脚,好灵活。好牛逼啊感觉。

2. device tree的结构和语法

首先按照总线的主从关系,其次按照硬件的包含关系。(总线下面的节点都有地址,而如LED等外设虽然用到了GPIO但是没有地址,所以直接成为根节点下的子节点)device tree 的适用范围:device tree 描述的是板卡级别的硬件信息,当一个办卡上有两个MCU他们不能公用device tree. 当一个MCU有两个独立固件的core,他们不能共用device tree,等。设备树语法跟linux里面一样只是需要注意写法:device tree 的节点: name@address
节点的name可以是字母数字下划线,加减号标点等等。但是c代码得到设备树名称的时候会把不符合字母数字下划线的那部分特殊符号变成下划线。
如果有reg属性,则address必须和reg属性的第一个寄存器地址值相等。如果没有reg属性,则@address必须省略。dts可以引用其他的dts或dtsi,这样板卡级dts就可以引用芯片厂商写好的芯片级dtsi文件。缩短dts开发时间。
dts还可以引用c头文件来使用一些枚举值或宏定义。devicetree文件的位置:
1. 新建工程时可以选择板卡,板子的dts文件位于:zephyr/board/目录
2. 板卡引用的dtsi文件位于 zephyr/dts/arm/nordic 目录
3. 用户不需要自己写dts,直接在自己的工程目录中的board.overlay增删改节点属性。覆盖开发板的配置即可。
使用label指定节点并覆写其属性,/delete-property/ led1;可以删除属性,/delete-node/ leds;可以删除节点。
4. 最后,构建项目时,zephyr build system 会使用一系列脚本把board,soc,用户的overlay合并起来。如果构建目录的名称是build,最终合并后的完整文件位于:
${projector_folder}build/zephyr/zephyr.dts
5. 了解即可,zephyr最终会把zephyr.dts处理成c语言头文件,位于S{project_folder}/build/zephyr/include/generated/devicetree_generated.h。代码最后得到设备树的信息其实是通过这个文件。

3. device tree如何配置硬件信息

reg = <addr1 add1_leng addr2 add2_leng> 可见reg由多对总线上的地址和长度信息组合而成。
当一个节点定义的ranges属性,那他的子节点就可以使用相对地址而非绝对地址。 ranges = <子空间首地址 父空间首地址 长度信息>
status ,只需要注意"okay" 和 disabled, 决定是否初始化该外设。
compatible 设备会通过这个找到合适的驱动程序设备树中的"域", 除了地址树以外,比如GPIO树其下面挂载了LED BUTTON等,中断树,ADC树下面挂载了很多ADC设备。 这些树的出现是为了更好的描述网状的硬件关系。这些虚拟概念上的树被称为"域"。每个域都有自己的根节点被称为controller, 控制器控制了整个域的相关硬件。控制器节点通过给自己一个*-controller的属性(也可能没有),如:gpio-controller, 然后这个GPIO域的子节点就可以通过前面介绍的phandle-array属性来使用这个gpio控制器,如:gpios = <&gpio0 0x18 0x11> 第一个数值指向控制器节点,后续的值是节点在这个域中的配置,被称为specifier。 使用哪一个gpio,后一个指定gpio的配置信息。设备树有一些奇怪的规则,比如有的域控制器有controller有的没有,而且zephyr能自动检测你写的dts是不是符合这些规则的,怎么实现的呢?通过yaml这种描述语言进行检测的,比如这个 ./zephyr/dts/bindings/dma/nxp,mcux-edma.yaml ,这个就是device bingding 文件。 这个文件名字就是compatible属性,zephyr的编译器会用bing文件去dts检查符合这个compatible属性的设备树节点。这个binding文件可以约束子节点怎么写,还可以给specifier中的节点数值做含义解释。zephyr build system 会从一下位置寻找binding文件:
./zephyr/dts/bindings/
${board_dir}/dts/bindings/
${project_dir}/dts/bindings/
也可以在Cmakelist.txt中,用list{APPEND DTS_ROOT /path/to/your/dts}增加binging文件的目录
也可以在编译时增加选项 west build -b <board_name> --DTS_ROOT=<path/to/your/dts>还有一些特殊的节点:
chosen: 为device kernel 选择特定设备
aliase: 为节点起一个别名,别名是属性名
pinctrl: 直属于根节点,数字IO复用
/zephyr,usr: 方便用户开发的节点,直接写spicifier和配置项,不用写binding了

4. 如何在c代码中获取device tree??

通过API获取,需要包含头文件:#include <zephyr/device_tree.h>
为了获得节点属性,需要先获得节点id作为句柄(节点id就是node identifier),节点id本质上是devicetree_generated.h中的宏定义。方式如下:
DT_ROOT 						得到根节点id
DT_PATH(soc,serial_40001000)	得到/soc/serial@40001000
DT_NODELABLE(serial1)			根据dts中定义的lable找到节点
DT_CHOSEN(zephyr_console)		根据chosen节点配置:zephyr,console=&uart0
还有很多子节点找父节点,父节点找子节点的方式。 还有一种方式,是通过实例ID的方式获取节点ID,如DT_INST(0, nordic_nrf_timer)对应的就是nordic,nrf_timer的第0个实例节点。如果节点中有多个节点有同一个comaptible,就是一个compatible对应多个实例,好处是什么呢就是可以放到for循环中遍历了呀。(但是注意不是c语言中的for,因为这些API都是预编译后的结果,括号里的当然也不是真正的输入参数,哈哈)DT_PROP 宏可以得到普通属性
DT_REG_ADDR 和 DT_REG_SIZE 宏可以读取reg的地址和长度
如果一个节点的属性是其他节点可以通过DT_HANDLE_BY_IDX得到内个节点上面说了zephyr提供的获取节点的ID虽然都不能传入遍历以便于通过for循环调用,但是zephyr提供了可以遍历这些节点的API,如DT_FOREACH_NODE(fn)为设备树中的每一个节点调用宏函数fn。 还有很多API请看 https://docs.zephyrproject.org/latest/build/dts/api/api.html#for-each-macros (这些API看似是循环,其实是宏)
这里这个fn宏函数是用代码模板。再看看还有很多API方便你直接读取specifier等。具体参考
https://docs.zephyrproject.org/latest/build/dts/api/api.html#hardware-specific-apis
https://docs.zephyrproject.org/latest/hardware/index.heml

5. Zephyr Driver 的实现方式

什么是驱动程序?因为zephyr中驱动程序是“面向对象”的,他有个device结构体,结构体如下:struct device {const char *name;const void *config;const void *api;struct device_state *state;void *data;......};
可以看到这个结构体很宽泛,驱动程序需要在application启动之前把这个结构体填充好,然后application才能调用这些api。
根据zeyphr的启动流程,可以把驱动程序放在一下五个级别中的任何一个级别:
start up-> EARLY --> PRE_KERNEL_1 --> PRE_KERNEL_2 --> RTOS KERNEL 启动 --> POST_KERNEL --> APPLICATION -->MAIN| 		放这的不能有log 					| 					| 			放这里可以打log	|Application需要得到这个device结构体才能操作外设对吧,那app如何得到device呢?两种方式:通过name, 通过设备树node id。
第一种方式中,在驱动程序中通过DEVICE_DEFINE宏来定义device结构体,其中第二个参数为该结构体的lable吧。在app中通过device_get_binding这个API即可得到这个device.
第二种方式中,在驱动程序中通过DEVICE_DT_DEFINE来定义结构体,并与节点绑定。在app中通过DEVICE_DT_GET(node id)宏来获得device.(node id可以用DT_PATH(节点名)得到)prj.config中的CONFIG_*选项与dts中的status状态有什么关系?
前者决定是否编译进入固件(嵌入到硬件设备的软件代码),后者决定驱动程序使用宏遍历device结构体时能够为这个okay的节点创建device对象。只有两者都启用app才能操作这个节点。

6. Zephyr标准驱动

zephyr是个跨平台操作系统,少不了对标准硬件的跨平台支持。
详见:https://docs.zephyrproject.org/latest/hardware/peripherals/index.html
以DMA为例,在zephyr/include/zephyr/drivers/dma.h目录中规定好了dma驱动应该有哪些API。在zephyr/drivers/dma/目录下有各个厂家写好的对字节芯片的driver驱动。通过zephyr/drivers/dma/Kconfig.nxp_edma可以看到各个CONFIG选项的作用,在./zephyr/build/zephyr/.config设置好对应的CONFIG_*=y,则zephyr就会把板子对应厂商的dma驱动编译进来。zephyr标准驱动支持硬件的全部功能吗???
zephyr只支持最基础最标准的硬件驱动,不支持各个厂商的硬件特性。如果想要硬件特性功能,需要使用厂商自己的driver library, 或者直接写寄存器。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/482332.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【python】图像、音频、视频等文件数据采集

【python】图像、音频、视频等文件数据采集 先安装所需要的工具一、Tesseract-OCRTesseract-OCR环境变量设置验证是否配置成功示例语言包下载失败 二、ffmpeg验证是否安装成功示例 先安装所需要的工具 一、Tesseract-OCR Tesseract是一个 由HP实验室开发 由Google维护的开源的…

【青牛科技】2K02 电动工具专用调速电路芯片描述

概述&#xff1a; 2K02 是电动工具专用调速电路。内置稳压电路&#xff0c;温度系数好&#xff0c;可以调节输出频率以及占空比的振荡输出&#xff0c;广泛的应用于小型电钻&#xff0c;割草机等工具。 主要特点&#xff1a; ● 电源电压范围宽 ● 占空比可调 ● 温度系数好 …

内网穿透步骤

步骤 第一次需要验证token window和linux的方法不同。 然后 启动 cpolar 服务&#xff1a; 在命令窗口中输入 cpolar.exe htttp 8080&#xff0c;启动内网穿透服务。确保命令窗口保持开启状态&#xff0c;以维持穿透效果。 cpolar.exe hhttp 8080 成功后 注意事项 命令窗口…

FreeRTOS之vTaskStartScheduler实现分析

FreeRTOS之vTaskStartScheduler实现分析 1 FreeRTOS源码下载地址2 函数接口2.1 函数接口2.2 函数参数简介3 vTaskDelete的调用关系3.1 调用关系3.2 调用关系示意图 4 函数源码分析4.1 vTaskStartScheduler4.2 prvCreateIdleTasks4.2.1 prvCreateIdleTasks4.2.2 xTaskCreate 4.3…

基于群晖搭建个人图书架-TaleBook based on Docker

前言 在群晖Container Manager中部署失败&#xff0c;转通过ssh部署。 一、准备工作 名称备注群晖SSH“终端机和SNMP”中启用SSH软件secureCRT等docker-compose.ymlGithub下载并修改 二、过程 2.1 创建本地文件夹 本地路径为&#xff1a; /docker/Calibre/data 2.2 下载d…

Ubuntu24.04初始化教程(包含基础优化、ros2)

将会不断更新。但是所有都是基础且必要的操作。 为重装系统之后的环境配置提供便捷信息来源。记录一些错误的解决方案。 目录 构建系统建立系统备份**Timeshift: 系统快照和备份工具****安装 Timeshift****使用 Timeshift 创建快照****还原快照****自动创建快照** 最基本配置换…

【论文笔记】A Token-level Contrastive Framework for Sign Language Translation

&#x1f34e;个人主页&#xff1a;小嗷犬的个人主页 &#x1f34a;个人网站&#xff1a;小嗷犬的技术小站 &#x1f96d;个人信条&#xff1a;为天地立心&#xff0c;为生民立命&#xff0c;为往圣继绝学&#xff0c;为万世开太平。 基本信息 标题: A Token-level Contrastiv…

yolov5 解决:export GIT_PYTHON_REFRESH=quiet

当我们在第一次运行YOLOv5中的train.py程序时&#xff1a;可能会出现以下报错&#xff1a; This initial warning can be silenced or aggravated in the future by setting the $GIT_PYTHON_REFRESH environment variable. Use one of the following values: - quiet|q|silen…

基于springboot中小型制造企业质量管理系统源码和论文

信息数据从传统到当代&#xff0c;是一直在变革当中&#xff0c;突如其来的互联网让传统的信息管理看到了革命性的曙光&#xff0c;因为传统信息管理从时效性&#xff0c;还是安全性&#xff0c;还是可操作性等各个方面来讲&#xff0c;遇到了互联网时代才发现能补上自古以来的…

【实验13】使用预训练ResNet18进行CIFAR10分类

目录 1 数据处理 1.1 数据集介绍 1.2数据处理与划分 2 模型构建- Pytorch高层API中的Resnet18 3 模型训练 4 模型评价 5 比较“使用预训练模型”和“不使用预训练模型”的效果&#xff1a; 6 模型预测 7 完整代码 8 参考链接 1 数据处理 1.1 数据集介绍 数据规模&…

Java之链表1

文章目录 1. 链表1.11.2 链表的概念及其结构1.3 自己实现一个链表 1. 链表 1.1 之前我们学习了 顺序表ArrayList&#xff0c;并自己实现了 ArrayList &#xff0c;发现它在删除元素和添加元素时很麻烦&#xff0c;最坏的情况时&#xff0c;需要将所有的元素移动&#xff0c;因…

二分搜索(三)x的平方根

69. x 的平方根 给定一个排序数组和一个目标值&#xff0c;在数组中找到目标值&#xff0c;并返回其索引。如果目标值不存在于数组中&#xff0c;返回它将会被按顺序插入的位置。 请必须使用时间复杂度为 O(log n) 的算法。 示例 1: 输入: nums [1,3,5,6], target 5 输出: 2…

AI开发-数据可视化库-Seaborn

1 需求 概述 Seaborn 是一个基于 Python 的数据可视化库&#xff0c;它建立在 Matplotlib 之上。其主要目的是使数据可视化更加美观、方便和高效。它提供了高层次的接口和各种美观的默认主题&#xff0c;能够帮助用户快速创建出具有吸引力的统计图表&#xff0c;用于数据分析和…

使用docker-compose部署搜索引擎ElasticSearch6.8.10

背景 Elasticsearch 是一个开源的分布式搜索和分析引擎&#xff0c;基于 Apache Lucene 构建。它被广泛用于实时数据搜索、日志分析、全文检索等应用场景。 Elasticsearch 支持高效的全文搜索&#xff0c;并提供了强大的聚合功能&#xff0c;可以处理大规模的数据集并进行快速…

LeetCode—74. 搜索二维矩阵(中等)

仅供个人学习使用 题目描述&#xff1a; 给你一个满足下述两条属性的 m x n 整数矩阵&#xff1a; 每行中的整数从左到右按非严格递增顺序排列。 每行的第一个整数大于前一行的最后一个整数。 给你一个整数 target &#xff0c;如果 target 在矩阵中&#xff0c;返回 true…

Cento7 紧急模式无法正常启动,修复home挂载问题

Centos 7 开机失败进入紧急模式[emergency mode]&#xff0c;解决方案。 通过journalctl -xb查看启动日志&#xff0c;定位发现/home目录无法正常挂载。 退出启动日志检查&#xff0c;进行修复。 进行问题修复 # 修复挂载问题 mkdir /home mount /dev/mapper/centos-home /ho…

Matlab mex- setup报错—错误使用 mex,未检测到支持的编译器...

错误日志&#xff1a; 在使用mex编译时报错提示&#xff1a;错误使用 mex&#xff0c;未检测到支持的编译器。您可以安装免费提供的 MinGW-w64 C/C 编译器&#xff1b;请参阅安装 MinGW-w64 编译器。有关更多选项&#xff0c;请访问https://www.mathworks.com/support/compile…

【C语言】二叉树(BinaryTree)的创建、3种递归遍历、3种非递归遍历、结点度的实现

代码主要实现了以下功能&#xff1a; 二叉树相关数据结构定义 定义了二叉树节点结构体 BiTNode&#xff0c;包含节点数据值&#xff08;字符类型&#xff09;以及指向左右子树的指针。 定义了顺序栈结构体 SqStack&#xff0c;用于存储二叉树节点指针&#xff0c;实现非递归遍历…

Android -- 简易音乐播放器

Android – 简易音乐播放器 播放器功能&#xff1a;* 1. 播放模式&#xff1a;单曲、列表循环、列表随机&#xff1b;* 2. 后台播放&#xff08;单例模式&#xff09;&#xff1b;* 3. 多位置同步状态回调&#xff1b;处理模块&#xff1a;* 1. 提取文件信息&#xff1a;音频文…

Python语法基础(四)

&#x1f308;个人主页&#xff1a;羽晨同学 &#x1f4ab;个人格言:“成为自己未来的主人~” 高阶函数之map 高阶函数就是说&#xff0c;A函数作为B函数的参数&#xff0c;B函数就是高阶函数 map&#xff1a;映射 map(func,iterable) 这个是map的基本语法&#xff0c;…