历史上的今天
返回首页

历史上的今天

今天是:2024年09月02日(星期一)

正在发生

2021年09月02日 | S3C2440裸机------异常与中断__按键中断程序示例

2021-09-02 来源:eefocus

1.中断处理流程

首先我们要对中断进行初始化,中断初始化主要分为下面三块:


设置中断,让它能够发出中断信号,

设置中断控制器,让它能够发出中断信号给CPU,

设置CPU,CPSR有I位,它是总开关。

然后中断的处理过程和之前说的异常处理过程类似,不一样的就是处理完成之后要清中断。


2 start.s

首先我们CPSR寄存器的bit7清零,这是中断的总开关,只有这一位设置为0才能相应中断。

mrs r0, cpsr         /* 读出cpsr */

bic r0, r0, #(1<<7)  /* 清除I位, 使能中断 */

msr cpsr, r0

然后当发生中断时,程序会跳转到0x18的地方去执行程序,因此我们增加如下代码

ldr pc, irq_addr /* vector 0x18 : irq */

然后我们增加中断处理流程的代码,整体流程也是保存现场,处理中断,恢复现场。


 

.text

.global _start

 

_start:

b reset          /* vector 0 : reset */

ldr pc, und_addr /* vector 4 : und */

ldr pc, swi_addr /* vector 8 : swi */

b halt /* vector 0x0c : prefetch aboot */

b halt /* vector 0x10 : data abort */

b halt /* vector 0x14 : reserved */

ldr pc, irq_addr /* vector 0x18 : irq */

b halt /* vector 0x1c : fiq */

 

und_addr:

.word do_und

 

swi_addr:

.word do_swi

 

irq_addr:

.word do_irq

 

do_und:

/* 执行到这里之前:

* 1. lr_und保存有被中断模式中的下一条即将执行的指令的地址

* 2. SPSR_und保存有被中断模式的CPSR

* 3. CPSR中的M4-M0被设置为11011, 进入到und模式

* 4. 跳到0x4的地方执行程序 

*/

 

/* sp_und未设置, 先设置它 */

ldr sp, =0x34000000

 

/* 在und异常处理函数中有可能会修改r0-r12, 所以先保存 */

/* lr是异常处理完后的返回地址, 也要保存 */

stmdb sp!, {r0-r12, lr}  

/* 保存现场 */

/* 处理und异常 */

mrs r0, cpsr

ldr r1, =und_string

bl printException

/* 恢复现场 */

ldmia sp!, {r0-r12, pc}^  /* ^会把spsr的值恢复到cpsr里 */

und_string:

.string "undefined instruction exception"

 

.align 4

 

do_swi:

/* 执行到这里之前:

* 1. lr_svc保存有被中断模式中的下一条即将执行的指令的地址

* 2. SPSR_svc保存有被中断模式的CPSR

* 3. CPSR中的M4-M0被设置为10011, 进入到svc模式

* 4. 跳到0x08的地方执行程序 

*/

 

/* sp_svc未设置, 先设置它 */

ldr sp, =0x33e00000

 

/* 保存现场 */

/* 在swi异常处理函数中有可能会修改r0-r12, 所以先保存 */

/* lr是异常处理完后的返回地址, 也要保存 */

stmdb sp!, {r0-r12, lr}  

 

mov r4, lr

/* 处理swi异常 */

mrs r0, cpsr

ldr r1, =swi_string

bl printException

 

sub r0, r4, #4

bl printSWIVal

/* 恢复现场 */

ldmia sp!, {r0-r12, pc}^  /* ^会把spsr的值恢复到cpsr里 */

swi_string:

.string "swi exception"

 

.align 4

 

do_irq:

/* 执行到这里之前:

* 1. lr_irq保存有被中断模式中的下一条即将执行的指令的地址

* 2. SPSR_irq保存有被中断模式的CPSR

* 3. CPSR中的M4-M0被设置为10010, 进入到irq模式

* 4. 跳到0x18的地方执行程序 

*/

 

/* sp_irq未设置, 先设置它 */

ldr sp, =0x33d00000

 

/* 保存现场 */

/* 在irq异常处理函数中有可能会修改r0-r12, 所以先保存 */

/* lr-4是异常处理完后的返回地址, 也要保存 */

sub lr, lr, #4

stmdb sp!, {r0-r12, lr}  

/* 处理irq异常 */

bl handle_irq_c

/* 恢复现场 */

