2001-10-09 18:26:45 +04:00
/*
* Copyright ( C ) 2001 Sistina Software ( UK ) Limited .
*
2001-10-31 20:59:52 +03:00
* Translates between disk and in - core formats .
*
2001-10-31 15:47:01 +03:00
* This file is released under the LGPL .
2001-10-09 18:26:45 +04:00
*/
# include "disk-rep.h"
# include "dbg_malloc.h"
# include "pool.h"
# include "hash.h"
# include "list.h"
# include "log.h"
2001-10-11 17:22:51 +04:00
# include <time.h>
# include <sys/utsname.h>
2001-10-10 13:25:04 +04:00
2001-10-09 18:26:45 +04:00
static int _check_vg_name ( const char * name )
{
return strlen ( name ) < NAME_LEN ;
}
/*
* Extracts the last part of a path .
*/
static char * _create_lv_name ( struct pool * mem , const char * full_name )
{
2001-10-11 14:08:44 +04:00
const char * ptr = strrchr ( full_name , ' / ' ) ;
if ( ! ptr )
ptr = full_name ;
else
ptr + + ;
2001-10-09 18:26:45 +04:00
return pool_strdup ( mem , ptr ) ;
}
static int _fill_lv_array ( struct logical_volume * * lvs ,
struct volume_group * vg , struct disk_list * dl )
{
2001-10-31 15:47:01 +03:00
struct list * lvh ;
2001-10-09 18:26:45 +04:00
struct logical_volume * lv ;
int i = 0 ;
2001-10-31 20:59:52 +03:00
list_iterate ( lvh , & dl - > lvds ) {
2001-10-31 15:47:01 +03:00
struct lvd_list * ll = list_item ( lvh , struct lvd_list ) ;
2001-10-09 18:26:45 +04:00
2001-10-31 20:59:52 +03:00
if ( ! ( lv = find_lv ( vg , ll - > lvd . lv_name ) ) ) {
2001-10-09 18:26:45 +04:00
stack ;
return 0 ;
}
lvs [ i ] = lv ;
i + + ;
}
return 1 ;
}
int import_pv ( struct pool * mem , struct device * dev ,
struct physical_volume * pv , struct pv_disk * pvd )
{
memset ( pv , 0 , sizeof ( * pv ) ) ;
2001-10-11 14:08:44 +04:00
memcpy ( & pv - > id , pvd - > pv_uuid , ID_LEN ) ;
2001-10-09 18:26:45 +04:00
pv - > dev = dev ;
if ( ! ( pv - > vg_name = pool_strdup ( mem , pvd - > vg_name ) ) ) {
stack ;
return 0 ;
}
if ( pvd - > pv_status & PV_ACTIVE )
pv - > status | = ACTIVE ;
if ( pvd - > pv_allocatable )
2001-10-09 21:20:02 +04:00
pv - > status | = ALLOCATED_PV ;
2001-10-09 18:26:45 +04:00
pv - > size = pvd - > pv_size ;
pv - > pe_size = pvd - > pe_size ;
pv - > pe_start = pvd - > pe_start ;
pv - > pe_count = pvd - > pe_total ;
pv - > pe_allocated = pvd - > pe_allocated ;
return 1 ;
}
2001-10-11 17:22:51 +04:00
static int _system_id ( char * system_id )
{
struct utsname uts ;
if ( uname ( & uts ) ! = 0 ) {
log_sys_error ( " uname " , " _system_id " ) ;
return 0 ;
}
sprintf ( system_id , " %s%lu " , uts . nodename , time ( NULL ) ) ;
return 1 ;
}
2001-10-09 18:26:45 +04:00
int export_pv ( struct pv_disk * pvd , struct physical_volume * pv )
{
memset ( pvd , 0 , sizeof ( * pvd ) ) ;
pvd - > id [ 0 ] = ' H ' ;
pvd - > id [ 1 ] = ' M ' ;
pvd - > version = 1 ;
2001-10-11 14:08:44 +04:00
memcpy ( pvd - > pv_uuid , pv - > id . uuid , ID_LEN ) ;
2001-10-09 18:26:45 +04:00
if ( ! _check_vg_name ( pv - > vg_name ) ) {
stack ;
return 0 ;
}
2001-10-10 14:05:29 +04:00
memset ( pvd - > vg_name , 0 , sizeof ( pvd - > vg_name ) ) ;
if ( pv - > vg_name )
2001-10-10 17:09:40 +04:00
strncpy ( pvd - > vg_name , pv - > vg_name , sizeof ( pvd - > vg_name ) ) ;
2001-10-09 18:26:45 +04:00
//pvd->pv_major = MAJOR(pv->dev);
if ( pv - > status & ACTIVE )
pvd - > pv_status | = PV_ACTIVE ;
2001-10-09 21:20:02 +04:00
if ( pv - > status & ALLOCATED_PV )
2001-10-11 17:34:17 +04:00
pvd - > pv_allocatable = PV_ALLOCATABLE ;
2001-10-09 18:26:45 +04:00
pvd - > pv_size = pv - > size ;
pvd - > lv_cur = 0 ; /* this is set when exporting the lv list */
pvd - > pe_size = pv - > pe_size ;
pvd - > pe_total = pv - > pe_count ;
pvd - > pe_allocated = pv - > pe_allocated ;
pvd - > pe_start = pv - > pe_start ;
2001-10-11 17:22:51 +04:00
if ( ! _system_id ( pvd - > system_id ) ) {
stack ;
return 0 ;
}
2001-10-09 18:26:45 +04:00
return 1 ;
}
int import_vg ( struct pool * mem ,
struct volume_group * vg , struct disk_list * dl )
{
2001-10-31 20:59:52 +03:00
struct vg_disk * vgd = & dl - > vgd ;
2001-10-11 14:08:44 +04:00
memcpy ( vg - > id . uuid , vgd - > vg_uuid , ID_LEN ) ;
2001-10-09 18:26:45 +04:00
2001-10-31 20:59:52 +03:00
if ( ! _check_vg_name ( dl - > pvd . vg_name ) ) {
2001-10-09 18:26:45 +04:00
stack ;
return 0 ;
}
2001-10-31 20:59:52 +03:00
if ( ! ( vg - > name = pool_strdup ( mem , dl - > pvd . vg_name ) ) ) {
2001-10-09 18:26:45 +04:00
stack ;
return 0 ;
}
if ( vgd - > vg_status & VG_ACTIVE )
vg - > status | = ACTIVE ;
if ( vgd - > vg_status & VG_EXPORTED )
vg - > status | = EXPORTED_VG ;
if ( vgd - > vg_status & VG_EXTENDABLE )
vg - > status | = EXTENDABLE_VG ;
if ( vgd - > vg_access & VG_READ )
vg - > status | = LVM_READ ;
if ( vgd - > vg_access & VG_WRITE )
vg - > status | = LVM_WRITE ;
if ( vgd - > vg_access & VG_CLUSTERED )
vg - > status | = CLUSTERED ;
if ( vgd - > vg_access & VG_SHARED )
vg - > status | = SHARED ;
vg - > extent_size = vgd - > pe_size ;
vg - > extent_count = vgd - > pe_total ;
vg - > free_count = vgd - > pe_total - vgd - > pe_allocated ;
vg - > max_lv = vgd - > lv_max ;
vg - > max_pv = vgd - > pv_max ;
return 1 ;
}
int export_vg ( struct vg_disk * vgd , struct volume_group * vg )
{
memset ( vgd , 0 , sizeof ( * vgd ) ) ;
2001-10-11 14:08:44 +04:00
memcpy ( vgd - > vg_uuid , vg - > id . uuid , ID_LEN ) ;
2001-10-09 18:26:45 +04:00
2001-10-11 19:09:31 +04:00
if ( vg - > status & LVM_READ )
2001-10-09 18:26:45 +04:00
vgd - > vg_access | = VG_READ ;
if ( vg - > status & LVM_WRITE )
vgd - > vg_access | = VG_WRITE ;
if ( vg - > status & CLUSTERED )
vgd - > vg_access | = VG_CLUSTERED ;
if ( vg - > status & SHARED )
vgd - > vg_access | = VG_SHARED ;
if ( vg - > status & ACTIVE )
vgd - > vg_status | = VG_ACTIVE ;
if ( vg - > status & EXPORTED_VG )
vgd - > vg_status | = VG_EXPORTED ;
if ( vg - > status & EXTENDABLE_VG )
vgd - > vg_status | = VG_EXTENDABLE ;
vgd - > lv_max = vg - > max_lv ;
vgd - > lv_cur = vg - > lv_count ;
vgd - > pv_max = vg - > max_pv ;
vgd - > pv_cur = vg - > pv_count ;
vgd - > pe_size = vg - > extent_size ;
vgd - > pe_total = vg - > extent_count ;
vgd - > pe_allocated = vg - > extent_count - vg - > free_count ;
2001-10-11 17:22:51 +04:00
2001-10-09 18:26:45 +04:00
return 1 ;
}
int import_lv ( struct pool * mem , struct logical_volume * lv , struct lv_disk * lvd )
{
int len ;
2001-10-11 14:08:44 +04:00
memset ( & lv - > id , 0 , sizeof ( lv - > id ) ) ;
2001-10-09 18:26:45 +04:00
if ( ! ( lv - > name = _create_lv_name ( mem , lvd - > lv_name ) ) ) {
stack ;
return 0 ;
}
if ( lvd - > lv_status & LV_ACTIVE )
lv - > status | = ACTIVE ;
if ( lvd - > lv_status & LV_SPINDOWN )
lv - > status | = SPINDOWN_LV ;
if ( lvd - > lv_access & LV_READ )
lv - > status | = LVM_READ ;
if ( lvd - > lv_access & LV_WRITE )
lv - > status | = LVM_WRITE ;
if ( lvd - > lv_access & LV_SNAPSHOT )
lv - > status | = SNAPSHOT ;
if ( lvd - > lv_access & LV_SNAPSHOT_ORG )
lv - > status | = SNAPSHOT_ORG ;
if ( lvd - > lv_badblock )
lv - > status | = BADBLOCK_ON ;
2001-11-12 20:55:05 +03:00
if ( lvd - > lv_allocation & LV_STRICT )
2001-10-09 18:26:45 +04:00
lv - > status | = ALLOC_STRICT ;
2001-11-12 20:55:05 +03:00
if ( lvd - > lv_allocation & LV_CONTIGUOUS )
2001-10-09 18:26:45 +04:00
lv - > status | = ALLOC_CONTIGUOUS ;
2001-11-12 20:55:05 +03:00
else
lv - > status | = ALLOC_SIMPLE ;
2001-10-09 18:26:45 +04:00
2001-10-29 18:34:56 +03:00
lv - > read_ahead = lvd - > lv_read_ahead ;
lv - > stripes = lvd - > lv_stripes ;
2001-10-09 18:26:45 +04:00
lv - > size = lvd - > lv_size ;
lv - > le_count = lvd - > lv_allocated_le ;
len = sizeof ( struct pe_specifier ) * lv - > le_count ;
if ( ! ( lv - > map = pool_alloc ( mem , len ) ) ) {
stack ;
return 0 ;
}
memset ( lv - > map , 0 , len ) ;
return 1 ;
}
void export_lv ( struct lv_disk * lvd , struct volume_group * vg ,
2001-11-14 16:52:38 +03:00
struct logical_volume * lv , const char * dev_dir )
2001-10-09 18:26:45 +04:00
{
memset ( lvd , 0 , sizeof ( * lvd ) ) ;
2001-10-29 18:28:00 +03:00
snprintf ( lvd - > lv_name , sizeof ( lvd - > lv_name ) , " %s%s/%s " ,
2001-11-14 16:52:38 +03:00
dev_dir , vg - > name , lv - > name ) ;
2001-10-09 18:26:45 +04:00
2001-10-29 18:28:00 +03:00
/* FIXME: Add 'if' test */
2001-10-09 18:26:45 +04:00
_check_vg_name ( vg - > name ) ;
strcpy ( lvd - > vg_name , vg - > name ) ;
if ( lv - > status & LVM_READ )
lvd - > lv_access | = LV_READ ;
if ( lv - > status & LVM_WRITE )
lvd - > lv_access | = LV_WRITE ;
if ( lv - > status & SNAPSHOT )
lvd - > lv_access | = LV_SNAPSHOT ;
if ( lv - > status & SNAPSHOT_ORG )
lvd - > lv_access | = LV_SNAPSHOT_ORG ;
if ( lv - > status & ACTIVE )
lvd - > lv_status | = LV_ACTIVE ;
if ( lv - > status & SPINDOWN_LV )
lvd - > lv_status | = LV_SPINDOWN ;
2001-10-29 18:34:56 +03:00
lvd - > lv_read_ahead = lv - > read_ahead ;
lvd - > lv_stripes = lv - > stripes ;
2001-10-09 18:26:45 +04:00
lvd - > lv_size = lv - > size ;
lvd - > lv_allocated_le = lv - > le_count ;
if ( lv - > status & BADBLOCK_ON )
lvd - > lv_badblock = LV_BADBLOCK_ON ;
if ( lv - > status & ALLOC_STRICT )
2001-11-12 20:55:05 +03:00
lvd - > lv_allocation | = LV_STRICT ;
if ( lv - > status & ALLOC_CONTIGUOUS )
lvd - > lv_allocation | = LV_CONTIGUOUS ;
2001-10-09 18:26:45 +04:00
}
2001-11-14 16:52:38 +03:00
int import_extents ( struct pool * mem , struct volume_group * vg , struct list * pvds )
2001-10-09 18:26:45 +04:00
{
struct disk_list * dl ;
struct logical_volume * lv , * lvs [ MAX_LV ] ;
struct physical_volume * pv ;
struct pe_disk * e ;
int i ;
uint32_t lv_num , le ;
2001-11-14 16:52:38 +03:00
struct list * pvdh ;
2001-10-09 18:26:45 +04:00
2001-11-14 16:52:38 +03:00
list_iterate ( pvdh , pvds ) {
dl = list_item ( pvdh , struct disk_list ) ;
2001-10-09 18:26:45 +04:00
pv = _find_pv ( vg , dl - > dev ) ;
e = dl - > extents ;
/* build an array of lv's for this pv */
if ( ! _fill_lv_array ( lvs , vg , dl ) ) {
stack ;
return 0 ;
}
2001-10-31 20:59:52 +03:00
for ( i = 0 ; i < dl - > pvd . pe_total ; i + + ) {
2001-10-09 18:26:45 +04:00
lv_num = e [ i ] . lv_num ;
if ( lv_num = = UNMAPPED_EXTENT )
2001-10-09 21:36:48 +04:00
continue ;
2001-10-09 18:26:45 +04:00
2001-10-31 20:59:52 +03:00
else if ( lv_num > dl - > pvd . lv_cur ) {
2001-10-09 18:26:45 +04:00
log_err ( " invalid lv in extent map \n " ) ;
return 0 ;
} else {
lv_num - - ;
lv = lvs [ lv_num ] ;
le = e [ i ] . le_num ;
2001-11-07 18:02:07 +03:00
if ( le > = lv - > le_count ) {
log_err ( " logical extent number "
" out of bounds " ) ;
return 0 ;
}
2001-10-09 18:26:45 +04:00
lv - > map [ le ] . pv = pv ;
lv - > map [ le ] . pe = i ;
}
}
}
return 1 ;
}
int export_extents ( struct disk_list * dl , int lv_num ,
struct logical_volume * lv ,
struct physical_volume * pv )
{
struct pe_disk * ped ;
2001-10-10 18:56:53 +04:00
int le ;
2001-10-09 18:26:45 +04:00
for ( le = 0 ; le < lv - > le_count ; le + + ) {
if ( lv - > map [ le ] . pv = = pv ) {
ped = & dl - > extents [ lv - > map [ le ] . pe ] ;
ped - > lv_num = lv_num ;
ped - > le_num = le ;
}
}
return 1 ;
}
2001-11-14 16:52:38 +03:00
int import_pvs ( struct pool * mem , struct list * pvds ,
2001-10-31 15:47:01 +03:00
struct list * results , int * count )
2001-10-09 18:26:45 +04:00
{
2001-11-14 16:52:38 +03:00
struct list * pvdh ;
2001-10-09 18:26:45 +04:00
struct disk_list * dl ;
struct pv_list * pvl ;
* count = 0 ;
2001-11-14 16:52:38 +03:00
list_iterate ( pvdh , pvds ) {
dl = list_item ( pvdh , struct disk_list ) ;
2001-10-09 18:26:45 +04:00
pvl = pool_alloc ( mem , sizeof ( * pvl ) ) ;
if ( ! pvl ) {
stack ;
return 0 ;
}
2001-10-31 20:59:52 +03:00
if ( ! import_pv ( mem , dl - > dev , & pvl - > pv , & dl - > pvd ) ) {
2001-10-09 18:26:45 +04:00
stack ;
return 0 ;
}
2001-10-31 15:47:01 +03:00
list_add ( results , & pvl - > list ) ;
2001-10-09 18:26:45 +04:00
( * count ) + + ;
}
return 1 ;
}
static struct logical_volume * _add_lv ( struct pool * mem ,
struct volume_group * vg ,
struct lv_disk * lvd )
{
2001-10-29 18:28:00 +03:00
struct lv_list * ll = pool_zalloc ( mem , sizeof ( * ll ) ) ;
2001-10-09 18:26:45 +04:00
struct logical_volume * lv ;
if ( ! ll ) {
stack ;
return NULL ;
}
lv = & ll - > lv ;
if ( ! import_lv ( mem , & ll - > lv , lvd ) ) {
stack ;
return NULL ;
}
2001-10-31 15:47:01 +03:00
list_add ( & vg - > lvs , & ll - > list ) ;
2001-10-31 20:59:52 +03:00
lv - > vg = vg ;
2001-10-09 18:26:45 +04:00
vg - > lv_count + + ;
return lv ;
}
int import_lvs ( struct pool * mem , struct volume_group * vg ,
2001-11-14 16:52:38 +03:00
struct list * pvds )
2001-10-09 18:26:45 +04:00
{
struct disk_list * dl ;
struct lvd_list * ll ;
struct lv_disk * lvd ;
2001-11-14 16:52:38 +03:00
struct list * pvdh , * lvdh ;
2001-10-09 18:26:45 +04:00
2001-11-14 16:52:38 +03:00
list_iterate ( pvdh , pvds ) {
dl = list_item ( pvdh , struct disk_list ) ;
list_iterate ( lvdh , & dl - > lvds ) {
ll = list_item ( lvdh , struct lvd_list ) ;
2001-10-31 20:59:52 +03:00
lvd = & ll - > lvd ;
2001-10-09 18:26:45 +04:00
2001-10-29 16:52:23 +03:00
if ( ! find_lv ( vg , lvd - > lv_name ) & &
2001-10-09 18:26:45 +04:00
! _add_lv ( mem , vg , lvd ) ) {
stack ;
return 0 ;
}
}
}
return 1 ;
}
int export_lvs ( struct disk_list * dl , struct volume_group * vg ,
2001-11-14 16:52:38 +03:00
struct physical_volume * pv , const char * dev_dir )
2001-10-09 18:26:45 +04:00
{
2001-10-31 15:47:01 +03:00
struct list * lvh ;
2001-10-09 18:26:45 +04:00
struct lv_list * ll ;
struct lvd_list * lvdl ;
2001-11-13 21:52:52 +03:00
int lv_num = 0 , len ;
2001-10-10 18:56:53 +04:00
/*
* setup the pv ' s extents array
*/
2001-10-31 20:59:52 +03:00
len = sizeof ( struct pe_disk ) * dl - > pvd . pe_total ;
2001-10-10 18:56:53 +04:00
if ( ! ( dl - > extents = pool_alloc ( dl - > mem , len ) ) ) {
stack ;
return 0 ;
}
2001-10-11 14:08:44 +04:00
memset ( dl - > extents , 0 , len ) ;
2001-10-10 18:56:53 +04:00
2001-10-09 18:26:45 +04:00
2001-10-31 15:47:01 +03:00
list_iterate ( lvh , & vg - > lvs ) {
ll = list_item ( lvh , struct lv_list ) ;
2001-10-09 18:26:45 +04:00
if ( ! ( lvdl = pool_alloc ( dl - > mem , sizeof ( * lvdl ) ) ) ) {
stack ;
return 0 ;
}
2001-11-14 16:52:38 +03:00
export_lv ( & lvdl - > lvd , vg , & ll - > lv , dev_dir ) ;
2001-11-10 01:01:04 +03:00
lvdl - > lvd . lv_number = lv_num ;
2001-11-13 21:52:52 +03:00
if ( ! export_extents ( dl , lv_num + 1 , & ll - > lv , pv ) ) {
2001-10-09 18:26:45 +04:00
stack ;
return 0 ;
}
2001-10-31 20:59:52 +03:00
list_add ( & dl - > lvds , & lvdl - > list ) ;
dl - > pvd . lv_cur + + ;
2001-11-13 21:52:52 +03:00
lv_num + + ;
2001-10-09 18:26:45 +04:00
}
return 1 ;
}
int export_uuids ( struct disk_list * dl , struct volume_group * vg )
{
struct uuid_list * ul ;
struct pv_list * pvl ;
2001-10-31 15:47:01 +03:00
struct list * pvh ;
2001-10-09 18:26:45 +04:00
2001-10-31 15:47:01 +03:00
list_iterate ( pvh , & vg - > pvs ) {
pvl = list_item ( pvh , struct pv_list ) ;
2001-10-09 18:26:45 +04:00
if ( ! ( ul = pool_alloc ( dl - > mem , sizeof ( * ul ) ) ) ) {
stack ;
return 0 ;
}
2001-10-11 14:08:44 +04:00
memset ( ul - > uuid , 0 , sizeof ( ul - > uuid ) ) ;
memcpy ( ul - > uuid , pvl - > pv . id . uuid , ID_LEN ) ;
2001-10-09 18:26:45 +04:00
2001-10-31 15:47:01 +03:00
list_add ( & dl - > uuids , & ul - > list ) ;
2001-10-09 18:26:45 +04:00
}
return 1 ;
}
2001-10-11 14:55:19 +04:00
/*
* This calculates the nasty pv_number and
* lv_number fields used by LVM1 . Very
* inefficient code .
*/
2001-11-14 16:52:38 +03:00
void export_numbers ( struct list * pvds , struct volume_group * vg )
2001-10-11 14:55:19 +04:00
{
2001-11-14 16:52:38 +03:00
struct list * pvdh ;
2001-10-11 14:55:19 +04:00
struct disk_list * dl ;
2001-10-11 20:31:09 +04:00
int pv_num = 1 ;
2001-10-11 14:55:19 +04:00
2001-11-14 16:52:38 +03:00
list_iterate ( pvdh , pvds ) {
dl = list_item ( pvdh , struct disk_list ) ;
2001-10-31 20:59:52 +03:00
dl - > pvd . pv_number = pv_num + + ;
2001-10-11 14:55:19 +04:00
}
}
2001-10-11 17:05:55 +04:00
/*
* Calculate vg_disk - > pv_act .
*/
2001-11-14 16:52:38 +03:00
void export_pv_act ( struct list * pvds )
2001-10-11 17:05:55 +04:00
{
2001-11-14 16:52:38 +03:00
struct list * pvdh ;
2001-10-11 17:05:55 +04:00
struct disk_list * dl ;
int act = 0 ;
2001-11-14 16:52:38 +03:00
list_iterate ( pvdh , pvds ) {
dl = list_item ( pvdh , struct disk_list ) ;
2001-10-31 20:59:52 +03:00
if ( dl - > pvd . pv_status & PV_ACTIVE )
2001-10-11 17:05:55 +04:00
act + + ;
}
2001-11-14 16:52:38 +03:00
list_iterate ( pvdh , pvds ) {
dl = list_item ( pvdh , struct disk_list ) ;
2001-10-31 20:59:52 +03:00
dl - > vgd . pv_act = act ;
2001-10-11 17:05:55 +04:00
}
}
2001-10-11 18:10:18 +04:00
2001-11-14 16:52:38 +03:00
int export_vg_number ( struct list * pvds , const char * vg_name ,
2001-10-11 18:10:18 +04:00
struct dev_filter * filter )
{
2001-11-14 16:52:38 +03:00
struct list * pvdh ;
2001-10-11 18:10:18 +04:00
struct disk_list * dl ;
int vg_num ;
if ( ! get_free_vg_number ( filter , vg_name , & vg_num ) ) {
stack ;
return 0 ;
}
2001-11-14 16:52:38 +03:00
list_iterate ( pvdh , pvds ) {
dl = list_item ( pvdh , struct disk_list ) ;
2001-10-31 20:59:52 +03:00
dl - > vgd . vg_number = vg_num ;
2001-10-11 18:10:18 +04:00
}
return 1 ;
}