RTT(RT-Thread)IO设备模型

目录

IO设备模型

模型框架原理

IO设备类型

创建和注册IO设备

RTT设备管理程序实现原理

访问IO设备

查找设备

初始化设备

打开设备

关闭设备

 控制设备

 读写设备

数据收发回调

数据接收回调

数据发送回调

设备模型实例


IO设备模型

        RT-Thread 提供了一套简单的 I/O 设备模型框架,如下图所示,它位于硬件和应用程序之间,共分成三层,从上到下分别是 I/O 设备管理层、设备驱动框架层、设备驱动层。

  • 应用程序通过 I/O 设备管理接口获得正确的设备驱动,然后通过这个设备驱动与底层 I/O 硬件设备进行交互。
  • I/O 设备管理层实现了对设备驱动程序的封装
  • 设备驱动框架层是对同类硬件设备驱动的抽象,将不同厂家的同类硬件设备驱动中相同的部分抽取出来,将不同部分留出接口,由驱动程序实现。
  • 设备驱动层是一组驱使硬件设备工作的程序,实现访问硬件设备的功能。

简单设备的注册不经过设备驱动框架层,直接将设备注册到I/O设备管理器中

  • 设备驱动根据设备模型定义,创建出具备硬件访问能力的设备实例,将该设备通过rt_device_register()接口注册到 I/O 设备管理器中
  • 应用程序通过 rt_device_find()接口查找到设备,然后使用 I/O 设备管理接口来访问硬件

模型框架原理

图中:在左边是应用层代码,在右边是设备驱动代码,设备驱动层是与硬件最接近的(用于直接访问硬件)。而设备驱动和应用程序通过同一的IO设备管理器来统一管理起来。

如何去管理呢?设备驱动在要操作硬件的时候,要向IO设备管理器去注册,一旦注册完之后,设备管理器就知道了哪个设备的驱动注册到系统里面了。当注册成功以后,当应用层想要访问硬件的时候,通过调用rt_device_find函数来找到相应的设备驱动,一旦找到以后,就可以打开设备,接着进行读写操作,最后要记得关闭设备。

应用层想要去访问硬件的时候,只需要查找到相应的设备驱动,然后调用统一的接口就能对硬件设备进行操作,不需要关注硬件的实现原理。而对应硬件驱动来讲,我们只需要提供对硬件的访问方法。具体如何去访问,以及访问到的数据如何去处理也跟设备驱动没有关系。也就是说将不同的事情交给不同的层去完成,实现解耦(高内聚、低耦合)

对于一些复杂设备,需要使用到对应的设备驱动框架层,进行注册,它们拥有自己专属的注册函数 如:看门狗定时器

  • 看门狗设备驱动程序根据看门狗设备模型定义,创建出具备硬件访问能力的看门狗设备实例,并将该看门狗设备通过特定的函数 rt_hw_watchdog_register()接口注册到看门狗设备驱动框架中
  • 看门狗设备驱动框架通过 rt_device_register()接口将看门狗设备注册到 I/O 设备管理器中
  • 应用程序通过 I/O 设备管理接口来访问看门狗设备硬件

IO设备类型

  • RT-Thread 支持多种 I/O 设备类型,主要设备类型如下所示

RT_Device_Class_Char = 0, /**< character device */

RT_Device_Class_Block, /**< block device */

RT_Device_Class_NetIf, /**< net interface */

RT_Device_Class_MTD, /**< memory device */

RT_Device_Class_CAN, /**< CAN device */

RT_Device_Class_RTC, /**< RTC device */

RT_Device_Class_Sound, /**< Sound device */

RT_Device_Class_Graphic, /**< Graphic device */

RT_Device_Class_I2CBUS, /**< I2C bus device */

RT_Device_Class_USBDevice, /**< USB slave device */

RT_Device_Class_USBHost, /**< USB host bus */

RT_Device_Class_SPIBUS, /**< SPI bus device */

RT_Device_Class_SPIDevice, /**< SPI device */

RT_Device_Class_SDIO, /**< SDIO bus device */

RT_Device_Class_Timer, /**< Timer device */

RT_Device_Class_Miscellaneous, /**< misc device */

RT_Device_Class_Sensor, /**< Sensor device */

