2010-09-30 13:16:55 +00:00
/*
2010-09-30 20:47:18 +00:00
* Copyright ( C ) 2001 - 2004 Sistina Software , Inc . All rights reserved .
2013-01-15 15:16:16 +01:00
* Copyright ( C ) 2004 - 2013 Red Hat , Inc . All rights reserved .
2010-09-30 13:16:55 +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 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 . , 59 Temple Place , Suite 330 , Boston , MA 02111 - 1307 USA
*/
# include "lib.h"
# include "metadata.h"
2010-10-25 13:54:29 +00:00
# include "display.h"
2010-09-30 13:52:55 +00:00
# include "activate.h"
2010-10-12 16:11:34 +00:00
# include "toolcontext.h"
2010-10-12 16:12:50 +00:00
# include "segtype.h"
2010-10-12 16:13:06 +00:00
# include "str_list.h"
2012-01-19 15:31:45 +00:00
# include <time.h>
# include <sys/utsname.h>
static struct utsname _utsname ;
static int _utsinit = 0 ;
2011-04-12 12:24:29 +00:00
static char * _format_pvsegs ( struct dm_pool * mem , const struct lv_segment * seg ,
int range_format )
{
unsigned int s ;
const char * name = NULL ;
uint32_t extent = 0 ;
char extent_str [ 32 ] ;
if ( ! dm_pool_begin_object ( mem , 256 ) ) {
log_error ( " dm_pool_begin_object failed " ) ;
return NULL ;
}
for ( s = 0 ; s < seg - > area_count ; s + + ) {
switch ( seg_type ( seg , s ) ) {
case AREA_LV :
name = seg_lv ( seg , s ) - > name ;
extent = seg_le ( seg , s ) ;
break ;
case AREA_PV :
name = dev_name ( seg_dev ( seg , s ) ) ;
extent = seg_pe ( seg , s ) ;
break ;
case AREA_UNASSIGNED :
name = " unassigned " ;
extent = 0 ;
2012-02-27 11:13:48 +00:00
break ;
2012-02-13 11:25:56 +00:00
default :
log_error ( INTERNAL_ERROR " Unknown area segtype. " ) ;
return NULL ;
2011-04-12 12:24:29 +00:00
}
if ( ! dm_pool_grow_object ( mem , name , strlen ( name ) ) ) {
log_error ( " dm_pool_grow_object failed " ) ;
return NULL ;
}
if ( dm_snprintf ( extent_str , sizeof ( extent_str ) ,
" %s% " PRIu32 " %s " ,
range_format ? " : " : " ( " , extent ,
range_format ? " - " : " ) " ) < 0 ) {
log_error ( " Extent number dm_snprintf failed " ) ;
return NULL ;
}
if ( ! dm_pool_grow_object ( mem , extent_str , strlen ( extent_str ) ) ) {
log_error ( " dm_pool_grow_object failed " ) ;
return NULL ;
}
if ( range_format ) {
if ( dm_snprintf ( extent_str , sizeof ( extent_str ) ,
" % " PRIu32 , extent + seg - > area_len - 1 ) < 0 ) {
log_error ( " Extent number dm_snprintf failed " ) ;
return NULL ;
}
if ( ! dm_pool_grow_object ( mem , extent_str , strlen ( extent_str ) ) ) {
log_error ( " dm_pool_grow_object failed " ) ;
return NULL ;
}
}
if ( ( s ! = seg - > area_count - 1 ) & &
! dm_pool_grow_object ( mem , range_format ? " " : " , " , 1 ) ) {
log_error ( " dm_pool_grow_object failed " ) ;
return NULL ;
}
}
if ( ! dm_pool_grow_object ( mem , " \0 " , 1 ) ) {
log_error ( " dm_pool_grow_object failed " ) ;
return NULL ;
}
return dm_pool_end_object ( mem ) ;
}
char * lvseg_devices ( struct dm_pool * mem , const struct lv_segment * seg )
{
return _format_pvsegs ( mem , seg , 0 ) ;
}
char * lvseg_seg_pe_ranges ( struct dm_pool * mem , const struct lv_segment * seg )
{
return _format_pvsegs ( mem , seg , 1 ) ;
}
2010-11-17 20:08:14 +00:00
char * lvseg_tags_dup ( const struct lv_segment * seg )
{
return tags_format_and_copy ( seg - > lv - > vg - > vgmem , & seg - > tags ) ;
}
2011-03-05 12:14:00 +00:00
char * lvseg_segtype_dup ( struct dm_pool * mem , const struct lv_segment * seg )
2010-11-17 20:08:14 +00:00
{
2011-03-05 12:14:00 +00:00
return dm_pool_strdup ( mem , seg - > segtype - > ops - > name ( seg ) ) ;
2010-11-17 20:08:14 +00:00
}
2012-11-27 11:02:49 +01:00
char * lvseg_discards_dup ( struct dm_pool * mem , const struct lv_segment * seg )
{
return dm_pool_strdup ( mem , get_pool_discards_name ( seg - > discards ) ) ;
}
2013-04-25 12:07:57 +02:00
# ifdef DMEVENTD
# include "libdevmapper-event.h"
# endif
char * lvseg_monitor_dup ( struct dm_pool * mem , const struct lv_segment * seg )
{
const char * s = " " ;
# ifdef DMEVENTD
struct lvinfo info ;
int pending = 0 , monitored ;
struct lv_segment * segm = ( struct lv_segment * ) seg ;
if ( lv_is_cow ( seg - > lv ) & & ! lv_is_merging_cow ( seg - > lv ) )
segm = first_seg ( seg - > lv - > snapshot - > lv ) ;
else if ( seg - > log_lv )
segm = first_seg ( seg - > log_lv ) ;
// log_debug("Query LV:%s mon:%s segm:%s tgtm:%p segmon:%d statusm:%d", seg->lv->name, segm->lv->name, segm->segtype->name, segm->segtype->ops->target_monitored, seg_monitored(segm), (int)(segm->status & PVMOVE));
if ( ! segm - > segtype - > ops - > target_monitored )
/* Nothing to do, monitoring not supported */ ;
2013-05-27 10:20:06 +02:00
else if ( lv_is_cow_covering_origin ( seg - > lv ) )
/* Nothing to do, snapshot already covers origin */ ;
2013-04-25 12:07:57 +02:00
else if ( ! seg_monitored ( segm ) | | ( segm - > status & PVMOVE ) )
s = " not monitored " ;
else if ( lv_info ( seg - > lv - > vg - > cmd , seg - > lv , 1 , & info , 0 , 0 ) & & info . exists ) {
monitored = segm - > segtype - > ops - >
target_monitored ( ( struct lv_segment * ) segm , & pending ) ;
if ( pending )
s = " pending " ;
else
s = ( monitored ) ? " monitored " : " not monitored " ;
} // else log_debug("Not active");
# endif
return dm_pool_strdup ( mem , s ) ;
}
2010-11-17 20:08:14 +00:00
uint64_t lvseg_chunksize ( const struct lv_segment * seg )
{
uint64_t size ;
if ( lv_is_cow ( seg - > lv ) )
size = ( uint64_t ) find_cow ( seg - > lv ) - > chunk_size ;
2012-01-19 15:34:32 +00:00
else if ( lv_is_thin_pool ( seg - > lv ) )
2012-01-24 00:55:03 +00:00
size = ( uint64_t ) seg - > chunk_size ;
2010-11-17 20:08:14 +00:00
else
size = UINT64_C ( 0 ) ;
2012-01-19 15:34:32 +00:00
2010-11-17 20:08:14 +00:00
return size ;
}
uint64_t lvseg_start ( const struct lv_segment * seg )
{
return ( uint64_t ) seg - > le * seg - > lv - > vg - > extent_size ;
}
uint64_t lvseg_size ( const struct lv_segment * seg )
{
return ( uint64_t ) seg - > len * seg - > lv - > vg - > extent_size ;
}
2010-10-21 14:49:31 +00:00
uint32_t lv_kernel_read_ahead ( const struct logical_volume * lv )
{
struct lvinfo info ;
if ( ! lv_info ( lv - > vg - > cmd , lv , 0 , & info , 0 , 1 ) | | ! info . exists )
return UINT32_MAX ;
return info . read_ahead ;
}
2010-10-21 14:49:20 +00:00
char * lv_origin_dup ( struct dm_pool * mem , const struct logical_volume * lv )
{
if ( lv_is_cow ( lv ) )
return lv_name_dup ( mem , origin_from_cow ( lv ) ) ;
2012-10-12 12:02:15 +02:00
if ( lv_is_thin_volume ( lv ) & & first_seg ( lv ) - > origin )
return lv_name_dup ( mem , first_seg ( lv ) - > origin ) ;
2013-01-15 15:16:16 +01:00
if ( lv_is_thin_volume ( lv ) & & first_seg ( lv ) - > external_lv )
return lv_name_dup ( mem , first_seg ( lv ) - > external_lv ) ;
2010-10-21 14:49:20 +00:00
return NULL ;
}
2010-10-21 14:49:10 +00:00
char * lv_name_dup ( struct dm_pool * mem , const struct logical_volume * lv )
{
return dm_pool_strdup ( mem , lv - > name ) ;
}
2010-10-12 16:13:06 +00:00
char * lv_modules_dup ( struct dm_pool * mem , const struct logical_volume * lv )
{
struct dm_list * modules ;
if ( ! ( modules = str_list_create ( mem ) ) ) {
log_error ( " modules str_list allocation failed " ) ;
return NULL ;
}
if ( ! list_lv_modules ( mem , lv , modules ) )
return_NULL ;
2010-10-12 17:09:23 +00:00
return tags_format_and_copy ( mem , modules ) ;
2010-10-12 16:13:06 +00:00
}
2010-10-12 16:12:50 +00:00
char * lv_mirror_log_dup ( struct dm_pool * mem , const struct logical_volume * lv )
{
struct lv_segment * seg ;
2012-01-20 10:56:30 +00:00
dm_list_iterate_items ( seg , & lv - > segments )
if ( seg_is_mirrored ( seg ) & & seg - > log_lv )
return dm_pool_strdup ( mem , seg - > log_lv - > name ) ;
2010-10-12 16:12:50 +00:00
return NULL ;
}
2010-10-12 16:11:34 +00:00
2011-09-09 00:54:49 +00:00
char * lv_pool_lv_dup ( struct dm_pool * mem , const struct logical_volume * lv )
{
struct lv_segment * seg ;
2012-01-20 10:56:30 +00:00
dm_list_iterate_items ( seg , & lv - > segments )
if ( seg_is_thin_volume ( seg ) & & seg - > pool_lv )
return dm_pool_strdup ( mem , seg - > pool_lv - > name ) ;
2011-09-09 00:54:49 +00:00
return NULL ;
}
2012-01-19 15:34:32 +00:00
char * lv_data_lv_dup ( struct dm_pool * mem , const struct logical_volume * lv )
{
return lv_is_thin_pool ( lv ) ?
dm_pool_strdup ( mem , seg_lv ( first_seg ( lv ) , 0 ) - > name ) : NULL ;
}
char * lv_metadata_lv_dup ( struct dm_pool * mem , const struct logical_volume * lv )
{
return lv_is_thin_pool ( lv ) ?
dm_pool_strdup ( mem , first_seg ( lv ) - > metadata_lv - > name ) : NULL ;
}
2013-02-01 11:09:34 +01:00
const char * lv_layer ( const struct logical_volume * lv )
{
if ( lv_is_thin_pool ( lv ) )
return " tpool " ;
2013-02-21 10:25:44 +01:00
else if ( lv_is_origin ( lv ) | | lv_is_external_origin ( lv ) )
2013-02-01 11:09:34 +01:00
return " real " ;
return NULL ;
}
2010-10-12 16:12:33 +00:00
int lv_kernel_minor ( const struct logical_volume * lv )
{
struct lvinfo info ;
if ( lv_info ( lv - > vg - > cmd , lv , 0 , & info , 0 , 0 ) & & info . exists )
return info . minor ;
return - 1 ;
}
int lv_kernel_major ( const struct logical_volume * lv )
{
struct lvinfo info ;
if ( lv_info ( lv - > vg - > cmd , lv , 0 , & info , 0 , 0 ) & & info . exists )
return info . major ;
return - 1 ;
}
2010-10-12 16:12:18 +00:00
char * lv_convert_lv_dup ( struct dm_pool * mem , const struct logical_volume * lv )
{
struct lv_segment * seg ;
if ( lv - > status & ( CONVERTING | MIRRORED ) ) {
seg = first_seg ( lv ) ;
/* Temporary mirror is always area_num == 0 */
if ( seg_type ( seg , 0 ) = = AREA_LV & &
is_temporary_mirror_layer ( seg_lv ( seg , 0 ) ) )
return dm_pool_strdup ( mem , seg_lv ( seg , 0 ) - > name ) ;
}
return NULL ;
}
2010-10-12 16:12:02 +00:00
char * lv_move_pv_dup ( struct dm_pool * mem , const struct logical_volume * lv )
{
struct lv_segment * seg ;
dm_list_iterate_items ( seg , & lv - > segments ) {
if ( seg - > status & PVMOVE )
return dm_pool_strdup ( mem , dev_name ( seg_dev ( seg , 0 ) ) ) ;
}
return NULL ;
}
2010-10-12 16:11:48 +00:00
uint64_t lv_origin_size ( const struct logical_volume * lv )
{
if ( lv_is_cow ( lv ) )
return ( uint64_t ) find_cow ( lv ) - > len * lv - > vg - > extent_size ;
2013-01-15 15:16:16 +01:00
if ( lv_is_thin_volume ( lv ) & & first_seg ( lv ) - > external_lv )
return first_seg ( lv ) - > external_lv - > size ;
2010-10-12 16:11:48 +00:00
if ( lv_is_origin ( lv ) )
return lv - > size ;
return 0 ;
}
2012-01-19 15:34:32 +00:00
uint64_t lv_metadata_size ( const struct logical_volume * lv )
{
return lv_is_thin_pool ( lv ) ? first_seg ( lv ) - > metadata_lv - > size : 0 ;
}
2010-10-12 16:11:34 +00:00
char * lv_path_dup ( struct dm_pool * mem , const struct logical_volume * lv )
{
char * repstr ;
size_t len ;
2011-03-09 12:44:42 +00:00
if ( ! * lv - > vg - > name )
return dm_pool_strdup ( mem , " " ) ;
2010-10-12 16:11:34 +00:00
len = strlen ( lv - > vg - > cmd - > dev_dir ) + strlen ( lv - > vg - > name ) +
strlen ( lv - > name ) + 2 ;
if ( ! ( repstr = dm_pool_zalloc ( mem , len ) ) ) {
log_error ( " dm_pool_alloc failed " ) ;
return 0 ;
}
if ( dm_snprintf ( repstr , len , " %s%s/%s " ,
lv - > vg - > cmd - > dev_dir , lv - > vg - > name , lv - > name ) < 0 ) {
log_error ( " lvpath snprintf failed " ) ;
return 0 ;
}
2011-03-09 12:44:42 +00:00
2010-10-12 16:11:34 +00:00
return repstr ;
}
2010-09-30 13:16:55 +00:00
2010-09-30 14:07:47 +00:00
char * lv_uuid_dup ( const struct logical_volume * lv )
{
return id_format_and_copy ( lv - > vg - > vgmem , & lv - > lvid . id [ 1 ] ) ;
}
2010-09-30 14:08:19 +00:00
char * lv_tags_dup ( const struct logical_volume * lv )
{
return tags_format_and_copy ( lv - > vg - > vgmem , & lv - > tags ) ;
}
2010-09-30 13:16:55 +00:00
uint64_t lv_size ( const struct logical_volume * lv )
{
return lv - > size ;
}
2010-09-30 13:52:55 +00:00
static int _lv_mimage_in_sync ( const struct logical_volume * lv )
{
2010-11-30 11:53:31 +00:00
percent_t percent ;
2010-09-30 13:52:55 +00:00
struct lv_segment * mirror_seg = find_mirror_seg ( first_seg ( lv ) ) ;
if ( ! ( lv - > status & MIRROR_IMAGE ) | | ! mirror_seg )
return_0 ;
if ( ! lv_mirror_percent ( lv - > vg - > cmd , mirror_seg - > lv , 0 , & percent ,
2010-11-30 11:53:31 +00:00
NULL ) )
2010-09-30 13:52:55 +00:00
return_0 ;
2010-11-30 11:53:31 +00:00
return ( percent = = PERCENT_100 ) ? 1 : 0 ;
2010-09-30 13:52:55 +00:00
}
2011-09-23 15:17:54 +00:00
static int _lv_raid_image_in_sync ( const struct logical_volume * lv )
{
RAID: Improve 'lvs' attribute reporting of RAID LVs and sub-LVs
There are currently a few issues with the reporting done on RAID LVs and
sub-LVs. The most concerning is that 'lvs' does not always report the
correct failure status of individual RAID sub-LVs (devices). This can
occur when a device fails and is restored after the failure has been
detected by the kernel. In this case, 'lvs' would report all devices are
fine because it can read the labels on each device just fine.
Example:
[root@bp-01 lvm2]# lvs -a -o name,vg_name,attr,copy_percent,devices vg
LV VG Attr Cpy%Sync Devices
lv vg rwi-a-r-- 100.00 lv_rimage_0(0),lv_rimage_1(0)
[lv_rimage_0] vg iwi-aor-- /dev/sda1(1)
[lv_rimage_1] vg iwi-aor-- /dev/sdb1(1)
[lv_rmeta_0] vg ewi-aor-- /dev/sda1(0)
[lv_rmeta_1] vg ewi-aor-- /dev/sdb1(0)
However, 'dmsetup status' on the device tells us a different story:
[root@bp-01 lvm2]# dmsetup status vg-lv
0 1024000 raid raid1 2 DA 1024000/1024000
In this case, we must also be sure to check the RAID LVs kernel status
in order to get the proper information. Here is an example of the correct
output that is displayed after this patch is applied:
[root@bp-01 lvm2]# lvs -a -o name,vg_name,attr,copy_percent,devices vg
LV VG Attr Cpy%Sync Devices
lv vg rwi-a-r-p 100.00 lv_rimage_0(0),lv_rimage_1(0)
[lv_rimage_0] vg iwi-aor-p /dev/sda1(1)
[lv_rimage_1] vg iwi-aor-- /dev/sdb1(1)
[lv_rmeta_0] vg ewi-aor-p /dev/sda1(0)
[lv_rmeta_1] vg ewi-aor-- /dev/sdb1(0)
The other case where 'lvs' gives incomplete or improper output is when a
device is replaced or added to a RAID LV. It should display that the RAID
LV is in the process of sync'ing and that the new device is the only one
that is not-in-sync - as indicated by a leading 'I' in the Attr column.
(Remember that 'i' indicates an (i)mage that is in-sync and 'I' indicates
an (I)mage that is not in sync.) Here's an example of the old incorrect
behaviour:
[root@bp-01 lvm2]# lvs -a -o name,vg_name,attr,copy_percent,devices vg
LV VG Attr Cpy%Sync Devices
lv vg rwi-a-r-- 100.00 lv_rimage_0(0),lv_rimage_1(0)
[lv_rimage_0] vg iwi-aor-- /dev/sda1(1)
[lv_rimage_1] vg iwi-aor-- /dev/sdb1(1)
[lv_rmeta_0] vg ewi-aor-- /dev/sda1(0)
[lv_rmeta_1] vg ewi-aor-- /dev/sdb1(0)
[root@bp-01 lvm2]# lvconvert -m +1 vg/lv; lvs -a -o name,vg_name,attr,copy_percent,devices vg
LV VG Attr Cpy%Sync Devices
lv vg rwi-a-r-- 0.00 lv_rimage_0(0),lv_rimage_1(0),lv_rimage_2(0)
[lv_rimage_0] vg Iwi-aor-- /dev/sda1(1)
[lv_rimage_1] vg Iwi-aor-- /dev/sdb1(1)
[lv_rimage_2] vg Iwi-aor-- /dev/sdc1(1)
[lv_rmeta_0] vg ewi-aor-- /dev/sda1(0)
[lv_rmeta_1] vg ewi-aor-- /dev/sdb1(0)
[lv_rmeta_2] vg ewi-aor-- /dev/sdc1(0) ** Note that all the images currently are marked as 'I' even though it is
only the last device that has been added that should be marked.
Here is an example of the correct output after this patch is applied:
[root@bp-01 lvm2]# lvs -a -o name,vg_name,attr,copy_percent,devices vg
LV VG Attr Cpy%Sync Devices
lv vg rwi-a-r-- 100.00 lv_rimage_0(0),lv_rimage_1(0)
[lv_rimage_0] vg iwi-aor-- /dev/sda1(1)
[lv_rimage_1] vg iwi-aor-- /dev/sdb1(1)
[lv_rmeta_0] vg ewi-aor-- /dev/sda1(0)
[lv_rmeta_1] vg ewi-aor-- /dev/sdb1(0)
[root@bp-01 lvm2]# lvconvert -m +1 vg/lv; lvs -a -o name,vg_name,attr,copy_percent,devices vg
LV VG Attr Cpy%Sync Devices
lv vg rwi-a-r-- 0.00 lv_rimage_0(0),lv_rimage_1(0),lv_rimage_2(0)
[lv_rimage_0] vg iwi-aor-- /dev/sda1(1)
[lv_rimage_1] vg iwi-aor-- /dev/sdb1(1)
[lv_rimage_2] vg Iwi-aor-- /dev/sdc1(1)
[lv_rmeta_0] vg ewi-aor-- /dev/sda1(0)
[lv_rmeta_1] vg ewi-aor-- /dev/sdb1(0)
[lv_rmeta_2] vg ewi-aor-- /dev/sdc1(0)
** Note only the last image is marked with an 'I'. This is correct and we can
tell that it isn't the whole array that is sync'ing, but just the new
device.
It also works under snapshots...
[root@bp-01 lvm2]# lvs -a -o name,vg_name,attr,copy_percent,devices vg
LV VG Attr Cpy%Sync Devices
lv vg owi-a-r-p 33.47 lv_rimage_0(0),lv_rimage_1(0),lv_rimage_2(0)
[lv_rimage_0] vg iwi-aor-- /dev/sda1(1)
[lv_rimage_1] vg Iwi-aor-p /dev/sdb1(1)
[lv_rimage_2] vg Iwi-aor-- /dev/sdc1(1)
[lv_rmeta_0] vg ewi-aor-- /dev/sda1(0)
[lv_rmeta_1] vg ewi-aor-p /dev/sdb1(0)
[lv_rmeta_2] vg ewi-aor-- /dev/sdc1(0)
snap vg swi-a-s-- /dev/sda1(51201)
2013-02-01 11:33:54 -06:00
unsigned s ;
2011-09-23 15:17:54 +00:00
percent_t percent ;
RAID: Improve 'lvs' attribute reporting of RAID LVs and sub-LVs
There are currently a few issues with the reporting done on RAID LVs and
sub-LVs. The most concerning is that 'lvs' does not always report the
correct failure status of individual RAID sub-LVs (devices). This can
occur when a device fails and is restored after the failure has been
detected by the kernel. In this case, 'lvs' would report all devices are
fine because it can read the labels on each device just fine.
Example:
[root@bp-01 lvm2]# lvs -a -o name,vg_name,attr,copy_percent,devices vg
LV VG Attr Cpy%Sync Devices
lv vg rwi-a-r-- 100.00 lv_rimage_0(0),lv_rimage_1(0)
[lv_rimage_0] vg iwi-aor-- /dev/sda1(1)
[lv_rimage_1] vg iwi-aor-- /dev/sdb1(1)
[lv_rmeta_0] vg ewi-aor-- /dev/sda1(0)
[lv_rmeta_1] vg ewi-aor-- /dev/sdb1(0)
However, 'dmsetup status' on the device tells us a different story:
[root@bp-01 lvm2]# dmsetup status vg-lv
0 1024000 raid raid1 2 DA 1024000/1024000
In this case, we must also be sure to check the RAID LVs kernel status
in order to get the proper information. Here is an example of the correct
output that is displayed after this patch is applied:
[root@bp-01 lvm2]# lvs -a -o name,vg_name,attr,copy_percent,devices vg
LV VG Attr Cpy%Sync Devices
lv vg rwi-a-r-p 100.00 lv_rimage_0(0),lv_rimage_1(0)
[lv_rimage_0] vg iwi-aor-p /dev/sda1(1)
[lv_rimage_1] vg iwi-aor-- /dev/sdb1(1)
[lv_rmeta_0] vg ewi-aor-p /dev/sda1(0)
[lv_rmeta_1] vg ewi-aor-- /dev/sdb1(0)
The other case where 'lvs' gives incomplete or improper output is when a
device is replaced or added to a RAID LV. It should display that the RAID
LV is in the process of sync'ing and that the new device is the only one
that is not-in-sync - as indicated by a leading 'I' in the Attr column.
(Remember that 'i' indicates an (i)mage that is in-sync and 'I' indicates
an (I)mage that is not in sync.) Here's an example of the old incorrect
behaviour:
[root@bp-01 lvm2]# lvs -a -o name,vg_name,attr,copy_percent,devices vg
LV VG Attr Cpy%Sync Devices
lv vg rwi-a-r-- 100.00 lv_rimage_0(0),lv_rimage_1(0)
[lv_rimage_0] vg iwi-aor-- /dev/sda1(1)
[lv_rimage_1] vg iwi-aor-- /dev/sdb1(1)
[lv_rmeta_0] vg ewi-aor-- /dev/sda1(0)
[lv_rmeta_1] vg ewi-aor-- /dev/sdb1(0)
[root@bp-01 lvm2]# lvconvert -m +1 vg/lv; lvs -a -o name,vg_name,attr,copy_percent,devices vg
LV VG Attr Cpy%Sync Devices
lv vg rwi-a-r-- 0.00 lv_rimage_0(0),lv_rimage_1(0),lv_rimage_2(0)
[lv_rimage_0] vg Iwi-aor-- /dev/sda1(1)
[lv_rimage_1] vg Iwi-aor-- /dev/sdb1(1)
[lv_rimage_2] vg Iwi-aor-- /dev/sdc1(1)
[lv_rmeta_0] vg ewi-aor-- /dev/sda1(0)
[lv_rmeta_1] vg ewi-aor-- /dev/sdb1(0)
[lv_rmeta_2] vg ewi-aor-- /dev/sdc1(0) ** Note that all the images currently are marked as 'I' even though it is
only the last device that has been added that should be marked.
Here is an example of the correct output after this patch is applied:
[root@bp-01 lvm2]# lvs -a -o name,vg_name,attr,copy_percent,devices vg
LV VG Attr Cpy%Sync Devices
lv vg rwi-a-r-- 100.00 lv_rimage_0(0),lv_rimage_1(0)
[lv_rimage_0] vg iwi-aor-- /dev/sda1(1)
[lv_rimage_1] vg iwi-aor-- /dev/sdb1(1)
[lv_rmeta_0] vg ewi-aor-- /dev/sda1(0)
[lv_rmeta_1] vg ewi-aor-- /dev/sdb1(0)
[root@bp-01 lvm2]# lvconvert -m +1 vg/lv; lvs -a -o name,vg_name,attr,copy_percent,devices vg
LV VG Attr Cpy%Sync Devices
lv vg rwi-a-r-- 0.00 lv_rimage_0(0),lv_rimage_1(0),lv_rimage_2(0)
[lv_rimage_0] vg iwi-aor-- /dev/sda1(1)
[lv_rimage_1] vg iwi-aor-- /dev/sdb1(1)
[lv_rimage_2] vg Iwi-aor-- /dev/sdc1(1)
[lv_rmeta_0] vg ewi-aor-- /dev/sda1(0)
[lv_rmeta_1] vg ewi-aor-- /dev/sdb1(0)
[lv_rmeta_2] vg ewi-aor-- /dev/sdc1(0)
** Note only the last image is marked with an 'I'. This is correct and we can
tell that it isn't the whole array that is sync'ing, but just the new
device.
It also works under snapshots...
[root@bp-01 lvm2]# lvs -a -o name,vg_name,attr,copy_percent,devices vg
LV VG Attr Cpy%Sync Devices
lv vg owi-a-r-p 33.47 lv_rimage_0(0),lv_rimage_1(0),lv_rimage_2(0)
[lv_rimage_0] vg iwi-aor-- /dev/sda1(1)
[lv_rimage_1] vg Iwi-aor-p /dev/sdb1(1)
[lv_rimage_2] vg Iwi-aor-- /dev/sdc1(1)
[lv_rmeta_0] vg ewi-aor-- /dev/sda1(0)
[lv_rmeta_1] vg ewi-aor-p /dev/sdb1(0)
[lv_rmeta_2] vg ewi-aor-- /dev/sdc1(0)
snap vg swi-a-s-- /dev/sda1(51201)
2013-02-01 11:33:54 -06:00
char * raid_health ;
2011-09-23 15:17:54 +00:00
struct lv_segment * raid_seg ;
2013-05-16 10:36:56 -05:00
/*
* If the LV is not active locally ,
* it doesn ' t make sense to check status
*/
if ( ! lv_is_active_locally ( lv ) )
RAID: Improve 'lvs' attribute reporting of RAID LVs and sub-LVs
There are currently a few issues with the reporting done on RAID LVs and
sub-LVs. The most concerning is that 'lvs' does not always report the
correct failure status of individual RAID sub-LVs (devices). This can
occur when a device fails and is restored after the failure has been
detected by the kernel. In this case, 'lvs' would report all devices are
fine because it can read the labels on each device just fine.
Example:
[root@bp-01 lvm2]# lvs -a -o name,vg_name,attr,copy_percent,devices vg
LV VG Attr Cpy%Sync Devices
lv vg rwi-a-r-- 100.00 lv_rimage_0(0),lv_rimage_1(0)
[lv_rimage_0] vg iwi-aor-- /dev/sda1(1)
[lv_rimage_1] vg iwi-aor-- /dev/sdb1(1)
[lv_rmeta_0] vg ewi-aor-- /dev/sda1(0)
[lv_rmeta_1] vg ewi-aor-- /dev/sdb1(0)
However, 'dmsetup status' on the device tells us a different story:
[root@bp-01 lvm2]# dmsetup status vg-lv
0 1024000 raid raid1 2 DA 1024000/1024000
In this case, we must also be sure to check the RAID LVs kernel status
in order to get the proper information. Here is an example of the correct
output that is displayed after this patch is applied:
[root@bp-01 lvm2]# lvs -a -o name,vg_name,attr,copy_percent,devices vg
LV VG Attr Cpy%Sync Devices
lv vg rwi-a-r-p 100.00 lv_rimage_0(0),lv_rimage_1(0)
[lv_rimage_0] vg iwi-aor-p /dev/sda1(1)
[lv_rimage_1] vg iwi-aor-- /dev/sdb1(1)
[lv_rmeta_0] vg ewi-aor-p /dev/sda1(0)
[lv_rmeta_1] vg ewi-aor-- /dev/sdb1(0)
The other case where 'lvs' gives incomplete or improper output is when a
device is replaced or added to a RAID LV. It should display that the RAID
LV is in the process of sync'ing and that the new device is the only one
that is not-in-sync - as indicated by a leading 'I' in the Attr column.
(Remember that 'i' indicates an (i)mage that is in-sync and 'I' indicates
an (I)mage that is not in sync.) Here's an example of the old incorrect
behaviour:
[root@bp-01 lvm2]# lvs -a -o name,vg_name,attr,copy_percent,devices vg
LV VG Attr Cpy%Sync Devices
lv vg rwi-a-r-- 100.00 lv_rimage_0(0),lv_rimage_1(0)
[lv_rimage_0] vg iwi-aor-- /dev/sda1(1)
[lv_rimage_1] vg iwi-aor-- /dev/sdb1(1)
[lv_rmeta_0] vg ewi-aor-- /dev/sda1(0)
[lv_rmeta_1] vg ewi-aor-- /dev/sdb1(0)
[root@bp-01 lvm2]# lvconvert -m +1 vg/lv; lvs -a -o name,vg_name,attr,copy_percent,devices vg
LV VG Attr Cpy%Sync Devices
lv vg rwi-a-r-- 0.00 lv_rimage_0(0),lv_rimage_1(0),lv_rimage_2(0)
[lv_rimage_0] vg Iwi-aor-- /dev/sda1(1)
[lv_rimage_1] vg Iwi-aor-- /dev/sdb1(1)
[lv_rimage_2] vg Iwi-aor-- /dev/sdc1(1)
[lv_rmeta_0] vg ewi-aor-- /dev/sda1(0)
[lv_rmeta_1] vg ewi-aor-- /dev/sdb1(0)
[lv_rmeta_2] vg ewi-aor-- /dev/sdc1(0) ** Note that all the images currently are marked as 'I' even though it is
only the last device that has been added that should be marked.
Here is an example of the correct output after this patch is applied:
[root@bp-01 lvm2]# lvs -a -o name,vg_name,attr,copy_percent,devices vg
LV VG Attr Cpy%Sync Devices
lv vg rwi-a-r-- 100.00 lv_rimage_0(0),lv_rimage_1(0)
[lv_rimage_0] vg iwi-aor-- /dev/sda1(1)
[lv_rimage_1] vg iwi-aor-- /dev/sdb1(1)
[lv_rmeta_0] vg ewi-aor-- /dev/sda1(0)
[lv_rmeta_1] vg ewi-aor-- /dev/sdb1(0)
[root@bp-01 lvm2]# lvconvert -m +1 vg/lv; lvs -a -o name,vg_name,attr,copy_percent,devices vg
LV VG Attr Cpy%Sync Devices
lv vg rwi-a-r-- 0.00 lv_rimage_0(0),lv_rimage_1(0),lv_rimage_2(0)
[lv_rimage_0] vg iwi-aor-- /dev/sda1(1)
[lv_rimage_1] vg iwi-aor-- /dev/sdb1(1)
[lv_rimage_2] vg Iwi-aor-- /dev/sdc1(1)
[lv_rmeta_0] vg ewi-aor-- /dev/sda1(0)
[lv_rmeta_1] vg ewi-aor-- /dev/sdb1(0)
[lv_rmeta_2] vg ewi-aor-- /dev/sdc1(0)
** Note only the last image is marked with an 'I'. This is correct and we can
tell that it isn't the whole array that is sync'ing, but just the new
device.
It also works under snapshots...
[root@bp-01 lvm2]# lvs -a -o name,vg_name,attr,copy_percent,devices vg
LV VG Attr Cpy%Sync Devices
lv vg owi-a-r-p 33.47 lv_rimage_0(0),lv_rimage_1(0),lv_rimage_2(0)
[lv_rimage_0] vg iwi-aor-- /dev/sda1(1)
[lv_rimage_1] vg Iwi-aor-p /dev/sdb1(1)
[lv_rimage_2] vg Iwi-aor-- /dev/sdc1(1)
[lv_rmeta_0] vg ewi-aor-- /dev/sda1(0)
[lv_rmeta_1] vg ewi-aor-p /dev/sdb1(0)
[lv_rmeta_2] vg ewi-aor-- /dev/sdc1(0)
snap vg swi-a-s-- /dev/sda1(51201)
2013-02-01 11:33:54 -06:00
return 0 ; /* Assume not in-sync */
2011-09-23 15:17:54 +00:00
if ( ! ( lv - > status & RAID_IMAGE ) ) {
log_error ( INTERNAL_ERROR " %s is not a RAID image " , lv - > name ) ;
return 0 ;
}
raid_seg = get_only_segment_using_this_lv ( first_seg ( lv ) - > lv ) ;
if ( ! raid_seg ) {
log_error ( " Failed to find RAID segment for %s " , lv - > name ) ;
return 0 ;
}
if ( ! seg_is_raid ( raid_seg ) ) {
log_error ( " %s on %s is not a RAID segment " ,
raid_seg - > lv - > name , lv - > name ) ;
return 0 ;
}
if ( ! lv_raid_percent ( raid_seg - > lv , & percent ) )
return_0 ;
if ( percent = = PERCENT_100 )
return 1 ;
RAID: Improve 'lvs' attribute reporting of RAID LVs and sub-LVs
There are currently a few issues with the reporting done on RAID LVs and
sub-LVs. The most concerning is that 'lvs' does not always report the
correct failure status of individual RAID sub-LVs (devices). This can
occur when a device fails and is restored after the failure has been
detected by the kernel. In this case, 'lvs' would report all devices are
fine because it can read the labels on each device just fine.
Example:
[root@bp-01 lvm2]# lvs -a -o name,vg_name,attr,copy_percent,devices vg
LV VG Attr Cpy%Sync Devices
lv vg rwi-a-r-- 100.00 lv_rimage_0(0),lv_rimage_1(0)
[lv_rimage_0] vg iwi-aor-- /dev/sda1(1)
[lv_rimage_1] vg iwi-aor-- /dev/sdb1(1)
[lv_rmeta_0] vg ewi-aor-- /dev/sda1(0)
[lv_rmeta_1] vg ewi-aor-- /dev/sdb1(0)
However, 'dmsetup status' on the device tells us a different story:
[root@bp-01 lvm2]# dmsetup status vg-lv
0 1024000 raid raid1 2 DA 1024000/1024000
In this case, we must also be sure to check the RAID LVs kernel status
in order to get the proper information. Here is an example of the correct
output that is displayed after this patch is applied:
[root@bp-01 lvm2]# lvs -a -o name,vg_name,attr,copy_percent,devices vg
LV VG Attr Cpy%Sync Devices
lv vg rwi-a-r-p 100.00 lv_rimage_0(0),lv_rimage_1(0)
[lv_rimage_0] vg iwi-aor-p /dev/sda1(1)
[lv_rimage_1] vg iwi-aor-- /dev/sdb1(1)
[lv_rmeta_0] vg ewi-aor-p /dev/sda1(0)
[lv_rmeta_1] vg ewi-aor-- /dev/sdb1(0)
The other case where 'lvs' gives incomplete or improper output is when a
device is replaced or added to a RAID LV. It should display that the RAID
LV is in the process of sync'ing and that the new device is the only one
that is not-in-sync - as indicated by a leading 'I' in the Attr column.
(Remember that 'i' indicates an (i)mage that is in-sync and 'I' indicates
an (I)mage that is not in sync.) Here's an example of the old incorrect
behaviour:
[root@bp-01 lvm2]# lvs -a -o name,vg_name,attr,copy_percent,devices vg
LV VG Attr Cpy%Sync Devices
lv vg rwi-a-r-- 100.00 lv_rimage_0(0),lv_rimage_1(0)
[lv_rimage_0] vg iwi-aor-- /dev/sda1(1)
[lv_rimage_1] vg iwi-aor-- /dev/sdb1(1)
[lv_rmeta_0] vg ewi-aor-- /dev/sda1(0)
[lv_rmeta_1] vg ewi-aor-- /dev/sdb1(0)
[root@bp-01 lvm2]# lvconvert -m +1 vg/lv; lvs -a -o name,vg_name,attr,copy_percent,devices vg
LV VG Attr Cpy%Sync Devices
lv vg rwi-a-r-- 0.00 lv_rimage_0(0),lv_rimage_1(0),lv_rimage_2(0)
[lv_rimage_0] vg Iwi-aor-- /dev/sda1(1)
[lv_rimage_1] vg Iwi-aor-- /dev/sdb1(1)
[lv_rimage_2] vg Iwi-aor-- /dev/sdc1(1)
[lv_rmeta_0] vg ewi-aor-- /dev/sda1(0)
[lv_rmeta_1] vg ewi-aor-- /dev/sdb1(0)
[lv_rmeta_2] vg ewi-aor-- /dev/sdc1(0) ** Note that all the images currently are marked as 'I' even though it is
only the last device that has been added that should be marked.
Here is an example of the correct output after this patch is applied:
[root@bp-01 lvm2]# lvs -a -o name,vg_name,attr,copy_percent,devices vg
LV VG Attr Cpy%Sync Devices
lv vg rwi-a-r-- 100.00 lv_rimage_0(0),lv_rimage_1(0)
[lv_rimage_0] vg iwi-aor-- /dev/sda1(1)
[lv_rimage_1] vg iwi-aor-- /dev/sdb1(1)
[lv_rmeta_0] vg ewi-aor-- /dev/sda1(0)
[lv_rmeta_1] vg ewi-aor-- /dev/sdb1(0)
[root@bp-01 lvm2]# lvconvert -m +1 vg/lv; lvs -a -o name,vg_name,attr,copy_percent,devices vg
LV VG Attr Cpy%Sync Devices
lv vg rwi-a-r-- 0.00 lv_rimage_0(0),lv_rimage_1(0),lv_rimage_2(0)
[lv_rimage_0] vg iwi-aor-- /dev/sda1(1)
[lv_rimage_1] vg iwi-aor-- /dev/sdb1(1)
[lv_rimage_2] vg Iwi-aor-- /dev/sdc1(1)
[lv_rmeta_0] vg ewi-aor-- /dev/sda1(0)
[lv_rmeta_1] vg ewi-aor-- /dev/sdb1(0)
[lv_rmeta_2] vg ewi-aor-- /dev/sdc1(0)
** Note only the last image is marked with an 'I'. This is correct and we can
tell that it isn't the whole array that is sync'ing, but just the new
device.
It also works under snapshots...
[root@bp-01 lvm2]# lvs -a -o name,vg_name,attr,copy_percent,devices vg
LV VG Attr Cpy%Sync Devices
lv vg owi-a-r-p 33.47 lv_rimage_0(0),lv_rimage_1(0),lv_rimage_2(0)
[lv_rimage_0] vg iwi-aor-- /dev/sda1(1)
[lv_rimage_1] vg Iwi-aor-p /dev/sdb1(1)
[lv_rimage_2] vg Iwi-aor-- /dev/sdc1(1)
[lv_rmeta_0] vg ewi-aor-- /dev/sda1(0)
[lv_rmeta_1] vg ewi-aor-p /dev/sdb1(0)
[lv_rmeta_2] vg ewi-aor-- /dev/sdc1(0)
snap vg swi-a-s-- /dev/sda1(51201)
2013-02-01 11:33:54 -06:00
/* Find out which sub-LV this is. */
for ( s = 0 ; s < raid_seg - > area_count ; s + + )
if ( seg_lv ( raid_seg , s ) = = lv )
break ;
if ( s = = raid_seg - > area_count ) {
log_error ( INTERNAL_ERROR
" sub-LV %s was not found in raid segment " ,
lv - > name ) ;
return 0 ;
}
if ( ! lv_raid_dev_health ( raid_seg - > lv , & raid_health ) )
return_0 ;
if ( raid_health [ s ] = = ' A ' )
return 1 ;
2011-09-23 15:17:54 +00:00
return 0 ;
}
RAID: Improve 'lvs' attribute reporting of RAID LVs and sub-LVs
There are currently a few issues with the reporting done on RAID LVs and
sub-LVs. The most concerning is that 'lvs' does not always report the
correct failure status of individual RAID sub-LVs (devices). This can
occur when a device fails and is restored after the failure has been
detected by the kernel. In this case, 'lvs' would report all devices are
fine because it can read the labels on each device just fine.
Example:
[root@bp-01 lvm2]# lvs -a -o name,vg_name,attr,copy_percent,devices vg
LV VG Attr Cpy%Sync Devices
lv vg rwi-a-r-- 100.00 lv_rimage_0(0),lv_rimage_1(0)
[lv_rimage_0] vg iwi-aor-- /dev/sda1(1)
[lv_rimage_1] vg iwi-aor-- /dev/sdb1(1)
[lv_rmeta_0] vg ewi-aor-- /dev/sda1(0)
[lv_rmeta_1] vg ewi-aor-- /dev/sdb1(0)
However, 'dmsetup status' on the device tells us a different story:
[root@bp-01 lvm2]# dmsetup status vg-lv
0 1024000 raid raid1 2 DA 1024000/1024000
In this case, we must also be sure to check the RAID LVs kernel status
in order to get the proper information. Here is an example of the correct
output that is displayed after this patch is applied:
[root@bp-01 lvm2]# lvs -a -o name,vg_name,attr,copy_percent,devices vg
LV VG Attr Cpy%Sync Devices
lv vg rwi-a-r-p 100.00 lv_rimage_0(0),lv_rimage_1(0)
[lv_rimage_0] vg iwi-aor-p /dev/sda1(1)
[lv_rimage_1] vg iwi-aor-- /dev/sdb1(1)
[lv_rmeta_0] vg ewi-aor-p /dev/sda1(0)
[lv_rmeta_1] vg ewi-aor-- /dev/sdb1(0)
The other case where 'lvs' gives incomplete or improper output is when a
device is replaced or added to a RAID LV. It should display that the RAID
LV is in the process of sync'ing and that the new device is the only one
that is not-in-sync - as indicated by a leading 'I' in the Attr column.
(Remember that 'i' indicates an (i)mage that is in-sync and 'I' indicates
an (I)mage that is not in sync.) Here's an example of the old incorrect
behaviour:
[root@bp-01 lvm2]# lvs -a -o name,vg_name,attr,copy_percent,devices vg
LV VG Attr Cpy%Sync Devices
lv vg rwi-a-r-- 100.00 lv_rimage_0(0),lv_rimage_1(0)
[lv_rimage_0] vg iwi-aor-- /dev/sda1(1)
[lv_rimage_1] vg iwi-aor-- /dev/sdb1(1)
[lv_rmeta_0] vg ewi-aor-- /dev/sda1(0)
[lv_rmeta_1] vg ewi-aor-- /dev/sdb1(0)
[root@bp-01 lvm2]# lvconvert -m +1 vg/lv; lvs -a -o name,vg_name,attr,copy_percent,devices vg
LV VG Attr Cpy%Sync Devices
lv vg rwi-a-r-- 0.00 lv_rimage_0(0),lv_rimage_1(0),lv_rimage_2(0)
[lv_rimage_0] vg Iwi-aor-- /dev/sda1(1)
[lv_rimage_1] vg Iwi-aor-- /dev/sdb1(1)
[lv_rimage_2] vg Iwi-aor-- /dev/sdc1(1)
[lv_rmeta_0] vg ewi-aor-- /dev/sda1(0)
[lv_rmeta_1] vg ewi-aor-- /dev/sdb1(0)
[lv_rmeta_2] vg ewi-aor-- /dev/sdc1(0) ** Note that all the images currently are marked as 'I' even though it is
only the last device that has been added that should be marked.
Here is an example of the correct output after this patch is applied:
[root@bp-01 lvm2]# lvs -a -o name,vg_name,attr,copy_percent,devices vg
LV VG Attr Cpy%Sync Devices
lv vg rwi-a-r-- 100.00 lv_rimage_0(0),lv_rimage_1(0)
[lv_rimage_0] vg iwi-aor-- /dev/sda1(1)
[lv_rimage_1] vg iwi-aor-- /dev/sdb1(1)
[lv_rmeta_0] vg ewi-aor-- /dev/sda1(0)
[lv_rmeta_1] vg ewi-aor-- /dev/sdb1(0)
[root@bp-01 lvm2]# lvconvert -m +1 vg/lv; lvs -a -o name,vg_name,attr,copy_percent,devices vg
LV VG Attr Cpy%Sync Devices
lv vg rwi-a-r-- 0.00 lv_rimage_0(0),lv_rimage_1(0),lv_rimage_2(0)
[lv_rimage_0] vg iwi-aor-- /dev/sda1(1)
[lv_rimage_1] vg iwi-aor-- /dev/sdb1(1)
[lv_rimage_2] vg Iwi-aor-- /dev/sdc1(1)
[lv_rmeta_0] vg ewi-aor-- /dev/sda1(0)
[lv_rmeta_1] vg ewi-aor-- /dev/sdb1(0)
[lv_rmeta_2] vg ewi-aor-- /dev/sdc1(0)
** Note only the last image is marked with an 'I'. This is correct and we can
tell that it isn't the whole array that is sync'ing, but just the new
device.
It also works under snapshots...
[root@bp-01 lvm2]# lvs -a -o name,vg_name,attr,copy_percent,devices vg
LV VG Attr Cpy%Sync Devices
lv vg owi-a-r-p 33.47 lv_rimage_0(0),lv_rimage_1(0),lv_rimage_2(0)
[lv_rimage_0] vg iwi-aor-- /dev/sda1(1)
[lv_rimage_1] vg Iwi-aor-p /dev/sdb1(1)
[lv_rimage_2] vg Iwi-aor-- /dev/sdc1(1)
[lv_rmeta_0] vg ewi-aor-- /dev/sda1(0)
[lv_rmeta_1] vg ewi-aor-p /dev/sdb1(0)
[lv_rmeta_2] vg ewi-aor-- /dev/sdc1(0)
snap vg swi-a-s-- /dev/sda1(51201)
2013-02-01 11:33:54 -06:00
/*
* _lv_raid_healthy
* @ lv : A RAID_IMAGE , RAID_META , or RAID logical volume .
*
* Returns : 1 if healthy , 0 if device is not health
*/
static int _lv_raid_healthy ( const struct logical_volume * lv )
{
unsigned s ;
char * raid_health ;
struct lv_segment * raid_seg ;
2013-05-16 10:36:56 -05:00
/*
* If the LV is not active locally ,
* it doesn ' t make sense to check status
*/
if ( ! lv_is_active_locally ( lv ) )
RAID: Improve 'lvs' attribute reporting of RAID LVs and sub-LVs
There are currently a few issues with the reporting done on RAID LVs and
sub-LVs. The most concerning is that 'lvs' does not always report the
correct failure status of individual RAID sub-LVs (devices). This can
occur when a device fails and is restored after the failure has been
detected by the kernel. In this case, 'lvs' would report all devices are
fine because it can read the labels on each device just fine.
Example:
[root@bp-01 lvm2]# lvs -a -o name,vg_name,attr,copy_percent,devices vg
LV VG Attr Cpy%Sync Devices
lv vg rwi-a-r-- 100.00 lv_rimage_0(0),lv_rimage_1(0)
[lv_rimage_0] vg iwi-aor-- /dev/sda1(1)
[lv_rimage_1] vg iwi-aor-- /dev/sdb1(1)
[lv_rmeta_0] vg ewi-aor-- /dev/sda1(0)
[lv_rmeta_1] vg ewi-aor-- /dev/sdb1(0)
However, 'dmsetup status' on the device tells us a different story:
[root@bp-01 lvm2]# dmsetup status vg-lv
0 1024000 raid raid1 2 DA 1024000/1024000
In this case, we must also be sure to check the RAID LVs kernel status
in order to get the proper information. Here is an example of the correct
output that is displayed after this patch is applied:
[root@bp-01 lvm2]# lvs -a -o name,vg_name,attr,copy_percent,devices vg
LV VG Attr Cpy%Sync Devices
lv vg rwi-a-r-p 100.00 lv_rimage_0(0),lv_rimage_1(0)
[lv_rimage_0] vg iwi-aor-p /dev/sda1(1)
[lv_rimage_1] vg iwi-aor-- /dev/sdb1(1)
[lv_rmeta_0] vg ewi-aor-p /dev/sda1(0)
[lv_rmeta_1] vg ewi-aor-- /dev/sdb1(0)
The other case where 'lvs' gives incomplete or improper output is when a
device is replaced or added to a RAID LV. It should display that the RAID
LV is in the process of sync'ing and that the new device is the only one
that is not-in-sync - as indicated by a leading 'I' in the Attr column.
(Remember that 'i' indicates an (i)mage that is in-sync and 'I' indicates
an (I)mage that is not in sync.) Here's an example of the old incorrect
behaviour:
[root@bp-01 lvm2]# lvs -a -o name,vg_name,attr,copy_percent,devices vg
LV VG Attr Cpy%Sync Devices
lv vg rwi-a-r-- 100.00 lv_rimage_0(0),lv_rimage_1(0)
[lv_rimage_0] vg iwi-aor-- /dev/sda1(1)
[lv_rimage_1] vg iwi-aor-- /dev/sdb1(1)
[lv_rmeta_0] vg ewi-aor-- /dev/sda1(0)
[lv_rmeta_1] vg ewi-aor-- /dev/sdb1(0)
[root@bp-01 lvm2]# lvconvert -m +1 vg/lv; lvs -a -o name,vg_name,attr,copy_percent,devices vg
LV VG Attr Cpy%Sync Devices
lv vg rwi-a-r-- 0.00 lv_rimage_0(0),lv_rimage_1(0),lv_rimage_2(0)
[lv_rimage_0] vg Iwi-aor-- /dev/sda1(1)
[lv_rimage_1] vg Iwi-aor-- /dev/sdb1(1)
[lv_rimage_2] vg Iwi-aor-- /dev/sdc1(1)
[lv_rmeta_0] vg ewi-aor-- /dev/sda1(0)
[lv_rmeta_1] vg ewi-aor-- /dev/sdb1(0)
[lv_rmeta_2] vg ewi-aor-- /dev/sdc1(0) ** Note that all the images currently are marked as 'I' even though it is
only the last device that has been added that should be marked.
Here is an example of the correct output after this patch is applied:
[root@bp-01 lvm2]# lvs -a -o name,vg_name,attr,copy_percent,devices vg
LV VG Attr Cpy%Sync Devices
lv vg rwi-a-r-- 100.00 lv_rimage_0(0),lv_rimage_1(0)
[lv_rimage_0] vg iwi-aor-- /dev/sda1(1)
[lv_rimage_1] vg iwi-aor-- /dev/sdb1(1)
[lv_rmeta_0] vg ewi-aor-- /dev/sda1(0)
[lv_rmeta_1] vg ewi-aor-- /dev/sdb1(0)
[root@bp-01 lvm2]# lvconvert -m +1 vg/lv; lvs -a -o name,vg_name,attr,copy_percent,devices vg
LV VG Attr Cpy%Sync Devices
lv vg rwi-a-r-- 0.00 lv_rimage_0(0),lv_rimage_1(0),lv_rimage_2(0)
[lv_rimage_0] vg iwi-aor-- /dev/sda1(1)
[lv_rimage_1] vg iwi-aor-- /dev/sdb1(1)
[lv_rimage_2] vg Iwi-aor-- /dev/sdc1(1)
[lv_rmeta_0] vg ewi-aor-- /dev/sda1(0)
[lv_rmeta_1] vg ewi-aor-- /dev/sdb1(0)
[lv_rmeta_2] vg ewi-aor-- /dev/sdc1(0)
** Note only the last image is marked with an 'I'. This is correct and we can
tell that it isn't the whole array that is sync'ing, but just the new
device.
It also works under snapshots...
[root@bp-01 lvm2]# lvs -a -o name,vg_name,attr,copy_percent,devices vg
LV VG Attr Cpy%Sync Devices
lv vg owi-a-r-p 33.47 lv_rimage_0(0),lv_rimage_1(0),lv_rimage_2(0)
[lv_rimage_0] vg iwi-aor-- /dev/sda1(1)
[lv_rimage_1] vg Iwi-aor-p /dev/sdb1(1)
[lv_rimage_2] vg Iwi-aor-- /dev/sdc1(1)
[lv_rmeta_0] vg ewi-aor-- /dev/sda1(0)
[lv_rmeta_1] vg ewi-aor-p /dev/sdb1(0)
[lv_rmeta_2] vg ewi-aor-- /dev/sdc1(0)
snap vg swi-a-s-- /dev/sda1(51201)
2013-02-01 11:33:54 -06:00
return 1 ; /* assume healthy */
if ( ! lv_is_raid_type ( lv ) ) {
log_error ( INTERNAL_ERROR " %s is not of RAID type " , lv - > name ) ;
return 0 ;
}
if ( lv - > status & RAID )
raid_seg = first_seg ( lv ) ;
else
raid_seg = get_only_segment_using_this_lv ( first_seg ( lv ) - > lv ) ;
if ( ! raid_seg ) {
log_error ( " Failed to find RAID segment for %s " , lv - > name ) ;
return 0 ;
}
if ( ! seg_is_raid ( raid_seg ) ) {
log_error ( " %s on %s is not a RAID segment " ,
raid_seg - > lv - > name , lv - > name ) ;
return 0 ;
}
if ( ! lv_raid_dev_health ( raid_seg - > lv , & raid_health ) )
return_0 ;
if ( lv - > status & RAID ) {
if ( strchr ( raid_health , ' D ' ) )
return 0 ;
else
return 1 ;
}
/* Find out which sub-LV this is. */
for ( s = 0 ; s < raid_seg - > area_count ; s + + )
if ( ( ( lv - > status & RAID_IMAGE ) & & ( seg_lv ( raid_seg , s ) = = lv ) ) | |
( ( lv - > status & RAID_META ) & & ( seg_metalv ( raid_seg , s ) = = lv ) ) )
break ;
if ( s = = raid_seg - > area_count ) {
log_error ( INTERNAL_ERROR
" sub-LV %s was not found in raid segment " ,
lv - > name ) ;
return 0 ;
}
if ( raid_health [ s ] = = ' D ' )
return 0 ;
return 1 ;
}
2010-09-30 13:52:55 +00:00
char * lv_attr_dup ( struct dm_pool * mem , const struct logical_volume * lv )
{
2010-11-30 11:53:31 +00:00
percent_t snap_percent ;
2010-09-30 13:52:55 +00:00
struct lvinfo info ;
2011-09-09 01:15:18 +00:00
struct lv_segment * seg ;
2010-09-30 13:52:55 +00:00
char * repstr ;
2012-09-19 12:49:40 +01:00
if ( ! ( repstr = dm_pool_zalloc ( mem , 10 ) ) ) {
2010-09-30 13:52:55 +00:00
log_error ( " dm_pool_alloc failed " ) ;
return 0 ;
}
/* Blank if this is a "free space" LV. */
if ( ! * lv - > name )
goto out ;
if ( lv - > status & PVMOVE )
repstr [ 0 ] = ' p ' ;
else if ( lv - > status & CONVERTING )
repstr [ 0 ] = ' c ' ;
2011-09-08 20:55:39 +00:00
/* Origin takes precedence over mirror and thin volume */
2013-06-05 13:45:31 +02:00
else if ( lv_is_origin ( lv ) | | lv_is_external_origin ( lv ) )
2010-09-30 14:07:19 +00:00
repstr [ 0 ] = ( lv_is_merging_origin ( lv ) ) ? ' O ' : ' o ' ;
2013-06-05 13:46:38 +02:00
else if ( lv_is_thin_pool_metadata ( lv ) | |
( lv - > status & RAID_META ) )
repstr [ 0 ] = ' e ' ;
2011-09-23 15:17:54 +00:00
else if ( lv - > status & RAID )
repstr [ 0 ] = ( lv - > status & LV_NOTSYNCED ) ? ' R ' : ' r ' ;
2011-09-08 20:55:39 +00:00
else if ( lv - > status & MIRRORED )
2011-03-29 12:51:57 +00:00
repstr [ 0 ] = ( lv - > status & LV_NOTSYNCED ) ? ' M ' : ' m ' ;
2011-09-08 20:55:39 +00:00
else if ( lv_is_thin_volume ( lv ) )
repstr [ 0 ] = ' V ' ;
else if ( lv - > status & VIRTUAL )
repstr [ 0 ] = ' v ' ;
else if ( lv_is_thin_pool ( lv ) )
repstr [ 0 ] = ' t ' ;
else if ( lv_is_thin_pool_data ( lv ) )
repstr [ 0 ] = ' T ' ;
else if ( lv - > status & MIRROR_IMAGE )
2010-09-30 14:07:19 +00:00
repstr [ 0 ] = ( _lv_mimage_in_sync ( lv ) ) ? ' i ' : ' I ' ;
2011-09-23 15:17:54 +00:00
else if ( lv - > status & RAID_IMAGE )
repstr [ 0 ] = ( _lv_raid_image_in_sync ( lv ) ) ? ' i ' : ' I ' ;
2010-09-30 13:52:55 +00:00
else if ( lv - > status & MIRROR_LOG )
repstr [ 0 ] = ' l ' ;
2013-06-11 13:45:36 +02:00
else if ( lv_is_cow ( lv ) )
2010-09-30 14:07:19 +00:00
repstr [ 0 ] = ( lv_is_merging_cow ( lv ) ) ? ' S ' : ' s ' ;
2013-06-11 13:45:36 +02:00
else
2010-09-30 13:52:55 +00:00
repstr [ 0 ] = ' - ' ;
if ( lv - > status & PVMOVE )
repstr [ 1 ] = ' - ' ;
else if ( lv - > status & LVM_WRITE )
repstr [ 1 ] = ' w ' ;
else if ( lv - > status & LVM_READ )
repstr [ 1 ] = ' r ' ;
else
repstr [ 1 ] = ' - ' ;
repstr [ 2 ] = alloc_policy_char ( lv - > alloc ) ;
if ( lv - > status & LOCKED )
repstr [ 2 ] = toupper ( repstr [ 2 ] ) ;
2010-09-30 14:07:19 +00:00
repstr [ 3 ] = ( lv - > status & FIXED_MINOR ) ? ' m ' : ' - ' ;
2010-09-30 13:52:55 +00:00
if ( lv_info ( lv - > vg - > cmd , lv , 0 , & info , 1 , 0 ) & & info . exists ) {
if ( info . suspended )
repstr [ 4 ] = ' s ' ; /* Suspended */
else if ( info . live_table )
repstr [ 4 ] = ' a ' ; /* Active */
else if ( info . inactive_table )
repstr [ 4 ] = ' i ' ; /* Inactive with table */
else
repstr [ 4 ] = ' d ' ; /* Inactive without table */
/* Snapshot dropped? */
2012-01-20 22:03:03 +00:00
if ( info . live_table & & lv_is_cow ( lv ) ) {
if ( ! lv_snapshot_percent ( lv , & snap_percent ) | |
snap_percent = = PERCENT_INVALID ) {
if ( info . suspended )
repstr [ 4 ] = ' S ' ; /* Susp Inv snapshot */
else
repstr [ 4 ] = ' I ' ; /* Invalid snapshot */
}
else if ( snap_percent = = PERCENT_MERGE_FAILED ) {
if ( info . suspended )
repstr [ 4 ] = ' M ' ; /* Susp snapshot merge failed */
else
repstr [ 4 ] = ' m ' ; /* snapshot merge failed */
}
2010-09-30 13:52:55 +00:00
}
2012-01-12 16:58:43 +00:00
/*
* ' R ' indicates read - only activation of a device that
* does not have metadata flagging it as read - only .
*/
if ( repstr [ 1 ] ! = ' r ' & & info . read_only )
repstr [ 1 ] = ' R ' ;
2010-09-30 14:07:19 +00:00
repstr [ 5 ] = ( info . open_count ) ? ' o ' : ' - ' ;
2010-09-30 13:52:55 +00:00
} else {
repstr [ 4 ] = ' - ' ;
repstr [ 5 ] = ' - ' ;
}
2011-09-08 20:55:39 +00:00
2013-06-05 13:44:10 +02:00
if ( lv_is_thin_pool ( lv ) | | lv_is_thin_volume ( lv ) )
2011-09-08 20:55:39 +00:00
repstr [ 6 ] = ' t ' ;
else if ( lv_is_raid_type ( lv ) )
repstr [ 6 ] = ' r ' ;
2012-08-24 15:34:19 -05:00
else if ( lv_is_mirror_type ( lv ) )
repstr [ 6 ] = ' m ' ;
2011-09-08 20:55:39 +00:00
else if ( lv_is_cow ( lv ) | | lv_is_origin ( lv ) )
repstr [ 6 ] = ' s ' ;
else if ( lv_has_unknown_segments ( lv ) )
repstr [ 6 ] = ' u ' ;
else if ( lv_is_virtual ( lv ) )
repstr [ 6 ] = ' v ' ;
else
repstr [ 6 ] = ' - ' ;
2011-09-14 10:03:15 +00:00
if ( ( ( lv_is_thin_volume ( lv ) & & ( seg = first_seg ( lv ) ) & & seg - > pool_lv & & ( seg = first_seg ( seg - > pool_lv ) ) ) | |
( lv_is_thin_pool ( lv ) & & ( seg = first_seg ( lv ) ) ) ) & &
2011-09-09 01:15:18 +00:00
seg - > zero_new_blocks )
repstr [ 7 ] = ' z ' ;
else
repstr [ 7 ] = ' - ' ;
2013-04-11 15:33:59 -05:00
repstr [ 8 ] = ' - ' ;
if ( lv - > status & PARTIAL_LV )
2012-09-19 12:49:40 +01:00
repstr [ 8 ] = ' p ' ;
2013-04-11 15:33:59 -05:00
else if ( lv_is_raid_type ( lv ) ) {
uint64_t n ;
if ( ! _lv_raid_healthy ( lv ) )
repstr [ 8 ] = ' r ' ; /* RAID needs 'r'efresh */
RAID: Add writemostly/writebehind support for RAID1
'lvchange' is used to alter a RAID 1 logical volume's write-mostly and
write-behind characteristics. The '--writemostly' parameter takes a
PV as an argument with an optional trailing character to specify whether
to set ('y'), unset ('n'), or toggle ('t') the value. If no trailing
character is given, it will set the flag.
Synopsis:
lvchange [--writemostly <PV>:{t|y|n}] [--writebehind <count>] vg/lv
Example:
lvchange --writemostly /dev/sdb1:y --writebehind 512 vg/raid1_lv
The last character in the 'lv_attr' field is used to show whether a device
has the WriteMostly flag set. It is signified with a 'w'. If the device
has failed, the 'p'artial flag has priority.
Example ("nosync" raid1 with mismatch_cnt and writemostly):
[~]# lvs -a --segment vg
LV VG Attr #Str Type SSize
raid1 vg Rwi---r-m 2 raid1 500.00m
[raid1_rimage_0] vg Iwi---r-- 1 linear 500.00m
[raid1_rimage_1] vg Iwi---r-w 1 linear 500.00m
[raid1_rmeta_0] vg ewi---r-- 1 linear 4.00m
[raid1_rmeta_1] vg ewi---r-- 1 linear 4.00m
Example (raid1 with mismatch_cnt, writemostly - but failed drive):
[~]# lvs -a --segment vg
LV VG Attr #Str Type SSize
raid1 vg rwi---r-p 2 raid1 500.00m
[raid1_rimage_0] vg Iwi---r-- 1 linear 500.00m
[raid1_rimage_1] vg Iwi---r-p 1 linear 500.00m
[raid1_rmeta_0] vg ewi---r-- 1 linear 4.00m
[raid1_rmeta_1] vg ewi---r-p 1 linear 4.00m
A new reportable field has been added for writebehind as well. If
write-behind has not been set or the LV is not RAID1, the field will
be blank.
Example (writebehind is set):
[~]# lvs -a -o name,attr,writebehind vg
LV Attr WBehind
lv rwi-a-r-- 512
[lv_rimage_0] iwi-aor-w
[lv_rimage_1] iwi-aor--
[lv_rmeta_0] ewi-aor--
[lv_rmeta_1] ewi-aor--
Example (writebehind is not set):
[~]# lvs -a -o name,attr,writebehind vg
LV Attr WBehind
lv rwi-a-r--
[lv_rimage_0] iwi-aor-w
[lv_rimage_1] iwi-aor--
[lv_rmeta_0] ewi-aor--
[lv_rmeta_1] ewi-aor--
2013-04-15 13:59:46 -05:00
else if ( lv - > status & RAID ) {
if ( lv_raid_mismatch_count ( lv , & n ) & & n )
repstr [ 8 ] = ' m ' ; /* RAID has 'm'ismatches */
} else if ( lv - > status & LV_WRITEMOSTLY )
repstr [ 8 ] = ' w ' ; /* sub-LV has 'w'ritemostly */
2013-04-11 15:33:59 -05:00
}
2012-09-19 12:49:40 +01:00
2010-09-30 13:52:55 +00:00
out :
return repstr ;
}
2012-01-19 15:31:45 +00:00
int lv_set_creation ( struct logical_volume * lv ,
const char * hostname , uint64_t timestamp )
{
const char * hn ;
if ( ! hostname ) {
if ( ! _utsinit ) {
if ( uname ( & _utsname ) ) {
log_error ( " uname failed: %s " , strerror ( errno ) ) ;
memset ( & _utsname , 0 , sizeof ( _utsname ) ) ;
}
_utsinit = 1 ;
}
hostname = _utsname . nodename ;
}
if ( ! ( hn = dm_hash_lookup ( lv - > vg - > hostnames , hostname ) ) ) {
if ( ! ( hn = dm_pool_strdup ( lv - > vg - > vgmem , hostname ) ) ) {
log_error ( " Failed to duplicate hostname " ) ;
return 0 ;
}
if ( ! dm_hash_insert ( lv - > vg - > hostnames , hostname , ( void * ) hn ) )
return_0 ;
}
lv - > hostname = hn ;
2012-02-23 22:31:23 +00:00
lv - > timestamp = timestamp ? : ( uint64_t ) time ( NULL ) ;
2012-01-19 15:31:45 +00:00
return 1 ;
}
char * lv_time_dup ( struct dm_pool * mem , const struct logical_volume * lv )
{
char buffer [ 50 ] ;
struct tm * local_tm ;
time_t ts = ( time_t ) lv - > timestamp ;
if ( ! ts | |
! ( local_tm = localtime ( & ts ) ) | |
/* FIXME: make this lvm.conf configurable */
! strftime ( buffer , sizeof ( buffer ) ,
" %Y-%m-%d %T %z " , local_tm ) )
buffer [ 0 ] = 0 ;
return dm_pool_strdup ( mem , buffer ) ;
}
char * lv_host_dup ( struct dm_pool * mem , const struct logical_volume * lv )
{
return dm_pool_strdup ( mem , lv - > hostname ? : " " ) ;
}
2013-04-25 12:12:05 +02:00
2013-04-29 14:04:38 +02:00
int lv_active_change ( struct cmd_context * cmd , struct logical_volume * lv ,
2013-05-03 16:02:43 +02:00
enum activation_change activate )
2013-04-29 14:04:38 +02:00
{
if ( activate = = CHANGE_AN ) {
log_verbose ( " Deactivating logical volume \" %s \" " , lv - > name ) ;
if ( ! deactivate_lv ( cmd , lv ) )
return_0 ;
} else if ( ( activate = = CHANGE_AE ) | |
lv_is_origin ( lv ) | |
lv_is_thin_type ( lv ) ) {
if ( activate = = CHANGE_ALN ) {
/* origin or thin, all others have _AE */
/* other types of activation are implicitly exclusive */
/* Note: the order of tests is mandatory */
log_error ( " Cannot deactivate \" %s \" locally. " , lv - > name ) ;
return 0 ;
}
log_verbose ( " Activating logical volume \" %s \" exclusively. " , lv - > name ) ;
if ( ! activate_lv_excl ( cmd , lv ) )
return_0 ;
} else if ( activate = = CHANGE_ALN ) {
log_verbose ( " Deactivating logical volume \" %s \" locally. " , lv - > name ) ;
if ( ! deactivate_lv_local ( cmd , lv ) )
return_0 ;
} else if ( ( activate = = CHANGE_ALY ) | | ( activate = = CHANGE_AAY ) ) {
log_verbose ( " Activating logical volume \" %s \" locally. " , lv - > name ) ;
if ( ! activate_lv_local ( cmd , lv ) )
return_0 ;
} else { /* CHANGE_AY */
log_verbose ( " Activating logical volume \" %s \" . " , lv - > name ) ;
if ( ! activate_lv ( cmd , lv ) )
return_0 ;
}
return 1 ;
}
2013-04-25 12:12:05 +02:00
char * lv_active_dup ( struct dm_pool * mem , const struct logical_volume * lv )
{
const char * s ;
2013-05-02 18:06:50 +02:00
if ( vg_is_clustered ( lv - > vg ) ) {
//const struct logical_volume *lvo = lv;
lv = lv_lock_holder ( lv ) ;
//log_debug("Holder for %s => %s.", lvo->name, lv->name);
}
2013-04-25 12:12:05 +02:00
if ( ! lv_is_active ( lv ) )
s = " " ; /* not active */
else if ( ! vg_is_clustered ( lv - > vg ) )
s = " active " ;
else if ( lv_is_active_exclusive ( lv ) )
/* exclusive cluster activation */
s = lv_is_active_exclusive_locally ( lv ) ?
" local exclusive " : " remote exclusive " ;
else /* locally active */
s = lv_is_active_but_not_locally ( lv ) ?
2013-05-02 18:06:50 +02:00
" remotely " : " locally " ;
2013-04-25 12:12:05 +02:00
return dm_pool_strdup ( mem , s ) ;
}
2013-05-02 18:06:50 +02:00
/* For given LV find recursively the LV which holds lock for it */
const struct logical_volume * lv_lock_holder ( const struct logical_volume * lv )
{
const struct seg_list * sl ;
if ( lv_is_cow ( lv ) )
return lv_lock_holder ( origin_from_cow ( lv ) ) ;
if ( lv_is_thin_pool ( lv ) )
/* Find any active LV from the pool */
dm_list_iterate_items ( sl , & lv - > segs_using_this_lv )
if ( lv_is_active ( sl - > seg - > lv ) ) {
log_debug ( " Thin volume \" %s \" is active. " , sl - > seg - > lv - > name ) ;
return sl - > seg - > lv ;
}
/* For other types, by default look for the first user */
dm_list_iterate_items ( sl , & lv - > segs_using_this_lv ) {
/* FIXME: complete this exception list */
if ( lv_is_thin_volume ( lv ) & &
lv_is_thin_volume ( sl - > seg - > lv ) & &
first_seg ( lv ) - > pool_lv = = sl - > seg - > pool_lv )
continue ; /* Skip thin snaphost */
if ( lv_is_external_origin ( lv ) & &
lv_is_thin_volume ( sl - > seg - > lv ) )
continue ; /* Skip external origin */
return lv_lock_holder ( sl - > seg - > lv ) ;
}
return lv ;
}