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"
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_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
2014-01-24 10:49:31 +01:00
log_debug_metadata ( " Added %s message. " ,
2013-01-07 22:30:29 +00:00
( type = = DM_THIN_MESSAGE_CREATE_SNAP | |
2014-01-24 10:49:31 +01:00
type = = DM_THIN_MESSAGE_CREATE_THIN ) ? " create " :
2013-01-07 22:30:29 +00:00
( 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 ;
}
2013-11-29 15:51:28 +01:00
int lv_is_merging_thin_snapshot ( const struct logical_volume * lv )
{
2014-02-22 00:26:01 +00:00
struct lv_segment * seg = first_seg ( lv ) ;
return ( seg & & seg - > status & MERGING ) ? 1 : 0 ;
2013-11-29 15:51:28 +01:00
}
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 )
2014-03-11 22:50:23 +01:00
return ! dm_list_empty ( & seg - > thin_messages ) ;
2012-01-25 08:55:19 +00:00
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 ;
}
2014-01-23 11:47:10 +01:00
int thin_pool_feature_supported ( const struct logical_volume * lv , int feature )
2013-06-11 12:32:01 +02:00
{
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 ;
}
2014-01-23 11:47:10 +01:00
return ( attr & feature ) ? 1 : 0 ;
2013-06-11 12:32:01 +02:00
}
2012-02-08 13:05:38 +00:00
int pool_below_threshold ( const struct lv_segment * pool_seg )
{
2014-06-09 12:08:27 +02:00
dm_percent_t percent ;
int threshold = DM_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 )
2014-01-29 14:26:06 +01: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 )
2014-01-29 14:26:06 +01:00
return 0 ;
2012-02-08 13:05:38 +00:00
return 1 ;
}
2014-01-29 14:27:13 +01:00
/*
* Validate given external origin could be used with thin pool
*/
int pool_supports_external_origin ( const struct lv_segment * pool_seg , const struct logical_volume * external_lv )
{
uint32_t csize = pool_seg - > chunk_size ;
if ( ( external_lv - > size < csize ) | | ( external_lv - > size % csize ) ) {
/* TODO: Validate with thin feature flag once, it will be supported */
log_error ( " Can't use \" %s/%s \" as external origin with \" %s/%s \" pool. "
" Size %s is not a multiple of pool's chunk size %s. " ,
external_lv - > vg - > name , external_lv - > name ,
pool_seg - > lv - > vg - > name , pool_seg - > lv - > name ,
display_size ( external_lv - > vg - > cmd , external_lv - > size ) ,
display_size ( external_lv - > vg - > cmd , csize ) ) ;
return 0 ;
}
return 1 ;
}
2014-01-08 10:27:17 +01:00
struct logical_volume * find_pool_lv ( const struct logical_volume * lv )
2013-06-11 12:32:01 +02:00
{
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-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
2014-07-22 22:20:18 +02:00
int update_thin_pool_params ( struct volume_group * vg ,
unsigned attr , int passed_args , uint32_t data_extents ,
uint64_t * pool_metadata_size ,
int * chunk_size_calc_method , uint32_t * chunk_size ,
thin_discards_t * discards , int * zero )
2013-08-06 11:42:40 +02:00
{
2014-07-22 22:20:18 +02:00
struct cmd_context * cmd = vg - > cmd ;
struct profile * profile = vg - > profile ;
uint32_t extent_size = vg - > extent_size ;
size_t estimate_chunk_size ;
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 ) ) {
2014-05-07 10:52:00 +02:00
if ( ! ( str = find_config_tree_str ( cmd , allocation_thin_pool_chunk_size_policy_CFG , profile ) ) ) {
2014-05-13 10:33:17 +02:00
log_error ( INTERNAL_ERROR " Could not find configuration. " ) ;
2014-05-07 10:52:00 +02:00
return 0 ;
}
2014-03-04 11:10:59 +01:00
if ( ! strcasecmp ( str , " generic " ) )
* chunk_size_calc_method = THIN_CHUNK_SIZE_CALC_METHOD_GENERIC ;
else if ( ! strcasecmp ( str , " performance " ) )
* chunk_size_calc_method = THIN_CHUNK_SIZE_CALC_METHOD_PERFORMANCE ;
else {
log_error ( " Thin pool chunk size calculation policy \" %s \" is unrecognised. " , str ) ;
return 0 ;
}
2014-07-22 22:20:18 +02:00
if ( ! ( * chunk_size = get_default_allocation_thin_pool_chunk_size_CFG ( cmd , profile ) ) )
return_0 ;
2013-09-25 16:00:52 +02:00
}
}
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 ) ) {
2014-05-07 10:52:00 +02:00
if ( ! ( str = find_config_tree_str ( cmd , allocation_thin_pool_discards_CFG , profile ) ) ) {
2014-05-13 10:33:17 +02:00
log_error ( INTERNAL_ERROR " Could not find configuration. " ) ;
2014-05-07 10:52:00 +02:00
return 0 ;
}
2013-09-25 16:00:52 +02:00
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 ) ;
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 ) ) ;
2014-07-22 22:20:18 +02:00
} else if ( * pool_metadata_size > ( DEFAULT_THIN_POOL_MAX_METADATA_SIZE * 2 ) ) {
2013-03-11 12:37:09 +01:00
/* Suggest bigger chunk size */
estimate_chunk_size = ( uint64_t ) data_extents * extent_size /
2014-07-22 22:20:18 +02:00
( DEFAULT_THIN_POOL_MAX_METADATA_SIZE * 2 * ( SECTOR_SIZE / UINT64_C ( 64 ) ) ) ;
2013-03-11 12:37:09 +01:00
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 {
2014-07-22 22:20:18 +02:00
estimate_chunk_size = ( uint64_t ) data_extents * extent_size /
2013-03-11 12:37:09 +01:00
( * pool_metadata_size * ( SECTOR_SIZE / UINT64_C ( 64 ) ) ) ;
2014-07-22 22:20:18 +02:00
if ( estimate_chunk_size < DM_THIN_MIN_DATA_BLOCK_SIZE )
estimate_chunk_size = DM_THIN_MIN_DATA_BLOCK_SIZE ;
else if ( estimate_chunk_size > DM_THIN_MAX_DATA_BLOCK_SIZE )
estimate_chunk_size = DM_THIN_MAX_DATA_BLOCK_SIZE ;
2013-03-11 12:37:09 +01:00
/* Check to eventually use bigger chunk size */
if ( ! ( passed_args & PASS_ARG_CHUNK_SIZE ) ) {
* chunk_size = estimate_chunk_size ;
2014-07-22 22:20:18 +02:00
log_verbose ( " Setting chunk size %s. " , display_size ( cmd , * chunk_size ) ) ;
2013-03-11 12:37:09 +01:00
} 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 ( * pool_metadata_size > ( 2 * DEFAULT_THIN_POOL_MAX_METADATA_SIZE ) ) {
2014-07-07 22:24:35 +02:00
* pool_metadata_size = 2 * DEFAULT_THIN_POOL_MAX_METADATA_SIZE ;
2013-03-11 12:37:09 +01:00
if ( passed_args & PASS_ARG_POOL_METADATA_SIZE )
log_warn ( " WARNING: Maximum supported pool metadata size is %s. " ,
2014-07-07 22:24:35 +02:00
display_size ( cmd , * pool_metadata_size ) ) ;
2013-03-11 12:37:09 +01:00
} else if ( * pool_metadata_size < ( 2 * DEFAULT_THIN_POOL_MIN_METADATA_SIZE ) ) {
2014-07-07 22:24:35 +02:00
* pool_metadata_size = 2 * DEFAULT_THIN_POOL_MIN_METADATA_SIZE ;
2013-03-11 12:37:09 +01:00
if ( passed_args & PASS_ARG_POOL_METADATA_SIZE )
log_warn ( " WARNING: Minimum supported pool metadata size is %s. " ,
2014-07-07 22:24:35 +02:00
display_size ( cmd , * pool_metadata_size ) ) ;
2013-03-11 12:37:09 +01:00
}
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
}