嵌入式linux系统设备树实例分析

news2024/11/17 21:42:00

前言

我们可以从LED程序中榨取很多知识:基本的驱动框架、驱动的简单分层、驱动的分层+分离思想、总线设备驱动模型、设备树等。这大多都是结合韦老师的教程学的。

这篇笔记结合第6个demo(基于设备树)来学习、分析:

图片

框图

下面是LED程序的几个层次结构图:

图片

图片

图片

图片

图片

注意:层与层之间的箭头指向是相对的,从哪指向哪看你怎么理解。比如有两个函数:函数A和函数B,我们可以说函数A调用函数B,也可以说函数B被函数A调用。

本篇笔记基于第⑤个图来分析。

体验设备树

我们先来体验一下使用设备树描述引脚信息的方式来点灯。

以百问网开发板为例,修改内核目录Linux-4.9.88/arch/arm/boot/dts下的100ask_imx6ull-14x14.dts(百问科技开发板出厂带的设备树文件)设备树文件。

把出厂带的设备树文件的led相关节点给屏蔽掉,然后添加如下节点信息至根节点:

#define GROUP_PIN(g,p) ((g<<16) | (p))
100ask_led@0 {
    compatible = "100as,leddrv";
    pin = <GROUP_PIN(5, 3)>;
};

修改后的设备树文件内容如:

图片

在内核根目录下使用如下命令编译设备树源文件:

make dtbs V=1

图片

然后把设备树文件与可加载的led驱动模块、led应用程序上传到板子里:

图片

上传成功的文件如下:

图片

运行测试:

图片

实验过程分析

这个实验的led驱动同样依赖的是总线设备驱动模型

在linux设备驱动模型中也有提到描述设备有两种方法:一种是直接用platform_device结构体来指定,另一种是用设备树来指定。

在本次的实验中我们就是用设备树来描述设备。

之前我们用platform_device结构体来指定设备信息时,platform_driver是直接从platform_device结构体里拿资源的,如:

图片

图片

现在我们用设备树来指定设备信息时,platform_driver是如何获取相关资源的呢?大致过程如下:

图片

图片

图片

这里我们还需要注意的一点是:并不是所有的设备树节点都可以转换为platform_device。下面看看几条规则:

  • 根节点下含有 compatile 属性的子节点能转换为platform_device

  • 含有特定 compatile 属性(它的值是 "simple-bus","simplemfd","isa","arm,amba-bus" 四者之一)的节点的子节点能转换为platform_device

  • I2C、 SPI 总线节点下的子节点 不不不能转换为platform_device,这些总线下的子节点, 应该交给对应的总线驱动程序来处理。

下面看一个例子:

图片

接下来,我们简单来看一下platform_device与platform_driver匹配的函数:

图片

这里有几种匹配方式,其它几种匹配方式在之前的笔记【Linux笔记】总线设备驱动模型中也有简单地分析过。

这里,我们来看第二种匹配方式(使用设备树时的匹配方式)。下面看看具体如何匹配:

图片

其中过程①优先匹配,其次是过程②,最后是过程③。

但是,实际上现在主要使用的是过程①的匹配,即匹配compatible属性。

过程②与过程③已经过时了,Linux内核不推荐使用这两种匹配方法。

这一点我们在设备树基础知识中也有简单提到。

在本次实验中,我们的匹配示意图如下:

图片

实验代码

代码大多与之前的【Linux笔记】LED驱动(总线设备驱动模型)中的代码一样,这里也来简单看一下。

1、应用程序ledtest.c:

int main(int argc, char **argv)
{
 int fd;
 char status;
 
 /* 1. 判断参数 */
 if (argc != 3) 
 {
  printf("Usage: %s <dev> <on | off>\n", argv[0]);
  return -1;
 }

 /* 2. 打开文件 */
 fd = open(argv[1], O_RDWR);
 if (fd == -1)
 {
  printf("can not open file %s\n", argv[1]);
  return -1;
 }

 /* 3. 写文件 */
 if (0 == strcmp(argv[2], "on"))
 {
  status = 1;
  write(fd, &status, 1);
 }
 else
 {
  status = 0;
  write(fd, &status, 1);
 }
 
 close(fd);
 
 return 0;
}

运行测试命令:

./ledtest /dev/100ask_led0 on
./ledtest /dev/100ask_led0 off

int main(int argc, char **argv)形式的main函数相关笔记:main()函数有哪几种形式?。

2、驱动层leddrv.c

