为什么写系统调用在 linux x86_64 (nasm) 结束时打印 `%`?

Why do write syscalls print `%` at end on linux x86_64 (nasm)?

下面的 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 中无法重现)。问题是为什么会发生这种情况以及如何解决它。

发生这种情况是因为您的程序的输出没有以换行符结尾。


您的程序没有打印 %。您可以通过将其输出管道传输到 hexdump -C 或类似的方法来验证这一点。

你也可以用strace来追踪是什么系统调用了它的制作,但是那并没有显示输出,所以不排除内核神奇地添加了一个 %。 (但是你可以确定内核不会那样做。唯一可能的修改是 write 到 return 早期,没有写完整的缓冲区。这只可能有大缓冲区大小,特别是当写入缓冲区大于管道缓冲区(可能 64kiB)的管道时。


这是针对部分线路的 ZSH 功能: Why ZSH ends a line with a highlighted percent symbol?。您没有说它是突出显示的 % 符号。准确/完整的描述很重要。

(此答案的早期版本假定您在 ZSH 中使用 % 作为提示符。% 有时用作提示符,但更常见于 tcsh 等 C-shell 中,而不是Bourne 派生的 shell,例如 bash 和 zsh。)

你会从 echo -n "hello, world!" 得到完全相同的东西。例如在 bash:

peter@volta:/tmp$ echo -n 'hello world!'
hello world!peter@volta:/tmp$ 

在 ZSH 中:

volta:/tmp$ echo -n 'hello world!' 
hello world!%
volta:/tmp$      # and the % on the previous line is in inverse-video

Bash 在前台命令退出后只打印 $PS1(或运行 $PROMPT_COMMAND),无论光标位置如何。它不能直接检查程序的输出是否以换行符结尾。我猜 ZSH 使用 VT100 转义码来查询光标位置,以检测程序将光标留在行首之外的情况。

当您 cat 一个不以换行符结尾的文件或任何其他情况时,您也会遇到这种情况。


要解决此问题,请在您的邮件中包含一个换行符(ASCII 换行符,0xa):

section .rodata
    msg: db  "hello, world!", 0xa
    msglen equ $ - msg               ; use mov edx, msglen
    msgend:                          ; or  mov edx, msgend - msg

有关让汇编程序为您计算大小的更多详细信息,请参阅 How does $ work in NASM, exactly?

不要在 NASM 中省略数据标签上的 :;这是很好的风格,避免了与指令助记符和汇编程序指令的名称冲突。

NASM(但 YASM 不接受)在反引号内接受 C 风格的转义符,因此您可以编写

而不是换行符的数字 ASCII 代码
  msg: db  `hello, world!\n`