【S5PV210_视频编解码项目】裸机开发:实现按键的外部中断处理

加粗样式本文所作内容:
基于S5PV210芯片实现按键的外部中断处理程序,搭建中断处理流程框架

S5PV210对于中断处理的操作流程

1 外部中断得到触发:

1)外部中断在初始化阶段得到使能
2)外界达到了外部中断的触发条件

2 跳转到向量表:

1)异常向量表在初始化阶段进行了配置ISR处理程序
2)ISR处理程序包含了保护现场和回复现场的功能
3)ISR处理程序能够跳转到中断总ISR_handel程序

3 跳转到对应中断号的中断处理程序

1)对应中断号的中断得到初始化使能
2)在ISR_handel中找到对应出现的中断号
3)在ISR_handel中跳转到对应该中断号的isr中

S5PV210实现外部中断处理的程序流

初始化阶段

1)首先初始化异常向量表:

 //初始化异常向量表r_exception_reset = (unsigned int)reset_exception;r_exception_undef = (unsigned int)undef_exception;r_exception_sotf_int = (unsigned int)sotf_int_exception;r_exception_prefetch = (unsigned int)prefetch_exception;r_exception_data = (unsigned int)data_exception;r_exception_irq = (unsigned int)IRQ_handle;r_exception_fiq = (unsigned int)IRQ_handle;

2)然后初始化芯片的中断系统:先全部中断号禁止中断,然后中断工作模式为IRQ,然后清除中断跳转寄存器

 //禁止所有中断VIC0INTENCLEAR = 0xFFFFFFFF;VIC1INTENCLEAR = 0xFFFFFFFF;VIC2INTENCLEAR = 0xFFFFFFFF;VIC3INTENCLEAR = 0xFFFFFFFF;//配置中断类型 (默认全部为IRQ工作模式)VIC0INTSELECT = 0x0;VIC1INTSELECT = 0x0;VIC2INTSELECT = 0x0;VIC3INTSELECT = 0x0;//清除中断跳转寄存器VIC0ADDR = 0;VIC1ADDR = 0;VIC2ADDR = 0;VIC3ADDR = 0;

3)设置想要调试的外部中断:设置对应GPIO的工作模式外外部中断模式,使能对应的外部中断,设置外部中断的触发方式

//配置IO引脚为外部中断工作模式rGPH0CON |= (0xFF<<8); rGPH2CON |= (0xFFFF<<0);//配置对应外部中断号下中断触发形式:下降沿触发rEXT_INT_0_CON &= ~(0xFF<<8);	rEXT_INT_0_CON |= ((2<<8)|(2<<12));		rEXT_INT_2_CON &= ~(0xFFFF<<0);rEXT_INT_2_CON |= ((2<<0)|(2<<4)|(2<<8)|(2<<12));	/*设置为0表示使能中断*/rEXT_INT_0_MASK &= ~(3<<2);	rEXT_INT_2_MASK &= ~(0x0f<<0);/*软件写1 表示清除flag*/rEXT_INT_0_PEND |= (3<<2);rEXT_INT_2_PEND |= (0x0F<<0);

实现中断处理流程

1)实现第一阶段的异常向量表跳转程序:汇编实现(保存现场-跳转IRQ处理程序-现场恢复)

IRQ_handle:// 设置IRQ模式下的栈指针ldr sp, =IRQ_STACK// 保存现场// 保存LR寄存器 lr寄存器先减去4(ARM的流水线机制)sub lr, lr, #4// 保存R0-R12寄存器stmfd sp!, {r0-r12, lr}// 跳转到真正的中断处理程序bl irq_handler// 回复现场ldmfd sp!, {r0-r12, pc}^

2)实现中断处理第二阶段:寻找中断编号,寻找中断编号对应的isr

/*1-根据状态寄存器 找到对应的中断控制器位置(VIC0/1/2/3)*/unsigned long vicaddr[4] = {VIC0ADDR,VIC1ADDR,VIC2ADDR,VIC3ADDR};void (*isr)(void) = NULL;unsigned int i = 0;for(i=0; i<3 ; i++){if(intc_getvicirqstatus(i)){/*2-根据中断控制器位置,跳转到对应中断控制器的vectaddr中拿到对应的中断isr*/isr = (void (*)(void)) vicaddr[i];break; }}(*isr)();  

3)使能对应外部中断的中断编号,配置对应该中断编号的isr程序

