《Unix环境高级编程》初始化一个守护进程的调用,cmd可为守护进程名。
void daemonize( const char *cmd) { int i, fd0, fd1, fd2; pid_t pid; struct rlimit r1; struct sigaction sa;
/* 设置文件模式创建屏蔽字为0,因为继承得来的文件模式创建屏蔽字可能会拒绝设置某些权限。 */ umask(0);
/* 获取最大文件描述符。 */ if (getrlimit(RLIMIT_NOFILE, &r1) < 0) { err_sys(" %s: In demonize(), can't get file limit ", cmd); }
/*
* 父进程退出,第一个子进程继承了父进程的进程组ID,但该子进程有新的进程ID,
* 这就保证了该子进程不是进程组的组长进程。为setsid准备了条件(调用setsid的进程
* 不能是进程组的组长进程)。
*/ if ((pid = fork()) < 0) { err_sys(" %s: In demonize(), can't fork ", cmd); } else if (pid != 0) { exit (0); }
/*
* 创建新会话,使第一个子进程即当前进程(因为父进程已退出)成为新会话的首进程。
* 也使第一个子进程成为新进程组的组长进程, 没有控制终端 。
*/ setsid();
/*
* 确保将来的操作不分配终端(主要针对UNIX系统V派生的系统,当会话首进程打开
* 第一个尚未与上一个会话相关联的终端设备时,UNIX系统V派生的系统会将此作为控制
* 终端分配给此会话。这假定会话首进程在调用open时没指定O_NOCTTY标志。)
*/ sa.sa_handler = SIG_IGN; sigemptyset(&sa.sa_mask); sa.sa_flags = 0; if (sigaction(SIGHUP, &sa, NULL) < 0) { err_sys(" %s:In demonize(), can't ignore SIGHUP "); } if ((pid = fork()) < 0) { err_sys(" %s:In demonize(), sencond can't fork ", cmd); } else if (pid != 0) { exit (0); }
/* 改变工作目录为根目录 */ if (chdir(" / ") < 0) { err_sys(" %s:In demonize(), can't change directory to / ", cmd) }
/* 关闭所有打开的文件描述符 */ if (r1.rlim_max == RLIM_INFINITY) { r1.rlim_max = 1024; } for (i = 0; i < r1.rlim_max; i++) { close (i); }
/* 0 1 2描述符指向/dev/null */ fd0 = open (" /dev/null ", O_RDWR); fd1 = dup(0); fd2 = dup(1);
/* 初始化系统日志 */ openlog(cmd, LOG_CONS, LOG_DAEMON); if (fd0 != 0 || fd1 != 1 || fd2 != 2) { syslog(LOG_ERR, " unexpected file descriptiors %d %d %d ", fd0, fd1, fd2); exit (1); } }