我们的任务是比较整数值并打印出适当的提示以显示哪些值更大。
我在下面编写的代码将"i"和"j"分别初始化为 5 和 4。目标是比较"变量"而不是直接值本身。所以在这个例子中,我把"i"和"j"放在 cmp 中,而不是 5 和 4。
global _main
extern _printf, _system
section .text
_main:
; clear screen
push clr
call _system
add esp, 4
;Test prints out i. Successfully does the job.
push dword [i] ; why did it work here and not in *cmp*?
push prompt2
call _printf
add esp, 8
;compare values and do a conditional jump.
CMP dword [i], dword [j] ; compares values stored in i and j. i and j are base 10 nums. This is where the error appears.
JG igreat ; jumps if i is greater than j.
JL jgreat ; jumps if j is greater than i.
JE eks ; jumps if i and j are equal.
ret ; as a measure, this ends the code.
igreat: ; prints out a prompt if i is greater than j.
push dword [j]
push dword [i]
push ibig
call _printf
add esp, 16
ret
jgreat: ; prints out a prompt if j is greater than i.
push dword [i]
push dword [j]
push jbig
call _printf
add esp, 16
ret
eks: ; prints out a prompt if i is equal to j.
push dword [i]
push dword [j]
push ex
call _printf
add esp, 16
ret
last: ; terminates the code
ret
section .data
clr db "cls",0
i dd 5 ; i is initialized to 5
j dd 4 ; j is initialized to 4
prompt2 db "Value is %d",13,10,0
ibig db "Test 1. Comparisons: %d is bigger than %d",13,10,0
jbig db "Test 2. Comparisons: %d is bigger than %d",13,10,0
ex db "Test 3. Comparisons: %d is bigger than %d",13,10,0
我可以提出两个问题:
- CMP dword [
- i], dword [j] 产生错误:操作码和操作数的组合无效;但我使用printf 函数在前面的调用/代码行上。为什么?
- 我试图用一个直接的,比如CMP dword [i],9替换dword [j]。它确实打印出正确的提示,但使程序没有响应。为什么?
请注意,我运行的是Windows 8 Intel 32位,此代码已在DoSBox中运行的NASM和CMD中的GCC中"编译"。
我是一个完全的初学者,任何帮助将不胜感激。谢谢!
CMP
无法比较两个内存位置。它可以将寄存器与寄存器、常量或存储器进行比较,也可以将存储器位置与常量或寄存器进行比较。这就是为什么你得到一个错误,它是正确的。您必须将其中一个的值存储到寄存器中,然后进行比较。
程序"无响应"的原因是因为你搞砸了ESP
。你把东西推到堆叠上,调用_printf
然后出于某种原因向上移动ESP
16字节。 然后RET
将跳转到堆栈中的第一个值,我确定这是不正确的。
您应该将ESP
值存储在启动时和返回还原ESP
EBP
中,然后RET
。