linux - 在汇编中使用 printf 会导致管道传输时输出为空,但可以在终端上使用

标签 linux assembly printf x86-64 glibc

我尝试在汇编代码中使用 printf,这是一个最小的示例,应该只将 hello 打印到标准输出:

.section  .rodata
hello:
    .ascii  "hello\n\0"
.section .text
    .globl _start        
_start:
    movq $hello, %rdi     # first parameter
    xorl %eax, %eax       # 0 - number of used vector registers
    call printf        
#exit   
    movq $60, %rax
    movq $0, %rdi
    syscall

我用它构建

gcc -nostdlib try_printf.s -o try_printf -lc

当我运行它时,它似乎起作用了:打印出字符串hello,退出状态为0:

XXX$ ./try_printf
hello
XXX$ echo $?
0
XXX$

但是当我 try catch 文本时,很明显,有些东西无法正常工作:

XXX$ output=$(./try_printf) 
XXX$ echo $output

XXX$ 

变量output应具有值hello,但为空。

我使用printf有什么问题?

最佳答案

在使用 printf 等 stdio 函数后,使用 call exit 而不是原始的 _exit 系统调用。这会在进行 write 系统调用之前刷新 stdio 缓冲区(exit_group 系统调用)。

(或者,如果您的程序定义了 main 而不是 _start ,则从 main 返回相当于调用 exit 。您不能从 ret 调用 _start 。)调用 fflush(NULL) 也应该有效。

<小时/>

正如 Michael 所解释的,动态链接 C 库是可以的。 "Programming bottom up" 书中也是这样介绍的(参见第 8 章)。

但是,为了结束程序而不是绕过它,从 C 库调用 exit 很重要,这就是我通过调用 exit-syscall 错误所做的事情。正如 Michael 所暗示的,退出会执行很多 clean up 操作,例如刷新流。

这就是发生的事情:正如 here 所解释的,C 库按如下方式缓冲标准流:

  1. 没有标准错误缓冲。
  2. 如果标准输出/输入是终端,则它是行缓冲的。
  3. 如果标准输出/输入不是终端,则它是完全缓冲的,因此在原始退出系统调用之前需要刷新。

哪种情况适用是在第一次为流调用 printf 时决定的。

因此,如果直接在终端中调用printf_try,则可以看到程序的输出,因为hello末尾有\n(在行缓冲模式下触发刷新)并且它是一个终端,也是2。案例。

通过 printf_try 调用 $(./printf_try) 意味着 stdout 不再是终端(实际上我不知道它是临时文件还是内存文件),因此第 3 种情况有效 - 需要显式刷新,即调用 C- exit

关于linux - 在汇编中使用 printf 会导致管道传输时输出为空,但可以在终端上使用,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/56247270/

相关文章:

linux - 可以使用 kmalloc() 分配的最大大小是否取决于 free_area[] 的大小?

java - crontab 在 linux 上运行 java shell 脚本

gcc - 如何在 x64 计算机上使用 gcc 编译 x86 汇编代码

c - 实现可变参数检查自定义字符串格式化函数

java - printf,说,getName setName

c - 简单 SSL 连接的 libldap openldap 设置选项失败

java - rsync 以有限的深度递归

c++ - 'printf' 与 C++ 中的 'cout'

assembly - 使用 GDB 在本地标签处中断 NASM 程序集

linux - SYSENTER/SYSEXIT 与 INT 0x80