嵌入式 中断实验.docx
《嵌入式 中断实验.docx》由会员分享,可在线阅读,更多相关《嵌入式 中断实验.docx(17页珍藏版)》请在冰豆网上搜索。
嵌入式中断实验
5.3中断实验
实验目的
1.了解中断的作用;
2.掌握嵌入式系统中断的处理流程;
3.掌握ARM中断编程。
实验内容
1.编写中断处理程序,处理外部中断;
预备知识
1.了解ADTIDE集成开发环境的基本功能;
2.了解中断的作用以及基本处理过程。
实验设备
1.硬件:
JX44B0教学实验箱、PC机;
2.软件:
PC机操作系统Windows98(2000、XP)+ADTIDE集成开发环境。
基础知识
1.中断的基本概念
CPU与外设之间传输数据的控制方式通常有三种:
查询方式、中断方式和DMA方式。
DMA方式将在后续实验中说明。
查询方式的优点是硬件开销小,使用起来比较简单。
但在此方式下,CPU要不断地查询外设的状态,当外设未准备好时,CPU就只能循环等待,不能执行其它程序,这样就浪费了CPU的大量时间,降低了CPU的利用率。
为了解决这个矛盾,通常采用中断传送方式:
即当CPU进行主程序操作时,外设的数据已存入输入端口的数据寄存器;或端口的数据输出寄存器已空,由外设通过接口电路向CPU发出中断请求信号,CPU在满足一定的条件下,暂停执行当前正在执行的主程序,转入执行相应能够进行输入/输出操作的子程序,待输入/输出操作执行完毕之后CPU再返回并继续执行原来被中断的主程序。
这样CPU就避免了把大量时间耗费在等待、查询状态信号的操作上,使其工作效率得以大大地提高。
能够向CPU发出中断请求的设备或事件称为中断源。
系统引入中断机制后,CPU与外设(甚至多个外设)处于“并行”工作状态,便于实现信息的实时处理和系统的故障处理。
中断方式的原理示意图如下所示。
图5-7中断处理示意图
1)中断响应
中断源向CPU发出中断请求,若优先级别最高,CPU在满足一定的条件下,可以中断当前程序的运行,保护好被中断的主程序的断点及现场信息。
然后,根据中断源提供的信息,找到中断服务子程序的入口地址,转去执行新的程序段,这就是中断响应。
CPU响应中断是有条件的,如内部允许中断、中断未被屏蔽、当前指令执行完等。
2)中断服务子程序
CPU响应中断以后,就会中止当前的程序,转去执行一个中断服务子程序,以完成为相应设备的服务。
中断服务子程序的一般结构如下图所示。
图5-8中断服务子程序处理流程
▼保护现场(由一系列的压栈指令完成)。
目的是为了保护那些与主程序中有冲突的寄存器,(如R0,R1,R2等),如果中断服务子程序中所使用的寄存器与主程序中所使用的寄存器等没有冲突的话,这一步骤可以省略。
▼中断处理,中断处理程序在检查到相应的中断源后,调用对应的中断处理程序完成。
▼恢复现场并返回(由一系列的出栈指令完成)。
是与保护现场对应的,但要注意数据恢复的次序,以免混乱。
由于中断服务子程序需要打断主程序的执行,因此其处理应该及时完成,较长时间的延时将导致系统性能严重下降。
2.S3C44B0X中断控制器
S3C44B0X的中断控制器包括5类寄存器:
中断控制寄存器、中断状态寄存器、中断模式寄存器、中断屏蔽寄存器和中断清除寄存器。
1)中断控制寄存器
该控制寄存器是处理器总的中断控制,包括中断模式是矢量模式还是非矢量模式,是否使能IRQ模式的中断,是否使能FIQ模式的中断,具体说明如下:
表5-3中断控制寄存器
寄存器名称
地址
读写状态
描述
复位值
INTCON
0x01E00000
R/W
中断控制寄存器
0x7
INTCON
位
描述
初始状态
保留
3
0
0
V
2
IRQ禁止/使能向量模式
0:
向量中断模式1:
非向量中断模式
1
I
1
使能CPU的IRQ中断,在使用IRQ中断之前,必须清除该位
0:
IRQ中断使能1:
IRQ中断禁止
1
F
0
使能CPU的FIQ中断,在使用FIQ中断之前,必须清除该位
0:
FIQ中断使能1:
FIQ中断禁止
1
2)中断状态寄存器
该寄存器用于检查中断来源,该寄存器是只读属性的。
表5-4中断状态寄存器
寄存器名称
地址
读写状态
描述
复位值
INTPND
0x01E00004
R
指示中断请求状态
0:
中断已被响应
1:
有中断请求
0x0
3)中断模式寄存器
用于设置相应中断的工作模式,是IRQ模式还是FIQ模式。
表5-5中断模式寄存器
寄存器名称
地址
读写状态
描述
复位值
INTMOD
0x01E00008
R/W
中断模式寄存器
0:
IRQ模式
1:
FIQ模式
0x0
4)中断屏蔽寄存器
表5-6中断屏蔽寄存器
寄存器名称
地址
读写状态
描述
复位值
INTMSK
0x01E0000C
R/W
确定哪一个中断源被屏蔽,屏蔽的中断源将不引发中断
0:
中断服务有效
1:
中断服务屏蔽
0x7FFFFFF
5)中断清除寄存器
中断处理之后需要清除相应的标志位,中断清除寄存器说明如下:
表5-7中断清除寄存器
寄存器名称
地址
读写状态
描述
复位值
I_ISPC
0x01E00024
W
IRQ中断请求清0寄存器
未定义
F_ISPC
0x01E0003C
W
FIQ中断请求清0寄存器
未定义
4.JX44B0中断处理
S3C44B0X处理器的中断处理与其他CPU的处理模式基本上是一致的,只是由于它引入了几种不同的处理器模式,使中断处理变得更加容易。
其典型的步骤如下:
1)保存现场:
当系统出现中断时,处理器首先要做的就是保存现场,这一过程包括:
保存当前的PC值到lr中,保存当前的程序运行状态到spsr中。
值得注意的就是由于ARM7采用3级流水线结构,此时的PC值实际上等于当前指令地址加上8(ARM指令时),所以返回时还需要将保存的PC值减4;
2)模式切换:
当处理器完成现场保护后,就进入中断模式,并将PC值置为一个固定的值0X00000018,这也就是IRQ模式的中断入口地址。
在中断模式下,有两个独立的寄存器R13、R14,这样可以便于中断程序使用自己特有的堆栈。
但这样随之而来产生一个问题,就是中断处理时堆栈溢出保护的问题,需要我们认真地估计堆栈的大小,同时在中断处理时也要尽量减少函数调用的层次,否则将产生一些不可预知的错误;
3)获取中断源:
所有的IRQ中断都从0X00000018开始执行,通常在该地址处放一条跳转指令,进一步跳到我们的中断程序中;
4)处理中断:
在中断程序中需要进一步获取中断源,即谁引发了该中断,然后通过查表获取相应中断的处理程序入口,并调用对应的函数;
5)中断返回,恢复现场:
在返回时需要恢复处理器模式,包括恢复中断处理用到的所有寄存器、恢复被中断的程序运行状态到CPSR,并跳转到被中断的主程序。
下图为JX44B0教学实验系统中处理外部中断0的流程:
图5-8JX44B0中断处理示意图
中断的入口代码(汇编代码):
0X00000018:
LDRpc,=0X0C000020
……
0X0C000020:
bHandlerIRQ
HandlerIRQ:
subsp,sp,#4/*为中断分发例程入口地址预留栈空间*/
stmfdsp!
{r0}/*保存R0*/
ldrr0,=HandleIRQ/*将中断分发例程入口地址指针保存到R0中*/
ldrr0,[r0]/*将中断分发例程入口地址保存到R0中*/
strr0,[sp,#4]/*将中断分发例程入口地址保存到预留的堆栈空间*/
ldmfdsp!
{r0,pc}/*将R0和中断分发例程入口地址出栈,这条指令也*/
/*实现了一个跳转*/
上述代码实际上就是一个三级跳,即从FLASH中跳到了RAM的中断入口,然后又从中断入口跳到中断分发例程入口。
在此我们有一个前提条件,即必须在HandleIRQ地址处保存正确的分发例程入口地址,如使用下面代码后IsrIRQ就是中断分发例程:
ldrr0,=HandleIRQ
ldrr1,=IsrIRQ
strr1,[r0]
中断分发例程可以采用汇编语言和C语言两种格式编写,下面将分别列出这两种方式。
1)用汇编代码编写的中断分发例程:
IsrIRQ:
/*usingI_ISPRregister.*/
sublr,lr,#4
stmfdsp!
{lr}/*保存中断返回的PC值*/
stmfdsp!
{r0-r4}/*备份寄存器R0-R4*/
subsp,sp,#4/*为PC预留栈空间*/
stmfdsp!
{r8-r9}/*备份寄存器R8-R9*/
ldrr9,=I_ISPR/*读取中断状态*/
ldrr9,[r9]
cmpr9,#0x0/*检查中断状态*/
beqi2
movr8,#0x0/*R8保存中断表的偏移*/
i0:
/*逐位检查中断状态*/
movsr9,r9,lsr#1
bcsi1/*如果该位等于1,则处理这一中断*/
addr8,r8,#4/*修改当前的中断偏移*/
bi0/*处理下一比特*/
i1:
ldrr9,=HandleADC/*HandleADC位于中断向量表起始位置,我们将该地址用作是中断向量表的基地址*/
addr9,r9,r8/*计算入口地址指针:
中断基地址加上偏移*/
ldrr9,[r9]/*从地址向量表中获取入口地址*/
strr9,[sp,#8]/*将入口地址保存到堆栈,并移动堆栈指针*/
movlr,pc/*保存当前PC*/
ldmfdsp!
{r8-r9,pc}/*调用中断例程*/
ldmfdsp!
{r0-r4,pc}^/*中断返回,并恢复中断前的处理器模式*/
i2:
ldmfdsp!
{r8-r9}/*如果当前没有任何中断,直接返回*/
addsp,sp,#4/*移动堆栈指针,该空间由第4句指令预留*/
ldmfdsp!
{r0-r4,pc}^/*中断返回,并恢复中断前的处理器模式*/
2)用C代码编写的中断分发例程:
如果采用GNU编译器,需要将该函数定义为中断类型,使用关键字:
__attribute__((interrupt("IRQ")))。
如下所示代码为C语言的IsrIRQ实现:
typedef(*ISR_ROUTINE_ENTRY)(void);
voidIsrIRQ()__attribute__((interrupt("IRQ")));
voidIsrIRQ()
{
intcount=0;
unsignedintisr_pending;
unsignedintisr_mask=0x00000001;
unsignedintisr_mask_set=rINTMSK;/*读取中断掩码*/
ISR_ROUTINE_ENTRYisr_routine_entry=(ISR_ROUTINE_ENTRY)0x0;
isr_pending=(rINTPND&~isr_mask_set);/*读取中断状态*/
/*查表*/
while(isr_mask)
{
if(isr_pending&isr_mask)
{
/*找到中断源,获取中断例程入口地址*/
isr_routine_entry=(ISR_ROUTINE_ENTRY)(*(int*)(HandleADC+count));
break;
}
count+=4;
isr_mask<<=1;
}
/*调用中断服务例程*/
if(isr_routine_entry)(*isr_routine_entry)();
}
中断处理例程(该函数无需定义为中断类型)
voidEINT0_Isr()
{
rI_ISPC=BIT_EINT0;/*清除中断标志*/
……
}
中断向量表中各个中断的偏移:
表5-8中断向量表中各个中断的偏移
中断源
向量表的偏移
ADC(AD转换中断)
0X20
RTC(实时时钟中断)
0X24
……
4.实验参考代码及说明
/*包含文件*/
#include"44b.h"
typedef(*ISR_ROUTINE_ENTRY)(void);
voiddelay();
voidext0_int_isr(void);
voidIsrIRQ()__attribute__((interrupt("IRQ")));/*声明为中断服务里例程*/
intext0_count=0;
intdither_count=0;
/*****************************************************************************
//Functionname:
IsrIRQ
//Description:
非矢量方式下中断的查表处理
//中断地址表位于0x0c7fff00开始的256字节
//Returntype:
void
//Argument:
void
*****************************************************************************/
voidIsrIRQ()
{
intcount=0;
unsignedintisr_pending;
unsignedintisr_mask=0x00000001;
unsignedintisr_mask_set=rINTMSK;
ISR_ROUTINE_ENTRYisr_routine_entry=(ISR_ROUTINE_ENTRY)0x0;
__asm__(
"STMFDSP!
{r1,r4-r8}@SAVEr1,r4-r10\n"
"nop\n");
isr_pending=(rINTPND&~isr_mask_set);
while(isr_mask)
{
if(isr_pending&isr_mask)
{
isr_routine_entry=(ISR_ROUTINE_ENTRY)(*(int*)(HandleADC+count));
break;
}
count+=4;
isr_mask<<=1;
}
if(isr_routine_entry)(*isr_routine_entry)();
__asm__(
"LDMFDSP!
{r1,r4-r8}@RESTOREr1,r4-r10\n"
"nop\n");
}
/*****************************************************************************
//Functionname:
init_interrupt_handler
//Description:
非矢量方式下中断向量表初始化处理
//Returntype:
void
//Argument:
irq_handler
//中断处理函数入口
*****************************************************************************/
voidinit_interrupt_handler(unsignedintirq_handler)
{
inti;
rINTPND=0x00000000;/*清除所有未决的中断*/
rI_ISPC=0x03FFFFFF;
for(i=0;i<256;i+=4)/*清除中断表*/
{
*(unsignedint*)(_ISR_STARTADDRESS+i)=0;
}
*(unsignedint*)(HandleIRQ)=irq_handler;/*设置IRQ模式处理函数*/
}
/*****************************************************************************
//Functionname:
install_isr_handler
//Description:
非矢量方式下中断向量的安装
//Returntype:
void
//Argument:
irq_no,中断号
//irq_routine,中断处理函数地址
*****************************************************************************/
voidinstall_isr_handler(intirq_no,void*irq_routine)
{
*(unsignedint*)(irq_no)=(unsignedint)irq_routine;
}
/*****************************************************************************
//Functionname:
Main
//Description:
中断测试程序主函数
//实现外部中断0的测试,引发中断后在中断处理函数中显示跑马灯
//Returntype:
int
//Argument:
void
*****************************************************************************/
intMain()
{
intold_index;
rINTCON=0x7;/*Non-vect,IRQdisable,FIQdisable*/
init_interrupt_handler((unsignedint)IsrIRQ);
install_isr_handler(HandleEINT0,(void*)ext0_int_isr);
rINTMOD=0x0;/*设置所有中断为IRQ模式*/
rINTMSK=(0x07ffffff&~(BIT_GLOBAL|BIT_EINT0));/*使能外部中断0*/
rINTCON=0x5;/*Non-vect,IRQenable,FIQdisable*/
ext0_count=0;
dither_count=0;
while
(1)
{
delay();
dither_count++;
}
}
/*****************************************************************************
//Functionname:
ext0_int_isr
//Description:
外部中断0处理函数
//Returntype:
int
//Argument:
void
*****************************************************************************/
voidext0_int_isr(void)
{
rI_ISPC=BIT_EINT0;/*清除外部中断0*/
if(dither_count>5)
{
dither_count=0;
if(ext0_count&1)/*交替显示发光二极管*/
*(unsignedchar*)0x2000000=0xff;
else
*(unsignedchar*)0x2000000=0x0f;
ext0_count++;
}
}
voiddelay()
{
intindex=0;
for(index=0;index<100000;index++);
}
实验步骤
1.参照模板工程interrupt(modules\interrupt\interrupt.apj),新建一个工程interrupt,添加相应的文件,并修改interrupt的工程设置;
2.创建interrupt.c并加入到工程interrupt中;
3.编写中断分发例程IsrIRQ;
4.注册外部中断0处理函数ext0_int_isr;
5.实现外部中断0处理函数ext0_int_isr,在其中实现LED开关功能;
6.编译interrupt;
7.下载程序并运行,按下按键EXTINT0将引发一次外部中断,并在中断处理函数中闪灯。
实验报告要求
1.中断处理的主要步骤有哪些?
试说明每一步的主要工作。
2.编写程序处理定时中断;
3.简述S3C44B0X非矢量中断的处理步骤。