将stdout和stderr从fork进程复制到文件

时间:2014-12-04 13:10:40

标签: c linux stdout system-calls

我需要将子进程的stdoutstderr复制到多个文件中 我知道我可以使用tee(),但我没有找到相关的例子。 现在,它只是打印到stdout和stderr。 怎么做?

 pid_t childId = fork();
    switch(childId){
    case -1:
        perror("fork() error!\n");
        return;
    case 0:

        mysignal(SIGTTOU, SIG_DFL);
        mysignal(SIGTTIN, SIG_DFL);
        mysignal(SIGCHLD, SIG_DFL);
        if(!background)
            tcsetpgrp(cterm, getpid());
        setpgid(0, 0);
        if (isWriter) close(pipefd[0]);
        if (isReader!=-1) {
            close(0);
            dup(oldChannelOut);
        }

        if(isWriter){
            close(1);
            dup(pipefd[1]);
        }
        //exec, if program is in current directory
        execv(commandArgv[0], commandArgv);
        int i = 0;
        char buf[_POSIX_MAX_PATH];
        while(path[i] != NULL){
            buf[0] = '\0';
            strcat(buf, path[i]);
            if(path[i][ strlen(path[i])-1 ] != '/'){
                buf[strlen(path[i])] = '/';
                buf[strlen(path[i])+1] = '\0';
            }
            strcat(buf, commandArgv[0]);
            execv(buf, commandArgv);
            ++i;
        }
        fprintf(stderr,"\"%s\": command not found\n",commandArgv[0]);
        exit(1);

UPD:尝试修改后,不行。哪里有问题?

fdout=open("1", O_APPEND | O_WRONLY | O_CREAT, S_IRUSR | S_IWUSR);
fderr=open("2",O_APPEND | O_WRONLY | O_CREAT, S_IRUSR | S_IWUSR);
...
case 0:
if (isReader!=-1) {
close(0);
dup(oldChannelOut);
}
dup2(fdout, 1);
dup2(fderr, 2);
close(fdout);
close(fderr);
 exec....
default:
{

if(background) {
addJob(&jobListStartB,childId,commandArgv,BACKGROUND);
if (oldChannelOut != -1){
close(oldChannelOut);
oldChannelOut = -1;
}
}
else if(!background){
if (oldChannelOut != -1){

close(oldChannelOut);
oldChannelOut = -1;
}
} 

1 个答案:

答案 0 :(得分:1)

使用teesplice将句柄中的数据写入多个其他句柄:

// Needs at least two targets, sentinel is INVALID_HANDLE_VALUE
int push_to_all(int source, ssize_t count, ...) {
    va_list vl;
    va_start(vl, count);
    int target = va_arg(vl, int), next = va_arg(vl, int);
    count = tee(source, target, count, SPLICE_F_NONBLOCK);
    if(count <= 0) { va_end(vl); return count; }
    while((target = next, next = va_arg(vl, int)) > 0) {
        tee(source, target, count, 0);
    va_end(vl);
    return splice(source, 0, target, 0, count, 0);
}