ARM 架构中的 R15 程序计数器(PC)
本文来自于我关于 Arm Cortex-M 编程模型的系列文章。欢迎阅读、点评与交流~
1、Arm Cortex-M 处理器的编程模型
2、ARM 架构中的R13栈指针(SP)
3、ARM 架构中的R14链接寄存器(LR)
4、ARM 架构中的 R15 程序计数器(PC)
R15 程序计数器(PC) 是 ARM 架构处理器中一个极其特殊和核心的寄存器。它的主要作用和特点如下:
1. 核心功能:程序计数器
- 存放当前正在执行的指令的地址。这是它最基本、最重要的功能。
- 在指令执行过程中,处理器会自动更新 PC 的值,以指向下一条将要执行的指令。
- 因此,PC 控制着程序的执行流程。改变 PC 的值,就意味着进行跳转(如执行分支、调用函数、处理中断等)。
2. 在 ARM 架构中的特殊性
在经典的 ARM 状态(如 ARM7,ARM9)下,PC 有一个非常著名的特性:
- PC 的值 = 当前指令地址 + 8
- 这是因为 ARM 处理器采用了三级流水线(取指 -> 译码 -> 执行)。
- 当 CPU 正在执行 地址为
X的指令时:- 它已经在译码 地址为
X+4的指令。 - 它已经在预取(读取) 地址为
X+8的指令。
- 它已经在译码 地址为
- 此时,R15(PC)指向的是正在被取指的指令地址(
X+8) ,而不是正在执行的指令地址(X)。 - 这个"超前"的偏移量(8个字节)是硬件流水线机制导致的,程序员在编写汇编代码(尤其是计算相对跳转偏移量时)必须清楚这一点。
示例:
assembly
0x1000: MOV R0, #1 ; 正在执行这条指令
0x1004: ADD R1, R2, R3 ; 正在译码这条指令
0x1008: SUB R4, R5, #6 ; 正在取指这条指令,此时 PC = 0x1008
3. 作为通用寄存器的限制
虽然 R15 是寄存器文件中的一个,但它不能像 R0-R14 那样被随意使用:
- 不能随意写入 :直接向 PC 写入一个地址(例如
MOV PC, #0x1000)会立即导致程序跳转到该地址,这通常用于实现绝对跳转或函数返回。 - 读取有偏移:如上所述,读取 PC 得到的是(当前指令地址 + 8),而不是当前指令地址。
- 很多指令对 PC 的使用有特殊规定 :例如,
LDR PC, [Rn]常用于从内存加载跳转地址(如函数指针、中断向量),从而实现远距离跳转或模式切换。
4. 在较新的 ARM 架构中的变化
在更现代的 ARM 架构(如 Cortex-M, Cortex-A 系列)中:
- 流水线更深(例如五级、十三级流水线),但这个"PC 超前"的偏移量可能因微架构而异。
- 为了简化编程,ARM 在架构手册中明确规定,当以 PC 为源操作数读取其值时,返回的值是当前指令地址加上一个取决于架构的常量。程序员通常不需要再精确计算这个偏移,因为汇编器和链接器会处理与位置无关的跳转偏移计算。
- 在 Cortex-M 系列中,由于其精简和确定的特性,对 PC 的操作行为有更明确的定义,便于嵌入式开发。
总结
| 特性 | 说明 |
|---|---|
| 本质 | 程序计数器,存储指令地址。 |
| 核心作用 | 控制程序执行流程。改变 PC 即跳转。 |
| 经典 ARM 特性 | PC = 当前执行指令地址 + 8(三级流水线效应)。 |
| 访问限制 | 写入即跳转;读取值具有预定义的偏移。 |
| 编程意义 | 是理解程序流控制、函数调用、中断处理和位置无关代码的基础。 |
简单来说,R15(PC)就是 ARM 处理器的"指挥棒",它指向哪里,CPU 下一步就去哪里取指令执行。所有的函数调用、循环、条件判断、中断响应等流程改变,最终都体现为对 PC 寄存器的修改。