2001-10-04 21:48:55 +04:00
/*
* Copyright ( C ) 2001 Sistina Software ( UK ) Limited .
*
2001-10-31 15:47:01 +03:00
* This file is released under the LGPL .
2001-10-04 21:48:55 +04:00
*/
2002-11-18 17:01:16 +03:00
# include "lib.h"
2001-10-04 21:48:55 +04:00
# include "disk-rep.h"
# include "pool.h"
2001-10-08 13:45:16 +04:00
# include "xlate.h"
2002-01-24 16:36:33 +03:00
# include "filter.h"
2002-11-18 17:01:16 +03:00
# include "cache.h"
2001-10-04 21:48:55 +04:00
2001-11-14 13:01:52 +03:00
# include <sys/stat.h>
# include <fcntl.h>
2002-01-24 16:36:33 +03:00
# include <linux/kdev_t.h>
2001-11-14 13:01:52 +03:00
2001-10-05 19:48:05 +04:00
# define fail do {stack; return 0;} while(0)
2001-10-04 21:48:55 +04:00
# define xx16(v) disk->v = xlate16(disk->v)
# define xx32(v) disk->v = xlate32(disk->v)
# define xx64(v) disk->v = xlate64(disk->v)
/*
* Functions to perform the endian conversion
* between disk and core . The same code works
* both ways of course .
*/
2001-10-31 20:59:52 +03:00
static void _xlate_pvd ( struct pv_disk * disk )
2001-10-04 21:48:55 +04:00
{
2001-12-13 03:07:29 +03:00
xx16 ( version ) ;
xx32 ( pv_on_disk . base ) ;
xx32 ( pv_on_disk . size ) ;
xx32 ( vg_on_disk . base ) ;
xx32 ( vg_on_disk . size ) ;
xx32 ( pv_uuidlist_on_disk . base ) ;
xx32 ( pv_uuidlist_on_disk . size ) ;
xx32 ( lv_on_disk . base ) ;
xx32 ( lv_on_disk . size ) ;
xx32 ( pe_on_disk . base ) ;
xx32 ( pe_on_disk . size ) ;
xx32 ( pv_major ) ;
xx32 ( pv_number ) ;
xx32 ( pv_status ) ;
xx32 ( pv_allocatable ) ;
xx32 ( pv_size ) ;
xx32 ( lv_cur ) ;
xx32 ( pe_size ) ;
xx32 ( pe_total ) ;
xx32 ( pe_allocated ) ;
2001-10-04 21:48:55 +04:00
xx32 ( pe_start ) ;
}
2001-10-31 20:59:52 +03:00
static void _xlate_lvd ( struct lv_disk * disk )
2001-10-04 21:48:55 +04:00
{
2001-12-13 03:07:29 +03:00
xx32 ( lv_access ) ;
xx32 ( lv_status ) ;
xx32 ( lv_open ) ;
xx32 ( lv_dev ) ;
xx32 ( lv_number ) ;
xx32 ( lv_mirror_copies ) ;
xx32 ( lv_recovery ) ;
xx32 ( lv_schedule ) ;
xx32 ( lv_size ) ;
xx32 ( lv_snapshot_minor ) ;
xx16 ( lv_chunk_size ) ;
xx16 ( dummy ) ;
xx32 ( lv_allocated_le ) ;
xx32 ( lv_stripes ) ;
xx32 ( lv_stripesize ) ;
xx32 ( lv_badblock ) ;
xx32 ( lv_allocation ) ;
xx32 ( lv_io_timeout ) ;
xx32 ( lv_read_ahead ) ;
2001-10-04 21:48:55 +04:00
}
2001-10-31 20:59:52 +03:00
static void _xlate_vgd ( struct vg_disk * disk )
2001-10-04 21:48:55 +04:00
{
2001-12-13 03:07:29 +03:00
xx32 ( vg_number ) ;
xx32 ( vg_access ) ;
xx32 ( vg_status ) ;
xx32 ( lv_max ) ;
xx32 ( lv_cur ) ;
xx32 ( lv_open ) ;
xx32 ( pv_max ) ;
xx32 ( pv_cur ) ;
xx32 ( pv_act ) ;
xx32 ( dummy ) ;
xx32 ( vgda ) ;
xx32 ( pe_size ) ;
xx32 ( pe_total ) ;
xx32 ( pe_allocated ) ;
xx32 ( pvg_total ) ;
2001-10-04 21:48:55 +04:00
}
2002-12-20 02:25:55 +03:00
static void _xlate_extents ( struct pe_disk * extents , uint32_t count )
2001-10-04 21:48:55 +04:00
{
2001-10-05 19:20:40 +04:00
int i ;
for ( i = 0 ; i < count ; i + + ) {
extents [ i ] . lv_num = xlate16 ( extents [ i ] . lv_num ) ;
extents [ i ] . le_num = xlate16 ( extents [ i ] . le_num ) ;
2001-10-04 21:48:55 +04:00
}
2001-10-05 19:20:40 +04:00
}
2001-10-04 21:48:55 +04:00
2001-10-10 13:36:29 +04:00
/*
* Handle both minor metadata formats .
*/
static int _munge_formats ( struct pv_disk * pvd )
{
uint32_t pe_start ;
switch ( pvd - > version ) {
case 1 :
2001-10-11 14:08:44 +04:00
pvd - > pe_start = ( ( pvd - > pe_on_disk . base +
2002-11-18 17:01:16 +03:00
pvd - > pe_on_disk . size ) > > SECTOR_SHIFT ) ;
2001-10-10 13:36:29 +04:00
break ;
case 2 :
pvd - > version = 1 ;
2002-11-18 17:01:16 +03:00
pe_start = pvd - > pe_start < < SECTOR_SHIFT ;
2001-10-10 13:36:29 +04:00
pvd - > pe_on_disk . size = pe_start - pvd - > pe_on_disk . base ;
break ;
default :
return 0 ;
}
return 1 ;
}
2002-11-18 17:01:16 +03:00
static int _read_pvd ( struct device * dev , struct pv_disk * pvd )
2001-10-05 19:20:40 +04:00
{
2002-12-20 02:25:55 +03:00
if ( dev_read ( dev , __UINT64_C ( 0 ) , sizeof ( * pvd ) , pvd ) ! = sizeof ( * pvd ) ) {
2002-04-24 22:20:51 +04:00
log_very_verbose ( " Failed to read PV data from %s " ,
2002-01-28 00:30:47 +03:00
dev_name ( dev ) ) ;
2002-01-16 21:10:08 +03:00
return 0 ;
}
2001-10-31 20:59:52 +03:00
_xlate_pvd ( pvd ) ;
2001-10-10 13:36:29 +04:00
2002-01-16 21:10:08 +03:00
if ( pvd - > id [ 0 ] ! = ' H ' | | pvd - > id [ 1 ] ! = ' M ' ) {
2002-11-18 17:01:16 +03:00
log_very_verbose ( " %s does not have a valid LVM1 PV identifier " ,
2002-01-16 21:10:08 +03:00
dev_name ( dev ) ) ;
return 0 ;
}
if ( ! _munge_formats ( pvd ) ) {
2002-11-18 17:01:16 +03:00
log_very_verbose ( " format1: Unknown metadata version %d "
" found on %s " , pvd - > version , dev_name ( dev ) ) ;
2002-01-16 21:10:08 +03:00
return 0 ;
}
2001-10-10 16:45:20 +04:00
return 1 ;
2001-10-04 21:48:55 +04:00
}
2002-12-20 02:25:55 +03:00
static int _read_lvd ( struct device * dev , uint64_t pos , struct lv_disk * disk )
2001-10-04 21:48:55 +04:00
{
2001-10-05 19:48:05 +04:00
if ( dev_read ( dev , pos , sizeof ( * disk ) , disk ) ! = sizeof ( * disk ) )
fail ;
2001-10-31 20:59:52 +03:00
_xlate_lvd ( disk ) ;
2001-10-05 19:20:40 +04:00
2001-10-04 21:48:55 +04:00
return 1 ;
}
2001-10-31 20:59:52 +03:00
static int _read_vgd ( struct disk_list * data )
2001-10-04 21:48:55 +04:00
{
2001-10-31 20:59:52 +03:00
struct vg_disk * vgd = & data - > vgd ;
2002-12-20 02:25:55 +03:00
uint64_t pos = data - > pvd . vg_on_disk . base ;
2001-10-05 19:48:05 +04:00
if ( dev_read ( data - > dev , pos , sizeof ( * vgd ) , vgd ) ! = sizeof ( * vgd ) )
fail ;
2001-10-31 20:59:52 +03:00
_xlate_vgd ( vgd ) ;
2001-10-04 21:48:55 +04:00
return 1 ;
}
2001-10-05 19:20:40 +04:00
static int _read_uuids ( struct disk_list * data )
2001-10-04 21:48:55 +04:00
{
int num_read = 0 ;
struct uuid_list * ul ;
char buffer [ NAME_LEN ] ;
2002-12-20 02:25:55 +03:00
uint64_t pos = data - > pvd . pv_uuidlist_on_disk . base ;
uint64_t end = pos + data - > pvd . pv_uuidlist_on_disk . size ;
2001-10-04 21:48:55 +04:00
2001-12-13 03:07:29 +03:00
while ( pos < end & & num_read < data - > vgd . pv_cur ) {
2001-10-05 19:20:40 +04:00
if ( dev_read ( data - > dev , pos , sizeof ( buffer ) , buffer ) ! =
2001-10-05 19:48:05 +04:00
sizeof ( buffer ) )
fail ;
2001-10-04 21:48:55 +04:00
2001-10-05 19:48:05 +04:00
if ( ! ( ul = pool_alloc ( data - > mem , sizeof ( * ul ) ) ) )
fail ;
2001-10-04 21:48:55 +04:00
memcpy ( ul - > uuid , buffer , NAME_LEN ) ;
2001-11-28 02:12:06 +03:00
ul - > uuid [ NAME_LEN - 1 ] = ' \0 ' ;
2001-10-04 21:48:55 +04:00
2001-10-31 15:47:01 +03:00
list_add ( & data - > uuids , & ul - > list ) ;
2001-10-04 21:48:55 +04:00
pos + = NAME_LEN ;
num_read + + ;
}
2001-10-05 19:20:40 +04:00
2001-10-04 21:48:55 +04:00
return 1 ;
}
2001-11-13 21:52:52 +03:00
static inline int _check_lvd ( struct lv_disk * lvd )
2001-10-11 14:08:44 +04:00
{
2001-11-13 21:52:52 +03:00
return ! ( lvd - > lv_name [ 0 ] = = ' \0 ' ) ;
2001-10-11 14:08:44 +04:00
}
2001-10-05 19:20:40 +04:00
static int _read_lvs ( struct disk_list * data )
2001-10-04 21:48:55 +04:00
{
2002-12-20 02:25:55 +03:00
unsigned int i , read = 0 ;
uint64_t pos ;
2001-10-08 13:45:16 +04:00
struct lvd_list * ll ;
2001-11-13 21:52:52 +03:00
struct vg_disk * vgd = & data - > vgd ;
2001-10-04 21:48:55 +04:00
2001-12-13 03:07:29 +03:00
for ( i = 0 ; ( i < vgd - > lv_max ) & & ( read < vgd - > lv_cur ) ; i + + ) {
2001-10-31 20:59:52 +03:00
pos = data - > pvd . lv_on_disk . base + ( i * sizeof ( struct lv_disk ) ) ;
2001-10-08 13:45:16 +04:00
ll = pool_alloc ( data - > mem , sizeof ( * ll ) ) ;
2001-10-04 21:48:55 +04:00
2001-10-05 19:48:05 +04:00
if ( ! ll )
fail ;
2001-10-04 21:48:55 +04:00
2001-10-31 20:59:52 +03:00
if ( ! _read_lvd ( data - > dev , pos , & ll - > lvd ) )
2001-10-05 19:48:05 +04:00
fail ;
2001-10-04 21:48:55 +04:00
2001-10-31 20:59:52 +03:00
if ( ! _check_lvd ( & ll - > lvd ) )
2001-11-13 21:52:52 +03:00
continue ;
2001-10-11 14:08:44 +04:00
2001-11-13 21:52:52 +03:00
read + + ;
2001-10-31 20:59:52 +03:00
list_add ( & data - > lvds , & ll - > list ) ;
2001-10-04 21:48:55 +04:00
}
return 1 ;
}
2001-10-05 19:20:40 +04:00
static int _read_extents ( struct disk_list * data )
2001-10-04 21:48:55 +04:00
{
2001-10-31 20:59:52 +03:00
size_t len = sizeof ( struct pe_disk ) * data - > pvd . pe_total ;
2001-10-04 21:48:55 +04:00
struct pe_disk * extents = pool_alloc ( data - > mem , len ) ;
2002-12-20 02:25:55 +03:00
uint64_t pos = data - > pvd . pe_on_disk . base ;
2001-10-04 21:48:55 +04:00
2001-10-05 19:48:05 +04:00
if ( ! extents )
fail ;
if ( dev_read ( data - > dev , pos , len , extents ) ! = len )
fail ;
2001-10-04 21:48:55 +04:00
2001-10-31 20:59:52 +03:00
_xlate_extents ( extents , data - > pvd . pe_total ) ;
2001-10-04 21:48:55 +04:00
data - > extents = extents ;
2001-10-05 19:20:40 +04:00
2001-10-04 21:48:55 +04:00
return 1 ;
}
2002-01-29 20:23:33 +03:00
/*
* If exported , remove " PV_EXP " from end of VG name
*/
static void _munge_exported_vg ( struct disk_list * data )
{
2002-12-20 02:25:55 +03:00
int l ;
size_t s ;
2002-01-29 20:23:33 +03:00
/* Return if PV not in a VG or VG not exported */
2002-04-24 22:20:51 +04:00
if ( ( ! * data - > pvd . vg_name ) | | ! ( data - > vgd . vg_status & VG_EXPORTED ) )
2002-01-29 20:23:33 +03:00
return ;
l = strlen ( data - > pvd . vg_name ) ;
s = sizeof ( EXPORTED_TAG ) ;
if ( ! strncmp ( data - > pvd . vg_name + l - s + 1 , EXPORTED_TAG , s ) )
2002-04-24 22:20:51 +04:00
data - > pvd . vg_name [ l - s + 1 ] = ' \0 ' ;
2002-01-29 20:23:33 +03:00
data - > pvd . pv_status | = VG_EXPORTED ;
}
2002-12-20 02:25:55 +03:00
static struct disk_list * __read_disk ( const struct format_type * fmt ,
2002-04-24 22:20:51 +04:00
struct device * dev , struct pool * mem ,
2001-11-14 13:01:52 +03:00
const char * vg_name )
2001-10-04 21:48:55 +04:00
{
2002-04-24 22:20:51 +04:00
struct disk_list * dl = pool_alloc ( mem , sizeof ( * dl ) ) ;
2001-10-25 18:04:18 +04:00
const char * name = dev_name ( dev ) ;
2002-11-18 17:01:16 +03:00
struct cache_info * info ;
2001-10-25 18:04:18 +04:00
2002-04-24 22:20:51 +04:00
if ( ! dl ) {
2001-11-12 19:00:52 +03:00
stack ;
return NULL ;
}
2002-04-24 22:20:51 +04:00
dl - > dev = dev ;
dl - > mem = mem ;
list_init ( & dl - > uuids ) ;
list_init ( & dl - > lvds ) ;
2001-10-05 19:20:40 +04:00
2002-11-18 17:01:16 +03:00
if ( ! _read_pvd ( dev , & dl - > pvd ) ) {
2002-01-16 21:10:08 +03:00
stack ;
2001-10-10 16:45:20 +04:00
goto bad ;
}
2002-11-18 17:01:16 +03:00
if ( ! ( info = cache_add ( fmt - > labeller , dl - > pvd . pv_uuid , dev ,
dl - > pvd . vg_name , NULL ) ) )
stack ;
else {
info - > device_size = xlate32 ( dl - > pvd . pv_size ) < < SECTOR_SHIFT ;
list_init ( & info - > mdas ) ;
info - > status & = ~ CACHE_INVALID ;
}
2001-10-10 16:28:10 +04:00
/*
* is it an orphan ?
*/
2002-04-24 22:20:51 +04:00
if ( ! * dl - > pvd . vg_name ) {
log_very_verbose ( " %s is not a member of any format1 VG " , name ) ;
2002-01-29 20:23:33 +03:00
/* Update VG cache */
2002-11-18 17:01:16 +03:00
/* vgcache_add(dl->pvd.vg_name, NULL, dev, fmt); */
2002-01-29 20:23:33 +03:00
2002-04-24 22:20:51 +04:00
return ( vg_name ) ? NULL : dl ;
2001-10-10 16:45:20 +04:00
}
2001-10-10 16:28:10 +04:00
2002-04-24 22:20:51 +04:00
if ( ! _read_vgd ( dl ) ) {
2002-01-29 20:23:33 +03:00
log_error ( " Failed to read VG data from PV (%s) " , name ) ;
2001-10-05 19:20:40 +04:00
goto bad ;
2001-10-04 21:48:55 +04:00
}
2002-01-29 20:23:33 +03:00
/* If VG is exported, set VG name back to the real name */
2002-04-24 22:20:51 +04:00
_munge_exported_vg ( dl ) ;
2002-01-29 20:23:33 +03:00
/* Update VG cache with what we found */
2002-11-18 17:01:16 +03:00
/* vgcache_add(dl->pvd.vg_name, dl->vgd.vg_uuid, dev, fmt); */
2002-01-29 20:23:33 +03:00
2002-04-24 22:20:51 +04:00
if ( vg_name & & strcmp ( vg_name , dl - > pvd . vg_name ) ) {
2002-01-29 20:23:33 +03:00
log_very_verbose ( " %s is not a member of the VG %s " ,
name , vg_name ) ;
2001-10-05 19:20:40 +04:00
goto bad ;
2001-10-04 21:48:55 +04:00
}
2002-04-24 22:20:51 +04:00
if ( ! _read_uuids ( dl ) ) {
2001-10-25 18:04:18 +04:00
log_error ( " Failed to read PV uuid list from %s " , name ) ;
2001-10-05 19:20:40 +04:00
goto bad ;
2001-10-04 21:48:55 +04:00
}
2002-04-24 22:20:51 +04:00
if ( ! _read_lvs ( dl ) ) {
2001-10-25 18:04:18 +04:00
log_error ( " Failed to read LV's from %s " , name ) ;
2001-10-05 19:20:40 +04:00
goto bad ;
2001-10-04 21:48:55 +04:00
}
2002-04-24 22:20:51 +04:00
if ( ! _read_extents ( dl ) ) {
2001-10-25 18:04:18 +04:00
log_error ( " Failed to read extents from %s " , name ) ;
2001-10-05 19:20:40 +04:00
goto bad ;
2001-10-04 21:48:55 +04:00
}
2002-04-24 22:20:51 +04:00
log_very_verbose ( " Found %s in %sVG %s " , name ,
( dl - > vgd . vg_status & VG_EXPORTED ) ? " exported " : " " ,
dl - > pvd . vg_name ) ;
2001-10-29 16:52:23 +03:00
2002-04-24 22:20:51 +04:00
return dl ;
2001-10-05 19:20:40 +04:00
2001-12-13 03:07:29 +03:00
bad :
2002-04-24 22:20:51 +04:00
pool_free ( dl - > mem , dl ) ;
2001-10-05 19:20:40 +04:00
return NULL ;
2001-10-04 21:48:55 +04:00
}
2002-12-20 02:25:55 +03:00
struct disk_list * read_disk ( const struct format_type * fmt , struct device * dev ,
2002-04-24 22:20:51 +04:00
struct pool * mem , const char * vg_name )
2001-11-14 13:01:52 +03:00
{
struct disk_list * r ;
if ( ! dev_open ( dev , O_RDONLY ) ) {
stack ;
return NULL ;
}
2002-04-24 22:20:51 +04:00
r = __read_disk ( fmt , dev , mem , vg_name ) ;
2001-11-14 13:01:52 +03:00
if ( ! dev_close ( dev ) )
stack ;
return r ;
}
2002-01-24 16:36:33 +03:00
static void _add_pv_to_list ( struct list * head , struct disk_list * data )
{
struct list * pvdh ;
struct pv_disk * pvd ;
list_iterate ( pvdh , head ) {
pvd = & list_item ( pvdh , struct disk_list ) - > pvd ;
2002-04-24 22:20:51 +04:00
if ( ! strncmp ( data - > pvd . pv_uuid , pvd - > pv_uuid ,
2002-01-24 16:36:33 +03:00
sizeof ( pvd - > pv_uuid ) ) ) {
if ( MAJOR ( data - > dev - > dev ) ! = md_major ( ) ) {
2002-01-28 00:30:47 +03:00
log_very_verbose ( " Ignoring duplicate PV %s on "
2002-04-24 22:20:51 +04:00
" %s " , pvd - > pv_uuid ,
2002-01-28 00:30:47 +03:00
dev_name ( data - > dev ) ) ;
2002-01-24 16:36:33 +03:00
return ;
}
2002-04-24 22:20:51 +04:00
log_very_verbose ( " Duplicate PV %s - using md %s " ,
pvd - > pv_uuid , dev_name ( data - > dev ) ) ;
2002-01-24 16:36:33 +03:00
list_del ( pvdh ) ;
break ;
}
}
list_add ( head , & data - > list ) ;
}
2001-10-04 21:48:55 +04:00
/*
2002-01-10 21:12:26 +03:00
* Build a list of pv_d ' s structures , allocated from mem .
2001-12-13 03:07:29 +03:00
* We keep track of the first object allocated form the pool
* so we can free off all the memory if something goes wrong .
2001-10-04 21:48:55 +04:00
*/
2002-12-20 02:25:55 +03:00
int read_pvs_in_vg ( const struct format_type * fmt , const char * vg_name ,
2002-04-24 22:20:51 +04:00
struct dev_filter * filter , struct pool * mem ,
struct list * head )
2001-10-04 21:48:55 +04:00
{
2001-12-13 18:08:58 +03:00
struct dev_iter * iter ;
2001-10-04 21:48:55 +04:00
struct device * dev ;
struct disk_list * data = NULL ;
2002-11-18 17:01:16 +03:00
struct list * vgih ;
struct cache_vginfo * vginfo ;
2001-12-13 03:07:29 +03:00
/* Fast path if we already saw this VG and cached the list of PVs */
2002-11-18 17:01:16 +03:00
if ( vg_name & & ( vginfo = vginfo_from_vgname ( vg_name ) ) & &
vginfo - > infos . n ) {
list_iterate ( vgih , & vginfo - > infos ) {
dev = list_item ( vgih , struct cache_info ) - > dev ;
if ( dev & & ! ( data = read_disk ( fmt , dev , mem , vg_name ) ) )
2002-01-16 17:43:27 +03:00
break ;
2002-01-24 16:36:33 +03:00
_add_pv_to_list ( head , data ) ;
2001-12-13 03:07:29 +03:00
}
/* Did we find the whole VG? */
if ( ! vg_name | | ! * vg_name | |
( data & & * data - > pvd . vg_name & &
2002-11-18 17:01:16 +03:00
list_size ( head ) = = data - > vgd . pv_cur ) )
return 1 ;
2001-12-13 03:07:29 +03:00
2002-11-18 17:01:16 +03:00
/* Failed */
2001-12-13 03:07:29 +03:00
list_init ( head ) ;
2002-11-18 17:01:16 +03:00
/* vgcache_del(vg_name); */
2001-12-13 03:07:29 +03:00
}
2001-12-13 18:08:58 +03:00
if ( ! ( iter = dev_iter_create ( filter ) ) ) {
log_error ( " read_pvs_in_vg: dev_iter_create failed " ) ;
return 0 ;
}
2001-12-13 03:07:29 +03:00
/* Otherwise do a complete scan */
2001-10-04 21:48:55 +04:00
for ( dev = dev_iter_get ( iter ) ; dev ; dev = dev_iter_get ( iter ) ) {
2002-04-24 22:20:51 +04:00
if ( ( data = read_disk ( fmt , dev , mem , vg_name ) ) ) {
2002-01-24 16:36:33 +03:00
_add_pv_to_list ( head , data ) ;
2001-12-13 03:07:29 +03:00
}
2001-10-05 19:20:40 +04:00
}
2001-10-08 13:45:16 +04:00
dev_iter_destroy ( iter ) ;
2001-10-16 20:25:28 +04:00
if ( list_empty ( head ) )
return 0 ;
2001-10-05 19:20:40 +04:00
return 1 ;
}
2001-10-31 20:59:52 +03:00
static int _write_vgd ( struct disk_list * data )
2001-10-05 19:20:40 +04:00
{
2001-10-31 20:59:52 +03:00
struct vg_disk * vgd = & data - > vgd ;
2002-12-20 02:25:55 +03:00
uint64_t pos = data - > pvd . vg_on_disk . base ;
2001-10-05 19:20:40 +04:00
2001-10-31 20:59:52 +03:00
_xlate_vgd ( vgd ) ;
2001-10-05 19:48:05 +04:00
if ( dev_write ( data - > dev , pos , sizeof ( * vgd ) , vgd ) ! = sizeof ( * vgd ) )
fail ;
2001-10-31 20:59:52 +03:00
_xlate_vgd ( vgd ) ;
2001-10-05 19:20:40 +04:00
return 1 ;
}
static int _write_uuids ( struct disk_list * data )
{
struct uuid_list * ul ;
2001-10-31 15:47:01 +03:00
struct list * uh ;
2002-12-20 02:25:55 +03:00
uint64_t pos = data - > pvd . pv_uuidlist_on_disk . base ;
uint64_t end = pos + data - > pvd . pv_uuidlist_on_disk . size ;
2001-10-05 19:20:40 +04:00
2001-10-31 15:47:01 +03:00
list_iterate ( uh , & data - > uuids ) {
2001-10-05 19:20:40 +04:00
if ( pos > = end ) {
2001-10-10 16:45:20 +04:00
log_error ( " Too many uuids to fit on %s " ,
2001-10-25 18:04:18 +04:00
dev_name ( data - > dev ) ) ;
2001-10-05 19:20:40 +04:00
return 0 ;
}
2001-10-31 15:47:01 +03:00
ul = list_item ( uh , struct uuid_list ) ;
2001-10-10 17:24:16 +04:00
if ( dev_write ( data - > dev , pos , NAME_LEN , ul - > uuid ) ! = NAME_LEN )
2001-10-05 19:48:05 +04:00
fail ;
2001-10-05 19:20:40 +04:00
pos + = NAME_LEN ;
}
2001-10-04 21:48:55 +04:00
2001-10-05 19:20:40 +04:00
return 1 ;
}
2001-10-04 21:48:55 +04:00
2002-12-20 02:25:55 +03:00
static int _write_lvd ( struct device * dev , uint64_t pos , struct lv_disk * disk )
2001-10-05 19:20:40 +04:00
{
2001-10-31 20:59:52 +03:00
_xlate_lvd ( disk ) ;
2001-10-05 19:48:05 +04:00
if ( dev_write ( dev , pos , sizeof ( * disk ) , disk ) ! = sizeof ( * disk ) )
fail ;
2001-10-31 20:59:52 +03:00
_xlate_lvd ( disk ) ;
2001-10-05 19:20:40 +04:00
2001-10-04 21:48:55 +04:00
return 1 ;
2001-10-05 19:20:40 +04:00
}
2001-10-04 21:48:55 +04:00
2001-10-05 19:20:40 +04:00
static int _write_lvs ( struct disk_list * data )
{
2001-10-31 15:47:01 +03:00
struct list * lvh ;
2002-12-20 02:25:55 +03:00
uint64_t pos , offset ;
2001-10-05 19:20:40 +04:00
2001-11-10 01:01:04 +03:00
pos = data - > pvd . lv_on_disk . base ;
2001-11-13 21:52:52 +03:00
if ( ! dev_zero ( data - > dev , pos , data - > pvd . lv_on_disk . size ) ) {
2001-11-14 16:52:38 +03:00
log_error ( " Couldn't zero lv area on device '%s' " ,
2001-12-13 03:07:29 +03:00
dev_name ( data - > dev ) ) ;
2001-11-13 21:52:52 +03:00
return 0 ;
}
2001-10-31 20:59:52 +03:00
list_iterate ( lvh , & data - > lvds ) {
2001-10-31 15:47:01 +03:00
struct lvd_list * ll = list_item ( lvh , struct lvd_list ) ;
2001-10-05 19:20:40 +04:00
2002-05-23 15:37:51 +04:00
offset = sizeof ( struct lv_disk ) * ll - > lvd . lv_number ;
2002-11-18 17:01:16 +03:00
if ( offset + sizeof ( struct lv_disk ) > data - > pvd . lv_on_disk . size ) {
2002-05-23 15:37:51 +04:00
log_error ( " lv_number %d too large " , ll - > lvd . lv_number ) ;
return 0 ;
}
2001-10-05 19:20:40 +04:00
2002-05-23 15:37:51 +04:00
if ( ! _write_lvd ( data - > dev , pos + offset , & ll - > lvd ) )
fail ;
2001-10-05 19:20:40 +04:00
}
return 1 ;
}
static int _write_extents ( struct disk_list * data )
{
2001-10-31 20:59:52 +03:00
size_t len = sizeof ( struct pe_disk ) * data - > pvd . pe_total ;
2001-10-05 19:20:40 +04:00
struct pe_disk * extents = data - > extents ;
2002-12-20 02:25:55 +03:00
uint64_t pos = data - > pvd . pe_on_disk . base ;
2001-10-05 19:20:40 +04:00
2001-10-31 20:59:52 +03:00
_xlate_extents ( extents , data - > pvd . pe_total ) ;
2001-10-11 14:08:44 +04:00
if ( dev_write ( data - > dev , pos , len , extents ) ! = len )
2001-10-05 19:48:05 +04:00
fail ;
2001-10-31 20:59:52 +03:00
_xlate_extents ( extents , data - > pvd . pe_total ) ;
2001-10-05 19:20:40 +04:00
return 1 ;
}
2001-10-31 20:59:52 +03:00
static int _write_pvd ( struct disk_list * data )
2001-10-05 19:20:40 +04:00
{
2002-01-29 18:52:11 +03:00
char * buf ;
2002-12-20 02:25:55 +03:00
uint64_t pos = data - > pvd . pv_on_disk . base ;
size_t size = data - > pvd . pv_on_disk . size ;
2001-10-05 19:20:40 +04:00
2002-04-24 22:20:51 +04:00
if ( size < sizeof ( struct pv_disk ) ) {
2002-01-29 20:23:33 +03:00
log_error ( " Invalid PV structure size. " ) ;
2002-01-29 18:52:11 +03:00
return 0 ;
}
2001-10-05 19:48:05 +04:00
2002-04-24 22:20:51 +04:00
/* Make sure that the gap between the PV structure and
2002-01-29 18:52:11 +03:00
the next one is zeroed in order to make non LVM tools
happy ( idea from AED ) */
buf = dbg_malloc ( size ) ;
2002-04-24 22:20:51 +04:00
if ( ! buf ) {
2002-01-29 18:52:11 +03:00
log_err ( " Couldn't allocate temporary PV buffer. " ) ;
return 0 ;
}
memset ( buf , 0 , size ) ;
memcpy ( buf , & data - > pvd , sizeof ( struct pv_disk ) ) ;
2002-04-24 22:20:51 +04:00
_xlate_pvd ( ( struct pv_disk * ) buf ) ;
2002-01-29 18:52:11 +03:00
if ( dev_write ( data - > dev , pos , size , buf ) ! = size ) {
dbg_free ( buf ) ;
fail ;
}
2001-10-05 19:20:40 +04:00
2002-01-29 18:52:11 +03:00
dbg_free ( buf ) ;
2001-10-05 19:20:40 +04:00
return 1 ;
}
2001-11-14 13:01:52 +03:00
/*
* assumes the device has been opened .
*/
2002-12-20 02:25:55 +03:00
static int __write_all_pvd ( const struct format_type * fmt ,
struct disk_list * data )
2001-10-05 19:20:40 +04:00
{
2001-10-25 18:04:18 +04:00
const char * pv_name = dev_name ( data - > dev ) ;
2001-10-05 19:20:40 +04:00
2001-10-31 20:59:52 +03:00
if ( ! _write_pvd ( data ) ) {
2001-10-10 16:45:20 +04:00
log_error ( " Failed to write PV structure onto %s " , pv_name ) ;
2001-10-05 19:20:40 +04:00
return 0 ;
}
2002-11-18 17:01:16 +03:00
/* vgcache_add(data->pvd.vg_name, data->vgd.vg_uuid, data->dev, fmt); */
2001-10-10 14:05:29 +04:00
/*
* Stop here for orphan pv ' s .
*/
2002-03-05 23:03:09 +03:00
if ( data - > pvd . vg_name [ 0 ] = = ' \0 ' ) {
2002-11-18 17:01:16 +03:00
/* if (!test_mode())
vgcache_add ( data - > pvd . vg_name , NULL , data - > dev , fmt ) ; */
2001-10-10 14:05:29 +04:00
return 1 ;
2002-03-05 23:03:09 +03:00
}
2002-11-18 17:01:16 +03:00
/* if (!test_mode())
vgcache_add ( data - > pvd . vg_name , data - > vgd . vg_uuid , data - > dev ,
fmt ) ; */
2001-10-10 14:05:29 +04:00
2001-10-31 20:59:52 +03:00
if ( ! _write_vgd ( data ) ) {
2001-10-10 16:45:20 +04:00
log_error ( " Failed to write VG data to %s " , pv_name ) ;
2001-10-05 19:20:40 +04:00
return 0 ;
}
if ( ! _write_uuids ( data ) ) {
2001-10-10 16:45:20 +04:00
log_error ( " Failed to write PV uuid list to %s " , pv_name ) ;
2001-10-05 19:20:40 +04:00
return 0 ;
}
if ( ! _write_lvs ( data ) ) {
2001-10-10 16:45:20 +04:00
log_error ( " Failed to write LV's to %s " , pv_name ) ;
2001-10-05 19:20:40 +04:00
return 0 ;
}
if ( ! _write_extents ( data ) ) {
2001-10-10 16:45:20 +04:00
log_error ( " Failed to write extents to %s " , pv_name ) ;
2001-10-05 19:20:40 +04:00
return 0 ;
}
return 1 ;
}
2001-11-14 13:01:52 +03:00
/*
* opens the device and hands to the above fn .
*/
2002-12-20 02:25:55 +03:00
static int _write_all_pvd ( const struct format_type * fmt , struct disk_list * data )
2001-11-14 13:01:52 +03:00
{
int r ;
if ( ! dev_open ( data - > dev , O_WRONLY ) ) {
stack ;
return 0 ;
}
2002-04-24 22:20:51 +04:00
r = __write_all_pvd ( fmt , data ) ;
2001-11-14 13:01:52 +03:00
if ( ! dev_close ( data - > dev ) )
stack ;
return r ;
}
2001-10-05 19:20:40 +04:00
/*
* Writes all the given pv ' s to disk . Does very
* little sanity checking , so make sure correct
2001-10-05 19:48:05 +04:00
* data is passed to here .
*/
2002-12-20 02:25:55 +03:00
int write_disks ( const struct format_type * fmt , struct list * pvs )
2001-10-05 19:20:40 +04:00
{
2001-10-31 15:47:01 +03:00
struct list * pvh ;
2001-10-05 19:20:40 +04:00
struct disk_list * dl ;
2001-10-31 15:47:01 +03:00
list_iterate ( pvh , pvs ) {
dl = list_item ( pvh , struct disk_list ) ;
2002-04-24 22:20:51 +04:00
if ( ! ( _write_all_pvd ( fmt , dl ) ) )
2001-10-05 19:48:05 +04:00
fail ;
2002-04-24 22:20:51 +04:00
log_very_verbose ( " Successfully wrote data to %s " ,
2002-01-28 00:30:47 +03:00
dev_name ( dl - > dev ) ) ;
2001-10-05 19:20:40 +04:00
}
return 1 ;
2001-10-04 21:48:55 +04:00
}