2018-05-14 12:16:43 +01:00
/*
* Copyright ( C ) 2005 - 2015 Red Hat , Inc . All rights reserved .
*
* 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
* of the GNU Lesser General Public License v .2 .1 .
*
* You should have received a copy of the GNU Lesser General Public License
* along with this program ; if not , write to the Free Software Foundation ,
* Inc . , 51 Franklin Street , Fifth Floor , Boston , MA 02110 - 1301 USA
*/
# include "misc/dmlib.h"
# include "libdm-common.h"
int dm_get_status_snapshot ( struct dm_pool * mem , const char * params ,
struct dm_status_snapshot * * status )
{
struct dm_status_snapshot * s ;
int r ;
if ( ! params ) {
log_error ( " Failed to parse invalid snapshot params. " ) ;
return 0 ;
}
if ( ! ( s = dm_pool_zalloc ( mem , sizeof ( * s ) ) ) ) {
log_error ( " Failed to allocate snapshot status structure. " ) ;
return 0 ;
}
r = sscanf ( params , FMTu64 " / " FMTu64 " " FMTu64 ,
& s - > used_sectors , & s - > total_sectors ,
& s - > metadata_sectors ) ;
if ( r = = 3 | | r = = 2 )
s - > has_metadata_sectors = ( r = = 3 ) ;
else if ( ! strcmp ( params , " Invalid " ) )
s - > invalid = 1 ;
else if ( ! strcmp ( params , " Merge failed " ) )
s - > merge_failed = 1 ;
else if ( ! strcmp ( params , " Overflow " ) )
s - > overflow = 1 ;
else {
dm_pool_free ( mem , s ) ;
log_error ( " Failed to parse snapshot params: %s. " , params ) ;
return 0 ;
}
* status = s ;
return 1 ;
}
/*
* Skip nr fields each delimited by a single space .
* FIXME Don ' t assume single space .
*/
static const char * _skip_fields ( const char * p , unsigned nr )
{
while ( p & & nr - - & & ( p = strchr ( p , ' ' ) ) )
p + + ;
return p ;
}
/*
* Count number of single - space delimited fields .
* Number of fields is number of spaces plus one .
*/
static unsigned _count_fields ( const char * p )
{
unsigned nr = 1 ;
if ( ! p | | ! * p )
return 0 ;
while ( ( p = _skip_fields ( p , 1 ) ) )
nr + + ;
return nr ;
}
/*
* Various RAID status versions include :
* Versions < 1.5 .0 ( 4 fields ) :
* < raid_type > < # devs > < health_str > < sync_ratio >
* Versions 1.5 .0 + ( 6 fields ) :
* < raid_type > < # devs > < health_str > < sync_ratio > < sync_action > < mismatch_cnt >
* Versions 1.9 .0 + ( 7 fields ) :
* < raid_type > < # devs > < health_str > < sync_ratio > < sync_action > < mismatch_cnt > < data_offset >
*/
int dm_get_status_raid ( struct dm_pool * mem , const char * params ,
struct dm_status_raid * * status )
{
int i ;
unsigned num_fields ;
const char * p , * pp , * msg_fields = " " ;
struct dm_status_raid * s = NULL ;
unsigned a = 0 ;
if ( ( num_fields = _count_fields ( params ) ) < 4 )
goto_bad ;
/* Second field holds the device count */
msg_fields = " <#devs> " ;
2023-10-26 21:57:59 +02:00
if ( ! ( pp = _skip_fields ( params , 1 ) ) | | ( sscanf ( pp , " %d " , & i ) ! = 1 ) | | ! ( p = _skip_fields ( pp , 1 ) ) )
2018-05-14 12:16:43 +01:00
goto_bad ;
msg_fields = " " ;
if ( ! ( s = dm_pool_zalloc ( mem , sizeof ( struct dm_status_raid ) ) ) )
goto_bad ;
2023-10-26 21:57:59 +02:00
msg_fields = " <raid_type> <#devices> <health_chars> and <sync_ratio> " ;
if ( ! ( s - > raid_type = dm_pool_strndup ( mem , params , pp - params - 1 ) ) )
2018-05-14 12:16:43 +01:00
goto_bad ; /* memory is freed when pool is destroyed */
2024-03-24 22:42:16 +01:00
if ( ! ( pp = _skip_fields ( p , 1 ) ) )
2018-05-14 12:16:43 +01:00
goto_bad ;
2024-03-24 22:42:16 +01:00
/* Raid target can actually report more then real number of legs in a case
* raid legs have been removed during initial raid array resynchronization */
if ( i > ( pp - p - 1 ) )
i = pp - p - 1 ;
if ( ! ( s - > dev_health = dm_pool_strndup ( mem , p , i ) ) ) /* health chars */
goto_bad ;
p = pp ;
2023-10-26 21:57:59 +02:00
s - > dev_count = i ;
if ( sscanf ( p , FMTu64 " / " FMTu64 , & s - > insync_regions , & s - > total_regions ) ! = 2 )
2018-05-14 12:16:43 +01:00
goto_bad ;
/*
* All pre - 1.5 .0 version parameters are read . Now we check
* for additional 1.5 .0 + parameters ( i . e . num_fields at least 6 ) .
*
* Note that ' sync_action ' will be NULL ( and mismatch_count
* will be 0 ) if the kernel returns a pre - 1.5 .0 status .
*/
if ( num_fields < 6 )
goto out ;
msg_fields = " <sync_action> and <mismatch_cnt> " ;
/* Skip pre-1.5.0 params */
2023-10-26 21:57:59 +02:00
if ( ! ( pp = _skip_fields ( params , 4 ) ) | | ! ( p = _skip_fields ( pp , 1 ) ) )
2018-05-14 12:16:43 +01:00
goto_bad ;
2023-10-26 21:57:59 +02:00
if ( ! ( s - > sync_action = dm_pool_strndup ( mem , pp , p - pp - 1 ) ) )
2018-05-14 12:16:43 +01:00
goto_bad ;
2023-10-26 21:57:59 +02:00
if ( sscanf ( p , FMTu64 , & s - > mismatch_count ) ! = 1 )
2018-05-14 12:16:43 +01:00
goto_bad ;
if ( num_fields < 7 )
goto out ;
/*
* All pre - 1.9 .0 version parameters are read . Now we check
* for additional 1.9 .0 + parameters ( i . e . nr_fields at least 7 ) .
*
* Note that data_offset will be 0 if the
* kernel returns a pre - 1.9 .0 status .
*/
msg_fields = " <data_offset> " ;
if ( ! ( p = _skip_fields ( params , 6 ) ) ) /* skip pre-1.9.0 params */
goto bad ;
if ( sscanf ( p , FMTu64 , & s - > data_offset ) ! = 1 )
goto bad ;
2023-10-26 21:57:59 +02:00
/* <journal_char> - 'A' - active write-through journal device.
* - ' a ' - active write - back journal device .
* - ' D ' - dead journal device .
* - ' - ' - no journal device .
*/
2018-05-14 12:16:43 +01:00
out :
* status = s ;
2023-10-26 21:57:59 +02:00
while ( i - - > 0 )
if ( s - > dev_health [ i ] = = ' a ' )
a + + ; /* Count number of 'a' */
if ( a ) {
if ( ( a < s - > dev_count ) & & /* SOME legs are in 'a' */
/* FIXME: kernel gives misleading info here
* Trying to recognize a true state */
( s - > insync_regions = = s - > total_regions ) & &
( ! strcasecmp ( s - > sync_action , " recover " ) | |
! strcasecmp ( s - > sync_action , " idle " ) ) ) {
/* Kernel may possibly start some action
* in near - by future , do not report 100 % */
s - > insync_regions - - ;
}
if ( ( a = = s - > dev_count ) & & /* all legs are in 'a' */
( ! strcasecmp ( s - > sync_action , " resync " ) | |
! strcasecmp ( s - > sync_action , " idle " ) ) ) {
/* Mark 1st. leg in sync */
s - > dev_health [ 0 ] = ' A ' ;
2018-05-14 12:16:43 +01:00
}
}
return 1 ;
bad :
log_error ( " Failed to parse %sraid params: %s " , msg_fields , params ) ;
if ( s )
dm_pool_free ( mem , s ) ;
* status = NULL ;
return 0 ;
}
/*
* < metadata block size > < # used metadata blocks > / < # total metadata blocks >
* < cache block size > < # used cache blocks > / < # total cache blocks >
* < # read hits > < # read misses > < # write hits > < # write misses >
* < # demotions > < # promotions > < # dirty > < # features > < features > *
* < # core args > < core args > * < policy name > < # policy args > < policy args > *
*
* metadata block size : Fixed block size for each metadata block in
* sectors
* # used metadata blocks : Number of metadata blocks used
* # total metadata blocks : Total number of metadata blocks
* cache block size : Configurable block size for the cache device
* in sectors
* # used cache blocks : Number of blocks resident in the cache
* # total cache blocks : Total number of cache blocks
* # read hits : Number of times a READ bio has been mapped
* to the cache
* # read misses : Number of times a READ bio has been mapped
* to the origin
* # write hits : Number of times a WRITE bio has been mapped
* to the cache
* # write misses : Number of times a WRITE bio has been
* mapped to the origin
* # demotions : Number of times a block has been removed
* from the cache
* # promotions : Number of times a block has been moved to
* the cache
* # dirty : Number of blocks in the cache that differ
* from the origin
* # feature args : Number of feature args to follow
* feature args : ' writethrough ' ( optional )
* # core args : Number of core arguments ( must be even )
* core args : Key / value pairs for tuning the core
* e . g . migration_threshold
* * policy name : Name of the policy
* # policy args : Number of policy arguments to follow ( must be even )
* policy args : Key / value pairs
* e . g . sequential_threshold
*/
int dm_get_status_cache ( struct dm_pool * mem , const char * params ,
struct dm_status_cache * * status )
{
int i , feature_argc ;
char * str ;
const char * p , * pp ;
struct dm_status_cache * s ;
if ( ! ( s = dm_pool_zalloc ( mem , sizeof ( struct dm_status_cache ) ) ) )
return_0 ;
if ( strstr ( params , " Error " ) ) {
s - > error = 1 ;
s - > fail = 1 ; /* This is also I/O fail state */
goto out ;
}
if ( strstr ( params , " Fail " ) ) {
s - > fail = 1 ;
goto out ;
}
/* Read in args that have definitive placement */
if ( sscanf ( params ,
" " FMTu32
" " FMTu64 " / " FMTu64
" " FMTu32
" " FMTu64 " / " FMTu64
" " FMTu64 " " FMTu64
" " FMTu64 " " FMTu64
" " FMTu64 " " FMTu64
" " FMTu64
" %d " ,
& s - > metadata_block_size ,
& s - > metadata_used_blocks , & s - > metadata_total_blocks ,
& s - > block_size , /* AKA, chunk_size */
& s - > used_blocks , & s - > total_blocks ,
& s - > read_hits , & s - > read_misses ,
& s - > write_hits , & s - > write_misses ,
& s - > demotions , & s - > promotions ,
& s - > dirty_blocks ,
& feature_argc ) ! = 14 )
goto bad ;
/* Now jump to "features" section */
if ( ! ( p = _skip_fields ( params , 12 ) ) )
goto bad ;
/* Read in features */
for ( i = 0 ; i < feature_argc ; i + + ) {
if ( ! strncmp ( p , " writethrough " , 13 ) )
s - > feature_flags | = DM_CACHE_FEATURE_WRITETHROUGH ;
else if ( ! strncmp ( p , " writeback " , 10 ) )
s - > feature_flags | = DM_CACHE_FEATURE_WRITEBACK ;
else if ( ! strncmp ( p , " passthrough " , 12 ) )
s - > feature_flags | = DM_CACHE_FEATURE_PASSTHROUGH ;
else if ( ! strncmp ( p , " metadata2 " , 10 ) )
s - > feature_flags | = DM_CACHE_FEATURE_METADATA2 ;
2019-06-05 14:31:34 +02:00
else if ( ! strncmp ( p , " no_discard_passdown " , 20 ) )
s - > feature_flags | = DM_CACHE_FEATURE_NO_DISCARD_PASSDOWN ;
2018-05-14 12:16:43 +01:00
else
log_error ( " Unknown feature in status: %s " , params ) ;
if ( ! ( p = _skip_fields ( p , 1 ) ) )
goto bad ;
}
/* Read in core_args. */
if ( sscanf ( p , " %d " , & s - > core_argc ) ! = 1 )
goto bad ;
if ( ( s - > core_argc > 0 ) & &
( ! ( s - > core_argv = dm_pool_zalloc ( mem , sizeof ( char * ) * s - > core_argc ) ) | |
! ( p = _skip_fields ( p , 1 ) ) | |
! ( str = dm_pool_strdup ( mem , p ) ) | |
! ( p = _skip_fields ( p , ( unsigned ) s - > core_argc ) ) | |
( dm_split_words ( str , s - > core_argc , 0 , s - > core_argv ) ! = s - > core_argc ) ) )
goto bad ;
/* Read in policy args */
pp = p ;
if ( ! ( p = _skip_fields ( p , 1 ) ) | |
! ( s - > policy_name = dm_pool_zalloc ( mem , ( p - pp ) ) ) )
goto bad ;
if ( sscanf ( pp , " %s %d " , s - > policy_name , & s - > policy_argc ) ! = 2 )
goto bad ;
if ( s - > policy_argc & &
( ! ( s - > policy_argv = dm_pool_zalloc ( mem , sizeof ( char * ) * s - > policy_argc ) ) | |
! ( p = _skip_fields ( p , 1 ) ) | |
! ( str = dm_pool_strdup ( mem , p ) ) | |
( dm_split_words ( str , s - > policy_argc , 0 , s - > policy_argv ) ! = s - > policy_argc ) ) )
goto bad ;
/* TODO: improve this parser */
if ( strstr ( p , " ro " ) )
s - > read_only = 1 ;
if ( strstr ( p , " needs_check " ) )
s - > needs_check = 1 ;
out :
* status = s ;
return 1 ;
bad :
log_error ( " Failed to parse cache params: %s " , params ) ;
dm_pool_free ( mem , s ) ;
* status = NULL ;
return 0 ;
}
2018-08-27 14:53:09 -05:00
/*
* From linux / Documentation / device - mapper / writecache . txt
*
* Status :
* 1. error indicator - 0 if there was no error , otherwise error number
* 2. the number of blocks
* 3. the number of free blocks
* 4. the number of blocks under writeback
*/
int dm_get_status_writecache ( struct dm_pool * mem , const char * params ,
struct dm_status_writecache * * status )
{
struct dm_status_writecache * s ;
if ( ! ( s = dm_pool_zalloc ( mem , sizeof ( struct dm_status_writecache ) ) ) )
return_0 ;
2020-01-31 11:52:49 -06:00
if ( sscanf ( params , " %llu %llu %llu %llu " ,
( unsigned long long * ) & s - > error ,
2018-08-27 14:53:09 -05:00
( unsigned long long * ) & s - > total_blocks ,
( unsigned long long * ) & s - > free_blocks ,
( unsigned long long * ) & s - > writeback_blocks ) ! = 4 ) {
log_error ( " Failed to parse writecache params: %s. " , params ) ;
dm_pool_free ( mem , s ) ;
return 0 ;
}
* status = s ;
return 1 ;
}
2019-11-20 16:07:27 -06:00
int dm_get_status_integrity ( struct dm_pool * mem , const char * params ,
struct dm_status_integrity * * status )
{
struct dm_status_integrity * s ;
char recalc_str [ 16 ] = " \0 " ;
if ( ! ( s = dm_pool_zalloc ( mem , sizeof ( * s ) ) ) )
return_0 ;
if ( sscanf ( params , " %llu %llu %s " ,
( unsigned long long * ) & s - > number_of_mismatches ,
( unsigned long long * ) & s - > provided_data_sectors ,
recalc_str ) ! = 3 ) {
log_error ( " Failed to parse integrity params: %s. " , params ) ;
dm_pool_free ( mem , s ) ;
return 0 ;
}
if ( recalc_str [ 0 ] = = ' - ' )
s - > recalc_sector = 0 ;
else
s - > recalc_sector = strtoull ( recalc_str , NULL , 0 ) ;
* status = s ;
return 1 ;
}
2018-05-14 12:16:43 +01:00
int parse_thin_pool_status ( const char * params , struct dm_status_thin_pool * s )
{
int pos ;
memset ( s , 0 , sizeof ( * s ) ) ;
if ( ! params ) {
log_error ( " Failed to parse invalid thin params. " ) ;
return 0 ;
}
if ( strstr ( params , " Error " ) ) {
s - > error = 1 ;
s - > fail = 1 ; /* This is also I/O fail state */
return 1 ;
}
if ( strstr ( params , " Fail " ) ) {
s - > fail = 1 ;
return 1 ;
}
/* FIXME: add support for held metadata root */
if ( sscanf ( params , FMTu64 " " FMTu64 " / " FMTu64 " " FMTu64 " / " FMTu64 " %n " ,
& s - > transaction_id ,
& s - > used_metadata_blocks ,
& s - > total_metadata_blocks ,
& s - > used_data_blocks ,
& s - > total_data_blocks , & pos ) < 5 ) {
log_error ( " Failed to parse thin pool params: %s. " , params ) ;
return 0 ;
}
/* New status flags */
if ( strstr ( params + pos , " no_discard_passdown " ) )
s - > discards = DM_THIN_DISCARDS_NO_PASSDOWN ;
else if ( strstr ( params + pos , " ignore_discard " ) )
s - > discards = DM_THIN_DISCARDS_IGNORE ;
else /* default discard_passdown */
s - > discards = DM_THIN_DISCARDS_PASSDOWN ;
/* Default is 'writable' (rw) data */
if ( strstr ( params + pos , " out_of_data_space " ) )
s - > out_of_data_space = 1 ;
else if ( strstr ( params + pos , " ro " ) )
s - > read_only = 1 ;
/* Default is 'queue_if_no_space' */
if ( strstr ( params + pos , " error_if_no_space " ) )
s - > error_if_no_space = 1 ;
if ( strstr ( params + pos , " needs_check " ) )
s - > needs_check = 1 ;
return 1 ;
}
int dm_get_status_thin_pool ( struct dm_pool * mem , const char * params ,
struct dm_status_thin_pool * * status )
{
struct dm_status_thin_pool * s ;
if ( ! ( s = dm_pool_alloc ( mem , sizeof ( struct dm_status_thin_pool ) ) ) ) {
log_error ( " Failed to allocate thin_pool status structure. " ) ;
return 0 ;
}
if ( ! parse_thin_pool_status ( params , s ) ) {
dm_pool_free ( mem , s ) ;
return_0 ;
}
* status = s ;
return 1 ;
}
int dm_get_status_thin ( struct dm_pool * mem , const char * params ,
struct dm_status_thin * * status )
{
struct dm_status_thin * s ;
if ( ! ( s = dm_pool_zalloc ( mem , sizeof ( struct dm_status_thin ) ) ) ) {
log_error ( " Failed to allocate thin status structure. " ) ;
return 0 ;
}
if ( strchr ( params , ' - ' ) ) {
/* nothing to parse */
} else if ( strstr ( params , " Fail " ) ) {
s - > fail = 1 ;
} else if ( sscanf ( params , FMTu64 " " FMTu64 ,
& s - > mapped_sectors ,
& s - > highest_mapped_sector ) ! = 2 ) {
dm_pool_free ( mem , s ) ;
log_error ( " Failed to parse thin params: %s. " , params ) ;
return 0 ;
}
* status = s ;
return 1 ;
}
2024-05-07 16:33:03 +02:00
static dm_status_mirror_health_t _get_health ( char c )
{
switch ( c ) {
case ' A ' : return DM_STATUS_MIRROR_ALIVE ;
case ' F ' : return DM_STATUS_MIRROR_FLUSH_FAILED ;
case ' D ' : return DM_STATUS_MIRROR_WRITE_FAILED ;
case ' S ' : return DM_STATUS_MIRROR_SYNC_FAILED ;
case ' R ' : return DM_STATUS_MIRROR_READ_FAILED ;
default :
log_warn ( " WARNING: Unknown mirror health status char: %c " , c ) ;
2024-05-14 15:35:07 +02:00
/* fall through */
2024-05-07 16:33:03 +02:00
case ' U ' : return DM_STATUS_MIRROR_UNCLASSIFIED ;
}
}
2018-05-14 12:16:43 +01:00
/*
* dm core parms : 0 409600 mirror
* Mirror core parms : 2 253 : 4 253 : 5 400 / 400
* New - style failure params : 1 AA
* New - style log params : 3 cluster 253 : 3 A
* or 3 disk 253 : 3 A
* or 1 core
*/
# define DM_MIRROR_MAX_IMAGES 8 /* limited by kernel DM_KCOPYD_MAX_REGIONS */
int dm_get_status_mirror ( struct dm_pool * mem , const char * params ,
struct dm_status_mirror * * status )
{
struct dm_status_mirror * s ;
const char * p , * pos = params ;
unsigned num_devs , argc , i ;
int used ;
if ( ! ( s = dm_pool_zalloc ( mem , sizeof ( * s ) ) ) ) {
log_error ( " Failed to alloc mem pool to parse mirror status. " ) ;
return 0 ;
}
if ( sscanf ( pos , " %u %n " , & num_devs , & used ) ! = 1 )
goto_out ;
pos + = used ;
if ( num_devs > DM_MIRROR_MAX_IMAGES ) {
log_error ( INTERNAL_ERROR " More then " DM_TO_STRING ( DM_MIRROR_MAX_IMAGES )
" reported in mirror status. " ) ;
goto out ;
}
if ( ! ( s - > devs = dm_pool_alloc ( mem , num_devs * sizeof ( * ( s - > devs ) ) ) ) ) {
log_error ( " Allocation of devs failed. " ) ;
goto out ;
}
for ( i = 0 ; i < num_devs ; + + i , pos + = used )
if ( sscanf ( pos , " %u:%u %n " ,
& ( s - > devs [ i ] . major ) , & ( s - > devs [ i ] . minor ) , & used ) ! = 2 )
goto_out ;
if ( sscanf ( pos , FMTu64 " / " FMTu64 " %n " ,
& s - > insync_regions , & s - > total_regions , & used ) ! = 2 )
goto_out ;
pos + = used ;
if ( sscanf ( pos , " %u %n " , & argc , & used ) ! = 1 )
goto_out ;
pos + = used ;
for ( i = 0 ; i < num_devs ; + + i )
2024-05-07 16:33:03 +02:00
s - > devs [ i ] . health = _get_health ( pos [ i ] ) ;
2018-05-14 12:16:43 +01:00
if ( ! ( pos = _skip_fields ( pos , argc ) ) )
goto_out ;
if ( strncmp ( pos , " userspace " , 9 ) = = 0 ) {
pos + = 9 ;
/* FIXME: support status of userspace mirror implementation */
}
if ( sscanf ( pos , " %u %n " , & argc , & used ) ! = 1 )
goto_out ;
pos + = used ;
if ( argc = = 1 ) {
/* core, cluster-core */
if ( ! ( s - > log_type = dm_pool_strdup ( mem , pos ) ) ) {
log_error ( " Allocation of log type string failed. " ) ;
goto out ;
}
} else {
if ( ! ( p = _skip_fields ( pos , 1 ) ) )
goto_out ;
/* disk, cluster-disk */
if ( ! ( s - > log_type = dm_pool_strndup ( mem , pos , p - pos - 1 ) ) ) {
log_error ( " Allocation of log type string failed. " ) ;
goto out ;
}
pos = p ;
if ( ( argc > 2 ) & & ! strcmp ( s - > log_type , " disk " ) ) {
s - > log_count = argc - 2 ;
if ( ! ( s - > logs = dm_pool_alloc ( mem , s - > log_count * sizeof ( * ( s - > logs ) ) ) ) ) {
log_error ( " Allocation of logs failed. " ) ;
goto out ;
}
for ( i = 0 ; i < s - > log_count ; + + i , pos + = used )
if ( sscanf ( pos , " %u:%u %n " ,
& s - > logs [ i ] . major , & s - > logs [ i ] . minor , & used ) ! = 2 )
goto_out ;
for ( i = 0 ; i < s - > log_count ; + + i )
2024-05-07 16:33:03 +02:00
s - > logs [ i ] . health = _get_health ( pos [ i ] ) ;
2018-05-14 12:16:43 +01:00
}
}
s - > dev_count = num_devs ;
* status = s ;
return 1 ;
out :
log_error ( " Failed to parse mirror status %s. " , params ) ;
dm_pool_free ( mem , s ) ;
* status = NULL ;
return 0 ;
}