2004-05-04 21:25:57 +00:00
/*
* Copyright ( C ) 2003 - 2004 Sistina Software , Inc . All rights reserved .
* Copyright ( C ) 2004 Red Hat , Inc . All rights reserved .
*
* 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 General Public License v .2 .
*
* You should have received a copy of the GNU 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 "toolcontext.h"
# include "metadata.h"
2004-09-16 18:40:56 +00:00
# include "segtype.h"
2004-05-04 21:25:57 +00:00
# include "text_export.h"
# include "config.h"
2004-05-05 18:11:43 +00:00
# include "activate.h"
2006-10-03 17:55:20 +00:00
# include "str_list.h"
2004-05-04 21:25:57 +00:00
2006-04-19 15:33:07 +00:00
static const char * _snap_name ( const struct lv_segment * seg )
2004-05-04 21:25:57 +00:00
{
return seg - > segtype - > name ;
}
2006-04-19 15:33:07 +00:00
static int _snap_text_import ( struct lv_segment * seg , const struct config_node * sn ,
2006-05-11 17:58:58 +00:00
struct dm_hash_table * pv_hash __attribute ( ( unused ) ) )
2004-05-04 21:25:57 +00:00
{
2005-01-27 15:48:49 +00:00
uint32_t chunk_size ;
2004-05-04 21:25:57 +00:00
const char * org_name , * cow_name ;
struct logical_volume * org , * cow ;
seg - > lv - > status | = SNAPSHOT ;
if ( ! get_config_uint32 ( sn , " chunk_size " , & chunk_size ) ) {
log_error ( " Couldn't read chunk size for snapshot. " ) ;
return 0 ;
}
log_suppress ( 1 ) ;
if ( ! ( cow_name = find_config_str ( sn , " cow_store " , NULL ) ) ) {
log_suppress ( 0 ) ;
log_error ( " Snapshot cow storage not specified. " ) ;
return 0 ;
}
if ( ! ( org_name = find_config_str ( sn , " origin " , NULL ) ) ) {
log_suppress ( 0 ) ;
log_error ( " Snapshot origin not specified. " ) ;
return 0 ;
}
log_suppress ( 0 ) ;
if ( ! ( cow = find_lv ( seg - > lv - > vg , cow_name ) ) ) {
log_error ( " Unknown logical volume specified for "
" snapshot cow store. " ) ;
return 0 ;
}
if ( ! ( org = find_lv ( seg - > lv - > vg , org_name ) ) ) {
log_error ( " Unknown logical volume specified for "
" snapshot origin. " ) ;
return 0 ;
}
2005-04-07 12:39:44 +00:00
if ( ! vg_add_snapshot ( seg - > lv - > vg - > fid , seg - > lv - > name , org , cow ,
& seg - > lv - > lvid , seg - > len , chunk_size ) ) {
2004-05-04 21:25:57 +00:00
stack ;
return 0 ;
}
return 1 ;
}
2006-04-19 15:33:07 +00:00
static int _snap_text_export ( const struct lv_segment * seg , struct formatter * f )
2004-05-04 21:25:57 +00:00
{
outf ( f , " chunk_size = %u " , seg - > chunk_size ) ;
outf ( f , " origin = \" %s \" " , seg - > origin - > name ) ;
outf ( f , " cow_store = \" %s \" " , seg - > cow - > name ) ;
return 1 ;
}
2004-05-05 18:11:43 +00:00
# ifdef DEVMAPPER_SUPPORT
2006-05-11 17:58:58 +00:00
static int _snap_target_percent ( void * * target_state __attribute ( ( unused ) ) ,
struct dm_pool * mem __attribute ( ( unused ) ) ,
2006-05-16 20:53:13 +00:00
struct cmd_context * cmd __attribute ( ( unused ) ) ,
2006-05-11 17:58:58 +00:00
struct lv_segment * seg __attribute ( ( unused ) ) ,
2004-05-04 21:25:57 +00:00
char * params , uint64_t * total_numerator ,
uint64_t * total_denominator , float * percent )
{
float percent2 ;
uint64_t numerator , denominator ;
if ( index ( params , ' / ' ) ) {
if ( sscanf ( params , " % " PRIu64 " /% " PRIu64 ,
& numerator , & denominator ) = = 2 ) {
* total_numerator + = numerator ;
* total_denominator + = denominator ;
}
} else if ( sscanf ( params , " %f " , & percent2 ) = = 1 ) {
* percent + = percent2 ;
* percent / = 2 ;
}
return 1 ;
}
2006-10-18 18:01:53 +00:00
static int _snap_target_present ( const struct lv_segment * seg __attribute ( ( unused ) ) )
2004-05-05 18:11:43 +00:00
{
2006-04-19 15:33:07 +00:00
static int _snap_checked = 0 ;
static int _snap_present = 0 ;
2004-05-05 18:11:43 +00:00
2006-04-19 15:33:07 +00:00
if ( ! _snap_checked )
_snap_present = target_present ( " snapshot " , 1 ) & &
2005-11-08 22:52:26 +00:00
target_present ( " snapshot-origin " , 0 ) ;
2004-05-05 18:11:43 +00:00
2006-04-19 15:33:07 +00:00
_snap_checked = 1 ;
2004-05-05 18:11:43 +00:00
2006-04-19 15:33:07 +00:00
return _snap_present ;
2004-05-05 18:11:43 +00:00
}
# endif
2006-10-03 17:55:20 +00:00
static int _snap_modules_needed ( struct dm_pool * mem ,
const struct lv_segment * seg ,
struct list * modules )
{
if ( ! str_list_add ( mem , modules , " snapshot " ) ) {
log_error ( " snapshot string list allocation failed " ) ;
return 0 ;
}
return 1 ;
}
2006-04-19 15:33:07 +00:00
static void _snap_destroy ( const struct segment_type * segtype )
2004-05-04 21:25:57 +00:00
{
2006-05-11 17:58:58 +00:00
dm_free ( ( void * ) segtype ) ;
2004-05-04 21:25:57 +00:00
}
static struct segtype_handler _snapshot_ops = {
2006-05-09 21:23:51 +00:00
. name = _snap_name ,
. text_import = _snap_text_import ,
. text_export = _snap_text_export ,
2004-05-05 18:11:43 +00:00
# ifdef DEVMAPPER_SUPPORT
2006-05-09 21:23:51 +00:00
. target_percent = _snap_target_percent ,
. target_present = _snap_target_present ,
2004-05-05 18:11:43 +00:00
# endif
2006-10-03 17:55:20 +00:00
. modules_needed = _snap_modules_needed ,
2006-05-09 21:23:51 +00:00
. destroy = _snap_destroy ,
2004-05-04 21:25:57 +00:00
} ;
# ifdef SNAPSHOT_INTERNAL
struct segment_type * init_snapshot_segtype ( struct cmd_context * cmd )
# else /* Shared */
struct segment_type * init_segtype ( struct cmd_context * cmd ) ;
struct segment_type * init_segtype ( struct cmd_context * cmd )
# endif
{
2005-10-16 23:03:59 +00:00
struct segment_type * segtype = dm_malloc ( sizeof ( * segtype ) ) ;
2004-05-04 21:25:57 +00:00
if ( ! segtype ) {
stack ;
return NULL ;
}
segtype - > cmd = cmd ;
segtype - > ops = & _snapshot_ops ;
segtype - > name = " snapshot " ;
segtype - > private = NULL ;
2004-05-11 16:01:58 +00:00
segtype - > flags = SEG_SNAPSHOT ;
2004-05-04 21:25:57 +00:00
2004-09-14 17:37:51 +00:00
log_very_verbose ( " Initialised segtype: %s " , segtype - > name ) ;
2004-05-04 21:25:57 +00:00
return segtype ;
}