嗨,我正在尝试编写一些使用printf打印给定字符串的汇编代码。我在.data节中声明使用之前的字符串,并且测试示例如下所示:

extern printf
extern fflush

LINUX        equ     80H      ; interupt number for entering Linux kernel
EXIT         equ     60       ; Linux system call 1 i.e. exit ()

section .data
    outputstringfmt: db "%s", 0
    sentence0: db "Hello\nWorld\n", 0


segment .text
    global  main


main:
    mov r8, sentence0
    push r8
    call print_sentence
    add rsp, 8
    call os_return

print_sentence:
    push rbp
    mov rbp, rsp
    push r12
    mov r12, [rbp + 16]
    push rsi
    push rdi
    push r8
    push r9
    push r10
    mov rsi, r12
    mov rdi, outputstringfmt
    xor rax, rax
    call printf
    xor rax, rax
    call fflush
    pop r10
    pop r9
    pop r8
    pop rdi
    pop rsi
    pop r12
    pop rbp
    ret

os_return:
    mov  rax, EXIT      ; Linux system call 1 i.e. exit ()
    mov  rdi, 0     ; Error code 0 i.e. no errors
    syscall     ; Interrupt Linux kernel 64-bit

然后,我进行如下编译:
nasm -f elf64 test.asm; gcc -m64 -o test test.o

最后运行:
./test

我的输出如下:
Hello\nWorld\n

我真的不想将句子0分解为以下内容:
sentence0: db "Hello", 10, 0
sentence1: db "World", 10, 0

然后两次调用打印文件。有更好的方法吗?

提前致谢!

最佳答案

NASM接受单引号('...')或双引号("...")的字符串,它们是等效的,并且不提供任何转义符;或反引号(`...`),它们提供对C样式转义符的支持,而这正是您想要的。

(请参阅section 3.4.2, "Character Strings", in the documentation。)

要在内存中的数据中获取实际的ASCII换行符,而不是文字反斜杠n:

sentence0: db `Hello\nWorld\n`, 0

或手动执行:
sentence0: db 'Hello', 10, 'World`, 10, 0

YASM(另一个NASM语法汇编程序)不接受反引号,因此手动选项是您的唯一选择。

顺便说一句,如果格式字符串中没有任何实际格式,则可以调用puts而不是printf(省去结尾的换行符)。

07-24 09:44
查看更多