我正在尝试编写调用函数以在我的伪壳中调用,但是当简单(一个命令)和管道情况工作时,当我尝试重定向时,它给了我一个写错误,说文件描述符是错误的。 argv []包含第一个命令的名称和参数,而argv []包含第二个命令的相同内容;
int invoke(char *argv1[], char *argv2[], type_c tipo)
{
int pid, pfd[2], fid;
switch(tipo)
{
case(_SIMPLE):
pid=fork();
switch(pid) {
case 0:
execvp(argv1[0],argv1);
}
break;
case(_PIPE):
pipe(pfd);
pid=fork();
switch(pid) {
case 0:
close(1);
dup(pfd[1]);
close(pfd[0]);
close(pfd[1]);
execvp(argv1[0],argv1);
}
pid=fork();
switch(pid) {
case 0:
close(0);
dup(pfd[0]);
close(pfd[0]);
close(pfd[1]);
execvp(argv2[0],argv2);
}
close(pfd[0]);
close(pfd[1]);
break;
case(_REDIRECT):
pid=fork();
switch(pid) {
case 0:
close(fileno(stdout));
fid=open(argv2[0],O_CREAT|O_WRONLY);
dup(fid);
close(fid);
execvp(argv1[0],argv1);
}
break;
}
return pid;
}
测试它,在main()函数中,我编写了这段代码:
char *argv[2][3];
argv[0][0]="ls";
argv[0][1]="-l";
argv[0][2]=NULL;
argv[1][0]="test";
argv[1][1]=NULL;
pid=invoke(argv[0],argv[1],_REDIRECT);
那么,我该怎么办? :/ thanx提前全部
答案 0 :(得分:0)
如果是_REDIRECT
,
1
。 file
。这将直接打开描述符号1
。 dup
这个新的文件描述符,它将在描述符3
上打开相同的内容(2
用于stderror
)。fid
上的1
。 ls
程序然后尝试打印到fd 1
,在这种情况下已经关闭,从而使其成为bad-file descriptor.
有两种方法可以解决这个问题。
不需要额外的dup
,您的代码也能完美运行。
close(fileno(stdout));
int fid=open(argv2[0],O_CREAT|O_WRONLY);
execvp(argv1[0],argv1);
为了让我们的生活更轻松,还有另一个功能dup2
,它会隐式关闭target fd
,并将其作为fd
的副本。
int fid=open(argv2[0],O_CREAT|O_WRONLY);
dup2(fid,1);
close(fid);
execvp(argv1[0],argv1);