Fork——分道扬镳的开始
刺猬@http://blog.csdn.net/littlehedgehog
在copy_process的函数中有这样一段代码:
赵博注释道: 这就是当fork()返回时新进程会返回0的原因所在! 呃,很多人不明白这段注释含义,fork返回0?我怎么没看见fork在新进程返回了? 这里我介绍下我的看法权当是抛砖引玉,以后有新手路过浏览拙文,若有一丝领悟,此文也达到它目的了。
话说这个fork返回值问题,赵博论坛(http://oldlinux.org)也有不少人讨教。其实追根溯源,fork调用还得到unix那个年代说起。
且看代码:
试问,此程序打印结果?
打印出两个helllo,若是初次调用fork可能有稍许惊讶,不过细细想来便知: 当前进程运行到fork时候,产生了一个新进程,我们知道新的进程几乎全部是拷贝父进程所有内容,最为重要的便是进程的TSS段。cs、eip均是拷贝的父进程内容。请看这段内核代码(部分):
注意我加粗的两句,分别是给子进程cs、eip赋值的,全部是父进程相对应的值。那么当子进程运行的时候,eip还是指向的是父进程的代码,也就是说子进程运行时仍执行的是父进程代码,即是打印hello。那么我们上面可以看到父进程和子进程同时会执行printf这条代码,难怪会有两行hello的打印。
Fork的调用问题我们就探讨到这儿,下面我们来看看赵博为什么说这就是当fork()返回时新进程会返回0的原因所在!
由上面的例子我们可以看到其实当fork调用后,父进程和子进程都会按照原来的代码流执行下去。我们来设想一下: 现在假如是父进程在调用fork后,父进程执行,fork的返回值就是子进程的进程号。fork这段代码是段嵌入式汇编宏函数,也就是syscall0. 大家可以参照《完全剖析》p235,恕不细表。而如果是子进程被调度的话,我们看到由于前面我们设置了tss:
此时的tss中的eax设为了0,一旦子进程被调度,cpu会自动加载tss,然后eax会被自动设为0,而一般情况下我们是把eax当作程序的返回值存放地点的,所以这里类似于返回了0。
这样fork在父进程返回的是子进程的进程号,而在子进程“返回”的是0。我们就可以根据此区分当前是父进程在运行,还是子进程在运行了。比如下例: