2007-04-19 15:56:44 +00:00
/*
Unix SMB / CIFS implementation .
Samba utility functions
Copyright ( C ) Andrew Tridgell 1992 - 1998
Copyright ( C ) Jeremy Allison 2001 - 2002
Copyright ( C ) Simo Sorce 2001
Copyright ( C ) Jim McDonough ( jmcd @ us . ibm . com ) 2003.
Copyright ( C ) James J Myers 2003
This program is free software ; you can redistribute it and / or modify
it under the terms of the GNU General Public License as published by
2007-07-10 02:07:03 +00:00
the Free Software Foundation ; either version 3 of the License , or
2007-04-19 15:56:44 +00:00
( at your option ) any later version .
This program is distributed in the hope that it will be useful ,
but WITHOUT ANY WARRANTY ; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
GNU General Public License for more details .
You should have received a copy of the GNU General Public License
2007-07-10 02:07:03 +00:00
along with this program . If not , see < http : //www.gnu.org/licenses/>.
2007-04-19 15:56:44 +00:00
*/
# include "includes.h"
# include "system/filesys.h"
# include "system/locale.h"
2014-03-25 12:53:04 +02:00
# if HAVE_SYSTEMD
# include <systemd/sd-daemon.h>
# endif
2007-04-19 15:56:44 +00:00
/*******************************************************************
Close the low 3 fd ' s and open dev / null in their place .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2009-02-20 15:10:21 -08:00
2014-07-29 20:35:10 +02:00
_PUBLIC_ int close_low_fd ( int fd )
{
# ifndef VALGRIND
int ret , dev_null ;
dev_null = open ( " /dev/null " , O_RDWR , 0 ) ;
if ( ( dev_null = = - 1 ) & & ( errno = ENFILE ) ) {
/*
* Try to free up an fd
*/
ret = close ( fd ) ;
if ( ret ! = 0 ) {
return errno ;
}
}
dev_null = open ( " /dev/null " , O_RDWR , 0 ) ;
if ( dev_null = = - 1 ) {
dev_null = open ( " /dev/null " , O_WRONLY , 0 ) ;
}
if ( dev_null = = - 1 ) {
return errno ;
}
if ( dev_null = = fd ) {
/*
* This can happen in the ENFILE case above
*/
return 0 ;
}
ret = dup2 ( dev_null , fd ) ;
if ( ret = = - 1 ) {
int err = errno ;
close ( dev_null ) ;
return err ;
}
close ( dev_null ) ;
# endif
return 0 ;
}
2012-03-02 19:32:56 +11:00
_PUBLIC_ void close_low_fds ( bool stdin_too , bool stdout_too , bool stderr_too )
2007-04-19 15:56:44 +00:00
{
# ifndef VALGRIND
int fd ;
int i ;
2012-03-02 19:32:56 +11:00
if ( stdin_too )
close ( 0 ) ;
if ( stdout_too )
close ( 1 ) ;
2007-04-19 15:56:44 +00:00
if ( stderr_too )
close ( 2 ) ;
/* try and use up these file descriptors, so silly
library routines writing to stdout etc won ' t cause havoc */
for ( i = 0 ; i < 3 ; i + + ) {
2012-03-02 19:32:56 +11:00
if ( i = = 0 & & ! stdin_too )
continue ;
if ( i = = 1 & & ! stdout_too )
continue ;
2007-04-19 15:56:44 +00:00
if ( i = = 2 & & ! stderr_too )
continue ;
fd = open ( " /dev/null " , O_RDWR , 0 ) ;
if ( fd < 0 )
fd = open ( " /dev/null " , O_WRONLY , 0 ) ;
if ( fd < 0 ) {
DEBUG ( 0 , ( " Can't open /dev/null \n " ) ) ;
return ;
}
if ( fd ! = i ) {
DEBUG ( 0 , ( " Didn't get file descriptor %d \n " , i ) ) ;
2013-02-20 10:48:05 +01:00
close ( fd ) ;
2007-04-19 15:56:44 +00:00
return ;
}
}
# endif
}
2009-02-20 15:10:21 -08:00
/****************************************************************************
2007-04-19 15:56:44 +00:00
Become a daemon , discarding the controlling terminal .
2009-02-20 15:10:21 -08:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2007-04-19 15:56:44 +00:00
2010-03-26 11:17:37 +01:00
_PUBLIC_ void become_daemon ( bool do_fork , bool no_process_group , bool log_stdout )
2007-04-19 15:56:44 +00:00
{
2014-03-25 12:53:04 +02:00
pid_t newpid ;
2009-02-23 20:46:11 -08:00
if ( do_fork ) {
2014-03-25 12:53:04 +02:00
newpid = fork ( ) ;
if ( newpid ) {
# if HAVE_SYSTEMD
sd_notifyf ( 0 , " READY=0 \n STATUS=Starting process... \n MAINPID=%lu " , ( unsigned long ) newpid ) ;
# endif /* HAVE_SYSTEMD */
2007-04-19 15:56:44 +00:00
_exit ( 0 ) ;
}
}
2009-02-20 15:10:21 -08:00
/* detach from the terminal */
2007-04-19 15:56:44 +00:00
# ifdef HAVE_SETSID
2009-02-20 15:10:21 -08:00
if ( ! no_process_group ) setsid ( ) ;
2007-04-19 15:56:44 +00:00
# elif defined(TIOCNOTTY)
2009-02-20 15:10:21 -08:00
if ( ! no_process_group ) {
2012-03-28 12:48:00 +11:00
int i = open ( " /dev/tty " , O_RDWR , 0 ) ;
2007-04-19 15:56:44 +00:00
if ( i ! = - 1 ) {
2009-02-20 15:10:21 -08:00
ioctl ( i , ( int ) TIOCNOTTY , ( char * ) 0 ) ;
2007-04-19 15:56:44 +00:00
close ( i ) ;
}
}
# endif /* HAVE_SETSID */
2012-03-02 19:32:56 +11:00
/* Close fd's 0,1,2 as appropriate. Needed if started by rsh. */
/* stdin must be open if we do not fork, for monitoring for
* close . stdout must be open if we are logging there , and we
* never close stderr ( but debug might dup it onto a log file ) */
close_low_fds ( do_fork , ! log_stdout , false ) ;
2007-04-19 15:56:44 +00:00
}
2014-03-25 12:53:04 +02:00
_PUBLIC_ void exit_daemon ( const char * msg , int error )
{
# ifdef HAVE_SYSTEMD
if ( msg = = NULL ) {
msg = strerror ( error ) ;
}
sd_notifyf ( 0 , " STATUS=daemon failed to start: %s \n "
" ERRNO=%i " ,
msg ,
error ) ;
# endif
DEBUG ( 0 , ( " STATUS=daemon failed to start: %s, error code %d \n " , msg , error ) ) ;
exit ( 1 ) ;
}
2014-05-12 14:36:57 +02:00
_PUBLIC_ void daemon_ready ( const char * name )
2014-03-25 12:53:04 +02:00
{
2014-05-12 14:36:57 +02:00
if ( name = = NULL ) {
name = " Samba " ;
2014-03-25 12:53:04 +02:00
}
# ifdef HAVE_SYSTEMD
2014-05-12 14:36:57 +02:00
sd_notifyf ( 0 , " READY=1 \n STATUS=%s: ready to serve connections... " , name ) ;
2014-03-25 12:53:04 +02:00
# endif
2014-05-22 08:48:32 +02:00
DEBUG ( 0 , ( " STATUS=daemon '%s' finished starting up and ready to serve "
" connections \n " , name ) ) ;
2014-03-25 12:53:04 +02:00
}