使用AVR-GCC的switch语句的汇编代码

时间:2013-01-30 23:59:05

标签: assembly switch-statement avr avr-gcc

嘿,我无法理解编译器为以下交换机statemet省略的汇编代码。它与我使用gcc等看到的常用汇编代码不同。

switch(instr) { 
case OP_NOP: 
    break;

case OP_BIPUSH: 
    stack_push(arg0.z.bh); pc_inc = 2; break;
} 

上述C代码的汇编代码:

switch(instr){ 
+00001482: 8529 LDD R18,Y+9 Load indirect with displacement 
+00001483: 2F82 MOV R24,R18 Copy register 
+00001484: E090 LDI R25,0x00 Load immediate 
+00001485: 01FC MOVW R30,R24 Copy register pair 
+00001486: 9732 SBIW R30,0x02 Subtract immediate from word 
+00001487: 3BED CPI R30,0xBD Compare with immediate 
+00001488: 05F1 CPC R31,R1 Compare with carry 
+00001489: F008 BRCS PC+0x02 Branch if carry set 
+0000148A: C39D RJMP PC+0x039E Relative jump 
+0000148B: 57E0 SUBI R30,0x70 Subtract immediate 
+0000148C: 4FFF SBCI R31,0xFF Subtract immediate with carry 
+0000148D: 0FEE LSL R30 Logical Shift Left 
+0000148E: 1FFF ROL R31 Rotate Left Through Carry 
+0000148F: 9005 LPM R0,Z+ Load program memory and postincrement 
+00001490: 91F4 LPM R31,Z Load program memory 
+00001491: 2DE0 MOV R30,R0 Copy register 
+00001492: 9409 IJMP Indirect jump to (Z) 
2003: stack_push(arg0.z.bh); pc_inc = 2; 
+00001493: 2D85 MOV R24,R5 Copy register 
+00001494: 2799 CLR R25 Clear Register 
+00001495: FD87 SBRC R24,7 Skip if bit in register cleared 
+00001496: 9590 LAT R25 Load and Toggle 
+00001497: C338 RJMP PC+0x0339 Relative jump 

我假设代码是这样的:NOP正在被编译器优化掉。

但我刚刚理解代码是如何为案例设置查找表的?

任何帮助都会很棒,谢谢。

1 个答案:

答案 0 :(得分:2)

在您的示例中只检查一个案例,因此只检查instr是否等于OP_BIPUSH。如果是,则向前跳过以完成其余操作,否则从函数调用返回。

在调用开始时,Y寄存器(实际上是两个通用寄存器,我忘了哪个)指向堆栈的顶部。参数instr在堆栈下方有9个点,并被加载到r24中。 r25得0,因此该对包含instr的值。

我们只需要针对一件事进行测试,因此减去2并与0xBD进行比较。 (r1是零寄存器。它总是包含0。)

如果结果相同,则跳过下一条指令并继续完成其余指令。如果它们不相等,则执行下一条指令,该指令通过跳转从函数调用返回。