2011-09-06 19:25:42 +00:00
/*
2013-02-05 11:07:09 +01:00
* Copyright ( C ) 2011 - 2013 Red Hat , Inc . All rights reserved .
2011-09-06 19:25:42 +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"
2011-10-22 16:44:23 +00:00
# include "activate.h"
# include "locking.h"
2011-09-06 19:25:42 +00:00
# include "metadata.h"
2011-09-08 16:41:18 +00:00
# include "segtype.h"
# include "lv_alloc.h"
2012-02-08 13:05:38 +00:00
# include "defaults.h"
2013-03-11 12:37:09 +01:00
# include "display.h"
2011-09-06 19:25:42 +00:00
2012-01-25 08:55:19 +00:00
int attach_pool_metadata_lv ( struct lv_segment * pool_seg , struct logical_volume * metadata_lv )
2011-09-06 19:25:42 +00:00
{
2012-01-25 08:55:19 +00:00
pool_seg - > metadata_lv = metadata_lv ;
2012-01-19 15:23:50 +00:00
metadata_lv - > status | = THIN_POOL_METADATA ;
lv_set_hidden ( metadata_lv ) ;
2011-09-06 19:25:42 +00:00
2012-01-25 08:55:19 +00:00
return add_seg_to_segs_using_this_lv ( metadata_lv , pool_seg ) ;
2011-09-06 19:25:42 +00:00
}
2012-12-02 01:28:51 +01:00
int detach_pool_metadata_lv ( struct lv_segment * pool_seg , struct logical_volume * * metadata_lv )
{
struct logical_volume * lv = pool_seg - > metadata_lv ;
if ( ! lv | | ! lv_is_thin_pool_metadata ( lv ) | |
! remove_seg_from_segs_using_this_lv ( lv , pool_seg ) ) {
log_error ( INTERNAL_ERROR " LV %s is invalid thin pool. " , pool_seg - > lv - > name ) ;
return 0 ;
}
lv_set_visible ( lv ) ;
lv - > status & = ~ THIN_POOL_METADATA ;
* metadata_lv = lv ;
pool_seg - > metadata_lv = NULL ;
return 1 ;
}
2012-01-25 08:55:19 +00:00
int attach_pool_data_lv ( struct lv_segment * pool_seg , struct logical_volume * pool_data_lv )
2011-09-06 19:25:42 +00:00
{
2012-01-25 08:55:19 +00:00
if ( ! set_lv_segment_area_lv ( pool_seg , 0 , pool_data_lv , 0 , THIN_POOL_DATA ) )
2011-09-08 16:41:18 +00:00
return_0 ;
2013-07-31 15:13:54 +02:00
pool_seg - > lv - > status | = THIN_POOL ;
2011-10-19 16:36:39 +00:00
lv_set_hidden ( pool_data_lv ) ;
2011-09-06 19:25:42 +00:00
2011-10-19 16:36:39 +00:00
return 1 ;
2011-09-06 19:25:42 +00:00
}
2011-11-07 11:03:47 +00:00
int attach_pool_lv ( struct lv_segment * seg , struct logical_volume * pool_lv ,
struct logical_volume * origin )
2011-09-06 22:43:56 +00:00
{
seg - > pool_lv = pool_lv ;
2011-09-08 16:41:18 +00:00
seg - > lv - > status | = THIN_VOLUME ;
2011-11-07 11:03:47 +00:00
seg - > origin = origin ;
if ( origin & & ! add_seg_to_segs_using_this_lv ( origin , seg ) )
return_0 ;
2011-09-06 22:43:56 +00:00
2011-10-19 16:36:39 +00:00
return add_seg_to_segs_using_this_lv ( pool_lv , seg ) ;
2011-09-06 22:43:56 +00:00
}
2011-09-08 16:41:18 +00:00
int detach_pool_lv ( struct lv_segment * seg )
{
2011-11-03 14:36:40 +00:00
struct lv_thin_message * tmsg , * tmp ;
2011-11-07 11:03:47 +00:00
struct seg_list * sl , * tsl ;
2012-01-25 08:55:19 +00:00
int no_update = 0 ;
2011-10-19 16:37:30 +00:00
2011-11-03 14:36:40 +00:00
if ( ! seg - > pool_lv | | ! lv_is_thin_pool ( seg - > pool_lv ) ) {
2013-07-09 12:34:48 +01:00
log_error ( INTERNAL_ERROR " Cannot detach pool from non-thin LV %s " ,
2011-11-03 14:36:40 +00:00
seg - > lv - > name ) ;
2011-09-08 16:41:18 +00:00
return 0 ;
}
2011-10-19 16:37:30 +00:00
/* Drop any message referencing removed segment */
2012-01-25 08:55:19 +00:00
dm_list_iterate_items_safe ( tmsg , tmp , & ( first_seg ( seg - > pool_lv ) - > thin_messages ) ) {
2011-10-19 16:37:30 +00:00
switch ( tmsg - > type ) {
case DM_THIN_MESSAGE_CREATE_SNAP :
case DM_THIN_MESSAGE_CREATE_THIN :
2012-01-25 08:55:19 +00:00
if ( tmsg - > u . lv = = seg - > lv ) {
2013-01-07 22:30:29 +00:00
log_debug_metadata ( " Discarding message for LV %s. " ,
tmsg - > u . lv - > name ) ;
2011-10-19 16:37:30 +00:00
dm_list_del ( & tmsg - > list ) ;
2012-01-25 08:55:19 +00:00
no_update = 1 ; /* Replacing existing */
2011-10-19 16:37:30 +00:00
}
2011-11-07 11:04:45 +00:00
break ;
case DM_THIN_MESSAGE_DELETE :
if ( tmsg - > u . delete_id = = seg - > device_id ) {
log_error ( INTERNAL_ERROR " Trying to delete %u again. " ,
tmsg - > u . delete_id ) ;
return 0 ;
}
break ;
2011-10-19 16:37:30 +00:00
default :
2011-11-07 11:04:45 +00:00
log_error ( INTERNAL_ERROR " Unsupported message type %u. " , tmsg - > type ) ;
2011-10-19 16:37:30 +00:00
break ;
}
}
2013-02-21 10:25:44 +01:00
if ( ! detach_thin_external_origin ( seg ) )
return_0 ;
2011-10-17 14:17:09 +00:00
if ( ! attach_pool_message ( first_seg ( seg - > pool_lv ) ,
DM_THIN_MESSAGE_DELETE ,
2012-01-25 08:55:19 +00:00
NULL , seg - > device_id , no_update ) )
2011-10-17 14:17:09 +00:00
return_0 ;
2011-11-07 11:03:47 +00:00
if ( ! remove_seg_from_segs_using_this_lv ( seg - > pool_lv , seg ) )
return_0 ;
if ( seg - > origin & &
! remove_seg_from_segs_using_this_lv ( seg - > origin , seg ) )
return_0 ;
/* If thin origin, remove it from related thin snapshots */
/*
* TODO : map removal of origin as snapshot lvconvert - - merge ?
* i . e . rename thin snapshot to origin thin origin
*/
dm_list_iterate_items_safe ( sl , tsl , & seg - > lv - > segs_using_this_lv ) {
if ( ! seg_is_thin_volume ( sl - > seg ) | |
( seg - > lv ! = sl - > seg - > origin ) )
continue ;
if ( ! remove_seg_from_segs_using_this_lv ( seg - > lv , sl - > seg ) )
return_0 ;
/* Thin snapshot is now regular thin volume */
sl - > seg - > origin = NULL ;
}
2013-02-05 11:07:09 +01:00
seg - > lv - > status & = ~ THIN_VOLUME ;
seg - > pool_lv = NULL ;
seg - > origin = NULL ;
2011-11-07 11:03:47 +00:00
return 1 ;
2011-09-08 16:41:18 +00:00
}
2012-01-25 08:55:19 +00:00
int attach_pool_message ( struct lv_segment * pool_seg , dm_thin_message_t type ,
2011-10-19 16:39:09 +00:00
struct logical_volume * lv , uint32_t delete_id ,
2012-01-25 08:55:19 +00:00
int no_update )
2011-10-17 14:17:09 +00:00
{
struct lv_thin_message * tmsg ;
2012-01-25 08:55:19 +00:00
if ( ! seg_is_thin_pool ( pool_seg ) ) {
2013-07-09 12:34:48 +01:00
log_error ( INTERNAL_ERROR " Cannot attach message to non-pool LV %s. " , pool_seg - > lv - > name ) ;
2012-01-25 08:55:19 +00:00
return 0 ;
}
2011-11-07 10:59:07 +00:00
2012-01-25 08:55:19 +00:00
if ( pool_has_message ( pool_seg , lv , delete_id ) ) {
if ( lv )
log_error ( " Message referring LV %s already queued in pool %s. " ,
lv - > name , pool_seg - > lv - > name ) ;
else
log_error ( " Delete for device %u already queued in pool %s. " ,
delete_id , pool_seg - > lv - > name ) ;
return 0 ;
2011-10-19 16:39:09 +00:00
}
2012-01-25 08:55:19 +00:00
if ( ! ( tmsg = dm_pool_alloc ( pool_seg - > lv - > vg - > vgmem , sizeof ( * tmsg ) ) ) ) {
2011-10-17 14:17:09 +00:00
log_error ( " Failed to allocate memory for message. " ) ;
return 0 ;
}
switch ( type ) {
case DM_THIN_MESSAGE_CREATE_SNAP :
case DM_THIN_MESSAGE_CREATE_THIN :
tmsg - > u . lv = lv ;
break ;
case DM_THIN_MESSAGE_DELETE :
2011-10-19 16:39:09 +00:00
tmsg - > u . delete_id = delete_id ;
2011-10-17 14:17:09 +00:00
break ;
default :
2011-10-19 16:42:14 +00:00
log_error ( INTERNAL_ERROR " Unsupported message type %u. " , type ) ;
2011-10-17 14:17:09 +00:00
return 0 ;
}
tmsg - > type = type ;
/* If the 1st message is add in non-read-only mode, modify transaction_id */
2012-01-25 08:55:19 +00:00
if ( ! no_update & & dm_list_empty ( & pool_seg - > thin_messages ) )
pool_seg - > transaction_id + + ;
2011-10-17 14:17:09 +00:00
2012-01-25 08:55:19 +00:00
dm_list_add ( & pool_seg - > thin_messages , & tmsg - > list ) ;
2011-10-17 14:17:09 +00:00
2013-01-07 22:30:29 +00:00
log_debug_metadata ( " Added %s message " ,
( type = = DM_THIN_MESSAGE_CREATE_SNAP | |
type = = DM_THIN_MESSAGE_CREATE_THIN ) ? " create " :
( type = = DM_THIN_MESSAGE_DELETE ) ? " delete " : " unknown " ) ;
2011-10-17 14:17:09 +00:00
return 1 ;
}
2013-02-21 10:25:44 +01:00
int attach_thin_external_origin ( struct lv_segment * seg ,
struct logical_volume * external_lv )
{
if ( seg - > external_lv ) {
log_error ( INTERNAL_ERROR " LV \" %s \" already has external origin. " ,
seg - > lv - > name ) ;
return 0 ;
}
seg - > external_lv = external_lv ;
if ( external_lv ) {
if ( ! add_seg_to_segs_using_this_lv ( external_lv , seg ) )
return_0 ;
external_lv - > external_count + + ;
if ( external_lv - > status & LVM_WRITE ) {
log_verbose ( " Setting logical volume \" %s \" read-only. " ,
external_lv - > name ) ;
external_lv - > status & = ~ LVM_WRITE ;
}
}
return 1 ;
}
int detach_thin_external_origin ( struct lv_segment * seg )
{
if ( seg - > external_lv ) {
if ( ! lv_is_external_origin ( seg - > external_lv ) ) {
log_error ( INTERNAL_ERROR " Inconsitent external origin. " ) ;
return 0 ;
}
if ( ! remove_seg_from_segs_using_this_lv ( seg - > external_lv , seg ) )
return_0 ;
seg - > external_lv - > external_count - - ;
seg - > external_lv = NULL ;
}
return 1 ;
}
2012-01-25 08:55:19 +00:00
/*
* Check whether pool has some message queued for LV or for device_id
* When LV is NULL and device_id is 0 it just checks for any message .
*/
int pool_has_message ( const struct lv_segment * seg ,
const struct logical_volume * lv , uint32_t device_id )
{
const struct lv_thin_message * tmsg ;
if ( ! seg_is_thin_pool ( seg ) ) {
log_error ( INTERNAL_ERROR " LV %s is not pool. " , seg - > lv - > name ) ;
return 0 ;
}
if ( ! lv & & ! device_id )
return dm_list_empty ( & seg - > thin_messages ) ;
dm_list_iterate_items ( tmsg , & seg - > thin_messages ) {
switch ( tmsg - > type ) {
case DM_THIN_MESSAGE_CREATE_SNAP :
case DM_THIN_MESSAGE_CREATE_THIN :
if ( tmsg - > u . lv = = lv )
return 1 ;
break ;
case DM_THIN_MESSAGE_DELETE :
if ( tmsg - > u . delete_id = = device_id )
return 1 ;
break ;
default :
break ;
}
}
return 0 ;
}
2013-02-05 16:49:09 +01:00
int pool_is_active ( const struct logical_volume * lv )
2013-02-05 10:52:39 +01:00
{
struct lvinfo info ;
const struct seg_list * sl ;
2013-02-05 16:49:09 +01:00
if ( ! lv_is_thin_pool ( lv ) ) {
2013-07-09 12:34:48 +01:00
log_error ( INTERNAL_ERROR " pool_is_active called with non-pool LV %s. " , lv - > name ) ;
2013-02-05 10:52:39 +01:00
return 0 ;
}
/* On clustered VG, query every related thin pool volume */
if ( vg_is_clustered ( lv - > vg ) ) {
if ( lv_is_active ( lv ) )
return 1 ;
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 1 ;
}
} else if ( lv_info ( lv - > vg - > cmd , lv , 1 , & info , 0 , 0 ) & & info . exists )
return 1 ; /* Non clustered VG - just checks for '-tpool' */
return 0 ;
}
2013-06-11 12:32:01 +02:00
int pool_can_resize_metadata ( const struct logical_volume * lv )
{
static unsigned attr = 0U ;
struct lv_segment * seg ;
if ( ! lv_is_thin_pool ( lv ) ) {
log_error ( INTERNAL_ERROR " LV %s is not thin pool. " , lv - > name ) ;
return 0 ;
}
seg = first_seg ( lv ) ;
if ( ( attr = = 0U ) & & activation ( ) & & seg - > segtype & &
seg - > segtype - > ops - > target_present & &
! seg - > segtype - > ops - > target_present ( lv - > vg - > cmd , NULL , & attr ) ) {
log_error ( " %s: Required device-mapper target(s) not "
" detected in your kernel " , seg - > segtype - > name ) ;
return 0 ;
}
return ( attr & THIN_FEATURE_METADATA_RESIZE ) ? 1 : 0 ;
}
2012-02-08 13:05:38 +00:00
int pool_below_threshold ( const struct lv_segment * pool_seg )
{
percent_t percent ;
int threshold = PERCENT_1 *
2013-06-27 11:22:02 +02:00
find_config_tree_int ( pool_seg - > lv - > vg - > cmd , activation_thin_pool_autoextend_threshold_CFG ,
lv_config_profile ( pool_seg - > lv ) ) ;
2012-02-08 13:05:38 +00:00
/* Data */
2012-02-12 21:42:43 +00:00
if ( ! lv_thin_pool_percent ( pool_seg - > lv , 0 , & percent ) )
return_0 ;
2012-02-08 13:05:38 +00:00
if ( percent > = threshold )
2012-02-12 21:42:43 +00:00
return_0 ;
2012-02-08 13:05:38 +00:00
/* Metadata */
2012-02-12 21:42:43 +00:00
if ( ! lv_thin_pool_percent ( pool_seg - > lv , 1 , & percent ) )
return_0 ;
2012-02-08 13:05:38 +00:00
if ( percent > = threshold )
2012-02-12 21:42:43 +00:00
return_0 ;
2012-02-08 13:05:38 +00:00
return 1 ;
}
2011-09-09 01:15:18 +00:00
struct lv_segment * find_pool_seg ( const struct lv_segment * seg )
2011-09-08 16:41:18 +00:00
{
2011-10-19 16:36:39 +00:00
struct lv_segment * pool_seg ;
2011-09-08 16:41:18 +00:00
2011-10-19 16:36:39 +00:00
pool_seg = get_only_segment_using_this_lv ( seg - > lv ) ;
2011-09-08 16:41:18 +00:00
2011-10-19 16:36:39 +00:00
if ( ! pool_seg ) {
log_error ( " Failed to find pool_seg for %s " , seg - > lv - > name ) ;
return NULL ;
}
2011-09-08 16:41:18 +00:00
2011-10-19 16:36:39 +00:00
if ( ! seg_is_thin_pool ( pool_seg ) ) {
log_error ( " %s on %s is not a pool segment " ,
pool_seg - > lv - > name , seg - > lv - > name ) ;
return NULL ;
}
2011-09-08 16:41:18 +00:00
2011-10-19 16:36:39 +00:00
return pool_seg ;
2011-09-08 16:41:18 +00:00
}
2011-10-03 18:39:17 +00:00
2013-06-11 12:32:01 +02:00
struct logical_volume * find_pool_lv ( struct logical_volume * lv )
{
struct lv_segment * seg ;
2013-06-14 22:02:12 +02:00
if ( ! ( seg = first_seg ( lv ) ) ) {
log_error ( " LV %s has no segment " , lv - > name ) ;
return NULL ;
}
if ( ! ( seg = find_pool_seg ( seg ) ) )
2013-06-11 12:32:01 +02:00
return_NULL ;
return seg - > lv ;
}
2011-10-03 18:39:17 +00:00
/*
* Find a free device_id for given thin_pool segment .
*
* \ return
* Free device id , or 0 if free device_id is not found .
*
* FIXME : Improve naive search and keep the value cached
* and updated during VG lifetime ( so no const for lv_segment )
*/
uint32_t get_free_pool_device_id ( struct lv_segment * thin_pool_seg )
{
2011-11-03 14:36:40 +00:00
uint32_t max_id = 0 ;
struct seg_list * sl ;
2011-10-03 18:39:17 +00:00
if ( ! seg_is_thin_pool ( thin_pool_seg ) ) {
2011-10-30 22:00:57 +00:00
log_error ( INTERNAL_ERROR
" Segment in %s is not a thin pool segment. " ,
2011-10-03 19:10:52 +00:00
thin_pool_seg - > lv - > name ) ;
2011-10-03 18:39:17 +00:00
return 0 ;
}
2011-11-03 14:36:40 +00:00
dm_list_iterate_items ( sl , & thin_pool_seg - > lv - > segs_using_this_lv )
if ( sl - > seg - > device_id > max_id )
max_id = sl - > seg - > device_id ;
2011-10-03 18:39:17 +00:00
2011-10-30 22:00:57 +00:00
if ( + + max_id > DM_THIN_MAX_DEVICE_ID ) {
2012-01-23 17:46:31 +00:00
/* FIXME Find empty holes instead of aborting! */
2011-10-30 22:00:57 +00:00
log_error ( " Cannot find free device_id. " ) ;
2011-10-03 18:39:17 +00:00
return 0 ;
}
2013-01-07 22:30:29 +00:00
log_debug_metadata ( " Found free pool device_id %u. " , max_id ) ;
2011-10-03 18:39:17 +00:00
return max_id ;
}
2011-10-22 16:44:23 +00:00
2013-07-23 13:33:14 +01:00
int create_pool ( struct logical_volume * pool_lv , const struct segment_type * segtype ,
2011-10-28 20:32:54 +00:00
struct alloc_handle * ah , uint32_t stripes , uint32_t stripe_size )
2011-10-22 16:44:23 +00:00
{
const struct segment_type * striped ;
struct logical_volume * meta_lv , * data_lv ;
struct lv_segment * seg ;
2013-07-18 16:01:38 +02:00
char name [ NAME_LEN ] ;
2011-10-22 16:44:23 +00:00
2011-10-30 22:00:57 +00:00
if ( pool_lv - > le_count ) {
/* FIXME move code for manipulation from lv_manip.c */
log_error ( INTERNAL_ERROR " Pool %s has already extents. " , pool_lv - > name ) ;
2011-10-22 16:44:23 +00:00
return 0 ;
}
/* LV is not yet a pool, so it's extension from lvcreate */
if ( ! ( striped = get_segtype_from_string ( pool_lv - > vg - > cmd , " striped " ) ) )
return_0 ;
if ( activation ( ) & & segtype - > ops - > target_present & &
! segtype - > ops - > target_present ( pool_lv - > vg - > cmd , NULL , NULL ) ) {
log_error ( " %s: Required device-mapper target(s) not "
" detected in your kernel. " , segtype - > name ) ;
return 0 ;
}
/* Metadata segment */
2011-10-28 20:32:54 +00:00
if ( ! lv_add_segment ( ah , stripes , 1 , pool_lv , striped , 1 , 0 , 0 ) )
2011-10-22 16:44:23 +00:00
return_0 ;
2013-07-23 13:33:14 +01:00
if ( ! activation ( ) )
log_warn ( " WARNING: Pool %s is created without initialization. " , pool_lv - > name ) ;
else {
2011-10-22 16:44:23 +00:00
if ( ! vg_write ( pool_lv - > vg ) | | ! vg_commit ( pool_lv - > vg ) )
return_0 ;
/*
* If killed here , only the VISIBLE striped pool LV is left
* and user could easily remove it .
*
* FIXME : implement lazy clearing when activation is disabled
*/
2011-12-10 00:47:23 +00:00
/* pool_lv is a new LV so the VG lock protects us */
2011-10-22 16:44:23 +00:00
if ( ! activate_lv_local ( pool_lv - > vg - > cmd , pool_lv ) | |
/* Clear 4KB of metadata device for new thin-pool. */
! set_lv ( pool_lv - > vg - > cmd , pool_lv , UINT64_C ( 0 ) , 0 ) ) {
log_error ( " Aborting. Failed to wipe pool metadata %s. " ,
pool_lv - > name ) ;
2013-07-18 16:44:39 +02:00
goto bad ;
2011-10-22 16:44:23 +00:00
}
}
2013-07-18 16:01:38 +02:00
if ( dm_snprintf ( name , sizeof ( name ) , " %s_tmeta " , pool_lv - > name ) < 0 ) {
log_error ( " Name is too long to be a pool name. " ) ;
goto bad ;
}
2011-10-22 16:44:23 +00:00
if ( ! ( meta_lv = lv_create_empty ( name , NULL , LVM_READ | LVM_WRITE ,
ALLOC_INHERIT , pool_lv - > vg ) ) )
2013-07-18 16:01:38 +02:00
goto_bad ;
2011-10-22 16:44:23 +00:00
if ( ! move_lv_segments ( meta_lv , pool_lv , 0 , 0 ) )
2013-07-18 16:01:38 +02:00
goto_bad ;
2011-10-22 16:44:23 +00:00
/* Pool data segment */
2011-10-28 20:32:54 +00:00
if ( ! lv_add_segment ( ah , 0 , stripes , pool_lv , striped , stripe_size , 0 , 0 ) )
2013-07-18 16:01:38 +02:00
goto_bad ;
2011-10-22 16:44:23 +00:00
if ( ! ( data_lv = insert_layer_for_lv ( pool_lv - > vg - > cmd , pool_lv ,
2011-11-03 14:38:36 +00:00
pool_lv - > status , " _tdata " ) ) )
2013-07-18 16:01:38 +02:00
goto_bad ;
2011-10-22 16:44:23 +00:00
seg = first_seg ( pool_lv ) ;
/* Drop reference as attach_pool_data_lv() takes it again */
2013-04-21 13:18:53 +02:00
if ( ! remove_seg_from_segs_using_this_lv ( data_lv , seg ) )
2013-07-18 16:01:38 +02:00
goto_bad ;
2013-07-31 15:13:54 +02:00
2011-10-22 16:44:23 +00:00
if ( ! attach_pool_data_lv ( seg , data_lv ) )
2013-07-18 16:01:38 +02:00
goto_bad ;
2011-10-22 16:44:23 +00:00
2013-07-31 15:13:54 +02:00
if ( ! attach_pool_metadata_lv ( seg , meta_lv ) )
goto_bad ;
seg - > segtype = segtype ; /* Set as thin_pool segment */
2011-10-22 16:44:23 +00:00
return 1 ;
2013-07-23 13:33:14 +01:00
2013-07-18 16:01:38 +02:00
bad :
if ( activation ( ) ) {
if ( deactivate_lv_local ( pool_lv - > vg - > cmd , pool_lv ) ) {
log_error ( " Aborting. Could not deactivate pool %s. " ,
pool_lv - > name ) ;
return 0 ;
}
if ( ! lv_remove ( pool_lv ) | | ! vg_write ( pool_lv - > vg ) | | ! vg_commit ( pool_lv - > vg ) )
log_error ( " Manual intervention may be required to remove "
" abandoned LV(s) before retrying. " ) ;
}
return 0 ;
2011-10-22 16:44:23 +00:00
}
2011-11-03 14:53:58 +00:00
int update_pool_lv ( struct logical_volume * lv , int activate )
{
2012-03-23 09:58:04 +00:00
int monitored ;
2011-11-03 14:53:58 +00:00
if ( ! lv_is_thin_pool ( lv ) ) {
log_error ( INTERNAL_ERROR " Updated LV %s is not pool. " , lv - > name ) ;
return 0 ;
}
2012-01-25 09:17:15 +00:00
if ( dm_list_empty ( & ( first_seg ( lv ) - > thin_messages ) ) )
return 1 ; /* No messages */
2011-11-03 14:53:58 +00:00
if ( activate ) {
2011-11-10 12:43:05 +00:00
/* If the pool is not active, do activate deactivate */
2011-11-03 15:58:20 +00:00
if ( ! lv_is_active ( lv ) ) {
2012-03-23 09:58:04 +00:00
monitored = dmeventd_monitor_mode ( ) ;
init_dmeventd_monitor ( DMEVENTD_MONITOR_IGNORE ) ;
2011-11-03 15:58:20 +00:00
if ( ! activate_lv_excl ( lv - > vg - > cmd , lv ) )
return_0 ;
if ( ! deactivate_lv ( lv - > vg - > cmd , lv ) )
return_0 ;
2012-03-23 09:58:04 +00:00
init_dmeventd_monitor ( monitored ) ;
2011-11-03 14:53:58 +00:00
}
2012-01-25 09:13:10 +00:00
/*
* Resume active pool to send thin messages .
* origin_only is used to skip check for resumed state
2011-11-03 14:53:58 +00:00
*/
2012-01-25 09:13:10 +00:00
else if ( ! resume_lv_origin ( lv - > vg - > cmd , lv ) ) {
2011-11-03 14:53:58 +00:00
log_error ( " Failed to resume %s. " , lv - > name ) ;
return 0 ;
}
}
2012-01-25 09:17:15 +00:00
dm_list_init ( & ( first_seg ( lv ) - > thin_messages ) ) ;
2011-11-03 14:53:58 +00:00
2012-01-25 09:17:15 +00:00
if ( ! vg_write ( lv - > vg ) | | ! vg_commit ( lv - > vg ) )
return_0 ;
2011-11-03 14:53:58 +00:00
return 1 ;
}
2012-06-28 14:47:34 +02:00
2013-09-25 16:00:52 +02:00
static int _get_pool_chunk_size_calc ( const char * str ,
int * chunk_size_calc_method ,
uint32_t * chunk_size )
{
if ( ! strcasecmp ( str , " default " ) ) {
* chunk_size_calc_method = THIN_CHUNK_SIZE_CALC_METHOD_DEFAULT ;
* chunk_size = DEFAULT_THIN_POOL_CHUNK_SIZE * 2 ;
}
else if ( ! strcasecmp ( str , " performance " ) ) {
* chunk_size_calc_method = THIN_CHUNK_SIZE_CALC_METHOD_PERFORMANCE ;
* chunk_size = DEFAULT_THIN_POOL_CHUNK_SIZE_PERFORMANCE * 2 ;
}
else {
log_error ( " Thin pool chunk size calculation method \" %s \" is unknown. " , str ) ;
return 0 ;
}
return 1 ;
}
2013-08-06 11:42:40 +02:00
int update_profilable_pool_params ( struct cmd_context * cmd , struct profile * profile ,
2013-09-25 16:00:52 +02:00
int passed_args , int * chunk_size_calc_method ,
uint32_t * chunk_size , thin_discards_t * discards ,
int * zero )
2013-08-06 11:42:40 +02:00
{
2013-09-25 16:00:52 +02:00
const char * str ;
2013-08-06 11:42:40 +02:00
2013-09-25 16:00:52 +02:00
if ( ! ( passed_args & PASS_ARG_CHUNK_SIZE ) ) {
if ( ! ( * chunk_size = find_config_tree_int ( cmd , allocation_thin_pool_chunk_size_CFG , profile ) * 2 ) ) {
str = find_config_tree_str ( cmd , allocation_thin_pool_chunk_size_calculation_CFG , profile ) ;
if ( ! _get_pool_chunk_size_calc ( str , chunk_size_calc_method , chunk_size ) )
return_0 ;
}
}
2013-08-06 16:28:12 +02:00
if ( ( * chunk_size < DM_THIN_MIN_DATA_BLOCK_SIZE ) | |
( * chunk_size > DM_THIN_MAX_DATA_BLOCK_SIZE ) ) {
log_error ( " Chunk size must be in the range %s to %s. " ,
display_size ( cmd , DM_THIN_MIN_DATA_BLOCK_SIZE ) ,
display_size ( cmd , DM_THIN_MAX_DATA_BLOCK_SIZE ) ) ;
return 0 ;
2013-08-06 11:42:40 +02:00
}
if ( ! ( passed_args & PASS_ARG_DISCARDS ) ) {
2013-09-25 16:00:52 +02:00
str = find_config_tree_str ( cmd , allocation_thin_pool_discards_CFG , profile ) ;
if ( ! get_pool_discards ( str , discards ) )
2013-08-06 11:42:40 +02:00
return_0 ;
}
if ( ! ( passed_args & PASS_ARG_ZERO ) )
* zero = find_config_tree_bool ( cmd , allocation_thin_pool_zero_CFG , profile ) ;
return 1 ;
}
int update_pool_params ( struct volume_group * vg , unsigned attr , int passed_args ,
2013-03-11 12:37:09 +01:00
uint32_t data_extents , uint32_t extent_size ,
2013-09-25 16:00:52 +02:00
int * chunk_size_calc_method , uint32_t * chunk_size ,
thin_discards_t * discards , uint64_t * pool_metadata_size ,
int * zero )
2013-03-11 12:37:09 +01:00
{
size_t estimate_chunk_size ;
2013-08-06 11:42:40 +02:00
struct cmd_context * cmd = vg - > cmd ;
if ( ! update_profilable_pool_params ( cmd , vg - > profile , passed_args ,
2013-09-25 16:00:52 +02:00
chunk_size_calc_method , chunk_size ,
discards , zero ) )
2013-08-06 11:42:40 +02:00
return_0 ;
2013-03-11 12:37:09 +01:00
if ( ! ( attr & THIN_FEATURE_BLOCK_SIZE ) & &
( * chunk_size & ( * chunk_size - 1 ) ) ) {
log_error ( " Chunk size must be a power of 2 for this thin target version. " ) ;
return 0 ;
} else if ( * chunk_size & ( DM_THIN_MIN_DATA_BLOCK_SIZE - 1 ) ) {
log_error ( " Chunk size must be multiple of %s. " ,
display_size ( cmd , DM_THIN_MIN_DATA_BLOCK_SIZE ) ) ;
return 0 ;
}
if ( ! * pool_metadata_size ) {
/* Defaults to nr_pool_blocks * 64b converted to size in sectors */
* pool_metadata_size = ( uint64_t ) data_extents * extent_size /
( * chunk_size * ( SECTOR_SIZE / UINT64_C ( 64 ) ) ) ;
/* Check if we could eventually use bigger chunk size */
if ( ! ( passed_args & PASS_ARG_CHUNK_SIZE ) ) {
while ( ( * pool_metadata_size >
( DEFAULT_THIN_POOL_OPTIMAL_SIZE / SECTOR_SIZE ) ) & &
( * chunk_size < DM_THIN_MAX_DATA_BLOCK_SIZE ) ) {
* chunk_size < < = 1 ;
* pool_metadata_size > > = 1 ;
}
log_verbose ( " Setting chunk size to %s. " ,
display_size ( cmd , * chunk_size ) ) ;
} else if ( * pool_metadata_size > ( 2 * DEFAULT_THIN_POOL_MAX_METADATA_SIZE ) ) {
/* Suggest bigger chunk size */
estimate_chunk_size = ( uint64_t ) data_extents * extent_size /
( 2 * DEFAULT_THIN_POOL_MAX_METADATA_SIZE *
( SECTOR_SIZE / UINT64_C ( 64 ) ) ) ;
log_warn ( " WARNING: Chunk size is too small for pool, suggested minimum is %s. " ,
display_size ( cmd , UINT64_C ( 1 ) < < ( ffs ( estimate_chunk_size ) + 1 ) ) ) ;
}
/* Round up to extent size */
if ( * pool_metadata_size % extent_size )
* pool_metadata_size + = extent_size - * pool_metadata_size % extent_size ;
} else {
estimate_chunk_size = ( uint64_t ) data_extents * extent_size /
( * pool_metadata_size * ( SECTOR_SIZE / UINT64_C ( 64 ) ) ) ;
/* Check to eventually use bigger chunk size */
if ( ! ( passed_args & PASS_ARG_CHUNK_SIZE ) ) {
* chunk_size = estimate_chunk_size ;
if ( * chunk_size < DM_THIN_MIN_DATA_BLOCK_SIZE )
* chunk_size = DM_THIN_MIN_DATA_BLOCK_SIZE ;
else if ( * chunk_size > DM_THIN_MAX_DATA_BLOCK_SIZE )
* chunk_size = DM_THIN_MAX_DATA_BLOCK_SIZE ;
log_verbose ( " Setting chunk size %s. " ,
display_size ( cmd , * chunk_size ) ) ;
} else if ( * chunk_size < estimate_chunk_size ) {
/* Suggest bigger chunk size */
log_warn ( " WARNING: Chunk size is smaller then suggested minimum size %s. " ,
display_size ( cmd , estimate_chunk_size ) ) ;
}
}
if ( ( uint64_t ) * chunk_size > ( uint64_t ) data_extents * extent_size ) {
log_error ( " Chunk size is bigger then pool data size. " ) ;
return 0 ;
}
if ( * pool_metadata_size > ( 2 * DEFAULT_THIN_POOL_MAX_METADATA_SIZE ) ) {
if ( passed_args & PASS_ARG_POOL_METADATA_SIZE )
log_warn ( " WARNING: Maximum supported pool metadata size is %s. " ,
display_size ( cmd , 2 * DEFAULT_THIN_POOL_MAX_METADATA_SIZE ) ) ;
* pool_metadata_size = 2 * DEFAULT_THIN_POOL_MAX_METADATA_SIZE ;
} else if ( * pool_metadata_size < ( 2 * DEFAULT_THIN_POOL_MIN_METADATA_SIZE ) ) {
if ( passed_args & PASS_ARG_POOL_METADATA_SIZE )
log_warn ( " WARNING: Minimum supported pool metadata size is %s. " ,
display_size ( cmd , 2 * DEFAULT_THIN_POOL_MIN_METADATA_SIZE ) ) ;
* pool_metadata_size = 2 * DEFAULT_THIN_POOL_MIN_METADATA_SIZE ;
}
log_verbose ( " Setting pool metadata size to %s. " ,
display_size ( cmd , * pool_metadata_size ) ) ;
return 1 ;
}
2012-08-07 21:24:41 +01:00
int get_pool_discards ( const char * str , thin_discards_t * discards )
2012-06-28 14:47:34 +02:00
{
if ( ! strcasecmp ( str , " passdown " ) )
2012-08-07 21:24:41 +01:00
* discards = THIN_DISCARDS_PASSDOWN ;
2012-08-07 18:37:35 +01:00
else if ( ! strcasecmp ( str , " nopassdown " ) )
2012-08-07 21:24:41 +01:00
* discards = THIN_DISCARDS_NO_PASSDOWN ;
2012-06-28 14:47:34 +02:00
else if ( ! strcasecmp ( str , " ignore " ) )
2012-08-07 21:24:41 +01:00
* discards = THIN_DISCARDS_IGNORE ;
2012-06-28 14:47:34 +02:00
else {
2013-06-24 12:00:48 +02:00
log_error ( " Thin pool discards type \" %s \" is unknown. " , str ) ;
2012-06-28 14:47:34 +02:00
return 0 ;
}
return 1 ;
}
2012-08-07 21:24:41 +01:00
const char * get_pool_discards_name ( thin_discards_t discards )
2012-06-28 14:47:34 +02:00
{
2012-08-07 21:24:41 +01:00
switch ( discards ) {
case THIN_DISCARDS_PASSDOWN :
2012-06-28 14:47:34 +02:00
return " passdown " ;
2012-08-07 21:24:41 +01:00
case THIN_DISCARDS_NO_PASSDOWN :
2012-06-28 14:47:34 +02:00
return " nopassdown " ;
2012-08-07 21:24:41 +01:00
case THIN_DISCARDS_IGNORE :
2012-06-28 14:47:34 +02:00
return " ignore " ;
}
2012-10-09 19:42:26 +02:00
log_error ( INTERNAL_ERROR " Unknown discards type encountered. " ) ;
2012-06-28 14:47:34 +02:00
2012-08-07 18:37:35 +01:00
return " unknown " ;
2012-06-28 14:47:34 +02:00
}
2013-07-04 11:04:05 +02:00
struct logical_volume * alloc_pool_metadata ( struct logical_volume * pool_lv ,
const char * name , uint32_t read_ahead ,
uint32_t stripes , uint32_t stripe_size ,
uint64_t size , alloc_policy_t alloc ,
struct dm_list * pvh )
{
struct logical_volume * metadata_lv ;
/* FIXME: Make lvm2api usable */
struct lvcreate_params lvc = {
. activate = CHANGE_ALY ,
. alloc = alloc ,
. lv_name = name ,
. major = - 1 ,
. minor = - 1 ,
. permission = LVM_READ | LVM_WRITE ,
. pvh = pvh ,
. read_ahead = read_ahead ,
. stripe_size = stripe_size ,
. stripes = stripes ,
. vg_name = pool_lv - > vg - > name ,
. zero = 1 ,
} ;
dm_list_init ( & lvc . tags ) ;
if ( ! ( lvc . extents = extents_from_size ( pool_lv - > vg - > cmd , size ,
pool_lv - > vg - > extent_size ) ) )
return_0 ;
if ( ! ( lvc . segtype = get_segtype_from_string ( pool_lv - > vg - > cmd , " striped " ) ) )
return_0 ;
/* FIXME: allocate properly space for metadata_lv */
if ( ! ( metadata_lv = lv_create_single ( pool_lv - > vg , & lvc ) ) )
return_0 ;
return metadata_lv ;
}
2013-07-05 17:10:47 +02:00
2013-06-25 13:34:31 +02:00
static struct logical_volume * _alloc_pool_metadata_spare ( struct volume_group * vg ,
uint32_t extents ,
struct dm_list * pvh )
{
struct logical_volume * lv ;
/* FIXME: Make lvm2api usable */
struct lvcreate_params lp = {
. activate = CHANGE_ALY ,
. alloc = ALLOC_INHERIT ,
. extents = extents ,
. major = - 1 ,
. minor = - 1 ,
. permission = LVM_READ | LVM_WRITE ,
. pvh = pvh ? : & vg - > pvs ,
. read_ahead = DM_READ_AHEAD_AUTO ,
. stripes = 1 ,
. vg_name = vg - > name ,
. zero = 1 ,
} ;
dm_list_init ( & lp . tags ) ;
if ( ! ( lp . segtype = get_segtype_from_string ( vg - > cmd , " striped " ) ) )
return_0 ;
/* FIXME: Maybe using silent mode ? */
if ( ! ( lv = lv_create_single ( vg , & lp ) ) )
return_0 ;
/* Spare LV should not be active */
if ( ! deactivate_lv_local ( vg - > cmd , lv ) ) {
log_error ( " Unable to deactivate pool metadata spare LV. "
" Manual intervention required. " ) ;
return 0 ;
}
if ( ! vg_set_pool_metadata_spare ( lv ) )
return_0 ;
return lv ;
}
/*
* Create / resize pool metadata spare LV
* Caller does vg_write ( ) , vg_commit ( ) with pool creation
* extents is 0 , max size is determined
*/
int handle_pool_metadata_spare ( struct volume_group * vg , uint32_t extents ,
struct dm_list * pvh , int poolmetadataspare )
{
struct logical_volume * lv = vg - > pool_metadata_spare_lv ;
uint32_t seg_mirrors ;
struct lv_segment * seg ;
const struct lv_list * lvl ;
if ( ! extents )
/* Find maximal size of metadata LV */
dm_list_iterate_items ( lvl , & vg - > lvs )
if ( lv_is_thin_pool_metadata ( lvl - > lv ) & &
( lvl - > lv - > le_count > extents ) )
extents = lvl - > lv - > le_count ;
if ( ! poolmetadataspare ) {
/* TODO: Not showing when lvm.conf would define 'n' ? */
if ( DEFAULT_POOL_METADATA_SPARE & & extents )
/* Warn if there would be any user */
log_warn ( " WARNING: recovery of pools without pool "
" metadata spare LV is not automated. " ) ;
return 1 ;
}
if ( ! lv ) {
if ( ! _alloc_pool_metadata_spare ( vg , extents , pvh ) )
return_0 ;
return 1 ;
}
seg = last_seg ( lv ) ;
seg_mirrors = lv_mirror_count ( lv ) ;
/* Check spare LV is big enough and preserve segtype */
if ( ( lv - > le_count < extents ) & & seg & &
! lv_extend ( lv , seg - > segtype ,
seg - > area_count / seg_mirrors ,
seg - > stripe_size ,
seg_mirrors ,
seg - > region_size ,
extents - lv - > le_count , NULL ,
pvh , lv - > alloc ) )
return_0 ;
return 1 ;
}
2013-07-05 17:10:47 +02:00
int vg_set_pool_metadata_spare ( struct logical_volume * lv )
{
char new_name [ NAME_LEN ] ;
struct volume_group * vg = lv - > vg ;
if ( vg - > pool_metadata_spare_lv ) {
if ( vg - > pool_metadata_spare_lv = = lv )
return 1 ;
if ( ! vg_remove_pool_metadata_spare ( vg ) )
return_0 ;
}
if ( dm_snprintf ( new_name , sizeof ( new_name ) , " %s_pmspare " , lv - > name ) < 0 ) {
log_error ( " Can't create pool metadata spare. Name of pool LV "
" %s is too long. " , lv - > name ) ;
return 0 ;
}
if ( ! lv_rename_update ( vg - > cmd , lv , new_name , 0 ) )
return_0 ;
lv_set_hidden ( lv ) ;
lv - > status | = POOL_METADATA_SPARE ;
vg - > pool_metadata_spare_lv = lv ;
return 1 ;
}
int vg_remove_pool_metadata_spare ( struct volume_group * vg )
{
char new_name [ NAME_LEN ] ;
char * c ;
struct logical_volume * lv = vg - > pool_metadata_spare_lv ;
if ( ! ( lv - > status & POOL_METADATA_SPARE ) ) {
log_error ( INTERNAL_ERROR " LV %s is not pool metadata spare. " ,
lv - > name ) ;
return 0 ;
}
vg - > pool_metadata_spare_lv = NULL ;
lv - > status & = ~ POOL_METADATA_SPARE ;
lv_set_visible ( lv ) ;
/* Cut off suffix _pmspare */
( void ) dm_strncpy ( new_name , lv - > name , sizeof ( new_name ) ) ;
if ( ! ( c = strchr ( new_name , ' _ ' ) ) ) {
log_error ( INTERNAL_ERROR " LV %s has no suffix for pool metadata spare. " ,
new_name ) ;
return 0 ;
}
* c = 0 ;
/* If the name is in use, generate new lvol%d */
if ( find_lv_in_vg ( vg , new_name ) & &
! generate_lv_name ( vg , " lvol%d " , new_name , sizeof ( new_name ) ) ) {
log_error ( " Failed to generate unique name for "
" pool metadata spare logical volume. " ) ;
return 0 ;
}
log_print_unless_silent ( " Renaming existing pool metadata spare "
" logical volume \" %s/%s \" to \" %s/%s \" . " ,
vg - > name , lv - > name , vg - > name , new_name ) ;
if ( ! lv_rename_update ( vg - > cmd , lv , new_name , 0 ) )
return_0 ;
2013-06-25 13:34:31 +02:00
/* To display default warning */
( void ) handle_pool_metadata_spare ( vg , 0 , 0 , 0 ) ;
2013-07-05 17:10:47 +02:00
return 1 ;
}