本文介绍如何使用 C 语言创建 Linux 系统中 SysV 风格的 daemon 程序。注意:这是一种旧式的 daemon 程序写法,进入 systemd 时代后是不需要通过这样的方式创建 daemon 程序的。 本文的演示环境为 ubuntu 18.04。

创建 daemon 程序的流程

通过前文《Linux session(会话)》我们了解到,如果要让程序运行在后台,必须处理好进程的 session。所以在创建 daemon 程序的过程中处理 session 问题是很重要的一步,当然除此之外还需要其它的步骤。下面是在 Linux 系统中创建一个 SysV 风格的 daemon 的基本流程:

  1. 从父进程 fork 出一个子进程
  2. 为子进程创建新的 session ID
  3. 在子进程中再 fork 一次
  4. 修改 umask
  5. 修改进程的当前工作目录
  6. 关闭进程中的文件描述符

接下来我们通过代码来介绍这些操作的含义。

创建 daemon 程序

从父进程 fork 出一个子进程
创建一个子进程,如果成功就让父进程退出,此时的子进程已经成为了 init 进程的子进程:

pid_t pid;

pid = fork();
if (pid < )
exit(EXIT_FAILURE);
if (pid > )
exit(EXIT_SUCCESS);

为子进程创建新的 session ID
运行在后台的进程需要摆脱 session 终端的束缚,通过 setsid() 函数为进程设置新的 session ID 可以做到这一点:

pid_t pid;

pid = fork();
if (pid < )
exit(EXIT_FAILURE);
if (pid > )
exit(EXIT_SUCCESS); if (setsid() < 0)
exit(EXIT_FAILURE);

********************************
执行到这里时,PID==PGID==SID

********************************

在子进程中再 fork 一次
这次 fork 的目的是防止进程再次获得终端。因为只有 session leader 才能获得终端,而这次 fork 使子进程变成了非 session leader:

pid_t pid;

pid = fork();
if (pid < )
exit(EXIT_FAILURE);
if (pid > )
exit(EXIT_SUCCESS); if (setsid() < )
exit(EXIT_FAILURE); /* 第二次 fork */
pid = fork();
if (pid < 0)
exit(EXIT_FAILURE); if (pid > 0)
exit(EXIT_SUCCESS);

********************************
执行到这里时,PGID==SID 但是已经不等于 PID 了,说明进程已经不是 session leader

********************************

修改 umask
为了能够向 daemon 进程创建的任何文件中写入内容(包括日志),必须重置 umask(file mode mask, umask),以确保能够正确地写入或读取这些文件:

umask(0);

修改进程的当前工作目录
必须保证进程的当前工作目录是存在的。因为众多的 Linux 发行版中很多都没有完全遵守标准的文件目录结构,所以最好是把进程的当前工作目录设置为 /,这样可以避免因设置了某个目录而导致它无法被 unmount:

chdir("/");

关闭进程中的文件描述符
关闭进程中所有打开的文件描述符:

int x;
for (x = sysconf(_SC_OPEN_MAX); x>=0; x--)
{
close (x);
}

把日志写入 syslog
Daemon 程序的日志非常重要,我们可以通过 openlog、syslog 和 closelog 三个函数把日志内容写入到 syslog  中:

openlog ("daemondemo", LOG_PID, LOG_DAEMON);
syslog (LOG_NOTICE, "Daemon demo is running, number: %d", count);
closelog();

本文 demo 输出的日志如下所示:

