后台进程返回后再次提示用户输入

时间:2012-05-27 20:36:24

标签: c unix process background-process

我正在尝试构建一个作业控件shell,目前正在处理后台进程。在这个程序中,我分叉一个子进程来处理每个后台进程,我在fork之后立即得到一个提示。但问题是,当后台进程返回时,它会再次打印出提示消息。有人可以帮我解决这个问题吗?谢谢##

char prompt[] = "myShell";
set_sighandler_SIGCHLD();  //wait for background process return
while(1){
       char cmd[BUFFERSIZE] = "";
       write(1, prompt, sizeof(prompt));
       if(read(0, cmd, 1024) <= 1)){
              //parse command
              //and execute
       }
}
//Here is the background process.
int put_to_background(int (*func)(char** arg), char ** cmd){
     pid_t pid;
     if((pid = fork()) < 0){
         perror("fork");
         return -1;
     }
     else if(pid == 0){
         func(cmd);     //call the function the execute the command
         _exit(0);
     }
     else{
         if(setpgid(pid, 0)){
             perror("setpgid");
             return -1;
         }
         printf("running: %d\n", pid);

     }

在我调用函数后,它立即打印出提示“myShell”(这就是我所期望的),但是在后台进程返回后再次打印出来。

我还在处理信号处理......

 JobList list;
 void sighandler(int signum, siginfo_t *sip, void *ucp){
     if(signum == SIGCHLD){
        pid_t pid;
        while((pid = waitpid(-1, NULL, WNOHANG)) > 0){
           Job * job = (Job*)malloc(sizeof(Job));
           job->pid = pid;
           insert(&list, job);  
       }
     }
     else if(signum == SIGTTOU){
       printf("SIGTTOU: pid = %d\n", (int) sip->si_pid);
     }
     else if(signum == SIGTTIN){
      printf("SIGTTIN: pid = %d\n", (int) sip->si_pid);
     }
 }  

int set_sighandler_SIGCHLD(){

struct sigaction sa;
sigemptyset(&sa.sa_mask);

sigaddset(&sa.sa_mask, SIGCHLD); 
sigaddset(&sa.sa_mask, SIGTTIN);
sigaddset(&sa.sa_mask, SIGTTOU);
sa.sa_sigaction = sighandler;
sa.sa_flags = SA_SIGINFO;
//sigprocmask(SIG_BLOCK, &sa.sa_mask, NULL);

if(sigaction(SIGCHLD, &sa, NULL)){
    perror("sigaction");
    return -1;
}
if(sigaction(SIGTTOU, &sa, NULL)){
    perror("sigaction");
    return -1;
}
if(sigaction(SIGTTIN, &sa, NULL)){
    perror("sigaction");
    return -1;
}
return 0;

}

1 个答案:

答案 0 :(得分:0)

read循环被信号中断,因此您打印提示。 (也就是说,当信号到达时,read会立即返回。)您可以检查读取是否返回-1并且由于EINTR而返回,或者尝试将SA_RESTART设置为信号处理程序。请注意SA_RESTART并不总是有效(依赖于实现,各种实现中存在许多错误),最好避免使用IMO。