《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);
}
}