【北京迅为】《i.MX8MM嵌入式Linux开发指南》-第三篇 嵌入式Linux驱动开发篇-第五十三章 设备树下的platform驱动

i.MX8MM处理器采用了先进的14LPCFinFET工艺,提供更快的速度和更高的电源效率;四核Cortex-A53,单核Cortex-M4,多达五个内核 ,主频高达1.8GHz,2G DDR4内存、8G EMMC存储。千兆工业级以太网、MIPI-DSI、USB HOST、WIFI/BT、4G模块、CAN、RS485等接口一应俱全。H264、VP8视频硬编码,H.264、H.265、VP8、VP9视频硬解码,并提供相关历程,支持8路PDM接口、5路SAI接口、2路Speaker。系统支持Android9.0(支持获取root限)Linux4.14.78+Qt5.10.1、Yocto、Ubuntu20、Debian9系统。适用于智能充电桩,物联网,工业控制,医疗,智能交通等,可用于任何通用工业和物联网应用、

【公众号】迅为电子

【粉丝群】258811263


第五十三章 设备树下的platform驱动

本章导读

本章节我们来学习设备树下的platform驱动,之前我们学习了linux下的平台总线模型但是我们是使用传统的方法进行学习。什么是传统的方法呢?传统的方法就是把我们的驱动分为两个部分,第一部分是device.c,第二部分是driver.c,当device.c和driver.c匹配成功以后,进入probe函数后就可以获取硬件资源了,然后可以注册杂项设备,注册字符设备。

我们现在使用的是设备树,设备树相当于之前学习的device.c。之前传统的方法是使用“name”进行匹配的,我们使用设备树要怎么和我们的driver进行匹配呢?之前讲设备树语法的时候我们学习过compitable属性,那么这个compitable属性就是和driver.c匹配的。我们打开设备树文件,在设备树的根节点下,也有一个compitable属性,比如说内核在iTOP-3399开发板上面可以运行,那么这个内核放到4412开发板,imx6Q开发板上可以运行吗?答案肯定是不可以的,因为内核运行之前会进行一次匹配,内核在运行之前会检查下这个板子是否支持运行,那么他是根据根节点下的compitable属性来进行判断的。

53.1章节在前面52章节的基础上修改设备树文件,并查看是否生成设备节点。

53.2章节编写了驱动程序,该程序是设备树下的Platform驱动,匹配成功后在probe函数中获取到硬件资源,映射寄存器物理地址等等。

本章内容对应视频讲解链接(在线观看):

设备树下的platform总线  https://www.bilibili.com/video/BV1Vy4y1B7ta?p=28

程序源码在网盘资料“iTOP-i.MX8MM开发板\02-i.MX8MM开发板网盘资料汇总(不含光盘内容)\嵌入式Linux开发指南(iTOP-i.MX8MM)手册配套资料\2.驱动程序例程\011-设备树下的platform驱动”路径下。

53.1 设备树下的Platform设备

Linux 系统中 platform 平台框架包括总线、设备和驱动,其中总线不用我们去操心,Linux 内核中会自动管理,我们只需要关心设备和驱动如何实现。在不支持设备树的内核中,我们需要分别实现 platform_device和 platform_driver,其中 platform_device 是在平台文件中实现的。在支持设备树的内核中,我们就不用实现 platform_device 了,而是在设备树文件中添加设备信息。下面看一下在设备树文件中添加设备信息。

在之前关于设备树语法的章节中,我们学习了如何在根节点“/”下去添加一个设备节点信息。其中最重要的就是 compatible 属性值,compatible 属性使用来和驱动进行匹配的。下面是本实验用到的设备的设备节点:

 

在编写驱动以前,有一个地方需要注意一下,我们在加载driver.ko之前,一定要在开发板上已经成功地添加了test的节点,你可以在linux系统里面查看到你添加的节点,查看节点方法请参考51.1 查看设备树节点方法章节,添加自定义节点请参考51.2添加自定义节点章节。查看到test节点的comtabile属性的值为test1234,如下图所示: 

53.2 实验程序编写

53.2.1 Platform驱动程序

