我正在开发一个嵌入式系统,其中包括一个 ARM Cortex-M4 CPU 和一些外围设备。其中一个外设包含可从 CPU 端访问(通过 AHB 总线)的 SRAM 块,但访问必须是字长事务(使用 LDR)。如果执行字节事务 (LDRB),则会生成异常。
在我的代码中,我从该内存中的一个数组中读取一个值并将其分配给一个局部变量。声明是这样的:
typedef enum
{
eType0 = 0,
eType1 = 1,
} type_t;
type_t arr_type;
uint32_t *array = BUF_ADDR; // array on periph. memory
uint32_t offset = 0;
arr_type = (type_t) array[offset]; // exception!
运行此代码时,读取内存时出现异常。碰巧这个赋值生成了汇编代码:
LDRB R1, [R2, R3, LSL #2]; // R2=array, R3=offset
即使我添加括号并显式转换表达式也是如此:
type = (uint32_t) (array[offset]);
解决这个问题的方法是声明arr_type
为uint32_t
而不是type_t
。现在,代码是:
LDR R1, [R2, R3, LSL #2];
这是预期的行为吗?我假设括号和强制转换(如果不是array
指针的自然类型)将使编译器生成和LDR
指令。因此,这看起来像一个错误。