完整的代码

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <signal.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <syslog.h> static void demo_daemon()
{
pid_t pid; /* Fork off the parent process */
pid = fork(); /* An error occurred */
if (pid < )
exit(EXIT_FAILURE); /* Success: Let the parent terminate */
if (pid > )
exit(EXIT_SUCCESS); /* On success: The child process becomes session leader */
if (setsid() < )
exit(EXIT_FAILURE); /* Catch, ignore and handle signals */
//TODO: Implement a working signal handler */
//signal(SIGCHLD, SIG_IGN);
//signal(SIGHUP, SIG_IGN); /* Fork off for the second time*/
pid = fork(); /* An error occurred */
if (pid < )
exit(EXIT_FAILURE); /* Success: Let the parent terminate */
if (pid > )
exit(EXIT_SUCCESS); /* Set new file permissions */
umask(); /* Change the working directory to the root directory */
/* or another appropriated directory */
chdir("/"); /* Close all open file descriptors */
int x;
for (x = sysconf(_SC_OPEN_MAX); x>=; x--)
{
close (x);
} /* Open the log file */
openlog ("daemondemo", LOG_PID, LOG_DAEMON);
} int main()
{
int count = ;
demo_daemon(); while ()
{
//TODO: Insert daemon code here.
count ++;
syslog (LOG_NOTICE, "Daemon demo is running, number: %d", count);
sleep ();
if(count > )
{
break;
}
} syslog (LOG_NOTICE, "Daemon demo terminated.");
closelog(); return EXIT_SUCCESS;
}

把上面的代码保存到文件 daemondemo.c 中(也可以从这里下代码),然后执行下面的命令进行编译就可以得到可执行文件 daemondemo:

$ gcc -Wall daemondemo.c -o daemondemo

关于 fork 两次

这是一个很有意思的话题,有人说需要 fork 两次,有人说第二次是可选的,究竟该如何做呢?当我们理解了第二次 fork 的用途后就可以自行决定是否需要第二次 fork 了。
这还需要从 session 的控制终端说起。控制终端是进程的一个属性,通过 fork 系统调用创建的子进程会从父进程那里继承控制终端。这样,session 中的所有进程都从 session 领头进程那里继承控制终端。前面已经说过了,要把程序变成 daemon,就得让进程摆脱 session 的终端。而这些在第一次 fork 后调用 setsid() 函数就搞定了。那么如果接下来不小心再给进程添加了终端该怎么办?答案是不让你添加!这就是第二次 fork 的作用。只有 session leader 才能获得终端,而第二次 fork 使子进程变成了非 session leader,你想犯错也不给你机会了。

像 nginx 和 gblic 的 daemon 函数的实现都是 fork 一次,所以说第二次 fork 是可选的,你可以根据自己的实际情况来决定。

参考:
Linux Daemon Writing HOWTO
Creating a daemon in Linux
daemon man page
daemon 函数
Unix Daemon Server Programming
glibc daemon.c

