嵌入式Linux应用开发-基础知识-第十八章系统对中断的处理②

嵌入式Linux应用开发-基础知识-第十八章系统对中断的处理②

  • 第十八章 Linux系统对中断的处理 ②
    • 18.3 Linux中断系统中的重要数据结构
      • 18.3.1 irq_desc数组
      • 18.3.2 irqaction结构体
      • 18.3.3 irq_data结构体
      • 18.3.4 irq_domain结构体
      • 18.3.5 irq_chip结构体
    • 18.4 在设备树中指定中断_在代码中获得中断
      • 18.4.1 设备树里中断节点的语法
        • 18.4.1.1 设备树里的中断控制器
        • 18.4.1.2 设备树里使用中断
      • 18.4.2 设备树里中断节点的示例
      • 18.4.3 在代码中获得中断
        • 18.4.3.1 对于 platform_device
        • 18.4.3.2 对于 I2C设备、SPI设备
        • 18.4.3.3 调用 of_irq_get获得中断号
        • 18.4.3.4 对于 GPIO

第十八章 Linux系统对中断的处理 ②

在这里插入图片描述

18.3 Linux中断系统中的重要数据结构

本节内容,可以从 request_irq(include/linux/interrupt.h)函数一路分析得到。 能弄清楚下面这个图,对 Linux中断系统的掌握也基本到位了。
在这里插入图片描述

最核心的结构体是 irq_desc,之前为了易于理解,我们说在 Linux内核中有一个中断数组,对于每一个硬件中断,都有一个数组项,这个数组就是 irq_desc数组。

注意:如果内核配置了 CONFIG_SPARSE_IRQ,那么它就会用基数树(radix tree)来代替 irq_desc数组。SPARSE的意思是“稀疏”,假设大小为 1000的数组中只用到 2个数组项,那不是浪费嘛?所以在中断比较“稀疏”的情况下可以用基数树来代替数组。

18.3.1 irq_desc数组

irq_desc结构体在 include/linux/irqdesc.h中定义,主要内容如下图:
在这里插入图片描述

每一个 irq_desc数组项中都有一个函数:handle_irq,还有一个 action链表。要理解它们,需要先看中断结构图:
在这里插入图片描述

外部设备 1、外部设备 n共享一个 GPIO中断 B,多个 GPIO中断汇聚到 GIC(通用中断控制器)的 A号中断,GIC再去中断 CPU。那么软件处理时就是反过来,先读取 GIC获得中断号 A,再细分出 GPIO中断 B,最后判断是哪一个外部芯片发生了中断。
所以,中断的处理函数来源有三:
① GIC的处理函数:
假设 irq_desc[A].handle_irq是 XXX_gpio_irq_handler(XXX指厂家),这个函数需要读取芯片的 GPIO控制器,细分发生的是哪一个 GPIO中断(假设是 B),再去调用 irq_desc[B]. handle_irq。
注意:irq_desc[A].handle_irq细分出中断后 B,调用对应的 irq_desc[B].handle_irq。
显然中断 A是 CPU感受到的顶层的中断,GIC中断 CPU时,CPU读取 GIC状态得到中断 A。
② 模块的中断处理函数:
比如对于 GPIO模块向 GIC发出的中断 B,它的处理函数是 irq_desc[B].handle_irq。
BSP开发人员会设置对应的处理函数,一般是 handle_level_irq或 handle_edge_irq,从名字上看是用来处理电平触发的中断、边沿触发的中断。
注意:导致 GPIO中断 B发生的原因很多,可能是外部设备 1,可能是外部设备 n,可能只是某一个设备,也可能是多个设备。所以 irq_desc[B].handle_irq会调用某个链表里的函数,这些函数由外部设备提供。这些函数自行判断该中断是否自己产生,若是则处理。
③ 外部设备提供的处理函数:
这里说的“外部设备”可能是芯片,也可能总是简单的按键。它们的处理函数由自己驱动程序提供,这是最熟悉这个设备的“人”:它知道如何判断设备是否发生了中断,如何处理中断。
对于共享中断,比如 GPIO中断 B,它的中断来源可能有多个,每个中断源对应一个中断处理函数。所以 irq_desc[B]中应该有一个链表,存放着多个中断源的处理函数。
一旦程序确定发生了 GPIO中断 B,那么就会从链表里把那些函数取出来,一一执行。
这个链表就是 action链表。
对于我们举的这个例子来说,irq_desc数组如下:
在这里插入图片描述

