LPC2478中断控制器以及串口详解
扫描二维码
随时随地手机看文章
LPC2478的中断系统
LPC2478使用的是ARM PrimeCell向量中断控制器,一共支持32个中断向量,处于AHB空间便于系统快速访问,在中断向量的硬件优先级上还有一层可以用户自己设计的软件优先级,
由于是ARM7内核,所以,2478的中断有两种,分别是FIQ中断和IRQ中断,IRQ有多个中断通道,FIQ的中断通道却只有一个,中断系统框图如下
当FIQ中断进入时,会经历多个中断想或的过程,得到的中断状态不管是不是使能了中断,都会存放至RAWINT里面,然后判断中断使能,被使能的中断写入到FIQSTATUS,并向系统发出快速中断请求,系统响应中断的模式则是自动读取FIQSTATUS判断哪个中断发生,并直接跳转到FIQ处理程序,我们可以看到,FIQ是没有中断优先级的,他只需要判断最高位为1是哪一位就能快速跳转到FIQ处理程序了
当IRQ中断发生的时候,首先会有一个优先级鉴别,只有软件优先级最高的才能进入系统,当有多个中断进入系统的时候,这多个中断优先级相同,则会经过一次硬件优先级鉴别,选出优先级最高的那个进入系统,系统直接到相应的中断处理程序处理
相关寄存器如下
在实际的硬件中断没发生的情况下又想调用对应的中断处理程序,这需要用这个软件中断
移除有软件中断引发的中断
这个应该叫做中断事件标志位,当有中断发生不管是什么中断不管是否使能对应位都要写1
中断使能
清除中断使能
选择中断的模式,可以看到,初始化的时候都是IRQ中断
IRQ中断状态,并不是说这是中断标志,只是一种状态表示
FIQ的中断状态, 并不是说这是中断标志,只是一种状态表示
有32个这样的寄存器,指定中断的服务程序的地址,系统发生中断之后,会自动调用
中断的优先级,四位优先级,有16个优先级,一共有32个寄存器和中断向量一一映射
当前正在响应的中断优先级的地址
将VIC中断寄存器保护起来,防止用户在用户模式下修改,只能在特权模式下修改
使用VIC向量中断寄存器一般分为以下几步
1.关闭对应中断使能以及对应的软件中断
2.清除中断状态
3.选择中断模式,设置中断级别,设置中断地址,使能中断
4.注意将相应的中断优先级屏蔽进行处理
中断处理时
1.清除中断标志
2.中断处理
3.完成之后将当前处理函数指针清零
具体代码如下
#ifndef__VIC_H_
#define__VIC_H_
#include"common.h"
#include"lpc24xx.h"
voidInitVic(void);
#endif
#include"vic.h"
voidInitVic(void)
{
//禁止所有中断
VICIntEnClr=0xffffffff;
//设置当前中断指向为0
VICVectAddr=0;
//所有中断都是IRQ模式
VICIntSelect=0;
//软件中断全部清除
VICSoftIntClr=0xffffffff;
}
LPC2478串口使用
Lpc2478有三个串口,是通用的异步串口,分别是串口0 2 3,都带有FIFO,深度可控制,1 2 4 8 均可触发,同时内置波特率发生器,并有小数波特率分频器用于实现自动波特率
一般而言,对于串口的控制分为设置格式,设置fifo,设置波特率,配置中断,打开串口
主要有以下寄存器
分别用来接收数据,发送数据,选择波特率分频,其中,波特率分频系数的计算需要小数波特率的参与,如下
计算公式如下
自动波特率一般用于测量基于AT指令的协议,用的不多,
配置串口的过程如下
选择格式,数据位长度,停止位,校验位,同时,里面的除数访问锁关闭,否则无法计算波特率
配置相应的中断使能,
配置串口的FIFO
配置波特率,除数寄存器前面已经描述
设置发送使能
具体查看代码
#ifndef__DEBUGSERIAL_H_
#define__DEBUGSERIAL_H_
#include"lpc24xx.h"
#include"common.h"
#include"clock.h"
#include"stdio.h"
#defineSERIALS_BUFFER_MAX_LENGTH0X3FF
#defineSERIAL_RECV_PACKAGE(1<<15)
#defineSERIAL_RECV_LENGTH(serialRecvStatus&0X3FF)
#defineSERIAL_RECV_CLEAR(serialRecvStatus=0)
#defineSERIAL_RECV_LF(1<<14)
voidDebugSerialInit(u32baud);
voidDebugSerialSendChar(u8value);
voidUartSendBuffer(u8*bufferStart,u8length);
externu8serialsBuffer[SERIALS_BUFFER_MAX_LENGTH];
externu16serialRecvStatus;
#endif
#include"debugSerial.h"
u8serialsBuffer[SERIALS_BUFFER_MAX_LENGTH];
u16serialRecvStatus=0;
void__irqUART0_IRQHandler(void)
{
u8Res;
Res=U0RBR;//读取接收数据
IENABLE;/*handlesnestedinterrupt*/
if(serialRecvStatus&SERIAL_RECV_PACKAGE)//已经收到完整的一包未处理
{
//丢弃
}
elseif(serialRecvStatus&SERIAL_RECV_LF)//已经收到0X0D
{
if('n'==Res)//收到0X0A
{
serialRecvStatus|=SERIAL_RECV_PACKAGE;
}
else//没收到,包错误,丢弃
{
SERIAL_RECV_CLEAR;
}
}
else
{
if('r'==Res)//收到0X0D
{
serialRecvStatus|=SERIAL_RECV_LF;
}
else//正常数据
{
if(SERIAL_RECV_LENGTH>=SERIALS_BUFFER_MAX_LENGTH)
{
//已经到最大包长度,数据错误
SERIAL_RECV_CLEAR;
}
else
{
serialsBuffer[serialRecvStatus]=Res;
serialRecvStatus++;
}
}
}
IDISABLE;
VICVectAddr=0;/*AcknowledgeInterrupt*/
}
//加入printf支持
#if1
#pragmaimport(__use_no_semihosting)
struct__FILE
{
int handle;