Linux编程之守护进程
0x00 守护进程概念
Daemon(精灵)进程,是Linux中的后台服务进程,生存期较长的进程,通常独立于控制终
端并且周期性地执行某种任务或等待处理某些发生的事件
0x01 守护进程创建步骤
1.创建子进程,父进程退出
所有工作在子进程中进行
形式上脱离了控制终端
2.在子进程中创建新会话
setsid()函数
使子进程完全独立出来,脱离控制
3.改变当前目录为根目录
chdir()函数
防止占用可卸载的文件系统
也可以换成其它路径
4.重设文件权限掩码
umask()函数
防止继承的文件创建屏蔽字拒绝某些权限
增加守护进程灵活性
5.关闭文件描述符
继承的打开文件不会用到,浪费系统资源,无法卸载
6.开始执行守护进程核心工作
7.守护进程退出处理
0x02 代码实例
/**
* @brief: 程序以守护进程方式启动
* @email: binbin_erices@163.com
*/
#include <unistd.h>
#include <signal.h>
#include <stdlib.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <iostream>
void daemon_run()
{
int pid;
signal(SIGCHLD, SIG_IGN);
//1)在父进程中,fork返回新创建子进程的进程ID;
//2)在子进程中,fork返回0;
//3)如果出现错误,fork返回一个负值;
pid = fork();
if (pid < 0)
{
std::cout << "fork error" << std::endl;
exit(-1);
}
//父进程退出,子进程独立运行
else if (pid > 0)
{
exit(0);
}
//之前parent和child运行在同一个session里,parent是会话(session)的组长进程,
//parent进程作为会话的组长进程,如果exit结束执行的话,那么子进程会成为孤儿进程,并被init收养。
//执行setsid()之后,child将重新获得一个新的会话(session)id。
//这时parent退出之后,将不会影响到child了。
setsid();
//改变当前目录到根目录下,防止占用可卸载的文件系统,也可换成其他路径
if (chdir("/") < 0)
{
exit(0);
}
//重设文件权限掩码,设置umask为0,
//防止继承的文件创建屏蔽字拒绝某些权限,增加守护进程灵活性
umask(0);
//关闭文件描述符
int fd;
fd = open("/dev/null", O_RDWR, 0);
if (fd != -1)
{
dup2(fd, STDIN_FILENO);
dup2(fd, STDOUT_FILENO);
dup2(fd, STDERR_FILENO);
}
if (fd > 2)
close(fd);
//开始执行守护进程核心工作
//...
//守护进程退出处理
//...
}
0x03 注意
setsid函数
#include <unistd.h>
pid_t setsid(void)
1.调用进程不能是进程组组长,该进程变成新会话首进程(session header)
2.该进程成为一个新进程组的组长进程
3.需有root权限(ubuntu不需要)
4.新会话丢弃原有的控制终端,该会话没有控制终端
5.该调用进程是组长进程,则出错返回
6.建立新会话时,先调用fork, 父进程终止,子进程调用
dup2函数
#include <unistd.h>
int dup2(int oldfd, int newfd);
返回值: 若dup2调用成功则返回新的文件描述符,出错则返回-1.
dup2与dup区别是dup2可以用参数newfd指定新文件描述符的数值。若参数newfd已经被程序使用,则系统就会将newfd所指的文件关闭,若newfd等于oldfd,则返回newfd,而不关闭newfd所指的文件。dup2所复制的文件描述符与原来的文件描述符共享各种文件状态。共享所有的锁定,读写位置和各项权限或flags等.
dup2(oldfd, newfd)等效于
close(newfd);
fcntl(oldfd, F_DUPFD, newfd);
在shell的重定向功能中,(输入重定向”<”和输出重定向”>”)就是通过调用dup或dup2函数对标准输入和标准输出的操作来实现的。