2002-01-10 12:22:17 +00:00
/*
2004-03-30 19:35:44 +00:00
* Copyright ( C ) 2001 - 2004 Sistina Software , Inc . All rights reserved .
* Copyright ( C ) 2004 Red Hat , Inc . All rights reserved .
2002-01-10 12:22:17 +00:00
*
2004-03-30 19:35:44 +00:00
* This file is part of LVM2 .
*
* This copyrighted material is made available to anyone wishing to use ,
* modify , copy , or redistribute it subject to the terms and conditions
* of the GNU General Public License v .2 .
*
* You should have received a copy of the GNU General Public License
* along with this program ; if not , write to the Free Software Foundation ,
* Inc . , 59 Temple Place , Suite 330 , Boston , MA 02111 - 1307 USA
2002-01-10 12:22:17 +00:00
*/
2002-11-18 14:01:16 +00:00
# include "lib.h"
2002-01-10 12:22:17 +00:00
# include "lvm-file.h"
# include "lvm-string.h"
# include <unistd.h>
# include <sys/stat.h>
# include <sys/file.h>
# include <fcntl.h>
2002-03-11 22:23:24 +00:00
# include <dirent.h>
2002-01-10 12:22:17 +00:00
/*
* Creates a temporary filename , and opens a descriptor to the
* file . Both the filename and descriptor are needed so we can
* rename the file after successfully writing it . Grab
* NFS - supported exclusive fcntl discretionary lock .
*/
2002-03-11 22:23:24 +00:00
int create_temp_name ( const char * dir , char * buffer , size_t len , int * fd )
2002-01-10 12:22:17 +00:00
{
int i , num ;
pid_t pid ;
char hostname [ 255 ] ;
struct flock lock = {
2006-05-09 21:23:51 +00:00
. l_type = F_WRLCK ,
. l_whence = 0 ,
. l_start = 0 ,
. l_len = 0
2002-01-10 12:22:17 +00:00
} ;
num = rand ( ) ;
pid = getpid ( ) ;
if ( gethostname ( hostname , sizeof ( hostname ) ) < 0 ) {
log_sys_error ( " gethostname " , " " ) ;
strcpy ( hostname , " nohostname " ) ;
}
for ( i = 0 ; i < 20 ; i + + , num + + ) {
2006-08-21 12:54:53 +00:00
if ( dm_snprintf ( buffer , len , " %s/.lvm_%s_%d_%d " ,
2002-01-10 12:22:17 +00:00
dir , hostname , pid , num ) = = - 1 ) {
log_err ( " Not enough space to build temporary file "
" string. " ) ;
return 0 ;
}
* fd = open ( buffer , O_CREAT | O_EXCL | O_WRONLY | O_APPEND ,
S_IRUSR | S_IRGRP | S_IROTH |
S_IWUSR | S_IWGRP | S_IWOTH ) ;
if ( * fd < 0 )
continue ;
if ( ! fcntl ( * fd , F_SETLK , & lock ) )
return 1 ;
close ( * fd ) ;
}
return 0 ;
}
/*
* NFS - safe rename of a temporary file to a common name , designed
* to avoid race conditions and not overwrite the destination if
* it exists .
*
* Try to create the new filename as a hard link to the original .
* Check the link count of the original file to see if it worked .
* ( Assumes nothing else touches our temporary file ! ) If it
* worked , unlink the old filename .
*/
int lvm_rename ( const char * old , const char * new )
{
struct stat buf ;
2005-03-10 22:31:10 +00:00
if ( link ( old , new ) ) {
log_error ( " %s: rename to %s failed: %s " , old , new ,
strerror ( errno ) ) ;
return 0 ;
}
2002-01-10 12:22:17 +00:00
if ( stat ( old , & buf ) ) {
log_sys_error ( " stat " , old ) ;
return 0 ;
}
if ( buf . st_nlink ! = 2 ) {
log_error ( " %s: rename to %s failed " , old , new ) ;
return 0 ;
}
if ( unlink ( old ) ) {
log_sys_error ( " unlink " , old ) ;
return 0 ;
}
return 1 ;
}
2002-02-11 15:42:34 +00:00
int path_exists ( const char * path )
{
struct stat info ;
if ( ! * path )
return 0 ;
if ( stat ( path , & info ) < 0 )
return 0 ;
return 1 ;
}
int dir_exists ( const char * path )
{
struct stat info ;
if ( ! * path )
return 0 ;
if ( stat ( path , & info ) < 0 )
return 0 ;
if ( ! S_ISDIR ( info . st_mode ) )
return 0 ;
return 1 ;
}
2002-05-19 03:46:34 +00:00
static int _create_dir_recursive ( const char * dir )
{
char * orig , * s ;
int rc ;
2002-05-31 19:28:37 +00:00
log_verbose ( " Creating directory \" %s \" " , dir ) ;
/* Create parent directories */
2005-10-16 23:03:59 +00:00
orig = s = dm_strdup ( dir ) ;
2002-05-19 03:46:34 +00:00
while ( ( s = strchr ( s , ' / ' ) ) ! = NULL ) {
* s = ' \0 ' ;
if ( * orig ) {
rc = mkdir ( orig , 0777 ) ;
if ( rc < 0 & & errno ! = EEXIST ) {
2005-05-17 13:44:02 +00:00
if ( errno ! = EROFS )
log_sys_error ( " mkdir " , orig ) ;
2005-10-16 23:03:59 +00:00
dm_free ( orig ) ;
2002-05-19 03:46:34 +00:00
return 0 ;
}
}
* s + + = ' / ' ;
}
2005-10-16 23:03:59 +00:00
dm_free ( orig ) ;
2002-05-19 03:46:34 +00:00
2002-05-31 19:28:37 +00:00
/* Create final directory */
2002-05-19 03:46:34 +00:00
rc = mkdir ( dir , 0777 ) ;
if ( rc < 0 & & errno ! = EEXIST ) {
2005-05-17 13:44:02 +00:00
if ( errno ! = EROFS )
log_sys_error ( " mkdir " , dir ) ;
2002-05-19 03:46:34 +00:00
return 0 ;
}
return 1 ;
}
2002-02-11 15:42:34 +00:00
int create_dir ( const char * dir )
{
struct stat info ;
if ( ! * dir )
return 1 ;
2002-05-19 03:46:34 +00:00
if ( stat ( dir , & info ) < 0 )
return _create_dir_recursive ( dir ) ;
2002-02-11 15:42:34 +00:00
if ( S_ISDIR ( info . st_mode ) )
return 1 ;
log_error ( " Directory \" %s \" not found " , dir ) ;
return 0 ;
}
2002-03-11 22:23:24 +00:00
int is_empty_dir ( const char * dir )
{
struct dirent * dirent ;
DIR * d ;
if ( ! ( d = opendir ( dir ) ) ) {
log_sys_error ( " opendir " , dir ) ;
return 0 ;
}
while ( ( dirent = readdir ( d ) ) )
if ( strcmp ( dirent - > d_name , " . " ) & & strcmp ( dirent - > d_name , " .. " ) )
break ;
if ( closedir ( d ) ) {
log_sys_error ( " closedir " , dir ) ;
}
return dirent ? 0 : 1 ;
}
2002-12-05 22:51:15 +00:00
void sync_dir ( const char * file )
{
int fd ;
char * dir , * c ;
2005-10-16 23:03:59 +00:00
if ( ! ( dir = dm_strdup ( file ) ) ) {
2002-12-05 22:51:15 +00:00
log_error ( " sync_dir failed in strdup " ) ;
return ;
}
if ( ! dir_exists ( dir ) ) {
c = dir + strlen ( dir ) ;
while ( * c ! = ' / ' & & c > dir )
c - - ;
2004-06-19 19:24:33 +00:00
if ( c = = dir )
* c + + = ' . ' ;
2002-12-05 22:51:15 +00:00
* c = ' \0 ' ;
}
if ( ( fd = open ( dir , O_RDONLY ) ) = = - 1 ) {
log_sys_error ( " open " , dir ) ;
goto out ;
}
2005-09-01 18:37:22 +00:00
if ( fsync ( fd ) & & ( errno ! = EROFS ) & & ( errno ! = EINVAL ) )
2002-12-05 22:51:15 +00:00
log_sys_error ( " fsync " , dir ) ;
close ( fd ) ;
out :
2005-10-16 23:03:59 +00:00
dm_free ( dir ) ;
2002-12-05 22:51:15 +00:00
}