浅析daemon精灵守护进程创建背后的故事
#include <unistd.h> int main(int argc, char *argv[]) { ... if (daemon(0, 0)) {//调用glibc库函数daemon,创建daemon守护进程 perror("daemon"); return -1; } 好了执行到这里的就是daemon的子进程了[luther.gliethttp]. ... } ========================================================= int daemon( int nochdir, int noclose ) { pid_t pid;
if ( !nochdir && chdir("/") != 0 )//如果nochdir=0,那么改变到"/"根目录 return -1; if ( !noclose )//如果没有close { int fd = open("/dev/null", O_RDWR);//打开空洞文件.
if ( fd < 0 ) return -1;
//对于每个进程,它的fds文件描述符表中:0,1和2文件句柄位置对应的fops文件操作函数集, //fdt->fd[0],fdt->fd[1],fdt->fd[2], //规定将分别与标准输入,标准输出,标准错误输出相关联[luther.gliethttp]. //所以用户应用程序调用open函数打开文件时,默认都是以3索引为开始句柄,故当前open返回的文件句柄最小值为3[luther.gliethttp]. //dup2(unsigned int oldfd, unsigned int newfd)系统调用就是用oldfd的fops操作文件集file,复制到newfd所在处 //即:fdt->fd[newfd] = fdt->fd[oldfd]; if ( dup2( fd, 0 ) < 0 || //使用字符设备/dev/null的fops函数操作集,替换0句柄对应的文件操作集. dup2( fd, 1 ) < 0 || //使用字符设备/dev/null的fops函数操作集,替换1句柄对应的文件操作集. dup2( fd, 2 ) < 0 ) //使用字符设备/dev/null的fops函数操作集,替换2句柄对应的文件操作集. { close(fd); return -1; } //如果上面替换成功,那么键盘的任何操作将不会对该进程产生任何影响,因为0,1,2句柄所在处的fops文件操作集已经都变成了,
//被重定向为"/dev/null"空洞设备的fops.所以对0,1,2句柄的读写操作,也就是在对/dev/null设备作读写操作. close(fd);//关闭打开的/dev/null } pid = fork();//创建子进程. if (pid < 0) return -1;
if (pid > 0) _exit(0);//返回执行的是父进程,那么父进程退出,让子进程变成真正的孤儿进程.
//ok,我们期望的daemon子进程执行到这里了. if ( setsid() < 0 )//设置session id. return -1;
return 0;//成功创建daemon子进程[luther.gliethttp]. }
|