以下hello-world程序在打印字符串的末尾显示%
符号。为什么这样,我该如何删除它?
这是我的计划:
section .data
msg db "hello, world!"
section .text
global _start
_start:
mov rax, 1 ; syscall 1 (write)
mov rdi, 1 ; arg 1 = 1 (stdout)
mov rsi, msg ; arg 2 = msg ("hello, world!")
mov rdx, 13 ; arg 3 = 13 (char count)
syscall ; call write
mov rax, 60 ; syscall 60 (exit)
mov rdi, 0 ; arg 1 = 0 (OK)
syscall ; call exit
以下是运行可执行文件时的输出:hello, world!%
提前致谢。
编辑:它似乎是由zsh引起的(在bash中不可重现)。问题是为什么会发生这种情况以及如何解决它。
答案 0 :(得分:3)
这是因为您的程序输出不以换行符结束。
您的程序未打印%
。您可以通过将其输出汇总为hexdump -C
或类似的来验证这一点。
你也可以使用strace
来跟踪它所调用的系统,但是没有显示输出,所以它不会排除内核神奇地添加{{ 1}}。 (但你可以肯定内核不会这样做。唯一可能的是%
提前返回,而没有编写完整的缓冲区。这只有大缓冲区大小,特别是在写入时缓冲区大于管道缓冲区的管道(可能是64kiB)。
这是部分线条的ZSH功能: Why ZSH ends a line with a highlighted percent symbol?。你没有说它是一个突出显示的%符号。准确/完整的描述很重要。
(这个答案的早期版本假设您在ZSH中使用write
作为提示。%
有时用作提示,但更常见于C-shell,如tcsh,而不是Bourne-派生的shell,如bash和zsh。)
你会从%
得到完全相同的东西。例如在bash:
echo -n "hello, world!"
在ZSH:
peter@volta:/tmp$ echo -n 'hello world!'
hello world!peter@volta:/tmp$
Bash只是在前台命令退出后打印volta:/tmp$ echo -n 'hello world!'
hello world!%
volta:/tmp$ # and the % on the previous line is in inverse-video
(或运行$PS1
),无论光标位置如何。它无法直接检查程序的输出是否以换行符结束。我猜ZSH使用VT100转义码来查询光标位置,以检测程序将光标留在行开头的情况。
当你$PROMPT_COMMAND
一个不以换行符结尾的文件或任何其他案例时,你也会得到这个。
要解决此问题,请在邮件中添加换行符(ASCII换行符,0xa):
cat
有关让汇编程序为您计算大小的详细信息,请参阅How does $ work in NASM, exactly?。
不要忽略NASM中数据标签上的section .rodata
msg: db "hello, world!", 0xa
msglen equ $ - msg ; use mov edx, msglen
msgend: ; or mov edx, msgend - msg
;它的风格很好,避免了与指令助记符和汇编程序指令的名称冲突。
NASM(但不是YASM)接受反引号内的C风格的esacapes,所以你可以编写
而不是换行符的数字ASCII代码: