LCOV - code coverage report
Current view: top level - lib/util - become_daemon.c (source / functions) Hit Total Coverage
Test: coverage report for master 70ed9daf Lines: 38 48 79.2 %
Date: 2024-01-11 09:59:51 Functions: 5 5 100.0 %

          Line data    Source code
       1             : /* 
       2             :    Unix SMB/CIFS implementation.
       3             :    Samba utility functions
       4             :    Copyright (C) Andrew Tridgell 1992-1998
       5             :    Copyright (C) Jeremy Allison 2001-2002
       6             :    Copyright (C) Simo Sorce 2001
       7             :    Copyright (C) Jim McDonough (jmcd@us.ibm.com)  2003.
       8             :    Copyright (C) James J Myers 2003
       9             :    
      10             :    This program is free software; you can redistribute it and/or modify
      11             :    it under the terms of the GNU General Public License as published by
      12             :    the Free Software Foundation; either version 3 of the License, or
      13             :    (at your option) any later version.
      14             :    
      15             :    This program is distributed in the hope that it will be useful,
      16             :    but WITHOUT ANY WARRANTY; without even the implied warranty of
      17             :    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
      18             :    GNU General Public License for more details.
      19             :    
      20             :    You should have received a copy of the GNU General Public License
      21             :    along with this program.  If not, see <http://www.gnu.org/licenses/>.
      22             : */
      23             : 
      24             : #include "replace.h"
      25             : #include "system/filesys.h"
      26             : #include "system/locale.h"
      27             : #if defined(HAVE_LIBSYSTEMD_DAEMON) || defined(HAVE_LIBSYSTEMD)
      28             : #include <systemd/sd-daemon.h>
      29             : #endif
      30             : 
      31             : #include "close_low_fd.h"
      32             : #include "debug.h"
      33             : 
      34             : #include "become_daemon.h"
      35             : 
      36             : static bool sd_notifications = true;
      37             : 
      38             : /*******************************************************************
      39             :  Enable or disable daemon status systemd notifications
      40             : ********************************************************************/
      41          22 : void daemon_sd_notifications(bool enable)
      42             : {
      43          22 :         sd_notifications = enable;
      44          22 :         DBG_DEBUG("Daemon status systemd notifications %s\n",
      45             :                   sd_notifications ? "enabled" : "disabled");
      46          22 : }
      47             : 
      48             : /****************************************************************************
      49             :  Become a daemon, discarding the controlling terminal.
      50             : ****************************************************************************/
      51             : 
      52          80 : void become_daemon(bool do_fork, bool no_session, bool log_stdout)
      53             : {
      54           0 :         pid_t newpid;
      55          80 :         if (do_fork) {
      56           0 :                 newpid = fork();
      57         145 :                 if (newpid == -1) {
      58           0 :                         exit_daemon("Fork failed", errno);
      59             :                 }
      60         145 :                 if (newpid) {
      61           0 :                         _exit(0);
      62             :                 }
      63             : #if defined(HAVE_LIBSYSTEMD_DAEMON) || defined(HAVE_LIBSYSTEMD)
      64          80 :         } else if (sd_notifications) {
      65          80 :                 sd_notify(0, "STATUS=Starting process...");
      66             : #endif
      67             :         }
      68             : 
      69             :         /* detach from the terminal */
      70             : #ifdef HAVE_SETSID
      71         225 :         if (!no_session) {
      72         145 :                 int ret = setsid();
      73         145 :                 if (ret == -1) {
      74           0 :                         exit_daemon("Failed to create session", errno);
      75             :                 }
      76             :         }
      77             : #elif defined(TIOCNOTTY)
      78             :         if (!no_session) {
      79             :                 int i = open("/dev/tty", O_RDWR, 0);
      80             :                 if (i != -1) {
      81             :                         ioctl(i, (int) TIOCNOTTY, (char *)0);
      82             :                         close(i);
      83             :                 }
      84             :         }
      85             : #endif /* HAVE_SETSID */
      86             : 
      87             :         /* Close fd's 0,1,2 as appropriate. Needed if started by rsh. */
      88             :         /* stdin must be open if we do not fork, for monitoring for
      89             :          * close.  stdout must be open if we are logging there, and we
      90             :          * never close stderr (but debug might dup it onto a log file) */
      91         225 :         if (do_fork) {
      92         145 :                 int ret = close_low_fd(0);
      93         145 :                 if (ret != 0) {
      94           0 :                         exit_daemon("close_low_fd(0) failed: %s\n", errno);
      95             :                 }
      96             :         }
      97         225 :         if (!log_stdout) {
      98         145 :                 int ret = close_low_fd(1);
      99         145 :                 if (ret != 0) {
     100           0 :                         exit_daemon("close_low_fd(1) failed: %s\n", errno);
     101             :                 }
     102             :         }
     103         225 : }
     104             : 
     105           1 : void exit_daemon(const char *msg, int error)
     106             : {
     107           1 :         if (msg == NULL) {
     108           0 :                 msg = strerror(error);
     109             :         }
     110             : 
     111             : #if defined(HAVE_LIBSYSTEMD_DAEMON) || defined(HAVE_LIBSYSTEMD)
     112           1 :         if (sd_notifications) {
     113           1 :                 sd_notifyf(0, "STATUS=daemon failed to start: %s\n"
     114             :                                   "ERRNO=%i",
     115             :                                   msg,
     116             :                                   error);
     117             :         }
     118             : #endif
     119           1 :         DBG_ERR("daemon failed to start: %s, error code %d\n",
     120             :                 msg, error);
     121           1 :         exit(1);
     122             : }
     123             : 
     124          92 : void daemon_ready(const char *daemon)
     125             : {
     126          92 :         if (daemon == NULL) {
     127           0 :                 daemon = "Samba";
     128             :         }
     129             : #if defined(HAVE_LIBSYSTEMD_DAEMON) || defined(HAVE_LIBSYSTEMD)
     130          92 :         if (sd_notifications) {
     131          92 :                 sd_notifyf(0,
     132             :                            "READY=1\nSTATUS=%s: ready to serve connections...",
     133             :                            daemon);
     134             :         }
     135             : #endif
     136          92 :         DBG_INFO("daemon '%s' finished starting up and ready to serve "
     137             :                 "connections\n", daemon);
     138          92 : }
     139             : 
     140          34 : void daemon_status(const char *daemon, const char *msg)
     141             : {
     142          34 :         if (daemon == NULL) {
     143           0 :                 daemon = "Samba";
     144             :         }
     145             : #if defined(HAVE_LIBSYSTEMD_DAEMON) || defined(HAVE_LIBSYSTEMD)
     146          34 :         if (sd_notifications) {
     147          12 :                 sd_notifyf(0, "STATUS=%s: %s", daemon, msg);
     148             :         }
     149             : #endif
     150          34 :         DBG_STARTUP_NOTICE("daemon '%s' : %s\n", daemon, msg);
     151          34 : }

Generated by: LCOV version 1.14