Linux/C: перенаправление каналов в STDIN/STDOUT
У меня были проблемы с имитацией сценария оболочки "env | grep HOME" с помощью C-программы. Я обнаружил, что комментирование строки 29 решило эту проблему, но я не совсем уверен, почему! На другом вопросе я читал, что это произошло потому, что dup2() закрывал fd у потомка, но страница руководства не указывает на это. Кто-нибудь может дать мне определенную причину и помочь мне понять это поведение? Спасибо!
#include <errno.h>
#include <stdio.h>
#include <unistd.h>
#include <sys/types.h>
int main(void){
pid_t childpid;
int fd[2];
if(pipe(fd) == -1){ /*setup a pipe*/
perror("Failed to setup pipeline");
return 1;
}
if((childpid = fork()) == -1){ /*fork a child*/
perror("Failed to fork a child");
return 1;
}
if(childpid == 0){ /*env is the child*/
if(dup2(fd[1],STDOUT_FILENO)==-1)
perror("Failed to redirect stdout of env");
else if(close(fd[0] == -1)) /*close unused file descriptor*/
perror("Failed to close extra pipe descriptors on env");
else{
execl("/usr/bin/env", "env", NULL); /*execute env*/
perror("Failed to exec env");
}
return 1;
}
if(dup2(fd[0],STDIN_FILENO)==-1) /*grep is the parent*/
perror("Failed to redirect stdin of grep");
//else if(close(fd[1]==-1))
//perror("Failed to close extra pipe file descriptors on grep");
else{
execl("/bin/grep", "grep", "HOME", NULL); /*execute "grep HOME"*/
perror("Failed to exec grep");
}
return 1;
}
1 ответ
Решение
Я нашел вашу ошибку. Вот что выходит правильно для меня. Это распространенная ошибка:
...
else if (close(fd[0]) == -1) /*close unused file descriptor*/
...
else if(close(fd[1]) == -1)
...
Изначально вы делали закрытие дескриптора файла на логическое значение fd[x] == -1
и что вы хотели сделать, это проверить -1
в возвращаемом значении close()
,