2003-08-13 01:53:07 +00:00
/*
* Unix SMB / CIFS implementation .
* SMB parameters and setup
* Copyright ( C ) Andrew Tridgell 1992 - 1998 Modified by Jeremy Allison 1995.
2005-09-28 21:54:29 +00:00
*
* Added afdgets ( ) Jelmer Vernooij 2005
2014-12-07 14:13:51 +01:00
*
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 the Free
2007-07-10 03:04:16 +00:00
* Software Foundation ; either version 3 of the License , or ( at your option )
2003-08-13 01:53:07 +00:00
* any later version .
2014-12-07 14:13:51 +01:00
*
2003-08-13 01:53:07 +00:00
* 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 .
2014-12-07 14:13:51 +01:00
*
2003-08-13 01:53:07 +00:00
* You should have received a copy of the GNU General Public License along with
2007-07-10 05:23:25 +00:00
* this program ; if not , see < http : //www.gnu.org/licenses/>.
2003-08-13 01:53:07 +00:00
*/
2011-10-21 13:42:34 +02:00
# include "replace.h"
2004-11-02 00:24:21 +00:00
# include "system/shmem.h"
2005-02-10 05:09:35 +00:00
# include "system/filesys.h"
2011-10-21 13:42:34 +02:00
# include <talloc.h>
# include "lib/util/samba_util.h"
# include "lib/util/debug.h"
2003-08-13 01:53:07 +00:00
2006-02-28 13:12:39 +00:00
/**
* @ file
* @ brief File - related utility functions
*/
/**
2014-12-07 14:13:51 +01:00
read a line from a file with possible \ continuation chars .
2003-08-13 01:53:07 +00:00
Blanks at the start or end of a line are stripped .
The string will be allocated if s2 is NULL
2006-02-28 13:12:39 +00:00
* */
2006-03-05 17:15:19 +00:00
_PUBLIC_ char * fgets_slash ( char * s2 , int maxlen , XFILE * f )
2003-08-13 01:53:07 +00:00
{
char * s = s2 ;
int len = 0 ;
int c ;
2007-08-27 17:21:16 +00:00
bool start_of_line = true ;
2003-08-13 01:53:07 +00:00
if ( x_feof ( f ) )
return ( NULL ) ;
if ( maxlen < 2 ) return ( NULL ) ;
if ( ! s2 )
{
maxlen = MIN ( maxlen , 8 ) ;
s = ( char * ) malloc ( maxlen ) ;
}
if ( ! s ) return ( NULL ) ;
* s = 0 ;
while ( len < maxlen - 1 )
{
c = x_getc ( f ) ;
switch ( c )
{
case ' \r ' :
break ;
case ' \n ' :
while ( len > 0 & & s [ len - 1 ] = = ' ' )
{
s [ - - len ] = 0 ;
}
if ( len > 0 & & s [ len - 1 ] = = ' \\ ' )
{
s [ - - len ] = 0 ;
2007-08-27 17:21:16 +00:00
start_of_line = true ;
2003-08-13 01:53:07 +00:00
break ;
}
return ( s ) ;
case EOF :
2014-12-07 14:13:51 +01:00
if ( len < = 0 & & ! s2 )
2003-08-13 01:53:07 +00:00
SAFE_FREE ( s ) ;
return ( len > 0 ? s : NULL ) ;
case ' ' :
if ( start_of_line )
break ;
2006-04-08 02:40:15 +00:00
/* fall through */
2003-08-13 01:53:07 +00:00
default :
2007-08-27 17:21:16 +00:00
start_of_line = false ;
2003-08-13 01:53:07 +00:00
s [ len + + ] = c ;
s [ len ] = 0 ;
}
if ( ! s2 & & len > maxlen - 3 )
{
char * t ;
2014-12-07 14:13:51 +01:00
2003-08-13 01:53:07 +00:00
maxlen * = 2 ;
2004-12-03 06:42:06 +00:00
t = realloc_p ( s , char , maxlen ) ;
2003-08-13 01:53:07 +00:00
if ( ! t ) {
DEBUG ( 0 , ( " fgets_slash: failed to expand buffer! \n " ) ) ;
SAFE_FREE ( s ) ;
return ( NULL ) ;
} else s = t ;
}
}
return ( s ) ;
}
2006-02-28 13:12:39 +00:00
/**
2014-12-07 14:13:51 +01:00
* Read one line ( data until next newline or eof ) and allocate it
2006-02-28 13:12:39 +00:00
*/
2006-03-05 17:15:19 +00:00
_PUBLIC_ char * afdgets ( int fd , TALLOC_CTX * mem_ctx , size_t hint )
2005-09-28 21:54:29 +00:00
{
char * data = NULL ;
ssize_t alloc_size = 0 , offset = 0 , ret ;
int p ;
if ( hint < = 0 ) hint = 0x100 ;
do {
alloc_size + = hint ;
data = talloc_realloc ( mem_ctx , data , char , alloc_size ) ;
if ( ! data )
return NULL ;
ret = read ( fd , data + offset , hint ) ;
2006-08-20 21:57:08 +00:00
if ( ret = = 0 ) {
return NULL ;
}
2005-09-28 21:54:29 +00:00
if ( ret = = - 1 ) {
talloc_free ( data ) ;
return NULL ;
}
/* Find newline */
for ( p = 0 ; p < ret ; p + + ) {
if ( data [ offset + p ] = = ' \n ' )
break ;
}
if ( p < ret ) {
data [ offset + p ] = ' \0 ' ;
/* Go back to position of newline */
lseek ( fd , p - ret + 1 , SEEK_CUR ) ;
return data ;
}
offset + = ret ;
} while ( ret = = hint ) ;
data [ offset ] = ' \0 ' ;
return data ;
}
2003-08-13 01:53:07 +00:00
2006-02-28 13:12:39 +00:00
/**
2003-08-13 01:53:07 +00:00
load a file into memory from a fd .
2006-02-28 13:12:39 +00:00
* */
2008-10-12 17:34:43 +02:00
_PUBLIC_ char * fd_load ( int fd , size_t * psize , size_t maxsize , TALLOC_CTX * mem_ctx )
2003-08-13 01:53:07 +00:00
{
2004-11-01 20:21:54 +00:00
struct stat sbuf ;
2003-08-13 01:53:07 +00:00
char * p ;
2008-10-12 17:34:43 +02:00
size_t size ;
2003-08-13 01:53:07 +00:00
2004-11-01 20:21:54 +00:00
if ( fstat ( fd , & sbuf ) ! = 0 ) return NULL ;
2003-08-13 01:53:07 +00:00
2008-10-12 17:34:43 +02:00
size = sbuf . st_size ;
if ( maxsize ) {
size = MIN ( size , maxsize ) ;
}
p = ( char * ) talloc_size ( mem_ctx , size + 1 ) ;
2003-08-13 01:53:07 +00:00
if ( ! p ) return NULL ;
2008-10-12 17:34:43 +02:00
if ( read ( fd , p , size ) ! = size ) {
2005-07-10 01:10:09 +00:00
talloc_free ( p ) ;
2003-08-13 01:53:07 +00:00
return NULL ;
}
2008-10-12 17:34:43 +02:00
p [ size ] = 0 ;
2003-08-13 01:53:07 +00:00
2008-10-12 17:34:43 +02:00
if ( psize ) * psize = size ;
2003-08-13 01:53:07 +00:00
return p ;
}
2006-02-28 13:12:39 +00:00
/**
2003-08-13 01:53:07 +00:00
load a file into memory
2006-02-28 13:12:39 +00:00
* */
2008-10-12 17:34:43 +02:00
_PUBLIC_ char * file_load ( const char * fname , size_t * size , size_t maxsize , TALLOC_CTX * mem_ctx )
2003-08-13 01:53:07 +00:00
{
int fd ;
char * p ;
if ( ! fname | | ! * fname ) return NULL ;
2014-12-07 14:13:51 +01:00
2003-08-13 01:53:07 +00:00
fd = open ( fname , O_RDONLY ) ;
if ( fd = = - 1 ) return NULL ;
2008-10-12 17:34:43 +02:00
p = fd_load ( fd , size , maxsize , mem_ctx ) ;
2003-08-13 01:53:07 +00:00
close ( fd ) ;
return p ;
}
2006-02-28 13:12:39 +00:00
/**
2003-08-13 01:53:07 +00:00
parse a buffer into lines
2006-10-28 06:06:15 +00:00
' p ' will be freed on error , and otherwise will be made a child of the returned array
2006-02-28 13:12:39 +00:00
* */
2008-10-12 17:34:43 +02:00
char * * file_lines_parse ( char * p , size_t size , int * numlines , TALLOC_CTX * mem_ctx )
2003-08-13 01:53:07 +00:00
{
int i ;
char * s , * * ret ;
if ( ! p ) return NULL ;
for ( s = p , i = 0 ; s < p + size ; s + + ) {
if ( s [ 0 ] = = ' \n ' ) i + + ;
}
2014-12-26 19:41:23 +00:00
ret = talloc_zero_array ( mem_ctx , char * , i + 2 ) ;
2003-08-13 01:53:07 +00:00
if ( ! ret ) {
2005-08-30 17:21:41 +00:00
talloc_free ( p ) ;
2003-08-13 01:53:07 +00:00
return NULL ;
2014-12-07 14:13:51 +01:00
}
2006-10-28 06:06:15 +00:00
talloc_steal ( ret , p ) ;
2014-12-07 14:13:51 +01:00
2003-08-13 01:53:07 +00:00
ret [ 0 ] = p ;
for ( s = p , i = 0 ; s < p + size ; s + + ) {
if ( s [ 0 ] = = ' \n ' ) {
s [ 0 ] = 0 ;
i + + ;
ret [ i ] = s + 1 ;
}
if ( s [ 0 ] = = ' \r ' ) s [ 0 ] = 0 ;
}
2006-12-09 03:12:33 +00:00
/* remove any blank lines at the end */
while ( i > 0 & & ret [ i - 1 ] [ 0 ] = = 0 ) {
i - - ;
}
if ( numlines ) * numlines = i ;
2003-08-13 01:53:07 +00:00
return ret ;
}
2006-02-28 13:12:39 +00:00
/**
2003-08-13 01:53:07 +00:00
load a file into memory and return an array of pointers to lines in the file
2014-12-07 14:13:51 +01:00
must be freed with talloc_free ( ) .
2006-02-28 13:12:39 +00:00
* */
2008-10-12 17:34:43 +02:00
_PUBLIC_ char * * file_lines_load ( const char * fname , int * numlines , size_t maxsize , TALLOC_CTX * mem_ctx )
2003-08-13 01:53:07 +00:00
{
char * p ;
size_t size ;
2008-10-12 17:34:43 +02:00
p = file_load ( fname , & size , maxsize , mem_ctx ) ;
2003-08-13 01:53:07 +00:00
if ( ! p ) return NULL ;
2006-10-28 06:06:15 +00:00
return file_lines_parse ( p , size , numlines , mem_ctx ) ;
2003-08-13 01:53:07 +00:00
}
2006-02-28 13:12:39 +00:00
/**
2003-08-13 01:53:07 +00:00
load a fd into memory and return an array of pointers to lines in the file
2005-08-30 17:21:41 +00:00
must be freed with talloc_free ( ) . If convert is true calls unix_to_dos on
2003-08-13 01:53:07 +00:00
the list .
2006-02-28 13:12:39 +00:00
* */
2008-10-12 17:34:43 +02:00
_PUBLIC_ char * * fd_lines_load ( int fd , int * numlines , size_t maxsize , TALLOC_CTX * mem_ctx )
2003-08-13 01:53:07 +00:00
{
char * p ;
size_t size ;
2008-10-12 17:34:43 +02:00
p = fd_load ( fd , & size , maxsize , mem_ctx ) ;
2003-08-13 01:53:07 +00:00
if ( ! p ) return NULL ;
2006-10-28 06:06:15 +00:00
return file_lines_parse ( p , size , numlines , mem_ctx ) ;
2003-08-13 01:53:07 +00:00
}
2013-10-29 17:48:11 +01:00
_PUBLIC_ bool file_save_mode ( const char * fname , const void * packet ,
size_t length , mode_t mode )
2003-08-13 01:53:07 +00:00
{
int fd ;
2013-10-29 17:48:11 +01:00
fd = open ( fname , O_WRONLY | O_CREAT | O_TRUNC , mode ) ;
2003-08-13 01:53:07 +00:00
if ( fd = = - 1 ) {
2007-08-27 17:21:16 +00:00
return false ;
2003-08-13 01:53:07 +00:00
}
if ( write ( fd , packet , length ) ! = ( size_t ) length ) {
2009-07-18 16:28:53 +07:00
close ( fd ) ;
2007-08-27 17:21:16 +00:00
return false ;
2003-08-13 01:53:07 +00:00
}
close ( fd ) ;
2007-08-27 17:21:16 +00:00
return true ;
2003-08-13 01:53:07 +00:00
}
2005-04-14 08:24:36 +00:00
2013-10-29 17:48:11 +01:00
/**
save a lump of data into a file . Mostly used for debugging
*/
_PUBLIC_ bool file_save ( const char * fname , const void * packet , size_t length )
{
return file_save_mode ( fname , packet , length , 0644 ) ;
}
2008-04-02 04:53:27 +02:00
_PUBLIC_ int vfdprintf ( int fd , const char * format , va_list ap )
2005-09-15 19:52:13 +00:00
{
char * p ;
int len , ret ;
va_list ap2 ;
2006-04-08 16:05:21 +00:00
va_copy ( ap2 , ap ) ;
2005-09-15 19:52:13 +00:00
len = vasprintf ( & p , format , ap2 ) ;
2007-02-06 05:26:25 +00:00
va_end ( ap2 ) ;
2005-09-15 19:52:13 +00:00
if ( len < = 0 ) return len ;
ret = write ( fd , p , len ) ;
SAFE_FREE ( p ) ;
return ret ;
}
2008-04-02 04:53:27 +02:00
_PUBLIC_ int fdprintf ( int fd , const char * format , . . . )
2005-09-15 19:52:13 +00:00
{
va_list ap ;
int ret ;
va_start ( ap , format ) ;
ret = vfdprintf ( fd , format , ap ) ;
va_end ( ap ) ;
return ret ;
}
2006-10-28 06:06:15 +00:00
2010-02-11 20:19:40 +11:00
/*
compare two files , return true if the two files have the same content
*/
bool file_compare ( const char * path1 , const char * path2 )
{
size_t size1 , size2 ;
char * p1 , * p2 ;
TALLOC_CTX * mem_ctx = talloc_new ( NULL ) ;
p1 = file_load ( path1 , & size1 , 0 , mem_ctx ) ;
p2 = file_load ( path2 , & size2 , 0 , mem_ctx ) ;
if ( ! p1 | | ! p2 | | size1 ! = size2 ) {
talloc_free ( mem_ctx ) ;
return false ;
}
if ( memcmp ( p1 , p2 , size1 ) ! = 0 ) {
talloc_free ( mem_ctx ) ;
return false ;
}
talloc_free ( mem_ctx ) ;
return true ;
}