RT_Device_Class_Touch, /**< Touch device */

RT_Device_Class_Unknown /**< unknown device */

创建和注册IO设备

  • 驱动层负责创建设备实例,并注册到 I/O 设备管理器中
/*** This function creates a device object with user data size.** @param type, the kind type of this device object.* @param attach_size, the size of user data.** @return the allocated device object, or RT_NULL when failed.*/
rt_device_t rt_device_create(int type, int attach_size)
  • 当一个动态创建的设备不再需要使用时可以通过如下函数来销毁
/*** This function destroy the specific device object.** @param dev, the specific device object.*/
void rt_device_destroy(rt_device_t dev)
  • 设备被创建后,需要实现它访问硬件的操作方法
struct rt_device_ops
{/* common device interface */rt_err_t  (*init)   (rt_device_t dev);rt_err_t  (*open)   (rt_device_t dev, rt_uint16_t oflag);rt_err_t  (*close)  (rt_device_t dev);rt_size_t (*read)   (rt_device_t dev, rt_off_t pos, void *buffer, rt_size_t size);rt_size_t (*write)  (rt_device_t dev, rt_off_t pos, const void *buffer, rt_size_t size);rt_err_t  (*control)(rt_device_t dev, int cmd, void *args);
};
  • 设备被创建后,需要注册到 I/O 设备管理器中,应用程序才能够访问
/*** This function registers a device driver withspecified name.** @param dev the pointer of device driverstructure* @param name the device driver's name* @param flags the capabilities flag ofdevice   设备模式标志** @return the error code, RT_EOK oninitialization successfully.*/
rt_err_t rt_device_register(rt_device_t dev,const char *name,rt_uint16_t flags)#define RT_DEVICE_FLAG_RDONLY 0x001/*只读*/
#define RT_DEVICE_FLAG_WRONLY 0x002/*只写*/
#define RT_DEVICE_FLAG_RDWR 0x003 /*读写*/
#define RT_DEVICE_FLAG_REMOVABLE0x004 /*可移除*/
#define RT_DEVICE_FLAG_STANDALONE0x008 /*独立*/
#define RT_DEVICE_FLAG_SUSPENDED0x020 /*挂起*/
#define RT_DEVICE_FLAG_STREAM 0x040/*流模式*/
#define RT_DEVICE_FLAG_INT_RX 0x100/*中断接收*/
#define RT_DEVICE_FLAG_DMA_RX 0x200/*DMA接收*/
#define RT_DEVICE_FLAG_INT_TX 0x400/*中断发送*/
#define RT_DEVICE_FLAG_DMA_TX 0x800/* DMA发送*/
  • 设备注销后的,设备将从设备管理器中移除,也就不能再通过设备查找搜索到该设备。注销设备不会释放设备控制块占用的内存
/*** This function removes a previouslyregistered device driver** @param dev the pointer of device driverstructure** @return the error code, RT_EOK onsuccessfully.*/
rt_err_t rt_device_unregister(rt_device_t dev)

RTT设备管理程序实现原理

当我们创建一个设备的时候,系统会使用一个结构体描述这个设备的所有信息。当我们创建并注册多个设备的时候,系统就会通过列表的方式将这些结构体统一管理起来。当我们应用层想要找到某个设备的时候,就会调用find函数来找到列表头来根据name遍历查找,找到以后就会调用相应的方法来操作设备。当设备不用的时候,调用unregister函数来移除(将结构体变量从列表中移除),但结构体的空间依然存在,如果我们想要将结构体的空间释放掉,就需要调用destroy函数来进行删除释放。

访问IO设备

应用程序通过 I/O 设备管理接口来访问硬件设备,当设备驱动实现后,应用程序就可以访问该硬件,I/O 设备管理接口与 I/O 设备的操作方法的映射关系下图所示

相应接口函数在设备句柄结构体中

使用应用层接口前,首先要查找到设备

查找设备

返回值为设备句柄指针

/*** This function finds a device driver byspecified name.** @param name the device driver's name** @return the registered device driver onsuccessful, or RT_NULL on failure.*/
rt_device_t rt_device_find(const char*name)

操作接口如下:

初始化设备

/*** This function will initialize the specified device** @param dev the pointer of device driver structure** @return the result*/
rt_err_t rt_device_init(rt_device_t dev)

打开设备

/*** This function will open a device** @param dev the pointer of device driver structure* @param oflag the flags for device open** @return the result*/
rt_err_t rt_device_open(rt_device_t dev, rt_uint16_t oflag)

注:RT_DEVICE_FLAG_STREAM:流模式用于向串口终端输出字符串:当输出的字符是 "\n"(对应 16 进制值为 0x0A)时,自动在前面输出一个 "\r"(对应 16 进制值为 0x0D)做分行。

流模式 RT_DEVICE_FLAG_STREAM 可以和接收发送模式参数使用或 “|” 运算符一起使用

关闭设备

/*** This function will close a device** @param dev the pointer of device driver structure** @return the result*/
rt_err_t rt_device_close(rt_device_t dev)

 控制设备

/*** This function will perform a variety of control functions on devices.** @param dev the pointer of device driver structure* @param cmd the command sent to device* @param arg the argument of command** @return the result*/
rt_err_t rt_device_control(rt_device_t dev, int cmd, void *arg)

 读写设备

/*** This function will read some data from adevice.** @param dev the pointer of device driverstructure* @param pos the position of reading* @param buffer the data buffer to save read data* @param size the size of buffer** @return the actually read size onsuccessful, otherwise negative returned.** @note since 0.4.0, the unit of size/pos is ablock for block device.*/
rt_size_t rt_device_read(rt_device_tdev,rt_off_t    pos,void       *buffer,rt_size_t   size)
/*** This function will write some data to adevice.** @param dev the pointer of device driverstructure* @param pos the position of written* @param buffer the data buffer to be writtento device* @param size the size of buffer** @return the actually written size onsuccessful, otherwise negative returned.** @note since 0.4.0, the unit of size/pos is ablock for block device.*/
rt_size_trt_device_write(rt_device_t dev,rt_off_t    pos,const void *buffer,rt_size_t   size)

数据收发回调

当硬件设备收到数据时,可以通过如下函数回调另一个函数来设置数据接收指示,通知上层应用线程有数据到达

原理:通过设置数据的收发回调来通知我们的应用层某个线程来接收数据。这样就不需要再接收数据的时候写一个while循环一直去读。让它在没有数据的时候去休眠阻塞,一旦有数据的时候,这个函数就会被回调去通知相应线程唤醒去读写,这样可以减少系统的调用,提高系统调用效率。

数据接收回调

/*** This function will set the receptionindication callback function. * This callback function* is invoked when this device receives data.** @param dev the pointer of device driverstructure* @param rx_ind the indication callbackfunction** @return RT_EOK*/
rt_err_t
rt_device_set_rx_indicate(rt_device_tdev,rt_err_t (*rx_ind)(rt_device_t dev, rt_size_t size))

数据发送回调

/*** This function will set the indicationcallback function when device has* written data to physical hardware.** @param dev the pointer of device driverstructure* @param tx_done the indication callbackfunction** @return RT_EOK*/
rt_err_t
rt_device_set_tx_complete(rt_device_tdev,rt_err_t (*tx_done)(rt_device_t dev,void *buffer))

设备模型实例

(1)首先在drivers下创建一个drv_demo.c的驱动文件

(2)然后刷新工程目录,打开drv_demo.c

(3)我们可以参考drv_wdt.c的设备驱动模型来写

在文件最后我们可以发现有一个板级初始化的导出函数:我们使用INIT_BOARD_EXPORT宏将rt_wdt_init函数导出,那么在板级初始化的时候就会调用rt_wdt_init函数

我们再drv_demo.c中导出自己的设备初始化函数,并编写设备初始化函数

(4)接着我们创建设备,使用rt_device_create函数

第一个参数是设备类型,这里用字符设备

第二个参数是用户的数据大小,如果要传入用户数据的话,我们就根据用户数据的大小来传参。因为我们不需要传入用户数据,因此大小可以随便写一个,如32。

返回值为指针类型rt_device_t

(5)设备创建成功以后,我们需要对设备编写相应的接口函数,我们以init、open、close为例

我们可以按F3跳转到rt_device_t中,并将相应接口函数指针复制到我们自己的设备驱动文件下

(6)将函数进行简单完善

