中止分为硬件中止和软件中止SWI
而硬件中止有内部中止,即中止操控器中所列出的那些中止。而外部中止在中止操控器里列的是EINT。
总共有60个中止源,而在中止操控器中列出32个,还有一些子中止在子中止操控器里列出。这些操控器咱们都是经过寄存器的某一位来操控。
中止发生之后的流程:
1把下一条指令的地址放入LR,不过PC是5级流水线,所以LR的值要减去4才是正真的下一条指令的地址。
2把cpsr复制到中止形式的spsr
3pc设置到0x18的方位,这个方位是中止向量表,他指引程序跳转到中止处理程序。而中止处理程序的作用是1:保存上下文2:持续跳转到中止服务程序。
4退出的时分要康复上下文,最重要的是把中止的spsr复制到cpsr
以下介绍下用到的寄存器:
1 SRCPND 它只需发送一次中止,该相应位就变成1。即便那个中止源被屏蔽了,这个位仍是置1.
2 SUBSRCPND 子中止源未决寄存器。
3 INTMSK 中止源屏蔽
4 INTSUBMSK 子中止源屏蔽
5 INTOFFSET 读这个寄存器 便利读出是哪个中止,用10进制表明。
以下来做个比如:
在主程序中小灯一向闪,然后串口接受到数据,发生中止,然后回显数据。
这个crt0.s 是文件的进口,该程序放在0x0的方位,所以一开始进入的是中止向量表
.extern main
.extern EINT_Handle
.externinit_irq
.text
.global _start
_start:
bReset
HandleUndef:
bHandleUndef
HandleSWI:
bHandleSWI
HandlePrefetchAbort:
bHandlePrefetchAbort
HandleDataAbort:
bHandleDataAbort
HandleNotUsed:
bHandleNotUsed
bHandleIRQ@0x18
HandleFIQ:
bHandleFIQ
Reset:
ldr r0, =0x53000000 @ WATCHDOG close
mov r1, #0x0
str r1, [r0]
msr cpsr_c, #0xd2 @进入中止形式
ldr sp, =3072 @设置中止形式的sp
msr cpsr_c, #0xd3 @进入svc形式
ldr sp, =4096
bl init_irq
msr cpsr_c, #0x53 @把svc的i方位0
ldr lr, =halt_loop @设置回来地址
ldr pc, =main @ 进入主程序
HandleIRQ:@中止处理程序
sub lr, lr, #4
stmdb sp!, { r0-r12,lr } @ 保存环境
ldr lr, =int_return
ldr pc, =ISR_Handle @进入中止服务程序
int_return:
ldmia sp!, { r0-r12,pc }^ @康复环境
halt_loop:
bhalt_loop
这个是程序的主结构。
以下是中止服务程序int.c
#include “addr.h”
void init_irq()
{
INTMSK = ~(1<<28);
INTMOD = 0;
INTSUBMSK = 0xfffffffe;
}
void EINT_Handle()
{
unsigned long oft = INTOFFSET;
if(oft == 28){
if(SUBSRCPND & 1){
UTXH0 = URXH0;//这儿RX里边的数据必定要读走,否则中止有问题。否则会依据RX interpute type 挑选形式的不同有不同的作用。
SUBSRCPND |= 1;//这儿清中止,要置1,清中止有必定次序。从src子到src再到int
SRCPND |= 1<
}
}
以下是main主程序:首要初始化灯和串口,履行小灯闪
#include”addr.h”
#define UART_CLK 50000000
#define UART_BAUD_RATE 115200
#define UART_BRD (int)(UART_CLK/(UART_BAUD_RATE *16))-1
void wait(int time)
{
do{
time–;
}
while(time>0);
}
void init_uart()
{
GPHCON |=0xa0;
GPHUP = 0x0c;
ULCON0 = 0x3;
UCON0 = 0x5+ (1<<8);
UFCON0 = 0;
UMCON0 = 0;
UBRDIV0 = UART_BRD;
}
void init_led()
{
GPECON = GPE12_out|GPE13_out;
}
void uart_write(char *data)
{
while (*data != \0) {
while (!(UTRSTAT0 & 0x4));
UTXH0 = *data;
data++;
}
}
int main()
{
init_uart();
init_led();
uart_write(“This is in main “);
while(1){
GPEDAT = 0;
wait(300000);
GPEDAT = (3<<12);
wait(300000);
}
return 0;
}
这儿省掉了addr.h makefile int.lds文件。
以下再也一个外部中止的比如,即用到EINT这几位。咱们经过开发板上的6个按键 上下左右 ab 按下,经过串口输出key up key down。
crt0.s仍是那个文件
但中止服务程序 和 中止初始化的东西不同了
#include “addr.h”
void init_irq()
{
INTMOD = 0;
GPFCON = 0xaa;//初始化6个按键的引脚到 INT形式
GPGCON = 0xa;
EXTINT0 = 0x4444;//设为上跳沿,触发中止
EXTINT1 = 0x44;
INTMSK = ~(0x2f); //设置中止
EINTMASK = ~(0x30f);//设置子中止
}
void EINT_Handle()
{
unsigned long oft = INTOFFSET;//有4个按键用EINT0 EINT1 EINT2 EINT3
unsigned long val = EINTPEND; //有2个按键是共用EINT5,所以在这个寄存器里能区别它们
switch (oft)
{
case 0:
uart_write(“key b\r\n”);break;
case 1:
uart_write(“key a\r\n”);break;
case 2:
uart_write(“key down\r\n”);break;
case 3:
uart_write(“key up\r\n”);break;
case 5:
if(val == 0x100){
uart_write(“key right\r\n”);break;
}
if(val == 0x200){
uart_write(“key left\r\n”);break;
}
default:
uart_write(“Unknown interrupt!\r\n”);
break;
}
if((oft >=0) && (oft<=3)) } 其他都相同。
EINTPEND |= 1<
EINTPEND |= val;
SRCPND |= (1<