在ARM程序集中,我部分能够使以下代码正常工作。
它应该打印出从 1 到 10,946(包括 1)的所有偶数斐波那契数。 我能够打印出所有这些偶数的总和。 但是,我在让它打印出"2"作为斐波那契数之一时遇到了问题 - 它从"8"开始。 我试过改变东西,但它只是把它搞砸了。
这是我的代码:
.equ SWI_PrStr, 0x69
.equ SWI_PrInt,0x6b
.equ Stdout, 1
.equ SWI_Exit, 0x11
.global _start
.text
_start:
MOV R2,#0 @Starting value of 0 to register 1
MOV R3,#1 @starting val of 1 to register 2
LDR R4,=10946 @register gets this value
LOOP:
CMP R5,R4
BGE After
ADD R5,R3,R2 @adding to check if its odd
MOV R2,R3 @ answer is 0
MOV R3,R5 @result is in R5 adds to R2 is 0=1
MOV R7, R5
AND R8, R7, #1
CMP R8, #1
BGE LOOP
MOV R1, R5
SWI SWI_PrInt
LDR R1, =NL
SWI SWI_PrStr
ADD R6,R6,R5 @sum of the even numbers
MOV R0,#Stdout
LDR R1, =NL
SWI SWI_PrStr
B LOOP
After:
MOV R1,R6
SWI SWI_PrInt
SWI SWI_Exit
.data
NL: .asciz"n"
如何重写代码,使其打印出"2"以及所有其他数字?
我怀疑这是麻烦的行:
MOV R0,#Stdout
由于它的位置,它只有在打印偶数后才会执行 - 如果R0
最初包含错误的值,则"2"可能被打印到错误的位置。当你到达下一个值("8")时,R0
已经设置好了(大概没有其他东西碰它,所以它保持这种状态),所以以后的输出会正确地Stdout
。
将此行移动到上面的"设置"代码LOOP:
应该可以修复它。
当我在做这件事时,你可以通过使用测试而不是比较来进行优化 - 而不是这个:
MOV R7, R5
AND R8, R7, #1
CMP R8, #1
BGE LOOP
你可以简单地拥有这个:
TST R5, #1 @ TST is a bitwise AND
BNE LOOP @ Not equal? What?
这种明显的误导性条件是因为我们正在测试结果是 1 还是 0 - 因为与CMP
的整数比较是作为减法完成的,"等于"是"零结果"的同义词,"不等于"是"非零结果"的同义词。