0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
会员中心
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

openEuler Kernel技术内核中断子系统介绍

openEuler 来源:openEuler 作者:openEuler 2022-04-20 09:06 次阅读

很多人在学习中断子系统的过程中,在对基本概念与整体不太了解的情况下,过早的陷入了各种架构的实现细节,如同盲人摸象。这里主要给大家明确中断的各个基本概念,希望从这个角度能让大家更好的理解中断子系统。

什么是中断

在计算机科学中,中断(英语:Interrupt)是指处理器接收到来自硬件或软件的信号,提示发生了某个事件,应该被注意,这种情况就称为中断。中断子系统中的中断指的是其中硬件的一方,后续中断均按此理解。

中断处理的参与对象和流程

中断处理中有着多个对象的参与,理解每个对象在其中是如何参与是很重要的。以下列举了中断处理的参与对象。

中断事件:指中断事件本身的抽象。

中断号:用于硬件和软件识别并区分中断事件,需要注意同一个中断事件在中断处理的不同阶段未必是同一个中断号。

中断源:有中断事件需要 cpu 处理的硬件。

中断控制器:非必须,用于解决系统拥有多中断源场景的硬件;从中断源接收中断事件并传递到 cpu;可以级联。

cpu:收到中断,cpu 跳转到特定的地址——中断向量。由中断向量开始软件对中断的处理。

中断事件在硬件中的流程如下,上一行是中断事件的体现形式,下一行是所在的硬件:

307ba222-c045-11ec-bce3-dac502259ad0.png

再把软件处理结合起来,形成一个硬件软件切换的过程:

30992400-c045-11ec-bce3-dac502259ad0.png

相邻的中断事件体现形式的映射方式可以在所在的对象的连接的实现中找到。

中断子系统

现在把之前的流程具有的部分对比内核中断子系统,可以发现还多出了一个通用中断处理层。因为内核需要支持各种不同的架构与外设,需要解耦架构硬件相关部分(cpu 与中断控制器)与非架构相关(外设),使得开发外设驱动并不需要了解架构相关部分。另一方面,系统硬件拓扑结构的信息一般由设备树源码 DTS 体现。

30abe1bc-c045-11ec-bce3-dac502259ad0.png

硬件封装层

硬件封装层包括 cpu 和中断控制器两部分。区分开 cpu 和中断控制器相当重要,希望大家能更明确 cpu 和中断控制器的概念。软件在 cpu 方面主要需要按架构实现中断向量的处理,可以看 arch/「/kernel/entry」.S 的汇编实现。另外还需要为通用的开关中断方法提供架构实现:

通用中断开关方法 具体架构中断开关实现
local_irq_enable arch_local_irq_enable
local_irq_disable arch_local_irq_disable

这个 local 指的是 local_cpu,表示的是当前 cpu 是否响应中断:当前 cpu 关中断的情况下,中断控制器不管怎么玩都是徒劳的。事实上 cpu 对中断开关的实现还包含着很多条件,类似特权态、非屏蔽中断 NMI 之类的,可以在后文找下具体分析。软件对中断控制器的抽象是 struct irq_chip,体现的是中断控制器所具体的行为。这里列举部分重要成员讲解:

起因 struct irq_chip 成员 说明
怎么控制中断控制器是否屏蔽某个中断事件? irq_enable/ irq_disable
中断控制器如何配置中断事件的触发方式 irq_set_type 控制各个中断的电气触发条件,例如边沿触发或者是电平触发。
中断控制器如何得知中断事件被 cpu 响应? irq_ack 中断控制器在实现中会根据中断事件被 cpu 开始响应或完成响应来决定该中断事件类型是否会再度通知 cpu 处理。
中断控制器如何得知 cpu 完成处理中断事件? irq_eoi 中断控制器在实现中会根据中断事件被 cpu 开始响应或完成响应来决定该中断事件类型是否会再度通知 cpu 处理。
在 smp 系统中,中断事件应该通知哪个 cpu? irq_set_affinity affinity 表示了中断事件在中断控制器中配置的目标 cpu,根据具体实现可以是 1 个或多个。

此外,当多个中断事件同时发生,中断控制器会根据其优先级的实现来决定中断事件通知给 cpu 的顺序,某些实现是可配置的。另一方面,考虑到系统中可能存在多个中断控制器,使得单一中断控制器的中断号不足以区分中断事件,所以引入了软件中断号的概念。加上硬件中断号映射中断号的软件抽象 struct irq_domain,再看中断控制器软件抽象到中断源软件抽象的流程:

30d10938-c045-11ec-bce3-dac502259ad0.png

##中断流控处理层这一层主要是隐藏了中断控制器在具体中断事件处理函数调用前后的一些处理逻辑,包括:

何时对中断控制器发出 ack 回应?

mask_irq 和 unmask_irq 的处理;

中断控制器是否需要 eoi 回应?

何时打开 cpu 的本地 irq 中断?以便允许 irq 的嵌套;

类似于在用洗衣机洗衣服的时候,我们不关心衣服可能要经历过的洗涤多久、脱水多久、漂洗多久诸如此类的步骤细节,只需要按衣服类型选择流程;内核引入中断流程的抽象类型 irq_flow_handler_t 屏蔽了中断事件相关的 cpu、中断控制器和中断源的属性的不同带来的处理流程差异。这里举例部分内核实现:

handle_simple_irq:用于简易流控处理。

handle_level_irq:用于电平触发中断的流控处理。

handle_edge_irq:用于边沿触发中断的流控处理。

handle_fasteoi_irq:用于需要响应 eoi 的中断控制器

handle_percpu_irq:用于只在单一 cpu 响应的中断。

驱动程序 API 与中断通用逻辑

对于中断事件本身,内核使用 struct irq_desc 进行描述,它包含着所有的信息。而对于中断控制器与中断源的驱动来说,关注的信息都只是其中的一部分。中断事件从中断源到中断控制器的映射的描述一般事先会静态定义好并存放在设备树源码里,即中断源的设备树节点包含着相连的中断控制器和中断事件对应在中断控制器中断号的信息;而作为驱动程序需要对软件中断号 irq 和中断事件处理函数建立映射。那么要把设备树节点中的中断控制器和中断控制器中断号转换成软件中断号 irq,内核给驱动程序提供了接口

irq_of_parse_and_map:驱动由设备树节点获得 irq。

当中断控制器和中断控制器中断号转换成软件中断号 irq 映射不存在时,这个接口会申请 irq_desc 并建立映射,根据连接着的中断控制器的驱动提供的硬件中断号映射中断号的软件抽象 irq_domain 完成映射。在映射过程中会包括对 irq_desc 的一些属性的设置,如:

irq_set_handler:驱动选择 irq_flow_handler。

irq_set_chip:驱动选择 irq 连接的中断控制器。

irq_alloc_desc 系列:驱动直接申请 irq_desc。

中断源驱动获取到 irq,还需要将 irq 与中断处理函数建立映射:

request_irq/request_threaded_irq:驱动将中断处理函数注册到 irq。

enable_irq & disable_irq:驱动开关 irq。

接下来将对一些具体的架构实现做介绍。这里介绍两个处理器 armv8 和 x86,以及两个中断控制器 arm-gicv3 和 x86-APIC 的实现。希望帮助大家得出诸如“arm 内核有中断嵌套吗”“arm cpu eoi 是做什么”这类问题的答案。

armv8

arm 核心拥有 2 个外部中断线,IRQ 和 FIQ;这两根中断线连接到中断控制器上,中断控制器通过拉高和拉低这两根中断线触发中断。一个中断应该触发 IRQ 还是 FIQ 中断线,由其 GROUP 属性和当前的特权级和安全域决定。

