《深入理解计算机系统》Exceptional Control Flow-Exception and Processes
当发生一些异常事件(除0,段错误,缺页,算数溢出,IO请求完成)时,控制权从用户态进程交给内核。由内核的异常处理函数来执行。执行结束后,有三种方式处理:1.回到原来的地址重新执行(缺页异常);2.跳到下一条指令继续执行(中断服务程序执行完成后,或按了ctrl-c后,或执行系统调用);3.程序退出(段错误等致命错误)。
异常的机制由软件和硬件共同组成。跳转到%rip的动作是硬件完成的。但跳转后执行的内容是操作系统软件完成的。
内核中有一个异常表。不同异常对应的处理函数的地址放在异常表中,每一个value对应一个异常index。
2. 故障faults:如果异常是可修复的(如缺页异常),则会修复完成后,返回原地址重新执行;如果异常是不可修复的(访问空地址等),则走abort流程。
3. 中止aborts:不可恢复的异常(非法指令,dram校验错误等很底层的错误,程序将遭到破坏)。
然后将最多6个参数依次赋值给%rdi,%rsi,%rdx,%r10,%r9,%r8
执行完成后,返回值被写在%rax中。正数为正常,负数代表出错,需要查看当前errno。
进程上下文包括:通用cpu寄存器、浮点寄存器、程序计数器、用户栈、状态寄存器内核栈(保存了当前进程的各种信息如当前task_struct,打开的文件表等)。
并发的概念:某一个时间段内,都在执行。(逻辑流在时间上重叠)
子进程退出后,需要父进程调用wait或waitpid,来回收子进程的资源。否则子进程会变为僵尸进程。
如果子进程退出是父进程已经退出(无父进程),则init进程会直接回收子进程的资源,避免其称为僵尸进程。
wait系统调用,会阻塞直到其中一个子进程退出。传入的int*的整型会被修改为子进程的退出状态。
int execve(char* filename, char* argv[], char* envp[]);
它会覆盖子进程所有地址空间,只保留了pid,打开的文件的信息。