并对接口进行赋值

(7)然后我们以读写的方式注册我们的设备模型

(8)最后我们在main函数中使用,首先查找设备,如果查找成功,返回一个设备对象指针,如果查找失败,我们返回错误:变量无效

然后调用相应的应用层接口函数

(10)编译之后发现有一个警告LOG_E未定义

我们需要将添加调试头文件#include

(11)运行结果

通过列举设备,可以查看到demo是我们自己创建的设备;uart2是用来监控当前STM32单片机的,用作用户的调试接口;pin是GPIO引脚。(其中要注意串口也是属于字符设备类型的)

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

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

相关文章

网络编程(TFTP协议实验)

#include <stdio.h> #include <string.h> #include <stdlib.h> #include <head.h> #include <sys/types.h> #include <sys/socket.h> #include <arpa/inet.h> #include <netinet/in.h>#define PORT 69 //端口号&#xf…

网络安全威胁与防御策略

第一章&#xff1a;引言 随着数字化时代的快速发展&#xff0c;网络已经成为人们生活和工作中不可或缺的一部分。然而&#xff0c;网络的广泛应用也引发了一系列严峻的网络安全威胁。恶意软件、网络攻击、数据泄露等问题层出不穷&#xff0c;给个人和企业带来了巨大的风险。本文…

mysql基础之触发器的简单使用

1.建立学生信息表 -- 触发器 -- 建立学生信息表 create table s1(id int unsigned auto_increment,name varchar(30),score tinyint unsigned,dept varchar(50),primary key(id) );2.建立学生补考信息表 -- 建立学生补考信息表 create table s2 like s1;3.建立触发器&#xf…

java 自定义xss校验注解实现

自定义一个注解Xss。名字随意 import javax.validation.Constraint; import javax.validation.Payload; import java.lang.annotation.ElementType; import java.lang.annotation.Retention; import java.lang.annotation.RetentionPolicy; import java.lang.annotation.Targe…

C字符串练习题(6.3.1)

编写一个程序&#xff0c;从键盘上读入一个小于1000的正整数&#xff0c;然后创建并输出一个字符串&#xff0c;说明该整数的值。例如&#xff0c;输入941&#xff0c;程序产生的字符串是“Nine hundred and forty one”。 #include<stdlib.h> #include<string.h>…

Maven基础总结

前言 Maven 是一个项目管理工具&#xff0c;可以对 Java 项目进行构建、依赖管理。 基本要求掌握 配置Maven环境直接查。 得会在IDEA创建Maven的java项目吧、会创建Maven的web项目吧、会创建多模块项目吧。 得会配置插件pligin、依赖dependency吧 一、Maven四大特性 1、…

CSS:服务器字体 与 响应式布局(用法 + 例子 + 效果)

文章目录 服务器字体定义 服务器字体使用例子 响应式布局设备类型设备特性例子 服务器字体 解决字体不一致而产生的。 首先&#xff0c;在网上把字体下载好。 定义 服务器字体 font-face{font-family:字体名称;src:url(字体资源路径); }使用 在需要使用的选择器里加上 font…

数学建模(一)前继概念

课程推荐&#xff1a;数学建模老哥_哔哩哔哩_bilibili 目录 一、什么是数学建模&#xff1f; 二、数学建模的一般步骤 三、数学建模赛题类型 1.预测型 2. 评价类 3.机理分析类 4. 优化类 一、什么是数学建模&#xff1f; 数学建模是利用数学方法解决实际问题的一种实践。…

什么是多线程?进程和线程的区别是什么?如何使用Java实现多线程?

文章目录 前言我们为什么要使用线程而不是进程来实现并发编程什么是线程进程和线程的区别如何使用Java实现多线程创建线程1.创建一个继承 Thread 类的线程类2.实现 Runnable 接口匿名内部类方式实现 Runnable 接口lambda 表达式实现 Runnable 接口 Thread 类的常见构造方法Thre…

T113-S3-RTL8211网口phy芯片调试

目录 前言 一、RTL8211介绍 二、硬件连接 三、设备树配置 四、内核配置 五、phy芯片配置 六、调试问题 总结 前言 在嵌入式系统开发中&#xff0c;网络连接是至关重要的一部分。T113-S3开发板搭载了RTL8211系列的网口PHY芯片&#xff0c;用于实现以太网连接。在开发过程…

