当前位置:首页 > 单片机 > 单片机
[导读] 在篇文章中主要讲讲调试中断过程中获得的知识,也许会对大家有用。大家应该都知道在使用S3C2440这块芯片时,有一个十分大的问题,就是对于keil软件自带的初始化代码没有给全,在初始化代码中主要做了以

在篇文章中主要讲讲调试中断过程中获得的知识,也许会对大家有用。

大家应该都知道在使用S3C2440这块芯片时,有一个十分大的问题,就是对于keil软件自带的初始化代码没有给全,在初始化代码中主要做了以下几件事

可以看出,也就是对I/O口进行了配置,看门狗,时钟进行了初始化,同时也对内存块进行了一定的配置,是十分基本的一些初始化,其中对于中断向量表根本就没有进行建立,因此如果要用他的文件进行中断的实验,必须自己去改动初始化代码,完成他没有完成的工作,自己尝试了一下,没有成功,因此就对ADS事例程序的初始化代码进行移植(参看某大神的教程实现,)

一、ARM中断的执行方式

这里中断有两种,一种外部中断,一种内部中断,对于两种中断稍有不同,原理相似,我以外部中断进行介绍。

1)当然首先必须配置好各种寄存器,当外部中断在满足条件的时候会发生中断,这时EINTPEND这个寄存器的对应位会被写为1,其中这个寄存器可以有多位置一

2)再继续和EINTMASK这个寄存器进行操作,如果触发的中断没有被屏蔽,则可以产生,这时SRCPND对应的位置会置1,当然SRCPND这个寄存器也可能会多位被置1,仅仅表示此时有多个中断发生了。

3)查看中断模式,是普通中断还是快速中断,如果是快速中断直接进入FIQ异常,如果是普通中断还需要进行后续操作。

4)如果是普通中断,再与INTMSK进行操作,操作后,如果还有多个中断,则进入中断判优,最终对INTPND中的某一位进行置1,表示某个中断产生并相应。注意INTPND这个寄存器有且仅有一位能够被置1。

以上就是有中断来了后,整个硬件操作的过程,因此我们可以看出,其实我们有时候可以不通过中断服务的方式来进行,直接进行中断位这些标志的查询就可以了,(类似单片机),但是这样没有任何效率的提升了。

对于复位、未定义指令、软件中断、指令异常、数据异常、普通中断、快速中断,在这里我们叫他们异常

对于普通中断中具体的中断,我们才叫中断。

对于以上异常发生时,都会有一个固定的跳转地址,即异常向量表:


地址异常0x0000,0000复位0x0000,0004未定义指令0x0000,0008软件中断0x0000,000c指令异常0x0000,0010数据异常0x0000,0014保留0x0000,0018普通中断0x0000,001c快速中断


下面是当进入IRQ中断发生后,软件方面的一些操作:

1)当中断产生时,程序会自动的跳转到0x0000,0018这个地址上去,在这个地址上一般会有一个跳转指令,就可以直接跳转到异常处理函数。

