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.mfbz.cn/a/449105.html

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

相关文章

(4)可执行文件

我们把.o文件链接起来得到可执行文件&#xff0c;然后一开始没有指定函数执行入口&#xff0c;连接器显示如下 这时候我们看最终的可执行文件会发现这个位置是main。也就是说连接器自动帮我们把入口识别为main 所以我们重新用-e main来指定连接器入口为main&#xff0c;再看看 …

李彦宏:程序员职业将不复存在,会说话就能当程序员;ChatGPT 日耗电超 50 万度丨 RTE 开发者日报 Vol.161

开发者朋友们大家好&#xff1a; 这里是 「RTE 开发者日报」 &#xff0c;每天和大家一起看新闻、聊八卦。我们的社区编辑团队会整理分享 RTE &#xff08;Real Time Engagement&#xff09; 领域内「有话题的新闻」、「有态度的观点」、「有意思的数据」、「有思考的文章」、…

springcloud2022 feign超时时间配置

spring:application:name: order-webcloud:openfeign:client:config:default:connectTimeout: 60000readTimeout: 60000 默认connection10秒,readTimeout 60秒

nacos服务中心和注册中心

前言 Nacos 是阿里巴巴开源的一个更易于构建云原生应用的动态服务发现、配置管理和服务管理 平台。它是使用 java 编写。所以也是需要依赖 java 环境的&#xff1a; Java环境变量配置详解-CSDN博客 Nacos 文档地址&#xff1a; https://nacos.io/zh-cn/docs/quick-star…

力扣刷题日志-Day2 (力扣151、43、14)

151. 反转字符串中的单词 给你一个字符串 s &#xff0c;请你反转字符串中 单词 的顺序。 单词 是由非空格字符组成的字符串。s 中使用至少一个空格将字符串中的 单词 分隔开 思路&#xff1a;根据题目大意&#xff0c;空格之间的就是一个单词&#xff0c;所以我们需要利用…

JMeter 简介及安装详细教程(全网独家)

JMeter 简介 全名为 Apache JMeter JMeter 是一个软件&#xff0c;使负载测试或业绩为导向的业务&#xff08;功能&#xff09;测试不同的协议或技术。 它是 Apache 软件基金会的Stefano Mazzocchi JMeter 最初开发的。 它主要对 Apache JServ&#xff08;现在称为如 Apache T…

Git版本工具学习

目录 版本控制git配置工作区域文件状态git对象模型基础命令.gitignore忽略文件IDEA集成Git 版本控制 本地版本控制&#xff1a;在本地记录每一次版本更新。 集中版本控制&#xff1a;版本数据都保存在单一服务器&#xff0c;不联网就看不到版本信息。SVN 分布式版本控制&…

flink的分组聚合、over聚合、窗口聚合对比

【背景】 flink有几种聚合&#xff0c;使用上是有一些不同&#xff0c;需要加以区分&#xff1a; 分组聚合&#xff1a;group agg over聚合&#xff1a;over agg 窗口聚合&#xff1a;window agg 省流版&#xff1a; 触发计算时机 结果流类型 状态大小 分组聚合group ag…

MongoDB的count() 统计文档数量非常慢

在MongoDB中&#xff0c;count()函数用于统计文档的数量。但是count()函数通常不会使用索引来计算文档数量&#xff0c;而是扫描集合中的文档来计数。当数据量较大的时候&#xff0c;就不适合使用了。 解决方案&#xff1a; 1、使用聚合框架&#xff08;aggregation framewor…

EasyNVR级联EasyCVR,在EasyCVR播放视频会导致EasyNVR崩溃的原因排查与解决

视频综合管理平台EasyCVR视频监控系统支持多协议接入、兼容多类型设备&#xff0c;平台可以将监控区域内所有部署的监控设备进行统一接入与集中汇聚管理&#xff0c;实现对监控区域的实时视频监控、录像与存储、设备管理、云台控制、语音对讲、级联共享等&#xff0c;在监控中心…

从零搭建NodeJS项目(小白教程)

