为什么写入系统调用在linux x86_64(nasm)的末尾打印`%`?

时间:2018-01-12 11:40:03

标签: linux nasm x86-64 zsh system-calls

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

1 个答案:

答案 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代码
: