2006-08-21 16:07:03 +04:00
/*
2012-02-23 22:05:12 +04:00
* Copyright ( C ) 2006 - 2012 Red Hat , Inc . All rights reserved .
2006-08-21 16:07:03 +04:00
*
* This file is part of the device - mapper userspace tools .
*
* This copyrighted material is made available to anyone wishing to use ,
* modify , copy , or redistribute it subject to the terms and conditions
2007-08-21 20:26:07 +04:00
* of the GNU Lesser General Public License v .2 .1 .
2006-08-21 16:07:03 +04:00
*
2007-08-21 20:26:07 +04:00
* You should have received a copy of the GNU Lesser General Public License
2006-08-21 16:07:03 +04:00
* along with this program ; if not , write to the Free Software Foundation ,
* Inc . , 59 Temple Place , Suite 330 , Boston , MA 02111 - 1307 USA
*/
2008-11-03 21:59:59 +03:00
# include "dmlib.h"
2006-08-21 16:07:03 +04:00
# include <ctype.h>
2014-04-15 15:27:47 +04:00
# include <stdarg.h>
2014-04-28 12:25:43 +04:00
# include <math.h> /* fabs() */
# include <float.h> /* DBL_EPSILON */
2006-08-21 16:07:03 +04:00
/*
* consume characters while they match the predicate function .
*/
static char * _consume ( char * buffer , int ( * fn ) ( int ) )
{
while ( * buffer & & fn ( * buffer ) )
buffer + + ;
return buffer ;
}
static int _isword ( int c )
{
return ! isspace ( c ) ;
}
/*
* Split buffer into NULL - separated words in argv .
* Returns number of words .
*/
2007-01-22 18:03:57 +03:00
int dm_split_words ( char * buffer , unsigned max ,
2010-07-09 19:34:40 +04:00
unsigned ignore_comments __attribute__ ( ( unused ) ) ,
2006-08-21 16:07:03 +04:00
char * * argv )
{
unsigned arg ;
for ( arg = 0 ; arg < max ; arg + + ) {
buffer = _consume ( buffer , isspace ) ;
if ( ! * buffer )
break ;
argv [ arg ] = buffer ;
buffer = _consume ( buffer , _isword ) ;
if ( * buffer ) {
* buffer = ' \0 ' ;
buffer + + ;
}
}
return arg ;
}
/*
* Remove hyphen quoting from a component of a name .
* NULL - terminates the component and returns start of next component .
*/
static char * _unquote ( char * component )
{
char * c = component ;
char * o = c ;
char * r ;
while ( * c ) {
if ( * ( c + 1 ) ) {
if ( * c = = ' - ' ) {
if ( * ( c + 1 ) = = ' - ' )
c + + ;
else
break ;
}
}
* o = * c ;
o + + ;
c + + ;
}
r = ( * c ) ? c + 1 : c ;
* o = ' \0 ' ;
return r ;
}
int dm_split_lvm_name ( struct dm_pool * mem , const char * dmname ,
char * * vgname , char * * lvname , char * * layer )
{
2013-04-21 14:48:24 +04:00
if ( mem )
* vgname = dm_pool_strdup ( mem , dmname ) ;
if ( ! * vgname )
2006-08-21 16:07:03 +04:00
return 0 ;
_unquote ( * layer = _unquote ( * lvname = _unquote ( * vgname ) ) ) ;
return 1 ;
}
2006-08-21 16:52:39 +04:00
/*
* On error , up to glibc 2.0 .6 , snprintf returned - 1 if buffer was too small ;
* From glibc 2.1 it returns number of chars ( excl . trailing null ) that would
* have been written had there been room .
*
* dm_snprintf reverts to the old behaviour .
*/
int dm_snprintf ( char * buf , size_t bufsize , const char * format , . . . )
{
int n ;
va_list ap ;
va_start ( ap , format ) ;
n = vsnprintf ( buf , bufsize , format , ap ) ;
va_end ( ap ) ;
2007-04-27 18:52:41 +04:00
if ( n < 0 | | ( ( unsigned ) n + 1 > bufsize ) )
2006-08-21 16:52:39 +04:00
return - 1 ;
return n ;
}
2007-01-08 18:18:52 +03:00
2010-10-25 17:13:53 +04:00
const char * dm_basename ( const char * path )
2007-01-08 18:18:52 +03:00
{
2010-10-25 17:13:53 +04:00
const char * p = strrchr ( path , ' / ' ) ;
2007-01-08 18:18:52 +03:00
2010-10-25 17:13:53 +04:00
return p ? p + 1 : path ;
2007-01-08 18:18:52 +03:00
}
2012-07-30 18:41:15 +04:00
int dm_vasprintf ( char * * result , const char * format , va_list aq )
2007-01-12 00:54:53 +03:00
{
2012-02-23 22:05:12 +04:00
int i , n , size = 16 ;
2007-01-12 00:54:53 +03:00
va_list ap ;
char * buf = dm_malloc ( size ) ;
* result = 0 ;
if ( ! buf )
return - 1 ;
2012-02-23 22:05:12 +04:00
for ( i = 0 ; ; i + + ) {
2012-07-30 18:41:15 +04:00
va_copy ( ap , aq ) ;
2007-01-12 00:54:53 +03:00
n = vsnprintf ( buf , size , format , ap ) ;
2010-11-23 18:00:52 +03:00
va_end ( ap ) ;
2007-01-12 00:54:53 +03:00
if ( 0 < = n & & n < size )
2012-02-23 22:05:12 +04:00
break ;
dm_free ( buf ) ;
/* Up to glibc 2.0.6 returns -1 */
size = ( n < 0 ) ? size * 2 : n + 1 ;
if ( ! ( buf = dm_malloc ( size ) ) )
return - 1 ;
2007-01-12 00:54:53 +03:00
}
2012-02-23 22:05:12 +04:00
if ( i > 1 ) {
/* Reallocating more then once? */
if ( ! ( * result = dm_strdup ( buf ) ) ) {
dm_free ( buf ) ;
return - 1 ;
}
2012-02-23 22:19:32 +04:00
dm_free ( buf ) ;
2012-02-23 22:05:12 +04:00
} else
* result = buf ;
2012-02-10 17:56:19 +04:00
2007-01-12 00:54:53 +03:00
return n + 1 ;
}
2011-08-30 18:55:15 +04:00
2012-07-30 18:41:15 +04:00
int dm_asprintf ( char * * result , const char * format , . . . )
{
int r ;
va_list ap ;
va_start ( ap , format ) ;
r = dm_vasprintf ( result , format , ap ) ;
va_end ( ap ) ;
return r ;
}
2011-08-30 18:55:15 +04:00
/*
* Count occurences of ' c ' in ' str ' until we reach a null char .
*
* Returns :
* len - incremented for each char we encounter .
* count - number of occurrences of ' c ' and ' c2 ' .
*/
static void _count_chars ( const char * str , size_t * len , int * count ,
const int c1 , const int c2 )
{
const char * ptr ;
for ( ptr = str ; * ptr ; ptr + + , ( * len ) + + )
if ( * ptr = = c1 | | * ptr = = c2 )
( * count ) + + ;
}
2011-09-01 21:58:27 +04:00
/*
* Count occurrences of ' c ' in ' str ' of length ' size ' .
*
* Returns :
* Number of occurrences of ' c '
*/
2011-08-30 18:55:15 +04:00
unsigned dm_count_chars ( const char * str , size_t len , const int c )
{
size_t i ;
unsigned count = 0 ;
for ( i = 0 ; i < len ; i + + )
if ( str [ i ] = = c )
count + + ;
return count ;
}
2011-09-01 21:58:27 +04:00
/*
* Length of string after escaping double quotes and backslashes .
*/
2011-08-30 18:55:15 +04:00
size_t dm_escaped_len ( const char * str )
{
size_t len = 1 ;
int count = 0 ;
_count_chars ( str , & len , & count , ' \" ' , ' \\ ' ) ;
return count + len ;
}
/*
* Copies a string , quoting orig_char with quote_char .
* Optionally also quote quote_char .
*/
static void _quote_characters ( char * * out , const char * src ,
const int orig_char , const int quote_char ,
int quote_quote_char )
{
while ( * src ) {
if ( * src = = orig_char | |
( * src = = quote_char & & quote_quote_char ) )
* ( * out ) + + = quote_char ;
* ( * out ) + + = * src + + ;
}
}
static void _unquote_one_character ( char * src , const char orig_char ,
const char quote_char )
{
char * out ;
char s , n ;
/* Optimise for the common case where no changes are needed. */
while ( ( s = * src + + ) ) {
if ( s = = quote_char & &
( ( n = * src ) = = orig_char | | n = = quote_char ) ) {
out = src + + ;
* ( out - 1 ) = n ;
while ( ( s = * src + + ) ) {
if ( s = = quote_char & &
( ( n = * src ) = = orig_char | | n = = quote_char ) ) {
s = n ;
src + + ;
}
* out = s ;
out + + ;
}
* out = ' \0 ' ;
return ;
}
}
}
/*
* Unquote each character given in orig_char array and unquote quote_char
* as well . Also save the first occurrence of each character from orig_char
* that was found unquoted in arr_substr_first_unquoted array . This way we can
* process several characters in one go .
*/
static void _unquote_characters ( char * src , const char * orig_chars ,
size_t num_orig_chars ,
const char quote_char ,
char * arr_substr_first_unquoted [ ] )
{
char * out = src ;
char c , s , n ;
unsigned i ;
while ( ( s = * src + + ) ) {
for ( i = 0 ; i < num_orig_chars ; i + + ) {
c = orig_chars [ i ] ;
if ( s = = quote_char & &
( ( n = * src ) = = c | | n = = quote_char ) ) {
s = n ;
src + + ;
break ;
}
if ( arr_substr_first_unquoted & & ( s = = c ) & &
! arr_substr_first_unquoted [ i ] )
arr_substr_first_unquoted [ i ] = out ;
} ;
* out + + = s ;
}
* out = ' \0 ' ;
}
/*
* Copies a string , quoting hyphens with hyphens .
*/
static void _quote_hyphens ( char * * out , const char * src )
{
_quote_characters ( out , src , ' - ' , ' - ' , 0 ) ;
}
2011-09-01 21:58:27 +04:00
/*
* < vg > - < lv > - < layer > or if ! layer just < vg > - < lv > .
*/
2011-08-30 18:55:15 +04:00
char * dm_build_dm_name ( struct dm_pool * mem , const char * vgname ,
const char * lvname , const char * layer )
{
size_t len = 1 ;
int hyphens = 1 ;
char * r , * out ;
_count_chars ( vgname , & len , & hyphens , ' - ' , 0 ) ;
_count_chars ( lvname , & len , & hyphens , ' - ' , 0 ) ;
if ( layer & & * layer ) {
_count_chars ( layer , & len , & hyphens , ' - ' , 0 ) ;
hyphens + + ;
}
len + = hyphens ;
if ( ! ( r = dm_pool_alloc ( mem , len ) ) ) {
log_error ( " build_dm_name: Allocation failed for % " PRIsize_t
" for %s %s %s. " , len , vgname , lvname , layer ) ;
return NULL ;
}
out = r ;
_quote_hyphens ( & out , vgname ) ;
* out + + = ' - ' ;
_quote_hyphens ( & out , lvname ) ;
if ( layer & & * layer ) {
/* No hyphen if the layer begins with _ e.g. _mlog */
if ( * layer ! = ' _ ' )
* out + + = ' - ' ;
_quote_hyphens ( & out , layer ) ;
}
* out = ' \0 ' ;
return r ;
}
2011-09-01 21:58:27 +04:00
char * dm_build_dm_uuid ( struct dm_pool * mem , const char * uuid_prefix , const char * lvid , const char * layer )
2011-08-30 18:55:15 +04:00
{
char * dmuuid ;
size_t len ;
if ( ! layer )
layer = " " ;
2011-09-14 20:07:07 +04:00
len = strlen ( uuid_prefix ) + strlen ( lvid ) + strlen ( layer ) + 2 ;
2011-08-30 18:55:15 +04:00
if ( ! ( dmuuid = dm_pool_alloc ( mem , len ) ) ) {
log_error ( " build_dm_name: Allocation failed for % " PRIsize_t
" %s %s. " , len , lvid , layer ) ;
return NULL ;
}
2011-09-01 21:58:27 +04:00
sprintf ( dmuuid , " %s%s%s%s " , uuid_prefix , lvid , ( * layer ) ? " - " : " " , layer ) ;
2011-08-30 18:55:15 +04:00
return dmuuid ;
}
/*
* Copies a string , quoting double quotes with backslashes .
*/
char * dm_escape_double_quotes ( char * out , const char * src )
{
char * buf = out ;
_quote_characters ( & buf , src , ' \" ' , ' \\ ' , 1 ) ;
* buf = ' \0 ' ;
return out ;
}
2011-09-01 21:58:27 +04:00
/*
* Undo quoting in situ .
*/
2011-08-30 18:55:15 +04:00
void dm_unescape_double_quotes ( char * src )
{
_unquote_one_character ( src , ' \" ' , ' \\ ' ) ;
}
2011-09-01 21:58:27 +04:00
/*
* Unescape colons and " at " signs in situ and save the substrings
* starting at the position of the first unescaped colon and the
* first unescaped " at " sign . This is normally used to unescape
* device names used as PVs .
*/
2011-08-30 18:55:15 +04:00
void dm_unescape_colons_and_at_signs ( char * src ,
char * * substr_first_unquoted_colon ,
char * * substr_first_unquoted_at_sign )
{
const char * orig_chars = " :@ " ;
char * arr_substr_first_unquoted [ ] = { NULL , NULL , NULL } ;
_unquote_characters ( src , orig_chars , 2 , ' \\ ' , arr_substr_first_unquoted ) ;
if ( substr_first_unquoted_colon )
* substr_first_unquoted_colon = arr_substr_first_unquoted [ 0 ] ;
if ( substr_first_unquoted_at_sign )
* substr_first_unquoted_at_sign = arr_substr_first_unquoted [ 1 ] ;
}
2012-02-24 02:45:43 +04:00
int dm_strncpy ( char * dest , const char * src , size_t n )
{
if ( memccpy ( dest , src , 0 , n ) )
return 1 ;
if ( n > 0 )
dest [ n - 1 ] = ' \0 ' ;
return 0 ;
}
2014-04-28 12:25:43 +04:00
/* Test if the doubles are close enough to be considered equal */
static int _close_enough ( double d1 , double d2 )
{
return fabs ( d1 - d2 ) < DBL_EPSILON ;
}
uint64_t dm_units_to_factor ( const char * units , char * unit_type ,
int strict , char * * endptr )
{
char * ptr = NULL ;
uint64_t v ;
double custom_value = 0 ;
uint64_t multiplier ;
if ( endptr )
* endptr = ( char * ) units ;
if ( isdigit ( * units ) ) {
custom_value = strtod ( units , & ptr ) ;
if ( ptr = = units )
return 0 ;
v = ( uint64_t ) strtoull ( units , NULL , 10 ) ;
if ( _close_enough ( ( double ) v , custom_value ) )
custom_value = 0 ; /* Use integer arithmetic */
units = ptr ;
} else
v = 1 ;
/* Only one units char permitted in strict mode. */
if ( strict & & units [ 0 ] & & units [ 1 ] )
return 0 ;
if ( v = = 1 )
* unit_type = * units ;
else
* unit_type = ' U ' ;
switch ( * units ) {
case ' h ' :
case ' H ' :
multiplier = v = UINT64_C ( 1 ) ;
* unit_type = * units ;
break ;
case ' b ' :
case ' B ' :
multiplier = UINT64_C ( 1 ) ;
break ;
# define KILO UINT64_C(1024)
case ' s ' :
case ' S ' :
multiplier = ( KILO / 2 ) ;
break ;
case ' k ' :
multiplier = KILO ;
break ;
case ' m ' :
multiplier = KILO * KILO ;
break ;
case ' g ' :
multiplier = KILO * KILO * KILO ;
break ;
case ' t ' :
multiplier = KILO * KILO * KILO * KILO ;
break ;
case ' p ' :
multiplier = KILO * KILO * KILO * KILO * KILO ;
break ;
case ' e ' :
multiplier = KILO * KILO * KILO * KILO * KILO * KILO ;
break ;
# undef KILO
# define KILO UINT64_C(1000)
case ' K ' :
multiplier = KILO ;
break ;
case ' M ' :
multiplier = KILO * KILO ;
break ;
case ' G ' :
multiplier = KILO * KILO * KILO ;
break ;
case ' T ' :
multiplier = KILO * KILO * KILO * KILO ;
break ;
case ' P ' :
multiplier = KILO * KILO * KILO * KILO * KILO ;
break ;
case ' E ' :
multiplier = KILO * KILO * KILO * KILO * KILO * KILO ;
break ;
# undef KILO
default :
return 0 ;
}
if ( endptr )
* endptr = ( char * ) units + 1 ;
if ( _close_enough ( custom_value , 0. ) )
return v * multiplier ; /* Use integer arithmetic */
else
return ( uint64_t ) ( custom_value * multiplier ) ;
}