微波EDA网,见证研发工程师的成长!
首页 > 硬件设计 > 嵌入式设计 > ARM·中断控制器

ARM·中断控制器

时间:11-24 来源:互联网 点击:

之前在基础篇里面看过中断概念的视频,不过因为介绍的十分繁琐,效果不是很好。今天看了高级开发篇,决定把中断的内容简化一些。

我觉得可以把这篇笔记分成2个区,寄存器区和代码区。寄存器区方便以后查看,代码区方便对中断控制的分析

@******************************************************************************

@ File:head.S

@ 功能:初始化,设置中断模式、管理模式的栈,设置好中断处理函数

@******************************************************************************

.extern main @extern 声明引用main函数

.text

.global _start

_start:

@******************************************************************************

@ 中断向量,本程序中,除Reset和HandleIRQ外,其它异常都没有使用

@******************************************************************************

b Reset

@ 0x04: 未定义指令中止模式的向量地址

HandleUndef:

b HandleUndef

@ 0x08: 管理模式的向量地址,通过SWI指令进入此模式

HandleSWI:

b HandleSWI

@ 0x0c: 指令预取终止导致的异常的向量地址

HandlePrefetchAbort:

b HandlePrefetchAbort

@ 0x10: 数据访问终止导致的异常的向量地址

HandleDataAbort:

b HandleDataAbort

@ 0x14: 保留

HandleNotUsed:

b HandleNotUsed

@ 0x18: 中断模式的向量地址

b HandleIRQ

@ 0x1c: 快中断模式的向量地址

HandleFIQ:

b HandleFIQ

Reset:

ldr sp, =4096 @ 设置栈指针,以下都是C函数,调用前需要设好栈

bl disable_watch_dog @ 关闭WATCHDOG,否则CPU会不断重启

msr cpsr_c, #0xd2 @ 进入中断模式

ldr sp, =3072 @ 设置中断模式栈指针

msr cpsr_c, #0xd3 @ 进入管理模式

ldr sp, =4096 @ 设置管理模式栈指针,

@ 其实复位之后,CPU就处于管理模式,

@ 前面的“ldr sp, =4096”完成同样的功能,此句可省略

bl init_led @ 初始化LED的GPIO管脚

bl init_irq @ 调用中断初始化函数,在init.c中

msr cpsr_c, #0x5f @ 设置I-bit=0,开IRQ中断

ldr lr, =halt_loop @ 设置返回地址

ldr pc, =main @ 调用main函数

halt_loop:

b halt_loop

HandleIRQ:

sub lr, lr, #4 @ 计算返回地址

stmdb sp!, { r0-r12,lr } @ 保存使用到的寄存器

@ 注意,此时的sp是中断模式的sp

@ 初始值是上面设置的3072

ldr lr, =int_return @ 设置调用ISR即EINT_Handle函数后的返回地址

ldr pc, =EINT_Handle @ 调用中断服务函数,在interrupt.c中

int_return:

ldmia sp!, { r0-r12,pc }^ @ 中断返回, ^表示将spsr的值复制到cpsr

注意:

(1)异常向量表(这里只用到了reset和IRQ异常)和各工作模式的CPRS



(2)

msr cpsr_c, #0xd2 (11010010) @ 进入中断模式

ldr sp, =3072 @ 设置中断模式栈指针

msr cpsr_c, #0xd3(11010011) @ 进入管理模式

ldr sp, =4096 @ 设置管理模式栈指针,

@ 其实复位之后,CPU就处于管理模式,

@ 前面的“ldr sp, =4096”完成同样的功能,此句可省略

bl init_led @ 初始化LED的GPIO管脚

bl init_irq @ 调用中断初始化函数,在init.c中

msr cpsr_c, #0x53(01001110) @ 设置I-bit=0,开IRQ中断

1.在管理模式允许中断,在中断模式屏蔽中断,防止中断嵌套;且在管理模式用来Thumb,可以对表查看,做了解。

2.在每一中异常下都必须设置堆栈指针,便于调用C函数,因为pc在每种状态下都是独立的。

3.中断初始化的同时,也顺便将硬件GPIO管脚初始化,把中断函数初始化 ;

#include "s3c24xx.h"

#define GPF4_out (1<(4*2))

#define GPF5_out (1<(5*2))

#define GPF6_out (1<(6*2))

#define GPF4_msk (3<(4*2))

#define GPF5_msk (3<(5*2))

#define GPF6_msk (3<(6*2))

#define GPF0_eint (0x2<(0*2))

#define GPF2_eint (0x2<(2*2))

#define GPG3_eint (0x2<(3*2))

#define GPF0_msk (3<(0*2))

#define GPF2_msk (3<(2*2))

#define GPG3_msk (3<(3*2))

void disable_watch_dog(void)

{

WTCON = 0; // 关闭WATCHDOG很简单,往这个寄存器写0即可

}

void init_led(void)

{

// LED1,LED2,LED4对应的3根引脚设为输出

GPFCON &= ~(GPF4_msk | GPF5_msk | GPF6_msk);

GPFCON |= GPF4_out | GPF5_out | GPF6_out;

}

void init_irq( )

{

// S2,S3对应的2根引脚设为中断引脚 EINT0,ENT2

GPFCON &= ~(GPF0_msk | GPF2_msk);

Copyright © 2017-2020 微波EDA网 版权所有

网站地图

Top