uboot中添加FIQ中断及相关问题

本文主要说明了在uboot中添加FIQ中断时遇到的问题以及对应的解决办法。

首先交代一下项目的软硬件环境。硬件方面,使用s3c2440作为主控芯片,外接串口、网卡等设备。软件方面,主控芯片上电后运行uboot程序,之后通过网口在线烧写应用程序至RAM中运行。为了使设备始终处于可控状态,需要分别在uboot及应用程序之中添加遥控程序,遥控程序使用FIQ中断来实现。uboot程序的修改主要在\arch\arm\cpu\arm920t\start.s文件及arch\arm\lib\board.c文件中。

问题:

一、发生FIQ中断之后处理器做了哪些事情?

发生中断之后ARM处理器需要处理完当前的指令,然后自动完成以下事情:

1.将当前程序状态寄存器CPSR保存到FIQ模式下的备份程序状态寄存器SPSR_fiq之中(执行中断返回时其逆过程不能自动完成)。

2.将程序计数器PC(R15)值减4存入FIQ模式下的链接寄存器R14_fiq之中(也就是lr)。

3.将PC的值强制改为ox0000001c(异常向量表中FIQ的入口地址),并跳转到该地址执行。

说明:

1.ARM处理器定义了undefined_instruction(0x00000004)、software_interrupt(0x00000008)、prefetch_abort(0x0000000c)、data_abort(0x00000010)、irq(0x00000018)、fiq(0x0000001c)等几种异常,当异常发生时,芯片会自动完成上述操作并跳转到相应的地址去执行,其中芯片上电后自动跳转至0x00000000这个地址执行,一般会在这些异常的地址处存放跳转指令,以实现后续的操作。

1 _start:    b    start_code
2     ldr    pc, _undefined_instruction
3     ldr    pc, _software_interrupt
4     ldr    pc, _prefetch_abort
5     ldr    pc, _data_abort
6     ldr    pc, _not_used
7     ldr    pc, _irq
8     ldr    pc, _fiq

其中start_code为启动代码。

二、跳转至异常向量表后需要做那些事情?

处理器将PC强制指向0x0000001c之后,再次跳转至_fiq标号处,接下来用户需要编写程序完成以下事情:

1.计算程序的返回地址。因为ARM9采用5级流水线结构,且程序的执行阶段处在流水线的第3级,那么PC始终指向正在执行指令的下两条指令处,也就是正在执行的指令地址为PC-8(程序向高地址方向执行)。由于FIQ发生时处理器会处理完正在执行的指令,这时PC值也被更新,也就是PC=PC+4,然后处理器自动将PC值减4存入R14_fiq寄存器之中。程序返回的地址应该是FIQ发生时正在处理的指令的下一条指令处,所以将R14_fiq中的地址减4就是程序应该返回的地址。

2.根据需要,有选择地保存寄存器lr(使用bl指令时会更改该值)、r0-r12(均为通用寄存器,若被更改则无法恢复至FIQ发生前的状态,一般将这些寄存器的值进行入栈保护,这就要求事先在启动代码中设置FIQ模式的堆栈)。

3.跳转至中断处理函数运行,注意使用bl(保存下一条指令地址,可以返回),而不能使用b(不保存下条指令地址,不能返回)。

4.在执行完中断处理函数之后,需要将lr、r0-r12出栈恢复。

