2011-03-22 08:17:39 +03:00
/*
Unix SMB / CIFS implementation .
Samba utility functions
Copyright ( C ) Andrew Tridgell 1992 - 2011
based on old fault . c code , which had :
Copyright ( C ) Jeremy Allison 2001 - 2007
Copyright ( C ) Simo Sorce 2001
Copyright ( C ) Jim McDonough < jmcd @ us . ibm . com > 2003
Copyright ( C ) James Peach 2006
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
the Free Software Foundation ; either version 3 of the License , or
( 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
along with this program . If not , see < http : //www.gnu.org/licenses/>.
*/
# include "includes.h"
2011-02-26 01:20:06 +03:00
# include "system/filesys.h"
2011-03-22 08:17:39 +03:00
2011-03-23 19:24:25 +03:00
# ifdef HAVE_SYS_SYSCTL_H
# include <sys/sysctl.h>
# endif
2011-03-23 19:27:21 +03:00
# ifdef HAVE_SYS_PRCTL_H
# include <sys/prctl.h>
# endif
2011-03-22 08:17:39 +03:00
static char * corepath ;
/**
* 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 ;
}
/**
* 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
# if defined(HAVE_SYS_KERNEL_PROC_CORE_PATTERN)
/**
* Get the Linux corepath .
*
* On Linux the contents of / proc / sys / kernel / core_pattern indicates the
* location of the core path .
*/
static char * get_linux_corepath ( void )
{
char * end ;
int fd ;
char * result ;
fd = open ( " /proc/sys/kernel/core_pattern " , O_RDONLY , 0 ) ;
if ( fd = = - 1 ) {
return NULL ;
}
result = afdgets ( fd , NULL , 0 ) ;
close ( fd ) ;
if ( result = = NULL ) {
return NULL ;
}
if ( result [ 0 ] ! = ' / ' ) {
/*
* No absolute path , use the default ( cwd )
*/
TALLOC_FREE ( result ) ;
return NULL ;
}
/* Strip off the common filename expansion */
end = strrchr_m ( result , ' / ' ) ;
if ( ( end ! = result ) /* this would be the only / */
& & ( end ! = NULL ) ) {
* end = ' \0 ' ;
}
return result ;
}
# 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 )
{
# if (defined(FREEBSD) && defined(HAVE_SYSCTLBYNAME))
char * tmp_corepath = NULL ;
tmp_corepath = get_freebsd_corepath ( ) ;
/* If this has been set correctly, we're done. */
if ( tmp_corepath ) {
return tmp_corepath ;
}
# endif
# if defined(HAVE_SYS_KERNEL_PROC_CORE_PATTERN)
char * tmp_corepath = NULL ;
tmp_corepath = get_linux_corepath ( ) ;
/* If this has been set correctly, we're done. */
if ( tmp_corepath ) {
return tmp_corepath ;
}
# endif
/* Fall back to the default. */
return get_default_corepath ( logbase , progname ) ;
}
/*******************************************************************
make all the preparations to safely dump a core file
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2011-03-23 19:27:57 +03:00
void dump_core_setup ( const char * progname , const char * log_file )
2011-03-22 08:17:39 +03:00
{
char * logbase = NULL ;
char * end = NULL ;
2011-03-23 19:27:57 +03:00
if ( log_file & & * log_file ) {
if ( asprintf ( & logbase , " %s " , log_file ) < 0 ) {
2011-03-22 08:17:39 +03:00
return ;
}
if ( ( end = strrchr_m ( logbase , ' / ' ) ) ) {
* end = ' \0 ' ;
}
} else {
/* We will end up here if the log file is given on the command
* line by the - l option but the " log file " option is not set
* in smb . conf .
*/
if ( asprintf ( & logbase , " %s " , get_dyn_LOGFILEBASE ( ) ) < 0 ) {
return ;
}
}
SMB_ASSERT ( progname ! = NULL ) ;
corepath = get_corepath ( logbase , progname ) ;
if ( ! corepath ) {
DEBUG ( 0 , ( " Unable to setup corepath for %s: %s \n " , progname ,
strerror ( errno ) ) ) ;
goto out ;
}
# 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
/* FIXME: if we have a core-plus-pid facility, configurably set
* this up here .
*/
out :
SAFE_FREE ( logbase ) ;
}
void dump_core ( void )
{
static bool called ;
if ( called ) {
DEBUG ( 0 , ( " dump_core() called recursive \n " ) ) ;
exit ( 1 ) ;
}
called = true ;
/* 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 ( ) ) {
DEBUG ( 0 , ( " Exiting on internal error (core file administratively disabled) \n " ) ) ;
exit ( 1 ) ;
}
# if DUMP_CORE
/* 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 ( ) . */
if ( geteuid ( ) ! = sec_initial_uid ( ) ) {
become_root ( ) ;
}
if ( corepath = = NULL ) {
DEBUG ( 0 , ( " Can not dump core: corepath not set up \n " ) ) ;
exit ( 1 ) ;
}
if ( * corepath ! = ' \0 ' ) {
/* The chdir might fail if we dump core before we finish
* processing the config file .
*/
if ( chdir ( corepath ) ! = 0 ) {
DEBUG ( 0 , ( " unable to change to %s \n " , corepath ) ) ;
DEBUGADD ( 0 , ( " refusing to dump core \n " ) ) ;
exit ( 1 ) ;
}
DEBUG ( 0 , ( " dumping core in %s \n " , corepath ) ) ;
}
umask ( ~ ( 0700 ) ) ;
dbgflush ( ) ;
# 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
/* Ensure we don't have a signal handler for abort. */
# ifdef SIGABRT
CatchSignal ( SIGABRT , SIG_DFL ) ;
# endif
abort ( ) ;
# else /* DUMP_CORE */
exit ( 1 ) ;
# endif /* DUMP_CORE */
}