2017-03-17 09:18:50 +03:00
// SPDX-License-Identifier: GPL-2.0
/*
* Copyright ( C ) 2010 Kent Overstreet < kent . overstreet @ gmail . com >
* Copyright ( C ) 2014 Datera Inc .
*/
# include "bcachefs.h"
2018-10-06 07:46:55 +03:00
# include "alloc_background.h"
2018-10-06 11:12:42 +03:00
# include "alloc_foreground.h"
2017-03-17 09:18:50 +03:00
# include "bkey_methods.h"
2020-12-17 23:08:58 +03:00
# include "bkey_buf.h"
2017-03-17 09:18:50 +03:00
# include "btree_locking.h"
# include "btree_update_interior.h"
# include "btree_io.h"
# include "btree_gc.h"
# include "buckets.h"
# include "clock.h"
# include "debug.h"
2018-11-01 22:13:19 +03:00
# include "ec.h"
2017-03-17 09:18:50 +03:00
# include "error.h"
# include "extents.h"
# include "journal.h"
# include "keylist.h"
# include "move.h"
2019-04-12 05:39:39 +03:00
# include "recovery.h"
2017-03-17 09:18:50 +03:00
# include "replicas.h"
# include "super-io.h"
# include "trace.h"
# include <linux/slab.h>
# include <linux/bitops.h>
# include <linux/freezer.h>
# include <linux/kthread.h>
# include <linux/preempt.h>
# include <linux/rcupdate.h>
# include <linux/sched/task.h>
2018-10-21 17:56:11 +03:00
static inline void __gc_pos_set ( struct bch_fs * c , struct gc_pos new_pos )
{
preempt_disable ( ) ;
write_seqcount_begin ( & c - > gc_pos_lock ) ;
c - > gc_pos = new_pos ;
write_seqcount_end ( & c - > gc_pos_lock ) ;
preempt_enable ( ) ;
}
static inline void gc_pos_set ( struct bch_fs * c , struct gc_pos new_pos )
{
BUG_ON ( gc_pos_cmp ( new_pos , c - > gc_pos ) < = 0 ) ;
__gc_pos_set ( c , new_pos ) ;
}
2021-01-27 04:59:00 +03:00
/*
* Missing : if an interior btree node is empty , we need to do something -
* perhaps just kill it
*/
2020-03-29 23:48:53 +03:00
static int bch2_gc_check_topology ( struct bch_fs * c ,
2021-01-27 00:04:38 +03:00
struct btree * b ,
struct bkey_buf * prev ,
struct bkey_buf cur ,
2020-03-29 23:48:53 +03:00
bool is_last )
2017-03-17 09:18:50 +03:00
{
2021-01-27 00:04:38 +03:00
struct bpos node_start = b - > data - > min_key ;
struct bpos node_end = b - > data - > max_key ;
struct bpos expected_start = bkey_deleted ( & prev - > k - > k )
? node_start
bcachefs: Start using bpos.snapshot field
This patch starts treating the bpos.snapshot field like part of the key
in the btree code:
* bpos_successor() and bpos_predecessor() now include the snapshot field
* Keys in btrees that will be using snapshots (extents, inodes, dirents
and xattrs) now always have their snapshot field set to U32_MAX
The btree iterator code gets a new flag, BTREE_ITER_ALL_SNAPSHOTS, that
determines whether we're iterating over keys in all snapshots or not -
internally, this controlls whether bkey_(successor|predecessor)
increment/decrement the snapshot field, or only the higher bits of the
key.
We add a new member to struct btree_iter, iter->snapshot: when
BTREE_ITER_ALL_SNAPSHOTS is not set, iter->pos.snapshot should always
equal iter->snapshot, which will be 0 for btrees that don't use
snapshots, and alsways U32_MAX for btrees that will use snapshots
(until we enable snapshot creation).
This patch also introduces a new metadata version number, and compat
code for reading from/writing to older versions - this isn't a forced
upgrade (yet).
Signed-off-by: Kent Overstreet <kent.overstreet@gmail.com>
Signed-off-by: Kent Overstreet <kent.overstreet@linux.dev>
2021-03-25 01:02:16 +03:00
: bpos_successor ( prev - > k - > k . p ) ;
2021-01-27 00:04:38 +03:00
char buf1 [ 200 ] , buf2 [ 200 ] ;
2021-01-27 04:59:00 +03:00
bool update_min = false ;
bool update_max = false ;
2020-03-29 23:48:53 +03:00
int ret = 0 ;
2017-03-17 09:18:50 +03:00
2021-01-27 00:04:38 +03:00
if ( cur . k - > k . type = = KEY_TYPE_btree_ptr_v2 ) {
struct bkey_i_btree_ptr_v2 * bp = bkey_i_to_btree_ptr_v2 ( cur . k ) ;
2017-03-17 09:18:50 +03:00
2021-03-04 23:20:22 +03:00
if ( bkey_deleted ( & prev - > k - > k ) ) {
struct printbuf out = PBUF ( buf1 ) ;
pr_buf ( & out , " start of node: " ) ;
bch2_bpos_to_text ( & out , node_start ) ;
} else {
2021-01-27 00:04:38 +03:00
bch2_bkey_val_to_text ( & PBUF ( buf1 ) , c , bkey_i_to_s_c ( prev - > k ) ) ;
2021-03-04 23:20:22 +03:00
}
2021-01-27 00:04:38 +03:00
2021-03-05 00:20:16 +03:00
if ( fsck_err_on ( bpos_cmp ( expected_start , bp - > v . min_key ) , c ,
2021-01-27 04:59:00 +03:00
" btree node with incorrect min_key at btree %s level %u: \n "
" prev %s \n "
" cur %s " ,
bch2_btree_ids [ b - > c . btree_id ] , b - > c . level ,
2021-01-27 00:04:38 +03:00
buf1 ,
2021-01-27 04:59:00 +03:00
( bch2_bkey_val_to_text ( & PBUF ( buf2 ) , c , bkey_i_to_s_c ( cur . k ) ) , buf2 ) ) )
update_min = true ;
2020-03-29 23:48:53 +03:00
}
2017-03-17 09:18:50 +03:00
2020-03-29 23:48:53 +03:00
if ( fsck_err_on ( is_last & &
2021-03-05 00:20:16 +03:00
bpos_cmp ( cur . k - > k . p , node_end ) , c ,
2021-01-27 04:59:00 +03:00
" btree node with incorrect max_key at btree %s level %u: \n "
" %s \n "
" expected %s " ,
bch2_btree_ids [ b - > c . btree_id ] , b - > c . level ,
2021-01-27 00:04:38 +03:00
( bch2_bkey_val_to_text ( & PBUF ( buf1 ) , c , bkey_i_to_s_c ( cur . k ) ) , buf1 ) ,
2021-01-27 04:59:00 +03:00
( bch2_bpos_to_text ( & PBUF ( buf2 ) , node_end ) , buf2 ) ) )
update_max = true ;
2021-01-27 00:04:38 +03:00
bch2_bkey_buf_copy ( prev , c , cur . k ) ;
2021-01-27 04:59:00 +03:00
if ( update_min | | update_max ) {
struct bkey_i * new ;
struct bkey_i_btree_ptr_v2 * bp = NULL ;
struct btree * n ;
if ( update_max ) {
ret = bch2_journal_key_delete ( c , b - > c . btree_id ,
b - > c . level , cur . k - > k . p ) ;
if ( ret )
return ret ;
}
new = kmalloc ( bkey_bytes ( & cur . k - > k ) , GFP_KERNEL ) ;
2021-02-23 23:16:41 +03:00
if ( ! new ) {
bch_err ( c , " %s: error allocating new key " , __func__ ) ;
2021-01-27 04:59:00 +03:00
return - ENOMEM ;
2021-02-23 23:16:41 +03:00
}
2021-01-27 04:59:00 +03:00
bkey_copy ( new , cur . k ) ;
if ( new - > k . type = = KEY_TYPE_btree_ptr_v2 )
bp = bkey_i_to_btree_ptr_v2 ( new ) ;
if ( update_min )
bp - > v . min_key = expected_start ;
if ( update_max )
new - > k . p = node_end ;
if ( bp )
SET_BTREE_PTR_RANGE_UPDATED ( & bp - > v , true ) ;
ret = bch2_journal_key_insert ( c , b - > c . btree_id , b - > c . level , new ) ;
if ( ret ) {
kfree ( new ) ;
return ret ;
}
n = bch2_btree_node_get_noiter ( c , cur . k , b - > c . btree_id ,
b - > c . level - 1 , true ) ;
if ( n ) {
mutex_lock ( & c - > btree_cache . lock ) ;
bch2_btree_node_hash_remove ( & c - > btree_cache , n ) ;
bkey_copy ( & n - > key , new ) ;
if ( update_min )
n - > data - > min_key = expected_start ;
if ( update_max )
n - > data - > max_key = node_end ;
ret = __bch2_btree_node_hash_insert ( & c - > btree_cache , n ) ;
BUG_ON ( ret ) ;
mutex_unlock ( & c - > btree_cache . lock ) ;
six_unlock_read ( & n - > c . lock ) ;
}
}
2020-03-29 23:48:53 +03:00
fsck_err :
return ret ;
2017-03-17 09:18:50 +03:00
}
2021-01-28 03:08:54 +03:00
static int bch2_check_fix_ptrs ( struct bch_fs * c , enum btree_id btree_id ,
unsigned level , bool is_root ,
struct bkey_s_c * k )
{
struct bkey_ptrs_c ptrs = bch2_bkey_ptrs_c ( * k ) ;
2021-02-17 21:37:22 +03:00
const union bch_extent_entry * entry ;
2021-03-25 05:11:22 +03:00
struct extent_ptr_decoded p = { 0 } ;
2021-01-28 03:08:54 +03:00
bool do_update = false ;
int ret = 0 ;
2021-02-17 21:37:22 +03:00
bkey_for_each_ptr_decode ( k - > k , ptrs , p , entry ) {
struct bch_dev * ca = bch_dev_bkey_exists ( c , p . ptr . dev ) ;
struct bucket * g = PTR_BUCKET ( ca , & p . ptr , true ) ;
struct bucket * g2 = PTR_BUCKET ( ca , & p . ptr , false ) ;
2021-01-28 03:08:54 +03:00
if ( fsck_err_on ( ! g - > gen_valid , c ,
" bucket %u:%zu data type %s ptr gen %u missing in alloc btree " ,
2021-02-17 21:37:22 +03:00
p . ptr . dev , PTR_BUCKET_NR ( ca , & p . ptr ) ,
bch2_data_types [ ptr_data_type ( k - > k , & p . ptr ) ] ,
p . ptr . gen ) ) {
if ( ! p . ptr . cached ) {
g2 - > _mark . gen = g - > _mark . gen = p . ptr . gen ;
2021-01-28 03:08:54 +03:00
g2 - > gen_valid = g - > gen_valid = true ;
set_bit ( BCH_FS_NEED_ALLOC_WRITE , & c - > flags ) ;
} else {
do_update = true ;
}
}
2021-02-17 21:37:22 +03:00
if ( fsck_err_on ( gen_cmp ( p . ptr . gen , g - > mark . gen ) > 0 , c ,
2021-01-28 03:08:54 +03:00
" bucket %u:%zu data type %s ptr gen in the future: %u > %u " ,
2021-02-17 21:37:22 +03:00
p . ptr . dev , PTR_BUCKET_NR ( ca , & p . ptr ) ,
bch2_data_types [ ptr_data_type ( k - > k , & p . ptr ) ] ,
p . ptr . gen , g - > mark . gen ) ) {
if ( ! p . ptr . cached ) {
g2 - > _mark . gen = g - > _mark . gen = p . ptr . gen ;
2021-01-28 03:08:54 +03:00
g2 - > gen_valid = g - > gen_valid = true ;
g2 - > _mark . data_type = 0 ;
g2 - > _mark . dirty_sectors = 0 ;
g2 - > _mark . cached_sectors = 0 ;
set_bit ( BCH_FS_NEED_ANOTHER_GC , & c - > flags ) ;
set_bit ( BCH_FS_NEED_ALLOC_WRITE , & c - > flags ) ;
} else {
do_update = true ;
}
}
2021-02-17 21:37:22 +03:00
if ( fsck_err_on ( ! p . ptr . cached & &
gen_cmp ( p . ptr . gen , g - > mark . gen ) < 0 , c ,
2021-01-28 03:08:54 +03:00
" bucket %u:%zu data type %s stale dirty ptr: %u < %u " ,
2021-02-17 21:37:22 +03:00
p . ptr . dev , PTR_BUCKET_NR ( ca , & p . ptr ) ,
bch2_data_types [ ptr_data_type ( k - > k , & p . ptr ) ] ,
p . ptr . gen , g - > mark . gen ) )
2021-01-28 03:08:54 +03:00
do_update = true ;
2021-02-17 21:37:22 +03:00
if ( p . has_ec ) {
struct stripe * m = genradix_ptr ( & c - > stripes [ true ] , p . ec . idx ) ;
if ( fsck_err_on ( ! m | | ! m - > alive , c ,
" pointer to nonexistent stripe %llu " ,
( u64 ) p . ec . idx ) )
do_update = true ;
2021-03-13 00:55:28 +03:00
if ( fsck_err_on ( ! bch2_ptr_matches_stripe_m ( m , p ) , c ,
" pointer does not match stripe %llu " ,
( u64 ) p . ec . idx ) )
do_update = true ;
2021-02-17 21:37:22 +03:00
}
2021-01-28 03:08:54 +03:00
}
if ( do_update ) {
2021-02-17 21:37:22 +03:00
struct bkey_ptrs ptrs ;
union bch_extent_entry * entry ;
2021-01-28 03:08:54 +03:00
struct bch_extent_ptr * ptr ;
struct bkey_i * new ;
if ( is_root ) {
bch_err ( c , " cannot update btree roots yet " ) ;
return - EINVAL ;
}
new = kmalloc ( bkey_bytes ( k - > k ) , GFP_KERNEL ) ;
2021-02-23 23:16:41 +03:00
if ( ! new ) {
bch_err ( c , " %s: error allocating new key " , __func__ ) ;
2021-01-28 03:08:54 +03:00
return - ENOMEM ;
2021-02-23 23:16:41 +03:00
}
2021-01-28 03:08:54 +03:00
bkey_reassemble ( new , * k ) ;
2021-04-17 01:59:54 +03:00
if ( level ) {
/*
* We don ' t want to drop btree node pointers - if the
* btree node isn ' t there anymore , the read path will
* sort it out :
*/
ptrs = bch2_bkey_ptrs ( bkey_i_to_s ( new ) ) ;
bkey_for_each_ptr ( ptrs , ptr ) {
struct bch_dev * ca = bch_dev_bkey_exists ( c , ptr - > dev ) ;
struct bucket * g = PTR_BUCKET ( ca , ptr , true ) ;
ptr - > gen = g - > mark . gen ;
}
} else {
bch2_bkey_drop_ptrs ( bkey_i_to_s ( new ) , ptr , ( {
struct bch_dev * ca = bch_dev_bkey_exists ( c , ptr - > dev ) ;
struct bucket * g = PTR_BUCKET ( ca , ptr , true ) ;
( ptr - > cached & &
( ! g - > gen_valid | | gen_cmp ( ptr - > gen , g - > mark . gen ) > 0 ) ) | |
( ! ptr - > cached & &
gen_cmp ( ptr - > gen , g - > mark . gen ) < 0 ) ;
} ) ) ;
2021-02-17 21:37:22 +03:00
again :
2021-04-17 01:59:54 +03:00
ptrs = bch2_bkey_ptrs ( bkey_i_to_s ( new ) ) ;
bkey_extent_entry_for_each ( ptrs , entry ) {
if ( extent_entry_type ( entry ) = = BCH_EXTENT_ENTRY_stripe_ptr ) {
struct stripe * m = genradix_ptr ( & c - > stripes [ true ] ,
entry - > stripe_ptr . idx ) ;
2021-03-13 00:55:28 +03:00
union bch_extent_entry * next_ptr ;
bkey_extent_entry_for_each_from ( ptrs , next_ptr , entry )
if ( extent_entry_type ( next_ptr ) = = BCH_EXTENT_ENTRY_ptr )
goto found ;
next_ptr = NULL ;
found :
if ( ! next_ptr ) {
bch_err ( c , " aieee, found stripe ptr with no data ptr " ) ;
continue ;
}
2021-04-17 01:59:54 +03:00
2021-03-13 00:55:28 +03:00
if ( ! m | | ! m - > alive | |
! __bch2_ptr_matches_stripe ( & m - > ptrs [ entry - > stripe_ptr . block ] ,
& next_ptr - > ptr ,
m - > sectors ) ) {
2021-04-17 01:59:54 +03:00
bch2_bkey_extent_entry_drop ( new , entry ) ;
goto again ;
}
2021-02-17 21:37:22 +03:00
}
}
}
2021-01-28 03:08:54 +03:00
ret = bch2_journal_key_insert ( c , btree_id , level , new ) ;
if ( ret )
kfree ( new ) ;
else
* k = bkey_i_to_s_c ( new ) ;
}
fsck_err :
return ret ;
}
2018-10-21 17:56:11 +03:00
/* marking of btree keys/nodes: */
2021-01-28 03:08:54 +03:00
static int bch2_gc_mark_key ( struct bch_fs * c , enum btree_id btree_id ,
unsigned level , bool is_root ,
2021-04-17 01:02:57 +03:00
struct bkey_s_c * k ,
2018-11-27 16:14:51 +03:00
u8 * max_stale , bool initial )
2018-10-21 17:56:11 +03:00
{
2021-04-17 01:02:57 +03:00
struct bkey_ptrs_c ptrs ;
2018-11-01 22:10:01 +03:00
const struct bch_extent_ptr * ptr ;
2018-11-01 22:21:48 +03:00
unsigned flags =
2020-01-01 00:17:42 +03:00
BTREE_TRIGGER_GC |
( initial ? BTREE_TRIGGER_NOATOMIC : 0 ) ;
2018-10-21 17:56:11 +03:00
int ret = 0 ;
2018-11-13 01:26:36 +03:00
if ( initial ) {
2020-11-03 02:20:44 +03:00
BUG_ON ( bch2_journal_seq_verify & &
2021-04-17 01:02:57 +03:00
k - > k - > version . lo > journal_cur_seq ( & c - > journal ) ) ;
2018-11-13 01:26:36 +03:00
2021-04-17 01:02:57 +03:00
if ( fsck_err_on ( k - > k - > version . lo > atomic64_read ( & c - > key_version ) , c ,
2020-02-19 04:02:41 +03:00
" key version number higher than recorded: %llu > %llu " ,
2021-04-17 01:02:57 +03:00
k - > k - > version . lo ,
2020-02-19 04:02:41 +03:00
atomic64_read ( & c - > key_version ) ) )
2021-04-17 01:02:57 +03:00
atomic64_set ( & c - > key_version , k - > k - > version . lo ) ;
2018-11-13 01:26:36 +03:00
if ( test_bit ( BCH_FS_REBUILD_REPLICAS , & c - > flags ) | |
2021-04-17 01:02:57 +03:00
fsck_err_on ( ! bch2_bkey_replicas_marked ( c , * k ) , c ,
2018-11-13 01:26:36 +03:00
" superblock not marked as containing replicas (type %u) " ,
2021-04-17 01:02:57 +03:00
k - > k - > type ) ) {
ret = bch2_mark_bkey_replicas ( c , * k ) ;
2021-02-23 23:16:41 +03:00
if ( ret ) {
bch_err ( c , " error marking bkey replicas: %i " , ret ) ;
goto err ;
}
2018-10-21 17:56:11 +03:00
}
2018-11-01 22:21:48 +03:00
2021-04-17 01:02:57 +03:00
ret = bch2_check_fix_ptrs ( c , btree_id , level , is_root , k ) ;
2018-10-21 17:56:11 +03:00
}
2021-04-17 01:02:57 +03:00
ptrs = bch2_bkey_ptrs_c ( * k ) ;
2018-11-01 22:10:01 +03:00
bkey_for_each_ptr ( ptrs , ptr ) {
struct bch_dev * ca = bch_dev_bkey_exists ( c , ptr - > dev ) ;
2019-02-12 06:08:09 +03:00
struct bucket * g = PTR_BUCKET ( ca , ptr , true ) ;
2018-11-01 22:10:01 +03:00
2019-02-12 03:27:33 +03:00
if ( gen_after ( g - > oldest_gen , ptr - > gen ) )
g - > oldest_gen = ptr - > gen ;
2018-11-01 22:10:01 +03:00
* max_stale = max ( * max_stale , ptr_stale ( ca , ptr ) ) ;
}
2018-11-13 01:26:36 +03:00
2021-04-17 01:02:57 +03:00
bch2_mark_key ( c , * k , 0 , k - > k - > size , NULL , 0 , flags ) ;
2018-11-13 01:26:36 +03:00
fsck_err :
2021-02-23 23:16:41 +03:00
err :
if ( ret )
bch_err ( c , " %s: ret %i " , __func__ , ret ) ;
2018-10-21 17:56:11 +03:00
return ret ;
}
2019-12-30 22:37:25 +03:00
static int btree_gc_mark_node ( struct bch_fs * c , struct btree * b , u8 * max_stale ,
2020-03-16 06:29:43 +03:00
bool initial )
2017-03-17 09:18:50 +03:00
{
struct btree_node_iter iter ;
struct bkey unpacked ;
struct bkey_s_c k ;
2021-01-27 00:04:38 +03:00
struct bkey_buf prev , cur ;
2018-11-27 16:14:51 +03:00
int ret = 0 ;
* max_stale = 0 ;
2017-03-17 09:18:50 +03:00
2018-11-01 22:10:01 +03:00
if ( ! btree_node_type_needs_gc ( btree_node_type ( b ) ) )
2018-10-21 17:56:11 +03:00
return 0 ;
2017-03-17 09:18:50 +03:00
2020-03-29 23:48:53 +03:00
bch2_btree_node_iter_init_from_start ( & iter , b ) ;
2021-01-27 00:04:38 +03:00
bch2_bkey_buf_init ( & prev ) ;
bch2_bkey_buf_init ( & cur ) ;
bkey_init ( & prev . k - > k ) ;
2020-03-29 23:48:53 +03:00
while ( ( k = bch2_btree_node_iter_peek_unpack ( & iter , b , & unpacked ) ) . k ) {
2021-01-28 03:08:54 +03:00
ret = bch2_gc_mark_key ( c , b - > c . btree_id , b - > c . level , false ,
2021-04-17 01:02:57 +03:00
& k , max_stale , initial ) ;
2018-11-27 16:14:51 +03:00
if ( ret )
break ;
2020-03-29 23:48:53 +03:00
bch2_btree_node_iter_advance ( & iter , b ) ;
if ( b - > c . level ) {
2021-01-27 00:04:38 +03:00
bch2_bkey_buf_reassemble ( & cur , c , k ) ;
ret = bch2_gc_check_topology ( c , b , & prev , cur ,
2020-03-29 23:48:53 +03:00
bch2_btree_node_iter_end ( & iter ) ) ;
if ( ret )
break ;
}
2018-10-21 17:56:11 +03:00
}
2021-01-27 00:04:38 +03:00
bch2_bkey_buf_exit ( & cur , c ) ;
bch2_bkey_buf_exit ( & prev , c ) ;
2018-11-27 16:14:51 +03:00
return ret ;
2017-03-17 09:18:50 +03:00
}
2018-10-21 17:56:11 +03:00
static int bch2_gc_btree ( struct bch_fs * c , enum btree_id btree_id ,
2021-04-16 23:54:11 +03:00
bool initial , bool metadata_only )
2017-03-17 09:18:50 +03:00
{
2019-03-25 22:10:15 +03:00
struct btree_trans trans ;
struct btree_iter * iter ;
2017-03-17 09:18:50 +03:00
struct btree * b ;
2021-04-16 23:54:11 +03:00
unsigned depth = metadata_only ? 1
: bch2_expensive_debug_checks ? 0
2019-04-05 06:01:54 +03:00
: ! btree_node_type_needs_gc ( btree_id ) ? 1
: 0 ;
2019-10-09 05:16:28 +03:00
u8 max_stale = 0 ;
2017-03-17 09:18:50 +03:00
int ret = 0 ;
2019-05-15 17:54:43 +03:00
bch2_trans_init ( & trans , c , 0 , 0 ) ;
2019-03-25 22:10:15 +03:00
2017-03-17 09:18:50 +03:00
gc_pos_set ( c , gc_pos_btree ( btree_id , POS_MIN , 0 ) ) ;
2019-03-25 22:10:15 +03:00
__for_each_btree_node ( & trans , iter , btree_id , POS_MIN ,
2017-03-17 09:18:50 +03:00
0 , depth , BTREE_ITER_PREFETCH , b ) {
bch2_verify_btree_nr_keys ( b ) ;
2019-02-12 23:03:47 +03:00
gc_pos_set ( c , gc_pos_btree_node ( b ) ) ;
2020-03-16 06:29:43 +03:00
ret = btree_gc_mark_node ( c , b , & max_stale , initial ) ;
2018-11-27 16:14:51 +03:00
if ( ret )
break ;
2017-03-17 09:18:50 +03:00
2018-10-21 17:56:11 +03:00
if ( ! initial ) {
if ( max_stale > 64 )
2019-03-25 22:10:15 +03:00
bch2_btree_node_rewrite ( c , iter ,
2018-10-21 17:56:11 +03:00
b - > data - > keys . seq ,
BTREE_INSERT_NOWAIT |
BTREE_INSERT_GC_LOCK_HELD ) ;
2020-11-03 02:20:44 +03:00
else if ( ! bch2_btree_gc_rewrite_disabled & &
( bch2_btree_gc_always_rewrite | | max_stale > 16 ) )
2019-03-25 22:10:15 +03:00
bch2_btree_node_rewrite ( c , iter ,
2018-10-21 17:56:11 +03:00
b - > data - > keys . seq ,
BTREE_INSERT_NOWAIT |
BTREE_INSERT_GC_LOCK_HELD ) ;
}
2017-03-17 09:18:50 +03:00
2019-03-25 22:10:15 +03:00
bch2_trans_cond_resched ( & trans ) ;
2017-03-17 09:18:50 +03:00
}
2021-03-20 03:29:11 +03:00
bch2_trans_iter_put ( & trans , iter ) ;
2019-03-25 22:10:15 +03:00
ret = bch2_trans_exit ( & trans ) ? : ret ;
2017-03-17 09:18:50 +03:00
if ( ret )
return ret ;
mutex_lock ( & c - > btree_root_lock ) ;
b = c - > btree_roots [ btree_id ] . b ;
2021-04-17 01:02:57 +03:00
if ( ! btree_node_fake ( b ) ) {
struct bkey_s_c k = bkey_i_to_s_c ( & b - > key ) ;
2021-01-28 03:08:54 +03:00
ret = bch2_gc_mark_key ( c , b - > c . btree_id , b - > c . level , true ,
2021-04-17 01:02:57 +03:00
& k , & max_stale , initial ) ;
}
2020-06-06 19:28:01 +03:00
gc_pos_set ( c , gc_pos_btree_root ( b - > c . btree_id ) ) ;
2017-03-17 09:18:50 +03:00
mutex_unlock ( & c - > btree_root_lock ) ;
2019-03-22 02:12:52 +03:00
return ret ;
2017-03-17 09:18:50 +03:00
}
2020-03-16 06:29:43 +03:00
static int bch2_gc_btree_init_recurse ( struct bch_fs * c , struct btree * b ,
2020-03-29 23:48:53 +03:00
unsigned target_depth )
2020-03-16 06:29:43 +03:00
{
struct btree_and_journal_iter iter ;
struct bkey_s_c k ;
2021-01-27 00:04:38 +03:00
struct bkey_buf cur , prev ;
2020-03-16 06:29:43 +03:00
u8 max_stale = 0 ;
int ret = 0 ;
2021-01-27 04:15:46 +03:00
bch2_btree_and_journal_iter_init_node_iter ( & iter , c , b ) ;
2021-01-27 00:04:38 +03:00
bch2_bkey_buf_init ( & prev ) ;
bch2_bkey_buf_init ( & cur ) ;
bkey_init ( & prev . k - > k ) ;
2020-03-16 06:29:43 +03:00
while ( ( k = bch2_btree_and_journal_iter_peek ( & iter ) ) . k ) {
2021-03-05 00:20:16 +03:00
BUG_ON ( bpos_cmp ( k . k - > p , b - > data - > min_key ) < 0 ) ;
BUG_ON ( bpos_cmp ( k . k - > p , b - > data - > max_key ) > 0 ) ;
2020-03-29 23:48:53 +03:00
2021-01-28 03:08:54 +03:00
ret = bch2_gc_mark_key ( c , b - > c . btree_id , b - > c . level , false ,
2021-04-17 01:02:57 +03:00
& k , & max_stale , true ) ;
2021-02-23 23:16:41 +03:00
if ( ret ) {
bch_err ( c , " %s: error %i from bch2_gc_mark_key " , __func__ , ret ) ;
2020-03-16 06:29:43 +03:00
break ;
2021-02-23 23:16:41 +03:00
}
2020-03-16 06:29:43 +03:00
2020-03-29 23:48:53 +03:00
if ( b - > c . level ) {
2021-01-27 00:04:38 +03:00
bch2_bkey_buf_reassemble ( & cur , c , k ) ;
k = bkey_i_to_s_c ( cur . k ) ;
2020-03-29 23:48:53 +03:00
bch2_btree_and_journal_iter_advance ( & iter ) ;
2020-03-16 06:29:43 +03:00
2021-01-27 00:04:38 +03:00
ret = bch2_gc_check_topology ( c , b ,
& prev , cur ,
2020-03-29 23:48:53 +03:00
! bch2_btree_and_journal_iter_peek ( & iter ) . k ) ;
2020-03-16 06:29:43 +03:00
if ( ret )
break ;
2021-01-27 04:59:00 +03:00
} else {
bch2_btree_and_journal_iter_advance ( & iter ) ;
}
}
2020-03-16 06:29:43 +03:00
2021-01-27 04:59:00 +03:00
if ( b - > c . level > target_depth ) {
bch2_btree_and_journal_iter_exit ( & iter ) ;
bch2_btree_and_journal_iter_init_node_iter ( & iter , c , b ) ;
while ( ( k = bch2_btree_and_journal_iter_peek ( & iter ) ) . k ) {
struct btree * child ;
bch2_bkey_buf_reassemble ( & cur , c , k ) ;
bch2_btree_and_journal_iter_advance ( & iter ) ;
2020-03-16 06:29:43 +03:00
2021-01-27 04:59:00 +03:00
child = bch2_btree_node_get_noiter ( c , cur . k ,
b - > c . btree_id , b - > c . level - 1 ,
false ) ;
ret = PTR_ERR_OR_ZERO ( child ) ;
2020-03-29 23:48:53 +03:00
2021-01-27 04:59:00 +03:00
if ( fsck_err_on ( ret = = - EIO , c ,
" unreadable btree node " ) ) {
ret = bch2_journal_key_delete ( c , b - > c . btree_id ,
b - > c . level , cur . k - > k . p ) ;
2020-03-29 23:48:53 +03:00
if ( ret )
2021-01-27 04:59:00 +03:00
return ret ;
set_bit ( BCH_FS_NEED_ANOTHER_GC , & c - > flags ) ;
continue ;
2020-03-29 23:48:53 +03:00
}
2021-01-27 04:59:00 +03:00
2021-02-23 23:16:41 +03:00
if ( ret ) {
bch_err ( c , " %s: error %i getting btree node " ,
__func__ , ret ) ;
2021-01-27 04:59:00 +03:00
break ;
2021-02-23 23:16:41 +03:00
}
2021-01-27 04:59:00 +03:00
ret = bch2_gc_btree_init_recurse ( c , child ,
target_depth ) ;
six_unlock_read ( & child - > c . lock ) ;
if ( ret )
break ;
2020-03-29 23:48:53 +03:00
}
2020-03-16 06:29:43 +03:00
}
2021-01-27 04:59:00 +03:00
fsck_err :
2021-01-27 00:04:38 +03:00
bch2_bkey_buf_exit ( & cur , c ) ;
bch2_bkey_buf_exit ( & prev , c ) ;
2021-01-27 04:15:46 +03:00
bch2_btree_and_journal_iter_exit ( & iter ) ;
2020-03-16 06:29:43 +03:00
return ret ;
}
static int bch2_gc_btree_init ( struct bch_fs * c ,
2021-04-16 23:54:11 +03:00
enum btree_id btree_id ,
bool metadata_only )
2020-03-16 06:29:43 +03:00
{
struct btree * b ;
2021-04-16 23:54:11 +03:00
unsigned target_depth = metadata_only ? 1
: bch2_expensive_debug_checks ? 0
: ! btree_node_type_needs_gc ( btree_id ) ? 1
2020-03-16 06:29:43 +03:00
: 0 ;
u8 max_stale = 0 ;
2021-03-04 23:20:22 +03:00
char buf [ 100 ] ;
2020-03-16 06:29:43 +03:00
int ret = 0 ;
b = c - > btree_roots [ btree_id ] . b ;
if ( btree_node_fake ( b ) )
return 0 ;
six_lock_read ( & b - > c . lock , NULL , NULL ) ;
2021-03-05 00:20:16 +03:00
if ( fsck_err_on ( bpos_cmp ( b - > data - > min_key , POS_MIN ) , c ,
2021-03-04 23:20:22 +03:00
" btree root with incorrect min_key: %s " ,
( bch2_bpos_to_text ( & PBUF ( buf ) , b - > data - > min_key ) , buf ) ) ) {
2020-03-29 23:48:53 +03:00
BUG ( ) ;
}
2021-03-05 00:20:16 +03:00
if ( fsck_err_on ( bpos_cmp ( b - > data - > max_key , POS_MAX ) , c ,
2021-03-04 23:20:22 +03:00
" btree root with incorrect max_key: %s " ,
( bch2_bpos_to_text ( & PBUF ( buf ) , b - > data - > max_key ) , buf ) ) ) {
2020-03-29 23:48:53 +03:00
BUG ( ) ;
}
2020-03-16 06:29:43 +03:00
if ( b - > c . level > = target_depth )
2021-01-27 04:15:46 +03:00
ret = bch2_gc_btree_init_recurse ( c , b , target_depth ) ;
2020-03-16 06:29:43 +03:00
2021-04-17 01:02:57 +03:00
if ( ! ret ) {
struct bkey_s_c k = bkey_i_to_s_c ( & b - > key ) ;
2021-01-28 03:08:54 +03:00
ret = bch2_gc_mark_key ( c , b - > c . btree_id , b - > c . level , true ,
2021-04-17 01:02:57 +03:00
& k , & max_stale , true ) ;
}
2020-03-29 23:48:53 +03:00
fsck_err :
2020-03-16 06:29:43 +03:00
six_unlock_read ( & b - > c . lock ) ;
2021-02-23 23:16:41 +03:00
if ( ret )
bch_err ( c , " %s: ret %i " , __func__ , ret ) ;
2020-03-16 06:29:43 +03:00
return ret ;
}
2018-11-01 22:13:19 +03:00
static inline int btree_id_gc_phase_cmp ( enum btree_id l , enum btree_id r )
{
return ( int ) btree_id_to_gc_phase ( l ) -
( int ) btree_id_to_gc_phase ( r ) ;
}
2021-04-16 23:54:11 +03:00
static int bch2_gc_btrees ( struct bch_fs * c , bool initial , bool metadata_only )
2018-10-21 17:56:11 +03:00
{
2018-11-01 22:13:19 +03:00
enum btree_id ids [ BTREE_ID_NR ] ;
2018-10-21 17:56:11 +03:00
unsigned i ;
2018-11-01 22:13:19 +03:00
for ( i = 0 ; i < BTREE_ID_NR ; i + + )
ids [ i ] = i ;
bubble_sort ( ids , BTREE_ID_NR , btree_id_gc_phase_cmp ) ;
2018-10-21 17:56:11 +03:00
for ( i = 0 ; i < BTREE_ID_NR ; i + + ) {
2018-11-01 22:13:19 +03:00
enum btree_id id = ids [ i ] ;
2020-03-16 06:29:43 +03:00
int ret = initial
2021-04-16 23:54:11 +03:00
? bch2_gc_btree_init ( c , id , metadata_only )
: bch2_gc_btree ( c , id , initial , metadata_only ) ;
2021-02-23 23:16:41 +03:00
if ( ret ) {
bch_err ( c , " %s: ret %i " , __func__ , ret ) ;
2018-10-21 17:56:11 +03:00
return ret ;
2021-02-23 23:16:41 +03:00
}
2018-10-21 17:56:11 +03:00
}
return 0 ;
}
2017-03-17 09:18:50 +03:00
static void mark_metadata_sectors ( struct bch_fs * c , struct bch_dev * ca ,
u64 start , u64 end ,
enum bch_data_type type ,
unsigned flags )
{
u64 b = sector_to_bucket ( ca , start ) ;
do {
unsigned sectors =
min_t ( u64 , bucket_to_sector ( ca , b + 1 ) , end ) - start ;
bch2_mark_metadata_bucket ( c , ca , b , type , sectors ,
gc_phase ( GC_PHASE_SB ) , flags ) ;
b + + ;
start + = sectors ;
} while ( start < end ) ;
}
void bch2_mark_dev_superblock ( struct bch_fs * c , struct bch_dev * ca ,
unsigned flags )
{
struct bch_sb_layout * layout = & ca - > disk_sb . sb - > layout ;
unsigned i ;
u64 b ;
2018-07-21 05:08:17 +03:00
/*
* This conditional is kind of gross , but we may be called from the
* device add path , before the new device has actually been added to the
* running filesystem :
*/
2017-03-17 09:18:50 +03:00
if ( c ) {
lockdep_assert_held ( & c - > sb_lock ) ;
2018-11-26 08:13:33 +03:00
percpu_down_read ( & c - > mark_lock ) ;
2017-03-17 09:18:50 +03:00
}
for ( i = 0 ; i < layout - > nr_superblocks ; i + + ) {
u64 offset = le64_to_cpu ( layout - > sb_offset [ i ] ) ;
if ( offset = = BCH_SB_SECTOR )
mark_metadata_sectors ( c , ca , 0 , BCH_SB_SECTOR ,
2020-07-10 01:28:11 +03:00
BCH_DATA_sb , flags ) ;
2017-03-17 09:18:50 +03:00
mark_metadata_sectors ( c , ca , offset ,
offset + ( 1 < < layout - > sb_max_size_bits ) ,
2020-07-10 01:28:11 +03:00
BCH_DATA_sb , flags ) ;
2017-03-17 09:18:50 +03:00
}
for ( i = 0 ; i < ca - > journal . nr ; i + + ) {
b = ca - > journal . buckets [ i ] ;
2020-07-10 01:28:11 +03:00
bch2_mark_metadata_bucket ( c , ca , b , BCH_DATA_journal ,
2017-03-17 09:18:50 +03:00
ca - > mi . bucket_size ,
gc_phase ( GC_PHASE_SB ) , flags ) ;
}
2022-12-25 06:44:56 +03:00
if ( c )
2018-11-26 08:13:33 +03:00
percpu_up_read ( & c - > mark_lock ) ;
2017-03-17 09:18:50 +03:00
}
static void bch2_mark_superblocks ( struct bch_fs * c )
{
struct bch_dev * ca ;
unsigned i ;
mutex_lock ( & c - > sb_lock ) ;
gc_pos_set ( c , gc_phase ( GC_PHASE_SB ) ) ;
for_each_online_member ( ca , c , i )
2020-01-01 00:17:42 +03:00
bch2_mark_dev_superblock ( c , ca , BTREE_TRIGGER_GC ) ;
2017-03-17 09:18:50 +03:00
mutex_unlock ( & c - > sb_lock ) ;
}
2020-05-25 21:57:06 +03:00
#if 0
2017-03-17 09:18:50 +03:00
/* Also see bch2_pending_btree_node_free_insert_done() */
static void bch2_mark_pending_btree_node_frees ( struct bch_fs * c )
{
struct btree_update * as ;
struct pending_btree_node_free * d ;
mutex_lock ( & c - > btree_interior_update_lock ) ;
gc_pos_set ( c , gc_phase ( GC_PHASE_PENDING_DELETE ) ) ;
for_each_pending_btree_node_free ( c , as , d )
if ( d - > index_update_done )
2019-08-09 20:01:10 +03:00
bch2_mark_key ( c , bkey_i_to_s_c ( & d - > key ) ,
0 , 0 , NULL , 0 ,
2020-01-01 00:17:42 +03:00
BTREE_TRIGGER_GC ) ;
2017-03-17 09:18:50 +03:00
mutex_unlock ( & c - > btree_interior_update_lock ) ;
}
2020-05-25 21:57:06 +03:00
# endif
2017-03-17 09:18:50 +03:00
static void bch2_mark_allocator_buckets ( struct bch_fs * c )
{
struct bch_dev * ca ;
struct open_bucket * ob ;
size_t i , j , iter ;
unsigned ci ;
2018-11-26 08:13:33 +03:00
percpu_down_read ( & c - > mark_lock ) ;
2017-03-17 09:18:50 +03:00
spin_lock ( & c - > freelist_lock ) ;
gc_pos_set ( c , gc_pos_alloc ( c , NULL ) ) ;
for_each_member_device ( ca , c , ci ) {
fifo_for_each_entry ( i , & ca - > free_inc , iter )
bch2_mark_alloc_bucket ( c , ca , i , true ,
gc_pos_alloc ( c , NULL ) ,
2020-01-01 00:17:42 +03:00
BTREE_TRIGGER_GC ) ;
2017-03-17 09:18:50 +03:00
for ( j = 0 ; j < RESERVE_NR ; j + + )
fifo_for_each_entry ( i , & ca - > free [ j ] , iter )
bch2_mark_alloc_bucket ( c , ca , i , true ,
gc_pos_alloc ( c , NULL ) ,
2020-01-01 00:17:42 +03:00
BTREE_TRIGGER_GC ) ;
2017-03-17 09:18:50 +03:00
}
spin_unlock ( & c - > freelist_lock ) ;
for ( ob = c - > open_buckets ;
ob < c - > open_buckets + ARRAY_SIZE ( c - > open_buckets ) ;
ob + + ) {
spin_lock ( & ob - > lock ) ;
if ( ob - > valid ) {
gc_pos_set ( c , gc_pos_alloc ( c , ob ) ) ;
ca = bch_dev_bkey_exists ( c , ob - > ptr . dev ) ;
bch2_mark_alloc_bucket ( c , ca , PTR_BUCKET_NR ( ca , & ob - > ptr ) , true ,
gc_pos_alloc ( c , ob ) ,
2020-01-01 00:17:42 +03:00
BTREE_TRIGGER_GC ) ;
2017-03-17 09:18:50 +03:00
}
spin_unlock ( & ob - > lock ) ;
}
2018-11-26 08:13:33 +03:00
percpu_up_read ( & c - > mark_lock ) ;
2017-03-17 09:18:50 +03:00
}
2018-07-23 12:32:01 +03:00
static void bch2_gc_free ( struct bch_fs * c )
{
struct bch_dev * ca ;
unsigned i ;
2018-11-25 01:09:44 +03:00
genradix_free ( & c - > stripes [ 1 ] ) ;
2018-07-23 12:32:01 +03:00
for_each_member_device ( ca , c , i ) {
kvpfree ( rcu_dereference_protected ( ca - > buckets [ 1 ] , 1 ) ,
sizeof ( struct bucket_array ) +
ca - > mi . nbuckets * sizeof ( struct bucket ) ) ;
ca - > buckets [ 1 ] = NULL ;
2021-01-22 05:52:06 +03:00
free_percpu ( ca - > usage_gc ) ;
ca - > usage_gc = NULL ;
2018-07-23 12:32:01 +03:00
}
2019-02-11 03:34:47 +03:00
free_percpu ( c - > usage_gc ) ;
c - > usage_gc = NULL ;
2018-12-01 19:32:12 +03:00
}
2019-03-30 05:22:45 +03:00
static int bch2_gc_done ( struct bch_fs * c ,
2021-04-16 23:54:11 +03:00
bool initial , bool metadata_only )
2018-07-23 12:32:01 +03:00
{
struct bch_dev * ca ;
2021-04-16 23:54:11 +03:00
bool verify = ! metadata_only & & ( ! initial | |
2021-04-05 04:57:35 +03:00
( c - > sb . compat & ( 1ULL < < BCH_COMPAT_alloc_info ) ) ) ;
2021-01-22 05:52:06 +03:00
unsigned i , dev ;
2019-03-28 10:13:27 +03:00
int ret = 0 ;
2018-07-23 12:32:01 +03:00
# define copy_field(_f, _msg, ...) \
2018-12-17 16:44:56 +03:00
if ( dst - > _f ! = src - > _f ) { \
2019-02-12 03:27:33 +03:00
if ( verify ) \
2019-03-28 10:13:27 +03:00
fsck_err ( c , _msg " : got %llu, should be %llu " \
2019-02-12 03:27:33 +03:00
, # # __VA_ARGS__ , dst - > _f , src - > _f ) ; \
2018-12-17 16:44:56 +03:00
dst - > _f = src - > _f ; \
2021-01-09 05:20:58 +03:00
set_bit ( BCH_FS_NEED_ALLOC_WRITE , & c - > flags ) ; \
2018-07-23 12:32:01 +03:00
}
2018-11-25 01:09:44 +03:00
# define copy_stripe_field(_f, _msg, ...) \
if ( dst - > _f ! = src - > _f ) { \
2019-02-12 03:27:33 +03:00
if ( verify ) \
2019-03-28 10:13:27 +03:00
fsck_err ( c , " stripe %zu has wrong " _msg \
" : got %u, should be %u " , \
2021-01-18 00:16:37 +03:00
iter . pos , # # __VA_ARGS__ , \
2019-02-12 03:27:33 +03:00
dst - > _f , src - > _f ) ; \
2018-11-25 01:09:44 +03:00
dst - > _f = src - > _f ; \
2021-01-09 05:20:58 +03:00
set_bit ( BCH_FS_NEED_ALLOC_WRITE , & c - > flags ) ; \
2018-11-25 01:09:44 +03:00
}
2018-07-23 12:32:01 +03:00
# define copy_bucket_field(_f) \
if ( dst - > b [ b ] . mark . _f ! = src - > b [ b ] . mark . _f ) { \
2019-02-12 03:27:33 +03:00
if ( verify ) \
2020-05-24 21:20:00 +03:00
fsck_err ( c , " bucket %u:%zu gen %u data type %s has wrong " # _f \
2019-03-28 10:13:27 +03:00
" : got %u, should be %u " , i , b , \
2020-05-24 21:20:00 +03:00
dst - > b [ b ] . mark . gen , \
bch2_data_types [ dst - > b [ b ] . mark . data_type ] , \
2019-02-12 03:27:33 +03:00
dst - > b [ b ] . mark . _f , src - > b [ b ] . mark . _f ) ; \
2018-07-23 12:32:01 +03:00
dst - > b [ b ] . _mark . _f = src - > b [ b ] . mark . _f ; \
2021-01-09 05:20:58 +03:00
set_bit ( BCH_FS_NEED_ALLOC_WRITE , & c - > flags ) ; \
2018-07-23 12:32:01 +03:00
}
# define copy_dev_field(_f, _msg, ...) \
copy_field ( _f , " dev %u has wrong " _msg , i , # # __VA_ARGS__ )
# define copy_fs_field(_f, _msg, ...) \
copy_field ( _f , " fs has wrong " _msg , # # __VA_ARGS__ )
2021-04-16 23:54:11 +03:00
if ( ! metadata_only ) {
2021-01-18 00:16:37 +03:00
struct genradix_iter iter = genradix_iter_init ( & c - > stripes [ 1 ] , 0 ) ;
2018-11-25 01:09:44 +03:00
struct stripe * dst , * src ;
2021-01-18 00:16:37 +03:00
while ( ( src = genradix_iter_peek ( & iter , & c - > stripes [ 1 ] ) ) ) {
dst = genradix_ptr_alloc ( & c - > stripes [ 0 ] , iter . pos , GFP_KERNEL ) ;
2018-11-26 04:53:51 +03:00
2021-01-18 00:45:19 +03:00
if ( dst - > alive ! = src - > alive | |
dst - > sectors ! = src - > sectors | |
dst - > algorithm ! = src - > algorithm | |
dst - > nr_blocks ! = src - > nr_blocks | |
dst - > nr_redundant ! = src - > nr_redundant ) {
bch_err ( c , " unexpected stripe inconsistency at bch2_gc_done, confused " ) ;
ret = - EINVAL ;
goto fsck_err ;
}
2018-11-25 01:09:44 +03:00
for ( i = 0 ; i < ARRAY_SIZE ( dst - > block_sectors ) ; i + + )
2018-11-26 04:53:51 +03:00
copy_stripe_field ( block_sectors [ i ] ,
2018-11-25 01:09:44 +03:00
" block_sectors[%u] " , i ) ;
2021-01-18 00:45:19 +03:00
dst - > blocks_nonempty = 0 ;
for ( i = 0 ; i < dst - > nr_blocks ; i + + )
dst - > blocks_nonempty + = dst - > block_sectors [ i ] ! = 0 ;
2021-01-18 00:16:37 +03:00
genradix_iter_advance ( & iter , & c - > stripes [ 1 ] ) ;
2018-11-25 01:09:44 +03:00
}
}
2021-01-22 05:52:06 +03:00
for ( i = 0 ; i < ARRAY_SIZE ( c - > usage ) ; i + + )
bch2_fs_usage_acc_to_base ( c , i ) ;
for_each_member_device ( ca , c , dev ) {
2018-07-23 12:32:01 +03:00
struct bucket_array * dst = __bucket_array ( ca , 0 ) ;
struct bucket_array * src = __bucket_array ( ca , 1 ) ;
size_t b ;
for ( b = 0 ; b < src - > nbuckets ; b + + ) {
copy_bucket_field ( gen ) ;
copy_bucket_field ( data_type ) ;
copy_bucket_field ( owned_by_allocator ) ;
copy_bucket_field ( stripe ) ;
copy_bucket_field ( dirty_sectors ) ;
copy_bucket_field ( cached_sectors ) ;
2019-02-12 03:27:33 +03:00
2019-08-28 00:45:42 +03:00
dst - > b [ b ] . oldest_gen = src - > b [ b ] . oldest_gen ;
2018-07-23 12:32:01 +03:00
}
2021-01-22 05:52:06 +03:00
{
struct bch_dev_usage * dst = ca - > usage_base ;
struct bch_dev_usage * src = ( void * )
bch2_acc_percpu_u64s ( ( void * ) ca - > usage_gc ,
dev_usage_u64s ( ) ) ;
copy_dev_field ( buckets_ec , " buckets_ec " ) ;
copy_dev_field ( buckets_unavailable , " buckets_unavailable " ) ;
2019-02-11 03:34:47 +03:00
2021-01-22 05:52:06 +03:00
for ( i = 0 ; i < BCH_DATA_NR ; i + + ) {
copy_dev_field ( d [ i ] . buckets , " %s buckets " , bch2_data_types [ i ] ) ;
copy_dev_field ( d [ i ] . sectors , " %s sectors " , bch2_data_types [ i ] ) ;
copy_dev_field ( d [ i ] . fragmented , " %s fragmented " , bch2_data_types [ i ] ) ;
}
}
} ;
2018-07-23 12:32:01 +03:00
{
2019-02-15 04:39:17 +03:00
unsigned nr = fs_usage_u64s ( c ) ;
2019-02-11 03:34:47 +03:00
struct bch_fs_usage * dst = c - > usage_base ;
2018-12-17 16:44:56 +03:00
struct bch_fs_usage * src = ( void * )
2019-02-11 03:34:47 +03:00
bch2_acc_percpu_u64s ( ( void * ) c - > usage_gc , nr ) ;
2018-07-23 12:32:01 +03:00
2019-02-15 02:38:52 +03:00
copy_fs_field ( hidden , " hidden " ) ;
2019-03-30 05:22:45 +03:00
copy_fs_field ( btree , " btree " ) ;
2018-12-01 19:32:12 +03:00
2021-04-16 23:54:11 +03:00
if ( ! metadata_only ) {
copy_fs_field ( data , " data " ) ;
copy_fs_field ( cached , " cached " ) ;
copy_fs_field ( reserved , " reserved " ) ;
copy_fs_field ( nr_inodes , " nr_inodes " ) ;
for ( i = 0 ; i < BCH_REPLICAS_MAX ; i + + )
copy_fs_field ( persistent_reserved [ i ] ,
" persistent_reserved[%i] " , i ) ;
}
2018-07-23 12:32:01 +03:00
2019-01-21 23:32:13 +03:00
for ( i = 0 ; i < c - > replicas . nr ; i + + ) {
2019-02-12 23:03:47 +03:00
struct bch_replicas_entry * e =
cpu_replicas_entry ( & c - > replicas , i ) ;
char buf [ 80 ] ;
2021-04-16 23:54:11 +03:00
if ( metadata_only & &
( e - > data_type = = BCH_DATA_user | |
e - > data_type = = BCH_DATA_cached ) )
continue ;
2019-02-12 23:03:47 +03:00
bch2_replicas_entry_to_text ( & PBUF ( buf ) , e ) ;
2019-02-15 02:38:52 +03:00
copy_fs_field ( replicas [ i ] , " %s " , buf ) ;
2019-01-21 23:32:13 +03:00
}
2018-07-23 12:32:01 +03:00
}
2019-02-12 03:27:33 +03:00
2018-07-23 12:32:01 +03:00
# undef copy_fs_field
# undef copy_dev_field
# undef copy_bucket_field
2018-11-25 01:09:44 +03:00
# undef copy_stripe_field
# undef copy_field
2019-03-28 10:13:27 +03:00
fsck_err :
2021-02-23 23:16:41 +03:00
if ( ret )
bch_err ( c , " %s: ret %i " , __func__ , ret ) ;
2019-03-28 10:13:27 +03:00
return ret ;
2018-07-23 12:32:01 +03:00
}
2021-04-16 23:54:11 +03:00
static int bch2_gc_start ( struct bch_fs * c ,
bool metadata_only )
2018-07-23 12:32:01 +03:00
{
struct bch_dev * ca ;
unsigned i ;
2019-10-05 02:14:43 +03:00
int ret ;
2018-11-25 01:09:44 +03:00
2019-02-11 03:34:47 +03:00
BUG_ON ( c - > usage_gc ) ;
2018-07-23 12:32:01 +03:00
2019-02-11 03:34:47 +03:00
c - > usage_gc = __alloc_percpu_gfp ( fs_usage_u64s ( c ) * sizeof ( u64 ) ,
2019-02-15 04:39:17 +03:00
sizeof ( u64 ) , GFP_KERNEL ) ;
2020-04-29 19:57:04 +03:00
if ( ! c - > usage_gc ) {
bch_err ( c , " error allocating c->usage_gc " ) ;
2018-07-23 12:32:01 +03:00
return - ENOMEM ;
2020-04-29 19:57:04 +03:00
}
2018-07-23 12:32:01 +03:00
2017-03-17 09:18:50 +03:00
for_each_member_device ( ca , c , i ) {
2018-07-23 12:32:01 +03:00
BUG_ON ( ca - > buckets [ 1 ] ) ;
2021-01-22 05:52:06 +03:00
BUG_ON ( ca - > usage_gc ) ;
2018-07-23 12:32:01 +03:00
ca - > buckets [ 1 ] = kvpmalloc ( sizeof ( struct bucket_array ) +
ca - > mi . nbuckets * sizeof ( struct bucket ) ,
GFP_KERNEL | __GFP_ZERO ) ;
if ( ! ca - > buckets [ 1 ] ) {
percpu_ref_put ( & ca - > ref ) ;
2020-04-29 19:57:04 +03:00
bch_err ( c , " error allocating ca->buckets[gc] " ) ;
2018-07-23 12:32:01 +03:00
return - ENOMEM ;
}
2021-01-22 05:52:06 +03:00
ca - > usage_gc = alloc_percpu ( struct bch_dev_usage ) ;
if ( ! ca - > usage_gc ) {
bch_err ( c , " error allocating ca->usage_gc " ) ;
2018-07-23 12:32:01 +03:00
percpu_ref_put ( & ca - > ref ) ;
return - ENOMEM ;
2017-03-17 09:18:50 +03:00
}
}
2018-07-23 12:32:01 +03:00
2019-10-05 02:14:43 +03:00
ret = bch2_ec_mem_alloc ( c , true ) ;
2020-04-29 19:57:04 +03:00
if ( ret ) {
bch_err ( c , " error allocating ec gc mem " ) ;
2019-10-05 02:14:43 +03:00
return ret ;
2020-04-29 19:57:04 +03:00
}
2019-10-05 02:14:43 +03:00
percpu_down_write ( & c - > mark_lock ) ;
/*
* indicate to stripe code that we need to allocate for the gc stripes
* radix tree , too
*/
gc_pos_set ( c , gc_phase ( GC_PHASE_START ) ) ;
2018-07-23 12:32:01 +03:00
for_each_member_device ( ca , c , i ) {
struct bucket_array * dst = __bucket_array ( ca , 1 ) ;
struct bucket_array * src = __bucket_array ( ca , 0 ) ;
size_t b ;
dst - > first_bucket = src - > first_bucket ;
dst - > nbuckets = src - > nbuckets ;
2019-02-21 01:57:06 +03:00
for ( b = 0 ; b < src - > nbuckets ; b + + ) {
2019-03-30 05:22:45 +03:00
struct bucket * d = & dst - > b [ b ] ;
struct bucket * s = & src - > b [ b ] ;
d - > _mark . gen = dst - > b [ b ] . oldest_gen = s - > mark . gen ;
d - > gen_valid = s - > gen_valid ;
2021-04-16 23:54:11 +03:00
if ( metadata_only & &
( s - > mark . data_type = = BCH_DATA_user | |
s - > mark . data_type = = BCH_DATA_cached ) ) {
d - > _mark = s - > mark ;
d - > _mark . owned_by_allocator = 0 ;
}
2019-02-21 01:57:06 +03:00
}
2018-07-23 12:32:01 +03:00
} ;
2019-10-05 02:14:43 +03:00
percpu_up_write ( & c - > mark_lock ) ;
return 0 ;
2017-03-17 09:18:50 +03:00
}
/**
2018-07-23 12:32:01 +03:00
* bch2_gc - walk _all_ references to buckets , and recompute them :
*
* Order matters here :
* - Concurrent GC relies on the fact that we have a total ordering for
* everything that GC walks - see gc_will_visit_node ( ) ,
* gc_will_visit_root ( )
*
* - also , references move around in the course of index updates and
* various other crap : everything needs to agree on the ordering
* references are allowed to move around in - e . g . , we ' re allowed to
* start with a reference owned by an open_bucket ( the allocator ) and
* move it to the btree , but not the reverse .
*
* This is necessary to ensure that gc doesn ' t miss references that
* move around - if references move backwards in the ordering GC
* uses , GC could skip past them
2017-03-17 09:18:50 +03:00
*/
2021-04-16 23:54:11 +03:00
int bch2_gc ( struct bch_fs * c , bool initial , bool metadata_only )
2017-03-17 09:18:50 +03:00
{
struct bch_dev * ca ;
u64 start_time = local_clock ( ) ;
2018-07-23 12:32:01 +03:00
unsigned i , iter = 0 ;
2018-10-21 17:56:11 +03:00
int ret ;
2017-03-17 09:18:50 +03:00
2020-06-15 21:58:47 +03:00
lockdep_assert_held ( & c - > state_lock ) ;
2017-03-17 09:18:50 +03:00
trace_gc_start ( c ) ;
down_write ( & c - > gc_lock ) ;
2020-05-25 21:57:06 +03:00
/* flush interior btree updates: */
closure_wait_event ( & c - > btree_interior_update_wait ,
! bch2_btree_interior_updates_nr_pending ( c ) ) ;
2018-07-23 12:32:01 +03:00
again :
2021-04-16 23:54:11 +03:00
ret = bch2_gc_start ( c , metadata_only ) ;
2018-07-23 12:32:01 +03:00
if ( ret )
2017-03-17 09:18:50 +03:00
goto out ;
bch2_mark_superblocks ( c ) ;
2021-04-16 23:54:11 +03:00
ret = bch2_gc_btrees ( c , initial , metadata_only ) ;
2018-07-23 12:32:01 +03:00
if ( ret )
2018-10-21 17:56:11 +03:00
goto out ;
2017-03-17 09:18:50 +03:00
2020-05-25 21:57:06 +03:00
#if 0
2017-03-17 09:18:50 +03:00
bch2_mark_pending_btree_node_frees ( c ) ;
2020-05-25 21:57:06 +03:00
# endif
2017-03-17 09:18:50 +03:00
bch2_mark_allocator_buckets ( c ) ;
c - > gc_count + + ;
2021-01-27 04:59:00 +03:00
if ( test_bit ( BCH_FS_NEED_ANOTHER_GC , & c - > flags ) | |
( ! iter & & bch2_test_restart_gc ) ) {
2018-07-23 12:32:01 +03:00
/*
* XXX : make sure gens we fixed got saved
*/
if ( iter + + < = 2 ) {
2021-01-27 04:59:00 +03:00
bch_info ( c , " Second GC pass needed, restarting: " ) ;
clear_bit ( BCH_FS_NEED_ANOTHER_GC , & c - > flags ) ;
2019-02-21 01:57:06 +03:00
__gc_pos_set ( c , gc_phase ( GC_PHASE_NOT_RUNNING ) ) ;
2019-03-28 10:08:40 +03:00
percpu_down_write ( & c - > mark_lock ) ;
2019-02-21 01:57:06 +03:00
bch2_gc_free ( c ) ;
2019-03-28 10:08:40 +03:00
percpu_up_write ( & c - > mark_lock ) ;
2019-09-07 19:42:27 +03:00
/* flush fsck errors, reset counters */
bch2_flush_fsck_errs ( c ) ;
2019-03-28 10:08:40 +03:00
2018-07-23 12:32:01 +03:00
goto again ;
}
bch_info ( c , " Unable to fix bucket gens, looping " ) ;
ret = - EINVAL ;
}
2021-01-27 04:59:00 +03:00
out :
2019-02-11 03:34:47 +03:00
if ( ! ret ) {
bch2_journal_block ( & c - > journal ) ;
2019-03-28 10:08:40 +03:00
2019-02-11 03:34:47 +03:00
percpu_down_write ( & c - > mark_lock ) ;
2021-04-16 23:54:11 +03:00
ret = bch2_gc_done ( c , initial , metadata_only ) ;
2018-07-23 12:32:01 +03:00
2019-02-11 03:34:47 +03:00
bch2_journal_unblock ( & c - > journal ) ;
} else {
percpu_down_write ( & c - > mark_lock ) ;
}
2018-07-23 12:32:01 +03:00
/* Indicates that gc is no longer in progress: */
2018-11-25 01:09:44 +03:00
__gc_pos_set ( c , gc_phase ( GC_PHASE_NOT_RUNNING ) ) ;
2018-07-23 12:32:01 +03:00
bch2_gc_free ( c ) ;
2019-03-28 10:08:40 +03:00
percpu_up_write ( & c - > mark_lock ) ;
2017-03-17 09:18:50 +03:00
up_write ( & c - > gc_lock ) ;
2018-07-23 12:32:01 +03:00
2017-03-17 09:18:50 +03:00
trace_gc_end ( c ) ;
bch2_time_stats_update ( & c - > times [ BCH_TIME_btree_gc ] , start_time ) ;
/*
* Wake up allocator in case it was waiting for buckets
* because of not being able to inc gens
*/
for_each_member_device ( ca , c , i )
bch2_wake_allocator ( ca ) ;
/*
* At startup , allocations can happen directly instead of via the
* allocator thread - issue wakeup in case they blocked on gc_lock :
*/
closure_wake_up ( & c - > freelist_wait ) ;
2018-07-23 12:32:01 +03:00
return ret ;
2017-03-17 09:18:50 +03:00
}
2020-10-13 10:58:50 +03:00
static bool gc_btree_gens_key ( struct bch_fs * c , struct bkey_s_c k )
{
struct bkey_ptrs_c ptrs = bch2_bkey_ptrs_c ( k ) ;
const struct bch_extent_ptr * ptr ;
percpu_down_read ( & c - > mark_lock ) ;
bkey_for_each_ptr ( ptrs , ptr ) {
struct bch_dev * ca = bch_dev_bkey_exists ( c , ptr - > dev ) ;
struct bucket * g = PTR_BUCKET ( ca , ptr , false ) ;
if ( gen_after ( g - > mark . gen , ptr - > gen ) > 16 ) {
percpu_up_read ( & c - > mark_lock ) ;
return true ;
}
}
bkey_for_each_ptr ( ptrs , ptr ) {
struct bch_dev * ca = bch_dev_bkey_exists ( c , ptr - > dev ) ;
struct bucket * g = PTR_BUCKET ( ca , ptr , false ) ;
if ( gen_after ( g - > gc_gen , ptr - > gen ) )
g - > gc_gen = ptr - > gen ;
}
percpu_up_read ( & c - > mark_lock ) ;
return false ;
}
2020-06-15 22:10:54 +03:00
/*
* For recalculating oldest gen , we only need to walk keys in leaf nodes ; btree
* node pointers currently never have cached pointers that can become stale :
*/
2020-10-13 10:58:50 +03:00
static int bch2_gc_btree_gens ( struct bch_fs * c , enum btree_id btree_id )
2020-06-15 22:10:54 +03:00
{
struct btree_trans trans ;
struct btree_iter * iter ;
struct bkey_s_c k ;
2020-12-17 23:08:58 +03:00
struct bkey_buf sk ;
2020-10-13 10:58:50 +03:00
int ret = 0 ;
2020-06-15 22:10:54 +03:00
2020-12-17 23:08:58 +03:00
bch2_bkey_buf_init ( & sk ) ;
2020-06-15 22:10:54 +03:00
bch2_trans_init ( & trans , c , 0 , 0 ) ;
2020-10-13 10:58:50 +03:00
iter = bch2_trans_get_iter ( & trans , btree_id , POS_MIN ,
bcachefs: Start using bpos.snapshot field
This patch starts treating the bpos.snapshot field like part of the key
in the btree code:
* bpos_successor() and bpos_predecessor() now include the snapshot field
* Keys in btrees that will be using snapshots (extents, inodes, dirents
and xattrs) now always have their snapshot field set to U32_MAX
The btree iterator code gets a new flag, BTREE_ITER_ALL_SNAPSHOTS, that
determines whether we're iterating over keys in all snapshots or not -
internally, this controlls whether bkey_(successor|predecessor)
increment/decrement the snapshot field, or only the higher bits of the
key.
We add a new member to struct btree_iter, iter->snapshot: when
BTREE_ITER_ALL_SNAPSHOTS is not set, iter->pos.snapshot should always
equal iter->snapshot, which will be 0 for btrees that don't use
snapshots, and alsways U32_MAX for btrees that will use snapshots
(until we enable snapshot creation).
This patch also introduces a new metadata version number, and compat
code for reading from/writing to older versions - this isn't a forced
upgrade (yet).
Signed-off-by: Kent Overstreet <kent.overstreet@gmail.com>
Signed-off-by: Kent Overstreet <kent.overstreet@linux.dev>
2021-03-25 01:02:16 +03:00
BTREE_ITER_PREFETCH |
BTREE_ITER_NOT_EXTENTS |
BTREE_ITER_ALL_SNAPSHOTS ) ;
2020-06-15 22:10:54 +03:00
2020-10-13 10:58:50 +03:00
while ( ( k = bch2_btree_iter_peek ( iter ) ) . k & &
! ( ret = bkey_err ( k ) ) ) {
if ( gc_btree_gens_key ( c , k ) ) {
2020-12-17 23:08:58 +03:00
bch2_bkey_buf_reassemble ( & sk , c , k ) ;
2020-10-13 10:58:50 +03:00
bch2_extent_normalize ( c , bkey_i_to_s ( sk . k ) ) ;
2020-06-15 22:10:54 +03:00
2020-10-13 10:58:50 +03:00
bch2_btree_iter_set_pos ( iter , bkey_start_pos ( & sk . k - > k ) ) ;
2020-06-15 22:10:54 +03:00
2020-10-13 10:58:50 +03:00
bch2_trans_update ( & trans , iter , sk . k , 0 ) ;
2020-06-15 22:10:54 +03:00
2020-10-13 10:58:50 +03:00
ret = bch2_trans_commit ( & trans , NULL , NULL ,
BTREE_INSERT_NOFAIL ) ;
if ( ret = = - EINTR )
continue ;
if ( ret ) {
break ;
2020-06-15 22:10:54 +03:00
}
}
2020-10-13 10:58:50 +03:00
2021-03-21 23:55:25 +03:00
bch2_btree_iter_advance ( iter ) ;
2020-06-15 22:10:54 +03:00
}
2021-03-20 03:29:11 +03:00
bch2_trans_iter_put ( & trans , iter ) ;
2020-06-15 22:10:54 +03:00
bch2_trans_exit ( & trans ) ;
2020-12-17 23:08:58 +03:00
bch2_bkey_buf_exit ( & sk , c ) ;
2020-10-13 10:58:50 +03:00
2020-06-15 22:10:54 +03:00
return ret ;
}
int bch2_gc_gens ( struct bch_fs * c )
{
struct bch_dev * ca ;
struct bucket_array * buckets ;
struct bucket * g ;
unsigned i ;
int ret ;
2020-06-18 00:30:38 +03:00
/*
* Ideally we would be using state_lock and not gc_lock here , but that
* introduces a deadlock in the RO path - we currently take the state
* lock at the start of going RO , thus the gc thread may get stuck :
*/
down_read ( & c - > gc_lock ) ;
2020-06-15 22:10:54 +03:00
for_each_member_device ( ca , c , i ) {
down_read ( & ca - > bucket_lock ) ;
buckets = bucket_array ( ca ) ;
for_each_bucket ( g , buckets )
g - > gc_gen = g - > mark . gen ;
up_read ( & ca - > bucket_lock ) ;
}
for ( i = 0 ; i < BTREE_ID_NR ; i + + )
if ( btree_node_type_needs_gc ( i ) ) {
ret = bch2_gc_btree_gens ( c , i ) ;
2020-07-22 00:12:39 +03:00
if ( ret ) {
bch_err ( c , " error recalculating oldest_gen: %i " , ret ) ;
2020-06-15 22:10:54 +03:00
goto err ;
2020-07-22 00:12:39 +03:00
}
2020-06-15 22:10:54 +03:00
}
for_each_member_device ( ca , c , i ) {
down_read ( & ca - > bucket_lock ) ;
buckets = bucket_array ( ca ) ;
for_each_bucket ( g , buckets )
g - > oldest_gen = g - > gc_gen ;
up_read ( & ca - > bucket_lock ) ;
}
2020-07-22 00:12:39 +03:00
c - > gc_count + + ;
2020-06-15 22:10:54 +03:00
err :
2020-06-18 00:30:38 +03:00
up_read ( & c - > gc_lock ) ;
2020-06-15 22:10:54 +03:00
return ret ;
}
2017-03-17 09:18:50 +03:00
/* Btree coalescing */
static void recalc_packed_keys ( struct btree * b )
{
2018-08-12 02:12:05 +03:00
struct bset * i = btree_bset_first ( b ) ;
2017-03-17 09:18:50 +03:00
struct bkey_packed * k ;
memset ( & b - > nr , 0 , sizeof ( b - > nr ) ) ;
BUG_ON ( b - > nsets ! = 1 ) ;
2018-08-12 02:12:05 +03:00
vstruct_for_each ( i , k )
2017-03-17 09:18:50 +03:00
btree_keys_account_key_add ( & b - > nr , 0 , k ) ;
}
static void bch2_coalesce_nodes ( struct bch_fs * c , struct btree_iter * iter ,
struct btree * old_nodes [ GC_MERGE_NODES ] )
{
struct btree * parent = btree_node_parent ( iter , old_nodes [ 0 ] ) ;
unsigned i , nr_old_nodes , nr_new_nodes , u64s = 0 ;
unsigned blocks = btree_blocks ( c ) * 2 / 3 ;
struct btree * new_nodes [ GC_MERGE_NODES ] ;
struct btree_update * as ;
struct keylist keylist ;
struct bkey_format_state format_state ;
struct bkey_format new_format ;
memset ( new_nodes , 0 , sizeof ( new_nodes ) ) ;
bch2_keylist_init ( & keylist , NULL ) ;
/* Count keys that are not deleted */
for ( i = 0 ; i < GC_MERGE_NODES & & old_nodes [ i ] ; i + + )
u64s + = old_nodes [ i ] - > nr . live_u64s ;
nr_old_nodes = nr_new_nodes = i ;
/* Check if all keys in @old_nodes could fit in one fewer node */
if ( nr_old_nodes < = 1 | |
__vstruct_blocks ( struct btree_node , c - > block_bits ,
DIV_ROUND_UP ( u64s , nr_old_nodes - 1 ) ) > blocks )
return ;
/* Find a format that all keys in @old_nodes can pack into */
bch2_bkey_format_init ( & format_state ) ;
2021-03-27 03:29:04 +03:00
/*
* XXX : this won ' t correctly take it account the new min / max keys :
*/
2017-03-17 09:18:50 +03:00
for ( i = 0 ; i < nr_old_nodes ; i + + )
__bch2_btree_calc_format ( & format_state , old_nodes [ i ] ) ;
new_format = bch2_bkey_format_done ( & format_state ) ;
/* Check if repacking would make any nodes too big to fit */
for ( i = 0 ; i < nr_old_nodes ; i + + )
if ( ! bch2_btree_node_format_fits ( c , old_nodes [ i ] , & new_format ) ) {
trace_btree_gc_coalesce_fail ( c ,
BTREE_GC_COALESCE_FAIL_FORMAT_FITS ) ;
return ;
}
if ( bch2_keylist_realloc ( & keylist , NULL , 0 ,
2020-12-17 23:08:58 +03:00
BKEY_BTREE_PTR_U64s_MAX * nr_old_nodes ) ) {
2017-03-17 09:18:50 +03:00
trace_btree_gc_coalesce_fail ( c ,
BTREE_GC_COALESCE_FAIL_KEYLIST_REALLOC ) ;
return ;
}
2020-04-06 04:49:17 +03:00
as = bch2_btree_update_start ( iter - > trans , iter - > btree_id ,
2017-03-17 09:18:50 +03:00
btree_update_reserve_required ( c , parent ) + nr_old_nodes ,
BTREE_INSERT_NOFAIL |
BTREE_INSERT_USE_RESERVE ,
NULL ) ;
if ( IS_ERR ( as ) ) {
trace_btree_gc_coalesce_fail ( c ,
BTREE_GC_COALESCE_FAIL_RESERVE_GET ) ;
bch2_keylist_free ( & keylist , NULL ) ;
return ;
}
trace_btree_gc_coalesce ( c , old_nodes [ 0 ] ) ;
for ( i = 0 ; i < nr_old_nodes ; i + + )
bch2_btree_interior_update_will_free_node ( as , old_nodes [ i ] ) ;
/* Repack everything with @new_format and sort down to one bset */
for ( i = 0 ; i < nr_old_nodes ; i + + )
new_nodes [ i ] =
__bch2_btree_node_alloc_replacement ( as , old_nodes [ i ] ,
new_format ) ;
/*
* Conceptually we concatenate the nodes together and slice them
* up at different boundaries .
*/
for ( i = nr_new_nodes - 1 ; i > 0 ; - - i ) {
struct btree * n1 = new_nodes [ i ] ;
struct btree * n2 = new_nodes [ i - 1 ] ;
struct bset * s1 = btree_bset_first ( n1 ) ;
struct bset * s2 = btree_bset_first ( n2 ) ;
struct bkey_packed * k , * last = NULL ;
/* Calculate how many keys from @n2 we could fit inside @n1 */
u64s = 0 ;
for ( k = s2 - > start ;
k < vstruct_last ( s2 ) & &
vstruct_blocks_plus ( n1 - > data , c - > block_bits ,
u64s + k - > u64s ) < = blocks ;
2021-03-25 03:22:51 +03:00
k = bkey_next ( k ) ) {
2017-03-17 09:18:50 +03:00
last = k ;
u64s + = k - > u64s ;
}
if ( u64s = = le16_to_cpu ( s2 - > u64s ) ) {
/* n2 fits entirely in n1 */
n1 - > key . k . p = n1 - > data - > max_key = n2 - > data - > max_key ;
memcpy_u64s ( vstruct_last ( s1 ) ,
s2 - > start ,
le16_to_cpu ( s2 - > u64s ) ) ;
le16_add_cpu ( & s1 - > u64s , le16_to_cpu ( s2 - > u64s ) ) ;
set_btree_bset_end ( n1 , n1 - > set ) ;
2020-06-06 19:28:01 +03:00
six_unlock_write ( & n2 - > c . lock ) ;
2017-03-17 09:18:50 +03:00
bch2_btree_node_free_never_inserted ( c , n2 ) ;
2020-06-06 19:28:01 +03:00
six_unlock_intent ( & n2 - > c . lock ) ;
2017-03-17 09:18:50 +03:00
memmove ( new_nodes + i - 1 ,
new_nodes + i ,
sizeof ( new_nodes [ 0 ] ) * ( nr_new_nodes - i ) ) ;
new_nodes [ - - nr_new_nodes ] = NULL ;
} else if ( u64s ) {
/* move part of n2 into n1 */
n1 - > key . k . p = n1 - > data - > max_key =
bkey_unpack_pos ( n1 , last ) ;
bcachefs: Start using bpos.snapshot field
This patch starts treating the bpos.snapshot field like part of the key
in the btree code:
* bpos_successor() and bpos_predecessor() now include the snapshot field
* Keys in btrees that will be using snapshots (extents, inodes, dirents
and xattrs) now always have their snapshot field set to U32_MAX
The btree iterator code gets a new flag, BTREE_ITER_ALL_SNAPSHOTS, that
determines whether we're iterating over keys in all snapshots or not -
internally, this controlls whether bkey_(successor|predecessor)
increment/decrement the snapshot field, or only the higher bits of the
key.
We add a new member to struct btree_iter, iter->snapshot: when
BTREE_ITER_ALL_SNAPSHOTS is not set, iter->pos.snapshot should always
equal iter->snapshot, which will be 0 for btrees that don't use
snapshots, and alsways U32_MAX for btrees that will use snapshots
(until we enable snapshot creation).
This patch also introduces a new metadata version number, and compat
code for reading from/writing to older versions - this isn't a forced
upgrade (yet).
Signed-off-by: Kent Overstreet <kent.overstreet@gmail.com>
Signed-off-by: Kent Overstreet <kent.overstreet@linux.dev>
2021-03-25 01:02:16 +03:00
n2 - > data - > min_key = bpos_successor ( n1 - > data - > max_key ) ;
2017-03-17 09:18:50 +03:00
memcpy_u64s ( vstruct_last ( s1 ) ,
s2 - > start , u64s ) ;
le16_add_cpu ( & s1 - > u64s , u64s ) ;
memmove ( s2 - > start ,
vstruct_idx ( s2 , u64s ) ,
( le16_to_cpu ( s2 - > u64s ) - u64s ) * sizeof ( u64 ) ) ;
s2 - > u64s = cpu_to_le16 ( le16_to_cpu ( s2 - > u64s ) - u64s ) ;
set_btree_bset_end ( n1 , n1 - > set ) ;
set_btree_bset_end ( n2 , n2 - > set ) ;
}
}
for ( i = 0 ; i < nr_new_nodes ; i + + ) {
struct btree * n = new_nodes [ i ] ;
recalc_packed_keys ( n ) ;
btree_node_reset_sib_u64s ( n ) ;
bch2_btree_build_aux_trees ( n ) ;
2020-05-25 21:57:06 +03:00
bch2_btree_update_add_new_node ( as , n ) ;
2020-06-06 19:28:01 +03:00
six_unlock_write ( & n - > c . lock ) ;
2017-03-17 09:18:50 +03:00
bch2_btree_node_write ( c , n , SIX_LOCK_intent ) ;
}
/*
* The keys for the old nodes get deleted . We don ' t want to insert keys
* that compare equal to the keys for the new nodes we ' ll also be
* inserting - we can ' t because keys on a keylist must be strictly
* greater than the previous keys , and we also don ' t need to since the
* key for the new node will serve the same purpose ( overwriting the key
* for the old node ) .
*/
for ( i = 0 ; i < nr_old_nodes ; i + + ) {
struct bkey_i delete ;
unsigned j ;
for ( j = 0 ; j < nr_new_nodes ; j + + )
2021-03-05 00:20:16 +03:00
if ( ! bpos_cmp ( old_nodes [ i ] - > key . k . p ,
2017-03-17 09:18:50 +03:00
new_nodes [ j ] - > key . k . p ) )
goto next ;
bkey_init ( & delete . k ) ;
delete . k . p = old_nodes [ i ] - > key . k . p ;
bch2_keylist_add_in_order ( & keylist , & delete ) ;
next :
i = i ;
}
/*
* Keys for the new nodes get inserted : bch2_btree_insert_keys ( ) only
* does the lookup once and thus expects the keys to be in sorted order
* so we have to make sure the new keys are correctly ordered with
* respect to the deleted keys added in the previous loop
*/
for ( i = 0 ; i < nr_new_nodes ; i + + )
bch2_keylist_add_in_order ( & keylist , & new_nodes [ i ] - > key ) ;
/* Insert the newly coalesced nodes */
bch2_btree_insert_node ( as , parent , iter , & keylist , 0 ) ;
BUG_ON ( ! bch2_keylist_empty ( & keylist ) ) ;
2020-06-06 19:28:01 +03:00
BUG_ON ( iter - > l [ old_nodes [ 0 ] - > c . level ] . b ! = old_nodes [ 0 ] ) ;
2017-03-17 09:18:50 +03:00
bch2_btree_iter_node_replace ( iter , new_nodes [ 0 ] ) ;
for ( i = 0 ; i < nr_new_nodes ; i + + )
2020-05-25 21:57:06 +03:00
bch2_btree_update_get_open_buckets ( as , new_nodes [ i ] ) ;
2017-03-17 09:18:50 +03:00
/* Free the old nodes and update our sliding window */
for ( i = 0 ; i < nr_old_nodes ; i + + ) {
bch2_btree_node_free_inmem ( c , old_nodes [ i ] , iter ) ;
/*
* the index update might have triggered a split , in which case
* the nodes we coalesced - the new nodes we just created -
* might not be sibling nodes anymore - don ' t add them to the
* sliding window ( except the first ) :
*/
if ( ! i ) {
old_nodes [ i ] = new_nodes [ i ] ;
} else {
old_nodes [ i ] = NULL ;
}
}
2019-10-11 21:45:22 +03:00
for ( i = 0 ; i < nr_new_nodes ; i + + )
six_unlock_intent ( & new_nodes [ i ] - > c . lock ) ;
2017-03-17 09:18:50 +03:00
bch2_btree_update_done ( as ) ;
bch2_keylist_free ( & keylist , NULL ) ;
}
static int bch2_coalesce_btree ( struct bch_fs * c , enum btree_id btree_id )
{
2019-03-25 22:10:15 +03:00
struct btree_trans trans ;
struct btree_iter * iter ;
2017-03-17 09:18:50 +03:00
struct btree * b ;
bool kthread = ( current - > flags & PF_KTHREAD ) ! = 0 ;
unsigned i ;
2021-03-20 03:29:11 +03:00
int ret = 0 ;
2017-03-17 09:18:50 +03:00
/* Sliding window of adjacent btree nodes */
struct btree * merge [ GC_MERGE_NODES ] ;
u32 lock_seq [ GC_MERGE_NODES ] ;
2019-05-15 17:54:43 +03:00
bch2_trans_init ( & trans , c , 0 , 0 ) ;
2019-03-25 22:10:15 +03:00
2017-03-17 09:18:50 +03:00
/*
* XXX : We don ' t have a good way of positively matching on sibling nodes
* that have the same parent - this code works by handling the cases
* where they might not have the same parent , and is thus fragile . Ugh .
*
* Perhaps redo this to use multiple linked iterators ?
*/
memset ( merge , 0 , sizeof ( merge ) ) ;
2019-03-25 22:10:15 +03:00
__for_each_btree_node ( & trans , iter , btree_id , POS_MIN ,
2017-03-17 09:18:50 +03:00
BTREE_MAX_DEPTH , 0 ,
BTREE_ITER_PREFETCH , b ) {
memmove ( merge + 1 , merge ,
sizeof ( merge ) - sizeof ( merge [ 0 ] ) ) ;
memmove ( lock_seq + 1 , lock_seq ,
sizeof ( lock_seq ) - sizeof ( lock_seq [ 0 ] ) ) ;
merge [ 0 ] = b ;
for ( i = 1 ; i < GC_MERGE_NODES ; i + + ) {
if ( ! merge [ i ] | |
2020-06-06 19:28:01 +03:00
! six_relock_intent ( & merge [ i ] - > c . lock , lock_seq [ i ] ) )
2017-03-17 09:18:50 +03:00
break ;
2020-06-06 19:28:01 +03:00
if ( merge [ i ] - > c . level ! = merge [ 0 ] - > c . level ) {
six_unlock_intent ( & merge [ i ] - > c . lock ) ;
2017-03-17 09:18:50 +03:00
break ;
}
}
memset ( merge + i , 0 , ( GC_MERGE_NODES - i ) * sizeof ( merge [ 0 ] ) ) ;
2019-03-25 22:10:15 +03:00
bch2_coalesce_nodes ( c , iter , merge ) ;
2017-03-17 09:18:50 +03:00
for ( i = 1 ; i < GC_MERGE_NODES & & merge [ i ] ; i + + ) {
2020-06-06 19:28:01 +03:00
lock_seq [ i ] = merge [ i ] - > c . lock . state . seq ;
six_unlock_intent ( & merge [ i ] - > c . lock ) ;
2017-03-17 09:18:50 +03:00
}
2020-06-06 19:28:01 +03:00
lock_seq [ 0 ] = merge [ 0 ] - > c . lock . state . seq ;
2017-03-17 09:18:50 +03:00
if ( kthread & & kthread_should_stop ( ) ) {
2021-03-20 03:29:11 +03:00
ret = - ESHUTDOWN ;
break ;
2017-03-17 09:18:50 +03:00
}
2019-03-25 22:10:15 +03:00
bch2_trans_cond_resched ( & trans ) ;
2017-03-17 09:18:50 +03:00
/*
* If the parent node wasn ' t relocked , it might have been split
* and the nodes in our sliding window might not have the same
* parent anymore - blow away the sliding window :
*/
2019-03-25 22:10:15 +03:00
if ( btree_iter_node ( iter , iter - > level + 1 ) & &
! btree_node_intent_locked ( iter , iter - > level + 1 ) )
2017-03-17 09:18:50 +03:00
memset ( merge + 1 , 0 ,
( GC_MERGE_NODES - 1 ) * sizeof ( merge [ 0 ] ) ) ;
}
2021-03-20 03:29:11 +03:00
bch2_trans_iter_put ( & trans , iter ) ;
return bch2_trans_exit ( & trans ) ? : ret ;
2017-03-17 09:18:50 +03:00
}
/**
* bch_coalesce - coalesce adjacent nodes with low occupancy
*/
void bch2_coalesce ( struct bch_fs * c )
{
enum btree_id id ;
down_read ( & c - > gc_lock ) ;
trace_gc_coalesce_start ( c ) ;
for ( id = 0 ; id < BTREE_ID_NR ; id + + ) {
int ret = c - > btree_roots [ id ] . b
? bch2_coalesce_btree ( c , id )
: 0 ;
if ( ret ) {
if ( ret ! = - ESHUTDOWN )
bch_err ( c , " btree coalescing failed: %d " , ret ) ;
return ;
}
}
trace_gc_coalesce_end ( c ) ;
up_read ( & c - > gc_lock ) ;
}
static int bch2_gc_thread ( void * arg )
{
struct bch_fs * c = arg ;
struct io_clock * clock = & c - > io_clock [ WRITE ] ;
2021-01-21 23:28:59 +03:00
unsigned long last = atomic64_read ( & clock - > now ) ;
2017-03-17 09:18:50 +03:00
unsigned last_kick = atomic_read ( & c - > kick_gc ) ;
2018-07-23 12:32:01 +03:00
int ret ;
2017-03-17 09:18:50 +03:00
set_freezable ( ) ;
while ( 1 ) {
while ( 1 ) {
set_current_state ( TASK_INTERRUPTIBLE ) ;
if ( kthread_should_stop ( ) ) {
__set_current_state ( TASK_RUNNING ) ;
return 0 ;
}
if ( atomic_read ( & c - > kick_gc ) ! = last_kick )
break ;
if ( c - > btree_gc_periodic ) {
unsigned long next = last + c - > capacity / 16 ;
2021-01-21 23:28:59 +03:00
if ( atomic64_read ( & clock - > now ) > = next )
2017-03-17 09:18:50 +03:00
break ;
bch2_io_clock_schedule_timeout ( clock , next ) ;
} else {
schedule ( ) ;
}
try_to_freeze ( ) ;
}
__set_current_state ( TASK_RUNNING ) ;
2021-01-21 23:28:59 +03:00
last = atomic64_read ( & clock - > now ) ;
2017-03-17 09:18:50 +03:00
last_kick = atomic_read ( & c - > kick_gc ) ;
2020-06-15 22:10:54 +03:00
/*
* Full gc is currently incompatible with btree key cache :
*/
#if 0
2021-01-27 04:15:46 +03:00
ret = bch2_gc ( c , false , false ) ;
2020-06-15 22:10:54 +03:00
# else
ret = bch2_gc_gens ( c ) ;
# endif
2020-10-17 04:36:26 +03:00
if ( ret < 0 )
2018-07-23 12:32:01 +03:00
bch_err ( c , " btree gc failed: %i " , ret ) ;
2017-03-17 09:18:50 +03:00
debug_check_no_locks_held ( ) ;
}
return 0 ;
}
void bch2_gc_thread_stop ( struct bch_fs * c )
{
struct task_struct * p ;
p = c - > gc_thread ;
c - > gc_thread = NULL ;
if ( p ) {
kthread_stop ( p ) ;
put_task_struct ( p ) ;
}
}
int bch2_gc_thread_start ( struct bch_fs * c )
{
struct task_struct * p ;
2021-03-23 01:39:16 +03:00
if ( c - > gc_thread )
return 0 ;
2017-03-17 09:18:50 +03:00
2020-11-20 04:55:33 +03:00
p = kthread_create ( bch2_gc_thread , c , " bch-gc/%s " , c - > name ) ;
2021-02-23 23:16:41 +03:00
if ( IS_ERR ( p ) ) {
bch_err ( c , " error creating gc thread: %li " , PTR_ERR ( p ) ) ;
2017-03-17 09:18:50 +03:00
return PTR_ERR ( p ) ;
2021-02-23 23:16:41 +03:00
}
2017-03-17 09:18:50 +03:00
get_task_struct ( p ) ;
c - > gc_thread = p ;
wake_up_process ( p ) ;
return 0 ;
}