首页 > 系统相关 >嵌入式linux系统设备树实例分析

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

时间:2023-10-30 11:01:41浏览次数:46  
标签:led 函数 GPIO5 嵌入式 platform 实例 linux 节点 设备


前言

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

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

嵌入式linux系统设备树实例分析_寄存器

框图

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

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

嵌入式linux系统设备树实例分析_运维_03

嵌入式linux系统设备树实例分析_运维_04

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

嵌入式linux系统设备树实例分析_寄存器_06

注意:层与层之间的箭头指向是相对的,从哪指向哪看你怎么理解。比如有两个函数:函数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)>;
};

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

嵌入式linux系统设备树实例分析_寄存器_07

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

make dtbs V=1

嵌入式linux系统设备树实例分析_运维_08

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

嵌入式linux系统设备树实例分析_服务器_09

上传成功的文件如下:

嵌入式linux系统设备树实例分析_服务器_10

运行测试:

嵌入式linux系统设备树实例分析_寄存器_11

实验过程分析

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

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

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

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

嵌入式linux系统设备树实例分析_服务器_12

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

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

嵌入式linux系统设备树实例分析_寄存器_14

嵌入式linux系统设备树实例分析_运维_15

嵌入式linux系统设备树实例分析_寄存器_16

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

  • 根节点下含有 compatile 属性的子节点能转换为platform_device
  • 含有特定 compatile 属性(它的值是 "simple-bus","simplemfd","isa","arm,amba-bus" 四者之一)的节点的子节点能转换为platform_device
  • I2C、 SPI 总线节点下的子节点 不不不能转换为platform_device,这些总线下的子节点, 应该交给对应的总线驱动程序来处理。

下面看一个例子:

嵌入式linux系统设备树实例分析_服务器_17

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

嵌入式linux系统设备树实例分析_寄存器_18

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

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

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

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

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

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

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

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

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

实验代码

代码大多与之前的【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

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

驱动程序入口函数:

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

open、write函数:

嵌入式linux系统设备树实例分析_寄存器_22

其它代码:

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

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

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

3、硬件层:chip_demo_gpio.c

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

(1)驱动初始化函数:

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

(2)probe函数:

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

嵌入式linux系统设备树实例分析_寄存器_26

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

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

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

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

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

(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文件

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

运行测试

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

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

嵌入式linux系统设备树实例分析_寄存器_30

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

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

嵌入式linux系统设备树实例分析_服务器_32

嵌入式linux系统设备树实例分析_运维_33

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

嵌入式linux系统设备树实例分析_服务器_34

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

嵌入式linux系统设备树实例分析_服务器_35

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

最后

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

标签:led,函数,GPIO5,嵌入式,platform,实例,linux,节点,设备
From: https://blog.51cto.com/u_11947739/8086589

相关文章

  • 【Python】在linux环境下, Django 3.0+版本遇到的一些问题
    1.Django3.2版本升级后的一些报错,导致项目运行失败1.1site-packages/rest_framework/serializers.py fromdjango.db.models.fieldsimportFieldDoesNotExistImportError:cannotimportname'FieldDoesNotExist'解决办法:改变FieldDoesNotExist的导入方式fromdjango......
  • 嵌入式硬件中常见的100种硬件选型方式
    1请列举您知道的电阻、电容、电感品牌(最好包括国内、国外品牌)。电阻:美国:AVX、VISHAY威世日本:KOA兴亚、Kyocera京瓷、muRata村田、Panasonic松下、ROHM罗姆、susumu、TDK台湾:LIZ丽智、PHYCOM飞元、RALEC旺诠、ROYALOHM厚生、SUPEROHM美隆、TA-I大毅、TMTEC泰铭、TOK......
  • 嵌入式硬件库的基本操作方式与分析
    本次要介绍的开源软件是c-periphery:https://github.com/vsergeev/c-periphery一个用C语言编写的硬件外设访问库。我们可以用它来读写Serial、SPI、I2C等,非常适合在嵌入式产品上使用。我们可以基于它优秀的代码框架,不断地扩展出更多的功能模块,最终形成自己产品适用的Linux硬......
  • 实战攻防演练-Linux写入ssh密钥,利用密钥登录
    前言密钥形式登录的原理是利用密钥生成器制作一对密钥,一只公钥和一只私钥。将公钥添加到服务器的某个账户上,然后在客户端利用私钥即可完成认证并登录。这样一来,没有私钥,任何人都无法通过SSH暴力破解你的密码来远程登录到系统。此外,如果将公钥复制到其他账户甚至主机,利用私钥也......
  • linux基本文件命令复习笔记
    1,放大缩小终端窗口字体  放大 ctrlshift+=   缩小  ctrl-2,6个常见终端命令 (1)ls  查看当前文件夹下的内容 (2)pwd 查看当前所在文件夹  (3)cd目录名 切换文件夹 (4)touch文件名 如果文件不存在,新建文件。和mkdir不同的是,mkdir创......
  • python进阶14大模块200页知识体系md笔记,第3篇:linux命令进阶
    本完整笔记从14大模块展示了python高级用的应用。分别有Linux命令,多任务编程、网络编程、Http协议和静态Web编程、html+css、JavaScript、jQuery、MySql数据库的各种用法、python的闭包和装饰器、mini-web框架、正则表达式等相关文章的详细讲述。完整版笔记直接地址:请移步这里......
  • 第九周 Linux课后技术总结
    5.1初始进程什么是进程?进程是已启动的可执行程序的运行实例。进程有以下组成部分。已分配内存的地址空间。安全属性,包括所有权凭据和特权。程序代码的一个或多个的执行线程。进程状态。每个进程都有唯一的进程标识PID,一个PID只能标识一个进程。(使用pidof命令查看sshd......
  • Linux用户管理
    1用户/组概览Linux系统是多用户、多任务的分时操作系统,系统上每一个进程都有一个特定的文件,每个文件都被一个特定的用户所拥有。如果需要使用系统资源,首先必须向系统超级用户申请成为普通用户,然后以普通用户的身份进入系统。超级用户可以对普通用户进行跟踪,并设置他们的访问权限,......
  • 创建一个自己的 Linux系统
    简单来说就是一个文件传递的机制,首先创建/安装一个硬盘,然后把前硬盘中的一部分文件先转移到Linux系统上,再通过Linux系统转移到创建的新硬盘,之后用虚拟机,把新硬盘装在其中,就可以在新硬盘上做到一些功能了前知识准备:Linux启动流程:1、首先Linux要通过自检,检查硬件设备有......
  • 第一章 初识Linux
    1.Linux简介1.1Linux是多用户多任务的操作系统,免费开源,主要用于搭建服务器,性能稳定。但是Linux系统是专业系统,原始环境较为简单,大部分功能的实现都需要自我搭建。1.2  Linux系统都是由文件构成。对于操作系统内核而言,命令、硬件、软件设备都被视为一个拥有特定功能的......