18.3.2 irqaction结构体

irqaction结构体在 include/linux/interrupt.h中定义,主要内容如下图:
在这里插入图片描述

当调用 request_irq、request_threaded_irq注册中断处理函数时,内核就会构造一个 irqaction结构体。在里面保存 name、dev_id等,最重要的是 handler、thread_fn、thread。
handler是中断处理的上半部函数,用来处理紧急的事情。
thread_fn对应一个内核线程 thread,当 handler执行完毕,Linux内核会唤醒对应的内核线程。在内核线程里,会调用 thread_fn函数。
可以提供 handler而不提供 thread_fn,就退化为一般的 request_irq函数。
可以不提供 handler只提供 thread_fn,完全由内核线程来处理中断。
也可以既提供 handler也提供 thread_fn,这就是中断上半部、下半部。
里面还有一个名为 sedondary的 irqaction结构体,它的作用以后再分析。
在 reqeust_irq时可以传入 dev_id,为何需要 dev_id?作用有 2:
① 中断处理函数执行时,可以使用 dev_id
② 卸载中断时要传入 dev_id,这样才能在 action链表中根据 dev_id找到对应项 所以在共享中断中必须提供 dev_id,非共享中断可以不提供。

18.3.3 irq_data结构体

irq_data结构体在 include/linux/irq.h中定义,主要内容如下图:
在这里插入图片描述

它就是个中转站,里面有 irq_chip指针 irq_domain指针,都是指向别的结构体。
比较有意思的是 irq、hwirq,irq是软件中断号,hwirq是硬件中断号。比如上面我们举的例子,在 GPIO中断 B是软件中断号,可以找到 irq_desc[B]这个数组项;GPIO里的第 x号中断,这就是 hwirq。
谁来建立 irq、hwirq之间的联系呢?由 irq_domain来建立。irq_domain会把本地的 hwirq映射为全局的 irq,什么意思?比如 GPIO控制器里有第 1号中断,UART模块里也有第 1号中断,这两个“第 1号中断”是不一样的,它们属于不同的“域”──irq_domain。

18.3.4 irq_domain结构体

irq_domain结构体在 include/linux/irqdomain.h中定义,主要内容如下图:
在这里插入图片描述

当我们后面从设备树讲起,如何在设备树中指定中断,设备树的中断如何被转换为 irq时,irq_domain将会起到极大的作为。
这里基于入门的解度简单讲讲,在设备树中你会看到这样的属性:

interrupt-parent = <&gpio1>; 
interrupts = <5 IRQ_TYPE_EDGE_RISING>; 

它表示要使用 gpio1里的第 5号中断,hwirq就是 5。
但是我们在驱动中会使用 request_irq(irq, handler)这样的函数来注册中断,irq是什么?它是软件中断号,它应该从“gpio1的第 5号中断”转换得来。
谁把 hwirq转换为 irq?由 gpio1的相关数据结构,就是 gpio1对应的 irq_domain结构体。
irq_domain结构体中有一个 irq_domain_ops结构体,里面有各种操作函数,主要是:
① xlate
用来解析设备树的中断属性,提取出 hwirq、type等信息。
② map
把 hwirq转换为 irq。

18.3.5 irq_chip结构体

irq_chip结构体在 include/linux/irq.h中定义,主要内容如下图:
在这里插入图片描述

这个结构体跟“chip”即芯片相关,里面各成员的作用在头文件中也列得很清楚,摘录部分如下:

