shlab

深入理解计算机系统ch8---shlab

感觉不熟悉异常控制流中章节中的内容时,写起来比较费劲,代码量不大但是要理清楚什么时候该阻塞信号,什么时候该取消阻塞。

# eval函数主要参照书中的方式
void eval(char *cmdline) 
{
    char *argv[MAXARGS];
    char buf[MAXLINE];  //这个buf感觉可以不需要
    int bg;
    pid_t pid;

    sigset_t mask_all, mask_one, prev;
    sigfillset(&mask_all);
    sigemptyset(&mask_one);  
    sigaddset(&mask_one, SIGCHLD);

    strcpy(buf, cmdline);
    bg = parseline(buf, argv);
    if (argv[0] == NULL)
        return;

    if (!builtin_cmd(argv)) {  //如果不是内置命令

        sigprocmask(SIG_BLOCK, &mask_one, &prev);
        if ((pid = fork()) < 0)
            unix_error("fork error");

        if (pid == 0) { /* child process run the jobs */
            sigprocmask(SIG_SETMASK, &prev, NULL);
            setpgid(0, 0);  //这里将每个Job设置成单独一个组,且组ID等于其PID,后面发送信号的时候就比较方便
            if (execve(argv[0], argv, environ) < 0) {
                printf("%s: Command not found.\n", argv[0]);
                exit(0);
            }
        } 

        sigprocmask(SIG_BLOCK, &mask_all, NULL);

        /* father process wait for foreground job to terminate */
        if (!bg) {
            addjob(jobs, pid, FG, buf);
            sigprocmask(SIG_SETMASK, &prev, NULL); //这里解除阻塞,我的想法是防止waitfg函数执行时,因为信号的阻塞导致循环的条件一直成立从而
            /* wait */
            waitfg(pid);
        } else {
            addjob(jobs, pid, BG, buf);
            int jid = pid2jid(pid);
            printf("[%d] (%d) %s", jid, pid, buf);
            sigprocmask(SIG_SETMASK, &prev, NULL);
        }
    }

    return;
}
int builtin_cmd(char **argv) 
{

    sigset_t mask_all, prev;
    sigfillset(&mask_all);
    if (!strcmp(argv[0], "quit")) {
        exit(0);
    }

    if (!strcmp(argv[0], "jobs")) {
        sigprocmask(SIG_BLOCK, &mask_all, &prev);
        listjobs(jobs);  //涉及到全局变量的读取,阻塞信号
        sigprocmask(SIG_SETMASK, &prev, NULL);
        return 1;
    }

    if (!strcmp(argv[0], "fg") || !strcmp(argv[0], "bg")) {
        do_bgfg(argv);
        return 1;
    }

    return 0;     /* not a builtin command */
}
void do_bgfg(char **argv) 
{
    char *jobstr = argv[1];
    if (jobstr == NULL) {
        printf("%s command requires PID or %%jid argument\n", argv[0]);
        return;
    }

    int isjid = 0, jid; //isjid用来判断fg,bg命令之后跟的是%jid还是PID
    pid_t pid;
    if (jobstr[0] == '%') 
        isjid = 1;

    jobstr += isjid;
    if (!(strspn(jobstr, "0123456789") == strlen(jobstr))) {
        printf("%s: argument must be a PID or %%jid\n", jobstr);
        return;
    }

    sigset_t mask, prev;
    sigfillset(&mask);
    sigprocmask(SIG_BLOCK, &mask, &prev);

    if (isjid) {
        jid = atoi(jobstr);
    }
    else {
        pid = atoi(jobstr);
        jid = pid2jid(pid);
    }

    struct job_t *job = getjobjid(jobs, jid);
    if (job == NULL) {
        if (isjid) {
            printf("%%%d: No such Job\n", jid);
            sigprocmask(SIG_SETMASK, &prev, NULL);
            return;
        }
        printf("%d: No such Process\n", pid);
        sigprocmask(SIG_SETMASK, &prev, NULL);
        return;
    }

    if (argv[0][0] == 'b') {
        if (job->state == ST) {
            job->state = BG;
            kill(-(job->pid), SIGCONT);
            printf("[%d] (%d) %s", job->jid, job->pid, job->cmdline);
            sigprocmask(SIG_SETMASK, &prev, NULL);
        }
    }
    
    if (argv[0][0] == 'f') {
        if (job->state == ST) {
            job->state = FG;
            kill(-(job->pid), SIGCONT);
            sigprocmask(SIG_SETMASK, &prev, NULL); //同上
            waitfg(job->pid);
        }
        else if (job->state == BG) {
            job->state = FG;
            sigprocmask(SIG_SETMASK, &prev, NULL); //理由和上面相同
            waitfg(job->pid);
        }
    }

    return;
}
# waitfg的写法,作业给的提示中说是使用忙等待,也就是循环
# 使用sigsuspend的话,需要额外增加一个全局变量
# 参考了其他博客之后,感觉使用循环更好
void waitfg(pid_t pid)
{
    /*
    sigset_t mask;
    sigemptyset(&mask);
    fg_pid = 0;
    while (!fg_pid)
        sigsuspend(&mask);

    */
    sigset_t mask, prev;
    sigfillset(&mask);
    sigprocmask(SIG_BLOCK, &mask, &prev);
    struct job_t *job = getjobpid(jobs, pid);  //每次要从全局变量jobs中读取内容是都要阻塞信号
    sigprocmask(SIG_SETMASK, &prev, NULL);
    while (job != NULL && job->state == FG) {  //判断pid对应的job是否为空,为空则不用再等待,如果工作非空,但是其状态不再为FG时,也退出等待
        sigfillset(&mask);
        sigprocmask(SIG_BLOCK, &mask, &prev);
        job = getjobpid(jobs, pid);
        sigprocmask(SIG_SETMASK, &prev, NULL);
    }

    return;
}
# 关键在于识别好发送SIGCHLD信号的原因,也就是进程是停止(stopped),终止(terminated)还是自然退出的(exited)
# waitpid(-1, &status, WNOHANG | WUNTRACED)如果子进程中有任何的进程停止,终止或者自然退出,都会立即返回。然后根据state来判断返回的pid对应的进程是因为什么原因导致SIGCHLD信号产生的
void sigchld_handler(int sig) 
{
    int olderrno = errno;    
    pid_t pid;
    int status;

    while (((pid = waitpid(-1, &status, WNOHANG | WUNTRACED)) > 0)) {
        sigset_t mask_all, prev;
        sigfillset(&mask_all);
        sigprocmask(SIG_BLOCK, &mask_all, &prev);
        if (WIFSIGNALED(status)) {  //因为ctrl + c导致的进程停止
            pid_t curr_fg_pid = fgpid(jobs);
            if (pid == curr_fg_pid) {
                int jid = pid2jid(pid);
                printf("Job [%d] (%d) Terminated by signal %d\n", jid, pid, WTERMSIG(status));
                deletejob(jobs, pid);
            }
        } 
        else if (WIFEXITED(status)) {   /* 处理正常终止的进程 */
            deletejob(jobs, pid);
        }
        else if (WIFSTOPPED(status)) {  /* ctrl + z导致的进程停止 */
            struct job_t *job = getjobpid(jobs, pid);
            job->state = ST;
            int jid = pid2jid(pid);
            printf("Job [%d] (%d) Stopped by signal %d\n", jid, pid, WSTOPSIG(status));
        }
        sigprocmask(SIG_SETMASK, &prev, NULL);
    }
    errno = olderrno;

    return;
}
void sigint_handler(int sig) 
{   /* remove FG job and terminate the process */
    int olderrno = errno;
    sigset_t mask_all, prev;
    sigfillset(&mask_all);
    sigprocmask(SIG_BLOCK, &mask_all, &prev);
    pid_t pid = fgpid(jobs);
    if (pid) {  //判断一下是不是前台进程,不是的话就不生效
        kill(-pid, sig); //这里是-pid,与之前的将每个工作单独放到一个进程组对应起来
    }
    sigprocmask(SIG_SETMASK, &prev, NULL);
    errno = olderrno;

    return;
}

# sigtstp_hadler类似
void sigtstp_handler(int sig) 
{
    int olderrno = errno;
    sigset_t mask_all, prev;
    sigprocmask(SIG_BLOCK, &mask_all, &prev);

    pid_t pid = fgpid(jobs);
    if (pid) {
        kill(-pid, sig);
    }
    sigprocmask(SIG_SETMASK, &prev, NULL);
    errno = olderrno;

    return;
}

参考博客链接
https://www.cnblogs.com/jjppp/p/14686584.html
https://blog.csdn.net/xiaolian_hust/article/details/80087376

上一篇:如何查看某个端口被占用


下一篇:CSS列表样式