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"
2011-10-30 22:52:08 +00:00
# include "archiver.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 ;
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 ) ) {
log_error ( INTERNAL_ERROR " LV %s is not a thin volume " ,
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 ) ) {
log_error ( INTERNAL_ERROR " LV %s is not pool. " , pool_seg - > lv - > name ) ;
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-02-05 10:52:39 +01:00
log_error ( INTERNAL_ERROR " LV %s is not pool. " , lv - > name ) ;
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
2011-12-10 00:47:23 +00:00
// FIXME Rename this fn: it doesn't extend an already-existing pool AFAICT
2011-10-22 16:44:23 +00:00
int extend_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 ;
const size_t len = strlen ( pool_lv - > name ) + 16 ;
char name [ len ] ;
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 ;
if ( activation ( ) ) {
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 ) ;
return 0 ;
}
if ( ! deactivate_lv_local ( pool_lv - > vg - > cmd , pool_lv ) ) {
log_error ( " Aborting. Could not deactivate pool metadata %s. " ,
pool_lv - > name ) ;
return 0 ;
}
} else {
2012-10-09 19:42:26 +02:00
log_warn ( " WARNING: Pool %s is created without initialization. " , pool_lv - > name ) ;
2011-10-22 16:44:23 +00:00
}
if ( dm_snprintf ( name , len , " %s_tmeta " , pool_lv - > name ) < 0 )
return_0 ;
if ( ! ( meta_lv = lv_create_empty ( name , NULL , LVM_READ | LVM_WRITE ,
ALLOC_INHERIT , pool_lv - > vg ) ) )
return_0 ;
if ( ! move_lv_segments ( meta_lv , pool_lv , 0 , 0 ) )
return_0 ;
/* 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 ) )
2011-10-22 16:44:23 +00:00
return_0 ;
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 " ) ) )
2011-10-22 16:44:23 +00:00
return_0 ;
seg = first_seg ( pool_lv ) ;
seg - > segtype = segtype ; /* Set as thin_pool segment */
seg - > lv - > status | = THIN_POOL ;
if ( ! attach_pool_metadata_lv ( seg , meta_lv ) )
return_0 ;
/* 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 ) )
return_0 ;
2011-10-22 16:44:23 +00:00
if ( ! attach_pool_data_lv ( seg , data_lv ) )
return_0 ;
return 1 ;
}
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
2012-01-25 09:17:15 +00:00
backup ( lv - > vg ) ;
2011-11-03 14:53:58 +00:00
return 1 ;
}
2012-06-28 14:47:34 +02:00
2013-03-11 12:37:09 +01:00
int update_pool_params ( struct cmd_context * cmd , unsigned attr , int passed_args ,
uint32_t data_extents , uint32_t extent_size ,
uint32_t * chunk_size , thin_discards_t * discards ,
uint64_t * pool_metadata_size )
{
size_t estimate_chunk_size ;
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 ;
}