1996-05-04 07:50:46 +00:00
/*
2002-01-30 06:08:46 +00:00
Unix SMB / CIFS implementation .
1996-05-04 07:50:46 +00:00
Critical Fault handling
1998-01-22 13:27:43 +00:00
Copyright ( C ) Andrew Tridgell 1992 - 1998
2009-02-24 16:42:18 -08:00
Copyright ( C ) Tim Prouty 2009
1996-05-04 07:50:46 +00:00
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-09 19:25:36 +00:00
the Free Software Foundation ; either version 3 of the License , or
1996-05-04 07:50:46 +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 00:52:41 +00:00
along with this program . If not , see < http : //www.gnu.org/licenses/>.
1996-05-04 07:50:46 +00:00
*/
# include "includes.h"
2006-04-04 00:27:50 +00:00
# ifdef HAVE_SYS_PRCTL_H
# include <sys/prctl.h>
# endif
1998-04-13 19:24:06 +00:00
static void ( * cont_fn ) ( void * ) ;
2007-11-15 14:19:52 -08:00
static char * corepath ;
1996-05-04 07:50:46 +00:00
/*******************************************************************
report a fault
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static void fault_report ( int sig )
{
1998-08-11 13:55:16 +00:00
static int counter ;
if ( counter ) _exit ( 1 ) ;
counter + + ;
2006-04-04 00:27:50 +00:00
DEBUGSEP ( 0 ) ;
2009-01-15 22:27:52 +01:00
DEBUG ( 0 , ( " INTERNAL ERROR: Signal %d in pid %d (%s) " , sig , ( int ) sys_getpid ( ) , samba_version_string ( ) ) ) ;
2005-08-17 06:01:38 +00:00
DEBUG ( 0 , ( " \n Please read the Trouble-Shooting section of the Samba3-HOWTO \n " ) ) ;
DEBUG ( 0 , ( " \n From: http://www.samba.org/samba/docs/Samba3-HOWTO.pdf \n " ) ) ;
2006-04-04 00:27:50 +00:00
DEBUGSEP ( 0 ) ;
1996-05-04 07:50:46 +00:00
1998-08-21 11:37:40 +00:00
smb_panic ( " internal error " ) ;
1996-05-04 07:50:46 +00:00
1998-08-11 13:55:16 +00:00
if ( cont_fn ) {
cont_fn ( NULL ) ;
1996-05-04 07:50:46 +00:00
# ifdef SIGSEGV
1998-08-11 13:55:16 +00:00
CatchSignal ( SIGSEGV , SIGNAL_CAST SIG_DFL ) ;
1996-05-04 07:50:46 +00:00
# endif
# ifdef SIGBUS
1998-08-11 13:55:16 +00:00
CatchSignal ( SIGBUS , SIGNAL_CAST SIG_DFL ) ;
2004-01-22 01:50:58 +00:00
# endif
# ifdef SIGABRT
CatchSignal ( SIGABRT , SIGNAL_CAST SIG_DFL ) ;
1996-05-04 07:50:46 +00:00
# endif
1998-08-11 13:55:16 +00:00
return ; /* this should cause a core dump */
}
exit ( 1 ) ;
1996-05-04 07:50:46 +00:00
}
/****************************************************************************
catch serious errors
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static void sig_fault ( int sig )
{
1998-08-11 13:55:16 +00:00
fault_report ( sig ) ;
1996-05-04 07:50:46 +00:00
}
/*******************************************************************
setup our fault handlers
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
1998-04-13 19:24:06 +00:00
void fault_setup ( void ( * fn ) ( void * ) )
1996-05-04 07:50:46 +00:00
{
1998-08-11 13:55:16 +00:00
cont_fn = fn ;
1996-05-04 07:50:46 +00:00
# ifdef SIGSEGV
1998-08-11 13:55:16 +00:00
CatchSignal ( SIGSEGV , SIGNAL_CAST sig_fault ) ;
1996-05-04 07:50:46 +00:00
# endif
# ifdef SIGBUS
1998-08-11 13:55:16 +00:00
CatchSignal ( SIGBUS , SIGNAL_CAST sig_fault ) ;
1996-05-04 07:50:46 +00:00
# endif
2004-01-22 01:50:58 +00:00
# ifdef SIGABRT
CatchSignal ( SIGABRT , SIGNAL_CAST sig_fault ) ;
# endif
1996-05-04 07:50:46 +00:00
}
2006-04-04 00:27:50 +00:00
2009-02-24 14:45:46 -08:00
/**
* Build up the default corepath as " <logbase>/cores/<progname> "
*/
static char * get_default_corepath ( const char * logbase , const char * progname )
{
char * tmp_corepath ;
/* Setup core dir in logbase. */
tmp_corepath = talloc_asprintf ( NULL , " %s/cores " , logbase ) ;
if ( ! tmp_corepath )
return NULL ;
if ( ( mkdir ( tmp_corepath , 0700 ) = = - 1 ) & & errno ! = EEXIST )
goto err_out ;
if ( chmod ( tmp_corepath , 0700 ) = = - 1 )
goto err_out ;
talloc_free ( tmp_corepath ) ;
/* Setup progname-specific core subdir */
tmp_corepath = talloc_asprintf ( NULL , " %s/cores/%s " , logbase , progname ) ;
if ( ! tmp_corepath )
return NULL ;
if ( mkdir ( tmp_corepath , 0700 ) = = - 1 & & errno ! = EEXIST )
goto err_out ;
if ( chown ( tmp_corepath , getuid ( ) , getgid ( ) ) = = - 1 )
goto err_out ;
if ( chmod ( tmp_corepath , 0700 ) = = - 1 )
goto err_out ;
return tmp_corepath ;
err_out :
talloc_free ( tmp_corepath ) ;
return NULL ;
}
2009-02-24 16:42:18 -08:00
/**
* Get the FreeBSD corepath .
*
* On FreeBSD the current working directory is ignored when creating a core
* file . Instead the core directory is controlled via sysctl . This consults
* the value of " kern.corefile " so the correct corepath can be printed out
* before dump_core ( ) calls abort .
*/
# if (defined(FREEBSD) && defined(HAVE_SYSCTLBYNAME))
static char * get_freebsd_corepath ( void )
{
char * tmp_corepath = NULL ;
char * end = NULL ;
size_t len = 128 ;
int ret ;
/* Loop with increasing sizes so we don't allocate too much. */
do {
if ( len > 1024 ) {
goto err_out ;
}
tmp_corepath = ( char * ) talloc_realloc ( NULL , tmp_corepath ,
char , len ) ;
if ( ! tmp_corepath ) {
return NULL ;
}
ret = sysctlbyname ( " kern.corefile " , tmp_corepath , & len , NULL ,
0 ) ;
if ( ret = = - 1 ) {
if ( errno ! = ENOMEM ) {
DEBUG ( 0 , ( " sysctlbyname failed getting "
" kern.corefile %s \n " ,
strerror ( errno ) ) ) ;
goto err_out ;
}
/* Not a large enough array, try a bigger one. */
len = len < < 1 ;
}
} while ( ret = = - 1 ) ;
/* Strip off the common filename expansion */
if ( ( end = strrchr_m ( tmp_corepath , ' / ' ) ) ) {
* end = ' \0 ' ;
}
return tmp_corepath ;
err_out :
if ( tmp_corepath ) {
talloc_free ( tmp_corepath ) ;
}
return NULL ;
}
# endif
/**
* Try getting system - specific corepath if one exists .
*
* If the system doesn ' t define a corepath , then the default is used .
*/
static char * get_corepath ( const char * logbase , const char * progname )
{
char * tmp_corepath = NULL ;
/* @todo: Add support for the linux corepath. */
# if (defined(FREEBSD) && defined(HAVE_SYSCTLBYNAME))
tmp_corepath = get_freebsd_corepath ( ) ;
# endif
/* If this has been set correctly, we're done. */
if ( tmp_corepath ) {
return tmp_corepath ;
}
/* Fall back to the default. */
return get_default_corepath ( logbase , progname ) ;
}
2006-04-04 00:27:50 +00:00
/*******************************************************************
make all the preparations to safely dump a core file
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
void dump_core_setup ( const char * progname )
{
2007-11-15 14:19:52 -08:00
char * logbase = NULL ;
char * end = NULL ;
2006-04-04 00:27:50 +00:00
if ( lp_logfile ( ) & & * lp_logfile ( ) ) {
2007-11-15 14:19:52 -08:00
if ( asprintf ( & logbase , " %s " , lp_logfile ( ) ) < 0 ) {
return ;
}
2006-04-04 00:27:50 +00:00
if ( ( end = strrchr_m ( logbase , ' / ' ) ) ) {
* end = ' \0 ' ;
}
} else {
2009-02-24 14:45:46 -08:00
/* We will end up here if the log file is given on the command
2006-04-04 00:27:50 +00:00
* line by the - l option but the " log file " option is not set
* in smb . conf .
*/
2007-12-10 11:30:37 -08:00
if ( asprintf ( & logbase , " %s " , get_dyn_LOGFILEBASE ( ) ) < 0 ) {
2007-11-15 14:19:52 -08:00
return ;
}
2006-04-04 00:27:50 +00:00
}
SMB_ASSERT ( progname ! = NULL ) ;
2009-02-24 16:42:18 -08:00
corepath = get_corepath ( logbase , progname ) ;
2009-02-24 14:45:46 -08:00
if ( ! corepath ) {
DEBUG ( 0 , ( " Unable to setup corepath for %s: %s \n " , progname ,
strerror ( errno ) ) ) ;
goto out ;
2008-12-31 18:06:57 -08:00
}
2006-04-04 00:27:50 +00:00
2007-11-15 14:19:52 -08:00
2006-04-04 00:27:50 +00:00
# ifdef HAVE_GETRLIMIT
# ifdef RLIMIT_CORE
{
struct rlimit rlp ;
getrlimit ( RLIMIT_CORE , & rlp ) ;
rlp . rlim_cur = MAX ( 16 * 1024 * 1024 , rlp . rlim_cur ) ;
setrlimit ( RLIMIT_CORE , & rlp ) ;
getrlimit ( RLIMIT_CORE , & rlp ) ;
DEBUG ( 3 , ( " Maximum core file size limits now %d(soft) %d(hard) \n " ,
( int ) rlp . rlim_cur , ( int ) rlp . rlim_max ) ) ;
}
# endif
# endif
# if defined(HAVE_PRCTL) && defined(PR_SET_DUMPABLE)
/* On Linux we lose the ability to dump core when we change our user
* ID . We know how to dump core safely , so let ' s make sure we have our
* dumpable flag set .
*/
prctl ( PR_SET_DUMPABLE , 1 ) ;
# endif
/* FIXME: if we have a core-plus-pid facility, configurably set
* this up here .
*/
2009-02-24 14:45:46 -08:00
out :
SAFE_FREE ( logbase ) ;
2006-04-04 00:27:50 +00:00
}
void dump_core ( void )
{
2007-09-14 11:07:02 +00:00
static bool called ;
if ( called ) {
DEBUG ( 0 , ( " dump_core() called recursive \n " ) ) ;
exit ( 1 ) ;
}
called = true ;
2006-05-16 02:50:49 +00:00
/* Note that even if core dumping has been disabled, we still set up
* the core path . This is to handle the case where core dumping is
* turned on in smb . conf and the relevant daemon is not restarted .
*/
if ( ! lp_enable_core_files ( ) ) {
2007-04-06 19:17:15 +00:00
DEBUG ( 0 , ( " Exiting on internal error (core file administratively disabled) \n " ) ) ;
2006-05-16 02:50:49 +00:00
exit ( 1 ) ;
}
2006-11-07 17:18:00 +00:00
# if DUMP_CORE
2007-04-17 17:17:19 +00:00
/* If we're running as non root we might not be able to dump the core
* file to the corepath . There must not be an unbecome_root ( ) before
* we call abort ( ) . */
2007-04-17 18:35:10 +00:00
if ( geteuid ( ) ! = 0 ) {
2007-04-17 17:17:19 +00:00
become_root ( ) ;
}
2008-06-13 11:56:40 +02:00
if ( corepath = = NULL ) {
DEBUG ( 0 , ( " Can not dump core: corepath not set up \n " ) ) ;
exit ( 1 ) ;
}
2006-04-04 00:27:50 +00:00
if ( * corepath ! = ' \0 ' ) {
/* The chdir might fail if we dump core before we finish
* processing the config file .
*/
if ( chdir ( corepath ) ! = 0 ) {
2006-11-07 17:18:00 +00:00
DEBUG ( 0 , ( " unable to change to %s \n " , corepath ) ) ;
2006-04-04 00:27:50 +00:00
DEBUGADD ( 0 , ( " refusing to dump core \n " ) ) ;
exit ( 1 ) ;
}
DEBUG ( 0 , ( " dumping core in %s \n " , corepath ) ) ;
}
umask ( ~ ( 0700 ) ) ;
dbgflush ( ) ;
/* Ensure we don't have a signal handler for abort. */
# ifdef SIGABRT
CatchSignal ( SIGABRT , SIGNAL_CAST SIG_DFL ) ;
# endif
abort ( ) ;
2006-11-07 17:18:00 +00:00
# else /* DUMP_CORE */
exit ( 1 ) ;
# endif /* DUMP_CORE */
2006-04-04 00:27:50 +00:00
}