2004-05-05 01:25:57 +04:00
/*
* Copyright ( C ) 2003 - 2004 Sistina Software , Inc . All rights reserved .
* Copyright ( C ) 2004 Red Hat , Inc . All rights reserved .
*
* 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
*/
# include "lib.h"
# include "pool.h"
# include "list.h"
# include "toolcontext.h"
2004-09-16 22:40:56 +04:00
# include "segtype.h"
2004-05-05 01:25:57 +04:00
# include "display.h"
# include "text_export.h"
# include "text_import.h"
# include "config.h"
# include "str_list.h"
# include "targets.h"
# include "lvm-string.h"
2004-05-05 22:11:43 +04:00
# include "activate.h"
2005-05-03 21:28:23 +04:00
# include "pv_alloc.h"
2004-05-05 01:25:57 +04:00
static const char * _name ( const struct lv_segment * seg )
{
return ( seg - > area_count = = 1 ) ? " linear " : seg - > segtype - > name ;
}
static void _display ( const struct lv_segment * seg )
{
uint32_t s ;
if ( seg - > area_count = = 1 )
display_stripe ( seg , 0 , " " ) ;
else {
log_print ( " Stripes \t \t %u " , seg - > area_count ) ;
log_print ( " Stripe size \t \t %u KB " , seg - > stripe_size / 2 ) ;
for ( s = 0 ; s < seg - > area_count ; s + + ) {
log_print ( " Stripe %d: " , s ) ;
display_stripe ( seg , s , " " ) ;
}
}
log_print ( " " ) ;
}
static int _text_import_area_count ( struct config_node * sn , uint32_t * area_count )
{
if ( ! get_config_uint32 ( sn , " stripe_count " , area_count ) ) {
log_error ( " Couldn't read 'stripe_count' for "
" segment '%s'. " , sn - > key ) ;
return 0 ;
}
return 1 ;
}
static int _text_import ( struct lv_segment * seg , const struct config_node * sn ,
struct hash_table * pv_hash )
{
struct config_node * cn ;
if ( ( seg - > area_count ! = 1 ) & &
! get_config_uint32 ( sn , " stripe_size " , & seg - > stripe_size ) ) {
log_error ( " Couldn't read stripe_size for segment '%s'. " ,
sn - > key ) ;
return 0 ;
}
if ( ! ( cn = find_config_node ( sn , " stripes " ) ) ) {
log_error ( " Couldn't find stripes array for segment "
" '%s'. " , sn - > key ) ;
return 0 ;
}
seg - > area_len / = seg - > area_count ;
2005-06-03 18:49:51 +04:00
return text_import_areas ( seg , sn , cn , pv_hash , 0 ) ;
2004-05-05 01:25:57 +04:00
}
static int _text_export ( const struct lv_segment * seg , struct formatter * f )
{
outf ( f , " stripe_count = %u%s " , seg - > area_count ,
( seg - > area_count = = 1 ) ? " \t # linear " : " " ) ;
if ( seg - > area_count > 1 )
out_size ( f , ( uint64_t ) seg - > stripe_size ,
" stripe_size = %u " , seg - > stripe_size ) ;
return out_areas ( f , seg , " stripe " ) ;
}
/*
* Test whether two segments could be merged by the current merging code
*/
static int _segments_compatible ( struct lv_segment * first ,
struct lv_segment * second )
{
uint32_t width ;
unsigned s ;
if ( ( first - > area_count ! = second - > area_count ) | |
( first - > stripe_size ! = second - > stripe_size ) ) return 0 ;
for ( s = 0 ; s < first - > area_count ; s + + ) {
/* FIXME Relax this to first area type != second area type */
/* plus the additional AREA_LV checks needed */
if ( ( first - > area [ s ] . type ! = AREA_PV ) | |
( second - > area [ s ] . type ! = AREA_PV ) ) return 0 ;
width = first - > area_len ;
2005-06-01 20:51:55 +04:00
if ( ( seg_pv ( first , s ) ! =
seg_pv ( second , s ) ) | |
( seg_pe ( first , s ) + width ! =
seg_pe ( second , s ) ) )
2004-05-05 01:25:57 +04:00
return 0 ;
}
if ( ! str_list_lists_equal ( & first - > tags , & second - > tags ) )
return 0 ;
return 1 ;
}
static int _merge_segments ( struct lv_segment * seg1 , struct lv_segment * seg2 )
{
2005-05-03 21:28:23 +04:00
uint32_t s ;
2004-05-05 01:25:57 +04:00
if ( ! _segments_compatible ( seg1 , seg2 ) )
return 0 ;
seg1 - > len + = seg2 - > len ;
seg1 - > area_len + = seg2 - > area_len ;
2005-05-03 21:28:23 +04:00
for ( s = 0 ; s < seg1 - > area_count ; s + + )
if ( seg1 - > area [ s ] . type = = AREA_PV )
2005-06-01 20:51:55 +04:00
merge_pv_segments ( seg_pvseg ( seg1 , s ) ,
seg_pvseg ( seg2 , s ) ) ;
2005-05-03 21:28:23 +04:00
2004-05-05 01:25:57 +04:00
return 1 ;
}
# ifdef DEVMAPPER_SUPPORT
static int _compose_target_line ( struct dev_manager * dm , struct pool * mem ,
struct config_tree * cft , void * * target_state ,
struct lv_segment * seg , char * params ,
size_t paramsize , const char * * target , int * pos ,
uint32_t * pvmove_mirror_count )
{
/* linear [device offset]+
* striped # stripes stripe_size [ device offset ] + */
if ( seg - > area_count = = 1 )
* target = " linear " ;
else if ( seg - > area_count > 1 ) {
* target = " striped " ;
if ( ( * pos = lvm_snprintf ( params , paramsize , " %u %u " ,
seg - > area_count ,
seg - > stripe_size ) ) < 0 ) {
stack ;
return - 1 ;
}
} else {
log_error ( " Internal error: striped target with no stripes " ) ;
return 0 ;
}
return compose_areas_line ( dm , seg , params , paramsize , pos , 0u ,
seg - > area_count ) ;
}
2004-05-05 22:11:43 +04:00
static int _target_present ( void )
{
static int checked = 0 ;
static int present = 0 ;
if ( ! checked )
present = target_present ( " linear " ) & & target_present ( " striped " ) ;
checked = 1 ;
return present ;
}
2004-05-05 01:25:57 +04:00
# endif
static void _destroy ( const struct segment_type * segtype )
{
dbg_free ( ( void * ) segtype ) ;
}
static struct segtype_handler _striped_ops = {
name : _name ,
display : _display ,
text_import_area_count : _text_import_area_count ,
text_import : _text_import ,
text_export : _text_export ,
merge_segments : _merge_segments ,
# ifdef DEVMAPPER_SUPPORT
compose_target_line : _compose_target_line ,
2004-05-05 22:11:43 +04:00
target_present : _target_present ,
2004-05-05 01:25:57 +04:00
# endif
destroy : _destroy ,
} ;
struct segment_type * init_striped_segtype ( struct cmd_context * cmd )
{
struct segment_type * segtype = dbg_malloc ( sizeof ( * segtype ) ) ;
if ( ! segtype ) {
stack ;
return NULL ;
}
segtype - > cmd = cmd ;
segtype - > ops = & _striped_ops ;
segtype - > name = " striped " ;
segtype - > private = NULL ;
segtype - > flags =
SEG_CAN_SPLIT | SEG_AREAS_STRIPED | SEG_FORMAT1_SUPPORT ;
2004-09-14 21:37:51 +04:00
log_very_verbose ( " Initialised segtype: %s " , segtype - > name ) ;
2004-05-05 01:25:57 +04:00
return segtype ;
}