程序源码在网盘资料“iTOP-i.MX8MM开发板\02-i.MX8MM开发板网盘资料汇总(不含光盘内容)\嵌入式Linux开发指南(iTOP-i.MX8MM)手册配套资料\2.驱动程序例程\011-设备树下的platform驱动\001”路径下。

我们在Ubuntu的/home/topeet/imx8mm/11/001目录下新建driver.c文件,修改代码为如下所示 

/** @Author: topeet* @Description: 实现设备树下Platform驱动匹配进入probe函数*/
#include <linux/init.h>
#include <linux/module.h>
#include <linux/platform_device.h>
#include <linux/of.h>
#include <linux/of_address.h>/*** @description: platform 驱动的 probe 函数,当驱动与设备匹配以后此函数就会执行* @param {*}pdev : platform 设备* @return {*}0,成功;其他负值,失败*/
int led_probe(struct platform_device *pdev)
{ //匹配成功以后,进入到probe函数printk("led_probe\n");return 0;
}
int led_remove(struct platform_device *pdev)
{printk("led_remove\n");return 0;
}
const struct platform_device_id led_idtable = {.name = "led_test",
};
const struct of_device_id of_match_table_test[] = {{.compatible = "test1234"},{},
};
struct platform_driver led_driver = {//3. 在led_driver结构体中完成了led_probe和led_remove.probe = led_probe,.remove = led_remove,.driver = {.owner = THIS_MODULE,.name = "led_test",.of_match_table = of_match_table_test //接下来我们改一下驱动,让他来匹配设备树里面test的节点},.id_table = &led_idtable //4 .id_table的优先级要比driver.name的优先级要高,优先与.id_table进行匹配
};static int led_driver_init(void)
{// 1.我们看驱动文件要从init函数开始看int ret = 0;//2. 在init函数里面注册了platform_driverret = platform_driver_register(&led_driver);if (ret < 0){printk("platform_driver_register error \n");}printk("platform_driver_register ok \n");return 0;
}static void led_driver_exit(void)
{platform_driver_unregister(&led_driver);printk("goodbye! \n");
}
module_init(led_driver_init);
module_exit(led_driver_exit);MODULE_LICENSE("GPL");

保存driver.c文件,编译driver.c为驱动模块,如下图所示:

驱动编译完,我们通过nfs将编译好的驱动程序加载模块。我们进入共享目录,加载刚刚编译好的driver.ko,如下图所示:

insmod driver.ko

 

如上图所示,已经匹配成功进入到probe函数中。如果没有进入probe函数,可能出现匹配不成功的原因是1 device或者设备树根本没有加到我们系统里面2 名字不一样导致匹配不成功。

53.2.2 获取资源

程序源码在网盘资料“iTOP-i.MX8MM开发板\02-i.MX8MM开发板网盘资料汇总(不含光盘内容)\嵌入式Linux开发指南(iTOP-i.MX8MM)手册配套资料\2.驱动程序例程\011-设备树下的platform驱动\002”路径下。

我们进入了probe函数,可以在probe函数中获取资源,如下所示:

int led_probe(struct platform_device *pdev)
{ //匹配成功以后,进入到probe函数printk("led_probe\n");
/*********************方法一:直接获取节点**************************/printk("node name is %s\n",pdev->dev.of_node->name);return 0;
}

编译驱动,然后加载驱动后,如下图所示:

如上图所示,加载驱动以后,设备树上的节点和驱动程序匹配成功,进入了probe函数,并打印了节点的名字。

我们也可以用第52章学习过的of操作函数来获取我们的设备资源,修改driver.c为如下所示:

int led_probe(struct platform_device *pdev)
{ //匹配成功以后,进入到probe函数printk("led_probe\n");/*********************方法一:直接获取节点**************************/// printk("node name is %s\n",pdev->dev.of_node->name);/*********************方法二:通过函数获取硬件资源**************************///获得设备节点test_device_node = of_find_node_by_path("/test"); //获得设备节点if (test_device_node == NULL){printk("of_find_node_by_path is error \n");return -1;}//获取reg属性ret = of_property_read_u32_array(pdev->dev.of_node, "reg", out_values, 4);if (ret < 0){printk("of_property_read_u32_array is error \n");return -1;}printk("out_values[0] is 0x%08x\n", out_values[0]);printk("out_values[1] is 0x%08x\n", out_values[1]);printk("out_values[2] is 0x%08x\n", out_values[2]);printk("out_values[3] is 0x%08x\n", out_values[3]);return 0;
}