2)但是对于普通中断异常中还有相当多的中断,因此对于普通中断异常跳入的异常处理函数不是一个正真的处理函数,而是一个中断偏移在此跳转的中间过程。其函数如下:


  1. ;呵呵,来了来了.好戏来了,这一段程序就是用来进行第二次查表的过程了.

  2. ;如果说第一次查表是由硬件来完成的,那这一次查表就是由软件来实现的了.

  3. ;为什么要查两次表??

  4. ;没有办法,ARM把所有的中断都归纳成一个IRQ中断异常和一个FIRQ中断异常

  5. ;第一次查表主要是查出是什么异常,可我们总要知道是这个中断异常中的什么中断呀!

  6. ;没办法了,再查一次表呗!

  7. ;===================================================================================

  8. ;//外部中断号判断,通过中断服务程序入口地址存储器的地址偏移确定

  9. ;//PC=[HandleEINT0+[INTOFFSET]]

  10. ;H|------|

  11. ;|///|

  12. ;|--isr-|====>pc

  13. ;L|--r8--|

  14. ;|--r9--|<----sp

  15. IsrIRQ

  16. subsp,sp,#4;给PC寄存器保留reservedforPC

  17. stmfdsp!,{r8-r9};把r8-r9压入栈

  18. ldrr9,=INTOFFSET;把INTOFFSET的地址装入r9INTOFFSET是一个内部的寄存器,存着中断的偏移

  19. ldrr9,[r9];I_ISR

  20. ldrr8,=HandleEINT0;这就是我们第二个中断向量表的入口的,先装入r8

  21. ;===================================================================================

  22. ;哈哈,这查表方法够好了吧,r8(入口)+index*4(别望了一条指令是4bytes的喔),

  23. ;这不就是我们要找的那一项了吗.找到了表项,下一步做什么?肯定先装入了!

  24. ;==================================================================================

  25. addr8,r8,r9,lsl#2;地址对齐,因为每个中断向量占4个字节,即isr=IvectTable+Offeset*4

  26. ldrr8,[r8];装入中断服务程序的入口

  27. strr8,[sp,#8];把入口也入栈,准备用旧招

  28. ldmfdsp!,{r8-r9,pc};施招,弹出栈,哈哈,顺便把r8弹出到PC了,跳转成功!



这段语句就实现了普通中断异常到具体中断的偏移处理。

这样就可以跳转到具体的中断处理程序中了。

对于如果用S3C2440.S这个初始化文件,肯定是没有二次查表,并且也没有建立后续具体中断的偏移地址,这样如果我们自己仅仅在S3C2440.S添加出硬件中断发生后,异常跳转,让异常跳转到一个C程序中,再在C程序中检测INTPND这个寄存器的值,根据这个置调用不同的子函数也可以实现中断。(当然在跳转过程中,从正常情况进入中断异常,需要进行模式转换,栈的保存等)

二、在ARM中执行中断时,内存的映射情况

第一部分所说的直接跳转地址,都是硬件执行时直接使用的地址,当MMU没有开启的时候,上面的地址就是物理地址,直接去实际的那块地址,但是当MMU开启后,上面的地址就是虚拟地址(开启MMU之后,所有使用的地址都应该是虚拟地址了,都会被映射到某一块对应的物理地址中去)。但是从keil forARM的工程配置

我们程序代码是从ROM1的0x3000,0000这里开始存放的,因此如果产生中断了,MMU没有开启,那硬件直接去访问0x0000,0000这里,是不可能找到我们的代码,这时候程序就跑飞了。因此为了当硬件去访问0x0000,0000时,其实访问的是ROM1的0x3000,0000,我们必须开启MMU把0x0000,0000变成一个虚拟的地址,这个虚拟地址映射的实际物理地址是0x3000,0000,这里我们就需要介绍一个函数MMU_SetMTT,源代码如下


  1. voidMMU_SetMTT(intvaddrStart,intvaddrEnd,intpaddrStart,intattr)

  2. {

  3. volatileU32*pTT;

  4. volatileinti,nSec;

  5. pTT=(U32*)_MMUTT_STARTADDRESS+(vaddrStart>>20);

  6. nSec=(vaddrEnd>>20)-(vaddrStart>>20);

  7. for(i=0;i<=nSec;i++)*pTT++=attr|(((paddrStart>>20)+i)<<20);

  8. }

在内存映射中增加MMU_SetMTT(0x00000000,0x03F00000,0x30000000,RW_CB);

或者增加MMU_SetMTT(0x00000000,0x03f00000,(int)__ENTRY,RW_CB); 因为ENTRY就是等于0x30000000;

这样在中断产生的时候就不会跑飞了,并且能按照程序代码找到对应中断入口等。

三、特别提醒

在上面两个问题解决后,顺利的把中断调试通过,但是需要注意的小细节是:

1)在每次中断发生后,记得要对中断进行清除,并且清除的过程是从内部向外部清除,

2)在对于的中断函数一定要记住挂载对应的中断向量地址上,不然同样程序会跑飞,如果没有注意这里,可能会调试很久。