C++ QT(二)

目录 Qt 控件按钮QPushButton控件简介用法示例运行效果 QToolButton控件简介用法示例运行效果 QRadioButton控件简介用法示例运行效果 QCheckBox控件简介用法示例运行效果 QCommandLinkButton控件简介用法示例运行效果 QDialogButtonBox控件简介用法示例运行效果 输入窗口部件Q…

用 React+ts 实现无缝滚动的走马灯

一、走马灯的作用 走马灯是一种常见的网页交互组件&#xff0c;可以展示多张图片或者内容&#xff0c;通过自动播放或者手动切换的方式&#xff0c;让用户能够方便地浏览多张图片或者内容。 本次实现的不是轮播图而是像传送带一样的无限滚动的形式。 二、需求梳理 走马灯可设…

Go Gin 中使用 JWT

一、JWT JWT全称JSON Web Token是一种跨域认证解决方案&#xff0c;属于一个开放的标准&#xff0c;它规定了一种Token实现方式&#xff0c;目前多用于前后端分离项目和OAuth2.0业务场景下。 二、为什么要用在你的Gin中使用JWT 传统的Cookie-Sesson模式占用服务器内存, 拓展性…

uniapp实现自定义导航内容高度居中(兼容APP端以及小程序端与胶囊对齐)

①效果图如下 1.小程序端与胶囊对齐 2.APP端内容区域居中 注意&#xff1a;上面使用的是colorui里面的自定义导航样式。 ②思路&#xff1a; 1.APP端和小程序端走不同的方法&#xff0c;因为小程序端要计算不同屏幕下右侧胶囊的高度。 2.其次最重要的要清晰App端和小程序端…

【数学建模】清风数模更新5 灰色关联分析

灰色关联分析综述 诸如经济系统、生态系统、社会系统等抽象系统都包含许多因素&#xff0c;系统整体的发展受各个因素共同影响。 为了更好地推动系统发展&#xff0c;我们需要清楚哪些因素是主要的&#xff0c;哪些是次要的&#xff0c;哪些是积极的&#xff0c;哪些是消极的…

网络基础——网络的由来与发展史

作者&#xff1a;Insist-- 个人主页&#xff1a;insist--个人主页 作者会持续更新网络知识和python基础知识&#xff0c;期待你的关注 目录 一、网络的由来 二、计算机网络的发展史 1、第一阶段 2、第二阶段 3、第三阶段 前言 每天都是使用网络&#xff0c;那么你知道网络…

FPGA----Vivado SDK创建并使用静态链接库(C/C++代码移植)

1、在进行SoC开发时&#xff0c;PS端的C/C代码可能涉及到核心算法需要移植操作&#xff0c;为此&#xff0c;本文讲述了如何将C/C代码打包为.a文件供程序调用 2、文章以我的程序为例&#xff0c;逐步讲述代码生成静态链接库并调用的方法。 下面是我程序的目录结构&#xff0c…

RocketMQ 延迟消息

RocketMQ 延迟消息 RocketMQ 消费者启动流程 什么是延迟消息 RocketMQ 延迟消息是指&#xff0c;生产者发送消息给消费者消息&#xff0c;消费者需要等待一段时间后才能消费到。 使用场景 用户下单之后&#xff0c;15分钟未支付&#xff0c;对支付账单进行提醒或者关单处理…

PostgreSQL查询慢sql原因和优化方案

PostgreSQL sql查询慢优化方案有一下几种解决方案&#xff1a; 1.关闭会话 查询慢sql的执行会话&#xff0c;关闭进程。 查看数据库后台连接进程 SELECT count(*) FROM pg_stat_activity;SELECT * FROM pg_stat_activity; 查看数据库后台连接进程&#xff0c;但是此条SQL不…

用HARU-Net增强核分割:一种基于混合注意的残差u块网络

文章目录 Enhancing Nucleus Segmentation with HARU-Net: A Hybrid Attention Based Residual U-Blocks Network摘要本文方法损失函数后处理消融实验 Enhancing Nucleus Segmentation with HARU-Net: A Hybrid Attention Based Residual U-Blocks Network 摘要 核图像分割是…