2015-03-12 16:23:19 +01:00
/*
2015-04-16 17:31:07 +02:00
* Copyright ( C ) 2005 - 2015 Red Hat , Inc . All rights reserved .
2015-03-12 16:23:19 +01: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 ,
2016-01-21 11:49:46 +01:00
* Inc . , 51 Franklin Street , Fifth Floor , Boston , MA 02110 - 1301 USA
2015-03-12 16:23:19 +01:00
*/
# include "tools.h"
2015-07-06 17:30:18 +01:00
# include "lvconvert_poll.h"
2015-03-12 16:23:19 +01:00
int lvconvert_mirror_finish ( struct cmd_context * cmd ,
struct volume_group * vg ,
struct logical_volume * lv ,
struct dm_list * lvs_changed __attribute__ ( ( unused ) ) )
{
if ( ! lv_is_converting ( lv ) )
return 1 ;
if ( ! collapse_mirrored_lv ( lv ) ) {
log_error ( " Failed to remove temporary sync layer. " ) ;
return 0 ;
}
lv - > status & = ~ CONVERTING ;
if ( ! lv_update_and_reload ( lv ) )
return_0 ;
2017-06-27 08:28:36 +02:00
log_print_unless_silent ( " Logical volume %s converted. " , display_lvname ( lv ) ) ;
2015-03-12 16:23:19 +01:00
return 1 ;
}
/* Swap lvid and LV names */
int swap_lv_identifiers ( struct cmd_context * cmd ,
struct logical_volume * a , struct logical_volume * b )
{
2024-03-09 23:21:12 +01:00
struct logical_volume tlv = * a ;
2015-09-11 20:56:53 +02:00
const char * aname = a - > name , * bname = b - > name ;
2015-03-12 16:23:19 +01:00
a - > lvid = b - > lvid ;
2024-03-09 23:21:12 +01:00
b - > lvid = tlv . lvid ;
a - > alloc = b - > alloc ;
b - > alloc = tlv . alloc ;
a - > read_ahead = b - > read_ahead ;
b - > read_ahead = tlv . read_ahead ;
a - > profile = b - > profile ;
b - > profile = tlv . profile ;
a - > major = b - > major ;
b - > major = tlv . major ;
a - > minor = b - > minor ;
b - > minor = tlv . minor ;
a - > timestamp = b - > timestamp ;
b - > timestamp = tlv . timestamp ;
a - > hostname = b - > hostname ;
b - > hostname = tlv . hostname ;
/* swap tags */
dm_list_init ( & tlv . tags ) ;
dm_list_splice ( & tlv . tags , & a - > tags ) ;
dm_list_splice ( & a - > tags , & b - > tags ) ;
dm_list_splice ( & b - > tags , & tlv . tags ) ;
2015-03-12 16:23:19 +01:00
2015-09-11 20:56:53 +02:00
/* rename temporarily to 'unused' name */
2024-08-29 09:12:41 +00:00
if ( ! lv_rename_update ( cmd , a , " pvmove_tmeta " , 0 ) )
2015-09-11 20:56:53 +02:00
return_0 ;
/* name rename 'b' to unused name of 'a' */
if ( ! lv_rename_update ( cmd , b , aname , 0 ) )
return_0 ;
/* finish name swapping */
if ( ! lv_rename_update ( cmd , a , bname , 0 ) )
2015-03-12 16:23:19 +01:00
return_0 ;
return 1 ;
}
2024-08-29 23:05:41 +02:00
/* Finalize merging of lv into merge_lv */
2015-03-12 16:23:19 +01:00
int thin_merge_finish ( struct cmd_context * cmd ,
struct logical_volume * merge_lv ,
struct logical_volume * lv )
{
if ( ! swap_lv_identifiers ( cmd , merge_lv , lv ) ) {
log_error ( " Failed to swap %s with merging %s. " ,
2017-06-27 08:28:36 +02:00
display_lvname ( lv ) , display_lvname ( merge_lv ) ) ;
2015-03-12 16:23:19 +01:00
return 0 ;
}
2024-03-09 23:21:12 +01:00
/* Preserve status */
lv - > status = merge_lv - > status ;
2015-03-12 16:23:19 +01:00
/* Removed LV has to be visible */
if ( ! lv_remove_single ( cmd , merge_lv , DONT_PROMPT , 1 ) )
return_0 ;
return 1 ;
}
int lvconvert_merge_finish ( struct cmd_context * cmd ,
struct volume_group * vg ,
struct logical_volume * lv ,
struct dm_list * lvs_changed __attribute__ ( ( unused ) ) )
{
struct lv_segment * snap_seg = find_snapshot ( lv ) ;
if ( ! lv_is_merging_origin ( lv ) ) {
2021-03-14 22:03:41 +01:00
log_print ( " Logical volume %s is no longer merging origin, polling has finished. " ,
2017-06-27 08:28:36 +02:00
display_lvname ( lv ) ) ;
2021-03-14 22:03:41 +01:00
return 1 ;
2015-03-12 16:23:19 +01:00
}
2017-06-27 08:28:36 +02:00
log_print_unless_silent ( " Merge of snapshot into logical volume %s has finished. " ,
display_lvname ( lv ) ) ;
2015-03-12 16:23:19 +01:00
if ( seg_is_thin_volume ( snap_seg ) ) {
clear_snapshot_merge ( lv ) ;
if ( ! thin_merge_finish ( cmd , lv , snap_seg - > lv ) )
return_0 ;
} else if ( ! lv_remove_single ( cmd , snap_seg - > cow , DONT_PROMPT , 0 ) ) {
log_error ( " Could not remove snapshot %s merged into %s. " ,
2017-06-27 08:28:36 +02:00
display_lvname ( snap_seg - > cow ) , display_lvname ( lv ) ) ;
2015-03-12 16:23:19 +01:00
return 0 ;
}
return 1 ;
}
progress_t poll_merge_progress ( struct cmd_context * cmd ,
struct logical_volume * lv ,
const char * name __attribute__ ( ( unused ) ) ,
struct daemon_parms * parms )
{
dm_percent_t percent = DM_PERCENT_0 ;
2021-03-14 22:03:41 +01:00
if ( ! lv_is_merging_origin ( lv ) )
/* Nothing to monitor here */
return PROGRESS_FINISHED_ALL ;
if ( ! lv_snapshot_percent ( lv , & percent ) ) {
2017-06-27 08:28:36 +02:00
log_error ( " %s: Failed query for merging percentage. Aborting merge. " ,
display_lvname ( lv ) ) ;
2015-03-12 16:23:19 +01:00
return PROGRESS_CHECK_FAILED ;
} else if ( percent = = DM_PERCENT_INVALID ) {
2017-06-27 08:28:36 +02:00
log_error ( " %s: Merging snapshot invalidated. Aborting merge. " ,
display_lvname ( lv ) ) ;
2015-03-12 16:23:19 +01:00
return PROGRESS_CHECK_FAILED ;
} else if ( percent = = LVM_PERCENT_MERGE_FAILED ) {
2017-06-27 08:28:36 +02:00
log_error ( " %s: Merge failed. Retry merge or inspect manually. " ,
display_lvname ( lv ) ) ;
2015-03-12 16:23:19 +01:00
return PROGRESS_CHECK_FAILED ;
}
if ( parms - > progress_display )
2017-06-27 08:28:36 +02:00
log_print_unless_silent ( " %s: %s: %s%% " , display_lvname ( lv ) , parms - > progress_title ,
2017-06-24 16:22:36 +02:00
display_percent ( cmd , DM_PERCENT_100 - percent ) ) ;
2015-03-12 16:23:19 +01:00
else
2017-06-27 08:28:36 +02:00
log_verbose ( " %s: %s: %s%% " , display_lvname ( lv ) , parms - > progress_title ,
2017-06-24 16:22:36 +02:00
display_percent ( cmd , DM_PERCENT_100 - percent ) ) ;
2015-03-12 16:23:19 +01:00
if ( percent = = DM_PERCENT_0 )
return PROGRESS_FINISHED_ALL ;
return PROGRESS_UNFINISHED ;
}
progress_t poll_thin_merge_progress ( struct cmd_context * cmd ,
struct logical_volume * lv ,
const char * name __attribute__ ( ( unused ) ) ,
struct daemon_parms * parms )
{
2021-03-09 16:23:08 +01:00
uint32_t device_id = 0 ;
2015-03-12 16:23:19 +01:00
2021-03-17 11:17:32 +01:00
if ( ! lv - > snapshot )
return PROGRESS_FINISHED_ALL ; /* Already merged by someone else */
if ( ! lv_thin_device_id ( lv , & device_id ) ) {
2015-03-12 16:23:19 +01:00
stack ;
return PROGRESS_CHECK_FAILED ;
}
/*
* There is no need to poll more than once ,
* a thin snapshot merge is immediate .
*/
if ( device_id ! = find_snapshot ( lv ) - > device_id ) {
2017-06-27 08:28:36 +02:00
log_error ( " LV %s is not merged. " , display_lvname ( lv ) ) ;
2015-03-12 16:23:19 +01:00
return PROGRESS_CHECK_FAILED ;
}
2024-08-29 23:05:41 +02:00
return PROGRESS_FINISHED_ALL ; /* Merging happened */
2015-03-12 16:23:19 +01:00
}