daemon编程规则(为什么要fork两次)

文章出处:http://blog.csdn.net/jeffgeng/article/details/5798338

                    http://roz1987.blog.163.com/blog/static/116392958201011156149507/

 

    在linux中,每一个系统与用户进行交流的界面称为终端,每一个从此终端开始运行的进程,都会依附于这个终端,这个终端就称为这些进程的控制终端,当控制终端被关闭时,相应的进程都会自动关闭。 

 

    在编写精灵进程程序时需要遵循一些基本规则,以便防止产生并不希望的交互作用。下面先说明这些规则,然后是一个按照规则编写的函数daemon_init。

    

    1.  首先做的是调用fork,然后使父进程exit。这样做实现了下面几点:

    第一:如果该精灵进程是由一条简单shell命令启动的,那么使父进程终止使得shell认为这条命令已经执行完成;(之后的所有工作都在子进程中完成,而用户在Shell终端里则可以执行其他命令,从而在形式上做到了与控制终端的脱离。)

    第二:在Linux中父进程先于子进程退出会造成子进程成为孤儿进程,而每当系统发现一个孤儿进程,就会自动由1号进程(init)收养它,这样,原先的子进程就会变成init的子进程;

    第三:子进程继承了父进程的进程组ID,但具有一个新的进程ID,这就保证了子进程不是一个进程组的首进程。这对于下面就要做的setsid调用是必要的前提条件

   

    2.  调用setsid以创建一个新对话期。于是执行9.5节中列举的三个操作,使调用进程:

    第一:成为新对话期的首进程;

    第二:成为一个新进程组的首进程;

    第三:没有控制终端。

    在SVR之下,有些人建议在此时再调用fork,并使父进程终止。第二个子进程作为精灵进程继续运行。这样就保证了该精灵进程不是对话期的首进程,于是按照SVR规则(见9.6节)可以防止它取得控制终端。另一方面,为了避免取得控制终端,无论何时打开一个终端设备都要指定O_NOCTTY

 

    3.  将当前工作目录更改为根目录。父进程继承过来的当前工作目录可能在一个装配的文件系统中。因为精灵进程通常在系统再引导之前是一直存在的,所以如果精灵进程的当前工作目录在一个装配文件系统中,那么该文件系统就不能被拆卸。

    另外,某些精灵进程可能会把当前工作目录更改到某个指定位置,在此位置做它们的工作。

    例如,行式打印机假脱机精灵进程常常将其工作目录更改到它们的spool目录上。

 

    4.  将文件方式创建屏蔽字设置为0。由继承得来的文件方式创建屏蔽字可能会拒绝设置某些许可权。例如,若精灵进程要创建一个组可读、写的文件,而继承的文件方式创建屏蔽字,屏蔽了这两种许可权,则要求的组可读、写就不能起作用。

 

    5.  关闭不在需要的文件描述符。这样使精灵进程就不再持有从其父进程继承来的某些文件描述符(父进程可能是shell进程,或某个其他进程)。但是,究竟关闭那些描述符则与具体的精灵进程有关,所以在下面的例子中不包含此步骤。可以使用程序2-3中的open_max函数来决定最高文件描述符值,并关闭直到该值的所有描述符。

你可能感兴趣的:(编程,工作,linux,shell,终端,2010)