* @irq_startup: start up the interrupt (defaults to ->enable if NULL) 
* @irq_shutdown: shut down the interrupt (defaults to ->disable if NULL) 
* @irq_enable:  enable the interrupt (defaults to chip->unmask if NULL) 
* @irq_disable: disable the interrupt 
* @irq_ack:  start of a new interrupt 
* @irq_mask:  mask an interrupt source 
* @irq_mask_ack: ack and mask an interrupt source 
* @irq_unmask:  unmask an interrupt source 
* @irq_eoi:  end of interrupt 

我们在 request_irq后,并不需要手工去使能中断,原因就是系统调用对应的 irq_chip里的函数帮我们使能了中断。
我们提供的中断处理函数中,也不需要执行主芯片相关的清中断操作,也是系统帮我们调用 irq_chip中的相关函数。
但是对于外部设备相关的清中断操作,还是需要我们自己做的。
就像上面图里的“外部设备 1“、“外部设备 n”,外设备千变万化,内核里可没有对应的清除中断操作。

18.4 在设备树中指定中断_在代码中获得中断

18.4.1 设备树里中断节点的语法

参考文档:
内核 Documentation\devicetree\bindings\interrupt-controller\interrupts.txt

18.4.1.1 设备树里的中断控制器

中断的硬件框图如下:
在这里插入图片描述

在硬件上,“中断控制器”只有 GIC这一个,但是我们在软件上也可以把上图中的“GPIO”称为“中断控制器”。很多芯片有多个 GPIO模块,比如 GPIO1、GPIO2等等。所以软件上的“中断控制器”就有很多个:GIC、GPIO1、GPIO2等等。
GPIO1连接到 GIC,GPIO2连接到 GIC,所以 GPIO1的父亲是 GIC,GPIO2的父亲是 GIC。
假设 GPIO1有 32个中断源,但是它把其中的 16个汇聚起来向 GIC发出一个中断,把另外 16个汇聚起来向 GIC发出另一个中断。这就意味着 GPIO1会用到 GIC的两个中断,会涉及 GIC里的 2个 hwirq。
这些层级关系、中断号(hwirq),都会在设备树中有所体现。

在设备树中,中断控制器节点中必须有一个属性:interrupt-controller,表明它是“中断控制器”。 还必须有一个属性:#interrupt-cells,表明引用这个中断控制器的话需要多少个 cell。
#interrupt-cells的值一般有如下取值:
① #interrupt-cells=<1>
别的节点要使用这个中断控制器时,只需要一个 cell来表明使用“哪一个中断”。
② #interrupt-cells=<2>
别的节点要使用这个中断控制器时,需要一个 cell来表明使用“哪一个中断”;
还需要另一个 cell来描述中断,一般是表明触发类型:
第 2个 cell的 bits[3:0] 用来表示中断触发类型(trigger type and level flags):

1 = low-to-high edge triggered,上升沿触发 
2 = high-to-low edge triggered,下降沿触发 
4 = active high level-sensitive,高电平触发 
8 = active low level-sensitive,低电平触发 

示例如下:

vic: intc@10140000 { compatible = "arm,versatile-vic"; interrupt-controller; #interrupt-cells = <1>; reg = <0x10140000 0x1000>; 
}; 

如果中断控制器有级联关系,下级的中断控制器还需要表明它的“interrupt-parent”是谁,用了interrupt-parent”中的哪一个“interrupts”,请看下一小节。

18.4.1.2 设备树里使用中断

一个外设,它的中断信号接到哪个“中断控制器”的哪个“中断引脚”,这个中断的触发方式是怎样的? 这 3个问题,在设备树里使用中断时,都要有所体现。
① interrupt-parent=<&XXXX>
你要用哪一个中断控制器里的中断?
② interrupts
你要用哪一个中断?
Interrupts里要用几个 cell,由 interrupt-parent对应的中断控制器决定。在中断控制器里有
“#interrupt-cells”属性,它指明了要用几个 cell来描述中断。
比如:

