execve("/ bin/sh",0,0); 在管道里

A. *_*son 10 c shell freebsd pipe execve

我有以下示例程序:

#include <stdio.h>

int
main(int argc, char ** argv){
    char buf[100];

    printf("Please enter your name: ");
    fflush(stdout);
    gets(buf);
    printf("Hello \"%s\"\n", buf);

    execve("/bin/sh", 0, 0);
}
Run Code Online (Sandbox Code Playgroud)

我当我没有任何管道运行时,它应该工作,并返回一个shpromt:

bash$ ./a.out
Please enter your name: warning: this program uses gets() which is unsafe.
testName
Hello "testName"
$ exit
bash$
Run Code Online (Sandbox Code Playgroud)

但这不适用于管道,我想我知道为什么会这样,但我无法找到解决方案.示例运行波纹管.

bash$ echo -e "testName\npwd" | ./a.out
Please enter your name: warning: this program uses gets() which is unsafe.
Hello "testName"
bash$
Run Code Online (Sandbox Code Playgroud)

我认为这与以下事实有关:以一种接收EOF 的方式gets清空并且stdin/bin/sh没有错误消息的情况下退出.

但是我如何解决这个问题(如果可能的话,不修改程序,如果没有gets,则不删除),这样即使我通过管道提供输入,我也会得到一个承诺?

PS我在FreeBSD(4.8)机器DS上运行它

Ada*_*man 10

您可以在不进行任何修改的情况下运行程序:

(echo -e 'testName\n'; cat ) | ./a.out
Run Code Online (Sandbox Code Playgroud)

这样,您可以确保程序的标准输入不会在echo输出之后结束.相反,cat继续为您的程序提供输入.后续输入的来源是您的终端,因为这是cat读取的地方.

这是一个示例会话:

bash-3.2$ cc stdin_shell.c 
bash-3.2$ (echo -e 'testName\n'; cat ) | ./a.out 
Please enter your name: warning: this program uses gets(), which is unsafe.
Hello "testName"
pwd
/home/user/stackoverflow/stdin_shell_question
ls -l
total 32
-rwxr-xr-x  1 user  group  9024 Dec 14 18:53 a.out
-rw-r--r--  1 user  group   216 Dec 14 18:52 stdin_shell.c
ps -p $$
  PID TTY           TIME CMD
93759 ttys000    0:00.01 (sh)
exit

bash-3.2$
Run Code Online (Sandbox Code Playgroud)

请注意,由于shell的标准输入未连接到终端,因此sh认为它不是以交互方式执行的,因此不会显示提示.但是,您可以正常键入命令.