daemon 进程的创建

Table of Contents

Daemon进程生命周期长且在后台运行。编写daemon进程需要遵循哪些规则呢?

1 执行fork()函数,父进程退出,子进程继续。

执行这一步,原因有两个:

  • 父进程可能是进程组的组长,从而不能够执行后面要执行的setsid函数。子进程继承了父进程的进程组ID,一定不会是进程组组长,所以子进程一定可以执行setsid。
  • 如果daemon是从终端命令行启动的,那么父进程退出后,shell会显示shell提示符,让子进程在后台执行。

2 子进程执行下面三个步骤

(1) 修改当前目录为根目录 如果当前工作路径上包含根文件系统以外的文件系统,那么这个文件系统将不能被卸载。当然也可以改成其它合适的目录。这里使用函数chdir("/")。

(2) 调用setsid 这是为了切断与控制终端的所有关系,创建一个新的会话。此时无论终端是否发送SIGIN、SIGQUIT或者SIGTSTP或者断开,都与daemon进程无关。

(3) 使用umask(0)设置文件模式创建掩码为0 这一步的目的是让daemon进程创建文件的权限属性与shell脱离关系。因为默认情况下,进程的umask来源于父进程shell的umask。如果不执行umask(0),那么父进程的shell就会影响daemon,造成daemon每次执行的umask信息不一致。

3 再次执行fork,父进程退出,子进程继续

执行完前面两步之后,新建了会话,进程是会话的首进程,也是进程组的首进程;进程ID,进程组ID,会话ID相同;进程和终端失去联系。但是还差一步。daemon进程有可能会打开一个终端设备:

int fd = open("/dev/console", O_RDWR);

这个设备是否会成为daemon进程的控制终端,取决于两点:

  • daemon进程是不是会话的首进程。
  • 系统实现。(BSD的实现不会成为daemon的控制终端,但POSIX由具体实现决定)。

为了万无一失,需要使用fork()确保daemon不是会话的首进程。

4 关闭stdin,stdout,stderr

关闭之后应该打开/dev/null将0,1,2描述符指向它。这是为了防止后面执行0,1,2上的I/O时出现错误。

C库的daemon函数和这个流程相似,但没有第二次fork。

By .