30e3b93e-c045-11ec-bce3-dac502259ad0.png

GROUP 的定义:

311a2032-c045-11ec-bce3-dac502259ad0.png

arm 核,软件可以写 SCR、HCR 和 PSTATE.DAIF 寄存器以决定响应中断的特权级和屏蔽中断;arm 不支持 NMI。arm 核由于中断控制器的实现,同时只会有一个需要被响应的中断,因此不限制 IRQ/FIQ 响应顺序的实现。arm 核上处于触发状态的中断线需要结合 SCR、HCR 和 PSTATE.DAIF 寄存器判断是否触发中断,不论当前是否处于中断。在中断触发时,arm 核心根据 VBAR 系列寄存器的基地址,会按具体情况选择偏移跳转到对应的地址。

312d13d6-c045-11ec-bce3-dac502259ad0.png

x86

Intel x86 架构提供 INTR 和 NMI 两个中断引脚,他们通常与 Local APIC 相连, 用于接收 Local APIC 传递的中断信号。一个中断应该触发 INTR 还是 NMI 中断线由 Local APIC 实现。

315d2ada-c045-11ec-bce3-dac502259ad0.png

x86,中断都在 ring0 响应。x86 上软件使用 CLI 指令将本 CPU 的 EFLAGS 寄存器的 IF 位清 0,阻止接收中断;STI 指令将 IF 位置为 1,允许接收中断。这两条指令都只对当前 CPU 起作用,而不影响平台上的其他 CPU。x86 中断线的实现原生支持 NMI。x86 核上同时只会有一个需要被响应的中断,它由 Local APIC 从 IRR 中选择;当 Local APIC 不使能时,优先响应 NMI 中断线。不论当前是否处于中断,x86 核上若 INTR 处于触发且未屏蔽中断即会触发中断;NMI 处于触发则直接触发中断。中断触发时,x86 核根据寄存器 IDTR 记录的基址和中断控制器的寄存器 ISR 提供的中断向量号找到 IDT 表中对应的 Interrupt Gate 表项,跳转到相应的地址。

arm-gicv3

31782344-c045-11ec-bce3-dac502259ad0.png

从逻辑视图上看,gicv3 的核外部分统称 IRI,由 Distributor、ITS、Redistributor 这 3 种组件组成;gicv3 核内部分是 CPU interface,PE 可以理解为 cpu;IRI 与 CPU interface 通过 GIC Stream Protocol interface 交互。

不同的中断在 gic 上对应着不同的 INTID;gic 把中断类型分为 LPI、PPI、SPI、SGI,约束 INTID 取值对应的中断类型。SGI 指由 CPU 直接写对应的寄存器触发中断;PPI 指中断为特定一个 CPU 私有/专用,同一中断号的 PPI 在不同 CPU 可以指不同的中断源;SPI 对应 PPI,是所有 CPU 全局共享的,同一中断号的 SPI 在不同 CPU 均指相同的中断源;LPI 的区分是中断路由上的不同,主要是在 IRI 中由 ITS 路由的中断,其余 3 种中断均不经过 ITS;某些实现下还有直接在 Redistributor 触发的 LPI 中断。

319b65c0-c045-11ec-bce3-dac502259ad0.png

31b4e522-c045-11ec-bce3-dac502259ad0.png

一个外部中断从在外设上产生,依次经过 IRI、CPU interface 并最终通过中断线到达 PE;PE 产生的中断需要先经过 CPU interface 到 IRI,再到目标的 CPU interface 和 PE。逻辑上,IRI 可以对应多个 PE,因此对于需要被一个特定目标 PE 响应的中断,gicv3 通过引入 affinity routing 机制解决这种路由问题。同一时间,CPU interface 上只能存在一个待处理的中断,对于多个中断被发送到 CPU interface 上,gic 引入优先级的机制来决定如何选择保留的中断;这个优先级的机制还被运用在 IRI 上,优先级更高的中断会被优先发送到 CPU interface。另外,CPU interface 还负责将这个待处理的中断按照 GROUP 属性和当前的特权级和安全域决定触发 IRQ 还是 FIQ 中断线;并且当 PE 当前处于中断时,CPU interface 还需要通过中断优先级分组的机制判断待处理的中断是否需要被通知给 PE,即抢占。