ldmia sp!, {r0-r12, pc}^  /* ^会把spsr_irq的值恢复到cpsr里 */

 

 

reset:

/* 关闭看门狗 */

ldr r0, =0x53000000

ldr r1, =0

str r1, [r0]

 

/* 设置MPLL, FCLK : HCLK : PCLK = 400m : 100m : 50m */

/* LOCKTIME(0x4C000000) = 0xFFFFFFFF */

ldr r0, =0x4C000000

ldr r1, =0xFFFFFFFF

str r1, [r0]

 

/* CLKDIVN(0x4C000014) = 0X5, tFCLK:tHCLK:tPCLK = 1:4:8  */

ldr r0, =0x4C000014

ldr r1, =0x5

str r1, [r0]

 

/* 设置CPU工作于异步模式 */

mrc p15,0,r0,c1,c0,0

orr r0,r0,#0xc0000000   //R1_nF:OR:R1_iA

mcr p15,0,r0,c1,c0,0

 

/* 设置MPLLCON(0x4C000004) = (92<<12)|(1<<4)|(1<<0) 

*  m = MDIV+8 = 92+8=100

*  p = PDIV+2 = 1+2 = 3

*  s = SDIV = 1

*  FCLK = 2*m*Fin/(p*2^s) = 2*100*12/(3*2^1)=400M

*/

ldr r0, =0x4C000004

ldr r1, =(92<<12)|(1<<4)|(1<<0)

str r1, [r0]

 

/* 一旦设置PLL, 就会锁定lock time直到PLL输出稳定

* 然后CPU工作于新的频率FCLK

*/

 

/* 设置内存: sp 栈 */

/* 分辨是nor/nand启动

* 写0到0地址, 再读出来

* 如果得到0, 表示0地址上的内容被修改了, 它对应ram, 这就是nand启动

* 否则就是nor启动

*/

mov r1, #0

ldr r0, [r1] /* 读出原来的值备份 */

str r1, [r1] /* 0->[0] */ 

ldr r2, [r1] /* r2=[0] */

cmp r1, r2   /* r1==r2? 如果相等表示是NAND启动 */

ldr sp, =0x40000000+4096 /* 先假设是nor启动 */

moveq sp, #4096  /* nand启动 */

streq r0, [r1]   /* 恢复原来的值 */

 

bl sdram_init

//bl sdram_init2 /* 用到有初始值的数组, 不是位置无关码 */

 

/* 重定位text, rodata, data段整个程序 */

bl copy2sdram

 

/* 清除BSS段 */

bl clean_bss

 

/* 复位之后, cpu处于svc模式

* 现在, 切换到usr模式

*/

mrs r0, cpsr         /* 读出cpsr */

bic r0, r0, #0xf     /* 修改M4-M0为0b10000, 进入usr模式 */

bic r0, r0, #(1<<7)  /* 清除I位, 使能中断 */

msr cpsr, r0

 

/* 设置 sp_usr */

ldr sp, =0x33f00000

 

ldr pc, =sdram

sdram:

bl uart0_init

 

bl print1

/* 故意加入一条未定义指令 */

und_code:

.word 0xdeadc0de  /* 未定义指令 */

bl print2

 

swi 0x123  /* 执行此命令, 触发SWI异常, 进入0x8执行 */

 

//bl main  /* 使用BL命令相对跳转, 程序仍然在NOR/sram执行 */

ldr pc, =main  /* 绝对跳转, 跳到SDRAM */

 

halt:

b halt

3 interrupt.c

3.1中断初始化

我们在这里面完成初始化中断控制器和初始化按键中断的工作,

 我们首先把按键设置成中断引脚, 


/* 配置GPIO为中断引脚 */

GPFCON &= ~((3<<0) | (3<<4));

GPFCON |= ((2<<0) | (2<<4));   /* S2,S3被配置为中断引脚 */

 

GPGCON &= ~((3<<6) | (3<<11));

GPGCON |= ((2<<6) | (2<<11));   /* S4,S5被配置为中断引脚 */

然后将中断设置为双边沿触发,按下松开都会触发中断。


/* 设置中断触发方式: 双边沿触发 */

EXTINT0 |= (7<<0) | (7<<8);     /* S2,S3 */

EXTINT1 |= (7<<12);             /* S4 */

EXTINT2 |= (7<<12);             /* S5 */

然后设置外部中断mask,使能外部中断,外部中断0和1不需要设置,因为寄存器中是reserved。

/* 设置EINTMASK使能eint11,19 */

