Unix守护进程的创建示例

《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);
	}
}
原文地址:https://www.cnblogs.com/helloweworld/p/2769003.html