//根据中断号 将对应的中断回调函数绑定在VECTADDR寄存器中
void intc_setvectaddr(unsigned long intnum, void (*handler)(void))
{if(intnum < 32){( *((volatile unsigned long *)(VIC0VECTADDR + 4*(intnum - 0))) ) = (unsigned)handler;}else if(intnum < 64){( *((volatile unsigned long *)(VIC0VECTADDR + 4*(intnum - 32))) ) = (unsigned)handler;}else if(intnum < 96){( *((volatile unsigned long *)(VIC0VECTADDR + 4*(intnum - 64))) ) = (unsigned)handler;}else if(intnum < 128){( *((volatile unsigned long *)(VIC0VECTADDR + 4*(intnum - 96))) ) = (unsigned)handler;}else{printf("ERR!:intc_setvectaddr err: intnum out of max of SOC\n");}return;
}
//根据中断号将对应中断使能
void intc_enable(unsigned long intnum)
{unsigned int sTemp = 0;if(intnum < 32){sTemp = VIC0INTENABLE;sTemp |= (1<<(intnum - 0));VIC0INTENABLE = sTemp;}else if(intnum < 64){sTemp = VIC1INTENABLE;sTemp |= (1<<(intnum - 32));VIC1INTENABLE = sTemp;}else if(intnum < 96){sTemp = VIC2INTENABLE;sTemp |= (1<<(intnum - 64));VIC2INTENABLE = sTemp;}else if(intnum < 128){sTemp = VIC3INTENABLE;sTemp |= (1<<(intnum - 96));VIC3INTENABLE = sTemp;}else{VIC0INTENABLE = 0xFFFFFFFF;VIC1INTENABLE = 0xFFFFFFFF;VIC2INTENABLE = 0xFFFFFFFF;VIC3INTENABLE = 0xFFFFFFFF;printf("ERR!:intc_enable err: intnum out of max of SOC\n");}
}

在功能调试时遇到的问题

问题一:按下按键后程序没有跳转到中断处理程序

1)分析1:结合没有心跳信息打印,怀疑程序跑飞,需要调试出程序具体在哪一行飞了
2)行动1:在main函数中加入串口打印信息,看看程序在哪一行之后不能正常打印信息了
3)结果1:发现程序在配置中断向量表的第一行飞了
在这里插入图片描述

4)分析2:检查该行,还行的目的是将函数指针以地址的形式填充到中断向量表中,但是中断向量表只有一个首地址,需要根据中断号计算地址。明显偏移地址计算出错,地址是以字节形式递增的,应该×4
5)行动2:

6)结果2:问题解决

问题二:中断只能触发一次,之后就不能触发了,同时心跳信息消失

1)分析1:怀疑是中断返回时候现场恢复有问题,但是检查现场恢复程序为标准处理过程,问题出现在异常向量表的处理阶段,检查初始化代码的流程看是否缺少异常向量表初始化
2)行动1:检查初始化程序流,发现缺少了一步异常向量表初始化,加入异常向量表初始化
3)结果1:问题解决

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

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

相关文章

24考研数学最大教训❗️660/880过时了?

我没看错吧&#xff0c;说660题和880题过时了&#xff1f; 660题和880题好好用&#xff0c;这俩很经典不会过时。 660题是客观题训练必刷的一本题集&#xff0c;而880是强化阶段非常好的一本综合性题集。我本身在考研的时候使用的也是这两本题集&#xff0c;所以对这两本题集…

如何学习一个大型分布式Java项目

前言 很多同学在没有实习经验的时候看到一个多模块分布式项目总是有一种老虎吃天的无力感&#xff0c;就像我刚毕业去到公司接触项目的时候一样&#xff0c;模块多的夸张&#xff0c;想学都不知道从哪开始学&#xff0c;那么我们拿到一份代码后如何从头开始学习一个新项目呢。…

Oracle Primavera Analytics 是什么,与P6的关系?

前言 Oracle Primavera P6 Analytics 是与P6有关的一个相对较新的模块&#xff0c;Primavera 用户社区在很大程度上尚未对其进行探索。 那么它到底有什么作用呢&#xff1f; 通过了解得知它旨在通过深入了解组织的项目组合绩效&#xff0c;帮助高级管理层对其项目组合做出更好…

MySQL | 库的操作 | 表的操作

目录 1. 库的操作 1.1. 创建数据库 1.2. 字符集和校验规则 1.2.1. 查看系统默认字符集以及校验规则 1.2.2. 查看数据库支持的字符集 1.2.3. 查看数据库支持的字符校验规则 2. 操作数据库 2.1. 查看数据库 2.2. 显示创建语句 2.3. 修改数据库 2.4. 数据库的删除 2.4.…

维基百科推广秘诀13个方法助你成为行业领导者-华媒舍

维基百科&#xff08;Wikipedia&#xff09;作为全球最大、最权威的在线百科全书&#xff0c;拥有海量的知识内容&#xff0c;被广大用户广泛使用。对于任何一个领域的从业者来说&#xff0c;建立自己的维基百科页面&#xff0c;无疑是提升行业影响力的重要手段。本文将向您介绍…

Linux学习(4)——使用编辑器

1.gedit编辑器 简单易懂&#xff0c;依赖图形界面。可以使用ctrlc ctrlv等快捷键&#xff0c;ctrls进行保存&#xff0c;与windows系统中相类似。 2.vi/vim编辑器 vi/vim可以直接通过控制台的终端完成文本的编辑&#xff0c;不依赖图形界面&#xff0c;使用范围更广。它的编辑…

Redis数据结构对象之字符串对象

字符串对象 字符串对象的编码可以是int、raw或者embstr 如果一个字符串对象保存的是整数值&#xff0c;并且这个整数值可以用long类型来表示&#xff0c;那么字符串对象会将整数值保存在字符串对象结构的ptr属性里面(将void *转换成long)&#xff0c;并且将字符串对象的编码设…