四、中断向量表

这是初始化代码中用于上面二次偏移的标号,定义在RAM中


  1. ALIGN

  2. AREARamData,DATA,READWRITE

  3. ^_ISR_STARTADDRESS;_ISR_STARTADDRESS=0x33FF_FF00

  4. HandleReset#4

  5. HandleUndef#4

  6. HandleSWI#4

  7. HandlePabort#4

  8. HandleDabort#4

  9. HandleReserved#4

  10. HandleIRQ#4

  11. HandleFIQ#4

  12. ;Donotusethelabel'IntVectorTable',

  13. ;ThevalueofIntVectorTableisdifferentwiththeaddressyouthinkitmaybe.

  14. ;IntVectorTable

  15. ;@0x33FF_FF20

  16. HandleEINT0#4

  17. HandleEINT1 # 4

本站声明: 本文章由作者或相关机构授权发布,目的在于传递更多信息,并不代表本站赞同其观点,本站亦不保证或承诺内容真实性等。需要转载请联系该专栏作者,如若文章内容侵犯您的权益,请及时联系本站删除。
换一批
延伸阅读

LED驱动电源的输入包括高压工频交流(即市电)、低压直流、高压直流、低压高频交流(如电子变压器的输出)等。

关键字: 驱动电源

在工业自动化蓬勃发展的当下,工业电机作为核心动力设备,其驱动电源的性能直接关系到整个系统的稳定性和可靠性。其中,反电动势抑制与过流保护是驱动电源设计中至关重要的两个环节,集成化方案的设计成为提升电机驱动性能的关键。

关键字: 工业电机 驱动电源

LED 驱动电源作为 LED 照明系统的 “心脏”,其稳定性直接决定了整个照明设备的使用寿命。然而,在实际应用中,LED 驱动电源易损坏的问题却十分常见,不仅增加了维护成本,还影响了用户体验。要解决这一问题,需从设计、生...

关键字: 驱动电源 照明系统 散热

根据LED驱动电源的公式,电感内电流波动大小和电感值成反比,输出纹波和输出电容值成反比。所以加大电感值和输出电容值可以减小纹波。

关键字: LED 设计 驱动电源

电动汽车(EV)作为新能源汽车的重要代表,正逐渐成为全球汽车产业的重要发展方向。电动汽车的核心技术之一是电机驱动控制系统,而绝缘栅双极型晶体管(IGBT)作为电机驱动系统中的关键元件,其性能直接影响到电动汽车的动力性能和...

关键字: 电动汽车 新能源 驱动电源

在现代城市建设中,街道及停车场照明作为基础设施的重要组成部分,其质量和效率直接关系到城市的公共安全、居民生活质量和能源利用效率。随着科技的进步,高亮度白光发光二极管(LED)因其独特的优势逐渐取代传统光源,成为大功率区域...

关键字: 发光二极管 驱动电源 LED

LED通用照明设计工程师会遇到许多挑战,如功率密度、功率因数校正(PFC)、空间受限和可靠性等。

关键字: LED 驱动电源 功率因数校正

在LED照明技术日益普及的今天,LED驱动电源的电磁干扰(EMI)问题成为了一个不可忽视的挑战。电磁干扰不仅会影响LED灯具的正常工作,还可能对周围电子设备造成不利影响,甚至引发系统故障。因此,采取有效的硬件措施来解决L...

关键字: LED照明技术 电磁干扰 驱动电源

开关电源具有效率高的特性,而且开关电源的变压器体积比串联稳压型电源的要小得多,电源电路比较整洁,整机重量也有所下降,所以,现在的LED驱动电源

关键字: LED 驱动电源 开关电源

LED驱动电源是把电源供应转换为特定的电压电流以驱动LED发光的电压转换器,通常情况下:LED驱动电源的输入包括高压工频交流(即市电)、低压直流、高压直流、低压高频交流(如电子变压器的输出)等。

关键字: LED 隧道灯 驱动电源
关闭