基本概念
父进程ID为0的各进程通常是内核进程,他们作为系统引导装入过程的一部分而启动。
init是一个由内核载引导装入时启动的用户层次的命令。
常见守护进程:
- kthreadd,特殊的内核进程,来创建其它内核进程
- kswapd,内存换页守护进程
- rpcbind,提供RPC程序号映射为网络端口号的服务
- inetd,侦听网络接口,取得来自网络对各种网络服务器的请求
- cron,在定期安排的时期和时间执行命令
内核守护进程以无控制终端方式启动。用户层守护进程缺少控制终端可能是调用了setsid的结果。
大多数用户层守护进程都是进程组的组长进程和会话的受进程,而且是这些进程组和会话中的唯一进程。
用户层守护进程的父进程市init进程。
编程规则
- 首先要调用umask将文件模式创建屏蔽字设置为一个已知值(通常是0)
- 调用fork,然后使父进程exit
- 调用setsid创建一个新会话
- 将当前工作目录改为根目录
- 关闭不再需要的文件描述符。
- 某些守护进程打开/dev/null使其具有文件描述符0,1,2
例子程序daemonize中有两次fork:
//...
// Become a session leader to lose controlling TTY.
if ((pid = fork()) < 0)
err_quit("%s: con't fork", cmd);
else if (pid != 0)
exit(0);
setsid();
// Ensure future opens won't allocate controlling TTYS.
sa.sa_handler = SIG_IGN;
sigemptyset(&sa.sa_mask);
sa.sa_flags = 0;
if (sigaction(SIGHUP, &sa, NULL) < 0)
printf("%s: can't ignore SIGHUP", cmd);
if ((pid = fork()) < 0)
printf("%s: can't fork", cmd);
else if (pid != 0)
exit(0);
为什么是SIGHUB信号?
大多数用户进程(守护进程)调用syslog(3)函数来产生日志消息。日志级别,DEBUG->INFO->NOTICE->WARNING->ERR->CRIT->ALERT->EMERG
单实例守护进程
使用文件和记录锁机制,搞一把写锁。
#define LOCKFILE "/var/run/daemon.pid"
#define LOCKMODE (S_IRUSR | S_IWUSR | S_IRGRP | S_IROTH)
//...
fd = open(LOCKFILE, O_RDWR | O_CREAT, LOCKMODE);
//...
守护进程读取配置文件
守护进程如何让init重新启动该进程?/etc/inittab中为该进程包括respawn记录项。
某些守护进程将捕捉SIGHUP信号,当接受到此信号时reload配置文件。守护进程并不与终端想结合,它们是无控制终端的会话首进程或者孤儿进程组的成员,所以守护进程没有理由期望接收SIGHUP。(正是基于这样的理由,书中利用重新恢复SIGHUP为默认处理方式SIG_DFL,然后当一个线程sigwait到一个SIGHUP的时候,进程reload配置)【疑问:对信号的屏蔽,是意味着不会给进程/线程发送相应的信号,还是进程可以收到这些信号但是不作任何处理?“屏蔽”和“忽略”各自的意思?】
//...
// Restore SIGHUP default and block all signals.
sa.sa_handler = SIG_DFL;
sigemptyset(&sa.sa_mask);
sa.sa_flags = 0;
if (sigaction(SIGHUP, &sa, NULL) < 0)
err_quit("can't restore SIGHUP default\n");
sigfillset(&mask);
if ((err = pthread_sigmask(SIG_BLOCK, &mask, NULL)) != 0)
{
err_exit("SIG_BLOCK error\n");
exit(1);
}
//...
这段代码注意,首先要理解调用sigaction对SIGHUP的处理(实际做了什么事?留到复习信号的时候再理解),然后是调用pthread_sigmask对所有的信号进行了屏蔽?(主线程屏蔽了之后,之后创建的子线程也继承了这种屏蔽)
SIGHUP和SIGTERM的默认动作是终止进程。阻塞了这些信号之后,这两个信号被发送到守护进程的时候,守护进程不会消亡。
单线程读取配置文件的处理方式,仍然是理解对信号的处理:
//...
// Handle signals of interest.
sa.sa_handler = sigterm;
sigemptyset(&sa.sa_mask);
sigaddset(&sa.sa_mask, SIGHUP);
sa.sa_flags = 0;
if (sigaction(SIGTERM, &sa, NULL) < 0)
{
//...syslog and exit
}
sa.sa_handler = sighup;
sigemptyset(&sa.sa_mask);
sigaddset(&sa.sa_mask, SIGTERM);
sa.sa_flags = 0;
if (sigaction(SIGHUP, &sa, NULL) < 0)
{
// ...syslog and exit
}
//...