i2c@7000c000 { 
gpioext: gpio-adnp@41 { compatible = "ad,gpio-adnp"; 
interrupt-parent = <&gpio>; interrupts = <160 1>; 
gpio-controller; 
#gpio-cells = <1>; }; ...... }; 
interrupt-controller; 
#interrupt-cells = <2>; 

③ 新写法:interrupts-extended
一个“interrupts-extended”属性就可以既指定“interrupt-parent”,也指定“interrupts”,比如: interrupts-extended = <&intc1 5 1>, <&intc2 1 0>;

18.4.2 设备树里中断节点的示例

以 xxxxxx_IMX6ULL开发板为例,在 arch/arm/boot/dts目录下可以看到 2个文件:imx6ull.dtsi、xxxxxx_imx6ull-14x14.dts,把里面有关中断的部分内容抽取出来。
在这里插入图片描述

从设备树反推 IMX6ULL的中断体系,如下,比之前的框图多了一个“GPC INTC”:
在这里插入图片描述

GPC INTC的英文是:General Power Controller, Interrupt Controller。它提供中断屏蔽、中断状态查询功能,实际上这些功能在 GIC里也实现了,个人觉得有点多余。除此之外,它还提供唤醒功能,这才是保留它的原因。

18.4.3 在代码中获得中断

之前我们提到过,设备树中的节点有些能被转换为内核里的 platform_device,有些不能,回顾如下: A. 根节点下含有 compatile属性的子节点,会转换为 platform_device
B. 含有特定 compatile属性的节点的子节点,会转换为 platform_device
如果一个节点的 compatile属性,它的值是这 4者之一: “simple-bus”,“simple-mfd”,“isa”,“arm,amba-bus”,
那么它的子结点(需含 compatile属性)也可以转换为 platform_device。
C. 总线 I2C、SPI节点下的子节点:不转换为 platform_device
某个总线下到子节点,应该交给对应的总线驱动程序来处理, 它们不应该被转换为 platform_device。

18.4.3.1 对于 platform_device

一个节点能被转换为 platform_device,如果它的设备树里指定了中断属性,那么可以从
platform_device中获得“中断资源”,函数如下,可以使用下列函数获得 IORESOURCE_IRQ资源,即中断号:

/** * platform_get_resource - get a resource for a device * @dev: platform device * @type: resource type   // 取哪类资源?IORESOURCE_MEM、IORESOURCE_REG 
*                      // IORESOURCE_IRQ等 * @num: resource index  // 这类资源中的哪一个? */ 
struct resource *platform_get_resource(struct platform_device *dev, unsigned int type, unsigned int num); 
18.4.3.2 对于 I2C设备、SPI设备

对于 I2C设备节点,I2C总线驱动在处理设备树里的 I2C子节点时,也会处理其中的中断信息。一个I2C设备会被转换为一个 i2c_client结构体,中断号会保存在 i2c_client的 irq成员里,代码如下(drivers/i2c/i2c-core.c):
在这里插入图片描述
对于 SPI设备节点,SPI总线驱动在处理设备树里的 SPI子节点时,也会处理其中的中断信息。一个SPI设备会被转换为一个 spi_device结构体,中断号会保存在 spi_device的 irq成员里,代码如下(drivers/spi/spi.c):
在这里插入图片描述

18.4.3.3 调用 of_irq_get获得中断号

如果你的设备节点既不能转换为 platform_device,它也不是 I2C设备,不是 SPI设备,那么在驱动程序中可以自行调用 of_irq_get函数去解析设备树,得到中断号。

18.4.3.4 对于 GPIO

参考:drivers/input/keyboard/gpio_keys.c
可以使用 gpio_to_irq或 gpiod_to_irq获得中断号。 举例,假设在设备树中有如下节点:

gpio-keys { compatible = "gpio-keys"; pinctrl-names = "default"; user { label = "User Button"; gpios = <&gpio5 1 GPIO_ACTIVE_HIGH>;     gpio-key,wakeup; linux,code = <KEY_1>; }; 
}; 

那么可以使用下面的函数获得引脚和 flag:

button->gpio = of_get_gpio_flags(pp, 0, &flags); 
bdata->gpiod = gpio_to_desc(button->gpio); 

再去使用 gpiod_to_irq获得中断号:

irq = gpiod_to_irq(bdata->gpiod); 

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

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

相关文章

Spring Boot的自动装配中的@ConditionalOnBean条件装配注解在Spring启动过程中,是如何保证处理顺序靠后的

前言 为什么Spring Boot条件注解那么多&#xff0c;而标题中是ConditionalOnBean呢&#xff1f; 因为&#xff0c;相比之下我们用的比较多的条件装配注解也就是ConditionalOnClass、ConditionalOnBean了&#xff0c;而ConditionalOnClass对顺序并不敏感&#xff08;说白了就是判…

Halcon中灰度直方图的使用与学习

目录 第一步:当前打开窗口的显示灰度图或者mono图片第二步:激活后,我们可以去调整调整右边直方图灰阶值的中蓝色和红色竖线,获取左边图上的灰阶值的范围内的特征显示。第三步:插入代码:总结:它的直观目的,就是查看灰度的分布情况!灰度直方图,是我们经常使用,抓取不同…

matlab 计算数组中所有值的均值

目录 一、概述1、算法概述2、主要函数3、输入参数4、输出参数二、代码实现三、结果展示四、参考链接本文由CSDN点云侠翻译,放入付费专栏只为防不要脸的爬虫。专栏值钱的不是本文,切勿因本文而订阅。 一、概述 1、算法概述 矩阵元素的平均值或均值。 2、主要函数<

使用VBA实现快速模糊查询数据

实例需求&#xff1a;基础数据保存在Database工作表中&#xff0c;如下图所示。 基础数据有37个字段&#xff0c;上图仅展示部分字段内容&#xff0c;下图中黄色字段为需要提取的数据字段。 在Search工作表B1单元格输入查询关键字Title和Genre字段中搜索关键字&#xff0c;包…

Spring Boot事件机制浅析

1、概述 在设计模式中&#xff0c;观察者模式是一个比较常用的设计模式。维基百科解释如下&#xff1a; 观察者模式是软件设计模式的一种。在此种模式中&#xff0c;一个目标对象管理所有相依于它的观察者对象&#xff0c;并且在它本身的状态改变时主动发出通知。这通常透过呼…

五分钟k8s入门到实战-应用配置

ConfigMap.png 背景 在前面三节中已经讲到如何将我们的应用部署到 k8s 集群并提供对外访问的能力&#xff0c;x现在可以满足基本的应用开发需求了。 现在我们需要更进一步&#xff0c;使用 k8s 提供的一些其他对象来标准化我的应用开发。首先就是 ConfigMap&#xff0c;从它的名…

C++ 强制类型转换(int double)、查看数据类型、自动决定类型、三元表达式、取反、

强制类型转换&#xff08; int 与 double&#xff09; #include <iostream> using namespace std;int main() {// 数据类型转换char c1;short s1;int n 1;long l 1;float f 1;double d 1;int p 0;int cc (int)c;// 注意&#xff1a;字符 转 整形时 是有问题的// “…

Apache Derby的使用

Apache Derby是关系型数据库&#xff0c;可以嵌入式方式运行&#xff0c;也可以独立运行&#xff0c;当使用嵌入式方式运行时常用于单元测试&#xff0c;本篇我们就使用单元测试来探索Apache Derby的使用 一、使用IDEA创建Maven项目 打开IDEA创建Maven项目&#xff0c;这里我…

HTML详细基础(三)表单控件

本帖介绍web开发中非常核心的标签——表格标签。 在日常我们使用到的各种需要输入用户信息的场景——如下图&#xff0c;均是通过表格标签table创造出来的&#xff1a; 目录 一.表格标签 二.表格属性 三.合并单元格 四.无序列表 五.有序列表 六.自定义标签 七.表单域 …

BI神器Power Query(27)-- 使用PQ实现表格多列转换(3/3)

实例需求&#xff1a;原始表格包含多列属性数据,现在需要将不同属性分列展示在不同的行中&#xff0c;att1、att3、att5为一组&#xff0c;att2、att3、att6为另一组&#xff0c;数据如下所示。 更新表格数据 原始数据表&#xff1a; Col1Col2Att1Att2Att3Att4Att5Att6AAADD…

山西电力市场日前价格预测【2023-10-02】

日前价格预测 预测说明&#xff1a; 如上图所示&#xff0c;预测明日&#xff08;2023-10-02&#xff09;山西电力市场全天平均日前电价为355.35元/MWh。其中&#xff0c;最高日前电价为521.18元/MWh&#xff0c;预计出现在18: 45。最低日前电价为309.36元/MWh&#xff0c;预计…

Windows权限维持

Meterpreter权限维持 Metasploit 框架提供了一个后渗透模块&#xff0c;可实现自动化地利用沾滞键的权限维持技术。 该模块将用 CMD 替换辅助功能的二进制文件&#xff08; sethc, osk, disp, utilman &#xff09; use post/windows/manage/sticky_keys 设置session 提示&a…

双指针算法——复写零

双指针算法——复写零&#x1f60e; 前言&#x1f64c;复写零板书分析&#xff1a;解题代码&#xff1a;B站视频讲解 总结撒花&#x1f49e; &#x1f60e;博客昵称&#xff1a;博客小梦 &#x1f60a;最喜欢的座右铭&#xff1a;全神贯注的上吧&#xff01;&#xff01;&#…

Arm Cache学习资料大汇总

关键词&#xff1a;cache学习、mmu学习、cache资料、mmu资料、arm资料、armv8资料、armv9资料、 trustzone视频、tee视频、ATF视频、secureboot视频、安全启动视频、selinux视频&#xff0c;cache视频、mmu视频&#xff0c;armv8视频、armv9视频、FF-A视频、密码学视频、RME/CC…

3分钟学会设计模式 -- 单例模式

►使用场景 在编写软件时&#xff0c;对于某些类来说&#xff0c;只有一个实例很重要。例如&#xff0c;一个系统中可以存在多个打印任务&#xff0c;但是只能有一个正在工作的任务&#xff1b;一个系统中可以多次查询数据库&#xff0c;但是只需要一个连接&#xff0c;而不是…

2023年中国艺术涂料市场发展历程及趋势分析:艺术涂料市场规模将进一步扩大[图]

艺术涂料是一种用于绘画和装饰&#xff0c;具有各种纹理或通过涂装手段后具有高装饰性的新型涂料。由于具有高度饱和的颜色、良好的遮盖力和可塑性&#xff0c;呈现立体装饰效果好、色彩搭配适当、风格独具特色的特点&#xff0c;而使得涂装出的饰面自然贴合、更加美观漂亮&…

【EasyPoi】SpringBoot使用EasyPoi自定义模版导出Excel

EasyPoi 官方文档&#xff1a;http://doc.wupaas.com/docs/easypoi Excel模版导出 引入依赖 <dependencies><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-web</artifactId></dependency…

数据结构与算法-(7)---栈的应用-(3)表达式转换

&#x1f308;write in front&#x1f308; &#x1f9f8;大家好&#xff0c;我是Aileen&#x1f9f8;.希望你看完之后&#xff0c;能对你有所帮助&#xff0c;不足请指正&#xff01;共同学习交流. &#x1f194;本文由Aileen_0v0&#x1f9f8; 原创 CSDN首发&#x1f412; 如…

【刷题笔记10.2】LeetCode: 罗马数字转整数

LeetCode: 罗马数字转整数 一、题目描述 二、分析 方法一&#xff1a; 将给定字符串s中的"IV", “IX”, “XL”, “XC”, “CD”, “CM” 全部替换为其他字符如&#xff1a;a, b, c, d, e, f 这种&#xff0c;然后就可以遍历累加了。 s s.replace("IV",…

Grafana 开源了一款 eBPF 采集器 Beyla

eBPF 的发展如火如荼&#xff0c;在可观测性领域大放异彩&#xff0c;Grafana 近期也发布了一款 eBPF 采集器&#xff0c;可以采集服务的 RED 指标&#xff0c;本文做一个尝鲜介绍&#xff0c;让读者有个大概了解。 eBPF 基础介绍可以参考我之前的文章《eBPF Hello world》。理…