EINTMASK &= ~((1<<11) | (1<<19));

接下来我们需要设置中断控制器,

3.2 中断处理函数

在中断处理函数里面,首先需要分辨中断源,然后调用对应的处理函数。最后清中断。


#include "s3c2440_soc.h"

 

 

/* SRCPND 用来显示哪个中断产生了, 需要清除对应位

 * bit0-eint0

 * bit2-eint2

 * bit5-eint8_23

 */

 

/* INTMSK 用来屏蔽中断, 1-masked

 * bit0-eint0

 * bit2-eint2

 * bit5-eint8_23

 */

 

/* INTPND 用来显示当前优先级最高的、正在发生的中断, 需要清除对应位

 * bit0-eint0

 * bit2-eint2

 * bit5-eint8_23

 */

 

/* INTOFFSET : 用来显示INTPND中哪一位被设置为1

 */

 

/* 初始化中断控制器 */

void interrupt_init(void)

{

INTMSK &= ~((1<<0) | (1<<2) | (1<<5));

}

 

/* 初始化按键, 设为中断源 */

void key_eint_init(void)

{

/* 配置GPIO为中断引脚 */

GPFCON &= ~((3<<0) | (3<<4));

GPFCON |= ((2<<0) | (2<<4));   /* S2,S3被配置为中断引脚 */

 

GPGCON &= ~((3<<6) | (3<<22));

GPGCON |= ((2<<6) | (2<<22));   /* S4,S5被配置为中断引脚 */

 

/* 设置中断触发方式: 双边沿触发 */

EXTINT0 |= (7<<0) | (7<<8);     /* S2,S3 */

EXTINT1 |= (7<<12);             /* S4 */

EXTINT2 |= (7<<12);             /* S5 */

 

/* 设置EINTMASK使能eint11,19 */

EINTMASK &= ~((1<<11) | (1<<19));

}

 

/* 读EINTPEND分辨率哪个EINT产生(eint4~23)

 * 清除中断时, 写EINTPEND的相应位

 */

 

 

void key_eint_irq(int irq)

{

unsigned int val = EINTPEND;

unsigned int val1 = GPFDAT;

unsigned int val2 = GPGDAT;

if (irq == 0) /* eint0 : s2 控制 D12 */

{

if (val1 & (1<<0)) /* s2 --> gpf6 */

{

/* 松开 */

GPFDAT |= (1<<6);

}

else

{

/* 按下 */

GPFDAT &= ~(1<<6);

}

}

else if (irq == 2) /* eint2 : s3 控制 D11 */

{

if (val1 & (1<<2)) /* s3 --> gpf5 */

{

/* 松开 */

GPFDAT |= (1<<5);

}

else

{

/* 按下 */

GPFDAT &= ~(1<<5);

}

}

else if (irq == 5) /* eint8_23, eint11--s4 控制 D10, eint19---s5 控制所有LED */

{

if (val & (1<<11)) /* eint11 */

{

if (val2 & (1<<3)) /* s4 --> gpf4 */

{

/* 松开 */

GPFDAT |= (1<<4);

}

else

{

/* 按下 */

GPFDAT &= ~(1<<4);

}

}

else if (val & (1<<19)) /* eint19 */

{

if (val2 & (1<<11))

{

/* 松开 */

/* 熄灭所有LED */

GPFDAT |= ((1<<4) | (1<<5) | (1<<6));

}

else

{

/* 按下: 点亮所有LED */

GPFDAT &= ~((1<<4) | (1<<5) | (1<<6));

}

}

}

 

EINTPEND = val;

}

 

void handle_irq_c(void)

