本博客的参考文章及相关资料下载 :
参考手册 : A2.2 Processor modes
处理器的 七种 工作模式 :
处理器 工作模式 区别 :
Linux 操作系统运行模式 :
特权模式 :
五种 处理器工作模式 ( 异常模式 ) 对应的异常类型 :
七种 异常类型 对应的 处理器工作模式 : ARM 架构 支持 七种类型的异常,
系统模式 :
修改 处理器 工作模式 的 前提条件 :
参考手册 : A2.5 Program status registers
程序状态字寄存器 :
参考手册 : A2.5 Program status registers
程序状态字寄存器 位 类型简介 :
参考手册 : arm汇编手册(中文版).chm
修改程序状态字寄存器 使用到的汇编指令 :
bic , ,
, dest 存放位清除结果, op1 是被清除的对象, op2 是掩码; bic r0, r0, #0b1011
, 清除 r0 中的 第0, 1, 3 位, 其余位保持不变, 结果放入 r0 中;ORR{条件}{S} , ,
, dest 结果是 op 1 与 op 2 进行或运算的结果; ORR R0, R0, #3
, 将 立即数 3 与 R0 寄存器中的值 进行 或 运算, 然后将运算结果存放到 R0 中;MRS R0, CPSR
, 将 CPRS 寄存器的值 复制 到 R0 寄存器中; MSR CPSR, R0
, 将 R0 寄存器中的值 设置 到 CPSR 寄存器中 ;
代码 逻辑 分析 :
set_svc :
, 在标号下定义一组汇编指令; mrs r0 cpsr
将 CPSR 寄存器中的值导出到 R0 寄存器中; bic r0, r0, #0x1f
, 将 R0 寄存器的值 与 #0x1f 进行 与操作, 即 后5 位都设置成0, 然后将 与 操作的结果保存到 R0 寄存器中 ; orr r0, r0, #0xd3
语句设置, 将 R0 寄存器中的值 与 0x13 进行 或操作, 将 或操作的结果 存放到 R0 寄存器中; msr cpsr, r0
, 将处理完的 CPSR 寄存器值 设置给 CPSR 寄存器;汇编代码示例 :
@****************************
@File:start.S
@
@BootLoader 初始化代码
@****************************
.text @ 宏 指明代码段
.global _start @ 伪指令声明全局开始符号
_start: @ 程序入口标志
b reset @ reset 复位异常
ldr pc, _undefined_instruction @ 未定义异常, 将 _undefined_instruction 值装载到 pc 指针中
ldr pc, _software_interrupt @ 软中断异常
ldr pc, _prefetch_abort @ 预取指令异常
ldr pc, _data_abort @ 数据读取异常
ldr pc, _not_used @ 占用 0x00000014 地址
ldr pc, _irq @ 普通中断异常
ldr pc, _fiq @ 软中断异常
_undefined_instruction: .word undefined_instruction @ _undefined_instruction 标号存放了一个值, 该值是 32 位地址 undefined_instruction, undefined_instruction 是一个地址
_software_interrupt: .word software_interrupt @ 软中断异常
_prefetch_abort: .word prefetch_abort @ 预取指令异常 处理
_data_abort: .word data_abort @ 数据读取异常
_not_used: .word not_used @ 空位处理
_irq: .word irq @ 普通中断处理
_fiq: .word fiq @ 快速中断处理
undefined_instruction: @ undefined_instruction 地址存放要执行的内容
nop
software_interrupt: @ software_interrupt 地址存放要执行的内容
nop
prefetch_abort: @ prefetch_abort 地址存放要执行的内容
nop
data_abort: @ data_abort 地址存放要执行的内容
nop
not_used: @ not_used 地址存放要执行的内容
nop
irq: @ irq 地址存放要执行的内容
nop
fiq: @ fiq 地址存放要执行的内容
nop
reset: @ reset 地址存放要执行的内容
bl set_svc @ 跳转到 set_svc 标号处执行
set_svc:
mrs r0, cpsr @ 将 CPSR 寄存器中的值 导出到 R0 寄存器中
bic r0, r0, #0x1f @ 将 R0 寄存器中的值 与 #0x1f 立即数 进行与操作, 并将结果保存到 R0 寄存器中, 实际是将寄存器的 0 ~ 4 位 置 0
orr r0, r0, #0xd3 @ 将 R0 寄存器中的值 与 #0xd3 立即数 进行或操作, 并将结果保存到 R0 寄存器中, 实际是设置 0 ~ 4 位 寄存器值 的处理器工作模式代码
msr cpsr, r0 @ 将 R0 寄存器中的值 保存到 CPSR 寄存器中
gboot.lds 链接器脚本 代码解析 :
OUTPUT_ARCH(架构名称)
指明输出格式, 即处理器的架构, 这里是 arm 架构的, OUTPUT_ARCH(arm)
;ENTRY(入口位置)
, 在上面的 Start.S 中设置的程序入口是 _start
, 代码为 ENTRY(_start)
;.text :
设置代码段; .data :
设置数据段;.bss :
设置 BSS 段; bss_start = .;
;bss_end = .;
;. = ALIGN(4);
设置四字节对齐即可;OUTPUT_ARCH(arm) /*指明处理器结构*/
ENTRY(_start) /*指明程序入口 在 _start 标号处*/
SECTIONS {
. = 0x50008000; /*整个程序链接的起始位置, 根据开发板确定, 不同开发板地址不一致*/
. = ALIGN(4); /*对齐处理, 每段开始之前进行 4 字节对齐*/
.text : /*代码段*/
{
start.o (.text) /*start.S 转化来的代码段*/
*(.text) /*其它代码段*/
}
. = ALIGN(4); /*对齐处理, 每段开始之前进行 4 字节对齐*/
.data : /*数据段*/
{
*(.data)
}
. = ALIGN(4); /*对齐处理, 每段开始之前进行 4 字节对齐*/
bss_start = .; /*记录 bss 段起始位置*/
.bss : /*bss 段*/
{
*(.bss)
}
bss_end = .; /*记录 bss 段结束位置*/
}
makefile 文件编写 :
%.o : %.S
, 产生过程是 arm-linux-gcc -g -c $^
, 其中 ^
标识是所有的依赖文件, 在该规则下 start.S 会被变异成 start.o ; %.o : %.c
, 产生过程是 arm-linux-gcc -g -c $^
; all:
设置最终编译目标; all: start.o
表示最终目标需要依赖该文件; arm-linux-ld -Tgboot.lds -o gboot.elf $^
, 需要使用链接器脚本进行连接, ①链接工具是 arm-linux-ld 工具, ②使用 -Tgboot.lds
设置链接器脚本 是刚写的 gboot.lds 链接器脚本, ③输出文件是 gboot.elf 这是个中间文件, ④ 依赖文件是 $^
代表所有的依赖; arm-linux-objcopy -O binary gboot.elf gboot.bin
, 使用 -O binary
设置输出二进制文件, 依赖文件是 gboot.elf
, 输出的可执行二进制文件 即 结果是 gboot.bin
;all: start.o #依赖于 start.o
arm-linux-ld -Tgboot.lds -o gboot.elf $^ #使用链接器脚本, 将 start.o 转为 gboot.elf
arm-linux-objcopy -O binary gboot.elf gboot.bin #将 gboot.elf 转化为可以直接在板子上执行的 gboot.bin 文件
%.o : %.S #通用规则, 如 start.o 是由 start.S 编译来的, -c 是只编译不链接
arm-linux-gcc -g -c $^
%.o : %.c #通用规则, 如 start.o 是由 start.c 编译来的, -c 是只编译不链接
arm-linux-gcc -g -c $^
.PHONY: clean
clean: #清除编译信息
rm *.o *.elf *.bin
编译过程 :
make
; 本博客的参考文章及相关资料下载 :