编译驱动,然后加载驱动后,如下图所示:

如上图所示,我们已经成功地获得设备树里面的reg属性。

53.2.3 获取节点属性

程序源码在网盘资料“iTOP-i.MX8MM开发板\02-i.MX8MM开发板网盘资料汇总(不含光盘内容)\嵌入式Linux开发指南(iTOP-i.MX8MM)手册配套资料\2.驱动程序例程\011-设备树下的platform驱动\003”路径下。

我们修改driver.c如下所示: 

int led_probe(struct platform_device *pdev)
{ //匹配成功以后,进入到probe函数printk("led_probe\n");/*********************方法一:直接获取节点**************************/// printk("node name is %s\n",pdev->dev.of_node->name);/*********************方法二:通过函数获取硬件资源**************************///获得设备节点// test_device_node = of_find_node_by_path("/test"); //获得设备节点// if (test_device_node == NULL)// {//     printk("of_find_node_by_path is error \n");//     return -1;// }//获取reg属性ret = of_property_read_u32_array(pdev->dev.of_node, "reg", out_values, 4);if (ret < 0){printk("of_property_read_u32_array is error \n");return -1;}printk("out_values[0] is 0x%08x\n", out_values[0]);printk("out_values[1] is 0x%08x\n", out_values[1]);printk("out_values[2] is 0x%08x\n", out_values[2]);printk("out_values[3] is 0x%08x\n", out_values[3]);return 0;
}

编译驱动,然后加载驱动后,如下图所示:

如上图所示,可以直接通过节点获取到reg属性的值。

53.2.4 映射物理地址

程序源码在网盘资料“iTOP-i.MX8MM开发板\02-i.MX8MM开发板网盘资料汇总(不含光盘内容)\嵌入式Linux开发指南(iTOP-i.MX8MM)手册配套资料\2.驱动程序例程\011-设备树下的platform驱动\004”路径下。

现在我们已经拿到了寄存器的地址,接下来可以注册杂项设备或者字符设备,我们先将获取到的物理地址映射为虚拟地址,修改driver.c代码如下:

int led_probe(struct platform_device *pdev)
{ //匹配成功以后,进入到probe函数printk("led_probe\n");/*********************方法一:直接获取节点**************************/// printk("node name is %s\n",pdev->dev.of_node->name);/*********************方法二:通过函数获取硬件资源**************************///获得设备节点// test_device_node = of_find_node_by_path("/test"); //获得设备节点// if (test_device_node == NULL)// {//     printk("of_find_node_by_path is error \n");//     return -1;// }//获取reg属性ret = of_property_read_u32_array(pdev->dev.of_node, "reg", out_values, 4);if (ret < 0){printk("of_property_read_u32_array is error \n");return -1;}printk("out_values[0] is 0x%08x\n", out_values[0]);printk("out_values[1] is 0x%08x\n", out_values[1]);printk("out_values[2] is 0x%08x\n", out_values[2]);printk("out_values[3] is 0x%08x\n", out_values[3]);//映射GPIO资源vir_gpio1_io13 = of_iomap(pdev->dev.of_node, 0);if (vir_gpio1_io13 == NULL){printk("GPIO1_IO13 iomap is error \n");return EBUSY;}printk("GPIO1_IO13 iomap is ok \n");vir_gpio1_io13_gdir = of_iomap(pdev->dev.of_node, 0);if (vir_gpio1_io13_gdir == NULL){printk("GPIO1_IO13_GDIR iomap is error \n");return EBUSY;}printk("GPIO1_IO13_GDIR iomap is ok \n");return 0;
}

 编译驱动,然后加载驱动后,如下图所示:

如上图所示,物理地址已经映射为虚拟地址,接下来可以注册字符设备和杂项设备,流程和我们前面学习到的内容是一模一样的。

 

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

如若内容造成侵权/违法违规/事实不符,请联系一条长河网进行投诉反馈,一经查实,立即删除!

相关文章

微信小程序--页面框架布局

创建page之后&#xff0c;在wxml中写的页面内容&#xff0c;都会包裹在<page>标签之中 若想像网页一样进行顶部、底部、内容等页面框架布局&#xff0c;仅仅写以下样式是不足够的 <view style"height: 10%; background-color: blueviolet;">顶部 </v…

深入理解 Java 虚拟机第三版(周志明)

这次社招选的这本作为 JVM 资料查阅&#xff0c;记录一些重点 1. 虚拟机历史 Sun Classic VM &#xff1a;已退休 HotSpot VM&#xff1a;主流虚拟机&#xff0c;热点代码探测技术 Mobile / Embedded VM &#xff1a;移动端、嵌入式使用的虚拟机 2.2 运行时数据区域 程序计…

Vue入门记录(一)

效果 本文为实现如下前端效果的学习实践记录&#xff1a; 实践 入门的最佳实践我觉得是先去看官网&#xff0c;官网一般都会有快速入门指引。 根据官网的快速上手文档&#xff0c;构建一个新的Vue3TypeScript&#xff0c;查看新建的项目结构&#xff1a; 现在先重点关注comp…

学生信息管理系统详细设计文档

一、设计概述 学生信息管理系统是一个用于管理学生信息的软件系统&#xff0c;旨在提高学校对学生信息的管理效率。本系统主要包括学生信息管理、课程信息管理、成绩信息管理、班级信息管理等功能模块。详细设计阶段的目标是确定各个模块的实现算法&#xff0c;并精确地表达这…

【AIGC】Llama-3 官方技术报告

Llama-3 技术报告&#xff08;中文翻译&#xff09; 欢迎关注【youcans的AGI学习笔记】原创作品 0. 简介 现代人工智能&#xff08;AI&#xff09;系统的核心驱动力来自于基础模型。 本文介绍了一组新的基础模型&#xff0c;称为 Llama 3。它是一个语言模型系列&#xff0c;原…

基于STM32瑞士军刀--【FreeRTOS开发】学习笔记(二)|| 堆 / 栈

堆和栈 1. 堆 堆就是空闲的一块内存&#xff0c;可以通过malloc申请一小块内存&#xff0c;用完之后使用再free释放回去。管理堆需要用到链表操作。 比如需要分配100字节&#xff0c;实际所占108字节&#xff0c;因为为了方便后期的free&#xff0c;这一小块需要有个头部记录…

Mysql第五次作业 触发器和存储过程

1、建库建表 建立触发器&#xff0c;订单表中增加订单数量后&#xff0c;商品表商品数量同步减少对应的商品订单出数量,并测试 建立触发器&#xff0c;实现功能:客户取消订单&#xff0c;恢复商品表对应商品的数量 建立触发器&#xff0c;实现功能:客户修改订单&#xff0c;商品…

护眼大路灯哪个口碑最好?五款专业护眼大路灯分享

护眼大路灯哪个口碑最好&#xff1f;护眼大路灯作为一款能够真正改善光线环境&#xff0c;有效做到减少视觉疲劳的护眼大路灯&#xff0c;逐渐成为众多家庭的必备照明神器。然而&#xff0c;市面上的护眼大路灯品牌琳琅满目&#xff0c;性能参差不齐&#xff0c;部分低质产品在…

docker安装httpd服务

docker安装httpd 一、简介 1、docker Docker是一个开源的容器化平台&#xff0c;可以轻松构建、发布和运行应用程序 2、httpd Apache HTTP服务器&#xff08;httpd&#xff09;是一个流行的开源Web服务器软件&#xff0c;用于托管网站和Web应用 二、准备环境 1、CentOS …

Docker安全管理与HTTPS协议

1 Docker容器的安全管理注意事项 Docker本身的架构与机制就可能产生问题&#xff0c;例如这样一种攻击场景&#xff0c;黑客已经控制了宿主机上的一些容器&#xff0c;或者获得了通过在公有云上建立容器的方式&#xff0c;然后对宿主机或其他容器发起攻击。 1. 容器之间的局…

C++ Lambda表达式个人理解

1、Lambda概述 lambda表达式&#xff08;也称为lambda函数&#xff09;是在调用或作为函数参数传递的位置处定义匿名函数对象的便捷方法。通常&#xff0c;lambda用于封装传递给算法或异步方法的几行代码。 2、Lambda表达式定义 2.1 Lambda表达式实例 Lambda有很多叫法&…

按图搜索新体验:阿里巴巴拍立淘API返回值详解

阿里巴巴拍立淘API是一项基于图片搜索的商品搜索服务&#xff0c;它允许用户通过上传商品图片&#xff0c;系统自动识别图片中的商品信息&#xff0c;并返回与之相关的搜索结果。以下是对阿里巴巴拍立淘API返回值的详细解析&#xff1a; 一、主要返回值内容 商品信息 商品列表…

深度学习趋同性的量化探索:以多模态学习与联合嵌入为例

深度学习趋同性的量化探索&#xff1a;以多模态学习与联合嵌入为例 参考文献 据说是2024年最好的人工智能论文&#xff0c;是否有划时代的意义&#xff1f; [2405.07987] The Platonic Representation Hypothesis (arxiv.org) ​arxiv.org/abs/2405.07987 趋同性的量化表达 …

CentOS搭建Apache服务器

安装对应的软件包 [roothds ~]# yum install httpd mod_ssl -y 查看防火墙的状态和selinux [roothds ~]# systemctl status firewalld [roothds ~]# cat /etc/selinux/config 若未关闭&#xff0c;则关闭防火墙和selinux [roothds ~]# systemctl stop firewalld [roothds ~]# …

全新微软语音合成网页版源码,短视频影视解说配音网页版系统-仿真人语音

源码介绍 最新微软语音合成网页版源码&#xff0c;可以用来给影视解说和短视频配音。它是TTS文本转语言&#xff0c;API接口和PHP源码。 这个微软语音合成接口的源码&#xff0c;超级简单&#xff0c;就几个文件搞定。用的是官方的API&#xff0c;试过了&#xff0c;合成速度…

Github个人网站搭建详细教程【Github+Jekyll模板】

文章目录 前言一、介绍1 Github Pages是什么2 静态网站生成工具3 Jekyll简介Jekyll 和 GitHub 的关系 4 Mac系统Jekyll的安装及使用安装Jekyll的简单使用 二、快速搭建第一个Github Pages网站三、静态网站模板——Chirpy1 个人定制 四、WordPress迁移到Github参考资料 前言 23…

DMv8共享存储集群部署

DMv8共享存储集群部署 环境说明 操作系统&#xff1a;centos7.6 服务器&#xff1a;2台虚拟机 达梦数据库版本&#xff1a;达梦V8 安装前准备工作 参考达梦官方文档&#xff1a;https://eco.dameng.com/document/dm/zh-cn/ops/DSC-installation-cluster.html#%E4%B8%80%E3…

Java面试八股之什么是spring boot starter

什么是spring boot starter Spring Boot Starter是Spring Boot项目中的一个重要概念。它是一种依赖管理机制&#xff0c;用于简化Maven或Gradle配置文件中的依赖项声明。Spring Boot Starter提供了一组预定义的依赖关系&#xff0c;这些依赖关系被封装在一个单一的包中&#x…

昇思25天学习打卡营第22天|munger85

LSTMCRF序列标注 我们希望得到这个模型来对词进行标注&#xff0c;B是开始&#xff0c;I是实体词的非开始&#xff0c;O是非实体词。 我们首先需要lstm对序列里token的记忆&#xff0c;和计算每个token发到crf的分数&#xff0c;发完了再退出来&#xff0c;最后形成1模型。那么…

海山数据库(He3DB)技术解析:海山Redis定时任务与持久化管控设计

文章目录 引言一、背景介绍二、具体实现1、多副本容灾功能2、主备切换后任务断点续做功能3、持久化管控编排功能 三、总结作者 引言 云Redis数据库服务是目前广泛应用的模式&#xff0c;其数据持久化方案是现在研究的热点内容&#xff0c;数据持久化操作主要由参数设置自动触发…