2002-11-18 17:04:08 +03:00
/*
2004-03-30 23:35:44 +04:00
* Copyright ( C ) 2002 - 2004 Sistina Software , Inc . All rights reserved .
* Copyright ( C ) 2004 Red Hat , Inc . All rights reserved .
2002-11-18 17:04:08 +03:00
*
2004-03-30 23:35:44 +04: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 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
2002-11-18 17:04:08 +03:00
*/
# include "lib.h"
# include "format-text.h"
# include "layout.h"
# include "label.h"
# include "xlate.h"
2004-05-05 01:25:57 +04:00
# include "lvmcache.h"
2002-11-18 17:04:08 +03:00
# include <sys/stat.h>
# include <fcntl.h>
2006-05-11 21:58:58 +04:00
static int _text_can_handle ( struct labeller * l __attribute ( ( unused ) ) ,
2006-12-01 02:11:42 +03:00
void * buf ,
2006-05-11 21:58:58 +04:00
uint64_t sector __attribute ( ( unused ) ) )
2002-11-18 17:04:08 +03:00
{
struct label_header * lh = ( struct label_header * ) buf ;
2006-05-10 01:23:51 +04:00
if ( ! strncmp ( ( char * ) lh - > type , LVM2_LABEL , sizeof ( lh - > type ) ) )
2002-11-18 17:04:08 +03:00
return 1 ;
return 0 ;
}
2006-12-01 02:11:42 +03:00
static int _text_write ( struct label * label , void * buf )
2002-11-18 17:04:08 +03:00
{
struct label_header * lh = ( struct label_header * ) buf ;
struct pv_header * pvhdr ;
2003-07-05 02:34:56 +04:00
struct lvmcache_info * info ;
2002-11-18 17:04:08 +03:00
struct disk_locn * pvh_dlocn_xl ;
struct metadata_area * mda ;
struct mda_context * mdac ;
struct data_area_list * da ;
/* FIXME Move to where label is created */
strncpy ( label - > type , LVM2_LABEL , sizeof ( label - > type ) ) ;
2006-05-10 01:23:51 +04:00
strncpy ( ( char * ) lh - > type , label - > type , sizeof ( label - > type ) ) ;
2002-11-18 17:04:08 +03:00
pvhdr = ( struct pv_header * ) ( ( void * ) buf + xlate32 ( lh - > offset_xl ) ) ;
2003-07-05 02:34:56 +04:00
info = ( struct lvmcache_info * ) label - > info ;
2002-11-18 17:04:08 +03:00
pvhdr - > device_size_xl = xlate64 ( info - > device_size ) ;
memcpy ( pvhdr - > pv_uuid , & info - > dev - > pvid , sizeof ( struct id ) ) ;
pvh_dlocn_xl = & pvhdr - > disk_areas_xl [ 0 ] ;
/* List of data areas (holding PEs) */
2005-06-01 20:51:55 +04:00
list_iterate_items ( da , & info - > das ) {
2002-11-18 17:04:08 +03:00
pvh_dlocn_xl - > offset = xlate64 ( da - > disk_locn . offset ) ;
pvh_dlocn_xl - > size = xlate64 ( da - > disk_locn . size ) ;
pvh_dlocn_xl + + ;
}
/* NULL-termination */
2003-08-20 19:48:27 +04:00
pvh_dlocn_xl - > offset = xlate64 ( UINT64_C ( 0 ) ) ;
pvh_dlocn_xl - > size = xlate64 ( UINT64_C ( 0 ) ) ;
2002-11-18 17:04:08 +03:00
pvh_dlocn_xl + + ;
/* List of metadata area header locations */
2005-06-01 20:51:55 +04:00
list_iterate_items ( mda , & info - > mdas ) {
2002-11-18 17:04:08 +03:00
mdac = ( struct mda_context * ) mda - > metadata_locn ;
if ( mdac - > area . dev ! = info - > dev )
continue ;
pvh_dlocn_xl - > offset = xlate64 ( mdac - > area . start ) ;
pvh_dlocn_xl - > size = xlate64 ( mdac - > area . size ) ;
pvh_dlocn_xl + + ;
}
/* NULL-termination */
2003-08-20 19:48:27 +04:00
pvh_dlocn_xl - > offset = xlate64 ( UINT64_C ( 0 ) ) ;
pvh_dlocn_xl - > size = xlate64 ( UINT64_C ( 0 ) ) ;
2002-11-18 17:04:08 +03:00
return 1 ;
}
2006-05-11 21:58:58 +04:00
int add_da ( struct dm_pool * mem , struct list * das ,
2002-11-18 17:04:08 +03:00
uint64_t start , uint64_t size )
{
struct data_area_list * dal ;
if ( ! mem ) {
2005-10-17 03:03:59 +04:00
if ( ! ( dal = dm_malloc ( sizeof ( * dal ) ) ) ) {
2002-11-18 17:04:08 +03:00
log_error ( " struct data_area_list allocation failed " ) ;
return 0 ;
}
} else {
2005-10-17 03:03:59 +04:00
if ( ! ( dal = dm_pool_alloc ( mem , sizeof ( * dal ) ) ) ) {
2002-11-18 17:04:08 +03:00
log_error ( " struct data_area_list allocation failed " ) ;
return 0 ;
}
}
dal - > disk_locn . offset = start ;
dal - > disk_locn . size = size ;
list_add ( das , & dal - > list ) ;
return 1 ;
}
void del_das ( struct list * das )
{
struct list * dah , * tmp ;
struct data_area_list * da ;
list_iterate_safe ( dah , tmp , das ) {
da = list_item ( dah , struct data_area_list ) ;
list_del ( & da - > list ) ;
2005-10-17 03:03:59 +04:00
dm_free ( da ) ;
2002-11-18 17:04:08 +03:00
}
}
2005-10-17 03:03:59 +04:00
int add_mda ( const struct format_type * fmt , struct dm_pool * mem , struct list * mdas ,
2002-11-18 17:04:08 +03:00
struct device * dev , uint64_t start , uint64_t size )
{
/* FIXME List size restricted by pv_header SECTOR_SIZE */
struct metadata_area * mdal ;
struct mda_lists * mda_lists = ( struct mda_lists * ) fmt - > private ;
struct mda_context * mdac ;
if ( ! mem ) {
2005-10-17 03:03:59 +04:00
if ( ! ( mdal = dm_malloc ( sizeof ( struct metadata_area ) ) ) ) {
2002-11-18 17:04:08 +03:00
log_error ( " struct mda_list allocation failed " ) ;
return 0 ;
}
2005-10-17 03:03:59 +04:00
if ( ! ( mdac = dm_malloc ( sizeof ( struct mda_context ) ) ) ) {
2002-11-18 17:04:08 +03:00
log_error ( " struct mda_context allocation failed " ) ;
2005-10-17 03:03:59 +04:00
dm_free ( mdal ) ;
2002-11-18 17:04:08 +03:00
return 0 ;
}
} else {
2005-10-17 03:03:59 +04:00
if ( ! ( mdal = dm_pool_alloc ( mem , sizeof ( struct metadata_area ) ) ) ) {
2002-11-18 17:04:08 +03:00
log_error ( " struct mda_list allocation failed " ) ;
return 0 ;
}
2005-10-17 03:03:59 +04:00
if ( ! ( mdac = dm_pool_alloc ( mem , sizeof ( struct mda_context ) ) ) ) {
2002-11-18 17:04:08 +03:00
log_error ( " struct mda_context allocation failed " ) ;
return 0 ;
}
}
mdal - > ops = mda_lists - > raw_ops ;
mdal - > metadata_locn = mdac ;
mdac - > area . dev = dev ;
mdac - > area . start = start ;
mdac - > area . size = size ;
memset ( & mdac - > rlocn , 0 , sizeof ( mdac - > rlocn ) ) ;
list_add ( mdas , & mdal - > list ) ;
return 1 ;
}
void del_mdas ( struct list * mdas )
{
struct list * mdah , * tmp ;
struct metadata_area * mda ;
list_iterate_safe ( mdah , tmp , mdas ) {
mda = list_item ( mdah , struct metadata_area ) ;
2005-10-17 03:03:59 +04:00
dm_free ( mda - > metadata_locn ) ;
2002-11-18 17:04:08 +03:00
list_del ( & mda - > list ) ;
2005-10-17 03:03:59 +04:00
dm_free ( mda ) ;
2002-11-18 17:04:08 +03:00
}
}
2006-05-11 21:58:58 +04:00
static int _text_initialise_label ( struct labeller * l __attribute ( ( unused ) ) ,
struct label * label )
2002-11-18 17:04:08 +03:00
{
strncpy ( label - > type , LVM2_LABEL , sizeof ( label - > type ) ) ;
return 1 ;
}
2006-12-01 02:11:42 +03:00
static int _text_read ( struct labeller * l , struct device * dev , void * buf ,
2002-11-18 17:04:08 +03:00
struct label * * label )
{
struct label_header * lh = ( struct label_header * ) buf ;
struct pv_header * pvhdr ;
2003-07-05 02:34:56 +04:00
struct lvmcache_info * info ;
2002-11-18 17:04:08 +03:00
struct disk_locn * dlocn_xl ;
uint64_t offset ;
struct metadata_area * mda ;
2006-04-11 17:55:59 +04:00
struct id vgid ;
2002-11-18 17:04:08 +03:00
struct mda_context * mdac ;
2006-04-11 17:55:59 +04:00
const char * vgname ;
2006-04-11 21:42:15 +04:00
uint32_t vgstatus ;
2006-04-13 21:32:24 +04:00
char * creation_host ;
2002-11-18 17:04:08 +03:00
pvhdr = ( struct pv_header * ) ( ( void * ) buf + xlate32 ( lh - > offset_xl ) ) ;
2006-05-10 01:23:51 +04:00
if ( ! ( info = lvmcache_add ( l , ( char * ) pvhdr - > pv_uuid , dev , NULL , NULL , 0 ) ) )
2006-04-11 17:55:59 +04:00
return_0 ;
2002-11-18 17:04:08 +03:00
* label = info - > label ;
info - > device_size = xlate64 ( pvhdr - > device_size_xl ) ;
if ( info - > das . n )
del_das ( & info - > das ) ;
list_init ( & info - > das ) ;
if ( info - > mdas . n )
del_mdas ( & info - > mdas ) ;
list_init ( & info - > mdas ) ;
/* Data areas holding the PEs */
dlocn_xl = pvhdr - > disk_areas_xl ;
while ( ( offset = xlate64 ( dlocn_xl - > offset ) ) ) {
2006-05-11 21:58:58 +04:00
add_da ( NULL , & info - > das , offset ,
2002-11-18 17:04:08 +03:00
xlate64 ( dlocn_xl - > size ) ) ;
dlocn_xl + + ;
}
/* Metadata area headers */
dlocn_xl + + ;
while ( ( offset = xlate64 ( dlocn_xl - > offset ) ) ) {
add_mda ( info - > fmt , NULL , & info - > mdas , dev , offset ,
xlate64 ( dlocn_xl - > size ) ) ;
dlocn_xl + + ;
}
2005-06-01 20:51:55 +04:00
list_iterate_items ( mda , & info - > mdas ) {
2002-11-18 17:04:08 +03:00
mdac = ( struct mda_context * ) mda - > metadata_locn ;
2006-04-11 17:55:59 +04:00
if ( ( vgname = vgname_from_mda ( info - > fmt , & mdac - > area ,
2006-04-13 21:32:24 +04:00
& vgid , & vgstatus , & creation_host ) ) & &
2006-04-11 21:42:15 +04:00
! lvmcache_update_vgname_and_id ( info , vgname ,
2006-04-13 21:32:24 +04:00
( char * ) & vgid , vgstatus ,
creation_host ) )
2006-04-11 20:00:26 +04:00
return_0 ;
2002-11-18 17:04:08 +03:00
}
info - > status & = ~ CACHE_INVALID ;
return 1 ;
}
2006-05-11 21:58:58 +04:00
static void _text_destroy_label ( struct labeller * l __attribute ( ( unused ) ) ,
struct label * label )
2002-11-18 17:04:08 +03:00
{
2003-07-05 02:34:56 +04:00
struct lvmcache_info * info = ( struct lvmcache_info * ) label - > info ;
2002-11-18 17:04:08 +03:00
if ( info - > mdas . n )
del_mdas ( & info - > mdas ) ;
if ( info - > das . n )
del_das ( & info - > das ) ;
}
2006-04-19 19:33:07 +04:00
static void _fmt_text_destroy ( struct labeller * l )
2002-11-18 17:04:08 +03:00
{
2005-10-17 03:03:59 +04:00
dm_free ( l ) ;
2002-11-18 17:04:08 +03:00
}
struct label_ops _text_ops = {
2006-05-10 01:23:51 +04:00
. can_handle = _text_can_handle ,
. write = _text_write ,
. read = _text_read ,
. verify = _text_can_handle ,
. initialise_label = _text_initialise_label ,
. destroy_label = _text_destroy_label ,
. destroy = _fmt_text_destroy ,
2002-11-18 17:04:08 +03:00
} ;
2002-12-20 02:25:55 +03:00
struct labeller * text_labeller_create ( const struct format_type * fmt )
2002-11-18 17:04:08 +03:00
{
struct labeller * l ;
2005-10-17 03:03:59 +04:00
if ( ! ( l = dm_malloc ( sizeof ( * l ) ) ) ) {
2002-11-18 17:04:08 +03:00
log_err ( " Couldn't allocate labeller object. " ) ;
return NULL ;
}
l - > ops = & _text_ops ;
2002-12-20 02:25:55 +03:00
l - > private = ( const void * ) fmt ;
2002-11-18 17:04:08 +03:00
return l ;
}