【Cute】MMA抽象代码理解 c2d9bff3d88846eb8c523fb722166bc9

【Cute】MMA抽象代码理解 导读&#xff1a; cute 之 Layoutcute Layout 的代数和几何解释cute 之 Tensorcute 之 MMA抽象cute 之 简单GEMM实现 阅读本文前建议先读上面reed大神的数篇文章&#xff0c;文本逻辑主要是针对具体的代码&#xff0c;记录一下自己学习过程中的理解…

Atlas200板卡部署车道线

本博客包含推理的准备和部署代码&#xff0c;一步步实现部署。 这个运行时生成的一个batch的数据&#xff0c;NCHW,就是输入的N&#xff0c;单图片推理就是1&#xff0c;把里面的数量改成1&#xff0c;但是你可以多生成一些bin图片放到校准文件夹中&#xff0c;更改输出文件名…

“城市绿肺诊断:集成GIS、RS、VORS模型、CCDM模型、geodetecto、GWR模型技术深入解析生态系统与城镇化协调发展“

基于GIS、RS、VORS模型、CCDM模型、geodetecto、GWR模型集成的生态系统健康的耦合协调分析 城市群是一国经济发展水平的象征&#xff0c;也是一国经济发展到一定阶段的标志&#xff0c;我国城市群建设体量不断增加&#xff0c;将成为全球经济的核心&#xff0c;中国城市群的建…

MyFileServer

靶场下载地址 https://download.vulnhub.com/myfileserver/My_file_server_1.ova 信息收集 # nmap -sn 192.168.56.0/24 -oN live.nmap Starting Nmap 7.94 ( https://nmap.org ) at 2024-02-24 22:07 CST Nmap scan report for 192.168.56.2 (192.168.56.2) Host is up (0.…

HarmonyOS NEXT应用开发—状态栏显隐变化

介绍 本示例介绍使用Scroll组件的滚动事件 onScroll 实现状态栏显隐变化。该场景多用于各种软件的首页、我的等页面中。 效果预览图 使用说明 加载完成后显示状态栏显隐变化页面&#xff0c;上下拖动屏幕&#xff0c;顶端状态栏出现显隐变化。 实现思路 在置顶位置使用sta…

文件夹秒变应用程序?别慌,数据恢复有妙招!

在日常使用电脑的过程中&#xff0c;我们有时会遇到一个令人头疼的问题&#xff1a;原本好好的文件夹突然变成了应用程序的图标&#xff0c;点击也无法正常打开。这种“文件夹变应用程序”的现象不仅让人感到困惑&#xff0c;还可能导致重要文件的丢失或损坏。那么&#xff0c;…

vite ts vue 项目提示 . Projects must list all files or use an include pattern.

vite ts vue 项目提示 . Projects must list all files or use an include pattern. 在引用一个 ts 的时候&#xff0c;提示如下&#xff1a; 需要在 tsconfig.node.json 文件中添加&#xff1a; {"compilerOptions": {"composite": true,"skipLibC…

变量命名之函数命名

变量命名: 变量命名和函数名命名 方式一:camel命名 因相骆驼脊背形象命名 大骆驼法:当变量名或函数名由一个或多个单词连接在一起的,从第一个单词首字母也大写了,后面每个单词都大写. 例子: HI_S32 HI_MPI_VI_SetDevAttr(VI_DEV ViDev,const VI_DEV_ATTR_S* pstDevAttr);HI_S…

Vue2(四):Vue监测数据的原理

一、先来看一个问题 添加一个按钮点击更新马冬梅的信息&#xff1a; <button click"gengxin">点击更新马冬梅的信息</button> methods:{gengxin(){this.person[1].name马老师,this.person[1].age50,this.person[1].sex男}} 下面这种方式就不能奏效&a…

【前端】-css的详解

&#x1f496;作者&#xff1a;小树苗渴望变成参天大树&#x1f388; &#x1f389;作者宣言&#xff1a;认真写好每一篇博客&#x1f4a4; &#x1f38a;作者gitee:gitee✨ &#x1f49e;作者专栏&#xff1a;C语言,数据结构初阶,Linux,C 动态规划算法&#x1f384; 如 果 你 …

信号(Linux)

信号 前言1. 引入2. 概念3. 初步认识ctrlc信号4. 硬件中断 一、信号的产生1. 键盘组合键2. kill命令3. 系统调用①kill②raise③ abort 4. 异常①异常产生信号②原理 5. 软件条件6. 小结 二、信号的保存1. 引入2. 原理3. 接口①信号集——sigset_t②sigprocmask③sigpending④使…

spring注解驱动系列--AOP探究一

一、AOP--动态代理 指在程序运行期间动态的将某段代码切入到指定方法指定位置进行运行的编程方式 二、使用栗子 一、导入aop模块 <dependency><groupId>org.springframework</groupId><artifactId>spring-aspects</artifactId><version>4…

虚拟机开机字体变大,进入系统后字体模糊

问题 虚拟机开机字体变大&#xff0c;进入系统后字体模糊。 原因 虚拟机配置问题。 解决办法 修改配置为如下: