2018-04-30 18:16:58 +03:00
# include "target.h"
// For DM_ARRAY_SIZE!
2018-05-14 14:16:43 +03:00
# include "device_mapper/libdevmapper.h"
2018-04-30 18:16:58 +03:00
2018-05-02 13:15:35 +03:00
# include <ctype.h>
2018-04-30 18:16:58 +03:00
# include <stdlib.h>
2018-05-02 13:15:35 +03:00
# include <string.h>
2018-04-30 18:16:58 +03:00
//----------------------------------------------------------------
2018-05-02 13:15:35 +03:00
static char * _tok_cpy ( const char * b , const char * e )
2018-04-30 18:16:58 +03:00
{
char * new = malloc ( ( e - b ) + 1 ) ;
char * ptr = new ;
2018-05-10 15:01:26 +03:00
if ( new ) {
2018-04-30 18:16:58 +03:00
while ( b ! = e )
* ptr + + = * b + + ;
2018-05-10 15:01:26 +03:00
* ptr = ' \0 ' ;
}
2018-04-30 18:16:58 +03:00
return new ;
}
static bool _tok_eq ( const char * b , const char * e , const char * str )
{
while ( b ! = e ) {
if ( ! * str | | * b ! = * str )
return false ;
b + + ;
str + + ;
}
return ! * str ;
}
2018-05-02 13:15:35 +03:00
static bool _parse_operating_mode ( const char * b , const char * e , void * context )
2018-04-30 18:16:58 +03:00
{
static struct {
const char * str ;
enum vdo_operating_mode mode ;
} _table [ ] = {
{ " recovering " , VDO_MODE_RECOVERING } ,
{ " read-only " , VDO_MODE_READ_ONLY } ,
{ " normal " , VDO_MODE_NORMAL }
} ;
2018-05-02 13:15:35 +03:00
enum vdo_operating_mode * r = context ;
2018-04-30 18:16:58 +03:00
unsigned i ;
for ( i = 0 ; i < DM_ARRAY_SIZE ( _table ) ; i + + ) {
if ( _tok_eq ( b , e , _table [ i ] . str ) ) {
* r = _table [ i ] . mode ;
return true ;
}
}
return false ;
}
2018-05-02 13:15:35 +03:00
static bool _parse_compression_state ( const char * b , const char * e , void * context )
2018-04-30 18:16:58 +03:00
{
static struct {
const char * str ;
enum vdo_compression_state state ;
} _table [ ] = {
{ " online " , VDO_COMPRESSION_ONLINE } ,
{ " offline " , VDO_COMPRESSION_OFFLINE }
} ;
2018-05-02 13:15:35 +03:00
enum vdo_compression_state * r = context ;
2018-04-30 18:16:58 +03:00
unsigned i ;
for ( i = 0 ; i < DM_ARRAY_SIZE ( _table ) ; i + + ) {
if ( _tok_eq ( b , e , _table [ i ] . str ) ) {
* r = _table [ i ] . state ;
return true ;
}
}
return false ;
}
2018-05-02 13:15:35 +03:00
static bool _parse_recovering ( const char * b , const char * e , void * context )
2018-04-30 18:16:58 +03:00
{
2018-05-02 13:15:35 +03:00
bool * r = context ;
2018-04-30 18:16:58 +03:00
if ( _tok_eq ( b , e , " recovering " ) )
* r = true ;
else if ( _tok_eq ( b , e , " - " ) )
* r = false ;
else
return false ;
return true ;
}
2018-05-02 13:15:35 +03:00
static bool _parse_index_state ( const char * b , const char * e , void * context )
2018-04-30 18:16:58 +03:00
{
static struct {
const char * str ;
enum vdo_index_state state ;
} _table [ ] = {
{ " error " , VDO_INDEX_ERROR } ,
{ " closed " , VDO_INDEX_CLOSED } ,
{ " opening " , VDO_INDEX_OPENING } ,
{ " closing " , VDO_INDEX_CLOSING } ,
{ " offline " , VDO_INDEX_OFFLINE } ,
{ " online " , VDO_INDEX_ONLINE } ,
{ " unknown " , VDO_INDEX_UNKNOWN }
} ;
2018-05-02 13:15:35 +03:00
enum vdo_index_state * r = context ;
2018-04-30 18:16:58 +03:00
unsigned i ;
for ( i = 0 ; i < DM_ARRAY_SIZE ( _table ) ; i + + ) {
if ( _tok_eq ( b , e , _table [ i ] . str ) ) {
* r = _table [ i ] . state ;
return true ;
}
}
return false ;
}
2018-05-02 13:15:35 +03:00
static bool _parse_uint64 ( const char * b , const char * e , void * context )
{
uint64_t * r = context , n ;
n = 0 ;
while ( b ! = e ) {
if ( ! isdigit ( * b ) )
return false ;
2018-05-10 15:01:26 +03:00
n = ( n * 10 ) + ( * b - ' 0 ' ) ;
2018-05-02 13:15:35 +03:00
b + + ;
}
* r = n ;
return true ;
}
2018-04-30 18:16:58 +03:00
static const char * _eat_space ( const char * b , const char * e )
{
while ( b ! = e & & isspace ( * b ) )
b + + ;
return b ;
}
static const char * _next_tok ( const char * b , const char * e )
{
2018-05-10 15:01:26 +03:00
const char * te = b ;
while ( te ! = e & & ! isspace ( * te ) )
te + + ;
2018-04-30 18:16:58 +03:00
2018-05-10 15:01:26 +03:00
return te = = b ? NULL : te ;
2018-04-30 18:16:58 +03:00
}
2018-05-10 15:01:26 +03:00
static void _set_error ( struct vdo_status_parse_result * result , const char * fmt , . . . )
2018-05-02 13:15:35 +03:00
__attribute__ ( ( format ( printf , 2 , 3 ) ) ) ;
2018-05-10 15:01:26 +03:00
static void _set_error ( struct vdo_status_parse_result * result , const char * fmt , . . . )
2018-05-02 13:15:35 +03:00
{
va_list ap ;
va_start ( ap , fmt ) ;
vsnprintf ( result - > error , sizeof ( result - > error ) , fmt , ap ) ;
va_end ( ap ) ;
}
2018-04-30 18:16:58 +03:00
static bool _parse_field ( const char * * b , const char * e ,
2018-05-02 13:15:35 +03:00
bool ( * p_fn ) ( const char * , const char * , void * ) ,
2018-04-30 18:16:58 +03:00
void * field , const char * field_name ,
2018-05-10 15:01:26 +03:00
struct vdo_status_parse_result * result )
2018-04-30 18:16:58 +03:00
{
const char * te ;
te = _next_tok ( * b , e ) ;
if ( ! te ) {
2018-05-02 13:15:35 +03:00
_set_error ( result , " couldn't get token for '%s' " , field_name ) ;
2018-04-30 18:16:58 +03:00
return false ;
}
2018-05-02 13:15:35 +03:00
if ( ! p_fn ( * b , te , field ) ) {
_set_error ( result , " couldn't parse '%s' " , field_name ) ;
2018-04-30 18:16:58 +03:00
return false ;
}
2018-05-02 13:15:35 +03:00
* b = _eat_space ( te , e ) ;
2018-04-30 18:16:58 +03:00
return true ;
}
2018-05-10 15:01:26 +03:00
bool vdo_status_parse ( const char * input , struct vdo_status_parse_result * result )
2018-04-30 18:16:58 +03:00
{
2018-05-02 13:15:35 +03:00
const char * b = b = input ;
2018-04-30 18:16:58 +03:00
const char * e = input + strlen ( input ) ;
const char * te ;
struct vdo_status * s = malloc ( sizeof ( * s ) ) ;
if ( ! s ) {
2018-05-02 13:15:35 +03:00
_set_error ( result , " out of memory " ) ;
2018-04-30 18:16:58 +03:00
return false ;
}
2018-05-02 13:15:35 +03:00
b = _eat_space ( b , e ) ;
2018-04-30 18:16:58 +03:00
te = _next_tok ( b , e ) ;
if ( ! te ) {
2018-05-02 13:15:35 +03:00
_set_error ( result , " couldn't get token for device " ) ;
2018-04-30 18:16:58 +03:00
free ( s ) ;
return false ;
}
s - > device = _tok_cpy ( b , te ) ;
if ( ! s - > device ) {
2018-05-02 13:15:35 +03:00
_set_error ( result , " out of memory " ) ;
2018-04-30 18:16:58 +03:00
free ( s ) ;
return false ;
}
2018-05-02 13:15:35 +03:00
b = _eat_space ( te , e ) ;
2018-04-30 18:16:58 +03:00
# define XX(p, f, fn) if (!_parse_field(&b, e, p, f, fn, result)) goto bad;
2018-05-02 13:15:35 +03:00
XX ( _parse_operating_mode , & s - > operating_mode , " operating mode " ) ;
2018-04-30 18:16:58 +03:00
XX ( _parse_recovering , & s - > recovering , " recovering " ) ;
2018-05-02 13:15:35 +03:00
XX ( _parse_index_state , & s - > index_state , " index state " ) ;
2018-04-30 18:16:58 +03:00
XX ( _parse_compression_state , & s - > compression_state , " compression state " ) ;
XX ( _parse_uint64 , & s - > used_blocks , " used blocks " ) ;
XX ( _parse_uint64 , & s - > total_blocks , " total blocks " ) ;
# undef XX
2018-05-10 15:01:26 +03:00
if ( b ! = e ) {
_set_error ( result , " too many tokens " ) ;
goto bad ;
}
result - > status = s ;
2018-04-30 18:16:58 +03:00
return true ;
bad :
free ( s - > device ) ;
free ( s ) ;
return false ;
}
//----------------------------------------------------------------