在嵌入式开发中,我们经常需要对单个位进行操作,如设置某个标志位、清除中断标志等。在传统的方式下,操作一个位通常需要几个步骤:读取整个寄存器值,修改目标位,然后再写回寄存器。这种方法不仅繁琐,还容易出现竞态条件等问题。幸运的是,Cortex-M3内核为我们提供了一个强大的工具——位带区(Bit-Band),它可以让单个位操作变得更加高效和直观。
就Cortex-M3而言,定义了如下两个位带区:
1. SRAM位带区:地址范围为0x20000000到0x200FFFFF,用于操作SRAM中的位。
2. 外设位带区:地址范围为0x40000000到0x400FFFFF,用于操作外设寄存器中的位。
同样Cortex-M3还定义了两个位带别名区,分别映射到上述两个位带区:
SRAM位带别名区:0x22000000到0x23FFFFFF
外设位带别名区:0x42000000`到`0x43FFFFFF
为了访问一个具体的位带别名区地址,可以使用以下公式进行计算:
别名地址 = 基地址 + (偏移地址 × 32) + (位编号 × 4)
其中:
基地址是位带别名区的起始地址,例如0x42000000。
偏移地址是原始位带区地址相对于基地址的偏移。
位编号是该位在32位寄存器中的位置(0-31)。
例如,假设你要操作外设寄存器`0x40000000`中的第3位,它的别名地址计算如下:
别名地址 = 0x42000000 + (0x00000000 × 32) + (3 × 4)
= 0x42000000 + 0x0000000C
= 0x4200000C
通过访问地址0x4200000C,你可以直接操作0x40000000寄存器中的第3位。
下面是一个简单的小例子
#define BITBAND_PERI_BASE 0x42000000
#define PERIPH_BASE 0x40000000
// 计算外设寄存器 0x40000000 第 3 位的位带别名地址
#define BITBAND_PERI(addr, bitnum) ((BITBAND_PERI_BASE + ((addr - PERIPH_BASE) * 32) + (bitnum * 4)))
// 操作寄存器 0x40000000 的第 3 位
#define REG_ADDR 0x40000000
#define REG_BIT 3
int main(void) {
// 设置寄存器 0x40000000 的第 3 位
*(volatile unsigned int *)BITBAND_PERI(REG_ADDR, REG_BIT) = 1;
// 清除寄存器 0x40000000 的第 3 位
*(volatile unsigned int *)BITBAND_PERI(REG_ADDR, REG_BIT) = 0;
while(1);
}