3

我正在使用为 sam7s 处理器编译的 gcc 4.6.3。

我需要使用一些内联汇编:

int res;
asm volatile (" \
        MRS r0,CPSR     \n  \
        MOV %0, r0      \n  \
        BIC r0,r0,%1    \n  \
        MSR CPSR,r0     \n  \
        " : "=r" (res) : "r" (0xc0) : "r0" );
return res;

由 gcc 翻译为(我添加的评论):

mov r3, #192    ; load 0xc0 to r3
str r0, [sl, #56]   ; preserve value of r0?

mrs r0, CPSR    ; load CPSR to r0
mov r3, r0      ; save r0 to "res"; r3 overwritten!
bic r0, r0, r3  ; 
msr CPSR_fc, r0 ; 

问题是使用相同的寄存器“r3”代替“%0”(res)和“%1”(常数:0xc0)。由于这个原因,%1 在使用之前被覆盖,并且代码工作不正确。

问题是如何禁止 gcc 对输入/输出操作数使用相同的寄存器?

4

1 回答 1

5

好的,我终于在这里找到了

& 表示在
读取输入之前写入输出操作数,因此此输出不得与任何输入为同一寄存器。
如果没有这个,即使“0”约束不需要,gcc 也可以将输出和输入放在同一个寄存器中。这非常有用,但在这里提到是因为它特定于替代方案。与 = 和 % 不同,但与 ? 类似,您必须将它包含在它适用的每个备选方案中。

更改"=r" (res)"=&r" (res)一切正常后。

于 2012-10-05T12:06:44.997 回答