{

/* 分辨中断源 */

int bit = INTOFFSET;

 

/* 调用对应的处理函数 */

if (bit == 0 || bit == 2 || bit == 5)  /* eint0,2,eint8_23 */

{

key_eint_irq(bit); /* 处理中断, 清中断源EINTPEND */

}

 

/* 清中断 : 从源头开始清 */

SRCPND = (1< INTPND = (1<}

推荐阅读

史海拾趣

Good-Ark公司的发展小趣事
考虑电路的防水、防尘等性能,确保其在恶劣环境下也能正常工作。
Avel Lindberg公司的发展小趣事
检查声音传感器是否有损坏或被遮挡,确保其能够正常接收声音信号。
常州能动(ENDRIVE)公司的发展小趣事

常州能动(ENDRIVE)公司自创立之初,就将技术创新作为公司发展的核心驱动力。在公司发展的早期阶段,研发团队成功开发出一款高效能、低能耗的电源管理芯片,这一创新产品在市场上引起了广泛关注。通过不断的技术迭代和优化,这款芯片的性能逐渐提升,成本逐渐降低,成为了公司的明星产品,为公司的快速发展奠定了坚实基础。

DBM Optix公司的发展小趣事

DBM Optix公司成立之初,便致力于在光学通信领域取得技术突破。在初创阶段,公司面临着资金紧张、人才匮乏等多重挑战。然而,凭借对技术的执着追求和对市场的敏锐洞察,DBM Optix成功研发出了一款具有高性能、高可靠性的光学模块,这一技术突破迅速获得了市场的认可,为公司的发展奠定了坚实的基础。

启英泰伦(Chiplntelli)公司的发展小趣事

随着技术的不断进步和市场需求的日益增长,启英泰伦的芯片产品历经了多次迭代。从最初的AI语音芯片,到后来的AI语音Wi-Fi芯片、AI语音BLE芯片,公司形成了系列化的芯片产品布局。这些芯片广泛应用于智能家居、智能家电、智慧照明等领域,为客户提供了更自然、更简单、更智能的人机交互体验。启英泰伦的芯片产品在市场上获得了广泛的认可,服务客户超过5000家,累计装机量突破2000万台。

EPCOS/TDK公司的发展小趣事

自上世纪60年代起,TDK就开始在中国布局。首先,在台湾建立合资公司,随后在香港设立销售网络及生产线。进入80年代后,TDK正式踏足中国大陆,在华东及华南多个地区建立生产基地,并逐渐将业务扩展至全国。TDK在中国的发展不仅推动了当地电子工业的进步,也为公司自身赢得了广阔的市场空间。目前,TDK在中国的员工人数近6万人,占TDK全球员工总数的近55%。

问答坊 | AI 解惑

希望可以体验lpc114

最近在学习cortex 也正在从事这方面的开发!…

查看全部问答>

wince+directshow的视频采集的问题

因开发需要在做DirectShow视频采集的程序在开发中遇到为何老是提示:Error   Creating   Device   Enumerator。已安装硬件设备,请高手指点。      程序代码如下:       ...…

查看全部问答>

ORI与SBR的区别

大家好,本人初学avr,在看到汇编指令中ORI与SBR都是寄存器与立即数与的操作,不明白这两条指令的区别在哪里?多谢指导…

查看全部问答>

关于NRF24L01芯片功耗问题求救

大家好!  现搞一无线的东东,有源电子标签,用到2401芯片,但处于就收模式下功耗有点大,想让它处于待机状态下,这样功耗会变的很小,但会检测不到读写器的信号,~~郁闷啊!最理想的是平时处于待机或掉电状态,只要进入读写器范围就处于接收 ...…

查看全部问答>

ADUC7060 之I2C

/******************************************************************************            I2C 主机寄存器 I2CMCON (I2C 主机控制寄存器)I2CMSTA (I2C 主机状态寄存器)I2CMRX&nb ...…

查看全部问答>

简单说说我学430的过程

    开始接触430已经有一段时间了,其间断断续续,到现在才算差不多入门了,趁这个机会把这个过程 说一说。    之前学过51,后来在论坛上知道了430,经过一些简单的了解觉得430比51强大好多,于是准备开始学 习。& ...…

查看全部问答>

Windows驱动开发前景求教

       在坛子里也算有段时间了,在这里学到了很多东西。大四毕业的时候毕设做的是有关FPGA的,承蒙坛子里的坛友帮助,帮我解决了一个很大的问题。现在我在一所普通院校里读研,已经研一了。上半年搞了半年的FPGA相关的东 ...…

查看全部问答>

出5509开发板+YXxds510仿真器

如题,出5509开发板+YXxds510仿真器,自定义为8成新以上,包好一起300不包邮 QQ 282774356…

查看全部问答>

LM3S SSI AD采样?

程序如下,设置断点发下程序执行到红色部分就不再往下执行了,不知道为什么。 int main(void) {     unsigned long ulDataTx[NUM_SSI_DATA];     unsigned long ulDataRx[NUM_SSI_DATA];     unsigned long ul ...…

查看全部问答>

找郑州的430工程师做个板子

用MSP430做个简单的板子,控制个微型直流电机的旋转,要求电路板体积小,功耗好,用电池供电,有意者加企鹅详聊:616837780,外地的请绕行。…

查看全部问答>