【发布时间】:2021-12-18 12:46:33
【问题描述】:
在 64 位 x86 汇编 nasm 中,如何将单个字节从寄存器移动到 .data 节中定义的内存位置?
我知道这行得通
global _main
section .data
quotient db 0x0, 0x0, 0x30, 0xa ; 3 digit + newline
remainder db 0x0, 0x0, 0x30, 0xa; 3 digit + newline
section .text
_main:
mov rax, 0x2000004
mov rdi, 1
mov rsi, quotient
mov rdx, 8
syscall ; outputs 0 /newline 0 /newline as expected
exit:
mov rax, 0x2000001
mov rdi, 0
syscall
我也知道你可以使用 byte 关键字设置单个字节
mov rbx, quotient ; move location of quotient into memory
mov [rbx], byte 0x31 ; change first byte to ascii '1'
这将按预期输出 1 0 /newline 0 /newline
但是假设我想将其设置为寄存器值
mov rbx, quotient ; move location of quotient into memory
mov r8, 0x31 ; set another register to ascii '1'
mov [rbx], r8 ; write the register value into memory
这确实输出 1,但它似乎覆盖了内存中的每个字节,因此换行符被删除。
我的问题是如何将单个“字节”从寄存器写入内存位置?
【问题讨论】:
-
您可能希望看到这个:wiki.cdot.senecacollege.ca/wiki/…。特别是要移动一个字节,您可以这样做
mov [rbx], r8b -
哦,哇,谢谢,我从 64 位寄存器的文章中猜测,您每次都将 8 个字节写入内存,并在寄存器名称后加一个 b 字节?但是,当我尝试使用 rcxb 为 rcx 之类的寄存器添加后缀时,nasm 给了我符号“rcxb”未定义,这是我的汇编程序有问题还是只有编号的寄存器才有这种能力?
-
正确。如果源是 64 位寄存器,则将完整的 64 位数据写入目标内存地址,覆盖您真正想要写入的一个字节之后的 7 个额外字节。
-
啊,我发现另一篇文章来自谷歌搜索后缀的东西cs.brown.edu/courses/cs033/docs/guides/x64_cheatsheet.pdf 似乎使用不同的寄存器,后缀不同,可以从中选择不同的字节。所以 rcx 选择第一个字节变成 cl