2003-11-26 07:17:08 +08:00
|
|
|
/* detach.c -- routines to daemonize a process */
|
1999-09-09 03:06:24 +08:00
|
|
|
/* $OpenLDAP$ */
|
2003-11-26 07:17:08 +08:00
|
|
|
/* This work is part of OpenLDAP Software <http://www.openldap.org/>.
|
|
|
|
*
|
2004-01-02 03:15:16 +08:00
|
|
|
* Copyright 1998-2004 The OpenLDAP Foundation.
|
2003-11-26 07:17:08 +08:00
|
|
|
* All rights reserved.
|
|
|
|
*
|
|
|
|
* Redistribution and use in source and binary forms, with or without
|
|
|
|
* modification, are permitted only as authorized by the OpenLDAP
|
|
|
|
* Public License.
|
|
|
|
*
|
|
|
|
* A copy of this license is available in the file LICENSE in the
|
|
|
|
* top-level directory of the distribution or, alternatively, at
|
|
|
|
* <http://www.OpenLDAP.org/license.html>.
|
2003-05-25 11:31:21 +08:00
|
|
|
*/
|
1998-11-05 07:05:02 +08:00
|
|
|
/*
|
|
|
|
* Copyright (c) 1990, 1994 Regents of the University of Michigan.
|
|
|
|
* All rights reserved.
|
|
|
|
*
|
|
|
|
* Redistribution and use in source and binary forms are permitted
|
|
|
|
* provided that this notice is preserved and that due credit is given
|
|
|
|
* to the University of Michigan at Ann Arbor. The name of the University
|
|
|
|
* may not be used to endorse or promote products derived from this
|
|
|
|
* software without specific prior written permission. This software
|
|
|
|
* is provided ``as is'' without express or implied warranty.
|
|
|
|
*/
|
2003-11-26 07:17:08 +08:00
|
|
|
/* This work was originally developed by the University of Michigan
|
|
|
|
* and distributed as part of U-MICH LDAP.
|
|
|
|
*/
|
1998-11-05 07:05:02 +08:00
|
|
|
|
|
|
|
#include "portable.h"
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
|
1999-08-04 02:41:55 +08:00
|
|
|
#include <ac/stdlib.h>
|
1998-11-05 07:05:02 +08:00
|
|
|
#include <ac/signal.h>
|
1999-03-29 17:04:35 +08:00
|
|
|
#include <ac/socket.h>
|
1998-11-05 07:05:02 +08:00
|
|
|
#include <ac/unistd.h>
|
|
|
|
|
|
|
|
#include <sys/stat.h>
|
|
|
|
#include <fcntl.h>
|
1999-03-29 17:04:35 +08:00
|
|
|
|
|
|
|
#ifdef HAVE_SYS_FILE_H
|
1998-11-05 07:05:02 +08:00
|
|
|
#include <sys/file.h>
|
1999-03-29 17:04:35 +08:00
|
|
|
#endif
|
|
|
|
#ifdef HAVE_SYS_IOCTL_H
|
1998-11-05 07:05:02 +08:00
|
|
|
#include <sys/ioctl.h>
|
1999-03-29 17:04:35 +08:00
|
|
|
#endif
|
1998-11-05 07:05:02 +08:00
|
|
|
|
1998-11-07 06:03:15 +08:00
|
|
|
#include "lutil.h"
|
1998-11-05 07:05:02 +08:00
|
|
|
|
|
|
|
void
|
|
|
|
lutil_detach( int debug, int do_close )
|
|
|
|
{
|
|
|
|
int i, sd, nbits;
|
|
|
|
|
|
|
|
#ifdef HAVE_SYSCONF
|
|
|
|
nbits = sysconf( _SC_OPEN_MAX );
|
|
|
|
#elif HAVE_GETDTABLESIZE
|
|
|
|
nbits = getdtablesize();
|
|
|
|
#else
|
|
|
|
nbits = FD_SETSIZE;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef FD_SETSIZE
|
|
|
|
if ( nbits > FD_SETSIZE ) {
|
|
|
|
nbits = FD_SETSIZE;
|
|
|
|
}
|
|
|
|
#endif /* FD_SETSIZE */
|
|
|
|
|
|
|
|
if ( debug == 0 ) {
|
|
|
|
for ( i = 0; i < 5; i++ ) {
|
|
|
|
#if HAVE_THR
|
|
|
|
switch ( fork1() )
|
|
|
|
#else
|
|
|
|
switch ( fork() )
|
|
|
|
#endif
|
|
|
|
{
|
|
|
|
case -1:
|
|
|
|
sleep( 5 );
|
|
|
|
continue;
|
|
|
|
|
|
|
|
case 0:
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
1999-08-04 02:14:24 +08:00
|
|
|
_exit( EXIT_SUCCESS );
|
1998-11-05 07:05:02 +08:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2003-01-31 06:44:53 +08:00
|
|
|
if ( (sd = open( "/dev/null", O_RDWR )) == -1 &&
|
|
|
|
(sd = open( "/dev/null", O_RDONLY )) == -1 &&
|
|
|
|
/* Panic -- open *something* */
|
|
|
|
(sd = open( "/", O_RDONLY )) == -1 ) {
|
2000-01-11 16:04:59 +08:00
|
|
|
perror("/dev/null");
|
2003-01-31 06:44:53 +08:00
|
|
|
} else {
|
|
|
|
/* redirect stdin, stdout, stderr to /dev/null */
|
|
|
|
dup2( sd, STDIN_FILENO );
|
|
|
|
dup2( sd, STDOUT_FILENO );
|
|
|
|
dup2( sd, STDERR_FILENO );
|
2000-01-11 16:04:59 +08:00
|
|
|
|
2003-01-31 06:44:53 +08:00
|
|
|
switch( sd ) {
|
|
|
|
default:
|
|
|
|
close( sd );
|
|
|
|
case STDIN_FILENO:
|
|
|
|
case STDOUT_FILENO:
|
|
|
|
case STDERR_FILENO:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
1998-11-05 07:05:02 +08:00
|
|
|
|
2000-01-11 16:04:59 +08:00
|
|
|
if ( do_close ) {
|
|
|
|
/* close everything else */
|
|
|
|
for ( i = 0; i < nbits; i++ ) {
|
|
|
|
if( i != STDIN_FILENO &&
|
|
|
|
i != STDOUT_FILENO &&
|
|
|
|
i != STDERR_FILENO )
|
2000-01-11 10:20:01 +08:00
|
|
|
{
|
2000-01-11 16:04:59 +08:00
|
|
|
close( i );
|
2000-01-11 10:20:01 +08:00
|
|
|
}
|
|
|
|
}
|
1998-11-05 07:05:02 +08:00
|
|
|
}
|
2000-01-11 10:20:01 +08:00
|
|
|
|
2000-02-08 12:22:31 +08:00
|
|
|
#ifdef CHDIR_TO_ROOT
|
2000-01-11 10:20:01 +08:00
|
|
|
(void) chdir( "/" );
|
2000-02-08 12:22:31 +08:00
|
|
|
#endif
|
1998-11-05 07:05:02 +08:00
|
|
|
|
|
|
|
#ifdef HAVE_SETSID
|
|
|
|
(void) setsid();
|
1999-03-29 17:04:35 +08:00
|
|
|
#elif TIOCNOTTY
|
1998-11-05 07:05:02 +08:00
|
|
|
if ( (sd = open( "/dev/tty", O_RDWR )) != -1 ) {
|
|
|
|
(void) ioctl( sd, TIOCNOTTY, NULL );
|
|
|
|
(void) close( sd );
|
|
|
|
}
|
1999-03-29 17:04:35 +08:00
|
|
|
#endif
|
1998-11-05 07:05:02 +08:00
|
|
|
}
|
|
|
|
|
1999-03-29 17:04:35 +08:00
|
|
|
#ifdef SIGPIPE
|
1998-11-05 07:05:02 +08:00
|
|
|
(void) SIGNAL( SIGPIPE, SIG_IGN );
|
1999-03-29 17:04:35 +08:00
|
|
|
#endif
|
1998-11-05 07:05:02 +08:00
|
|
|
}
|