2011-10-31 20:19:11 +00:00
/*
* Copyright ( C ) 2011 Red Hat , Inc .
*
* This file is released under the GPL .
*/
# include "dm-space-map.h"
# include "dm-space-map-common.h"
# include "dm-space-map-metadata.h"
# include <linux/list.h>
# include <linux/slab.h>
# include <linux/device-mapper.h>
2017-10-01 15:30:49 -04:00
# include <linux/kernel.h>
2011-10-31 20:19:11 +00:00
# define DM_MSG_PREFIX "space map metadata"
/*----------------------------------------------------------------*/
2013-05-10 14:37:20 +01:00
/*
* An edge triggered threshold .
*/
struct threshold {
bool threshold_set ;
bool value_set ;
dm_block_t threshold ;
dm_block_t current_value ;
dm_sm_threshold_fn fn ;
void * context ;
} ;
static void threshold_init ( struct threshold * t )
{
t - > threshold_set = false ;
t - > value_set = false ;
}
static void set_threshold ( struct threshold * t , dm_block_t value ,
dm_sm_threshold_fn fn , void * context )
{
t - > threshold_set = true ;
t - > threshold = value ;
t - > fn = fn ;
t - > context = context ;
}
static bool below_threshold ( struct threshold * t , dm_block_t value )
{
return t - > threshold_set & & value < = t - > threshold ;
}
static bool threshold_already_triggered ( struct threshold * t )
{
return t - > value_set & & below_threshold ( t , t - > current_value ) ;
}
static void check_threshold ( struct threshold * t , dm_block_t value )
{
if ( below_threshold ( t , value ) & &
! threshold_already_triggered ( t ) )
t - > fn ( t - > context ) ;
t - > value_set = true ;
t - > current_value = value ;
}
/*----------------------------------------------------------------*/
2011-10-31 20:19:11 +00:00
/*
* Space map interface .
*
* The low level disk format is written using the standard btree and
* transaction manager . This means that performing disk operations may
* cause us to recurse into the space map in order to allocate new blocks .
* For this reason we have a pool of pre - allocated blocks large enough to
* service any metadata_ll_disk operation .
*/
/*
* FIXME : we should calculate this based on the size of the device .
* Only the metadata space map needs this functionality .
*/
# define MAX_RECURSIVE_ALLOCATIONS 1024
enum block_op_type {
BOP_INC ,
BOP_DEC
} ;
struct block_op {
enum block_op_type type ;
2021-04-13 11:03:45 +01:00
dm_block_t b ;
dm_block_t e ;
2011-10-31 20:19:11 +00:00
} ;
2014-03-07 14:57:19 +00:00
struct bop_ring_buffer {
unsigned begin ;
unsigned end ;
struct block_op bops [ MAX_RECURSIVE_ALLOCATIONS + 1 ] ;
} ;
static void brb_init ( struct bop_ring_buffer * brb )
{
brb - > begin = 0 ;
brb - > end = 0 ;
}
static bool brb_empty ( struct bop_ring_buffer * brb )
{
return brb - > begin = = brb - > end ;
}
static unsigned brb_next ( struct bop_ring_buffer * brb , unsigned old )
{
unsigned r = old + 1 ;
2017-10-01 15:30:49 -04:00
return r > = ARRAY_SIZE ( brb - > bops ) ? 0 : r ;
2014-03-07 14:57:19 +00:00
}
static int brb_push ( struct bop_ring_buffer * brb ,
2021-04-13 11:03:45 +01:00
enum block_op_type type , dm_block_t b , dm_block_t e )
2014-03-07 14:57:19 +00:00
{
struct block_op * bop ;
unsigned next = brb_next ( brb , brb - > end ) ;
/*
* We don ' t allow the last bop to be filled , this way we can
* differentiate between full and empty .
*/
if ( next = = brb - > begin )
return - ENOMEM ;
bop = brb - > bops + brb - > end ;
bop - > type = type ;
2021-04-13 11:03:45 +01:00
bop - > b = b ;
bop - > e = e ;
2014-03-07 14:57:19 +00:00
brb - > end = next ;
return 0 ;
}
2015-12-09 16:38:12 +00:00
static int brb_peek ( struct bop_ring_buffer * brb , struct block_op * result )
2014-03-07 14:57:19 +00:00
{
struct block_op * bop ;
if ( brb_empty ( brb ) )
return - ENODATA ;
bop = brb - > bops + brb - > begin ;
2021-04-13 11:03:45 +01:00
memcpy ( result , bop , sizeof ( * result ) ) ;
2015-12-09 16:38:12 +00:00
return 0 ;
}
static int brb_pop ( struct bop_ring_buffer * brb )
{
if ( brb_empty ( brb ) )
return - ENODATA ;
2014-03-07 14:57:19 +00:00
brb - > begin = brb_next ( brb , brb - > begin ) ;
return 0 ;
}
/*----------------------------------------------------------------*/
2011-10-31 20:19:11 +00:00
struct sm_metadata {
struct dm_space_map sm ;
struct ll_disk ll ;
struct ll_disk old_ll ;
dm_block_t begin ;
unsigned recursion_count ;
unsigned allocated_this_transaction ;
2014-03-07 14:57:19 +00:00
struct bop_ring_buffer uncommitted ;
2013-05-10 14:37:20 +01:00
struct threshold threshold ;
2011-10-31 20:19:11 +00:00
} ;
2021-04-13 11:03:45 +01:00
static int add_bop ( struct sm_metadata * smm , enum block_op_type type , dm_block_t b , dm_block_t e )
2011-10-31 20:19:11 +00:00
{
2021-04-13 11:03:45 +01:00
int r = brb_push ( & smm - > uncommitted , type , b , e ) ;
2014-03-07 14:57:19 +00:00
if ( r ) {
2011-10-31 20:19:11 +00:00
DMERR ( " too many recursive allocations " ) ;
return - ENOMEM ;
}
return 0 ;
}
static int commit_bop ( struct sm_metadata * smm , struct block_op * op )
{
int r = 0 ;
2021-04-13 11:03:45 +01:00
int32_t nr_allocations ;
2011-10-31 20:19:11 +00:00
switch ( op - > type ) {
case BOP_INC :
2021-04-13 11:03:45 +01:00
r = sm_ll_inc ( & smm - > ll , op - > b , op - > e , & nr_allocations ) ;
2011-10-31 20:19:11 +00:00
break ;
case BOP_DEC :
2021-04-13 11:03:45 +01:00
r = sm_ll_dec ( & smm - > ll , op - > b , op - > e , & nr_allocations ) ;
2011-10-31 20:19:11 +00:00
break ;
}
return r ;
}
static void in ( struct sm_metadata * smm )
{
smm - > recursion_count + + ;
}
2015-06-17 13:35:19 +01:00
static int apply_bops ( struct sm_metadata * smm )
{
int r = 0 ;
while ( ! brb_empty ( & smm - > uncommitted ) ) {
struct block_op bop ;
2015-12-09 16:38:12 +00:00
r = brb_peek ( & smm - > uncommitted , & bop ) ;
2015-06-17 13:35:19 +01:00
if ( r ) {
DMERR ( " bug in bop ring buffer " ) ;
break ;
}
r = commit_bop ( smm , & bop ) ;
if ( r )
break ;
2015-12-09 16:38:12 +00:00
brb_pop ( & smm - > uncommitted ) ;
2015-06-17 13:35:19 +01:00
}
return r ;
}
2011-10-31 20:19:11 +00:00
static int out ( struct sm_metadata * smm )
{
int r = 0 ;
/*
* If we ' re not recursing then very bad things are happening .
*/
if ( ! smm - > recursion_count ) {
DMERR ( " lost track of recursion depth " ) ;
return - ENOMEM ;
}
2015-06-17 13:35:19 +01:00
if ( smm - > recursion_count = = 1 )
2019-08-19 11:31:21 +08:00
r = apply_bops ( smm ) ;
2011-10-31 20:19:11 +00:00
smm - > recursion_count - - ;
return r ;
}
/*
* When using the out ( ) function above , we often want to combine an error
* code for the operation run in the recursive context with that from
* out ( ) .
*/
static int combine_errors ( int r1 , int r2 )
{
return r1 ? r1 : r2 ;
}
static int recursing ( struct sm_metadata * smm )
{
return smm - > recursion_count ;
}
static void sm_metadata_destroy ( struct dm_space_map * sm )
{
struct sm_metadata * smm = container_of ( sm , struct sm_metadata , sm ) ;
kfree ( smm ) ;
}
static int sm_metadata_get_nr_blocks ( struct dm_space_map * sm , dm_block_t * count )
{
struct sm_metadata * smm = container_of ( sm , struct sm_metadata , sm ) ;
* count = smm - > ll . nr_blocks ;
return 0 ;
}
static int sm_metadata_get_nr_free ( struct dm_space_map * sm , dm_block_t * count )
{
struct sm_metadata * smm = container_of ( sm , struct sm_metadata , sm ) ;
* count = smm - > old_ll . nr_blocks - smm - > old_ll . nr_allocated -
smm - > allocated_this_transaction ;
return 0 ;
}
static int sm_metadata_get_count ( struct dm_space_map * sm , dm_block_t b ,
uint32_t * result )
{
2014-03-07 14:57:19 +00:00
int r ;
unsigned i ;
2011-10-31 20:19:11 +00:00
struct sm_metadata * smm = container_of ( sm , struct sm_metadata , sm ) ;
unsigned adjustment = 0 ;
/*
* We may have some uncommitted adjustments to add . This list
* should always be really short .
*/
2014-03-07 14:57:19 +00:00
for ( i = smm - > uncommitted . begin ;
i ! = smm - > uncommitted . end ;
i = brb_next ( & smm - > uncommitted , i ) ) {
struct block_op * op = smm - > uncommitted . bops + i ;
2011-10-31 20:19:11 +00:00
2021-04-13 11:03:45 +01:00
if ( b < op - > b | | b > = op - > e )
2011-10-31 20:19:11 +00:00
continue ;
switch ( op - > type ) {
case BOP_INC :
adjustment + + ;
break ;
case BOP_DEC :
adjustment - - ;
break ;
}
}
r = sm_ll_lookup ( & smm - > ll , b , result ) ;
if ( r )
return r ;
* result + = adjustment ;
return 0 ;
}
static int sm_metadata_count_is_more_than_one ( struct dm_space_map * sm ,
dm_block_t b , int * result )
{
2014-03-07 14:57:19 +00:00
int r , adjustment = 0 ;
unsigned i ;
2011-10-31 20:19:11 +00:00
struct sm_metadata * smm = container_of ( sm , struct sm_metadata , sm ) ;
uint32_t rc ;
/*
* We may have some uncommitted adjustments to add . This list
* should always be really short .
*/
2014-03-07 14:57:19 +00:00
for ( i = smm - > uncommitted . begin ;
i ! = smm - > uncommitted . end ;
i = brb_next ( & smm - > uncommitted , i ) ) {
struct block_op * op = smm - > uncommitted . bops + i ;
2011-10-31 20:19:11 +00:00
2021-04-13 11:03:45 +01:00
if ( b < op - > b | | b > = op - > e )
2011-10-31 20:19:11 +00:00
continue ;
switch ( op - > type ) {
case BOP_INC :
adjustment + + ;
break ;
case BOP_DEC :
adjustment - - ;
break ;
}
}
if ( adjustment > 1 ) {
* result = 1 ;
return 0 ;
}
r = sm_ll_lookup_bitmap ( & smm - > ll , b , & rc ) ;
if ( r )
return r ;
if ( rc = = 3 )
/*
* We err on the side of caution , and always return true .
*/
* result = 1 ;
else
* result = rc + adjustment > 1 ;
return 0 ;
}
static int sm_metadata_set_count ( struct dm_space_map * sm , dm_block_t b ,
uint32_t count )
{
int r , r2 ;
2021-04-13 11:03:45 +01:00
int32_t nr_allocations ;
2011-10-31 20:19:11 +00:00
struct sm_metadata * smm = container_of ( sm , struct sm_metadata , sm ) ;
if ( smm - > recursion_count ) {
DMERR ( " cannot recurse set_count() " ) ;
return - EINVAL ;
}
in ( smm ) ;
2021-04-13 11:03:45 +01:00
r = sm_ll_insert ( & smm - > ll , b , count , & nr_allocations ) ;
2011-10-31 20:19:11 +00:00
r2 = out ( smm ) ;
return combine_errors ( r , r2 ) ;
}
2021-04-13 11:03:45 +01:00
static int sm_metadata_inc_blocks ( struct dm_space_map * sm , dm_block_t b , dm_block_t e )
2011-10-31 20:19:11 +00:00
{
int r , r2 = 0 ;
2021-04-13 11:03:45 +01:00
int32_t nr_allocations ;
2011-10-31 20:19:11 +00:00
struct sm_metadata * smm = container_of ( sm , struct sm_metadata , sm ) ;
2021-04-13 11:03:45 +01:00
if ( recursing ( smm ) ) {
r = add_bop ( smm , BOP_INC , b , e ) ;
if ( r )
return r ;
} else {
2011-10-31 20:19:11 +00:00
in ( smm ) ;
2021-04-13 11:03:45 +01:00
r = sm_ll_inc ( & smm - > ll , b , e , & nr_allocations ) ;
2011-10-31 20:19:11 +00:00
r2 = out ( smm ) ;
}
return combine_errors ( r , r2 ) ;
}
2021-04-13 11:03:45 +01:00
static int sm_metadata_dec_blocks ( struct dm_space_map * sm , dm_block_t b , dm_block_t e )
2011-10-31 20:19:11 +00:00
{
int r , r2 = 0 ;
2021-04-13 11:03:45 +01:00
int32_t nr_allocations ;
2011-10-31 20:19:11 +00:00
struct sm_metadata * smm = container_of ( sm , struct sm_metadata , sm ) ;
if ( recursing ( smm ) )
2021-04-13 11:03:45 +01:00
r = add_bop ( smm , BOP_DEC , b , e ) ;
2011-10-31 20:19:11 +00:00
else {
in ( smm ) ;
2021-04-13 11:03:45 +01:00
r = sm_ll_dec ( & smm - > ll , b , e , & nr_allocations ) ;
2011-10-31 20:19:11 +00:00
r2 = out ( smm ) ;
}
return combine_errors ( r , r2 ) ;
}
static int sm_metadata_new_block_ ( struct dm_space_map * sm , dm_block_t * b )
{
int r , r2 = 0 ;
2021-04-13 11:03:45 +01:00
int32_t nr_allocations ;
2011-10-31 20:19:11 +00:00
struct sm_metadata * smm = container_of ( sm , struct sm_metadata , sm ) ;
2020-01-07 11:58:42 +00:00
/*
* Any block we allocate has to be free in both the old and current ll .
*/
r = sm_ll_find_common_free_block ( & smm - > old_ll , & smm - > ll , smm - > begin , smm - > ll . nr_blocks , b ) ;
2021-04-13 09:03:49 +01:00
if ( r = = - ENOSPC ) {
/*
* There ' s no free block between smm - > begin and the end of the metadata device .
* We search before smm - > begin in case something has been freed .
*/
r = sm_ll_find_common_free_block ( & smm - > old_ll , & smm - > ll , 0 , smm - > begin , b ) ;
}
2011-10-31 20:19:11 +00:00
if ( r )
return r ;
smm - > begin = * b + 1 ;
if ( recursing ( smm ) )
2021-04-13 11:03:45 +01:00
r = add_bop ( smm , BOP_INC , * b , * b + 1 ) ;
2011-10-31 20:19:11 +00:00
else {
in ( smm ) ;
2021-04-13 11:03:45 +01:00
r = sm_ll_inc ( & smm - > ll , * b , * b + 1 , & nr_allocations ) ;
2011-10-31 20:19:11 +00:00
r2 = out ( smm ) ;
}
if ( ! r )
smm - > allocated_this_transaction + + ;
return combine_errors ( r , r2 ) ;
}
static int sm_metadata_new_block ( struct dm_space_map * sm , dm_block_t * b )
{
2013-05-10 14:37:20 +01:00
dm_block_t count ;
struct sm_metadata * smm = container_of ( sm , struct sm_metadata , sm ) ;
2011-10-31 20:19:11 +00:00
int r = sm_metadata_new_block_ ( sm , b ) ;
2013-12-02 16:47:01 -05:00
if ( r ) {
2013-12-13 09:58:46 -05:00
DMERR_LIMIT ( " unable to allocate new metadata block " ) ;
2013-12-02 16:47:01 -05:00
return r ;
}
2013-05-10 14:37:20 +01:00
r = sm_metadata_get_nr_free ( sm , & count ) ;
2013-12-02 16:47:01 -05:00
if ( r ) {
2013-12-13 09:58:46 -05:00
DMERR_LIMIT ( " couldn't get free block count " ) ;
2013-12-02 16:47:01 -05:00
return r ;
}
2013-05-10 14:37:20 +01:00
check_threshold ( & smm - > threshold , count ) ;
2011-10-31 20:19:11 +00:00
return r ;
}
static int sm_metadata_commit ( struct dm_space_map * sm )
{
int r ;
struct sm_metadata * smm = container_of ( sm , struct sm_metadata , sm ) ;
r = sm_ll_commit ( & smm - > ll ) ;
if ( r )
return r ;
memcpy ( & smm - > old_ll , & smm - > ll , sizeof ( smm - > old_ll ) ) ;
smm - > allocated_this_transaction = 0 ;
return 0 ;
}
2013-05-10 14:37:20 +01:00
static int sm_metadata_register_threshold_callback ( struct dm_space_map * sm ,
dm_block_t threshold ,
dm_sm_threshold_fn fn ,
void * context )
{
struct sm_metadata * smm = container_of ( sm , struct sm_metadata , sm ) ;
set_threshold ( & smm - > threshold , threshold , fn , context ) ;
return 0 ;
}
2011-10-31 20:19:11 +00:00
static int sm_metadata_root_size ( struct dm_space_map * sm , size_t * result )
{
* result = sizeof ( struct disk_sm_root ) ;
return 0 ;
}
static int sm_metadata_copy_root ( struct dm_space_map * sm , void * where_le , size_t max )
{
struct sm_metadata * smm = container_of ( sm , struct sm_metadata , sm ) ;
struct disk_sm_root root_le ;
root_le . nr_blocks = cpu_to_le64 ( smm - > ll . nr_blocks ) ;
root_le . nr_allocated = cpu_to_le64 ( smm - > ll . nr_allocated ) ;
root_le . bitmap_root = cpu_to_le64 ( smm - > ll . bitmap_root ) ;
root_le . ref_count_root = cpu_to_le64 ( smm - > ll . ref_count_root ) ;
if ( max < sizeof ( root_le ) )
return - ENOSPC ;
memcpy ( where_le , & root_le , sizeof ( root_le ) ) ;
return 0 ;
}
2013-05-10 14:37:19 +01:00
static int sm_metadata_extend ( struct dm_space_map * sm , dm_block_t extra_blocks ) ;
2017-02-15 23:43:28 +05:30
static const struct dm_space_map ops = {
2011-10-31 20:19:11 +00:00
. destroy = sm_metadata_destroy ,
. extend = sm_metadata_extend ,
. get_nr_blocks = sm_metadata_get_nr_blocks ,
. get_nr_free = sm_metadata_get_nr_free ,
. get_count = sm_metadata_get_count ,
. count_is_more_than_one = sm_metadata_count_is_more_than_one ,
. set_count = sm_metadata_set_count ,
2021-04-13 11:03:45 +01:00
. inc_blocks = sm_metadata_inc_blocks ,
. dec_blocks = sm_metadata_dec_blocks ,
2011-10-31 20:19:11 +00:00
. new_block = sm_metadata_new_block ,
. commit = sm_metadata_commit ,
. root_size = sm_metadata_root_size ,
2013-05-10 14:37:20 +01:00
. copy_root = sm_metadata_copy_root ,
2013-05-10 14:37:20 +01:00
. register_threshold_callback = sm_metadata_register_threshold_callback
2011-10-31 20:19:11 +00:00
} ;
/*----------------------------------------------------------------*/
/*
* When a new space map is created that manages its own space . We use
* this tiny bootstrap allocator .
*/
static void sm_bootstrap_destroy ( struct dm_space_map * sm )
{
}
static int sm_bootstrap_extend ( struct dm_space_map * sm , dm_block_t extra_blocks )
{
2013-05-10 14:37:17 +01:00
DMERR ( " bootstrap doesn't support extend " ) ;
2011-10-31 20:19:11 +00:00
return - EINVAL ;
}
static int sm_bootstrap_get_nr_blocks ( struct dm_space_map * sm , dm_block_t * count )
{
struct sm_metadata * smm = container_of ( sm , struct sm_metadata , sm ) ;
2014-11-29 15:50:21 +03:00
* count = smm - > ll . nr_blocks ;
return 0 ;
2011-10-31 20:19:11 +00:00
}
static int sm_bootstrap_get_nr_free ( struct dm_space_map * sm , dm_block_t * count )
{
struct sm_metadata * smm = container_of ( sm , struct sm_metadata , sm ) ;
* count = smm - > ll . nr_blocks - smm - > begin ;
return 0 ;
}
static int sm_bootstrap_get_count ( struct dm_space_map * sm , dm_block_t b ,
uint32_t * result )
{
struct sm_metadata * smm = container_of ( sm , struct sm_metadata , sm ) ;
2014-12-01 14:38:11 +00:00
* result = ( b < smm - > begin ) ? 1 : 0 ;
return 0 ;
2011-10-31 20:19:11 +00:00
}
static int sm_bootstrap_count_is_more_than_one ( struct dm_space_map * sm ,
dm_block_t b , int * result )
{
* result = 0 ;
return 0 ;
}
static int sm_bootstrap_set_count ( struct dm_space_map * sm , dm_block_t b ,
uint32_t count )
{
2013-05-10 14:37:17 +01:00
DMERR ( " bootstrap doesn't support set_count " ) ;
2011-10-31 20:19:11 +00:00
return - EINVAL ;
}
static int sm_bootstrap_new_block ( struct dm_space_map * sm , dm_block_t * b )
{
struct sm_metadata * smm = container_of ( sm , struct sm_metadata , sm ) ;
/*
* We know the entire device is unused .
*/
if ( smm - > begin = = smm - > ll . nr_blocks )
return - ENOSPC ;
* b = smm - > begin + + ;
return 0 ;
}
2021-04-13 11:03:45 +01:00
static int sm_bootstrap_inc_blocks ( struct dm_space_map * sm , dm_block_t b , dm_block_t e )
2011-10-31 20:19:11 +00:00
{
2021-04-13 11:03:45 +01:00
int r ;
2011-10-31 20:19:11 +00:00
struct sm_metadata * smm = container_of ( sm , struct sm_metadata , sm ) ;
2021-04-13 11:03:45 +01:00
r = add_bop ( smm , BOP_INC , b , e ) ;
if ( r )
return r ;
return 0 ;
2011-10-31 20:19:11 +00:00
}
2021-04-13 11:03:45 +01:00
static int sm_bootstrap_dec_blocks ( struct dm_space_map * sm , dm_block_t b , dm_block_t e )
2011-10-31 20:19:11 +00:00
{
2021-04-13 11:03:45 +01:00
int r ;
2011-10-31 20:19:11 +00:00
struct sm_metadata * smm = container_of ( sm , struct sm_metadata , sm ) ;
2021-04-13 11:03:45 +01:00
r = add_bop ( smm , BOP_DEC , b , e ) ;
if ( r )
return r ;
return 0 ;
2011-10-31 20:19:11 +00:00
}
static int sm_bootstrap_commit ( struct dm_space_map * sm )
{
return 0 ;
}
static int sm_bootstrap_root_size ( struct dm_space_map * sm , size_t * result )
{
2013-05-10 14:37:17 +01:00
DMERR ( " bootstrap doesn't support root_size " ) ;
2011-10-31 20:19:11 +00:00
return - EINVAL ;
}
static int sm_bootstrap_copy_root ( struct dm_space_map * sm , void * where ,
size_t max )
{
2013-05-10 14:37:17 +01:00
DMERR ( " bootstrap doesn't support copy_root " ) ;
2011-10-31 20:19:11 +00:00
return - EINVAL ;
}
2017-02-15 23:43:28 +05:30
static const struct dm_space_map bootstrap_ops = {
2011-10-31 20:19:11 +00:00
. destroy = sm_bootstrap_destroy ,
. extend = sm_bootstrap_extend ,
. get_nr_blocks = sm_bootstrap_get_nr_blocks ,
. get_nr_free = sm_bootstrap_get_nr_free ,
. get_count = sm_bootstrap_get_count ,
. count_is_more_than_one = sm_bootstrap_count_is_more_than_one ,
. set_count = sm_bootstrap_set_count ,
2021-04-13 11:03:45 +01:00
. inc_blocks = sm_bootstrap_inc_blocks ,
. dec_blocks = sm_bootstrap_dec_blocks ,
2011-10-31 20:19:11 +00:00
. new_block = sm_bootstrap_new_block ,
. commit = sm_bootstrap_commit ,
. root_size = sm_bootstrap_root_size ,
2013-05-10 14:37:20 +01:00
. copy_root = sm_bootstrap_copy_root ,
. register_threshold_callback = NULL
2011-10-31 20:19:11 +00:00
} ;
/*----------------------------------------------------------------*/
2013-05-10 14:37:19 +01:00
static int sm_metadata_extend ( struct dm_space_map * sm , dm_block_t extra_blocks )
{
2021-04-13 11:03:45 +01:00
int r ;
2013-05-10 14:37:19 +01:00
struct sm_metadata * smm = container_of ( sm , struct sm_metadata , sm ) ;
dm_block_t old_len = smm - > ll . nr_blocks ;
/*
* Flick into a mode where all blocks get allocated in the new area .
*/
smm - > begin = old_len ;
2014-01-07 15:49:02 +00:00
memcpy ( sm , & bootstrap_ops , sizeof ( * sm ) ) ;
2013-05-10 14:37:19 +01:00
/*
* Extend .
*/
r = sm_ll_extend ( & smm - > ll , extra_blocks ) ;
2014-01-07 15:49:02 +00:00
if ( r )
goto out ;
2014-01-21 11:07:32 +00:00
/*
* We repeatedly increment then commit until the commit doesn ' t
* allocate any new blocks .
*/
do {
2021-04-13 11:03:45 +01:00
r = add_bop ( smm , BOP_INC , old_len , smm - > begin ) ;
2015-12-09 16:38:12 +00:00
if ( r )
goto out ;
2014-01-21 11:07:32 +00:00
old_len = smm - > begin ;
2015-06-17 13:35:19 +01:00
r = apply_bops ( smm ) ;
if ( r ) {
DMERR ( " %s: apply_bops failed " , __func__ ) ;
goto out ;
}
2014-01-21 11:07:32 +00:00
r = sm_ll_commit ( & smm - > ll ) ;
2014-01-07 15:49:02 +00:00
if ( r )
goto out ;
2013-05-10 14:37:19 +01:00
2014-01-21 11:07:32 +00:00
} while ( old_len ! = smm - > begin ) ;
2014-01-07 15:49:02 +00:00
out :
2013-05-10 14:37:19 +01:00
/*
* Switch back to normal behaviour .
*/
2014-01-07 15:49:02 +00:00
memcpy ( sm , & ops , sizeof ( * sm ) ) ;
2013-05-10 14:37:19 +01:00
return r ;
}
/*----------------------------------------------------------------*/
2011-10-31 20:19:11 +00:00
struct dm_space_map * dm_sm_metadata_init ( void )
{
struct sm_metadata * smm ;
smm = kmalloc ( sizeof ( * smm ) , GFP_KERNEL ) ;
if ( ! smm )
return ERR_PTR ( - ENOMEM ) ;
memcpy ( & smm - > sm , & ops , sizeof ( smm - > sm ) ) ;
return & smm - > sm ;
}
int dm_sm_metadata_create ( struct dm_space_map * sm ,
struct dm_transaction_manager * tm ,
dm_block_t nr_blocks ,
dm_block_t superblock )
{
int r ;
struct sm_metadata * smm = container_of ( sm , struct sm_metadata , sm ) ;
smm - > begin = superblock + 1 ;
smm - > recursion_count = 0 ;
smm - > allocated_this_transaction = 0 ;
2014-03-07 14:57:19 +00:00
brb_init ( & smm - > uncommitted ) ;
2013-05-10 14:37:20 +01:00
threshold_init ( & smm - > threshold ) ;
2011-10-31 20:19:11 +00:00
memcpy ( & smm - > sm , & bootstrap_ops , sizeof ( smm - > sm ) ) ;
r = sm_ll_new_metadata ( & smm - > ll , tm ) ;
2016-11-30 17:56:14 -06:00
if ( ! r ) {
if ( nr_blocks > DM_SM_METADATA_MAX_BLOCKS )
nr_blocks = DM_SM_METADATA_MAX_BLOCKS ;
r = sm_ll_extend ( & smm - > ll , nr_blocks ) ;
}
memcpy ( & smm - > sm , & ops , sizeof ( smm - > sm ) ) ;
2011-10-31 20:19:11 +00:00
if ( r )
return r ;
/*
* Now we need to update the newly created data structures with the
* allocated blocks that they were built from .
*/
2021-04-13 11:03:45 +01:00
r = add_bop ( smm , BOP_INC , superblock , smm - > begin ) ;
2011-10-31 20:19:11 +00:00
if ( r )
return r ;
2015-06-17 13:35:19 +01:00
r = apply_bops ( smm ) ;
if ( r ) {
DMERR ( " %s: apply_bops failed " , __func__ ) ;
return r ;
}
2011-10-31 20:19:11 +00:00
return sm_metadata_commit ( sm ) ;
}
int dm_sm_metadata_open ( struct dm_space_map * sm ,
struct dm_transaction_manager * tm ,
void * root_le , size_t len )
{
int r ;
struct sm_metadata * smm = container_of ( sm , struct sm_metadata , sm ) ;
r = sm_ll_open_metadata ( & smm - > ll , tm , root_le , len ) ;
if ( r )
return r ;
smm - > begin = 0 ;
smm - > recursion_count = 0 ;
smm - > allocated_this_transaction = 0 ;
2014-03-07 14:57:19 +00:00
brb_init ( & smm - > uncommitted ) ;
2013-05-10 14:37:20 +01:00
threshold_init ( & smm - > threshold ) ;
2011-10-31 20:19:11 +00:00
memcpy ( & smm - > old_ll , & smm - > ll , sizeof ( smm - > old_ll ) ) ;
return 0 ;
}