首页 > 代码库 > LCD实验学习笔记(八):中断控制器

LCD实验学习笔记(八):中断控制器

s3c2440有60个中断源(其中15个为子中断源)。

程序状态寄存器(PSR)的F位设为1,禁用快速中断(FRQ)。

程序状态寄存器(PSR)的I位设为1,禁用普通中断(IRQ)。

相反,PSR寄存器F位为0开FRQ中断,I位为0开IRQ中断。

s3c2440中断控制器中有五个控制寄存器:中断源等待寄存器(SRCPND),中断模式寄存器(INTMOD),屏蔽寄存器(INTMSK),优先级寄存器(PRIORITY),中断等待寄存器(INTPND)。

SRCPND寄存器各位对应不同的中断,其中外部中断EINT4-7共用bit4,EINT8-23共用bin5。各位初始值为0,当中断发生时,对应位值改为1。

INTMOD寄存器各位用于标记该位对应的中断是模式,0为IRQ,1为FRQ。只允许有一个中断设为FRQ。初始值都为0。

INIMSK寄存器各位标记该位对应中断是否可用,0表示有效,1表示禁止。初始值都为1,即禁止所有中断。

PRIORITY寄存器用于设置6个中断优先权仲裁模块的工作方式,主要用来设定各中断优先策略。

INTPND寄存器的32位同一时刻最多只允许有一个值为1,这一位是有相就的中断请求发生,且该中断未被屏蔽,而且该中断请求通过了优先仲裁。这个中断请求将被送到CPU,CPU将跳转到中断服务程序执行。中断服务程序可以读取INTPND或INTOFFSET寄存器(当INTPND中某位标记中断请求发生了,INTOFFSET寄存器中的值为0-31,标记INTPND的哪一位被置为1了)的值,以确定哪个中断发生了。

有的几个中断共有一个SRCPND的中断位,比如INT_AC97和INT_WDT共用INT_WDT_AC97中断位。这种子中断共15个。子中断发生时,先在INTSRCPND寄存器相应位置1,如果INTSUBMSK寄存器相应位未屏蔽(即该位为0),则该中断送到SRCPND寄存器。

23个外部中断:

 EINTINT0-2三个寄存器用于设置23个外部中断的触发方式和过滤器是否有效。触发方式有低电平、高电平、下降沿、上升沿、电平高低变化。

EINTFLT0-3三个寄存器用于设置外部中断引脚的过滤参数。

EINTMASK寄存器用于设置23个外部中断的屏蔽情况,默认是禁用中断的。

EINTPEND寄存器记录着23个外部中断哪一个发生了。发生中断的对应位置1。初始值为0。但清除中断位标记时要写入1。

代码:


/*interrupt registes*/
#define SRCPND (*(volatile unsigned long *)0x4A000000)
#define INTMOD (*(volatile unsigned long *)0x4A000004)
#define INTMSK (*(volatile unsigned long *)0x4A000008)
#define PRIORITY (*(volatile unsigned long *)0x4A00000c)
#define INTPND (*(volatile unsigned long *)0x4A000010)
#define INTOFFSET (*(volatile unsigned long *)0x4A000014)
#define SUBSRCPND (*(volatile unsigned long *)0x4A000018)
#define INTSUBMSK (*(volatile unsigned long *)0x4A00001c)

/*external interrupt registers*/
#define EINTMASK (*(volatile unsigned long *)0x560000a4)
#define EINTPEND (*(volatile unsigned long *)0x560000a8)

#define BIT_ALLMSK (0xffffffff)

//声明50个中断服务函数指针

void (*isr_handle_array[50])(void);

//虚拟中断服务

void Dummy_isr(void)
{
  while(1);
}

//初始化中断

void init_irq(void)
{
  int i = 0;
  for (i = 0; i < sizeof(isr_handle_array) / sizeof(isr_handle_array[0]); i++)
  {
  isr_handle_array[i] = Dummy_isr;
  }

  INTMOD = 0x0; // 所有中断都设为IRQ模式
  INTMSK = BIT_ALLMSK; // 先屏蔽所有中断

}

//中断处理

void IRQ_Handle(void)
{
  unsigned long oft = INTOFFSET;
  //清中断
  if (oft == 4)
    EINTPEND = 1<<7; //EINT4-7合用IRQ4,注意EINTPEND[3:0]保留未用,向这些位写入1可能导致未知结果
  SRCPND = 1<<oft;
  INTPND = INTPND;

  /* 调用中断服务程序 */
  isr_handle_array[oft]();
}

 

LCD实验学习笔记(八):中断控制器