【嵌入式Linux】i.MX6ULL 外部中断服务函数的初始化

文章目录

  • 1. Cortex-A7 中断系统
    • 1.1 分析
    • 1.2 具体处理流程
  • 2. 外部中断服务函数的初始化
    • 2.1 基本流程分析
    • 2.2 具体代码分析
      • 2.2.1. 定义中断处理类型和结构体
      • 2.2.2. 初始化中断系统
      • 2.2.3. 注册中断处理函数
      • 2.2.4. 具体的中断处理逻辑
      • 2.2.5. 默认的中断处理函数
  • 3. 完整代码

本文章结合了正点原子的 i.mx6u嵌入式Linux开发指南和笔者的理解。

1. Cortex-A7 中断系统

1.1 分析

1. 中断向量表:

  • Cortex-A7 也有中断向量表,位于代码的最前面。
  • 包含 8 个异常中断,每个中断对应一个向量地址和中断模式。
  • 表 17.1.2.1 列出了这 8 个中断:
    • 复位中断 (Reset):特权模式 (SVC)
    • 未定义指令中断 (Undefined Instruction):未定义指令中止模式 (Undef)
    • 软中断 (Software Interrupt, SWI):特权模式 (SVC)
    • 指令预取中止中断 (Prefetch Abort):中止模式
    • 数据访问中止中断 (Data Abort):中止模式
    • 未使用 (Not Used):未使用
    • IRQ 中断 (IRQ Interrupt):外部中断模式 (IRQ)
    • FIQ 中断 (FIQ Interrupt):快速中断模式 (FIQ)

2. Cortex-A7 中断向量表与 Cortex-M 的区别:

  • Cortex-M 中断向量表列举了芯片的所有中断向量,包括外设中断。
  • Cortex-A7 中断向量表只列出了 8 个基本中断,其中 IRQ 中断包含了所有外部中断。
  • 也就是说,Cortex-A7 中,所有外部中断都通过 IRQ 中断进行处理。

3. IRQ 中断处理:

  • 当任意一个外部中断发生时,都会触发 IRQ 中断。
  • IRQ 中断服务函数需要读取指定的寄存器来判断具体是哪个外部中断发生了。
  • 根据具体的中断做出相应的处理。

4. 外部中断与 IRQ 中断的关系:
在这里插入图片描述

  • 图中左侧的 Software0_IRQn~PMU_IRQ2_IRQ 都是 I.MX6U 的外部中断,它们都属于 IRQ 中断。
  • 当任意一个外部中断发生时,IRQ 中断会被触发。
  • 在 IRQ 中断服务函数中需要判断是哪个外部中断发生了,并进行相应的处理。

1.2 具体处理流程

  1. 某个外部中断发生。
  2. IRQ 中断被触发。
  3. IRQ 中断服务函数执行。
  4. IRQ 中断服务函数调用 C 语言中断处理函数。
  5. C 语言中断处理函数根据中断号判断是哪个外部中断发生了。
  6. C 语言中断处理函数执行相应的处理逻辑。
  • IRQ 中断就像一个“总开关”,而外部中断就像“子开关”。当某个子开关打开时,会触发总开关,然后需要根据子开关的具体情况进行处理。
  • 前面的文章我们已经对IRQ中断服务函数进行了初始化,设置在IRQ中断服务函数中执行C语言的中断服务函数:
ldr r2, =system_irqhandler	/* 加载C语言中断处理函数到r2寄存器中*/
blx r2						/* 运行C语言中断处理函数,带有一个参数,保存在R0寄存器中 */

下面我们就开始在C语言中,根据不同的中断号进行不同的中断服务函数初始化和处理。

2. 外部中断服务函数的初始化

2.1 基本流程分析

  1. 定义中断处理类型和结构体
  2. 初始化中断系统
  3. 注册中断处理函数
  4. 具体的中断处理逻辑
  5. 默认的中断处理函数

2.2 具体代码分析

2.2.1. 定义中断处理类型和结构体

此步骤定义了中断处理函数的类型别名 system_irq_handler_t 和中断处理函数结构体 sys_irq_handle_t

typedef void (*system_valve_handler_t)(unsigned int gicciar, void *param);

typedef struct _sys_irq_handle{
    system_valve_handler_t irqHandler; // 中断处理函数
    void *usrparam;                    // 中断处理函数的参数
} sys_irq_handle_t;

/*中断处理函数表*/
static sys_irq_handle_t irqTable[NUMBER_OF_INT_VECTORS];

