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

	}

}

你可能感兴趣的:(unix)