gpt4 book ai didi

assembly - 如何在 GNU 汇编程序中使用字符串文字作为直接操作数(并将其移动到地址)?

转载 作者:行者123 更新时间:2023-12-04 02:35:04 24 4
gpt4 key购买 nike

我的意思是我在 NASM 中这样写的东西:

mov dword [0xA0BF17C], ' : )'

我在 GNU 汇编程序中尝试过这样的事情:

movd " : )", 0xB8000

movd $" : )", 0xB8000

movd ' : )', 0xB8000

movd " : )", $0xB8000

但是...他们都导致了这个错误:

Error: unbalanced parenthesis in operand 1.

最佳答案

GAS 仅支持将单字 rune 字作为数字。 UTF-8 多字节单个字符可以,但不能多个单独的字符。您可以执行 movb $' ', 0xB8000,但您不想对 4 个字节使用 4 条指令。

您有两个真正的选择:将单字 rune 字一起转换成一个数字,或者将其写成十六进制。 (两种方式都考虑到 x86 是小端)

# NASM   mov eax, "abcd"
movl $'a' + ('b'<<8) + ('c'<<16) + ('d'<<24), 0xB8000
movl $0x64636261, 0xB8000 # or manual ASCII -> hex, little-endian

shift/add 技巧适用于任意字节;您甚至可以制作一个 #define CPP 宏来执行此操作(采用 4 个参数)。

使用 EAX 目标而不是内存(以简化机器代码),反汇编回 GAS Intel 语法(objdump -drwC -Mintel),我们可以看到它们的组装方式相同(使用 as --32):

   0:   b8 61 62 63 64          mov    eax,0x64636261
5: b8 61 62 63 64 mov eax,0x64636261

或与您的内存目的地。同样,32 位模式,因为这会导致实模式下的#GP 错误超过 64k DS 段限制,偏移量为 0xb8000。
另请注意,机器代码中的立即字节的顺序与将作为数据存储到内存目标的顺序相同。 (如果您使用 NASM mov dst, "abcd",它们会匹配源代码顺序。

a:   c7 05 00 80 0b 00 61 62 63 64   mov    DWORD PTR ds:0xb8000,0x64636261

与 NASM 不同,GAS 不支持将多字符字 rune 字作为数字常量。不支持它们甚至会混淆 GAS 的解析器1! GAS 主要是为组装编译器输出而设计的,编译器不需要它。

GAS 仅支持(双)引用的多个字符的字符串作为 .ascii/.asciz/.string8/16/32 的参数,而不是 .byte(不像 NASM db)或作为指令的直接操作数。

如果支持,x86 AT&T 语法将是 movl $' : )', 0xB8000
不是 movd,立即操作数总是需要 $

参见 When using the MOV mnemonic to load/copy a string to a memory register in MASM, are the characters stored in reverse order?对于具有多字 rune 字的 NASM vs. MASM vs. GAS。只有 NASM 可以直观地工作。


双引号也不起作用:mov $"foo", %eax 汇编,但它的汇编与mov $foo, %eax 相同 - 将符号 foo 的地址放入寄存器。参见 relocation R_X86_64_8 against undefined symbol `ELF' can not be used when making a PIE object举个例子。


脚注 1: 因此会出现“不平衡括号”之类的错误,而不是“字 rune 字包含多个字符”之类的合理错误。

mov $'abcd', %eax

是另一个完全混淆解析器的例子。它将 b 视为局部标签的向后符号引用,例如 jmp 1b 以向后方向引用 1: 标签。但它在这里寻找的标签号是 97,即 'a' 的 ASCII 值。这完全是疯了

foo.s: Assembler messages:
foo.s:4: Error: backward ref to unknown label "97:"
foo.s:4: Error: junk `cd44%eax' after expression
foo.s:4: Error: number of operands mismatch for `mov'

所有这些都使用 as --version = GNU assembler (GNU Binutils) 2.34 进行了测试。

关于assembly - 如何在 GNU 汇编程序中使用字符串文字作为直接操作数(并将其移动到地址)?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/62357062/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com