2003-08-13 01:53:07 +00:00
/*
Unix SMB / CIFS implementation .
Samba utility functions
Copyright ( C ) Andrew Tridgell 1992 - 1998
Copyright ( C ) Jeremy Allison 2001 - 2002
2011-04-21 10:04:06 -04:00
Copyright ( C ) Simo Sorce 2001 - 2011
2004-07-28 13:08:08 +00:00
Copyright ( C ) Jim McDonough ( jmcd @ us . ibm . com ) 2003.
2003-08-13 01:53:07 +00:00
Copyright ( C ) James J Myers 2003
2011-04-21 09:45:27 -04:00
Copyright ( C ) Volker Lendecke 2010
2019-03-06 09:03:27 +01:00
Copyright ( C ) Swen Schillig 2019
2003-08-13 01:53:07 +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-10 02:07:03 +00:00
the Free Software Foundation ; either version 3 of the License , or
2003-08-13 01:53:07 +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/>.
2003-08-13 01:53:07 +00:00
*/
2014-09-22 18:28:53 +10:00
# include "replace.h"
2015-07-02 20:09:02 +03:00
# include <talloc.h>
2004-11-01 22:48:25 +00:00
# include "system/network.h"
2005-02-10 05:09:35 +00:00
# include "system/filesys.h"
2006-05-13 19:14:12 +00:00
# include "system/locale.h"
2010-03-22 08:27:58 +01:00
# include "system/shmem.h"
2011-10-27 12:00:53 +02:00
# include "system/passwd.h"
2014-09-22 18:28:53 +10:00
# include "system/time.h"
# include "system/wait.h"
# include "debug.h"
# include "samba_util.h"
2017-01-08 17:54:06 +00:00
# include "lib/util/select.h"
2010-03-22 08:27:58 +01:00
2008-10-17 12:26:46 +02:00
# undef malloc
# undef strcasecmp
2008-10-19 12:38:16 +02:00
# undef strncasecmp
2008-10-17 12:26:46 +02:00
# undef strdup
# undef realloc
2012-03-24 17:16:26 +01:00
# undef calloc
2003-08-13 01:53:07 +00:00
2006-02-28 13:12:39 +00:00
/**
* @ file
* @ brief Misc utility functions
*/
2019-01-28 09:42:13 +01:00
/**
* Convert a string to an unsigned long integer
*
* @ param nptr pointer to string which is to be converted
* @ param endptr [ optional ] reference to remainder of the string
* @ param base base of the numbering scheme
* @ param err error occured during conversion
2019-04-11 11:22:02 +02:00
* @ flags controlling conversion feature
2019-01-28 09:42:13 +01:00
* @ result result of the conversion as provided by strtoul
*
2019-04-11 11:22:02 +02:00
* The following flags are supported
* SMB_STR_STANDARD # raise error if negative or non - numeric
* SMB_STR_ALLOW_NEGATIVE # allow strings with a leading " - "
2019-04-11 14:42:37 +02:00
* SMB_STR_FULL_STR_CONV # entire string must be converted
2019-04-11 14:46:49 +02:00
* SMB_STR_ALLOW_NO_CONVERSION # allow empty strings or non - numeric
* SMB_STR_GLIBC_STANDARD # act exactly as the standard glibc strtoul
2019-04-11 11:22:02 +02:00
*
2019-02-05 08:39:14 +01:00
* The following errors are detected
* - wrong base
* - value overflow
* - string with a leading " - " indicating a negative number
2019-03-06 09:03:27 +01:00
* - no conversion due to empty string or not representing a number
2019-01-28 09:42:13 +01:00
*/
unsigned long int
2019-04-11 11:22:02 +02:00
smb_strtoul ( const char * nptr , char * * endptr , int base , int * err , int flags )
2019-01-28 09:42:13 +01:00
{
unsigned long int val ;
int saved_errno = errno ;
2019-02-05 08:39:14 +01:00
char * needle = NULL ;
char * tmp_endptr = NULL ;
2019-01-28 09:42:13 +01:00
errno = 0 ;
2019-02-05 08:39:14 +01:00
* err = 0 ;
val = strtoul ( nptr , & tmp_endptr , base ) ;
if ( endptr ! = NULL ) {
* endptr = tmp_endptr ;
}
if ( errno ! = 0 ) {
* err = errno ;
errno = saved_errno ;
return val ;
}
2019-04-11 14:46:49 +02:00
if ( ( flags & SMB_STR_ALLOW_NO_CONVERSION ) = = 0 ) {
/* got an invalid number-string resulting in no conversion */
if ( nptr = = tmp_endptr ) {
* err = EINVAL ;
goto out ;
}
2019-03-06 09:03:27 +01:00
}
2019-04-11 11:22:02 +02:00
if ( ( flags & SMB_STR_ALLOW_NEGATIVE ) = = 0 ) {
/* did we convert a negative "number" ? */
needle = strchr ( nptr , ' - ' ) ;
if ( needle ! = NULL & & needle < tmp_endptr ) {
* err = EINVAL ;
2019-04-11 14:42:37 +02:00
goto out ;
2019-04-11 11:22:02 +02:00
}
2019-02-05 08:39:14 +01:00
}
2019-01-28 09:42:13 +01:00
2019-04-11 14:42:37 +02:00
if ( ( flags & SMB_STR_FULL_STR_CONV ) ! = 0 ) {
/* did we convert the entire string ? */
if ( tmp_endptr [ 0 ] ! = ' \0 ' ) {
* err = EINVAL ;
goto out ;
}
}
out :
2019-01-28 09:42:13 +01:00
errno = saved_errno ;
return val ;
}
/**
* Convert a string to an unsigned long long integer
*
* @ param nptr pointer to string which is to be converted
* @ param endptr [ optional ] reference to remainder of the string
* @ param base base of the numbering scheme
* @ param err error occured during conversion
2019-04-11 11:22:02 +02:00
* @ flags controlling conversion feature
2019-01-28 09:42:13 +01:00
* @ result result of the conversion as provided by strtoull
*
2019-04-11 11:22:02 +02:00
* The following flags are supported
* SMB_STR_STANDARD # raise error if negative or non - numeric
* SMB_STR_ALLOW_NEGATIVE # allow strings with a leading " - "
2019-04-11 14:42:37 +02:00
* SMB_STR_FULL_STR_CONV # entire string must be converted
2019-04-11 14:46:49 +02:00
* SMB_STR_ALLOW_NO_CONVERSION # allow empty strings or non - numeric
* SMB_STR_GLIBC_STANDARD # act exactly as the standard glibc strtoul
2019-04-11 11:22:02 +02:00
*
2019-02-05 08:39:14 +01:00
* The following errors are detected
* - wrong base
* - value overflow
* - string with a leading " - " indicating a negative number
2019-03-06 09:03:27 +01:00
* - no conversion due to empty string or not representing a number
2019-01-28 09:42:13 +01:00
*/
unsigned long long int
2019-04-11 11:22:02 +02:00
smb_strtoull ( const char * nptr , char * * endptr , int base , int * err , int flags )
2019-01-28 09:42:13 +01:00
{
unsigned long long int val ;
int saved_errno = errno ;
2019-02-05 08:39:14 +01:00
char * needle = NULL ;
char * tmp_endptr = NULL ;
2019-01-28 09:42:13 +01:00
errno = 0 ;
2019-02-05 08:39:14 +01:00
* err = 0 ;
val = strtoull ( nptr , & tmp_endptr , base ) ;
if ( endptr ! = NULL ) {
* endptr = tmp_endptr ;
}
if ( errno ! = 0 ) {
* err = errno ;
errno = saved_errno ;
return val ;
}
2019-04-11 14:46:49 +02:00
if ( ( flags & SMB_STR_ALLOW_NO_CONVERSION ) = = 0 ) {
/* got an invalid number-string resulting in no conversion */
if ( nptr = = tmp_endptr ) {
* err = EINVAL ;
goto out ;
}
2019-03-06 09:03:27 +01:00
}
2019-04-11 11:22:02 +02:00
if ( ( flags & SMB_STR_ALLOW_NEGATIVE ) = = 0 ) {
/* did we convert a negative "number" ? */
needle = strchr ( nptr , ' - ' ) ;
if ( needle ! = NULL & & needle < tmp_endptr ) {
* err = EINVAL ;
2019-04-11 14:42:37 +02:00
goto out ;
}
}
if ( ( flags & SMB_STR_FULL_STR_CONV ) ! = 0 ) {
/* did we convert the entire string ? */
if ( tmp_endptr [ 0 ] ! = ' \0 ' ) {
* err = EINVAL ;
goto out ;
2019-04-11 11:22:02 +02:00
}
2019-02-05 08:39:14 +01:00
}
2019-01-28 09:42:13 +01:00
2019-04-11 14:42:37 +02:00
out :
2019-01-28 09:42:13 +01:00
errno = saved_errno ;
return val ;
}
2006-02-28 13:12:39 +00:00
/**
2003-08-13 01:53:07 +00:00
Find a suitable temporary directory . The result should be copied immediately
as it may be overwritten by a subsequent call .
2006-02-28 13:12:39 +00:00
* */
2006-03-05 17:15:19 +00:00
_PUBLIC_ const char * tmpdir ( void )
2003-08-13 01:53:07 +00:00
{
char * p ;
if ( ( p = getenv ( " TMPDIR " ) ) )
return p ;
return " /tmp " ;
}
2011-04-21 09:45:27 -04:00
/**
Create a tmp file , open it and immediately unlink it .
2011-04-21 10:04:06 -04:00
If dir is NULL uses tmpdir ( )
2011-04-21 09:45:27 -04:00
Returns the file descriptor or - 1 on error .
* */
int create_unlink_tmp ( const char * dir )
{
2016-02-25 14:11:10 +01:00
size_t len = strlen ( dir ? dir : ( dir = tmpdir ( ) ) ) ;
2014-09-19 16:58:52 +02:00
char fname [ len + 25 ] ;
2011-04-21 09:45:27 -04:00
int fd ;
2013-02-25 17:34:21 +01:00
mode_t mask ;
2011-04-21 09:45:27 -04:00
2014-09-19 16:58:52 +02:00
len = snprintf ( fname , sizeof ( fname ) , " %s/listenerlock_XXXXXX " , dir ) ;
if ( len > = sizeof ( fname ) ) {
2011-04-21 09:45:27 -04:00
errno = ENOMEM ;
return - 1 ;
}
2013-02-25 17:34:21 +01:00
mask = umask ( S_IRWXO | S_IRWXG ) ;
2011-04-21 09:45:27 -04:00
fd = mkstemp ( fname ) ;
2013-02-25 17:34:21 +01:00
umask ( mask ) ;
2011-04-21 09:45:27 -04:00
if ( fd = = - 1 ) {
return - 1 ;
}
if ( unlink ( fname ) = = - 1 ) {
int sys_errno = errno ;
close ( fd ) ;
errno = sys_errno ;
return - 1 ;
}
return fd ;
}
2006-02-28 13:12:39 +00:00
/**
2003-08-13 01:53:07 +00:00
Check if a file exists - call vfs_file_exist for samba files .
2006-02-28 13:12:39 +00:00
* */
2007-08-27 17:21:16 +00:00
_PUBLIC_ bool file_exist ( const char * fname )
2003-08-13 01:53:07 +00:00
{
2004-11-01 20:21:54 +00:00
struct stat st ;
2003-08-13 01:53:07 +00:00
2005-02-10 06:36:30 +00:00
if ( stat ( fname , & st ) ! = 0 ) {
2007-08-27 17:21:16 +00:00
return false ;
2005-02-10 06:36:30 +00:00
}
return ( ( S_ISREG ( st . st_mode ) ) | | ( S_ISFIFO ( st . st_mode ) ) ) ;
2003-08-13 01:53:07 +00:00
}
2006-02-28 13:12:39 +00:00
/**
2003-08-13 01:53:07 +00:00
Check a files mod time .
2006-02-28 13:12:39 +00:00
* */
2003-08-13 01:53:07 +00:00
2006-03-05 17:15:19 +00:00
_PUBLIC_ time_t file_modtime ( const char * fname )
2003-08-13 01:53:07 +00:00
{
2004-11-01 20:21:54 +00:00
struct stat st ;
2003-08-13 01:53:07 +00:00
2004-11-01 20:21:54 +00:00
if ( stat ( fname , & st ) ! = 0 )
2003-08-13 01:53:07 +00:00
return ( 0 ) ;
return ( st . st_mtime ) ;
}
2013-10-29 17:43:17 +01:00
/**
Check file permissions .
* */
_PUBLIC_ bool file_check_permissions ( const char * fname ,
uid_t uid ,
mode_t file_perms ,
struct stat * pst )
{
int ret ;
struct stat st ;
if ( pst = = NULL ) {
pst = & st ;
}
ZERO_STRUCTP ( pst ) ;
ret = stat ( fname , pst ) ;
if ( ret ! = 0 ) {
DEBUG ( 0 , ( " stat failed on file '%s': %s \n " ,
fname , strerror ( errno ) ) ) ;
return false ;
}
2014-01-17 14:43:01 +01:00
if ( pst - > st_uid ! = uid & & ! uid_wrapper_enabled ( ) ) {
2013-10-29 17:43:17 +01:00
DEBUG ( 0 , ( " invalid ownership of file '%s': "
" owned by uid %u, should be %u \n " ,
fname , ( unsigned int ) pst - > st_uid ,
( unsigned int ) uid ) ) ;
return false ;
}
if ( ( pst - > st_mode & 0777 ) ! = file_perms ) {
DEBUG ( 0 , ( " invalid permissions on file "
" '%s': has 0%o should be 0%o \n " , fname ,
( unsigned int ) ( pst - > st_mode & 0777 ) ,
( unsigned int ) file_perms ) ) ;
return false ;
}
return true ;
}
2006-02-28 13:12:39 +00:00
/**
2003-08-13 01:53:07 +00:00
Check if a directory exists .
2006-02-28 13:12:39 +00:00
* */
2003-08-13 01:53:07 +00:00
2007-08-27 17:21:16 +00:00
_PUBLIC_ bool directory_exist ( const char * dname )
2003-08-13 01:53:07 +00:00
{
2005-02-10 06:36:30 +00:00
struct stat st ;
2007-08-27 17:21:16 +00:00
bool ret ;
2003-08-13 01:53:07 +00:00
2005-02-10 06:36:30 +00:00
if ( stat ( dname , & st ) ! = 0 ) {
2007-08-27 17:21:16 +00:00
return false ;
2005-02-10 06:36:30 +00:00
}
2003-08-13 01:53:07 +00:00
2005-02-10 06:36:30 +00:00
ret = S_ISDIR ( st . st_mode ) ;
2003-08-13 01:53:07 +00:00
if ( ! ret )
errno = ENOTDIR ;
return ret ;
}
2006-03-06 00:24:51 +00:00
/**
* Try to create the specified directory if it didn ' t exist .
*
2014-07-28 13:03:54 -07:00
* @ retval true if the directory already existed
2006-03-06 00:24:51 +00:00
* or was successfully created .
*/
2013-01-08 14:21:00 +01:00
_PUBLIC_ bool directory_create_or_exist ( const char * dname ,
mode_t dir_perms )
2006-01-31 00:48:57 +00:00
{
2012-03-14 10:46:14 +01:00
int ret ;
2014-07-27 19:08:52 +02:00
mode_t old_umask ;
2013-01-08 14:21:00 +01:00
2014-07-27 19:08:52 +02:00
/* Create directory */
old_umask = umask ( 0 ) ;
ret = mkdir ( dname , dir_perms ) ;
if ( ret = = - 1 & & errno ! = EEXIST ) {
2020-01-27 14:58:10 +01:00
int dbg_level = geteuid ( ) = = 0 ? DBGLVL_ERR : DBGLVL_NOTICE ;
DBG_PREFIX ( dbg_level ,
( " mkdir failed on directory %s: %s \n " ,
2019-03-07 12:31:42 +01:00
dname ,
2020-01-27 14:58:10 +01:00
strerror ( errno ) ) ) ;
2012-03-14 10:52:50 +01:00
umask ( old_umask ) ;
2014-07-27 19:08:52 +02:00
return false ;
}
umask ( old_umask ) ;
2012-03-14 10:43:54 +01:00
2018-08-29 12:04:29 -07:00
if ( ret ! = 0 & & errno = = EEXIST ) {
struct stat sbuf ;
ret = lstat ( dname , & sbuf ) ;
if ( ret ! = 0 ) {
return false ;
}
if ( ! S_ISDIR ( sbuf . st_mode ) ) {
return false ;
}
2006-01-31 00:48:57 +00:00
}
2012-03-14 10:41:47 +01:00
2013-01-08 14:21:00 +01:00
return true ;
}
/**
* @ brief Try to create a specified directory if it doesn ' t exist .
*
* The function creates a directory with the given uid and permissions if it
2013-12-11 17:47:36 +01:00
* doesn ' t exist . If it exists it makes sure the uid and permissions are
2013-01-08 14:21:00 +01:00
* correct and it will fail if they are different .
*
* @ param [ in ] dname The directory to create .
*
* @ param [ in ] uid The uid the directory needs to belong too .
*
* @ param [ in ] dir_perms The expected permissions of the directory .
*
* @ return True on success , false on error .
*/
_PUBLIC_ bool directory_create_or_exist_strict ( const char * dname ,
uid_t uid ,
mode_t dir_perms )
{
struct stat st ;
bool ok ;
int rc ;
2014-07-27 19:18:09 +02:00
ok = directory_create_or_exist ( dname , dir_perms ) ;
2013-01-08 14:21:00 +01:00
if ( ! ok ) {
return false ;
}
rc = lstat ( dname , & st ) ;
if ( rc = = - 1 ) {
DEBUG ( 0 , ( " lstat failed on created directory %s: %s \n " ,
dname , strerror ( errno ) ) ) ;
return false ;
}
2012-03-14 10:41:47 +01:00
/* Check ownership and permission on existing directory */
if ( ! S_ISDIR ( st . st_mode ) ) {
DEBUG ( 0 , ( " directory %s isn't a directory \n " ,
dname ) ) ;
return false ;
}
2014-01-17 14:43:01 +01:00
if ( st . st_uid ! = uid & & ! uid_wrapper_enabled ( ) ) {
2015-10-27 17:08:50 -07:00
DBG_NOTICE ( " invalid ownership on directory "
" %s \n " , dname ) ;
2012-03-14 10:41:47 +01:00
return false ;
}
if ( ( st . st_mode & 0777 ) ! = dir_perms ) {
DEBUG ( 0 , ( " invalid permissions on directory "
" '%s': has 0%o should be 0%o \n " , dname ,
2013-03-14 20:14:19 +11:00
( unsigned int ) ( st . st_mode & 0777 ) , ( unsigned int ) dir_perms ) ) ;
2012-03-14 10:41:47 +01:00
return false ;
}
2007-08-27 17:21:16 +00:00
return true ;
2013-01-08 14:21:00 +01:00
}
2006-01-31 00:48:57 +00:00
2006-02-28 13:12:39 +00:00
/**
2003-08-13 01:53:07 +00:00
Sleep for a specified number of milliseconds .
2006-02-28 13:12:39 +00:00
* */
2003-08-13 01:53:07 +00:00
2010-09-16 21:36:37 +02:00
_PUBLIC_ void smb_msleep ( unsigned int t )
2003-08-13 01:53:07 +00:00
{
2017-01-08 17:54:06 +00:00
sys_poll_intr ( NULL , 0 , t ) ;
2003-08-13 01:53:07 +00:00
}
2006-02-28 13:12:39 +00:00
/**
2009-02-13 10:56:34 +01:00
Get my own name , return in talloc ' ed storage .
2006-02-28 13:12:39 +00:00
* */
2003-08-13 01:53:07 +00:00
2009-02-13 10:56:34 +01:00
_PUBLIC_ char * get_myname ( TALLOC_CTX * ctx )
2003-08-13 01:53:07 +00:00
{
char * p ;
2009-02-13 10:56:34 +01:00
char hostname [ HOST_NAME_MAX ] ;
2003-08-13 01:53:07 +00:00
/* get my host name */
2009-02-13 10:56:34 +01:00
if ( gethostname ( hostname , sizeof ( hostname ) ) = = - 1 ) {
2003-08-13 01:53:07 +00:00
DEBUG ( 0 , ( " gethostname failed \n " ) ) ;
return NULL ;
2009-02-13 10:56:34 +01:00
}
2003-08-13 01:53:07 +00:00
/* Ensure null termination. */
2009-02-13 10:56:34 +01:00
hostname [ sizeof ( hostname ) - 1 ] = ' \0 ' ;
2003-08-13 01:53:07 +00:00
/* split off any parts after an initial . */
2009-02-13 10:56:34 +01:00
p = strchr_m ( hostname , ' . ' ) ;
if ( p ) {
2003-08-13 01:53:07 +00:00
* p = 0 ;
2009-02-13 10:56:34 +01:00
}
return talloc_strdup ( ctx , hostname ) ;
2003-08-13 01:53:07 +00:00
}
2006-02-28 13:12:39 +00:00
/**
2003-08-13 01:53:07 +00:00
Check if a process exists . Does this work on all unixes ?
2006-02-28 13:12:39 +00:00
* */
2003-08-13 01:53:07 +00:00
2008-10-17 12:26:46 +02:00
_PUBLIC_ bool process_exists_by_pid ( pid_t pid )
2003-08-13 01:53:07 +00:00
{
/* Doing kill with a non-positive pid causes messages to be
* sent to places we don ' t want . */
2012-08-22 17:52:01 +02:00
if ( pid < = 0 ) {
return false ;
}
2003-08-13 01:53:07 +00:00
return ( kill ( pid , 0 ) = = 0 | | errno ! = ESRCH ) ;
}
2006-02-28 13:12:39 +00:00
/**
2003-08-13 01:53:07 +00:00
Simple routine to do POSIX file locking . Cruft in NFS and 64 - > 32 bit mapping
is dealt with in posix . c
2006-02-28 13:12:39 +00:00
* */
2003-08-13 01:53:07 +00:00
2007-08-27 17:21:16 +00:00
_PUBLIC_ bool fcntl_lock ( int fd , int op , off_t offset , off_t count , int type )
2003-08-13 01:53:07 +00:00
{
2004-11-01 20:21:54 +00:00
struct flock lock ;
2003-08-13 01:53:07 +00:00
int ret ;
DEBUG ( 8 , ( " fcntl_lock %d %d %.0f %.0f %d \n " , fd , op , ( double ) offset , ( double ) count , type ) ) ;
lock . l_type = type ;
lock . l_whence = SEEK_SET ;
lock . l_start = offset ;
lock . l_len = count ;
lock . l_pid = 0 ;
2004-11-01 20:21:54 +00:00
ret = fcntl ( fd , op , & lock ) ;
2003-08-13 01:53:07 +00:00
if ( ret = = - 1 & & errno ! = 0 )
DEBUG ( 3 , ( " fcntl_lock: fcntl lock gave errno %d (%s) \n " , errno , strerror ( errno ) ) ) ;
/* a lock query */
2004-11-01 20:21:54 +00:00
if ( op = = F_GETLK ) {
2003-08-13 01:53:07 +00:00
if ( ( ret ! = - 1 ) & &
( lock . l_type ! = F_UNLCK ) & &
( lock . l_pid ! = 0 ) & &
( lock . l_pid ! = getpid ( ) ) ) {
DEBUG ( 3 , ( " fcntl_lock: fd %d is locked by pid %d \n " , fd , ( int ) lock . l_pid ) ) ;
2007-08-27 17:21:16 +00:00
return true ;
2003-08-13 01:53:07 +00:00
}
/* it must be not locked or locked by me */
2007-08-27 17:21:16 +00:00
return false ;
2003-08-13 01:53:07 +00:00
}
/* a lock set or unset */
if ( ret = = - 1 ) {
DEBUG ( 3 , ( " fcntl_lock: lock failed at offset %.0f count %.0f op %d type %d (%s) \n " ,
( double ) offset , ( double ) count , op , type , strerror ( errno ) ) ) ;
2007-08-27 17:21:16 +00:00
return false ;
2003-08-13 01:53:07 +00:00
}
/* everything went OK */
DEBUG ( 8 , ( " fcntl_lock: Lock call successful \n " ) ) ;
2007-08-27 17:21:16 +00:00
return true ;
2003-08-13 01:53:07 +00:00
}
2013-01-18 09:44:02 +01:00
struct debug_channel_level {
int channel ;
int level ;
} ;
static void debugadd_channel_cb ( const char * buf , void * private_data )
{
struct debug_channel_level * dcl =
( struct debug_channel_level * ) private_data ;
DEBUGADDC ( dcl - > channel , dcl - > level , ( " %s " , buf ) ) ;
}
2010-10-24 15:23:41 +02:00
static void debugadd_cb ( const char * buf , void * private_data )
{
int * plevel = ( int * ) private_data ;
DEBUGADD ( * plevel , ( " %s " , buf ) ) ;
}
void print_asc_cb ( const uint8_t * buf , int len ,
void ( * cb ) ( const char * buf , void * private_data ) ,
void * private_data )
2003-08-13 01:53:07 +00:00
{
int i ;
2010-10-24 15:23:41 +02:00
char s [ 2 ] ;
s [ 1 ] = 0 ;
for ( i = 0 ; i < len ; i + + ) {
s [ 0 ] = isprint ( buf [ i ] ) ? buf [ i ] : ' . ' ;
cb ( s , private_data ) ;
}
}
void print_asc ( int level , const uint8_t * buf , int len )
{
print_asc_cb ( buf , len , debugadd_cb , & level ) ;
2003-08-13 01:53:07 +00:00
}
2006-03-06 00:24:51 +00:00
/**
2010-10-24 15:42:45 +02:00
* Write dump of binary data to a callback
2006-03-06 00:24:51 +00:00
*/
2010-10-24 15:42:45 +02:00
void dump_data_cb ( const uint8_t * buf , int len ,
bool omit_zero_bytes ,
void ( * cb ) ( const char * buf , void * private_data ) ,
void * private_data )
2003-08-13 01:53:07 +00:00
{
int i = 0 ;
2007-11-02 11:33:53 +01:00
bool skipped = false ;
2010-10-24 15:42:45 +02:00
char tmp [ 16 ] ;
2007-11-02 11:33:53 +01:00
2003-08-13 01:53:07 +00:00
if ( len < = 0 ) return ;
for ( i = 0 ; i < len ; ) {
2007-11-02 11:33:53 +01:00
if ( i % 16 = = 0 ) {
if ( ( omit_zero_bytes = = true ) & &
( i > 0 ) & &
( len > i + 16 ) & &
2017-10-19 17:52:31 +02:00
all_zero ( & buf [ i ] , 16 ) )
2007-11-02 11:33:53 +01:00
{
i + = 16 ;
continue ;
}
if ( i < len ) {
2010-10-24 15:42:45 +02:00
snprintf ( tmp , sizeof ( tmp ) , " [%04X] " , i ) ;
cb ( tmp , private_data ) ;
2007-11-02 11:33:53 +01:00
}
}
2010-10-24 15:42:45 +02:00
snprintf ( tmp , sizeof ( tmp ) , " %02X " , ( int ) buf [ i ] ) ;
cb ( tmp , private_data ) ;
2003-08-13 01:53:07 +00:00
i + + ;
2010-10-24 15:42:45 +02:00
if ( i % 8 = = 0 ) {
cb ( " " , private_data ) ;
}
2007-11-02 11:33:53 +01:00
if ( i % 16 = = 0 ) {
2010-10-24 15:42:45 +02:00
print_asc_cb ( & buf [ i - 16 ] , 8 , cb , private_data ) ;
cb ( " " , private_data ) ;
print_asc_cb ( & buf [ i - 8 ] , 8 , cb , private_data ) ;
cb ( " \n " , private_data ) ;
2007-11-02 11:33:53 +01:00
if ( ( omit_zero_bytes = = true ) & &
( len > i + 16 ) & &
2017-10-19 17:52:31 +02:00
all_zero ( & buf [ i ] , 16 ) ) {
2007-11-02 11:33:53 +01:00
if ( ! skipped ) {
2010-10-24 15:42:45 +02:00
cb ( " skipping zero buffer bytes \n " ,
private_data ) ;
2007-11-02 11:33:53 +01:00
skipped = true ;
}
}
2003-08-13 01:53:07 +00:00
}
}
2007-11-02 11:33:53 +01:00
2003-08-13 01:53:07 +00:00
if ( i % 16 ) {
int n ;
n = 16 - ( i % 16 ) ;
2015-06-25 13:53:41 +02:00
cb ( " " , private_data ) ;
2010-10-24 15:42:45 +02:00
if ( n > 8 ) {
cb ( " " , private_data ) ;
}
while ( n - - ) {
cb ( " " , private_data ) ;
}
2003-08-13 01:53:07 +00:00
n = MIN ( 8 , i % 16 ) ;
2010-10-24 15:42:45 +02:00
print_asc_cb ( & buf [ i - ( i % 16 ) ] , n , cb , private_data ) ;
cb ( " " , private_data ) ;
2003-08-13 01:53:07 +00:00
n = ( i % 16 ) - n ;
2010-10-24 15:42:45 +02:00
if ( n > 0 ) {
print_asc_cb ( & buf [ i - n ] , n , cb , private_data ) ;
}
cb ( " \n " , private_data ) ;
2007-11-02 11:33:53 +01:00
}
2003-08-13 01:53:07 +00:00
}
2007-11-02 11:33:53 +01:00
/**
* Write dump of binary data to the log file .
*
* The data is only written if the log level is at least level .
*/
_PUBLIC_ void dump_data ( int level , const uint8_t * buf , int len )
{
2010-10-24 15:42:45 +02:00
if ( ! DEBUGLVL ( level ) ) {
return ;
}
dump_data_cb ( buf , len , false , debugadd_cb , & level ) ;
2007-11-02 11:33:53 +01:00
}
2013-01-18 09:44:02 +01:00
/**
* Write dump of binary data to the log file .
*
* The data is only written if the log level is at least level for
* debug class dbgc_class .
*/
_PUBLIC_ void dump_data_dbgc ( int dbgc_class , int level , const uint8_t * buf , int len )
{
struct debug_channel_level dcl = { dbgc_class , level } ;
if ( ! DEBUGLVLC ( dbgc_class , level ) ) {
return ;
}
dump_data_cb ( buf , len , false , debugadd_channel_cb , & dcl ) ;
}
2007-11-02 11:33:53 +01:00
/**
* Write dump of binary data to the log file .
*
* The data is only written if the log level is at least level .
2010-02-21 16:58:07 +11:00
* 16 zero bytes in a row are omitted
2007-11-02 11:33:53 +01:00
*/
_PUBLIC_ void dump_data_skip_zeros ( int level , const uint8_t * buf , int len )
{
2010-10-24 15:42:45 +02:00
if ( ! DEBUGLVL ( level ) ) {
return ;
}
dump_data_cb ( buf , len , true , debugadd_cb , & level ) ;
2007-11-02 11:33:53 +01:00
}
2012-01-25 09:10:04 +01:00
static void fprintf_cb ( const char * buf , void * private_data )
{
FILE * f = ( FILE * ) private_data ;
fprintf ( f , " %s " , buf ) ;
}
void dump_data_file ( const uint8_t * buf , int len , bool omit_zero_bytes ,
FILE * f )
{
dump_data_cb ( buf , len , omit_zero_bytes , fprintf_cb , f ) ;
}
2007-11-02 11:33:53 +01:00
2006-02-28 13:12:39 +00:00
/**
2003-08-13 01:53:07 +00:00
malloc that aborts with smb_panic on fail or zero size .
2006-02-28 13:12:39 +00:00
* */
2003-08-13 01:53:07 +00:00
2006-03-05 17:15:19 +00:00
_PUBLIC_ void * smb_xmalloc ( size_t size )
2003-08-13 01:53:07 +00:00
{
void * p ;
if ( size = = 0 )
smb_panic ( " smb_xmalloc: called with zero size. \n " ) ;
if ( ( p = malloc ( size ) ) = = NULL )
smb_panic ( " smb_xmalloc: malloc fail. \n " ) ;
return p ;
}
/**
Memdup with smb_panic on fail .
* */
2006-03-05 17:15:19 +00:00
_PUBLIC_ void * smb_xmemdup ( const void * p , size_t size )
2003-08-13 01:53:07 +00:00
{
void * p2 ;
p2 = smb_xmalloc ( size ) ;
memcpy ( p2 , p , size ) ;
return p2 ;
}
/**
strdup that aborts on malloc fail .
* */
2008-10-17 12:26:46 +02:00
char * smb_xstrdup ( const char * s )
2003-08-13 01:53:07 +00:00
{
2008-10-17 12:26:46 +02:00
# if defined(PARANOID_MALLOC_CHECKER)
# ifdef strdup
# undef strdup
# endif
# endif
# ifndef HAVE_STRDUP
# define strdup rep_strdup
# endif
2003-08-13 01:53:07 +00:00
char * s1 = strdup ( s ) ;
2008-10-17 12:26:46 +02:00
# if defined(PARANOID_MALLOC_CHECKER)
# ifdef strdup
# undef strdup
# endif
# define strdup(s) __ERROR_DONT_USE_STRDUP_DIRECTLY
# endif
if ( ! s1 ) {
smb_panic ( " smb_xstrdup: malloc failed " ) ;
}
return s1 ;
}
/**
strndup that aborts on malloc fail .
* */
char * smb_xstrndup ( const char * s , size_t n )
{
# if defined(PARANOID_MALLOC_CHECKER)
# ifdef strndup
# undef strndup
# endif
# endif
# if (defined(BROKEN_STRNDUP) || !defined(HAVE_STRNDUP))
# undef HAVE_STRNDUP
# define strndup rep_strndup
# endif
char * s1 = strndup ( s , n ) ;
# if defined(PARANOID_MALLOC_CHECKER)
# ifdef strndup
# undef strndup
# endif
# define strndup(s,n) __ERROR_DONT_USE_STRNDUP_DIRECTLY
# endif
if ( ! s1 ) {
smb_panic ( " smb_xstrndup: malloc failed " ) ;
}
2003-08-13 01:53:07 +00:00
return s1 ;
}
2008-10-17 12:26:46 +02:00
2006-02-28 13:12:39 +00:00
/**
2003-08-13 01:53:07 +00:00
Like strdup but for memory .
2006-02-28 13:12:39 +00:00
* */
2003-08-13 01:53:07 +00:00
2014-04-14 14:37:29 +02:00
_PUBLIC_ void * smb_memdup ( const void * p , size_t size )
2003-08-13 01:53:07 +00:00
{
void * p2 ;
if ( size = = 0 )
return NULL ;
p2 = malloc ( size ) ;
if ( ! p2 )
return NULL ;
memcpy ( p2 , p , size ) ;
return p2 ;
}
2006-03-06 00:24:51 +00:00
/**
* Write a password to the log file .
*
* @ note Only actually does something if DEBUG_PASSWORD was defined during
* compile - time .
*/
2006-03-05 17:15:19 +00:00
_PUBLIC_ void dump_data_pw ( const char * msg , const uint8_t * data , size_t len )
2003-11-26 01:16:41 +00:00
{
# ifdef DEBUG_PASSWORD
DEBUG ( 11 , ( " %s " , msg ) ) ;
if ( data ! = NULL & & len > 0 )
{
dump_data ( 11 , data , len ) ;
}
# endif
}
2003-12-13 02:20:40 +00:00
2006-02-28 13:12:39 +00:00
/**
* see if a range of memory is all zero . A NULL pointer is considered
* to be all zero
*/
2007-08-27 17:21:16 +00:00
_PUBLIC_ bool all_zero ( const uint8_t * ptr , size_t size )
2003-12-13 02:20:40 +00:00
{
2014-12-03 15:54:19 +01:00
size_t i ;
2007-08-27 17:21:16 +00:00
if ( ! ptr ) return true ;
2003-12-13 02:20:40 +00:00
for ( i = 0 ; i < size ; i + + ) {
2007-08-27 17:21:16 +00:00
if ( ptr [ i ] ) return false ;
2003-12-13 02:20:40 +00:00
}
2007-08-27 17:21:16 +00:00
return true ;
2003-12-13 02:20:40 +00:00
}
2004-12-03 06:24:38 +00:00
2006-02-28 13:12:39 +00:00
/**
2004-12-03 06:24:38 +00:00
realloc an array , checking for integer overflow in the array size
*/
2008-10-18 14:12:56 +02:00
_PUBLIC_ void * realloc_array ( void * ptr , size_t el_size , unsigned count , bool free_on_fail )
2004-12-03 06:24:38 +00:00
{
# define MAX_MALLOC_SIZE 0x7fffffff
if ( count = = 0 | |
count > = MAX_MALLOC_SIZE / el_size ) {
2008-10-18 14:12:56 +02:00
if ( free_on_fail )
SAFE_FREE ( ptr ) ;
2004-12-03 06:24:38 +00:00
return NULL ;
}
if ( ! ptr ) {
return malloc ( el_size * count ) ;
}
return realloc ( ptr , el_size * count ) ;
}
2008-10-11 19:44:10 +02:00
/****************************************************************************
Type - safe malloc .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
void * malloc_array ( size_t el_size , unsigned int count )
{
2008-10-18 14:12:56 +02:00
return realloc_array ( NULL , el_size , count , false ) ;
2008-10-11 19:44:10 +02:00
}
2012-03-24 15:19:09 +01:00
/****************************************************************************
Type - safe memalign
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
void * memalign_array ( size_t el_size , size_t align , unsigned int count )
{
2015-11-18 13:13:57 +01:00
if ( el_size = = 0 | | count > = MAX_MALLOC_SIZE / el_size ) {
2012-03-24 15:19:09 +01:00
return NULL ;
}
return memalign ( align , el_size * count ) ;
}
/****************************************************************************
Type - safe calloc .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
void * calloc_array ( size_t size , size_t nmemb )
{
if ( nmemb > = MAX_MALLOC_SIZE / size ) {
return NULL ;
}
if ( size = = 0 | | nmemb = = 0 ) {
return NULL ;
}
return calloc ( nmemb , size ) ;
}
2008-10-19 12:38:16 +02:00
/**
Trim the specified elements off the front and back of a string .
* */
_PUBLIC_ bool trim_string ( char * s , const char * front , const char * back )
{
bool ret = false ;
size_t front_len ;
size_t back_len ;
size_t len ;
/* Ignore null or empty strings. */
2016-06-19 06:47:09 +10:00
if ( ! s | | ( s [ 0 ] = = ' \0 ' ) ) {
2008-10-19 12:38:16 +02:00
return false ;
2016-06-19 06:47:09 +10:00
}
len = strlen ( s ) ;
2008-10-19 12:38:16 +02:00
front_len = front ? strlen ( front ) : 0 ;
back_len = back ? strlen ( back ) : 0 ;
if ( front_len ) {
2016-06-19 06:47:09 +10:00
size_t front_trim = 0 ;
while ( strncmp ( s + front_trim , front , front_len ) = = 0 ) {
front_trim + = front_len ;
}
if ( front_trim > 0 ) {
2008-10-19 12:38:16 +02:00
/* Must use memmove here as src & dest can
* easily overlap . Found by valgrind . JRA . */
2016-06-19 06:47:09 +10:00
memmove ( s , s + front_trim , ( len - front_trim ) + 1 ) ;
len - = front_trim ;
2008-10-19 12:38:16 +02:00
ret = true ;
}
}
2016-06-19 06:47:09 +10:00
2008-10-19 12:38:16 +02:00
if ( back_len ) {
while ( ( len > = back_len ) & & strncmp ( s + len - back_len , back , back_len ) = = 0 ) {
s [ len - back_len ] = ' \0 ' ;
len - = back_len ;
ret = true ;
}
}
return ret ;
}
/**
Find the number of ' c ' chars in a string
* */
_PUBLIC_ _PURE_ size_t count_chars ( const char * s , char c )
{
size_t count = 0 ;
while ( * s ) {
if ( * s = = c ) count + + ;
s + + ;
}
return count ;
}
/**
2011-10-18 18:07:54 +02:00
* Routine to get hex characters and turn them into a byte array .
* the array can be variable length .
* - " 0xnn " or " 0Xnn " is specially catered for .
* - The first non - hex - digit character ( apart from possibly leading " 0x "
* finishes the conversion and skips the rest of the input .
2011-10-18 18:10:00 +02:00
* - A single hex - digit character at the end of the string is skipped .
2011-10-18 18:07:54 +02:00
*
* valid examples : " 0A5D15 " ; " 0x123456 "
*/
2008-10-19 12:38:16 +02:00
_PUBLIC_ size_t strhex_to_str ( char * p , size_t p_len , const char * strhex , size_t strhex_len )
{
2009-09-25 23:40:55 +03:00
size_t i = 0 ;
2008-10-19 12:38:16 +02:00
size_t num_chars = 0 ;
uint8_t lonybble , hinybble ;
const char * hexchars = " 0123456789ABCDEF " ;
char * p1 = NULL , * p2 = NULL ;
2009-09-25 23:40:55 +03:00
/* skip leading 0x prefix */
if ( strncasecmp ( strhex , " 0x " , 2 ) = = 0 ) {
i + = 2 ; /* skip two chars */
}
2008-10-19 12:38:16 +02:00
2011-10-18 18:10:00 +02:00
for ( ; i + 1 < strhex_len & & strhex [ i ] ! = 0 & & strhex [ i + 1 ] ! = 0 ; i + + ) {
2011-10-18 18:03:10 +02:00
p1 = strchr ( hexchars , toupper ( ( unsigned char ) strhex [ i ] ) ) ;
if ( p1 = = NULL ) {
2008-10-19 12:38:16 +02:00
break ;
2011-10-18 18:03:10 +02:00
}
2008-10-19 12:38:16 +02:00
i + + ; /* next hex digit */
2011-10-18 18:03:10 +02:00
p2 = strchr ( hexchars , toupper ( ( unsigned char ) strhex [ i ] ) ) ;
if ( p2 = = NULL ) {
2008-10-19 12:38:16 +02:00
break ;
2011-10-18 18:03:10 +02:00
}
2008-10-19 12:38:16 +02:00
/* get the two nybbles */
hinybble = PTR_DIFF ( p1 , hexchars ) ;
lonybble = PTR_DIFF ( p2 , hexchars ) ;
if ( num_chars > = p_len ) {
break ;
}
p [ num_chars ] = ( hinybble < < 4 ) | lonybble ;
num_chars + + ;
p1 = NULL ;
p2 = NULL ;
}
return num_chars ;
}
2016-08-28 03:01:18 +12:00
/**
* Parse a hex string and return a data blob .
2008-10-19 12:38:16 +02:00
*/
_PUBLIC_ _PURE_ DATA_BLOB strhex_to_data_blob ( TALLOC_CTX * mem_ctx , const char * strhex )
{
DATA_BLOB ret_blob = data_blob_talloc ( mem_ctx , NULL , strlen ( strhex ) / 2 + 1 ) ;
ret_blob . length = strhex_to_str ( ( char * ) ret_blob . data , ret_blob . length ,
strhex ,
strlen ( strhex ) ) ;
return ret_blob ;
}
2016-08-28 03:01:18 +12:00
/**
* Parse a hex dump and return a data blob . Hex dump is structured as
* is generated from dump_data_cb ( ) elsewhere in this file
*
*/
_PUBLIC_ _PURE_ DATA_BLOB hexdump_to_data_blob ( TALLOC_CTX * mem_ctx , const char * hexdump , size_t hexdump_len )
{
2017-03-16 13:29:18 +11:00
DATA_BLOB ret_blob = { 0 } ;
2016-08-28 03:01:18 +12:00
size_t i = 0 ;
size_t char_count = 0 ;
/* hexdump line length is 77 chars long. We then use the ASCII representation of the bytes
* at the end of the final line to calculate how many are in that line , minus the extra space
* and newline . */
size_t hexdump_byte_count = ( 16 * ( hexdump_len / 77 ) ) ;
if ( hexdump_len % 77 ) {
hexdump_byte_count + = ( ( hexdump_len % 77 ) - 59 - 2 ) ;
}
ret_blob = data_blob_talloc ( mem_ctx , NULL , hexdump_byte_count + 1 ) ;
for ( ; i + 1 < hexdump_len & & hexdump [ i ] ! = 0 & & hexdump [ i + 1 ] ! = 0 ; i + + ) {
if ( ( i % 77 ) = = 0 )
i + = 7 ; /* Skip the offset at the start of the line */
if ( ( i % 77 ) < 56 ) { /* position 56 is after both hex chunks */
if ( hexdump [ i ] ! = ' ' ) {
char_count + = strhex_to_str ( ( char * ) & ret_blob . data [ char_count ] ,
hexdump_byte_count - char_count ,
& hexdump [ i ] , 2 ) ;
i + = 2 ;
} else {
i + + ;
}
} else {
i + + ;
}
}
ret_blob . length = char_count ;
return ret_blob ;
}
2012-02-03 03:27:19 +01:00
/**
* Print a buf in hex . Assumes dst is at least ( srclen * 2 ) + 1 large .
*/
_PUBLIC_ void hex_encode_buf ( char * dst , const uint8_t * src , size_t srclen )
{
size_t i ;
for ( i = 0 ; i < srclen ; i + + ) {
snprintf ( dst + i * 2 , 3 , " %02X " , src [ i ] ) ;
}
/*
* Ensure 0 - termination for 0 - length buffers
*/
dst [ srclen * 2 ] = ' \0 ' ;
}
2008-10-19 12:38:16 +02:00
/**
2015-05-08 10:24:48 +00:00
* talloc version of hex_encode_buf ( )
2008-10-19 12:38:16 +02:00
*/
_PUBLIC_ char * hex_encode_talloc ( TALLOC_CTX * mem_ctx , const unsigned char * buff_in , size_t len )
{
char * hex_buffer ;
hex_buffer = talloc_array ( mem_ctx , char , ( len * 2 ) + 1 ) ;
2008-11-13 21:26:03 +01:00
if ( ! hex_buffer ) {
return NULL ;
}
2012-02-03 03:27:19 +01:00
hex_encode_buf ( hex_buffer , buff_in , len ) ;
2008-11-13 21:26:03 +01:00
talloc_set_name_const ( hex_buffer , hex_buffer ) ;
2008-10-19 12:38:16 +02:00
return hex_buffer ;
}
/**
varient of strcmp ( ) that handles NULL ptrs
* */
_PUBLIC_ int strcmp_safe ( const char * s1 , const char * s2 )
{
if ( s1 = = s2 ) {
return 0 ;
}
if ( s1 = = NULL | | s2 = = NULL ) {
return s1 ? - 1 : 1 ;
}
return strcmp ( s1 , s2 ) ;
}
/**
return the number of bytes occupied by a buffer in ASCII format
the result includes the null termination
limited by ' n ' bytes
* */
_PUBLIC_ size_t ascii_len_n ( const char * src , size_t n )
{
size_t len ;
len = strnlen ( src , n ) ;
if ( len + 1 < = n ) {
len + = 1 ;
}
return len ;
}
2010-10-26 22:45:19 +02:00
struct anonymous_shared_header {
union {
size_t length ;
uint8_t pad [ 16 ] ;
} u ;
} ;
2010-03-22 08:27:58 +01:00
/* Map a shared memory buffer of at least nelem counters. */
2010-10-26 22:45:19 +02:00
void * anonymous_shared_allocate ( size_t orig_bufsz )
2010-03-22 08:27:58 +01:00
{
2010-10-26 22:45:19 +02:00
void * ptr ;
2010-03-22 08:27:58 +01:00
void * buf ;
size_t pagesz = getpagesize ( ) ;
2011-01-19 17:55:13 +01:00
size_t pagecnt ;
2010-10-26 22:45:19 +02:00
size_t bufsz = orig_bufsz ;
struct anonymous_shared_header * hdr ;
bufsz + = sizeof ( * hdr ) ;
2010-03-22 08:27:58 +01:00
2011-01-19 17:55:13 +01:00
/* round up to full pages */
pagecnt = bufsz / pagesz ;
2010-03-22 08:27:58 +01:00
if ( bufsz % pagesz ) {
2011-01-19 17:55:13 +01:00
pagecnt + = 1 ;
2010-03-22 08:27:58 +01:00
}
2011-01-19 17:55:13 +01:00
bufsz = pagesz * pagecnt ;
2010-03-22 08:27:58 +01:00
2010-10-26 22:45:19 +02:00
if ( orig_bufsz > = bufsz ) {
/* integer wrap */
errno = ENOMEM ;
return NULL ;
}
2010-03-22 08:27:58 +01:00
# ifdef MAP_ANON
/* BSD */
buf = mmap ( NULL , bufsz , PROT_READ | PROT_WRITE , MAP_ANON | MAP_SHARED ,
- 1 /* fd */ , 0 /* offset */ ) ;
# else
2012-06-26 13:48:36 +02:00
{
int saved_errno ;
int fd ;
fd = open ( " /dev/zero " , O_RDWR ) ;
if ( fd = = - 1 ) {
return NULL ;
}
2010-03-22 08:27:58 +01:00
buf = mmap ( NULL , bufsz , PROT_READ | PROT_WRITE , MAP_FILE | MAP_SHARED ,
2012-06-26 13:48:36 +02:00
fd , 0 /* offset */ ) ;
saved_errno = errno ;
close ( fd ) ;
errno = saved_errno ;
}
2010-03-22 08:27:58 +01:00
# endif
if ( buf = = MAP_FAILED ) {
return NULL ;
}
2010-10-26 22:45:19 +02:00
hdr = ( struct anonymous_shared_header * ) buf ;
hdr - > u . length = bufsz ;
ptr = ( void * ) ( & hdr [ 1 ] ) ;
return ptr ;
}
2011-08-14 18:10:53 -04:00
void * anonymous_shared_resize ( void * ptr , size_t new_size , bool maymove )
{
# ifdef HAVE_MREMAP
void * buf ;
size_t pagesz = getpagesize ( ) ;
size_t pagecnt ;
size_t bufsz ;
struct anonymous_shared_header * hdr ;
int flags = 0 ;
if ( ptr = = NULL ) {
errno = EINVAL ;
return NULL ;
}
hdr = ( struct anonymous_shared_header * ) ptr ;
hdr - - ;
if ( hdr - > u . length > ( new_size + sizeof ( * hdr ) ) ) {
errno = EINVAL ;
return NULL ;
}
bufsz = new_size + sizeof ( * hdr ) ;
/* round up to full pages */
pagecnt = bufsz / pagesz ;
if ( bufsz % pagesz ) {
pagecnt + = 1 ;
}
bufsz = pagesz * pagecnt ;
if ( new_size > = bufsz ) {
/* integer wrap */
errno = ENOSPC ;
return NULL ;
}
if ( bufsz < = hdr - > u . length ) {
return ptr ;
}
if ( maymove ) {
flags = MREMAP_MAYMOVE ;
}
buf = mremap ( hdr , hdr - > u . length , bufsz , flags ) ;
if ( buf = = MAP_FAILED ) {
errno = ENOSPC ;
return NULL ;
}
hdr = ( struct anonymous_shared_header * ) buf ;
hdr - > u . length = bufsz ;
ptr = ( void * ) ( & hdr [ 1 ] ) ;
return ptr ;
# else
errno = ENOSPC ;
return NULL ;
# endif
}
2010-10-26 22:45:19 +02:00
void anonymous_shared_free ( void * ptr )
{
struct anonymous_shared_header * hdr ;
if ( ptr = = NULL ) {
return ;
}
hdr = ( struct anonymous_shared_header * ) ptr ;
hdr - - ;
2010-03-22 08:27:58 +01:00
2010-10-26 22:45:19 +02:00
munmap ( hdr , hdr - > u . length ) ;
2010-03-22 08:27:58 +01:00
}
2008-10-19 12:38:16 +02:00
2010-07-20 15:31:49 +10:00
# ifdef DEVELOPER
/* used when you want a debugger started at a particular point in the
code . Mostly useful in code that runs as a child process , where
normal gdb attach is harder to organise .
*/
void samba_start_debugger ( void )
{
char * cmd = NULL ;
2010-09-09 11:10:01 +02:00
if ( asprintf ( & cmd , " xterm -e \" gdb --pid %u \" & " , getpid ( ) ) = = - 1 ) {
return ;
}
if ( system ( cmd ) = = - 1 ) {
free ( cmd ) ;
return ;
}
2010-07-20 15:31:49 +10:00
free ( cmd ) ;
sleep ( 2 ) ;
}
# endif