x86-APIC

315d2ada-c045-11ec-bce3-dac502259ad0.png

从逻辑视图上看,APIC 的核外部分是 I/O APIC,核内部分是 Local APIC。I/O APIC 根据内部 PRT table 中的 RTE 发送中断消息给 Local APIC。I/O APIC 中 PRT table 由 24 个 RTE 项组成,每一项对应一个 IRQ 引脚。I/O APIC 可以有多个,当多个 I/O APIC 存在时,使用 GSI 代表每个 I/O APIC 管脚的编号:例如 I/O APIC1 有 24 个 IRQ,I/O APIC2 也有 24 个 IRQ,则 I/O APIC2 的 GSI 是从 24 开始,GSI = 24 + IRQ(I/O APIC2)。I/O APIC 的 24 个管脚没有优先级之分。一个外部中断经过 I/O APIC 再到 Local APIC,最后由 Local APIC 控制中断线在 CPU 上触发中断;CPU 内部的中断源由 Local APIC 管理,不需要经过 I/O APIC;IPI 也由 Local APIC 管理,同样不需要经过 I/O APIC。

31dc24ca-c045-11ec-bce3-dac502259ad0.png

Local APIC 支持 0-255 的中断向量号,它们可以同时存在于寄存器 IRR 上,引入中断优先级进行选择:优先级 = 中断向量号 / 16因为 32 以下的中断向量号是保留的,所以可用中断优先级范围为 2-15,数字越大优先级越高;当优先级高于寄存器 PPR 的情况下会操作 INTR 中断线,若当前已经处于中断则可能出现抢占。中断向量号的低 4 位会在当 PPR 改变的情况下,ISR 从 IRR 上选择中断向量号的比较中使用,同样也是数字越大优先级越高。

审核编辑 :李倩

声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • 处理器
    +关注

    关注

    68

    文章

    19160

    浏览量

    229115
  • 控制器
    +关注

    关注

    112

    文章

    16198

    浏览量

    177398

原文标题:openEuler Kernel 技术解读 | 内核中断子系统介绍

文章出处:【微信号:openEulercommunity,微信公众号:openEuler】欢迎添加关注!文章转载请注明出处。

收藏 人收藏

    评论

    相关推荐

    润和软件亮相操作系统大会openEuler Summit 2024

    近日,操作系统大会&openEuler Summit 2024在北京隆重举办。本次大会由openEuler社区和全球计算联盟主办,江苏润和软件股份有限公司(以下简称“润和软件”)参与协办。旨在汇聚全球产业发展力量,共同探讨操作
    的头像 发表于 11-17 09:08 354次阅读

    润和软件参与openEuler社区操作系统安全主题技术交流

    信息技术飞速发展,操作系统安全已成为保障数字世界稳定运行的基石,OpenAtom openEuler(简称"openEuler")致力于构建一个开放、可靠、安全的开源操作
    的头像 发表于 11-13 16:14 320次阅读

    解决视频采集子系统故障的工具和技术

    电子发烧友网站提供《解决视频采集子系统故障的工具和技术.pdf》资料免费下载
    发表于 10-10 09:24 0次下载
    解决视频采集<b class='flag-5'>子系统</b>故障的工具和<b class='flag-5'>技术</b>

    深度解析linux时钟子系统

    linux内核中实现了一个CLK子系统,用于对上层提供各模块(例如需要时钟信号的外设,USB等)的时钟驱动接口,对下层提供具体SOC的时钟操作细节。
    的头像 发表于 09-29 16:46 301次阅读
    深度解析linux时钟<b class='flag-5'>子系统</b>

    Linux内核测试技术

    内核测试技术是实现这一目标的关键手段。本文将详细介绍 Linux 内核测试的各种技术,包括单元测试、集成测试、功能测试和性能测试等,并讨论
    的头像 发表于 08-13 13:42 434次阅读
    Linux<b class='flag-5'>内核</b>测试<b class='flag-5'>技术</b>

    润和软件受邀参与openEuler 24.03 LTS发布

    发布会汇聚操作系统产业界顶尖力量,共探openEuler社区技术、生态、国际化发展,共建健康可持续发展的开源操作系统根社区。会上,openEuler
    的头像 发表于 06-15 10:07 745次阅读

    SoC芯片设计系列-ARM CPU子系统组件介绍

    在ARM架构的CPU子系统中,组件设计旨在高效地整合了多种功能模块,以支持处理器核心的运行、内存管理、中断处理、数据交换以及与外部设备的交互等。
    的头像 发表于 05-31 10:56 2488次阅读
    SoC芯片设计系列-ARM CPU<b class='flag-5'>子系统</b>组件<b class='flag-5'>介绍</b>

    鸿蒙开发学习:【驱动子系统

    OpenHarmony驱动子系统采用C面向对象编程模型构建,通过平台解耦、内核解耦,兼容不同内核,提供了归一化的驱动平台底座,旨在为开发者提供更精准、更高效的开发环境,力求做到一次开发,多系统
    的头像 发表于 03-17 22:05 542次阅读
    鸿蒙开发学习:【驱动<b class='flag-5'>子系统</b>】

    基于 PCIe 的多路视频采集与显示子系统介绍

    电子发烧友网站提供《基于 PCIe 的多路视频采集与显示子系统介绍.pdf》资料免费下载
    发表于 02-23 09:47 0次下载

    openEuler Code Camp圆满举办

    12月17日,2023开放原子开发者大会openEuler Code Camp在无锡成功举办。此次盛会吸引了众多行 业领袖、前沿开发者和用户共襄盛举,共同聆听openEuler在推动生态系统迅速发展
    的头像 发表于 12-20 09:45 279次阅读
    <b class='flag-5'>openEuler</b> Code Camp圆满举办

    如何修改内核设备树

    本文档介绍内核设备树的位置和包含关系 1.内核设备树位置 文件 备注 dts longan/device/config/chips/t507/configs/evb/board.dts
    发表于 12-14 13:42

    Linux内核自解压过程分析

    uboot完成系统引导以后,执行环境变量bootm中的命令;即,将Linux内核调入内存中并调用do_bootm函数启动内核,跳转至kernel的起始位置。
    的头像 发表于 12-08 14:00 852次阅读
    Linux<b class='flag-5'>内核</b>自解压过程分析

    51单片机中断系统介绍

    51单片机中断系统介绍与运用外部中断系统触发数码管显示数字的代码实验
    发表于 12-05 09:10 2次下载

    kernel到android核心启动过程

    kernel_init 与kthreadd,创建完后系统通过init_idle_bootup_task蜕化为idle进程(cpu_idle)。 调用kernel_thread()创建1号内核
    的头像 发表于 12-04 16:59 888次阅读
    <b class='flag-5'>kernel</b>到android核心启动过程

    菁蓉联创获得首个openEuler嵌入式操作系统技术测评证书

    近日,成都菁蓉联创科技有限公司(简称:菁蓉联创)携手四川鲲鹏&欧拉生态创新中心完成首个openEuler嵌入式操作系统技术测评证书。
    的头像 发表于 11-28 10:18 1012次阅读
    菁蓉联创获得首个<b class='flag-5'>openEuler</b>嵌入式操作<b class='flag-5'>系统</b><b class='flag-5'>技术</b>测评证书