当前位置:首页 > 单片机 > 单片机
[导读] 在篇文章中主要讲讲调试中断过程中获得的知识,也许会对大家有用。大家应该都知道在使用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

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

近日,Arm推出了Arm® Ethos™-U85神经网络处理器(NPU)和Arm Corstone™-320物联网参考设计平台,旨在满足海量的数据处理和大规模计算,加速推进边缘AI的发展进程。

关键字: ARM

为了赶超云计算市场上的竞争对手,谷歌正试图通过定制的Arm服务器芯片降低云计算服务成本。

关键字: 谷歌 ARM 定制芯片

嵌入式开发作为一种专业且技术密集型的领域,涵盖了从硬件底层驱动、中间件到应用层软件开发等多个层面的工作,其所需的工具种类繁多,各有针对性,旨在提升开发效率、保证代码质量以及简化调试过程。

关键字: 嵌入式开发 keil

嵌入式开发作为一个融合了计算机软硬件和系统工程的综合性领域,其成功与否往往取决于三个核心要素的有效整合与协调。这三个要素分别是:硬件平台的选择与设计、软件开发及其优化、以及系统级的设计与集成。深入理解并熟练掌握这三个方面...

关键字: 嵌入式开发 ARM

随着汽车软件数量爆发式的增长,整个行业都需要重新思考汽车产品的开发流程。为此,Arm推出了丰富的硬件IP、新的系统IP,以及全新的汽车计算与计算子系统产品路线图,旨在为各种汽车应用实现性能、功能安全、可扩展等方面的支持。

关键字: ARM 汽车电子

知名移动芯片设计公司ARM最近迈出重要一步,它正式推出汽车芯片设计。ARM推出的芯片设计方案名叫Neoverse,随同芯片一起推出的还有面向汽车制造商、汽车供应商的新系统。

关键字: ARM 汽车芯片 芯片

随着通用人工智能的发展,数据中心的计算需求逐步提高。针对多模态数据、大模型的推理和训练需要更高的算力支持,而随着算力提升与之而来的还需更关注在功耗方面的优化。对于头部云计算和服务厂商而言,针对专门用例提高每瓦性能变得至关...

关键字: ARM 服务器 AI Neoverse CSS

一直以来,riscv架构都是大家的关注焦点之一。因此针对大家的兴趣点所在,小编将为大家带来riscv架构的相关介绍,详细内容请看下文。

关键字: riscv ARM riscv架构

最新消息报道,知情人士透露Arm近日裁掉了中国70多名软件工程师,并会将部分职位转移到中国以外的地区。Arm通过“全球服务”部门已经将支持其中国客户的工作外包给安谋科技,该部门曾经拥有约200名员工。

关键字: ARM 裁员

凭借着在个人计算机领域的广泛应用打下的坚实基础,X86自始至终统治着整个服务器生态。而这并不是业界希望看到的,因此Arm服务器被给予厚望。业界期盼Arm能够带来新的服务器CPU替代:打破一个同质化的数据中心架构,实现更高...

关键字: ARM 服务器 AI
关闭
关闭