这一层主要是放一些通用的驱动操作函数,核心代码如:

驱动程序入口函数:

图片

open、write函数:

图片

其它代码:

图片

其中led的操作结构体如下:

图片

3、硬件层:chip_demo_gpio.c

这一层主要是一些寄存器相关的操作,及platform_driver相关。与上一个实验代码不同的部分就是这个文件。

(1)驱动初始化函数:

图片

(2)probe函数:

当设备树的compatible属性与platform_driver中的设备匹配表中的compatible成员互相匹配时会执行此函数获取设备信息。

图片

这里的pin属性与compatible属性(标准属性)类别不同,pin属性是个自定义属性。

我们可以使用of_property_read_u32函数来获取这些自定义属性的内容。

与设备树相关的读取函数我们在上一篇笔记【Linux笔记】设备树基础知识中也有详细介绍。

这些函数大多在文件 include/linux/of.h 中可以找到:

图片

(3)led寄存器操作相关的代码:

/* 寄存器物理地址 */
#define CCM_CCGR1_BASE    (0X020C406C) 
#define SW_MUX_GPIO5_IO03_BASE  (0X02290014)
#define GPIO5_DR_BASE    (0X020AC000)
#define GPIO5_GDIR_BASE    (0X020AC004)

/* 映射后的寄存器虚拟地址指针 */
static void __iomem *CCM_CCGR1;
static void __iomem *SW_MUX_GPIO5_IO03;
static void __iomem *GPIO5_DR;
static void __iomem *GPIO5_GDIR;

/* 初始化LED, which-哪个LED */    
static int board_demo_led_init (int which)    
{   
 int group, pin;
 unsigned int val;

 group = GROUP(g_ledpins[which]);
 pin = PIN(g_ledpins[which]);
 printk("init gpio: group %d, pin %d\n", group, pin);

 /* 100ask_IMX6uLL_Board LED:GPIO5_3 */
 if ((5 == group) && (3 == pin))
 {
  /* 相关寄存器物理地址与虚拟地址之间的映射 */
  /* 1、地址映射:时钟寄存器 */
  CCM_CCGR1 = ioremap(CCM_CCGR1_BASE, 4);  
  /* 2、地址映射:模式寄存器 */ 
  SW_MUX_GPIO5_IO03 = ioremap(SW_MUX_GPIO5_IO03_BASE, 4); 
  /* 3、地址映射:数据寄存器 */
  GPIO5_DR = ioremap(GPIO5_DR_BASE, 4); 
  /* 地址映射:方向寄存器 */
  GPIO5_GDIR = ioremap(GPIO5_GDIR_BASE, 4);

  /* 使能GPIO5时钟 */
  val = readl(CCM_CCGR1); /* 读出当前CCM_CCGR1配置值 */
  val &= ~(3 << 30);  /* 清除以前的设置 */
  val |= (3 << 30);  /* 设置新值 */
  writel(val, CCM_CCGR1);

  /* 设置GPIO5_IO03的为IO模式 */
  writel(5, SW_MUX_GPIO5_IO03);
  
  /* 设置GPIO5_IO03方向为输出 */
  val = readl(GPIO5_GDIR); 
  val &= ~(1 << 3);   
  val |= (1 << 3);   
  writel(val, GPIO5_GDIR);
 }
 else
 {
  printk("This is not 100ask_IMX6ULL_Board!\n");
 }
 
 return 0;
}

/* 控制LED, which-哪个LED, status:1-亮,0-灭 */
static int board_demo_led_ctl (int which, char status) 
{
 int group, pin;
 unsigned int val;

 group = GROUP(g_ledpins[which]);
 pin = PIN(g_ledpins[which]);
 printk("init gpio: group %d, pin %d\n", group, pin);

 /* 100ask_IMX6uLL_Board LED:GPIO5_3 */
 if ((5 == group) && (3 == pin))
 {
  /* 点灯 */
  if (1 == status)
  {
   printk("<<<<<<<<led on>>>>>>>>>>\n");
   val = readl(GPIO5_DR);
   val &= ~(1 << 3); 
   writel(val, GPIO5_DR);
  }
  /* 灭灯 */
  else if (0 == status)
  {
   printk("<<<<<<<<led off>>>>>>>>>>\n");
   val = readl(GPIO5_DR);
   val|= (1 << 3); 
   writel(val, GPIO5_DR);
  }
  else{}
 }
 else
 {
  printk("This is not 100ask_IMX6ULL_Board!\n");
 }
 
 return 0;
}

4、Makefile文件

图片

运行测试

这在文章开头的体验设备树一节中也有演示测试结果:

图片

图片

图片

(话说我好像还没给板子露过面,这下来点个灯露个面

图片

图片

同时,在目录/sys/firmware/devicetree/base下,我们可以查看设备树节点:

图片

可以看到,我们创建的设备树节点100ask_led@0也在该目录下。100ask_led@0节点本身就是一个文件夹,可以使用cd命令进入该文件夹查看该节点的属性信息:

图片

属性值是字符串时,用 cat 命令可以打印出来;属性值是数值时,用 hexdump 命令可以打印出来。

最后

以上就是本次的实验分享。如有错误,欢迎指出!谢谢

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

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

相关文章

10 创建型模式-原型模式

引言&#xff1a; 创建对象的五种方式&#xff1a; 通过new关键字通过Class类的newInstance()方法通过Constructor类的newInstance()方法利用Clone方法反序列化 Clone方法&#xff1a; 其实现方式正是通过调用 Object 类的 clone() 方法来完成。 protected native Object cl…

python安装.whl文件

python --version https://www.lfd.uci.edu/~gohlke/pythonlibs/ 用CtrlF找需要安装的包 下载对应版本的whl python3.8 把下载好的whl放到安装路径下&#xff1a;C:\Users\Administrator\AppData\Local\Programs\Python\Python38\Lib\site-packages 并在该路径下打开cmd执行…

GaussDB数据库管理系统介绍

1.GaussDB的发展 2.GaussDB的生态 内部&#xff1a; 云化自动化方案。通过数据库运行基础设施的云化将DBA(数据库管理员)和运维人员的日常工作 自动化。外部&#xff1a; 采用与数据库周边生态伙伴对接与认证的生态连接融合方案&#xff0c;解决开发者/DBA难获取、应用难对接等…

Linux绝对路径和相对路径

在 Linux 中&#xff0c;简单的理解一个文件的路径&#xff0c;指的就是该文件存放的位置。 只要我们告诉 Linux 系统某个文件存放的准确位置&#xff0c;那么它就可以找到这个文件。指明一个文件存放的位置&#xff0c;有 2 种方法&#xff0c;分别是使用绝对路径和相对路径。…

【驱动开发】LED灯的亮灭——通过字符设备驱动的分步实现编写LED驱动,实现设备文件和设备的绑定

头文件&#xff1a; #ifndef __HEAD_H__ #define __HEAD_H__typedef struct {unsigned int MODER;unsigned int OTYPER;unsigned int OSPEEDR;unsigned int PUPDR;unsigned int IDR;unsigned int ODR; }gpio_t;//LED灯的寄存器地址 #define LED1_ADDR 0X50006000 #define L…

深入理解Redis集群模式、协议、元数据维护方式

文章目录 &#x1f34a; 集群模式&#x1f34a; 集群协议&#x1f34a; 元数据维护方式&#x1f389; 集中式&#x1f389; gossip 协议 &#x1f4d5;我是廖志伟&#xff0c;一名Java开发工程师、Java领域优质创作者、CSDN博客专家、51CTO专家博主、阿里云专家博主、清华大学出…

适用于 Linux 和 Unix 的特权访问管理

凭据、SSH 密钥、服务帐户、数字签名、文件系统等内容构成了Linux 环境的关键部分&#xff0c;虽然大多数PAM供应商为基于Windows的环境提供无缝的特权访问管理&#xff0c;但它们的通用性不足以为Linux&#xff0c;Unix和*nix环境扩展相同的功能和功能。 Linux 中的root权限是…

redis的key超时策略和key淘汰机制(面试题详解)

ChatGPT给出的回答&#xff1a; Redis中的Key超时策略和Key淘汰机制是为了有效管理内存和控制数据的生命周期。 Key超时策略&#xff1a;Redis可以为每个Key设置过期时间&#xff0c;一旦Key过期&#xff0c;它将自动从Redis中删除。可以使用EXPIRE命令为Key设置过期时间&…

系统设计 - 我们如何通俗的理解那些技术的运行原理 - 第六部分:开发运维

本心、输入输出、结果 文章目录 系统设计 - 我们如何通俗的理解那些技术的运行原理 - 第六部分&#xff1a;开发运维前言DevOps vs. SRE vs. Platform Engineering。有什么区别&#xff1f;什么是k8s&#xff08;Kubernetes&#xff09;&#xff1f;控制面板组件节点 Docker vs…

关于阿里云服务器续费详细流程_优惠续费方法

阿里云服务器如何续费&#xff1f;续费流程来了&#xff0c;在云服务器ECS管理控制台选择续费实例、续费时长和续费优惠券&#xff0c;然后提交订单&#xff0c;分分钟即可完成阿里云服务器续费流程&#xff0c;阿里云服务器网分享阿里云服务器详细续费方法&#xff0c;看这一篇…

【代码随想录】算法训练营 第十三天 第五章 栈与队列 Part 3

239. 滑动窗口最大值 题目 给你一个整数数组 nums&#xff0c;有一个大小为 k 的滑动窗口从数组的最左侧移动到数组的最右侧。你只可以看到在滑动窗口内的 k 个数字。滑动窗口每次只向右移动一位。 返回 滑动窗口中的最大值 。 思路 一开始我是直接暴力两层循环的&#xff…

Python学习笔记——MYSQL,SQL核心

食用说明&#xff1a;本笔记适用于有一定编程基础的伙伴们。希望有助于各位&#xff01; SQL语言分类 SQL注释 库管理 表管理 数据操作 分组聚合 分页限制 需要注意的是关键字的顺序不可以错乱&#xff0c;否则会报错其中LIMIT关键字的n是指从第n个开始&#xff0c;m是指查…

http post协议实现简单的rpc协议,WireShark抓包分析

文章目录 1.http 客户端-RPC客户端1.http 服务端-RPC服务端3.WireShark抓包分析3.1客户端到服务端的HTTP/JSON报文3.2服务端到客户端的HTTP/JSON报文 1.http 客户端-RPC客户端 import json import requests# 定义 RPC 客户端类 class RPCClient:def __init__(self, server_url…

【蓝桥杯选拔赛真题43】python二进制位数 青少年组蓝桥杯python 选拔赛STEMA比赛真题解析

目录 python二进制位数 一、题目要求 1、编程实现 2、输入输出 二、算法分析

深入浅出Apache SeaTunnel SQL Server Sink Connector

在大数据时代&#xff0c;数据的迁移和流动已经变得日益重要。为了使数据能够更加高效地从一个源流向另一个目标&#xff0c;我们需要可靠、高效和易于配置的工具。今天&#xff0c;我们将介绍 JDBC SQL Server Sink Connector&#xff0c;这是一个专为 SQL Server 设计的连接器…

嵌入式linux总线设备驱动模型分析

嵌入式linux系统按照&#xff0c;分层&#xff0c;抽象的思想&#xff0c;按照这样的思想来设计我们的程序可以更容易写出耦合性低、独立性强、可重用性强的代码。 Linux内核中更是存在着更多的分离、分层思想的代码&#xff0c;platform平台设备驱动就是用了这样的思想。本篇…

机器学习(新手入门)-线性回归 #房价预测

题目&#xff1a;给定数据集dataSet&#xff0c;每一行代表一组数据记录,每组数据记录中&#xff0c;第一个值为房屋面积&#xff08;单位&#xff1a;平方英尺&#xff09;&#xff0c;第二个值为房屋中的房间数&#xff0c;第三个值为房价&#xff08;单位&#xff1a;千美元…

pv操作题目笔记

对于 pv 操作分以下几步走 什么是pv操作 PV操作在进程同步中通常指的是信号量&#xff08;Semaphore&#xff09;操作。信号量是一种用于控制多个并发进程或线程之间的同步和互斥访问的同步工具。PV操作通常涉及两个基本操作&#xff1a;P操作&#xff08;wait操作&#xff0…

算法通关村第十一关青铜挑战——移位运算详解

大家好&#xff0c;我是怒码少年小码。 计算机到底是怎么处理数字的&#xff1f; 数字在计算机中的表示 机器数 一个数在计算机中的二进制表示形式&#xff0c;叫做这个数的机器数。 机器数是带符号的&#xff0c;在计算机用一个数的最高位存放符号&#xff0c;正数为0&am…

【剑指Offer】:删除链表中的倒数第N个节点(此题是LeetCode上面的)剑指Offer上面是链表中的倒数第K个节点

给定一个链表&#xff0c;删除链表的倒数第 n 个结点&#xff0c;并且返回链表的头结点 示例 1&#xff1a; 输入&#xff1a;head [1,2,3,4,5], n 2 输出&#xff1a;[1,2,3,5] 示例 2&#xff1a; 输入&#xff1a;head [1], n 1 输出&#xff1a;[] 示例 3&#xff1a;…