5.将lr的值传递给PC,跳转至FIQ发生时的下一条指令(使用subs pc, lr, #0的目的是为了将SPSR_fiq中的值恢复至CPSR中)。

1 fiq:
2     subs lr, lr, #4
3     stmfd sp!, {r0-r12,lr}
4     bl roger_test
5     ldmfd sp!,{r0-r12,lr}
6     subs pc, lr, #0

三、中断处理函数的位置?

处理器在上电之后跳转至启动代码处运行,按顺序进行相关寄存器及外设的初始化工作。在arch\arm\lib\board.c的board_init_r( )函数的末尾打开FIQ中断,此时处理器的初始化工作已经完成且已处于C语言环境。

时间: 2024-10-05 22:46:01

uboot中添加FIQ中断及相关问题的相关文章

向linux内核中添加外部中断驱动模块

本文主要介绍外部中断驱动模块的编写,包括:1.linux模块的框架及混杂设备的注册.卸载.操作函数集.2.中断的申请及释放.3.等待队列的使用.4.工作队列的使用.5.定时器的使用.6.向linux内核中添加外部中断驱动模块.7.完整驱动程序代码.linux的内核版本为linux2.6.32.2. 一.linux模块的框架以及混杂设备相关知识 1.内核模块的框架如下图所示,其中module_init()(图中有误,不是modules_init)只有在使用insmod命令手动加载模块时才会被调用,

u-boot中添加mtdparts支持以及Linux的分区设置

简介 作者:彭东林 邮箱:[email protected] u-boot版本:u-boot-2015.04 Linux版本:Linux-3.14 硬件平台:tq2440, 内存:64M   NandFlash: 256MB 下面我们分两部分,u-boot和kernel,首先介绍u-boot中是如何支持mtdparts的,然后简单分析Linux内核设置分区的两种方式: 方式一 在平台代码中写死,然后在初始化NandFlash的时候设置. 方式二 在u-boot中设置,这个比较灵活,u-boot将

在U-Boot中添加自定义命令以实现自动下载程序【转】

本文转载自:https://gaomf.cn/2016/06/26/%E5%9C%A8U-Boot%E4%B8%AD%E6%B7%BB%E5%8A%A0%E8%87%AA%E5%AE%9A%E4%B9%89%E5%91%BD%E4%BB%A4%E4%BB%A5%E5%AE%9E%E7%8E%B0%E8%87%AA%E5%8A%A8%E4%B8%8B%E8%BD%BD%E7%A8%8B%E5%BA%8F/ U-Boot中通过NFS下载程序是一种很普遍的方式,然而下载程序的过程并不能只用一条命令实现

uboot之添加命令

1.添加命令 若要向Uboot中添加自己的命令,而自己又不想知道命令实现的原理,可以按照以下做法: 1.在uboot/common目录下,随便找一个cmd_xxx.c文件,将cmd_xxx.c文件拷贝一下,并重新命名为cmd_hello.c cp cmd_xxx.c cmd_hello.c 2.进入到cmd_hello.c中,修改 a:修改宏U_BOOT_CMD U_BOOT_CMD宏参数有6个: 第一个参数:添加的命令的名字 第二个参数:添加的命令最多有几个参数(注意,假如你设置的参数个数是3

uboot自定义添加命令

1.添加命令 1.u-boot的命令格式: U_BOOT_CMD(name,maxargs,repeatable,command,"usage","help") name:命令的名字,不是一个字符串: maxargs:最大的参数个数: repeatable:命令是可重复的: command:对应的函数指针 2.在uboot/common目录下,随便找一个cmd_xxx.c文件,将cmd_xxx.c文件拷贝一下,并重新命名为cmd_hello.c cp cmd_xxx

step4----->往工程中添加Spring的子项目spring IO Platform------->通过maven添加相关框架(pom.xml)

添加Spring IO Platform的目的: 避免自己的project的外部依赖(external dependencies)之间产生版本冲突问题.更多详细信息参见:Spring IO Platform概述 具体操作步骤: step1,往自己的工程中添加Spring IO Platform 编写project的pom.xml,添加如下代码,引入Spring IO Platform <dependencyManagement> <dependencies> <depende

u-boot中链接地址和加载地址的相关知识

以zc702开发板的u-boot为例 链接地址(运行地址):链接地址是在程序编译链接阶段就确定好的地址. u-boot的链接脚本由CONFIG_SYS_LDSCRIPT宏定义来指定,如在zynq_common.h当中有如下代码: #define CONFIG_SYS_LDSCRIPT "arch/arm/cpu/armv7/zynq/u-boot.lds" 在该链接脚本中指定了u-boot中各部分的链接顺序.同时zynq_common.h中的CONFIG_SYS_TEXT_BASE则指

SylixOS中GIC通用中断控制器(二)——GIC实现

1.概述 本篇文档主要介绍IMX6UL平台上基于SylixOS集成开发环境中GIC通用中断控制器的实现流程和方法. 2.GIC控制器基地址获取 GIC控制器基地址通过调用armPrivatePeriphBaseGet函数获得.如图 2.1所示,Ctrl+h局搜索armPrivatePeriphBaseGet函数,搜索结果如图 2.2所示. 图 2.1全局搜索armPrivatePeriphBaseGet函数 图 2.2 armPrivatePeriphBaseGet函数搜索结果 参考DDI046

uboot中gd的定义和使用

最近在做uboot中nand启动相关的工作,遇到一个问题一直纠结着.现在终于明白了这个问题,想想还有好多兄弟在某个黑暗的角落里或者某台电脑前纠结着呢,所以赶紧写下来以供查阅. uboot version 2014.4 /* Architecture-specific global data */ struct arch_global_data { #if defined(CONFIG_FSL_ESDHC) u32 sdhc_clk; #endif #ifdef CONFIG_AT91FAMILY