创建 SysV 风格的 linux daemon 程序的更多相关文章

  1. 嵌入式Linux应用程序开发详解------(创建守护进程)

    嵌入式Linux应用程序开发详解 华清远见 本文只是阅读文摘. 创建一个守护进程的步骤: 1.创建一个子进程,然后退出父进程: 2.在子进程中使用创建新会话---setsid(): 3.改变当前工作目 ...

  2. Linux Daemon 类程序

    1.后台daemon程序(精灵程序) 在Linux中专门提供了一个函数来完成这个daemon化的过程,这个函数的原型如下 int daemon (int __nochdir, int __noclos ...

  3. Linux C 程序 进程控制(17)

    进程控制 1.进程概述现代操作系统的特点在于程序的并行执行.Linux是一个多用户多任务的操作系统.ps .pstree 查看进程进程除了进程id外还有一些其他标识信息,可以通过相应的函数获得.// ...

  4. 使用PHP脚本来写Daemon程序

    什么是Daemon进程   这又是一个有趣的概念,daemon在英语中是"精灵"的意思,就像我们经常在迪斯尼动画里见到的那些,有些会飞,有些不会,经常围着动画片的主人公转来转去,啰 ...

  5. Linux 高性能服务器编程——Linux服务器程序规范

    问题聚焦:     除了网络通信外,服务器程序通常还必须考虑许多其他细节问题,这些细节问题涉及面逛且零碎,而且基本上是模板式的,所以称之为服务器程序规范.     工欲善其事,必先利其器,这篇主要来探 ...

  6. 构建 iOS 风格移动 Web 应用程序的8款开发框架

    使用 HTML5,CSS3 和 JavaScript 开发移动应用经过实践证明是一种可行的方式.这里收录了几款 iOS 风格的手机应用程序开发框架,帮助您使用擅长的 Web 技术来开发移动应用程序.这 ...

  7. 使用 iosOverlay.js 创建 iOS 风格的提示和通知

    iosOverlay.js 用于在 Web 项目中实现 iOS 风格的通知和提示效果.为了防止图标加载的时候闪烁,你需要预加载的图像资源.不兼容 CSS 动画的浏览器需要 jQuery 支持.浏览器兼 ...

  8. linux应用程序开发-文件编程-系统调用方式

    在看韦东山视频linux驱动方面有一些吃力,究其原因,虽然接触过linux应用程序编程,但是没有深入去理解,相关函数用法不清楚,正好看到国嵌视频对这一方面讲的比较透彻, 所以把学习过程记录下来,也作为 ...

  9. linux c程序中获取shell脚本输出的实现方法

    linux c程序中获取shell脚本输出的实现方法 1. 前言Unix界有一句名言:“一行shell脚本胜过万行C程序”,虽然这句话有些夸张,但不可否认的是,借助脚本确实能够极大的简化一些编程工作. ...

随机推荐

  1. Web_javaScript

    JavaScript 的使用 第1章 JavaScript起源 1.1 起源 N年前 拨号上网,网速很慢,数据提交到服务器端验证,体验很差 于是,就有人在想:能不能让这些数据在浏览器端验证呢? 20世 ...

  2. CentOS7系统服务管理systemctl

    目录 一.systemctl介绍 二.systemctl常用命令 1.启动服务 2.停止服务 3.重启服务 4.查看服务是否已启动 5.查看服务的状态 6.启用开机自启动服务 7.停用开机自启动服务 ...

  3. Beef xss神器

    KALI中启动BEEFXSS PAYLOAD为 <script src=”http://攻击机IP:3000/hook.js”></script> 将攻击代码插入到存储型XSS ...

  4. Python编写“求一元二次方程的解”

    #求一元二次方程的解 import math def equation(a,b,c): h=b*b-4*a*c #一元二次方程的解,百度来的 if h>=0: x1=(-b+math.sqrt( ...

  5. 第十六周Java实验作业

    实验十六  线程技术 实验时间 2017-12-8 1.实验目的与要求 (1) 掌握线程概念: 多线程是进程执行过程中产生的多条执行线索,线程是比进程执行更小的单位. 线程不能独立存在,必须存在于进程 ...

  6. JavaScript----简介及基础语法

    ##JavaScript *概念:一门客户端脚本语言 *运行在客户端浏览器中的.每一个浏览器都有JavaScript的解析引擎. *脚本语言:不需要编译,直接就可以被浏览器解析执行. *功能: *可以 ...

  7. Java并发编程(03):多线程并发访问,同步控制

    本文源码:GitHub·点这里 || GitEE·点这里 一.并发问题 多线程学习的时候,要面对的第一个复杂问题就是,并发模式下变量的访问,如果不理清楚内在流程和原因,经常会出现这样一个问题:线程处理 ...

  8. (一)iview的校验TypeError: Cannot read property 'validateField' of undefined"

    一.问题描述 我是在自己封装了一个地址级联选择,然后想要每次改变了其中数据的时候,就进行一次单独校验,所以用到了iview对部分表单字段进行校验的方法validateField.其实一开始使用的时候是 ...

  9. 动态规划/MinMax-Stone Game

    2019-09-07 16:34:48 877. Stone Game 问题描述: 问题求解: 典型的博弈问题,也是一个典型的min-max问题.通常使用算diff的方法把min-max转为求max. ...

  10. laravel中间件的创建思路分析

    网上有很多解析laravel中间件的实现原理,但是不知道有没有读者在读的时候不明白,作者是怎么想到要用array_reduce函数的? 本文从自己的角度出发,模拟了如果我是作者,我是怎么实现这个中间件 ...