这是将 2 和 3 相乘的代码的一部分:
section .text
global _start
_start:
mov al,'3'
sub al, '0'
mov bl, '2'
sub bl, '0'
mul bl
add al, '0'
;first question ->
mov [res], al
mov ecx,msg
mov edx, len
mov ebx,1 ;file descriptor (stdout)
mov eax,4 ;system call number (sys_write)
int 0x80
;second question ->
mov ecx,res
mov edx, 1
mov ebx,1 ;file descriptor (stdout)
mov eax,4 ;system call number (sys_write)
int 0x80
mov eax,1 ;system call number (sys_exit)
int 0x80
section .data
msg db "The result is:", 0xA,0xD
len equ $- msg
segment .bss
res resb 1
在上面的代码中:
- 为什么我们应该使用
mov [res], al
而不使用mov res, al
? - 为什么我们应该使用
mov ecx,res
而不使用mov ecx,[res]
?
****更新
我知道一个是地址,另一个是价值,但我不知道为什么......
res
是汇编程序的名称,只不过是表示值的符号。
您已经知道res
的值是它后面第一个字节的地址。
我们非正式地说res
是变量res.
在这种考虑下,指令mov res, ecx
没有意义,脱糖等同于mov <a number>, ecx
。
一个具体的例子:mov 0x12345678, ecx
- 立即进入是没有意义的。
mov ecx, res
有意义,它会加载ecx
res的地址。
我们需要它的地址而不是它的值,因为sys_write
将字节流写入文件(在您的情况下为stdout)——它与打印高级语言中的数字的函数完全不同。
实际上,乘法的个位数结果通过add al, '0'
的平均值转换为字符串。
转换后,sys_write
用于将字符串(不是数字!)写入标准输出。
由于sys_write
处理字节流,因此它期望ecx
要写入的第一个字节的地址。
您可以查看 C 包装器以检查参数的类型:第二个参数const void*
。所以必须给出一个地址(mov ecx, res
)而不是值1(mov ecx, [res]
)。
1请注意,地址和值之间的区别纯粹是语义上的,在汇编中没有这种划分。