为什么写系统调用在 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`
下面的 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`