2001-11-06 10:29:56 +00:00
/*
* Copyright ( C ) 2001 Sistina Software
*
* This file is released under the LGPL .
*/
# include "metadata.h"
# include "pv_map.h"
# include "log.h"
2001-11-29 18:45:35 +00:00
# include "dbg_malloc.h"
2001-11-06 10:29:56 +00:00
2001-11-27 16:37:33 +00:00
# include <assert.h>
/*
* These functions adjust the pe counts in pv ' s
* after we ' ve added or removed segments .
*/
static void _get_extents ( struct stripe_segment * seg )
{
int s , count ;
struct physical_volume * pv ;
2001-11-27 17:39:15 +00:00
for ( s = 0 ; s < seg - > stripes ; s + + ) {
pv = seg - > area [ s % seg - > stripes ] . pv ;
2001-11-27 16:37:33 +00:00
count = seg - > len / seg - > stripes ;
pv - > pe_allocated + = count ;
}
}
static void _put_extents ( struct stripe_segment * seg )
{
int s , count ;
struct physical_volume * pv ;
2001-11-27 17:39:15 +00:00
for ( s = 0 ; s < seg - > stripes ; s + + ) {
pv = seg - > area [ s % seg - > stripes ] . pv ;
2001-11-27 16:37:33 +00:00
count = seg - > len / seg - > stripes ;
assert ( pv - > pe_allocated > = count ) ;
pv - > pe_allocated - = count ;
}
}
2001-11-29 18:45:35 +00:00
static struct stripe_segment * _alloc_segment ( struct pool * mem , int stripes )
{
struct stripe_segment * seg ;
uint32_t len = sizeof ( * seg ) + ( stripes * sizeof ( seg - > area [ 0 ] ) ) ;
if ( ! ( seg = pool_zalloc ( mem , len ) ) ) {
stack ;
return NULL ;
}
return seg ;
}
static int _alloc_stripe_area ( struct logical_volume * lv , int stripes ,
struct pv_area * * areas , uint32_t * index )
{
uint32_t count = lv - > le_count - * index ;
uint32_t per_area = count / stripes ;
uint32_t smallest = areas [ stripes - 1 ] - > count ;
uint32_t s ;
struct stripe_segment * seg ;
if ( smallest < per_area )
per_area = smallest ;
if ( ! ( seg = _alloc_segment ( lv - > vg - > cmd - > mem , stripes ) ) ) {
log_err ( " Couldn't allocate new stripe segment. " ) ;
return 0 ;
}
seg - > lv = lv ;
seg - > le = * index ;
seg - > len = per_area * stripes ;
seg - > stripes = stripes ;
for ( s = 0 ; s < stripes ; s + + ) {
struct pv_area * pva = areas [ s ] ;
seg - > area [ s ] . pv = pva - > map - > pv ;
seg - > area [ s ] . pe = pva - > start ;
consume_pv_area ( pva , per_area ) ;
}
list_add ( & lv - > segments , & seg - > list ) ;
* index + = seg - > len ;
return 1 ;
}
static int _comp_area ( const void * l , const void * r )
{
struct pv_area * lhs = * ( ( struct pv_area * * ) l ) ;
struct pv_area * rhs = * ( ( struct pv_area * * ) r ) ;
if ( lhs - > count < rhs - > count )
2001-11-30 09:19:46 +00:00
return 1 ;
2001-11-29 18:45:35 +00:00
else if ( lhs - > count > rhs - > count )
2001-11-30 09:19:46 +00:00
return - 1 ;
2001-11-29 18:45:35 +00:00
return 0 ;
}
static int _alloc_striped ( struct logical_volume * lv ,
struct list * pvms , uint32_t allocated ,
uint32_t stripes , uint32_t stripe_size )
{
2001-11-30 09:19:46 +00:00
int r = 0 ;
2001-11-29 18:45:35 +00:00
struct list * pvmh ;
struct pv_area * * areas ;
int pv_count = 0 , index ;
struct pv_map * pvm ;
size_t len ;
list_iterate ( pvmh , pvms )
pv_count + + ;
/* allocate an array of pv_areas, one candidate per pv */
len = sizeof ( * areas ) * pv_count ;
if ( ! ( areas = dbg_malloc ( sizeof ( * areas ) * pv_count ) ) ) {
log_err ( " Couldn't allocate areas array. " ) ;
return 0 ;
}
while ( allocated ! = lv - > le_count ) {
index = 0 ;
list_iterate ( pvmh , pvms ) {
pvm = list_item ( pvmh , struct pv_map ) ;
if ( list_empty ( & pvm - > areas ) )
continue ;
areas [ index + + ] = list_item ( pvm - > areas . n ,
struct pv_area ) ;
}
2001-11-30 09:19:46 +00:00
if ( index < stripes ) {
log_error ( " Insufficient free extents (suitable for "
" striping) to allocate logical volume "
" %s: %u required " ,
lv - > name , lv - > le_count ) ;
goto out ;
}
2001-11-29 18:45:35 +00:00
/* sort the areas so we allocate from the biggest */
qsort ( areas , index , sizeof ( * areas ) , _comp_area ) ;
if ( ! _alloc_stripe_area ( lv , stripes , areas , & allocated ) ) {
stack ;
2001-11-30 09:19:46 +00:00
goto out ;
2001-11-29 18:45:35 +00:00
}
}
2001-11-30 09:19:46 +00:00
r = 1 ;
2001-11-29 18:45:35 +00:00
2001-11-30 09:19:46 +00:00
out :
dbg_free ( areas ) ;
return r ;
2001-11-29 18:45:35 +00:00
}
2001-11-06 10:29:56 +00:00
/*
* The heart of the allocation code . This
* function takes a pv_area and allocates it to
* the lv . If the lv doesn ' t need the complete
* area then the area is split , otherwise the area
* is unlinked from the pv_map .
*/
2001-11-27 16:37:33 +00:00
static int _alloc_linear_area ( struct logical_volume * lv , uint32_t * index ,
2001-11-29 18:45:35 +00:00
struct pv_map * map , struct pv_area * pva )
2001-11-06 10:29:56 +00:00
{
2001-11-29 18:45:35 +00:00
uint32_t count , remaining ;
2001-11-27 16:37:33 +00:00
struct stripe_segment * seg ;
2001-11-06 10:29:56 +00:00
count = pva - > count ;
2001-11-27 16:37:33 +00:00
remaining = lv - > le_count - * index ;
2001-11-29 18:45:35 +00:00
if ( count < remaining )
2001-11-06 10:29:56 +00:00
count = remaining ;
2001-11-29 18:45:35 +00:00
if ( ! ( seg = _alloc_segment ( lv - > vg - > cmd - > mem , 1 ) ) ) {
2001-11-27 16:37:33 +00:00
log_err ( " Couldn't allocate new stripe segment. " ) ;
return 0 ;
2001-11-06 10:29:56 +00:00
}
2001-11-27 16:37:33 +00:00
seg - > lv = lv ;
seg - > le = * index ;
seg - > len = count ;
seg - > stripe_size = 0 ;
seg - > stripes = 1 ;
2001-11-29 18:45:35 +00:00
seg - > area [ 0 ] . pv = map - > pv ;
seg - > area [ 0 ] . pe = pva - > start ;
2001-11-27 16:37:33 +00:00
list_add ( & lv - > segments , & seg - > list ) ;
2001-11-29 18:45:35 +00:00
consume_pv_area ( pva , count ) ;
2001-11-27 16:37:33 +00:00
* index + = count ;
return 1 ;
2001-11-06 10:29:56 +00:00
}
2001-11-06 10:55:01 +00:00
/*
* Only one area per pv is allowed , so we search
* for the biggest area , or the first area that
* can complete the allocation .
*/
2001-11-29 18:45:35 +00:00
/*
* FIXME : subsequent lvextends may not be contiguous .
*/
2001-11-06 11:19:33 +00:00
static int _alloc_contiguous ( struct logical_volume * lv ,
struct list * pvms , uint32_t allocated )
2001-11-06 10:29:56 +00:00
{
2001-11-29 18:45:35 +00:00
struct list * tmp1 ;
2001-11-06 10:55:01 +00:00
struct pv_map * pvm ;
2001-11-29 18:45:35 +00:00
struct pv_area * pva ;
2001-11-06 10:55:01 +00:00
2001-11-09 22:01:04 +00:00
list_iterate ( tmp1 , pvms ) {
2001-11-06 10:55:01 +00:00
pvm = list_item ( tmp1 , struct pv_map ) ;
2001-11-29 18:45:35 +00:00
if ( list_empty ( & pvm - > areas ) )
continue ;
2001-11-06 10:55:01 +00:00
2001-11-29 18:45:35 +00:00
/* first item in the list is the biggest */
pva = list_item ( pvm - > areas . n , struct pv_area ) ;
2001-11-06 10:55:01 +00:00
2001-11-29 18:45:35 +00:00
if ( ! _alloc_linear_area ( lv , & allocated , pvm , pva ) ) {
2001-11-27 16:37:33 +00:00
stack ;
return 0 ;
}
2001-11-06 10:55:01 +00:00
if ( allocated = = lv - > le_count )
break ;
}
if ( allocated ! = lv - > le_count ) {
2001-11-06 19:02:26 +00:00
log_error ( " Insufficient free extents to "
2001-11-09 22:01:04 +00:00
" allocate logical volume %s: %u required " ,
2001-11-06 19:02:26 +00:00
lv - > name , lv - > le_count ) ;
2001-11-06 10:55:01 +00:00
return 0 ;
}
return 1 ;
2001-11-06 10:29:56 +00:00
}
2001-11-06 10:55:01 +00:00
/*
* Areas just get allocated in order until the lv
* is full .
*/
2001-11-06 11:19:33 +00:00
static int _alloc_simple ( struct logical_volume * lv ,
struct list * pvms , uint32_t allocated )
2001-11-06 10:29:56 +00:00
{
struct list * tmp1 , * tmp2 ;
struct pv_map * pvm ;
struct pv_area * pva ;
2001-11-09 22:01:04 +00:00
list_iterate ( tmp1 , pvms ) {
2001-11-06 10:29:56 +00:00
pvm = list_item ( tmp1 , struct pv_map ) ;
2001-11-09 22:01:04 +00:00
list_iterate ( tmp2 , & pvm - > areas ) {
2001-11-06 10:29:56 +00:00
pva = list_item ( tmp2 , struct pv_area ) ;
2001-11-29 18:45:35 +00:00
if ( ! _alloc_linear_area ( lv , & allocated , pvm , pva ) | |
2001-11-27 16:37:33 +00:00
( allocated = = lv - > le_count ) )
2001-11-12 12:16:57 +00:00
goto done ;
2001-11-06 10:29:56 +00:00
}
}
2001-11-29 18:45:35 +00:00
done :
2001-11-06 10:29:56 +00:00
if ( allocated ! = lv - > le_count ) {
2001-11-06 19:02:26 +00:00
log_error ( " Insufficient free logical extents to "
2001-11-09 22:01:04 +00:00
" allocate logical volume %s: %u required " ,
2001-11-06 19:02:26 +00:00
lv - > name , lv - > le_count ) ;
2001-11-06 10:29:56 +00:00
return 0 ;
}
2001-11-29 18:45:35 +00:00
return 0 ;
2001-11-06 10:29:56 +00:00
}
2001-11-06 11:31:29 +00:00
/*
* Chooses a correct allocation policy .
*/
static int _allocate ( struct volume_group * vg , struct logical_volume * lv ,
2001-11-27 17:39:15 +00:00
struct list * acceptable_pvs , uint32_t allocated ,
uint32_t stripes , uint32_t stripe_size )
2001-11-06 11:31:29 +00:00
{
int r = 0 ;
struct pool * scratch ;
2001-11-27 17:39:15 +00:00
struct list * pvms , * old_tail = lv - > segments . p , * segh ;
2001-11-06 11:31:29 +00:00
if ( ! ( scratch = pool_create ( 1024 ) ) ) {
stack ;
return 0 ;
}
/*
* Build the sets of available areas on
* the pv ' s .
*/
if ( ! ( pvms = create_pv_maps ( scratch , vg , acceptable_pvs ) ) ) {
goto out ;
}
2001-11-27 17:39:15 +00:00
if ( stripes > 1 )
r = _alloc_striped ( lv , pvms , allocated , stripes , stripe_size ) ;
2001-11-06 11:31:29 +00:00
else if ( lv - > status & ALLOC_CONTIGUOUS )
r = _alloc_contiguous ( lv , pvms , allocated ) ;
2001-11-12 17:55:05 +00:00
else if ( lv - > status & ALLOC_SIMPLE )
2001-11-06 11:31:29 +00:00
r = _alloc_simple ( lv , pvms , allocated ) ;
2001-11-12 17:55:05 +00:00
else {
2001-11-14 13:52:38 +00:00
log_error ( " Unknown allocation policy: "
" unable to setup logical volume. " ) ;
2001-11-12 17:55:05 +00:00
goto out ;
}
2001-11-06 11:31:29 +00:00
if ( r ) {
2001-11-06 12:01:46 +00:00
vg - > free_count - = lv - > le_count - allocated ;
2001-11-27 16:37:33 +00:00
/* Iterate through the new segments,
* updating pe counts in pv ' s . */
for ( segh = lv - > segments . p ; segh ! = old_tail ; segh = segh - > p )
_get_extents ( list_item ( segh , struct stripe_segment ) ) ;
} else {
/* Put the segment list back how
* we found it . */
old_tail - > n = & lv - > segments ;
lv - > segments . p = old_tail ;
2001-11-06 11:31:29 +00:00
}
2001-11-13 14:17:50 +00:00
out :
2001-11-06 11:31:29 +00:00
pool_destroy ( scratch ) ;
return r ;
}
2001-11-14 14:12:01 +00:00
static char * _generate_lv_name ( struct volume_group * vg ,
2001-11-14 18:38:07 +00:00
char * buffer , size_t len )
2001-11-14 12:07:37 +00:00
{
struct list * lvh ;
struct logical_volume * lv ;
2001-11-14 14:12:01 +00:00
int high = - 1 , i , s ;
2001-11-14 12:07:37 +00:00
list_iterate ( lvh , & vg - > lvs ) {
lv = & ( list_item ( lvh , struct lv_list ) - > lv ) ;
if ( sscanf ( lv - > name , " lvol%d " , & i ) ! = 1 )
continue ;
if ( i > high )
2001-11-14 14:12:01 +00:00
high = i ;
2001-11-14 12:07:37 +00:00
}
2001-11-14 14:12:01 +00:00
if ( ( s = snprintf ( buffer , len , " lvol%d " , high + 1 ) ) < 0 | | s > = len )
2001-11-14 12:07:37 +00:00
return NULL ;
return buffer ;
}
2001-11-12 12:16:57 +00:00
struct logical_volume * lv_create ( const char * name ,
2001-11-06 10:29:56 +00:00
uint32_t status ,
uint32_t stripes ,
uint32_t stripe_size ,
uint32_t extents ,
struct volume_group * vg ,
2001-11-06 11:19:33 +00:00
struct list * acceptable_pvs )
2001-11-06 10:29:56 +00:00
{
2001-11-12 12:16:57 +00:00
struct cmd_context * cmd = vg - > cmd ;
2001-11-06 10:29:56 +00:00
struct lv_list * ll = NULL ;
struct logical_volume * lv ;
2001-11-14 12:07:37 +00:00
char dname [ 32 ] ;
2001-11-06 10:29:56 +00:00
if ( ! extents ) {
2001-11-14 13:52:38 +00:00
log_error ( " Unable to create logical volume %s with no extents " ,
name ) ;
2001-11-06 10:29:56 +00:00
return NULL ;
}
if ( vg - > free_count < extents ) {
2001-11-14 13:52:38 +00:00
log_error ( " Insufficient free extents (%u) in volume group %s: "
" %u required " , vg - > free_count , vg - > name , extents ) ;
2001-11-06 19:02:26 +00:00
return NULL ;
2001-11-06 10:29:56 +00:00
}
if ( vg - > max_lv = = vg - > lv_count ) {
2001-11-06 19:02:26 +00:00
log_error ( " Maximum number of logical volumes (%u) reached "
" in volume group %s " , vg - > max_lv , vg - > name ) ;
return NULL ;
2001-11-06 10:29:56 +00:00
}
2001-11-14 18:38:07 +00:00
if ( ! name & & ! ( name = _generate_lv_name ( vg , dname , sizeof ( dname ) ) ) ) {
2001-11-14 13:52:38 +00:00
log_error ( " Failed to generate unique name for the new "
" logical volume " ) ;
2001-11-14 12:07:37 +00:00
return NULL ;
}
log_verbose ( " Creating logical volume %s " , name ) ;
2001-11-12 12:16:57 +00:00
if ( ! ( ll = pool_zalloc ( cmd - > mem , sizeof ( * ll ) ) ) ) {
2001-11-06 10:29:56 +00:00
stack ;
return NULL ;
}
2001-11-06 19:02:26 +00:00
list_init ( & ll - > list ) ;
2001-11-06 10:29:56 +00:00
lv = & ll - > lv ;
strcpy ( lv - > id . uuid , " " ) ;
2001-11-12 12:16:57 +00:00
if ( ! ( lv - > name = pool_strdup ( cmd - > mem , name ) ) ) {
2001-11-06 10:29:56 +00:00
stack ;
goto bad ;
}
lv - > status = status ;
lv - > read_ahead = 0 ;
lv - > size = extents * vg - > extent_size ;
lv - > le_count = extents ;
2001-11-28 13:45:50 +00:00
lv - > vg = vg ;
2001-11-27 16:37:33 +00:00
list_init ( & lv - > segments ) ;
2001-11-06 10:29:56 +00:00
2001-11-27 17:39:15 +00:00
if ( ! _allocate ( vg , lv , acceptable_pvs , 0u , stripes , stripe_size ) ) {
2001-11-06 11:31:29 +00:00
stack ;
2001-11-06 10:29:56 +00:00
goto bad ;
}
2001-11-09 22:01:04 +00:00
vg - > lv_count + + ;
2001-11-06 19:02:26 +00:00
list_add ( & vg - > lvs , & ll - > list ) ;
2001-11-09 22:01:04 +00:00
2001-11-06 10:29:56 +00:00
return lv ;
2001-11-27 16:37:33 +00:00
bad :
2001-11-06 10:29:56 +00:00
if ( ll )
2001-11-12 12:16:57 +00:00
pool_free ( cmd - > mem , ll ) ;
2001-11-06 10:29:56 +00:00
return NULL ;
}
2001-11-06 11:19:33 +00:00
2001-11-12 12:16:57 +00:00
int lv_reduce ( struct logical_volume * lv , uint32_t extents )
2001-11-06 11:19:33 +00:00
{
2001-11-27 16:37:33 +00:00
struct list * segh ;
struct stripe_segment * seg ;
uint32_t count = extents ;
for ( segh = lv - > segments . p ;
( segh ! = & lv - > segments ) & & count ;
segh = segh - > p ) {
seg = list_item ( segh , struct stripe_segment ) ;
if ( seg - > len < = count ) {
/* remove this segment completely */
count - = seg - > len ;
_put_extents ( seg ) ;
list_del ( segh ) ;
} else {
/* reduce this segment */
_put_extents ( seg ) ;
seg - > len - = count ;
_get_extents ( seg ) ;
count = 0 ;
}
2001-11-06 11:19:33 +00:00
}
2001-11-27 16:37:33 +00:00
lv - > le_count - = extents ;
lv - > size = lv - > le_count * lv - > vg - > extent_size ;
2001-11-13 14:17:50 +00:00
2001-11-06 11:19:33 +00:00
return 1 ;
}
2001-11-27 17:39:15 +00:00
int lv_extend ( struct logical_volume * lv ,
uint32_t stripes , uint32_t stripe_size ,
uint32_t extents ,
2001-11-06 12:01:46 +00:00
struct list * acceptable_pvs )
2001-11-06 11:19:33 +00:00
{
2001-11-27 16:37:33 +00:00
uint32_t old_le_count = lv - > le_count ;
uint64_t old_size = lv - > size ;
2001-11-06 12:01:46 +00:00
2001-11-27 16:37:33 +00:00
lv - > le_count + = extents ;
lv - > size + = extents * lv - > vg - > extent_size ;
2001-11-06 12:01:46 +00:00
2001-11-27 17:39:15 +00:00
if ( ! _allocate ( lv - > vg , lv , acceptable_pvs , old_le_count ,
stripes , stripe_size ) ) {
2001-11-27 16:37:33 +00:00
lv - > le_count = old_le_count ;
2001-11-27 17:39:15 +00:00
lv - > size = old_size ;
2001-11-06 12:01:46 +00:00
return 0 ;
}
2001-11-07 15:02:07 +00:00
2001-11-06 12:01:46 +00:00
return 1 ;
2001-11-06 11:19:33 +00:00
}
2001-11-12 19:28:50 +00:00
2001-11-14 18:38:07 +00:00
int lv_remove ( struct volume_group * vg , struct logical_volume * lv )
2001-11-12 19:28:50 +00:00
{
2001-11-27 16:37:33 +00:00
struct list * segh ;
2001-11-12 19:28:50 +00:00
2001-11-27 16:37:33 +00:00
/* iterate through the lv's segments freeing off the pe's */
list_iterate ( segh , & lv - > segments )
_put_extents ( list_item ( segh , struct stripe_segment ) ) ;
2001-11-12 19:28:50 +00:00
vg - > lv_count - - ;
vg - > free_count + = lv - > le_count ;
2001-11-14 18:38:07 +00:00
list_del ( & list_head ( lv , struct lv_list , lv ) ) ;
2001-11-12 19:28:50 +00:00
return 1 ;
}