嗨,我正在尝试编写一些使用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
(省去结尾的换行符)。