这边文章将介绍如何从零开始创建一个基于Express框架的Node.js项目。Express是一个快速、无拘束且极简的Node.js web应用框架&#xff0c;它提供了一系列强大的功能&#xff0c;使得web开发变得更加高效。 目录 1. 环境准备 2. 安装Express脚手架 3. 创建项目 4. 初始化项…

Clearview X for mac v3.5.0 电子书阅读器 兼容 M1/M2/M3

应用介绍 Clearview X 是 macOS 上的一款简洁易用且美观大方的电子书阅读器。直观好用的图书管理功能&#xff0c;支持 PDF, Epub, MOBI, CHM, FB2, CBR, CBZ 等流行的电子书格式&#xff0c;可以方便地添加注解&#xff0c;插入书签&#xff0c;及迅速的搜索查找。支持在不同…

git init 执行后发生了什么?

首先在磁盘中创建一个新目录 Git&#xff0c;进入该目录后执行 git init 初始化。这个时候目录下会创建一个隐藏目录 ./git&#xff0c;这个./git 目录叫做 Git 版本库或者仓库 $ git init Initialized empty Git repository in D:/Git/.git/ 在讲解.git 目录内容前&#xff0…

【C++】关联式容器

目录 前言&#xff1a; 一&#xff0c;set容器 二&#xff0c;multiset容器 三&#xff0c;map容器 四&#xff0c;multimap容器 前言&#xff1a; 在C中&#xff0c;STL中的部分容器&#xff0c;比如&#xff1a;vector、list、deque、 forward_list(C11)等&#xff0c;这…

第五届国际信息技术与教育技术大会(ITET 2024)即将召开!

2024年第五届国际信息技术与教育技术大会&#xff08;ITET 2024&#xff09;将于5月10-12日在日本鸟取举行。本届会议由日本鸟取大学主办&#xff0c;冈山大学、湘南工业大学、名古屋工业大学、山口大学等提供技术支持。ITET 2024旨在探讨计算机领域的创新发展在教育环境中所带…

javase day03笔记

第三天课堂笔记 idea的使用★★★ 创建空工程创建模块创建包&#xff1a;package创建类idea的设置 file -> settings 快捷键 shift &#xff0b; 回车 &#xff1a; 光标切换到下一行psvm回车&#xff1a; main方法main回车&#xff1a;main方法sout回车&#xff1a;输…

快速入门:JS对象/BOM/DOM/事件监听

本贴介绍JS相对进阶的知识&#xff0c;对于JavaScript的基础语法&#xff0c;本文不再赘述~ 一.JavaScript对象 1.Array数组对象 定义 var arr new Array(1,2,3); var arr[1,2,3]; 访问 arr[0]1; Js数组类似Java中的集合&#xff0c;长度&#xff0c;类型都可以改变。 如…

Web端功能测试方法最有作用的5个点

对于web测试&#xff0c;较之其他软件测试又有所不同&#xff0c;这是细节的不同&#xff0c;这个不同需要我们在不停的测试中去总结的。 web测试正式测试之前&#xff0c;应先确定如何开展测试&#xff0c;不可盲目的测试&#xff0c;讲究方法才能行之有效的提高我们的效…

Linux——文件缓冲区与模拟实现stdio.h

前言 我们学习了系统层面上的文件操作&#xff0c;也明白了重定向的基本原理&#xff0c;在重定向中&#xff0c;我们使用fflush(stdout)刷新了缓冲区&#xff0c;当时我们仅仅知道重定向需要刷新缓冲区&#xff0c;但是不知道其所以然&#xff0c;今天我们来见识一下。 一、…

框架学了不会用?四小时做完一个完整的前后端分离demo(SpringBoot+Vue)

四小时做完一个完整的前后端分离demo&#xff08;SpringBootVue&#xff09; 分享一个看到的还不错的小项目&#xff0c;非常适合刚学完框架但是没有太多动手机会的的学生党用来练手。 优势 手把手写代码&#xff0c;有教学视频免费&#xff0c;有源代码项目周期短 视频教程…
最新文章