解释

  • system_irq_handler_t 是一个函数指针类型,指向的函数用于处理中断,接收中断号和用户参数。
  • sys_irq_handle_t 结构体包含一个中断处理函数和一个用于传递给该函数的用户参数。
  • irqTable是一个结构体数组,包含所有外部中断的结构体,共NUMBER_OF_INT_VECTORS=160个。

2.2.2. 初始化中断系统

初始化中断系统包括 GIC 初始化和中断向量表的初始化。

/*初始中断向量表*/
void system_irqtable_init(void)
{
    irqNesting = 0;//中断计数器清0
    unsigned int i;
    for(i=0; i<NUMBER_OF_INT_VECTORS; i++){
        irqTable[i].irqHandler = default_irqhandler;
        irqTable[i].usrparam = NULL;
    }
}

void int_Init(void)
{
    GIC_Init(); // 初始化GIC
    system_irqtable_init();
    __set_VBAR(0x87800000); // 中断向量偏移设置
}

解释

  • GIC_Init() 调用是用来初始化通用中断控制器(Generic Interrupt Controller)。
  • system_irqtable_init() 初始化中断处理函数表,为每个可能的中断分配默认处理函数。
  • __set_VBAR(0x87800000) 设置中断向量基地址寄存器(VBAR)。

2.2.3. 注册中断处理函数

此步骤允许用户为特定的中断号注册自定义的处理函数。

void system_register_irqhandler(IRQn_Type irq, system_irq_handler_t handler, void *param)
{
    irqTable[irq].irqHandler = handler;
    irqTable[irq].usrparam = param;
}

解释

  • 通过指定的 irq 中断号,用户可以将自定义的处理函数 handler 和参数 param 注册到中断处理函数表 irqTable 中。

2.2.4. 具体的中断处理逻辑

  • 该部分是实际的中断处理逻辑,由具体的中断处理函数 system_irqhandler 实现。
  • system_irqhandler实际上由IRQ中断调用,完成了选择不同外部中断函数进行处理中断的功能。
void system_irqhandler(unsigned int gicciar)
{
    irqNesting++; // 中断计数器
    uint32_t intNum = gicciar & 0x3FF; // 从低10位获取中断ID
    if(intNum >= NUMBER_OF_INT_VECTORS) return; // 检查中断ID是否正常
    irqTable[intNum].irqHandler(intNum, irqTable[intNum].usrparam); // 根据中断号,选取中断处理函数执行
    irqNesting--; // 中断执行完,中断计数器-1
}

解释

  • irqNesting 记录当前正在处理的中断嵌套层数。
  • 通过 gicciar & 0x3FF 提取中断号,并检查其有效性,然后调用相应的中断处理函数。

2.2.5. 默认的中断处理函数

提供一个默认的中断处理函数,当没有特定的处理函数注册时使用。

void default_irqhandler(unsigned int gicciar, void *param)
{
    while(1); // 死循环,用于捕捉未处理的中断
}

解释

  • 这是一个防止未注册中断导致系统异常退出的安全措施。如果执行到此函数,表明发生了未注册处理函数的中断。

3. 完整代码

bsp_int.c

#include "bsp_int.h"
static unsigned int irqNesting;//中断计数器,用于嵌套

/*中断处理函数表*/
static sys_irq_handle_t irqTable[NUMBER_OF_INT_VECTORS];
/*初始中断向量表*/
void system_irqtable_init(void)
{
    irqNesting = 0;//中断计数器清0
    unsigned int i;
    for(i=0; i<NUMBER_OF_INT_VECTORS; i++){
        irqTable[i].irqHandler = default_irqhandler;
        irqTable[i].usrparam = NULL;
    }
}

/*注册中断处理函数*/
void system_register_irqhandler(IRQn_Type irq, system_irq_handler_t handler, void *param)
{
    irqTable[irq].irqHandler = handler;
    irqTable[irq].usrparam = param;
}

//中断初始化函数
void int_Init(void)
{
    GIC_Init();//初始化GIC
    system_irqtable_init();

    __set_VBAR(0x87800000);//中断向量偏移设置

}


/*具体的中断处理函数,IRQ_Handler会调用此C函数*/
void system_irqhandler(unsigned int gicciar)
{
    irqNesting++;//中断计数器
    uint32_t intNum = gicciar & 0x3FF;//从低10位获取中断ID
    //检查中断ID是否正常
    if(intNum >= NUMBER_OF_INT_VECTORS) return;
    //根据中断号,选取中断处理函数执行
    irqTable[intNum].irqHandler(intNum, irqTable[intNum].usrparam);
    irqNesting++;//中断执行完,中断计数器-1
}


//默认中断处理函数
void default_irqhandler(unsigned int gicciar, void *param)
{
    while(1);
}

bsp_int.h

#ifndef __BSP_INT_H
#define __BSP_INT_H

#include "imx6u.h"

// 这段代码定义了一个类型别名 system_irq_handler_t,
// 用于表示一个指向中断处理函数的指针。
// 该函数指针指向的函数接收两个参数:
// 中断号
// 和一个指向任意类型数据的指针,用于传递额外的参数。
typedef void (*system_irq_handler_t)(unsigned int gicciar, void *param);

//中断处理函数结构体
typedef struct _sys_irq_handle{
    system_irq_handler_t irqHandler;//中断处理函数
    void *usrparam;//中断处理函数的参数
}sys_irq_handle_t;

void int_Init(void);
void system_irqtable_init();
void default_irqhandler(unsigned int gicciar, void *param);
void system_register_irqhandler(IRQn_Type irq, system_irq_handler_t handler, void *param);

#endif // !__BSP_INT_H

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

002_unsigned long数据比较的坑?

【背景】 unsigned long 类似数据的比较问题&#xff0c;先上一段代码&#xff0c;如下图所示&#xff1a; 就是图中框出的部分&#xff0c;眨眼一看&#xff0c;应该没啥问题&#xff0c;而且我也在本地的编译器vs2019上编译了&#xff0c;确实也没有报错&#xff0c;所以就修…

【Linux】静态库、动态库

动静态库里面包含的是源文件通过汇编阶段生成的后缀为.o的可重定位目标文件。我们在使用C语言&#xff0c;包含一个stdio.h头文件就可以使用scanf方法&#xff0c;其实都是系统调用了相应的头文件和库&#xff0c;库里面有开发者已经写好各种方法。也就是说我们在使用C语言时&a…

Java | Leetcode Java题解之第191题位1的个数

题目&#xff1a; 题解&#xff1a; public class Solution {public int hammingWeight(int n) {int ret 0;while (n ! 0) {n & n - 1;ret;}return ret;} }

【学习】软件测试中常见的文档类型及其作用

在软件开发的生命周期中&#xff0c;软件测试是确保产品质量的关键步骤。为了系统地进行测试活动&#xff0c;并保证测试结果的有效性和可追溯性&#xff0c;产生了一系列标准化的测试文档。这些文档不仅为测试人员提供了执行指南&#xff0c;而且为项目管理者和利益相关者提供…

【排序 队列】1585. 检查字符串是否可以通过排序子字符串得到另一个字符串

本文涉及知识点 排序 队列 LeetCode1585. 检查字符串是否可以通过排序子字符串得到另一个字符串 给你两个字符串 s 和 t &#xff0c;请你通过若干次以下操作将字符串 s 转化成字符串 t &#xff1a; 选择 s 中一个 非空 子字符串并将它包含的字符就地 升序 排序。 比方说&a…

Discourse 的 AI 内容分享

虽然 Discourse 的 AI 接口调用是需要比较高的用户权限或者管理员权限。 但是对已经生成的结果&#xff0c;Discourse 是可以保存并且分享的。 例如&#xff0c;我们搜索了一些美食的做法。 在页面的下面有一个分享 AI 对话的按钮。 在随后弹出的界面中&#xff0c;会又一个…

服务运营 | MS文章精选:线上点单,当真免排队?餐饮零售与医疗场景中的全渠道运营

编者按&#xff1a; 小A走进了一家奶茶店&#xff0c;准备向店员点单&#xff0c;但却在屏幕上看到还有98杯奶茶待制作&#xff08;因为线上订单突然暴增&#xff09;。因此&#xff0c;小A不满地嘟囔着离开了奶茶店。这个例子展示了线上渠道可能会对线下渠道造成一些负面影响…

链表数组遍历输出的辨析(二者都含指针的情况下)----PTA期末复习题

输入输出三位学生的学号和信息 一开始我认为是指针&#xff0c;直接背了指针输出的方式&#xff1b;p;p!NULL;pp->next 这个是错误的 下面这个输出是正确的方式 分析怎么区分这两个 举个例子来 数组遍历&#xff1a; 链表遍历&#xff1a; 输出的结果&#xff1a; 如果将…

第十次作业

1.登陆界面 2.导航页面 3.接口&#xff08;我负责的主要是管理员管理用户和密码的界面&#xff09; import request from /utils/request// 登录 export function login(data) {return request({url: /user/login,method: post,data}) }// 获取用户信息 export function getIn…

网关登录校验

如何在网关转发之前做登录校验&#xff1f; 网关请求处理流程 如何在网关转发之前做登录校验&#xff1f; 网关如何将用户信息传递给微服务&#xff1f; 如何在微服务之间传递用户信息&#xff1f; 自定义过滤器 网关过滤器有两种&#xff0c;分别是&#xff1a; GatewayFi…

春秋云境:CVE-2022-25411[漏洞复现]

根据题目提示和CNNVD优先寻找后台管理地址 靶机启动后&#xff0c;使用AWVS进行扫描查看网站结构 在这里可以看到后台管理的登录地址&#xff1a;/admin/&#xff0c;根据题目提示可知是弱口令 尝试admin、123456、admin666、admin123、admin888...等等常见弱口令 正确的账户…

论文导读 | Manufacturing Service Operations Management近期文章精选

编者按 在本系列文章中&#xff0c;我们梳理了顶刊Manufacturing & Service Operations Management5月份发布有关OR/OM以及相关应用的文章之基本信息&#xff0c;旨在帮助读者快速洞察行业/学界最新动态。 推荐文章1 ● 题目&#xff1a;Robust Drone Delivery with Weath…

KVM网络模式设置

一、KVM网络模式介绍 1、NAT ( 默认上网 ) 虚拟机利用host机器的ip进行上网,对外显示一个ip;virbr0是KVM 默认创建的一个 Bridge,其作用是为连接其上的虚机网卡提供NAT访问外网的功能,默认ip为192.168.122.1 2、自带的Bridge 将虚拟机桥接到host机器的网卡上,vm和ho…

【C++题解】1712. 输出满足条件的整数2

问题&#xff1a;1712. 输出满足条件的整数2 类型&#xff1a;简单循环 题目描述&#xff1a; 有这样的三位数&#xff0c;其百位、十位、个位的数字之和为偶数&#xff0c;且百位大于十位&#xff0c;十位大于个位&#xff0c;请输出满所有满足条件的整数。 输入&#xff1…

C++ | Leetcode C++题解之第191题位1的个数

题目&#xff1a; 题解&#xff1a; class Solution { public:int hammingWeight(uint32_t n) {int ret 0;while (n) {n & n - 1;ret;}return ret;} };

SpringBoot控制反转和依赖注入

目录 一、内聚和耦合 二、分层解耦 三、具体实现 四、bean的组件扫描 五、bean注入 一、内聚和耦合 在了解分层解耦的概念之前我们我们要去先了解一下内聚和耦合。内聚&#xff1a;通常将的是软件中各个模块之间的功能联系。耦合衡量软件各个模块之间的依赖、关联的程度。一…

【ai】tx2 nx : fix pip升级警告

jetson 环境同样出现:【原创】pip3 使用报警问题在对 Ubuntu 18.04 上的 pip3 9.0.1 版本使用 pip install -U pip 的方式进行升级后,再使用 pip 就会出现一堆警告信息。这个警告信息目前不影响使用,但从警告信息来看,会在未来版本中出现失败风险。 当前系统中存在了两个不…

Android反编译之Apktool

文章目录 简述工具操作步骤 简述 可以从apk安装包中提取出res、AndroidManifest、xml等文件&#xff1b;也可以修改资源文件后rebuild一个apk。 工具 1.官方下载地址 https://apktool.org/ 2.操作指令 // 解析apk包 $ apktool d test.apk // 重新rebuid apk包 $ apktool …

vscode_cmake_stm32_lvgl移植及显示优化

1 LVGL移植 本文使用的环境如下&#xff1a; STM32H743FreeRTOSst7789 lcd(320*240) 下载 LVGL源码&#xff0c;本文使用Release v9.1.0&#xff1b; 将压缩包解压到工程目录&#xff0c;例如stm32h7xx_cmake_project/components/lvgl-9.1.0&#xff0c;如下所示&#xff1a; …

vue3封装表格嵌套表单问题汇总

1.插槽嵌套多层数据ui组件怎么使用 思路&#xff1a;插槽具名【区分】后暴露传递&#xff0c;这个为神魔要区分&#xff0c;因为封装组件表格列表项也有插槽 步骤一&#xff1a;表单插槽暴露 <ElFormclass"form-search":model"formParams"ref"form…