2017-03-17 09:18:50 +03:00
// SPDX-License-Identifier: GPL-2.0
# include "bcachefs.h"
# include "bkey_methods.h"
# include "btree_cache.h"
# include "btree_iter.h"
2019-03-08 03:46:10 +03:00
# include "btree_key_cache.h"
2017-03-17 09:18:50 +03:00
# include "btree_locking.h"
2020-03-06 02:44:59 +03:00
# include "btree_update.h"
2017-03-17 09:18:50 +03:00
# include "debug.h"
# include "extents.h"
2019-03-08 03:46:10 +03:00
# include "journal.h"
2017-03-17 09:18:50 +03:00
# include "trace.h"
# include <linux/prefetch.h>
static inline bool is_btree_node ( struct btree_iter * iter , unsigned l )
{
return l < BTREE_MAX_DEPTH & &
2019-05-14 21:08:23 +03:00
( unsigned long ) iter - > l [ l ] . b > = 128 ;
2017-03-17 09:18:50 +03:00
}
2020-01-07 06:25:09 +03:00
static inline struct bpos btree_iter_search_key ( struct btree_iter * iter )
2018-08-21 23:30:14 +03:00
{
2020-01-07 06:25:09 +03:00
struct bpos pos = iter - > pos ;
2018-08-22 00:38:41 +03:00
2020-01-07 06:25:09 +03:00
if ( ( iter - > flags & BTREE_ITER_IS_EXTENTS ) & &
bkey_cmp ( pos , POS_MAX ) )
pos = bkey_successor ( pos ) ;
return pos ;
2018-08-21 23:30:14 +03:00
}
2020-03-02 21:38:19 +03:00
static inline bool btree_iter_pos_before_node ( struct btree_iter * iter ,
struct btree * b )
{
2020-01-07 21:29:32 +03:00
return bkey_cmp ( btree_iter_search_key ( iter ) , b - > data - > min_key ) < 0 ;
2020-03-02 21:38:19 +03:00
}
static inline bool btree_iter_pos_after_node ( struct btree_iter * iter ,
struct btree * b )
{
return bkey_cmp ( b - > key . k . p , btree_iter_search_key ( iter ) ) < 0 ;
}
static inline bool btree_iter_pos_in_node ( struct btree_iter * iter ,
struct btree * b )
{
return iter - > btree_id = = b - > c . btree_id & &
! btree_iter_pos_before_node ( iter , b ) & &
! btree_iter_pos_after_node ( iter , b ) ;
}
2017-03-17 09:18:50 +03:00
/* Btree node locking: */
void bch2_btree_node_unlock_write ( struct btree * b , struct btree_iter * iter )
{
2019-10-29 02:35:13 +03:00
bch2_btree_node_unlock_write_inlined ( b , iter ) ;
2017-03-17 09:18:50 +03:00
}
void __bch2_btree_node_lock_write ( struct btree * b , struct btree_iter * iter )
{
struct btree_iter * linked ;
unsigned readers = 0 ;
2019-08-12 21:35:34 +03:00
EBUG_ON ( ! btree_node_intent_locked ( iter , b - > c . level ) ) ;
2017-03-17 09:18:50 +03:00
2019-03-28 05:03:30 +03:00
trans_for_each_iter ( iter - > trans , linked )
2020-06-06 19:28:01 +03:00
if ( linked - > l [ b - > c . level ] . b = = b & &
btree_node_read_locked ( linked , b - > c . level ) )
2017-03-17 09:18:50 +03:00
readers + + ;
/*
* Must drop our read locks before calling six_lock_write ( ) -
* six_unlock ( ) won ' t do wakeups until the reader count
* goes to 0 , and it ' s safe because we have the node intent
* locked :
*/
atomic64_sub ( __SIX_VAL ( read_lock , readers ) ,
2020-06-06 19:28:01 +03:00
& b - > c . lock . state . counter ) ;
2019-03-26 00:06:42 +03:00
btree_node_lock_type ( iter - > trans - > c , b , SIX_LOCK_write ) ;
2017-03-17 09:18:50 +03:00
atomic64_add ( __SIX_VAL ( read_lock , readers ) ,
2020-06-06 19:28:01 +03:00
& b - > c . lock . state . counter ) ;
2017-03-17 09:18:50 +03:00
}
bool __bch2_btree_node_relock ( struct btree_iter * iter , unsigned level )
{
struct btree * b = btree_iter_node ( iter , level ) ;
int want = __btree_lock_want ( iter , level ) ;
2019-05-14 21:08:23 +03:00
if ( ! is_btree_node ( iter , level ) )
2017-03-17 09:18:50 +03:00
return false ;
if ( race_fault ( ) )
return false ;
2019-05-14 21:08:23 +03:00
if ( six_relock_type ( & b - > c . lock , want , iter - > l [ level ] . lock_seq ) | |
( btree_node_lock_seq_matches ( iter , b , level ) & &
2020-06-12 21:58:07 +03:00
btree_node_lock_increment ( iter - > trans , b , level , want ) ) ) {
2019-05-14 21:08:23 +03:00
mark_btree_node_locked ( iter , level , want ) ;
return true ;
} else {
2017-03-17 09:18:50 +03:00
return false ;
2019-05-14 21:08:23 +03:00
}
2017-03-17 09:18:50 +03:00
}
static bool bch2_btree_node_upgrade ( struct btree_iter * iter , unsigned level )
{
struct btree * b = iter - > l [ level ] . b ;
EBUG_ON ( btree_lock_want ( iter , level ) ! = BTREE_NODE_INTENT_LOCKED ) ;
if ( ! is_btree_node ( iter , level ) )
return false ;
if ( btree_node_intent_locked ( iter , level ) )
return true ;
if ( race_fault ( ) )
return false ;
if ( btree_node_locked ( iter , level )
2020-06-06 19:28:01 +03:00
? six_lock_tryupgrade ( & b - > c . lock )
: six_relock_type ( & b - > c . lock , SIX_LOCK_intent , iter - > l [ level ] . lock_seq ) )
2017-03-17 09:18:50 +03:00
goto success ;
2019-05-14 21:08:23 +03:00
if ( btree_node_lock_seq_matches ( iter , b , level ) & &
2020-06-12 21:58:07 +03:00
btree_node_lock_increment ( iter - > trans , b , level , BTREE_NODE_INTENT_LOCKED ) ) {
2017-03-17 09:18:50 +03:00
btree_node_unlock ( iter , level ) ;
goto success ;
}
return false ;
success :
mark_btree_node_intent_locked ( iter , level ) ;
return true ;
}
static inline bool btree_iter_get_locks ( struct btree_iter * iter ,
2019-05-15 17:08:55 +03:00
bool upgrade , bool trace )
2017-03-17 09:18:50 +03:00
{
unsigned l = iter - > level ;
int fail_idx = - 1 ;
do {
if ( ! btree_iter_node ( iter , l ) )
break ;
if ( ! ( upgrade
? bch2_btree_node_upgrade ( iter , l )
: bch2_btree_node_relock ( iter , l ) ) ) {
2019-05-15 17:08:55 +03:00
if ( trace )
( upgrade
? trace_node_upgrade_fail
: trace_node_relock_fail ) ( l , iter - > l [ l ] . lock_seq ,
2019-05-14 21:08:23 +03:00
is_btree_node ( iter , l )
? 0
: ( unsigned long ) iter - > l [ l ] . b ,
is_btree_node ( iter , l )
? iter - > l [ l ] . b - > c . lock . state . seq
: 0 ) ;
2017-03-17 09:18:50 +03:00
fail_idx = l ;
btree_iter_set_dirty ( iter , BTREE_ITER_NEED_TRAVERSE ) ;
}
l + + ;
} while ( l < iter - > locks_want ) ;
/*
* When we fail to get a lock , we have to ensure that any child nodes
* can ' t be relocked so bch2_btree_iter_traverse has to walk back up to
* the node that we failed to relock :
*/
while ( fail_idx > = 0 ) {
btree_node_unlock ( iter , fail_idx ) ;
2019-05-14 21:08:23 +03:00
iter - > l [ fail_idx ] . b = BTREE_ITER_NO_NODE_GET_LOCKS ;
2017-03-17 09:18:50 +03:00
- - fail_idx ;
}
if ( iter - > uptodate = = BTREE_ITER_NEED_RELOCK )
iter - > uptodate = BTREE_ITER_NEED_PEEK ;
2019-03-28 05:03:30 +03:00
bch2_btree_trans_verify_locks ( iter - > trans ) ;
2017-03-17 09:18:50 +03:00
return iter - > uptodate < BTREE_ITER_NEED_RELOCK ;
}
2020-06-16 02:53:46 +03:00
static struct bpos btree_node_pos ( struct btree_bkey_cached_common * _b ,
enum btree_iter_type type )
{
return type ! = BTREE_ITER_CACHED
? container_of ( _b , struct btree , c ) - > key . k . p
: container_of ( _b , struct bkey_cached , c ) - > key . pos ;
}
2017-03-17 09:18:50 +03:00
/* Slowpath: */
bool __bch2_btree_node_lock ( struct btree * b , struct bpos pos ,
2020-06-12 21:58:07 +03:00
unsigned level , struct btree_iter * iter ,
2020-06-13 05:29:48 +03:00
enum six_lock_type type ,
six_lock_should_sleep_fn should_sleep_fn ,
void * p )
2017-03-17 09:18:50 +03:00
{
2020-06-12 21:58:07 +03:00
struct btree_trans * trans = iter - > trans ;
2017-03-17 09:18:50 +03:00
struct btree_iter * linked ;
2020-06-13 05:29:48 +03:00
u64 start_time = local_clock ( ) ;
2017-03-17 09:18:50 +03:00
bool ret = true ;
2018-07-24 23:42:27 +03:00
/* Check if it's safe to block: */
2020-06-12 21:58:07 +03:00
trans_for_each_iter ( trans , linked ) {
2017-03-17 09:18:50 +03:00
if ( ! linked - > nodes_locked )
continue ;
/*
* Can ' t block taking an intent lock if we have _any_ nodes read
* locked :
*
* - Our read lock blocks another thread with an intent lock on
* the same node from getting a write lock , and thus from
* dropping its intent lock
*
* - And the other thread may have multiple nodes intent locked :
* both the node we want to intent lock , and the node we
* already have read locked - deadlock :
*/
if ( type = = SIX_LOCK_intent & &
linked - > nodes_locked ! = linked - > nodes_intent_locked ) {
2020-06-12 21:58:07 +03:00
if ( ! ( trans - > nounlock ) ) {
2017-03-17 09:18:50 +03:00
linked - > locks_want = max_t ( unsigned ,
linked - > locks_want ,
__fls ( linked - > nodes_locked ) + 1 ) ;
2020-06-12 21:58:07 +03:00
if ( ! btree_iter_get_locks ( linked , true , false ) )
ret = false ;
} else {
ret = false ;
2017-03-17 09:18:50 +03:00
}
}
/*
* Interior nodes must be locked before their descendants : if
* another iterator has possible descendants locked of the node
* we ' re about to lock , it must have the ancestors locked too :
*/
if ( linked - > btree_id = = iter - > btree_id & &
level > __fls ( linked - > nodes_locked ) ) {
2020-06-12 21:58:07 +03:00
if ( ! ( trans - > nounlock ) ) {
2018-07-24 23:42:27 +03:00
linked - > locks_want =
max ( level + 1 , max_t ( unsigned ,
linked - > locks_want ,
iter - > locks_want ) ) ;
2020-06-12 21:58:07 +03:00
if ( ! btree_iter_get_locks ( linked , true , false ) )
ret = false ;
} else {
ret = false ;
2017-03-17 09:18:50 +03:00
}
2020-06-12 21:58:07 +03:00
}
/* Must lock btree nodes in key order: */
2019-03-08 03:46:10 +03:00
if ( ( cmp_int ( iter - > btree_id , linked - > btree_id ) ? :
- cmp_int ( btree_iter_type ( iter ) , btree_iter_type ( linked ) ) ) < 0 )
2020-06-12 21:58:07 +03:00
ret = false ;
if ( iter - > btree_id = = linked - > btree_id & &
btree_node_locked ( linked , level ) & &
2020-06-16 02:53:46 +03:00
bkey_cmp ( pos , btree_node_pos ( ( void * ) linked - > l [ level ] . b ,
btree_iter_type ( linked ) ) ) < = 0 )
2017-03-17 09:18:50 +03:00
ret = false ;
2020-06-12 21:58:07 +03:00
/*
* Recheck if this is a node we already have locked - since one
* of the get_locks ( ) calls might ' ve successfully
* upgraded / relocked it :
*/
if ( linked - > l [ level ] . b = = b & &
btree_node_locked_type ( linked , level ) > = type ) {
six_lock_increment ( & b - > c . lock , type ) ;
return true ;
2017-03-17 09:18:50 +03:00
}
}
2019-04-23 07:10:08 +03:00
if ( unlikely ( ! ret ) ) {
2019-05-15 17:54:43 +03:00
trace_trans_restart_would_deadlock ( iter - > trans - > ip ) ;
2019-04-23 07:10:08 +03:00
return false ;
}
2018-07-13 06:30:45 +03:00
2020-06-13 05:29:48 +03:00
if ( six_trylock_type ( & b - > c . lock , type ) )
return true ;
if ( six_lock_type ( & b - > c . lock , type , should_sleep_fn , p ) )
return false ;
bch2_time_stats_update ( & trans - > c - > times [ lock_to_time_stat ( type ) ] ,
start_time ) ;
2019-04-23 07:10:08 +03:00
return true ;
2017-03-17 09:18:50 +03:00
}
/* Btree iterator locking: */
# ifdef CONFIG_BCACHEFS_DEBUG
2020-03-15 23:15:08 +03:00
static void bch2_btree_iter_verify_locks ( struct btree_iter * iter )
2017-03-17 09:18:50 +03:00
{
unsigned l ;
2020-06-13 05:29:48 +03:00
if ( ! ( iter - > trans - > iters_linked & ( 1ULL < < iter - > idx ) ) ) {
BUG_ON ( iter - > nodes_locked ) ;
return ;
}
2019-03-08 03:46:10 +03:00
for ( l = 0 ; is_btree_node ( iter , l ) ; l + + ) {
2017-03-17 09:18:50 +03:00
if ( iter - > uptodate > = BTREE_ITER_NEED_RELOCK & &
! btree_node_locked ( iter , l ) )
continue ;
BUG_ON ( btree_lock_want ( iter , l ) ! =
btree_node_locked_type ( iter , l ) ) ;
}
}
2018-11-23 13:19:25 +03:00
2019-03-28 05:03:30 +03:00
void bch2_btree_trans_verify_locks ( struct btree_trans * trans )
2018-11-23 13:19:25 +03:00
{
2019-03-28 05:03:30 +03:00
struct btree_iter * iter ;
2018-11-23 13:19:25 +03:00
2020-06-13 05:29:48 +03:00
trans_for_each_iter_all ( trans , iter )
2019-03-28 05:03:30 +03:00
bch2_btree_iter_verify_locks ( iter ) ;
2018-11-23 13:19:25 +03:00
}
2020-03-15 23:15:08 +03:00
# else
static inline void bch2_btree_iter_verify_locks ( struct btree_iter * iter ) { }
2017-03-17 09:18:50 +03:00
# endif
__flatten
2019-03-08 03:46:10 +03:00
bool bch2_btree_iter_relock ( struct btree_iter * iter , bool trace )
2017-03-17 09:18:50 +03:00
{
2019-11-04 23:56:04 +03:00
return btree_iter_get_locks ( iter , false , trace ) ;
2017-03-17 09:18:50 +03:00
}
bool __bch2_btree_iter_upgrade ( struct btree_iter * iter ,
unsigned new_locks_want )
{
struct btree_iter * linked ;
EBUG_ON ( iter - > locks_want > = new_locks_want ) ;
iter - > locks_want = new_locks_want ;
2019-05-15 17:08:55 +03:00
if ( btree_iter_get_locks ( iter , true , true ) )
2017-03-17 09:18:50 +03:00
return true ;
/*
* Ancestor nodes must be locked before child nodes , so set locks_want
* on iterators that might lock ancestors before us to avoid getting
* - EINTR later :
*/
2019-03-28 05:03:30 +03:00
trans_for_each_iter ( iter - > trans , linked )
if ( linked ! = iter & &
linked - > btree_id = = iter - > btree_id & &
2017-03-17 09:18:50 +03:00
linked - > locks_want < new_locks_want ) {
linked - > locks_want = new_locks_want ;
2019-05-15 17:08:55 +03:00
btree_iter_get_locks ( linked , true , false ) ;
2017-03-17 09:18:50 +03:00
}
return false ;
}
bool __bch2_btree_iter_upgrade_nounlock ( struct btree_iter * iter ,
unsigned new_locks_want )
{
unsigned l = iter - > level ;
EBUG_ON ( iter - > locks_want > = new_locks_want ) ;
iter - > locks_want = new_locks_want ;
do {
if ( ! btree_iter_node ( iter , l ) )
break ;
if ( ! bch2_btree_node_upgrade ( iter , l ) ) {
iter - > locks_want = l ;
return false ;
}
l + + ;
} while ( l < iter - > locks_want ) ;
return true ;
}
void __bch2_btree_iter_downgrade ( struct btree_iter * iter ,
unsigned downgrade_to )
{
2020-06-08 20:26:48 +03:00
unsigned l , new_locks_want = downgrade_to ? :
( iter - > flags & BTREE_ITER_INTENT ? 1 : 0 ) ;
2017-03-17 09:18:50 +03:00
2020-06-08 20:26:48 +03:00
if ( iter - > locks_want < downgrade_to ) {
iter - > locks_want = new_locks_want ;
2017-03-17 09:18:50 +03:00
2020-06-08 20:26:48 +03:00
while ( iter - > nodes_locked & &
( l = __fls ( iter - > nodes_locked ) ) > = iter - > locks_want ) {
if ( l > iter - > level ) {
btree_node_unlock ( iter , l ) ;
2017-03-17 09:18:50 +03:00
} else {
2020-06-08 20:26:48 +03:00
if ( btree_node_intent_locked ( iter , l ) ) {
six_lock_downgrade ( & iter - > l [ l ] . b - > c . lock ) ;
iter - > nodes_intent_locked ^ = 1 < < l ;
2017-03-17 09:18:50 +03:00
}
break ;
}
}
}
2018-11-23 13:19:25 +03:00
2019-03-28 05:03:30 +03:00
bch2_btree_trans_verify_locks ( iter - > trans ) ;
2017-03-17 09:18:50 +03:00
}
2020-06-08 20:26:48 +03:00
void bch2_trans_downgrade ( struct btree_trans * trans )
{
struct btree_iter * iter ;
trans_for_each_iter ( trans , iter )
bch2_btree_iter_downgrade ( iter ) ;
}
2019-05-15 16:47:40 +03:00
/* Btree transaction locking: */
bool bch2_trans_relock ( struct btree_trans * trans )
2019-03-28 05:03:30 +03:00
{
struct btree_iter * iter ;
bool ret = true ;
trans_for_each_iter ( trans , iter )
2019-05-15 17:08:55 +03:00
if ( iter - > uptodate = = BTREE_ITER_NEED_RELOCK )
ret & = bch2_btree_iter_relock ( iter , true ) ;
2019-03-28 05:03:30 +03:00
return ret ;
}
2019-05-15 16:47:40 +03:00
void bch2_trans_unlock ( struct btree_trans * trans )
2019-03-28 05:03:30 +03:00
{
struct btree_iter * iter ;
trans_for_each_iter ( trans , iter )
__bch2_btree_iter_unlock ( iter ) ;
}
2017-03-17 09:18:50 +03:00
/* Btree iterator: */
# ifdef CONFIG_BCACHEFS_DEBUG
2020-06-16 02:53:46 +03:00
static void bch2_btree_iter_verify_cached ( struct btree_iter * iter )
{
struct bkey_cached * ck ;
bool locked = btree_node_locked ( iter , 0 ) ;
if ( ! bch2_btree_node_relock ( iter , 0 ) )
return ;
ck = ( void * ) iter - > l [ 0 ] . b ;
BUG_ON ( ck - > key . btree_id ! = iter - > btree_id | |
bkey_cmp ( ck - > key . pos , iter - > pos ) ) ;
if ( ! locked )
btree_node_unlock ( iter , 0 ) ;
}
2020-02-19 00:17:55 +03:00
static void bch2_btree_iter_verify_level ( struct btree_iter * iter ,
unsigned level )
2017-03-17 09:18:50 +03:00
{
2020-01-07 06:25:09 +03:00
struct bpos pos = btree_iter_search_key ( iter ) ;
2020-02-19 00:17:55 +03:00
struct btree_iter_level * l = & iter - > l [ level ] ;
2017-03-17 09:18:50 +03:00
struct btree_node_iter tmp = l - > iter ;
2020-02-19 00:17:55 +03:00
bool locked = btree_node_locked ( iter , level ) ;
struct bkey_packed * p , * k ;
char buf1 [ 100 ] , buf2 [ 100 ] ;
const char * msg ;
2017-03-17 09:18:50 +03:00
2019-03-28 08:51:47 +03:00
if ( ! debug_check_iterators ( iter - > trans - > c ) )
return ;
2020-06-16 02:53:46 +03:00
if ( btree_iter_type ( iter ) = = BTREE_ITER_CACHED ) {
if ( ! level )
bch2_btree_iter_verify_cached ( iter ) ;
return ;
}
2020-02-19 00:17:55 +03:00
BUG_ON ( iter - > level < iter - > min_depth ) ;
if ( ! btree_iter_node ( iter , level ) )
return ;
if ( ! bch2_btree_node_relock ( iter , level ) )
2016-07-22 06:05:06 +03:00
return ;
2020-02-19 00:17:55 +03:00
/*
* Ideally this invariant would always be true , and hopefully in the
* future it will be , but for now set_pos_same_leaf ( ) breaks it :
*/
BUG_ON ( iter - > uptodate < BTREE_ITER_NEED_TRAVERSE & &
! btree_iter_pos_in_node ( iter , l - > b ) ) ;
/*
* node iterators don ' t use leaf node iterator :
*/
if ( btree_iter_type ( iter ) = = BTREE_ITER_NODES & &
level < = iter - > min_depth )
goto unlock ;
2020-03-02 21:38:19 +03:00
2020-02-19 00:17:55 +03:00
bch2_btree_node_iter_verify ( & l - > iter , l - > b ) ;
2017-03-17 09:18:50 +03:00
/*
* For interior nodes , the iterator will have skipped past
* deleted keys :
2016-07-22 06:05:06 +03:00
*
* For extents , the iterator may have skipped past deleted keys ( but not
* whiteouts )
2017-03-17 09:18:50 +03:00
*/
2020-02-19 00:17:55 +03:00
p = level | | btree_node_type_is_extents ( iter - > btree_id )
? bch2_btree_node_iter_prev_filter ( & tmp , l - > b , KEY_TYPE_discard )
: bch2_btree_node_iter_prev_all ( & tmp , l - > b ) ;
k = bch2_btree_node_iter_peek_all ( & l - > iter , l - > b ) ;
2017-03-17 09:18:50 +03:00
2020-02-19 00:17:55 +03:00
if ( p & & bkey_iter_pos_cmp ( l - > b , p , & pos ) > = 0 ) {
msg = " before " ;
goto err ;
2017-03-17 09:18:50 +03:00
}
2020-02-19 00:17:55 +03:00
if ( k & & bkey_iter_pos_cmp ( l - > b , k , & pos ) < 0 ) {
msg = " after " ;
goto err ;
}
unlock :
if ( ! locked )
btree_node_unlock ( iter , level ) ;
return ;
err :
strcpy ( buf1 , " (none) " ) ;
strcpy ( buf2 , " (none) " ) ;
if ( p ) {
struct bkey uk = bkey_unpack_key ( l - > b , p ) ;
bch2_bkey_to_text ( & PBUF ( buf1 ) , & uk ) ;
}
2017-03-17 09:18:50 +03:00
2020-02-19 00:17:55 +03:00
if ( k ) {
struct bkey uk = bkey_unpack_key ( l - > b , k ) ;
bch2_bkey_to_text ( & PBUF ( buf2 ) , & uk ) ;
2017-03-17 09:18:50 +03:00
}
2020-02-19 00:17:55 +03:00
panic ( " iterator should be %s key at level %u: \n "
" iter pos %s %llu:%llu \n "
" prev key %s \n "
" cur key %s \n " ,
msg , level ,
iter - > flags & BTREE_ITER_IS_EXTENTS ? " > " : " => " ,
iter - > pos . inode , iter - > pos . offset ,
buf1 , buf2 ) ;
2017-03-17 09:18:50 +03:00
}
2020-02-19 00:17:55 +03:00
static void bch2_btree_iter_verify ( struct btree_iter * iter )
2017-03-17 09:18:50 +03:00
{
2020-02-19 00:17:55 +03:00
unsigned i ;
2017-03-17 09:18:50 +03:00
2020-02-19 00:17:55 +03:00
bch2_btree_trans_verify_locks ( iter - > trans ) ;
for ( i = 0 ; i < BTREE_MAX_DEPTH ; i + + )
bch2_btree_iter_verify_level ( iter , i ) ;
}
void bch2_btree_trans_verify_iters ( struct btree_trans * trans , struct btree * b )
{
struct btree_iter * iter ;
if ( ! debug_check_iterators ( trans - > c ) )
2019-03-28 08:51:47 +03:00
return ;
2020-02-19 00:17:55 +03:00
trans_for_each_iter_with_node ( trans , b , iter )
bch2_btree_iter_verify_level ( iter , b - > c . level ) ;
2017-03-17 09:18:50 +03:00
}
2016-07-22 06:05:06 +03:00
# else
2020-03-15 23:15:08 +03:00
static inline void bch2_btree_iter_verify_level ( struct btree_iter * iter , unsigned l ) { }
static inline void bch2_btree_iter_verify ( struct btree_iter * iter ) { }
2016-07-22 06:05:06 +03:00
2017-03-17 09:18:50 +03:00
# endif
2019-08-21 00:43:47 +03:00
static void btree_node_iter_set_set_pos ( struct btree_node_iter * iter ,
struct btree * b ,
struct bset_tree * t ,
struct bkey_packed * k )
{
struct btree_node_iter_set * set ;
btree_node_iter_for_each ( iter , set )
if ( set - > end = = t - > end_offset ) {
set - > k = __btree_node_key_to_offset ( b , k ) ;
bch2_btree_node_iter_sort ( iter , b ) ;
return ;
}
bch2_btree_node_iter_push ( iter , b , k , btree_bkey_last ( b , t ) ) ;
}
2019-10-02 16:56:39 +03:00
static void __bch2_btree_iter_fix_key_modified ( struct btree_iter * iter ,
2020-01-07 06:25:09 +03:00
struct btree * b ,
struct bkey_packed * where )
2019-10-02 16:56:39 +03:00
{
2020-01-07 06:25:09 +03:00
struct btree_iter_level * l = & iter - > l [ b - > c . level ] ;
struct bpos pos = btree_iter_search_key ( iter ) ;
2019-10-02 16:56:39 +03:00
2020-01-07 06:25:09 +03:00
if ( where ! = bch2_btree_node_iter_peek_all ( & l - > iter , l - > b ) )
return ;
if ( bkey_iter_pos_cmp ( l - > b , where , & pos ) < 0 )
bch2_btree_node_iter_advance ( & l - > iter , l - > b ) ;
btree_iter_set_dirty ( iter , BTREE_ITER_NEED_PEEK ) ;
2019-10-02 16:56:39 +03:00
}
void bch2_btree_iter_fix_key_modified ( struct btree_iter * iter ,
struct btree * b ,
struct bkey_packed * where )
{
struct btree_iter * linked ;
trans_for_each_iter_with_node ( iter - > trans , b , linked ) {
__bch2_btree_iter_fix_key_modified ( linked , b , where ) ;
2020-02-19 00:17:55 +03:00
bch2_btree_iter_verify_level ( linked , b - > c . level ) ;
2019-10-02 16:56:39 +03:00
}
}
2017-03-17 09:18:50 +03:00
static void __bch2_btree_node_iter_fix ( struct btree_iter * iter ,
struct btree * b ,
struct btree_node_iter * node_iter ,
struct bset_tree * t ,
struct bkey_packed * where ,
unsigned clobber_u64s ,
unsigned new_u64s )
{
const struct bkey_packed * end = btree_bkey_last ( b , t ) ;
struct btree_node_iter_set * set ;
unsigned offset = __btree_node_key_to_offset ( b , where ) ;
int shift = new_u64s - clobber_u64s ;
2016-07-22 06:05:06 +03:00
unsigned old_end = t - > end_offset - shift ;
2019-09-08 02:17:40 +03:00
unsigned orig_iter_pos = node_iter - > data [ 0 ] . k ;
bool iter_current_key_modified =
orig_iter_pos > = offset & &
orig_iter_pos < = offset + clobber_u64s ;
2020-01-07 06:25:09 +03:00
struct bpos iter_pos = btree_iter_search_key ( iter ) ;
2017-03-17 09:18:50 +03:00
btree_node_iter_for_each ( node_iter , set )
if ( set - > end = = old_end )
goto found ;
/* didn't find the bset in the iterator - might have to readd it: */
if ( new_u64s & &
2020-01-07 06:25:09 +03:00
bkey_iter_pos_cmp ( b , where , & iter_pos ) > = 0 ) {
2017-03-17 09:18:50 +03:00
bch2_btree_node_iter_push ( node_iter , b , where , end ) ;
2019-09-08 02:17:40 +03:00
goto fixup_done ;
} else {
/* Iterator is after key that changed */
2019-09-19 23:07:41 +03:00
return ;
2017-03-17 09:18:50 +03:00
}
found :
2016-07-22 06:05:06 +03:00
set - > end = t - > end_offset ;
2017-03-17 09:18:50 +03:00
/* Iterator hasn't gotten to the key that changed yet: */
if ( set - > k < offset )
2019-09-19 23:07:41 +03:00
return ;
2017-03-17 09:18:50 +03:00
if ( new_u64s & &
2020-01-07 06:25:09 +03:00
bkey_iter_pos_cmp ( b , where , & iter_pos ) > = 0 ) {
2017-03-17 09:18:50 +03:00
set - > k = offset ;
} else if ( set - > k < offset + clobber_u64s ) {
set - > k = offset + new_u64s ;
if ( set - > k = = set - > end )
bch2_btree_node_iter_set_drop ( node_iter , set ) ;
} else {
2019-09-08 02:17:40 +03:00
/* Iterator is after key that changed */
2017-03-17 09:18:50 +03:00
set - > k = ( int ) set - > k + shift ;
2019-09-19 23:07:41 +03:00
return ;
2017-03-17 09:18:50 +03:00
}
bch2_btree_node_iter_sort ( node_iter , b ) ;
2019-09-08 02:17:40 +03:00
fixup_done :
if ( node_iter - > data [ 0 ] . k ! = orig_iter_pos )
iter_current_key_modified = true ;
2018-12-06 20:01:29 +03:00
2017-03-17 09:18:50 +03:00
/*
2019-08-21 00:43:47 +03:00
* When a new key is added , and the node iterator now points to that
* key , the iterator might have skipped past deleted keys that should
* come after the key the iterator now points to . We have to rewind to
2019-09-08 02:17:40 +03:00
* before those deleted keys - otherwise
* bch2_btree_node_iter_prev_all ( ) breaks :
2017-03-17 09:18:50 +03:00
*/
2019-08-21 00:43:47 +03:00
if ( ! bch2_btree_node_iter_end ( node_iter ) & &
2019-09-08 02:17:40 +03:00
iter_current_key_modified & &
2019-08-21 00:43:47 +03:00
( b - > c . level | |
2020-01-31 21:26:05 +03:00
btree_node_type_is_extents ( iter - > btree_id ) ) ) {
2019-08-21 00:43:47 +03:00
struct bset_tree * t ;
struct bkey_packed * k , * k2 , * p ;
k = bch2_btree_node_iter_peek_all ( node_iter , b ) ;
2017-03-17 09:18:50 +03:00
for_each_bset ( b , t ) {
2019-08-21 00:43:47 +03:00
bool set_pos = false ;
if ( node_iter - > data [ 0 ] . end = = t - > end_offset )
2017-03-17 09:18:50 +03:00
continue ;
2019-08-21 00:43:47 +03:00
k2 = bch2_btree_node_iter_bset_pos ( node_iter , b , t ) ;
while ( ( p = bch2_bkey_prev_all ( b , t , k2 ) ) & &
bkey_iter_cmp ( b , k , p ) < 0 ) {
k2 = p ;
set_pos = true ;
2017-03-17 09:18:50 +03:00
}
2019-08-21 00:43:47 +03:00
if ( set_pos )
btree_node_iter_set_set_pos ( node_iter ,
b , t , k2 ) ;
2017-03-17 09:18:50 +03:00
}
}
2019-08-21 00:43:47 +03:00
2019-09-08 02:17:40 +03:00
if ( ! b - > c . level & &
node_iter = = & iter - > l [ 0 ] . iter & &
2020-02-19 00:17:55 +03:00
iter_current_key_modified )
2019-09-08 02:17:40 +03:00
btree_iter_set_dirty ( iter , BTREE_ITER_NEED_PEEK ) ;
2017-03-17 09:18:50 +03:00
}
void bch2_btree_node_iter_fix ( struct btree_iter * iter ,
2018-08-12 02:12:05 +03:00
struct btree * b ,
struct btree_node_iter * node_iter ,
struct bkey_packed * where ,
unsigned clobber_u64s ,
unsigned new_u64s )
2017-03-17 09:18:50 +03:00
{
2018-08-12 02:12:05 +03:00
struct bset_tree * t = bch2_bkey_to_bset_inlined ( b , where ) ;
2017-03-17 09:18:50 +03:00
struct btree_iter * linked ;
2019-09-19 23:07:41 +03:00
if ( node_iter ! = & iter - > l [ b - > c . level ] . iter ) {
2017-03-17 09:18:50 +03:00
__bch2_btree_node_iter_fix ( iter , b , node_iter , t ,
2019-09-19 23:07:41 +03:00
where , clobber_u64s , new_u64s ) ;
2020-02-19 00:17:55 +03:00
if ( debug_check_iterators ( iter - > trans - > c ) )
bch2_btree_node_iter_verify ( node_iter , b ) ;
2019-09-19 23:07:41 +03:00
}
2017-03-17 09:18:50 +03:00
2019-09-19 23:07:41 +03:00
trans_for_each_iter_with_node ( iter - > trans , b , linked ) {
2017-03-17 09:18:50 +03:00
__bch2_btree_node_iter_fix ( linked , b ,
2019-09-19 23:07:41 +03:00
& linked - > l [ b - > c . level ] . iter , t ,
where , clobber_u64s , new_u64s ) ;
2020-02-19 00:17:55 +03:00
bch2_btree_iter_verify_level ( linked , b - > c . level ) ;
2019-09-19 23:07:41 +03:00
}
2017-03-17 09:18:50 +03:00
}
static inline struct bkey_s_c __btree_iter_unpack ( struct btree_iter * iter ,
struct btree_iter_level * l ,
struct bkey * u ,
struct bkey_packed * k )
{
struct bkey_s_c ret ;
if ( unlikely ( ! k ) ) {
/*
* signal to bch2_btree_iter_peek_slot ( ) that we ' re currently at
* a hole
*/
2018-11-01 22:10:01 +03:00
u - > type = KEY_TYPE_deleted ;
2017-03-17 09:18:50 +03:00
return bkey_s_c_null ;
}
ret = bkey_disassemble ( l - > b , k , u ) ;
2019-03-26 00:06:42 +03:00
if ( debug_check_bkeys ( iter - > trans - > c ) )
bch2_bkey_debugcheck ( iter - > trans - > c , l - > b , ret ) ;
2017-03-17 09:18:50 +03:00
return ret ;
}
/* peek_all() doesn't skip deleted keys */
static inline struct bkey_s_c __btree_iter_peek_all ( struct btree_iter * iter ,
struct btree_iter_level * l ,
struct bkey * u )
{
return __btree_iter_unpack ( iter , l , u ,
bch2_btree_node_iter_peek_all ( & l - > iter , l - > b ) ) ;
}
static inline struct bkey_s_c __btree_iter_peek ( struct btree_iter * iter ,
struct btree_iter_level * l )
{
return __btree_iter_unpack ( iter , l , & iter - > k ,
bch2_btree_node_iter_peek ( & l - > iter , l - > b ) ) ;
}
2019-09-08 00:17:21 +03:00
static inline struct bkey_s_c __btree_iter_prev ( struct btree_iter * iter ,
struct btree_iter_level * l )
{
return __btree_iter_unpack ( iter , l , & iter - > k ,
bch2_btree_node_iter_prev ( & l - > iter , l - > b ) ) ;
}
2018-08-21 23:30:14 +03:00
static inline bool btree_iter_advance_to_pos ( struct btree_iter * iter ,
struct btree_iter_level * l ,
int max_advance )
2017-03-17 09:18:50 +03:00
{
2020-01-07 06:25:09 +03:00
struct bpos pos = btree_iter_search_key ( iter ) ;
2018-08-21 23:30:14 +03:00
struct bkey_packed * k ;
int nr_advanced = 0 ;
while ( ( k = bch2_btree_node_iter_peek_all ( & l - > iter , l - > b ) ) & &
2020-01-07 06:25:09 +03:00
bkey_iter_pos_cmp ( l - > b , k , & pos ) < 0 ) {
2018-08-21 23:30:14 +03:00
if ( max_advance > 0 & & nr_advanced > = max_advance )
return false ;
bch2_btree_node_iter_advance ( & l - > iter , l - > b ) ;
nr_advanced + + ;
}
return true ;
2017-03-17 09:18:50 +03:00
}
/*
* Verify that iterator for parent node points to child node :
*/
static void btree_iter_verify_new_node ( struct btree_iter * iter , struct btree * b )
{
struct btree_iter_level * l ;
unsigned plevel ;
bool parent_locked ;
struct bkey_packed * k ;
if ( ! IS_ENABLED ( CONFIG_BCACHEFS_DEBUG ) )
return ;
2020-06-06 19:28:01 +03:00
plevel = b - > c . level + 1 ;
2017-03-17 09:18:50 +03:00
if ( ! btree_iter_node ( iter , plevel ) )
return ;
parent_locked = btree_node_locked ( iter , plevel ) ;
if ( ! bch2_btree_node_relock ( iter , plevel ) )
return ;
l = & iter - > l [ plevel ] ;
k = bch2_btree_node_iter_peek_all ( & l - > iter , l - > b ) ;
if ( ! k | |
bkey_deleted ( k ) | |
bkey_cmp_left_packed ( l - > b , k , & b - > key . k . p ) ) {
char buf [ 100 ] ;
struct bkey uk = bkey_unpack_key ( b , k ) ;
2018-11-09 09:24:07 +03:00
bch2_bkey_to_text ( & PBUF ( buf ) , & uk ) ;
2017-03-17 09:18:50 +03:00
panic ( " parent iter doesn't point to new node: \n %s \n %llu:%llu \n " ,
buf , b - > key . k . p . inode , b - > key . k . p . offset ) ;
}
if ( ! parent_locked )
2020-06-06 19:28:01 +03:00
btree_node_unlock ( iter , b - > c . level + 1 ) ;
2017-03-17 09:18:50 +03:00
}
static inline void __btree_iter_init ( struct btree_iter * iter ,
2018-08-21 23:30:14 +03:00
unsigned level )
2017-03-17 09:18:50 +03:00
{
2020-01-07 06:25:09 +03:00
struct bpos pos = btree_iter_search_key ( iter ) ;
2018-08-21 23:30:14 +03:00
struct btree_iter_level * l = & iter - > l [ level ] ;
2020-01-07 06:25:09 +03:00
bch2_btree_node_iter_init ( & l - > iter , l - > b , & pos ) ;
2017-03-17 09:18:50 +03:00
btree_iter_set_dirty ( iter , BTREE_ITER_NEED_PEEK ) ;
}
static inline void btree_iter_node_set ( struct btree_iter * iter ,
struct btree * b )
{
2019-03-08 03:46:10 +03:00
BUG_ON ( btree_iter_type ( iter ) = = BTREE_ITER_CACHED ) ;
2017-03-17 09:18:50 +03:00
btree_iter_verify_new_node ( iter , b ) ;
EBUG_ON ( ! btree_iter_pos_in_node ( iter , b ) ) ;
2020-06-06 19:28:01 +03:00
EBUG_ON ( b - > c . lock . state . seq & 1 ) ;
2017-03-17 09:18:50 +03:00
2020-06-06 19:28:01 +03:00
iter - > l [ b - > c . level ] . lock_seq = b - > c . lock . state . seq ;
iter - > l [ b - > c . level ] . b = b ;
__btree_iter_init ( iter , b - > c . level ) ;
2017-03-17 09:18:50 +03:00
}
/*
* A btree node is being replaced - update the iterator to point to the new
* node :
*/
void bch2_btree_iter_node_replace ( struct btree_iter * iter , struct btree * b )
{
enum btree_node_locked_type t ;
struct btree_iter * linked ;
2019-03-28 05:03:30 +03:00
trans_for_each_iter ( iter - > trans , linked )
2019-03-08 03:46:10 +03:00
if ( btree_iter_type ( linked ) ! = BTREE_ITER_CACHED & &
btree_iter_pos_in_node ( linked , b ) ) {
2017-03-17 09:18:50 +03:00
/*
* bch2_btree_iter_node_drop ( ) has already been called -
* the old node we ' re replacing has already been
* unlocked and the pointer invalidated
*/
2020-06-06 19:28:01 +03:00
BUG_ON ( btree_node_locked ( linked , b - > c . level ) ) ;
2017-03-17 09:18:50 +03:00
2020-06-06 19:28:01 +03:00
t = btree_lock_want ( linked , b - > c . level ) ;
2017-03-17 09:18:50 +03:00
if ( t ! = BTREE_NODE_UNLOCKED ) {
2020-06-06 19:28:01 +03:00
six_lock_increment ( & b - > c . lock , ( enum six_lock_type ) t ) ;
mark_btree_node_locked ( linked , b - > c . level , ( enum six_lock_type ) t ) ;
2017-03-17 09:18:50 +03:00
}
btree_iter_node_set ( linked , b ) ;
}
}
void bch2_btree_iter_node_drop ( struct btree_iter * iter , struct btree * b )
{
struct btree_iter * linked ;
2020-06-06 19:28:01 +03:00
unsigned level = b - > c . level ;
2017-03-17 09:18:50 +03:00
2019-03-28 05:03:30 +03:00
trans_for_each_iter ( iter - > trans , linked )
2017-03-17 09:18:50 +03:00
if ( linked - > l [ level ] . b = = b ) {
2018-11-23 13:19:25 +03:00
__btree_node_unlock ( linked , level ) ;
2019-05-14 21:08:23 +03:00
linked - > l [ level ] . b = BTREE_ITER_NO_NODE_DROP ;
2017-03-17 09:18:50 +03:00
}
}
/*
* A btree node has been modified in such a way as to invalidate iterators - fix
* them :
*/
void bch2_btree_iter_reinit_node ( struct btree_iter * iter , struct btree * b )
{
struct btree_iter * linked ;
2019-03-28 05:03:30 +03:00
trans_for_each_iter_with_node ( iter - > trans , b , linked )
2020-06-06 19:28:01 +03:00
__btree_iter_init ( linked , b - > c . level ) ;
2017-03-17 09:18:50 +03:00
}
2020-06-13 05:29:48 +03:00
static int lock_root_check_fn ( struct six_lock * lock , void * p )
{
struct btree * b = container_of ( lock , struct btree , c . lock ) ;
struct btree * * rootp = p ;
return b = = * rootp ? 0 : - 1 ;
}
2017-03-17 09:18:50 +03:00
static inline int btree_iter_lock_root ( struct btree_iter * iter ,
unsigned depth_want )
{
2019-03-26 00:06:42 +03:00
struct bch_fs * c = iter - > trans - > c ;
2020-06-13 05:29:48 +03:00
struct btree * b , * * rootp = & c - > btree_roots [ iter - > btree_id ] . b ;
2017-03-17 09:18:50 +03:00
enum six_lock_type lock_type ;
unsigned i ;
EBUG_ON ( iter - > nodes_locked ) ;
while ( 1 ) {
2020-06-13 05:29:48 +03:00
b = READ_ONCE ( * rootp ) ;
2020-06-06 19:28:01 +03:00
iter - > level = READ_ONCE ( b - > c . level ) ;
2017-03-17 09:18:50 +03:00
if ( unlikely ( iter - > level < depth_want ) ) {
/*
* the root is at a lower depth than the depth we want :
* got to the end of the btree , or we ' re walking nodes
* greater than some depth and there are no nodes > =
* that depth
*/
iter - > level = depth_want ;
2019-05-14 21:08:23 +03:00
for ( i = iter - > level ; i < BTREE_MAX_DEPTH ; i + + )
iter - > l [ i ] . b = NULL ;
2018-11-21 10:59:07 +03:00
return 1 ;
2017-03-17 09:18:50 +03:00
}
lock_type = __btree_lock_want ( iter , iter - > level ) ;
if ( unlikely ( ! btree_node_lock ( b , POS_MAX , iter - > level ,
2020-06-13 05:29:48 +03:00
iter , lock_type ,
lock_root_check_fn , rootp ) ) )
2017-03-17 09:18:50 +03:00
return - EINTR ;
2020-06-13 05:29:48 +03:00
if ( likely ( b = = READ_ONCE ( * rootp ) & &
2020-06-06 19:28:01 +03:00
b - > c . level = = iter - > level & &
2017-03-17 09:18:50 +03:00
! race_fault ( ) ) ) {
for ( i = 0 ; i < iter - > level ; i + + )
2019-05-14 21:08:23 +03:00
iter - > l [ i ] . b = BTREE_ITER_NO_NODE_LOCK_ROOT ;
2017-03-17 09:18:50 +03:00
iter - > l [ iter - > level ] . b = b ;
2019-05-14 21:08:23 +03:00
for ( i = iter - > level + 1 ; i < BTREE_MAX_DEPTH ; i + + )
iter - > l [ i ] . b = NULL ;
2017-03-17 09:18:50 +03:00
mark_btree_node_locked ( iter , iter - > level , lock_type ) ;
btree_iter_node_set ( iter , b ) ;
return 0 ;
}
2020-06-06 19:28:01 +03:00
six_unlock_type ( & b - > c . lock , lock_type ) ;
2017-03-17 09:18:50 +03:00
}
}
noinline
static void btree_iter_prefetch ( struct btree_iter * iter )
{
2019-03-26 00:06:42 +03:00
struct bch_fs * c = iter - > trans - > c ;
2017-03-17 09:18:50 +03:00
struct btree_iter_level * l = & iter - > l [ iter - > level ] ;
struct btree_node_iter node_iter = l - > iter ;
struct bkey_packed * k ;
BKEY_PADDED ( k ) tmp ;
2019-03-26 00:06:42 +03:00
unsigned nr = test_bit ( BCH_FS_STARTED , & c - > flags )
2017-03-17 09:18:50 +03:00
? ( iter - > level > 1 ? 0 : 2 )
: ( iter - > level > 1 ? 1 : 16 ) ;
bool was_locked = btree_node_locked ( iter , iter - > level ) ;
while ( nr ) {
if ( ! bch2_btree_node_relock ( iter , iter - > level ) )
return ;
bch2_btree_node_iter_advance ( & node_iter , l - > b ) ;
k = bch2_btree_node_iter_peek ( & node_iter , l - > b ) ;
if ( ! k )
break ;
bch2_bkey_unpack ( l - > b , & tmp . k , k ) ;
2019-03-26 00:06:42 +03:00
bch2_btree_node_prefetch ( c , iter , & tmp . k , iter - > level - 1 ) ;
2017-03-17 09:18:50 +03:00
}
if ( ! was_locked )
btree_node_unlock ( iter , iter - > level ) ;
}
2020-02-24 23:25:00 +03:00
static noinline void btree_node_mem_ptr_set ( struct btree_iter * iter ,
unsigned plevel , struct btree * b )
{
struct btree_iter_level * l = & iter - > l [ plevel ] ;
bool locked = btree_node_locked ( iter , plevel ) ;
struct bkey_packed * k ;
struct bch_btree_ptr_v2 * bp ;
if ( ! bch2_btree_node_relock ( iter , plevel ) )
return ;
k = bch2_btree_node_iter_peek_all ( & l - > iter , l - > b ) ;
BUG_ON ( k - > type ! = KEY_TYPE_btree_ptr_v2 ) ;
bp = ( void * ) bkeyp_val ( & l - > b - > format , k ) ;
bp - > mem_ptr = ( unsigned long ) b ;
if ( ! locked )
btree_node_unlock ( iter , plevel ) ;
}
2019-10-24 02:50:01 +03:00
static __always_inline int btree_iter_down ( struct btree_iter * iter )
2017-03-17 09:18:50 +03:00
{
2019-03-26 00:06:42 +03:00
struct bch_fs * c = iter - > trans - > c ;
2017-03-17 09:18:50 +03:00
struct btree_iter_level * l = & iter - > l [ iter - > level ] ;
struct btree * b ;
unsigned level = iter - > level - 1 ;
enum six_lock_type lock_type = __btree_lock_want ( iter , level ) ;
BKEY_PADDED ( k ) tmp ;
2019-10-24 02:50:01 +03:00
EBUG_ON ( ! btree_node_locked ( iter , iter - > level ) ) ;
2017-03-17 09:18:50 +03:00
bch2_bkey_unpack ( l - > b , & tmp . k ,
bch2_btree_node_iter_peek ( & l - > iter , l - > b ) ) ;
2019-05-11 00:09:42 +03:00
b = bch2_btree_node_get ( c , iter , & tmp . k , level , lock_type ) ;
2017-03-17 09:18:50 +03:00
if ( unlikely ( IS_ERR ( b ) ) )
return PTR_ERR ( b ) ;
mark_btree_node_locked ( iter , level , lock_type ) ;
btree_iter_node_set ( iter , b ) ;
2020-02-24 23:25:00 +03:00
if ( tmp . k . k . type = = KEY_TYPE_btree_ptr_v2 & &
unlikely ( b ! = btree_node_mem_ptr ( & tmp . k ) ) )
btree_node_mem_ptr_set ( iter , level + 1 , b ) ;
2017-03-17 09:18:50 +03:00
if ( iter - > flags & BTREE_ITER_PREFETCH )
btree_iter_prefetch ( iter ) ;
iter - > level = level ;
return 0 ;
}
static void btree_iter_up ( struct btree_iter * iter )
{
btree_node_unlock ( iter , iter - > level + + ) ;
}
2019-09-08 21:00:12 +03:00
static int btree_iter_traverse_one ( struct btree_iter * ) ;
2017-03-17 09:18:50 +03:00
2019-03-08 03:46:10 +03:00
static int __btree_iter_traverse_all ( struct btree_trans * trans , int ret )
2017-03-17 09:18:50 +03:00
{
2019-03-28 05:54:42 +03:00
struct bch_fs * c = trans - > c ;
2019-05-24 21:33:16 +03:00
struct btree_iter * iter ;
2019-03-28 05:54:42 +03:00
u8 sorted [ BTREE_ITER_MAX ] ;
unsigned i , nr_sorted = 0 ;
2019-03-08 03:46:10 +03:00
if ( trans - > in_traverse_all )
return - EINTR ;
trans - > in_traverse_all = true ;
retry_all :
nr_sorted = 0 ;
2019-03-28 05:54:42 +03:00
trans_for_each_iter ( trans , iter )
2019-03-08 03:46:10 +03:00
sorted [ nr_sorted + + ] = iter - > idx ;
2019-03-28 05:54:42 +03:00
# define btree_iter_cmp_by_idx(_l, _r) \
btree_iter_cmp ( & trans - > iters [ _l ] , & trans - > iters [ _r ] )
bubble_sort ( sorted , nr_sorted , btree_iter_cmp_by_idx ) ;
# undef btree_iter_cmp_by_idx
2019-05-15 16:47:40 +03:00
bch2_trans_unlock ( trans ) ;
2017-03-17 09:18:50 +03:00
2019-03-28 07:07:24 +03:00
if ( unlikely ( ret = = - ENOMEM ) ) {
2017-03-17 09:18:50 +03:00
struct closure cl ;
closure_init_stack ( & cl ) ;
do {
ret = bch2_btree_cache_cannibalize_lock ( c , & cl ) ;
closure_sync ( & cl ) ;
} while ( ret ) ;
}
2019-03-28 07:07:24 +03:00
if ( unlikely ( ret = = - EIO ) ) {
2019-04-05 04:28:16 +03:00
trans - > error = true ;
2019-03-28 07:07:24 +03:00
goto out ;
}
BUG_ON ( ret & & ret ! = - EINTR ) ;
2017-03-17 09:18:50 +03:00
/* Now, redo traversals in correct order: */
2019-03-28 05:54:42 +03:00
for ( i = 0 ; i < nr_sorted ; i + + ) {
2019-03-08 03:46:10 +03:00
unsigned idx = sorted [ i ] ;
/*
* sucessfully traversing one iterator can cause another to be
* unlinked , in btree_key_cache_fill ( )
*/
if ( ! ( trans - > iters_linked & ( 1ULL < < idx ) ) )
continue ;
2017-03-17 09:18:50 +03:00
2019-03-08 03:46:10 +03:00
ret = btree_iter_traverse_one ( & trans - > iters [ idx ] ) ;
2019-03-28 05:54:42 +03:00
if ( ret )
goto retry_all ;
}
2017-03-17 09:18:50 +03:00
2020-03-18 20:40:28 +03:00
if ( hweight64 ( trans - > iters_live ) > 1 )
ret = - EINTR ;
else
trans_for_each_iter ( trans , iter )
if ( iter - > flags & BTREE_ITER_KEEP_UNTIL_COMMIT ) {
ret = - EINTR ;
break ;
}
2017-03-17 09:18:50 +03:00
out :
bch2_btree_cache_cannibalize_unlock ( c ) ;
2019-03-08 03:46:10 +03:00
trans - > in_traverse_all = false ;
2017-03-17 09:18:50 +03:00
return ret ;
2019-03-28 07:07:24 +03:00
}
2017-03-17 09:18:50 +03:00
2019-03-28 07:07:24 +03:00
int bch2_btree_iter_traverse_all ( struct btree_trans * trans )
{
2019-03-08 03:46:10 +03:00
return __btree_iter_traverse_all ( trans , 0 ) ;
2017-03-17 09:18:50 +03:00
}
2019-09-08 02:19:57 +03:00
static inline bool btree_iter_good_node ( struct btree_iter * iter ,
unsigned l , int check_pos )
{
if ( ! is_btree_node ( iter , l ) | |
! bch2_btree_node_relock ( iter , l ) )
return false ;
if ( check_pos < = 0 & & btree_iter_pos_before_node ( iter , iter - > l [ l ] . b ) )
return false ;
if ( check_pos > = 0 & & btree_iter_pos_after_node ( iter , iter - > l [ l ] . b ) )
return false ;
return true ;
}
static inline unsigned btree_iter_up_until_good_node ( struct btree_iter * iter ,
int check_pos )
2017-03-17 09:18:50 +03:00
{
unsigned l = iter - > level ;
while ( btree_iter_node ( iter , l ) & &
2019-09-08 02:19:57 +03:00
! btree_iter_good_node ( iter , l , check_pos ) ) {
2017-03-17 09:18:50 +03:00
btree_node_unlock ( iter , l ) ;
2019-05-14 21:08:23 +03:00
iter - > l [ l ] . b = BTREE_ITER_NO_NODE_UP ;
2017-03-17 09:18:50 +03:00
l + + ;
}
return l ;
}
/*
* This is the main state machine for walking down the btree - walks down to a
* specified depth
*
* Returns 0 on success , - EIO on error ( error reading in a btree node ) .
*
* On error , caller ( peek_node ( ) / peek_key ( ) ) must return NULL ; the error is
2019-05-10 23:09:17 +03:00
* stashed in the iterator and returned from bch2_trans_exit ( ) .
2017-03-17 09:18:50 +03:00
*/
2019-09-08 21:00:12 +03:00
static int btree_iter_traverse_one ( struct btree_iter * iter )
2017-03-17 09:18:50 +03:00
{
unsigned depth_want = iter - > level ;
2019-11-04 23:56:04 +03:00
/*
* if we need interior nodes locked , call btree_iter_relock ( ) to make
* sure we walk back up enough that we lock them :
*/
if ( iter - > uptodate = = BTREE_ITER_NEED_RELOCK | |
iter - > locks_want > 1 )
bch2_btree_iter_relock ( iter , false ) ;
2019-03-08 03:46:10 +03:00
if ( btree_iter_type ( iter ) = = BTREE_ITER_CACHED )
return bch2_btree_iter_traverse_cached ( iter ) ;
2019-11-04 23:56:04 +03:00
if ( iter - > uptodate < BTREE_ITER_NEED_RELOCK )
2017-03-17 09:18:50 +03:00
return 0 ;
2019-03-08 03:46:10 +03:00
if ( unlikely ( iter - > level > = BTREE_MAX_DEPTH ) )
return 0 ;
2017-03-17 09:18:50 +03:00
/*
* XXX : correctly using BTREE_ITER_UPTODATE should make using check_pos
* here unnecessary
*/
2019-09-08 02:19:57 +03:00
iter - > level = btree_iter_up_until_good_node ( iter , 0 ) ;
2017-03-17 09:18:50 +03:00
/*
* If we ' ve got a btree node locked ( i . e . we aren ' t about to relock the
* root ) - advance its node iterator if necessary :
*
* XXX correctly using BTREE_ITER_UPTODATE should make this unnecessary
*/
2019-09-08 02:19:57 +03:00
if ( btree_iter_node ( iter , iter - > level ) ) {
BUG_ON ( ! btree_iter_pos_in_node ( iter , iter - > l [ iter - > level ] . b ) ) ;
2018-08-21 23:30:14 +03:00
btree_iter_advance_to_pos ( iter , & iter - > l [ iter - > level ] , - 1 ) ;
2019-09-08 02:19:57 +03:00
}
2017-03-17 09:18:50 +03:00
/*
* Note : iter - > nodes [ iter - > level ] may be temporarily NULL here - that
* would indicate to other code that we got to the end of the btree ,
* here it indicates that relocking the root failed - it ' s critical that
* btree_iter_lock_root ( ) comes next and that it can ' t fail
*/
while ( iter - > level > depth_want ) {
int ret = btree_iter_node ( iter , iter - > level )
? btree_iter_down ( iter )
: btree_iter_lock_root ( iter , depth_want ) ;
if ( unlikely ( ret ) ) {
2018-11-21 10:59:07 +03:00
if ( ret = = 1 )
return 0 ;
2017-03-17 09:18:50 +03:00
iter - > level = depth_want ;
2019-03-08 03:46:10 +03:00
if ( ret = = - EIO ) {
iter - > flags | = BTREE_ITER_ERROR ;
iter - > l [ iter - > level ] . b =
BTREE_ITER_NO_NODE_ERROR ;
} else {
iter - > l [ iter - > level ] . b =
BTREE_ITER_NO_NODE_DOWN ;
}
2017-03-17 09:18:50 +03:00
return ret ;
}
}
iter - > uptodate = BTREE_ITER_NEED_PEEK ;
2016-07-22 06:05:06 +03:00
2020-02-19 00:17:55 +03:00
bch2_btree_iter_verify ( iter ) ;
2017-03-17 09:18:50 +03:00
return 0 ;
}
2019-09-08 21:00:12 +03:00
int __must_check __bch2_btree_iter_traverse ( struct btree_iter * iter )
2017-03-17 09:18:50 +03:00
{
2019-03-08 03:46:10 +03:00
struct btree_trans * trans = iter - > trans ;
2017-03-17 09:18:50 +03:00
int ret ;
2019-03-08 03:46:10 +03:00
ret = bch2_trans_cond_resched ( trans ) ? :
2019-09-08 21:00:12 +03:00
btree_iter_traverse_one ( iter ) ;
2017-03-17 09:18:50 +03:00
if ( unlikely ( ret ) )
2019-03-08 03:46:10 +03:00
ret = __btree_iter_traverse_all ( trans , ret ) ;
2017-03-17 09:18:50 +03:00
return ret ;
}
2020-06-16 02:53:46 +03:00
static inline void bch2_btree_iter_checks ( struct btree_iter * iter )
2017-03-17 09:18:50 +03:00
{
2020-06-16 02:53:46 +03:00
enum btree_iter_type type = btree_iter_type ( iter ) ;
2017-03-17 09:18:50 +03:00
EBUG_ON ( iter - > btree_id > = BTREE_ID_NR ) ;
2020-06-16 02:53:46 +03:00
BUG_ON ( ( type = = BTREE_ITER_KEYS | |
type = = BTREE_ITER_CACHED ) & &
2020-02-19 00:17:55 +03:00
( bkey_cmp ( iter - > pos , bkey_start_pos ( & iter - > k ) ) < 0 | |
bkey_cmp ( iter - > pos , iter - > k . p ) > 0 ) ) ;
2020-02-19 00:17:55 +03:00
bch2_btree_iter_verify_locks ( iter ) ;
bch2_btree_iter_verify_level ( iter , iter - > level ) ;
2017-03-17 09:18:50 +03:00
}
/* Iterate across nodes (leaf and interior nodes) */
struct btree * bch2_btree_iter_peek_node ( struct btree_iter * iter )
{
struct btree * b ;
int ret ;
2020-06-16 02:53:46 +03:00
EBUG_ON ( btree_iter_type ( iter ) ! = BTREE_ITER_NODES ) ;
bch2_btree_iter_checks ( iter ) ;
2017-03-17 09:18:50 +03:00
if ( iter - > uptodate = = BTREE_ITER_UPTODATE )
return iter - > l [ iter - > level ] . b ;
ret = bch2_btree_iter_traverse ( iter ) ;
if ( ret )
return NULL ;
b = btree_iter_node ( iter , iter - > level ) ;
if ( ! b )
return NULL ;
BUG_ON ( bkey_cmp ( b - > key . k . p , iter - > pos ) < 0 ) ;
iter - > pos = b - > key . k . p ;
iter - > uptodate = BTREE_ITER_UPTODATE ;
2020-02-19 00:17:55 +03:00
bch2_btree_iter_verify ( iter ) ;
2017-03-17 09:18:50 +03:00
return b ;
}
2020-02-19 00:17:55 +03:00
struct btree * bch2_btree_iter_next_node ( struct btree_iter * iter )
2017-03-17 09:18:50 +03:00
{
struct btree * b ;
int ret ;
2020-06-16 02:53:46 +03:00
EBUG_ON ( btree_iter_type ( iter ) ! = BTREE_ITER_NODES ) ;
bch2_btree_iter_checks ( iter ) ;
2017-03-17 09:18:50 +03:00
/* already got to end? */
if ( ! btree_iter_node ( iter , iter - > level ) )
return NULL ;
2019-04-05 04:53:12 +03:00
bch2_trans_cond_resched ( iter - > trans ) ;
2017-03-17 09:18:50 +03:00
btree_iter_up ( iter ) ;
if ( ! bch2_btree_node_relock ( iter , iter - > level ) )
btree_iter_set_dirty ( iter , BTREE_ITER_NEED_RELOCK ) ;
ret = bch2_btree_iter_traverse ( iter ) ;
if ( ret )
return NULL ;
/* got to end? */
b = btree_iter_node ( iter , iter - > level ) ;
if ( ! b )
return NULL ;
if ( bkey_cmp ( iter - > pos , b - > key . k . p ) < 0 ) {
/*
* Haven ' t gotten to the end of the parent node : go back down to
* the next child node
*/
/*
* We don ' t really want to be unlocking here except we can ' t
* directly tell btree_iter_traverse ( ) " traverse to this level "
* except by setting iter - > level , so we have to unlock so we
* don ' t screw up our lock invariants :
*/
if ( btree_node_read_locked ( iter , iter - > level ) )
btree_node_unlock ( iter , iter - > level ) ;
2020-01-07 21:29:32 +03:00
iter - > pos = bkey_successor ( iter - > pos ) ;
2020-02-19 00:17:55 +03:00
iter - > level = iter - > min_depth ;
2017-03-17 09:18:50 +03:00
btree_iter_set_dirty ( iter , BTREE_ITER_NEED_TRAVERSE ) ;
ret = bch2_btree_iter_traverse ( iter ) ;
if ( ret )
return NULL ;
b = iter - > l [ iter - > level ] . b ;
}
iter - > pos = b - > key . k . p ;
iter - > uptodate = BTREE_ITER_UPTODATE ;
2020-02-19 00:17:55 +03:00
bch2_btree_iter_verify ( iter ) ;
2017-03-17 09:18:50 +03:00
return b ;
}
/* Iterate across keys (in leaf nodes only) */
void bch2_btree_iter_set_pos_same_leaf ( struct btree_iter * iter , struct bpos new_pos )
{
struct btree_iter_level * l = & iter - > l [ 0 ] ;
EBUG_ON ( iter - > level ! = 0 ) ;
EBUG_ON ( bkey_cmp ( new_pos , iter - > pos ) < 0 ) ;
EBUG_ON ( ! btree_node_locked ( iter , 0 ) ) ;
EBUG_ON ( bkey_cmp ( new_pos , l - > b - > key . k . p ) > 0 ) ;
2020-02-19 00:17:55 +03:00
bkey_init ( & iter - > k ) ;
iter - > k . p = iter - > pos = new_pos ;
2017-03-17 09:18:50 +03:00
btree_iter_set_dirty ( iter , BTREE_ITER_NEED_PEEK ) ;
2018-08-21 23:30:14 +03:00
btree_iter_advance_to_pos ( iter , l , - 1 ) ;
2017-03-17 09:18:50 +03:00
2020-06-03 02:41:47 +03:00
/*
* XXX :
* keeping a node locked that ' s outside ( even just outside ) iter - > pos
* breaks __bch2_btree_node_lock ( ) . This seems to only affect
* bch2_btree_node_get_sibling so for now it ' s fixed there , but we
* should try to get rid of this corner case .
*
* ( this behaviour is currently needed for BTREE_INSERT_NOUNLOCK )
*/
2018-08-21 23:30:14 +03:00
if ( bch2_btree_node_iter_end ( & l - > iter ) & &
btree_iter_pos_after_node ( iter , l - > b ) )
2017-03-17 09:18:50 +03:00
btree_iter_set_dirty ( iter , BTREE_ITER_NEED_TRAVERSE ) ;
}
2020-02-19 00:17:55 +03:00
static void btree_iter_pos_changed ( struct btree_iter * iter , int cmp )
2017-03-17 09:18:50 +03:00
{
2020-02-19 00:17:55 +03:00
unsigned l = iter - > level ;
if ( ! cmp )
goto out ;
2019-03-08 03:46:10 +03:00
if ( unlikely ( btree_iter_type ( iter ) = = BTREE_ITER_CACHED ) ) {
btree_node_unlock ( iter , 0 ) ;
iter - > l [ 0 ] . b = BTREE_ITER_NO_NODE_UP ;
btree_iter_set_dirty ( iter , BTREE_ITER_NEED_TRAVERSE ) ;
return ;
}
2020-02-19 00:17:55 +03:00
l = btree_iter_up_until_good_node ( iter , cmp ) ;
2017-03-17 09:18:50 +03:00
2019-09-08 02:19:57 +03:00
if ( btree_iter_node ( iter , l ) ) {
2017-03-17 09:18:50 +03:00
/*
* We might have to skip over many keys , or just a few : try
* advancing the node iterator , and if we have to skip over too
* many keys just reinit it ( or if we ' re rewinding , since that
* is expensive ) .
*/
2018-08-21 23:30:14 +03:00
if ( cmp < 0 | |
2019-09-08 02:19:57 +03:00
! btree_iter_advance_to_pos ( iter , & iter - > l [ l ] , 8 ) )
__btree_iter_init ( iter , l ) ;
2017-03-17 09:18:50 +03:00
/* Don't leave it locked if we're not supposed to: */
2019-09-08 02:19:57 +03:00
if ( btree_lock_want ( iter , l ) = = BTREE_NODE_UNLOCKED )
btree_node_unlock ( iter , l ) ;
2017-03-17 09:18:50 +03:00
}
2020-02-19 00:17:55 +03:00
out :
if ( l ! = iter - > level )
btree_iter_set_dirty ( iter , BTREE_ITER_NEED_TRAVERSE ) ;
else
btree_iter_set_dirty ( iter , BTREE_ITER_NEED_PEEK ) ;
2019-09-08 02:19:57 +03:00
}
2020-01-31 21:23:18 +03:00
void __bch2_btree_iter_set_pos ( struct btree_iter * iter , struct bpos new_pos ,
bool strictly_greater )
{
struct bpos old = btree_iter_search_key ( iter ) ;
int cmp ;
iter - > flags & = ~ BTREE_ITER_IS_EXTENTS ;
iter - > flags | = strictly_greater ? BTREE_ITER_IS_EXTENTS : 0 ;
2020-02-19 00:17:55 +03:00
bkey_init ( & iter - > k ) ;
iter - > k . p = iter - > pos = new_pos ;
2020-01-31 21:23:18 +03:00
2020-02-19 00:17:55 +03:00
cmp = bkey_cmp ( btree_iter_search_key ( iter ) , old ) ;
2020-01-31 21:23:18 +03:00
2020-02-19 00:17:55 +03:00
btree_iter_pos_changed ( iter , cmp ) ;
2020-01-31 21:23:18 +03:00
}
2019-09-08 02:19:57 +03:00
void bch2_btree_iter_set_pos ( struct btree_iter * iter , struct bpos new_pos )
{
int cmp = bkey_cmp ( new_pos , iter - > pos ) ;
2020-02-19 00:17:55 +03:00
bkey_init ( & iter - > k ) ;
iter - > k . p = iter - > pos = new_pos ;
2019-09-08 02:19:57 +03:00
2020-02-19 00:17:55 +03:00
btree_iter_pos_changed ( iter , cmp ) ;
2017-03-17 09:18:50 +03:00
}
2019-09-08 02:19:57 +03:00
static inline bool btree_iter_set_pos_to_next_leaf ( struct btree_iter * iter )
{
struct btree_iter_level * l = & iter - > l [ 0 ] ;
2020-02-19 00:17:55 +03:00
bool ret ;
2019-09-08 02:19:57 +03:00
2020-02-19 00:17:55 +03:00
bkey_init ( & iter - > k ) ;
iter - > k . p = iter - > pos = l - > b - > key . k . p ;
2019-09-08 02:19:57 +03:00
2020-02-19 00:17:55 +03:00
ret = bkey_cmp ( iter - > pos , POS_MAX ) ! = 0 ;
2020-01-07 21:29:32 +03:00
if ( ret & & ! ( iter - > flags & BTREE_ITER_IS_EXTENTS ) )
iter - > k . p = iter - > pos = bkey_successor ( iter - > pos ) ;
2019-09-08 02:19:57 +03:00
btree_iter_pos_changed ( iter , 1 ) ;
2020-02-19 00:17:55 +03:00
return ret ;
2019-09-08 02:19:57 +03:00
}
static inline bool btree_iter_set_pos_to_prev_leaf ( struct btree_iter * iter )
{
struct btree_iter_level * l = & iter - > l [ 0 ] ;
2020-02-19 00:17:55 +03:00
bool ret ;
2019-09-08 02:19:57 +03:00
2020-02-19 00:17:55 +03:00
bkey_init ( & iter - > k ) ;
iter - > k . p = iter - > pos = l - > b - > data - > min_key ;
2019-09-08 02:19:57 +03:00
iter - > uptodate = BTREE_ITER_NEED_TRAVERSE ;
2020-02-19 00:17:55 +03:00
ret = bkey_cmp ( iter - > pos , POS_MIN ) ! = 0 ;
2020-01-07 21:29:32 +03:00
if ( ret ) {
iter - > k . p = iter - > pos = bkey_predecessor ( iter - > pos ) ;
if ( iter - > flags & BTREE_ITER_IS_EXTENTS )
iter - > k . p = iter - > pos = bkey_predecessor ( iter - > pos ) ;
}
2019-09-08 02:19:57 +03:00
btree_iter_pos_changed ( iter , - 1 ) ;
2020-02-19 00:17:55 +03:00
return ret ;
2019-09-08 02:19:57 +03:00
}
2020-03-02 21:38:19 +03:00
/**
* btree_iter_peek_uptodate - given an iterator that is uptodate , return the key
* it currently points to
*/
2017-03-17 09:18:50 +03:00
static inline struct bkey_s_c btree_iter_peek_uptodate ( struct btree_iter * iter )
{
struct btree_iter_level * l = & iter - > l [ 0 ] ;
struct bkey_s_c ret = { . k = & iter - > k } ;
if ( ! bkey_deleted ( & iter - > k ) ) {
2019-09-19 23:07:41 +03:00
struct bkey_packed * _k =
__bch2_btree_node_iter_peek_all ( & l - > iter , l - > b ) ;
ret . v = bkeyp_val ( & l - > b - > format , _k ) ;
if ( debug_check_iterators ( iter - > trans - > c ) ) {
struct bkey k = bkey_unpack_key ( l - > b , _k ) ;
2019-10-12 21:13:45 +03:00
2019-09-19 23:07:41 +03:00
BUG_ON ( memcmp ( & k , & iter - > k , sizeof ( k ) ) ) ;
}
if ( debug_check_bkeys ( iter - > trans - > c ) )
bch2_bkey_debugcheck ( iter - > trans - > c , l - > b , ret ) ;
2017-03-17 09:18:50 +03:00
}
return ret ;
}
2019-09-08 02:19:57 +03:00
/**
* bch2_btree_iter_peek : returns first key greater than or equal to iterator ' s
* current position
*/
2017-03-17 09:18:50 +03:00
struct bkey_s_c bch2_btree_iter_peek ( struct btree_iter * iter )
{
struct btree_iter_level * l = & iter - > l [ 0 ] ;
struct bkey_s_c k ;
int ret ;
2020-06-16 02:53:46 +03:00
EBUG_ON ( btree_iter_type ( iter ) ! = BTREE_ITER_KEYS ) ;
bch2_btree_iter_checks ( iter ) ;
2017-03-17 09:18:50 +03:00
2020-03-02 21:38:19 +03:00
if ( iter - > uptodate = = BTREE_ITER_UPTODATE & &
! bkey_deleted ( & iter - > k ) )
2017-03-17 09:18:50 +03:00
return btree_iter_peek_uptodate ( iter ) ;
while ( 1 ) {
2019-09-08 21:00:12 +03:00
ret = bch2_btree_iter_traverse ( iter ) ;
if ( unlikely ( ret ) )
return bkey_s_c_err ( ret ) ;
2017-03-17 09:18:50 +03:00
k = __btree_iter_peek ( iter , l ) ;
if ( likely ( k . k ) )
break ;
2019-09-08 02:19:57 +03:00
if ( ! btree_iter_set_pos_to_next_leaf ( iter ) )
2017-03-17 09:18:50 +03:00
return bkey_s_c_null ;
}
/*
* iter - > pos should always be equal to the key we just
* returned - except extents can straddle iter - > pos :
*/
if ( ! ( iter - > flags & BTREE_ITER_IS_EXTENTS ) | |
bkey_cmp ( bkey_start_pos ( k . k ) , iter - > pos ) > 0 )
iter - > pos = bkey_start_pos ( k . k ) ;
iter - > uptodate = BTREE_ITER_UPTODATE ;
2020-02-19 00:17:55 +03:00
bch2_btree_iter_verify_level ( iter , 0 ) ;
2017-03-17 09:18:50 +03:00
return k ;
}
2019-09-08 02:19:57 +03:00
/**
* bch2_btree_iter_next : returns first key greater than iterator ' s current
* position
*/
2017-03-17 09:18:50 +03:00
struct bkey_s_c bch2_btree_iter_next ( struct btree_iter * iter )
{
2020-02-19 00:17:55 +03:00
if ( unlikely ( ! bkey_cmp ( iter - > k . p , POS_MAX ) ) )
return bkey_s_c_null ;
2019-09-08 02:19:57 +03:00
2020-02-19 00:17:55 +03:00
bch2_btree_iter_set_pos ( iter ,
2020-01-07 21:29:32 +03:00
( iter - > flags & BTREE_ITER_IS_EXTENTS )
? iter - > k . p
: bkey_successor ( iter - > k . p ) ) ;
2019-05-15 16:53:27 +03:00
2020-02-19 00:17:55 +03:00
return bch2_btree_iter_peek ( iter ) ;
2017-03-17 09:18:50 +03:00
}
2020-03-06 02:44:59 +03:00
static struct bkey_s_c __btree_trans_updates_peek ( struct btree_iter * iter )
{
struct bpos pos = btree_iter_search_key ( iter ) ;
struct btree_trans * trans = iter - > trans ;
struct btree_insert_entry * i ;
2019-12-30 22:37:25 +03:00
trans_for_each_update2 ( trans , i )
2020-03-06 02:44:59 +03:00
if ( ( cmp_int ( iter - > btree_id , i - > iter - > btree_id ) ? :
bkey_cmp ( pos , i - > k - > k . p ) ) < = 0 )
break ;
2019-12-30 22:37:25 +03:00
return i < trans - > updates2 + trans - > nr_updates2 & &
2020-03-06 02:44:59 +03:00
iter - > btree_id = = i - > iter - > btree_id
? bkey_i_to_s_c ( i - > k )
: bkey_s_c_null ;
}
static struct bkey_s_c __bch2_btree_iter_peek_with_updates ( struct btree_iter * iter )
{
struct btree_iter_level * l = & iter - > l [ 0 ] ;
struct bkey_s_c k = __btree_iter_peek ( iter , l ) ;
struct bkey_s_c u = __btree_trans_updates_peek ( iter ) ;
if ( k . k & & ( ! u . k | | bkey_cmp ( k . k - > p , u . k - > p ) < 0 ) )
return k ;
if ( u . k & & bkey_cmp ( u . k - > p , l - > b - > key . k . p ) < = 0 ) {
iter - > k = * u . k ;
return u ;
}
return bkey_s_c_null ;
}
struct bkey_s_c bch2_btree_iter_peek_with_updates ( struct btree_iter * iter )
{
struct bkey_s_c k ;
int ret ;
2020-06-16 02:53:46 +03:00
EBUG_ON ( btree_iter_type ( iter ) ! = BTREE_ITER_KEYS ) ;
bch2_btree_iter_checks ( iter ) ;
2020-03-06 02:44:59 +03:00
while ( 1 ) {
ret = bch2_btree_iter_traverse ( iter ) ;
if ( unlikely ( ret ) )
return bkey_s_c_err ( ret ) ;
k = __bch2_btree_iter_peek_with_updates ( iter ) ;
if ( k . k & & bkey_deleted ( k . k ) ) {
bch2_btree_iter_set_pos ( iter ,
2020-01-07 21:29:32 +03:00
( iter - > flags & BTREE_ITER_IS_EXTENTS )
? iter - > k . p
: bkey_successor ( iter - > k . p ) ) ;
2020-03-06 02:44:59 +03:00
continue ;
}
if ( likely ( k . k ) )
break ;
if ( ! btree_iter_set_pos_to_next_leaf ( iter ) )
return bkey_s_c_null ;
}
/*
* iter - > pos should always be equal to the key we just
* returned - except extents can straddle iter - > pos :
*/
if ( ! ( iter - > flags & BTREE_ITER_IS_EXTENTS ) | |
bkey_cmp ( bkey_start_pos ( k . k ) , iter - > pos ) > 0 )
iter - > pos = bkey_start_pos ( k . k ) ;
iter - > uptodate = BTREE_ITER_UPTODATE ;
return k ;
}
struct bkey_s_c bch2_btree_iter_next_with_updates ( struct btree_iter * iter )
{
if ( unlikely ( ! bkey_cmp ( iter - > k . p , POS_MAX ) ) )
return bkey_s_c_null ;
bch2_btree_iter_set_pos ( iter ,
2020-01-07 21:29:32 +03:00
( iter - > flags & BTREE_ITER_IS_EXTENTS )
? iter - > k . p
: bkey_successor ( iter - > k . p ) ) ;
2020-03-06 02:44:59 +03:00
return bch2_btree_iter_peek_with_updates ( iter ) ;
}
2019-09-08 00:17:21 +03:00
/**
* bch2_btree_iter_peek_prev : returns first key less than or equal to
* iterator ' s current position
*/
struct bkey_s_c bch2_btree_iter_peek_prev ( struct btree_iter * iter )
2017-03-17 09:18:50 +03:00
{
2020-02-19 00:17:55 +03:00
struct bpos pos = iter - > pos ;
2017-03-17 09:18:50 +03:00
struct btree_iter_level * l = & iter - > l [ 0 ] ;
struct bkey_s_c k ;
int ret ;
2020-06-16 02:53:46 +03:00
EBUG_ON ( btree_iter_type ( iter ) ! = BTREE_ITER_KEYS ) ;
bch2_btree_iter_checks ( iter ) ;
2017-03-17 09:18:50 +03:00
2020-03-02 21:38:19 +03:00
if ( iter - > uptodate = = BTREE_ITER_UPTODATE & &
! bkey_deleted ( & iter - > k ) )
2019-09-08 00:17:21 +03:00
return btree_iter_peek_uptodate ( iter ) ;
2017-03-17 09:18:50 +03:00
while ( 1 ) {
ret = bch2_btree_iter_traverse ( iter ) ;
if ( unlikely ( ret ) )
return bkey_s_c_err ( ret ) ;
2019-09-08 00:17:21 +03:00
k = __btree_iter_peek ( iter , l ) ;
2020-02-19 00:17:55 +03:00
if ( ! k . k | | bkey_cmp ( bkey_start_pos ( k . k ) , pos ) > 0 )
2019-09-08 00:17:21 +03:00
k = __btree_iter_prev ( iter , l ) ;
if ( likely ( k . k ) )
2017-03-17 09:18:50 +03:00
break ;
2019-09-08 00:17:21 +03:00
if ( ! btree_iter_set_pos_to_prev_leaf ( iter ) )
return bkey_s_c_null ;
}
2017-03-17 09:18:50 +03:00
2020-02-19 00:17:55 +03:00
EBUG_ON ( bkey_cmp ( bkey_start_pos ( k . k ) , pos ) > 0 ) ;
2017-03-17 09:18:50 +03:00
iter - > pos = bkey_start_pos ( k . k ) ;
iter - > uptodate = BTREE_ITER_UPTODATE ;
return k ;
}
2019-09-08 00:17:21 +03:00
/**
* bch2_btree_iter_prev : returns first key less than iterator ' s current
* position
*/
struct bkey_s_c bch2_btree_iter_prev ( struct btree_iter * iter )
{
2020-02-19 00:17:55 +03:00
struct bpos pos = bkey_start_pos ( & iter - > k ) ;
2019-09-08 00:17:21 +03:00
2020-06-16 02:53:46 +03:00
EBUG_ON ( btree_iter_type ( iter ) ! = BTREE_ITER_KEYS ) ;
bch2_btree_iter_checks ( iter ) ;
2019-09-08 00:17:21 +03:00
2020-02-19 00:17:55 +03:00
if ( unlikely ( ! bkey_cmp ( pos , POS_MIN ) ) )
return bkey_s_c_null ;
2019-09-08 00:17:21 +03:00
2020-02-19 00:17:55 +03:00
bch2_btree_iter_set_pos ( iter , bkey_predecessor ( pos ) ) ;
2019-09-08 00:17:21 +03:00
2020-02-19 00:17:55 +03:00
return bch2_btree_iter_peek_prev ( iter ) ;
2019-09-08 00:17:21 +03:00
}
2017-03-17 09:18:50 +03:00
static inline struct bkey_s_c
2016-07-22 06:05:06 +03:00
__bch2_btree_iter_peek_slot_extents ( struct btree_iter * iter )
2017-03-17 09:18:50 +03:00
{
struct btree_iter_level * l = & iter - > l [ 0 ] ;
2016-07-22 06:05:06 +03:00
struct btree_node_iter node_iter ;
2017-03-17 09:18:50 +03:00
struct bkey_s_c k ;
struct bkey n ;
int ret ;
2020-03-14 04:41:22 +03:00
/* keys & holes can't span inode numbers: */
if ( iter - > pos . offset = = KEY_OFFSET_MAX ) {
if ( iter - > pos . inode = = KEY_INODE_MAX )
return bkey_s_c_null ;
bch2_btree_iter_set_pos ( iter , bkey_successor ( iter - > pos ) ) ;
ret = bch2_btree_iter_traverse ( iter ) ;
if ( unlikely ( ret ) )
return bkey_s_c_err ( ret ) ;
}
2017-03-17 09:18:50 +03:00
2016-07-22 06:05:06 +03:00
/*
* iterator is now at the correct position for inserting at iter - > pos ,
* but we need to keep iterating until we find the first non whiteout so
* we know how big a hole we have , if any :
*/
node_iter = l - > iter ;
2020-01-07 06:25:09 +03:00
k = __btree_iter_unpack ( iter , l , & iter - > k ,
bch2_btree_node_iter_peek ( & node_iter , l - > b ) ) ;
if ( k . k & & bkey_cmp ( bkey_start_pos ( k . k ) , iter - > pos ) < = 0 ) {
/*
2020-03-14 04:41:22 +03:00
* We ' re not setting iter - > uptodate because the node iterator
* doesn ' t necessarily point at the key we ' re returning :
2020-01-07 06:25:09 +03:00
*/
EBUG_ON ( bkey_cmp ( k . k - > p , iter - > pos ) < = 0 ) ;
2020-02-19 00:17:55 +03:00
bch2_btree_iter_verify_level ( iter , 0 ) ;
2020-01-07 06:25:09 +03:00
return k ;
}
2016-07-22 06:05:06 +03:00
2017-03-17 09:18:50 +03:00
/* hole */
2016-07-22 06:05:06 +03:00
if ( ! k . k )
k . k = & l - > b - > key . k ;
2017-03-17 09:18:50 +03:00
bkey_init ( & n ) ;
n . p = iter - > pos ;
2016-07-22 06:05:06 +03:00
bch2_key_resize ( & n ,
min_t ( u64 , KEY_SIZE_MAX ,
( k . k - > p . inode = = n . p . inode
? bkey_start_offset ( k . k )
: KEY_OFFSET_MAX ) -
n . p . offset ) ) ;
2018-11-09 09:24:07 +03:00
EBUG_ON ( ! n . size ) ;
2017-03-17 09:18:50 +03:00
2016-07-22 06:05:06 +03:00
iter - > k = n ;
iter - > uptodate = BTREE_ITER_UPTODATE ;
2019-08-17 22:17:09 +03:00
2020-02-19 00:17:55 +03:00
bch2_btree_iter_verify_level ( iter , 0 ) ;
2016-07-22 06:05:06 +03:00
return ( struct bkey_s_c ) { & iter - > k , NULL } ;
}
2017-03-17 09:18:50 +03:00
2020-03-14 04:41:22 +03:00
struct bkey_s_c bch2_btree_iter_peek_slot ( struct btree_iter * iter )
2016-07-22 06:05:06 +03:00
{
struct btree_iter_level * l = & iter - > l [ 0 ] ;
struct bkey_s_c k ;
2020-03-14 04:41:22 +03:00
int ret ;
2020-06-16 02:53:46 +03:00
EBUG_ON ( btree_iter_type ( iter ) ! = BTREE_ITER_KEYS ) ;
bch2_btree_iter_checks ( iter ) ;
2020-03-14 04:41:22 +03:00
if ( iter - > uptodate = = BTREE_ITER_UPTODATE )
return btree_iter_peek_uptodate ( iter ) ;
ret = bch2_btree_iter_traverse ( iter ) ;
if ( unlikely ( ret ) )
return bkey_s_c_err ( ret ) ;
2017-03-17 09:18:50 +03:00
2016-07-22 06:05:06 +03:00
if ( iter - > flags & BTREE_ITER_IS_EXTENTS )
return __bch2_btree_iter_peek_slot_extents ( iter ) ;
2017-03-17 09:18:50 +03:00
2020-03-02 21:38:19 +03:00
k = __btree_iter_peek_all ( iter , l , & iter - > k ) ;
2017-03-17 09:18:50 +03:00
2020-03-02 21:38:19 +03:00
EBUG_ON ( k . k & & bkey_deleted ( k . k ) & & bkey_cmp ( k . k - > p , iter - > pos ) = = 0 ) ;
2017-03-17 09:18:50 +03:00
2020-03-02 21:38:19 +03:00
if ( ! k . k | | bkey_cmp ( iter - > pos , k . k - > p ) ) {
2016-07-22 06:05:06 +03:00
/* hole */
bkey_init ( & iter - > k ) ;
iter - > k . p = iter - > pos ;
2019-08-17 22:17:09 +03:00
k = ( struct bkey_s_c ) { & iter - > k , NULL } ;
2016-07-22 06:05:06 +03:00
}
2019-08-17 22:17:09 +03:00
iter - > uptodate = BTREE_ITER_UPTODATE ;
2020-02-19 00:17:55 +03:00
bch2_btree_iter_verify_level ( iter , 0 ) ;
2019-08-17 22:17:09 +03:00
return k ;
2017-03-17 09:18:50 +03:00
}
struct bkey_s_c bch2_btree_iter_next_slot ( struct btree_iter * iter )
{
2020-02-19 00:17:55 +03:00
if ( unlikely ( ! bkey_cmp ( iter - > k . p , POS_MAX ) ) )
return bkey_s_c_null ;
2017-03-17 09:18:50 +03:00
2020-02-19 00:17:55 +03:00
bch2_btree_iter_set_pos ( iter ,
2020-01-07 21:29:32 +03:00
( iter - > flags & BTREE_ITER_IS_EXTENTS )
? iter - > k . p
: bkey_successor ( iter - > k . p ) ) ;
2017-03-17 09:18:50 +03:00
2020-02-19 00:17:55 +03:00
return bch2_btree_iter_peek_slot ( iter ) ;
2017-03-17 09:18:50 +03:00
}
2019-03-08 03:46:10 +03:00
struct bkey_s_c bch2_btree_iter_peek_cached ( struct btree_iter * iter )
{
struct bkey_cached * ck ;
int ret ;
2020-06-16 02:53:46 +03:00
EBUG_ON ( btree_iter_type ( iter ) ! = BTREE_ITER_CACHED ) ;
bch2_btree_iter_checks ( iter ) ;
2019-03-08 03:46:10 +03:00
ret = bch2_btree_iter_traverse ( iter ) ;
if ( unlikely ( ret ) )
return bkey_s_c_err ( ret ) ;
ck = ( void * ) iter - > l [ 0 ] . b ;
EBUG_ON ( iter - > btree_id ! = ck - > key . btree_id | |
bkey_cmp ( iter - > pos , ck - > key . pos ) ) ;
BUG_ON ( ! ck - > valid ) ;
return bkey_i_to_s_c ( ck - > k ) ;
}
2019-03-26 00:06:42 +03:00
static inline void bch2_btree_iter_init ( struct btree_trans * trans ,
struct btree_iter * iter , enum btree_id btree_id ,
2019-03-25 22:10:15 +03:00
struct bpos pos , unsigned flags )
2017-03-17 09:18:50 +03:00
{
2019-03-26 00:06:42 +03:00
struct bch_fs * c = trans - > c ;
2017-03-17 09:18:50 +03:00
unsigned i ;
2019-05-12 00:29:20 +03:00
if ( btree_node_type_is_extents ( btree_id ) & &
2019-03-25 22:10:15 +03:00
! ( flags & BTREE_ITER_NODES ) )
flags | = BTREE_ITER_IS_EXTENTS ;
2017-03-17 09:18:50 +03:00
2019-03-26 00:06:42 +03:00
iter - > trans = trans ;
2017-03-17 09:18:50 +03:00
iter - > pos = pos ;
bkey_init ( & iter - > k ) ;
iter - > k . p = pos ;
iter - > flags = flags ;
iter - > uptodate = BTREE_ITER_NEED_TRAVERSE ;
iter - > btree_id = btree_id ;
2019-03-25 22:10:15 +03:00
iter - > level = 0 ;
2020-02-19 00:17:55 +03:00
iter - > min_depth = 0 ;
2019-03-25 22:10:15 +03:00
iter - > locks_want = flags & BTREE_ITER_INTENT ? 1 : 0 ;
2017-03-17 09:18:50 +03:00
iter - > nodes_locked = 0 ;
iter - > nodes_intent_locked = 0 ;
for ( i = 0 ; i < ARRAY_SIZE ( iter - > l ) ; i + + )
2020-02-19 03:29:33 +03:00
iter - > l [ i ] . b = BTREE_ITER_NO_NODE_INIT ;
2017-03-17 09:18:50 +03:00
prefetch ( c - > btree_roots [ btree_id ] . b ) ;
}
/* new transactional stuff: */
2019-03-25 22:34:48 +03:00
static inline void __bch2_trans_iter_free ( struct btree_trans * trans ,
unsigned idx )
{
2019-03-28 05:46:52 +03:00
__bch2_btree_iter_unlock ( & trans - > iters [ idx ] ) ;
2019-03-25 22:34:48 +03:00
trans - > iters_linked & = ~ ( 1ULL < < idx ) ;
trans - > iters_live & = ~ ( 1ULL < < idx ) ;
trans - > iters_touched & = ~ ( 1ULL < < idx ) ;
}
2019-09-27 05:21:39 +03:00
int bch2_trans_iter_put ( struct btree_trans * trans ,
struct btree_iter * iter )
2017-03-17 09:18:50 +03:00
{
2020-02-18 22:27:10 +03:00
int ret ;
if ( IS_ERR_OR_NULL ( iter ) )
return 0 ;
2020-02-26 23:39:46 +03:00
BUG_ON ( trans - > iters + iter - > idx ! = iter ) ;
2020-02-18 22:27:10 +03:00
ret = btree_iter_err ( iter ) ;
2019-03-25 22:10:15 +03:00
2019-09-27 05:21:39 +03:00
if ( ! ( trans - > iters_touched & ( 1ULL < < iter - > idx ) ) & &
! ( iter - > flags & BTREE_ITER_KEEP_UNTIL_COMMIT ) )
__bch2_trans_iter_free ( trans , iter - > idx ) ;
trans - > iters_live & = ~ ( 1ULL < < iter - > idx ) ;
2019-03-25 22:10:15 +03:00
return ret ;
2019-03-25 22:34:48 +03:00
}
2017-03-17 09:18:50 +03:00
2019-09-27 05:21:39 +03:00
int bch2_trans_iter_free ( struct btree_trans * trans ,
struct btree_iter * iter )
2019-03-25 22:34:48 +03:00
{
2020-02-18 22:27:10 +03:00
if ( IS_ERR_OR_NULL ( iter ) )
return 0 ;
2019-09-27 05:21:39 +03:00
trans - > iters_touched & = ~ ( 1ULL < < iter - > idx ) ;
2019-03-25 22:10:15 +03:00
2019-09-27 05:21:39 +03:00
return bch2_trans_iter_put ( trans , iter ) ;
2017-03-17 09:18:50 +03:00
}
2019-05-15 17:54:43 +03:00
static int bch2_trans_realloc_iters ( struct btree_trans * trans ,
2019-05-15 17:08:55 +03:00
unsigned new_size )
2017-03-17 09:18:50 +03:00
{
2019-12-30 22:37:25 +03:00
void * p , * new_iters , * new_updates , * new_updates2 ;
2019-09-07 21:16:00 +03:00
size_t iters_bytes ;
size_t updates_bytes ;
2017-03-17 09:18:50 +03:00
2019-05-15 17:08:55 +03:00
new_size = roundup_pow_of_two ( new_size ) ;
2019-03-08 07:13:39 +03:00
BUG_ON ( new_size > BTREE_ITER_MAX ) ;
if ( new_size < = trans - > size )
return 0 ;
BUG_ON ( trans - > used_mempool ) ;
2017-03-17 09:18:50 +03:00
bch2_trans_unlock ( trans ) ;
2019-09-07 21:16:00 +03:00
iters_bytes = sizeof ( struct btree_iter ) * new_size ;
2019-12-30 21:08:26 +03:00
updates_bytes = sizeof ( struct btree_insert_entry ) * new_size ;
2019-09-07 21:16:00 +03:00
2019-12-30 22:37:25 +03:00
p = kmalloc ( iters_bytes +
updates_bytes +
updates_bytes , GFP_NOFS ) ;
if ( p )
2019-03-08 07:13:39 +03:00
goto success ;
2019-12-30 22:37:25 +03:00
p = mempool_alloc ( & trans - > c - > btree_iters_pool , GFP_NOFS ) ;
2019-03-08 07:13:39 +03:00
new_size = BTREE_ITER_MAX ;
trans - > used_mempool = true ;
success :
2019-12-30 22:37:25 +03:00
new_iters = p ; p + = iters_bytes ;
new_updates = p ; p + = updates_bytes ;
new_updates2 = p ; p + = updates_bytes ;
2017-03-17 09:18:50 +03:00
memcpy ( new_iters , trans - > iters ,
sizeof ( struct btree_iter ) * trans - > nr_iters ) ;
2019-03-08 07:13:39 +03:00
memcpy ( new_updates , trans - > updates ,
sizeof ( struct btree_insert_entry ) * trans - > nr_updates ) ;
2019-12-30 22:37:25 +03:00
memcpy ( new_updates2 , trans - > updates2 ,
sizeof ( struct btree_insert_entry ) * trans - > nr_updates2 ) ;
2019-03-08 07:13:39 +03:00
2019-03-25 22:34:48 +03:00
if ( IS_ENABLED ( CONFIG_BCACHEFS_DEBUG ) )
memset ( trans - > iters , POISON_FREE ,
sizeof ( struct btree_iter ) * trans - > nr_iters +
sizeof ( struct btree_insert_entry ) * trans - > nr_iters ) ;
2019-03-08 07:13:39 +03:00
if ( trans - > iters ! = trans - > iters_onstack )
kfree ( trans - > iters ) ;
2019-09-07 21:16:00 +03:00
trans - > iters = new_iters ;
trans - > updates = new_updates ;
2019-12-30 22:37:25 +03:00
trans - > updates2 = new_updates2 ;
2019-09-07 21:16:00 +03:00
trans - > size = new_size ;
2017-03-17 09:18:50 +03:00
2018-07-13 06:30:45 +03:00
if ( trans - > iters_live ) {
2019-05-15 17:54:43 +03:00
trace_trans_restart_iters_realloced ( trans - > ip , trans - > size ) ;
2018-07-13 06:30:45 +03:00
return - EINTR ;
}
return 0 ;
2017-03-17 09:18:50 +03:00
}
2019-09-23 02:35:12 +03:00
static struct btree_iter * btree_trans_iter_alloc ( struct btree_trans * trans )
2019-03-26 05:43:26 +03:00
{
2019-04-01 01:40:01 +03:00
unsigned idx = __ffs64 ( ~ trans - > iters_linked ) ;
2019-03-26 05:43:26 +03:00
if ( idx < trans - > nr_iters )
goto got_slot ;
if ( trans - > nr_iters = = trans - > size ) {
2019-09-27 05:21:39 +03:00
int ret ;
if ( trans - > nr_iters > = BTREE_ITER_MAX ) {
struct btree_iter * iter ;
trans_for_each_iter ( trans , iter ) {
2020-06-02 23:36:11 +03:00
pr_err ( " iter: btree %s pos %llu:%llu%s%s%s %ps " ,
2019-09-27 05:21:39 +03:00
bch2_btree_ids [ iter - > btree_id ] ,
iter - > pos . inode ,
iter - > pos . offset ,
( trans - > iters_live & ( 1ULL < < iter - > idx ) ) ? " live " : " " ,
( trans - > iters_touched & ( 1ULL < < iter - > idx ) ) ? " touched " : " " ,
2020-04-02 00:14:14 +03:00
iter - > flags & BTREE_ITER_KEEP_UNTIL_COMMIT ? " keep " : " " ,
( void * ) iter - > ip_allocated ) ;
2019-09-27 05:21:39 +03:00
}
panic ( " trans iter oveflow \n " ) ;
}
ret = bch2_trans_realloc_iters ( trans , trans - > size * 2 ) ;
2019-03-26 05:43:26 +03:00
if ( ret )
2019-09-23 02:35:12 +03:00
return ERR_PTR ( ret ) ;
2019-03-26 05:43:26 +03:00
}
idx = trans - > nr_iters + + ;
BUG_ON ( trans - > nr_iters > trans - > size ) ;
2019-03-28 06:14:38 +03:00
trans - > iters [ idx ] . idx = idx ;
2019-03-26 05:43:26 +03:00
got_slot :
BUG_ON ( trans - > iters_linked & ( 1ULL < < idx ) ) ;
trans - > iters_linked | = 1ULL < < idx ;
2020-01-01 03:37:10 +03:00
trans - > iters [ idx ] . flags = 0 ;
2019-09-23 02:35:12 +03:00
return & trans - > iters [ idx ] ;
2019-03-26 05:43:26 +03:00
}
2019-09-27 05:21:39 +03:00
static inline void btree_iter_copy ( struct btree_iter * dst ,
struct btree_iter * src )
{
unsigned i , idx = dst - > idx ;
* dst = * src ;
dst - > idx = idx ;
2019-03-08 03:46:10 +03:00
dst - > flags & = ~ BTREE_ITER_KEEP_UNTIL_COMMIT ;
2019-09-27 05:21:39 +03:00
for ( i = 0 ; i < BTREE_MAX_DEPTH ; i + + )
if ( btree_node_locked ( dst , i ) )
six_lock_increment ( & dst - > l [ i ] . b - > c . lock ,
__btree_lock_want ( dst , i ) ) ;
2020-01-01 03:37:10 +03:00
dst - > flags & = ~ BTREE_ITER_KEEP_UNTIL_COMMIT ;
dst - > flags & = ~ BTREE_ITER_SET_POS_AFTER_COMMIT ;
2019-09-27 05:21:39 +03:00
}
static inline struct bpos bpos_diff ( struct bpos l , struct bpos r )
{
if ( bkey_cmp ( l , r ) > 0 )
swap ( l , r ) ;
return POS ( r . inode - l . inode , r . offset - l . offset ) ;
}
2017-03-17 09:18:50 +03:00
static struct btree_iter * __btree_trans_get_iter ( struct btree_trans * trans ,
2019-03-25 22:34:48 +03:00
unsigned btree_id , struct bpos pos ,
2019-09-27 05:21:39 +03:00
unsigned flags )
2017-03-17 09:18:50 +03:00
{
2019-09-27 05:21:39 +03:00
struct btree_iter * iter , * best = NULL ;
2017-03-17 09:18:50 +03:00
BUG_ON ( trans - > nr_iters > BTREE_ITER_MAX ) ;
2019-09-27 05:21:39 +03:00
trans_for_each_iter ( trans , iter ) {
if ( btree_iter_type ( iter ) ! = ( flags & BTREE_ITER_TYPE ) )
continue ;
2019-09-23 02:35:12 +03:00
2019-09-27 05:21:39 +03:00
if ( iter - > btree_id ! = btree_id )
continue ;
if ( best & &
bkey_cmp ( bpos_diff ( best - > pos , pos ) ,
bpos_diff ( iter - > pos , pos ) ) < 0 )
continue ;
best = iter ;
}
if ( ! best ) {
2019-09-23 02:35:12 +03:00
iter = btree_trans_iter_alloc ( trans ) ;
if ( IS_ERR ( iter ) )
return iter ;
2017-03-17 09:18:50 +03:00
2019-03-26 00:06:42 +03:00
bch2_btree_iter_init ( trans , iter , btree_id , pos , flags ) ;
2019-09-27 05:21:39 +03:00
} else if ( ( trans - > iters_live & ( 1ULL < < best - > idx ) ) | |
( best - > flags & BTREE_ITER_KEEP_UNTIL_COMMIT ) ) {
iter = btree_trans_iter_alloc ( trans ) ;
if ( IS_ERR ( iter ) )
return iter ;
2019-08-12 21:35:34 +03:00
2019-09-27 05:21:39 +03:00
btree_iter_copy ( iter , best ) ;
} else {
iter = best ;
2017-03-17 09:18:50 +03:00
}
2019-03-08 03:46:10 +03:00
iter - > flags & = ~ BTREE_ITER_KEEP_UNTIL_COMMIT ;
iter - > flags & = ~ BTREE_ITER_USER_FLAGS ;
iter - > flags | = flags & BTREE_ITER_USER_FLAGS ;
2019-09-27 05:21:39 +03:00
if ( iter - > flags & BTREE_ITER_INTENT )
bch2_btree_iter_upgrade ( iter , 1 ) ;
else
bch2_btree_iter_downgrade ( iter ) ;
2019-03-25 22:34:48 +03:00
BUG_ON ( iter - > btree_id ! = btree_id ) ;
2019-09-27 05:21:39 +03:00
BUG_ON ( ( iter - > flags ^ flags ) & BTREE_ITER_TYPE ) ;
BUG_ON ( iter - > flags & BTREE_ITER_KEEP_UNTIL_COMMIT ) ;
2020-01-01 03:37:10 +03:00
BUG_ON ( iter - > flags & BTREE_ITER_SET_POS_AFTER_COMMIT ) ;
2019-09-23 02:35:12 +03:00
BUG_ON ( trans - > iters_live & ( 1ULL < < iter - > idx ) ) ;
2019-09-27 05:21:39 +03:00
2019-09-23 02:35:12 +03:00
trans - > iters_live | = 1ULL < < iter - > idx ;
trans - > iters_touched | = 1ULL < < iter - > idx ;
2017-03-17 09:18:50 +03:00
return iter ;
}
2020-04-02 00:14:14 +03:00
struct btree_iter * __bch2_trans_get_iter ( struct btree_trans * trans ,
enum btree_id btree_id ,
struct bpos pos , unsigned flags )
2017-03-17 09:18:50 +03:00
{
struct btree_iter * iter =
2019-09-27 05:21:39 +03:00
__btree_trans_get_iter ( trans , btree_id , pos , flags ) ;
2017-03-17 09:18:50 +03:00
if ( ! IS_ERR ( iter ) )
2020-01-31 21:23:18 +03:00
__bch2_btree_iter_set_pos ( iter , pos ,
btree_node_type_is_extents ( btree_id ) ) ;
2017-03-17 09:18:50 +03:00
return iter ;
}
2019-03-25 22:10:15 +03:00
struct btree_iter * bch2_trans_get_node_iter ( struct btree_trans * trans ,
enum btree_id btree_id ,
struct bpos pos ,
unsigned locks_want ,
unsigned depth ,
unsigned flags )
{
struct btree_iter * iter =
__btree_trans_get_iter ( trans , btree_id , pos ,
2019-09-27 05:21:39 +03:00
flags | BTREE_ITER_NODES ) ;
2019-03-25 22:10:15 +03:00
unsigned i ;
BUG_ON ( IS_ERR ( iter ) ) ;
BUG_ON ( bkey_cmp ( iter - > pos , pos ) ) ;
iter - > locks_want = locks_want ;
iter - > level = depth ;
2020-02-19 00:17:55 +03:00
iter - > min_depth = depth ;
2019-03-25 22:10:15 +03:00
for ( i = 0 ; i < ARRAY_SIZE ( iter - > l ) ; i + + )
iter - > l [ i ] . b = NULL ;
2019-05-14 21:08:23 +03:00
iter - > l [ iter - > level ] . b = BTREE_ITER_NO_NODE_INIT ;
2019-03-25 22:10:15 +03:00
return iter ;
}
2020-04-02 00:14:14 +03:00
struct btree_iter * __bch2_trans_copy_iter ( struct btree_trans * trans ,
2019-03-26 05:43:26 +03:00
struct btree_iter * src )
2017-03-17 09:18:50 +03:00
{
2019-03-28 05:46:52 +03:00
struct btree_iter * iter ;
2017-03-17 09:18:50 +03:00
2019-09-23 02:35:12 +03:00
iter = btree_trans_iter_alloc ( trans ) ;
if ( IS_ERR ( iter ) )
return iter ;
2019-09-27 05:21:39 +03:00
btree_iter_copy ( iter , src ) ;
2019-03-26 05:43:26 +03:00
2019-09-27 05:21:39 +03:00
trans - > iters_live | = 1ULL < < iter - > idx ;
/*
2020-02-18 22:27:10 +03:00
* We don ' t need to preserve this iter since it ' s cheap to copy it
* again - this will cause trans_iter_put ( ) to free it right away :
2019-09-27 05:21:39 +03:00
*/
trans - > iters_touched & = ~ ( 1ULL < < iter - > idx ) ;
2019-03-26 05:43:26 +03:00
2019-09-23 02:35:12 +03:00
return iter ;
2017-03-17 09:18:50 +03:00
}
2019-05-15 17:54:43 +03:00
static int bch2_trans_preload_mem ( struct btree_trans * trans , size_t size )
2017-03-17 09:18:50 +03:00
{
2019-05-15 17:54:43 +03:00
if ( size > trans - > mem_bytes ) {
2017-03-17 09:18:50 +03:00
size_t old_bytes = trans - > mem_bytes ;
2019-05-15 17:54:43 +03:00
size_t new_bytes = roundup_pow_of_two ( size ) ;
2017-03-17 09:18:50 +03:00
void * new_mem = krealloc ( trans - > mem , new_bytes , GFP_NOFS ) ;
if ( ! new_mem )
2019-05-15 17:54:43 +03:00
return - ENOMEM ;
2017-03-17 09:18:50 +03:00
trans - > mem = new_mem ;
trans - > mem_bytes = new_bytes ;
2018-07-13 06:30:45 +03:00
if ( old_bytes ) {
2019-05-15 17:54:43 +03:00
trace_trans_restart_mem_realloced ( trans - > ip , new_bytes ) ;
return - EINTR ;
2018-07-13 06:30:45 +03:00
}
2017-03-17 09:18:50 +03:00
}
2019-05-15 17:54:43 +03:00
return 0 ;
}
void * bch2_trans_kmalloc ( struct btree_trans * trans , size_t size )
{
void * p ;
int ret ;
ret = bch2_trans_preload_mem ( trans , trans - > mem_top + size ) ;
if ( ret )
return ERR_PTR ( ret ) ;
p = trans - > mem + trans - > mem_top ;
2017-03-17 09:18:50 +03:00
trans - > mem_top + = size ;
2019-05-15 17:54:43 +03:00
return p ;
2017-03-17 09:18:50 +03:00
}
2019-09-27 05:21:39 +03:00
inline void bch2_trans_unlink_iters ( struct btree_trans * trans )
2019-03-25 22:34:48 +03:00
{
2019-09-27 05:21:39 +03:00
u64 iters = trans - > iters_linked &
~ trans - > iters_touched &
~ trans - > iters_live ;
2019-03-25 22:34:48 +03:00
while ( iters ) {
unsigned idx = __ffs64 ( iters ) ;
iters & = ~ ( 1ULL < < idx ) ;
__bch2_trans_iter_free ( trans , idx ) ;
}
}
2019-09-27 05:21:39 +03:00
void bch2_trans_reset ( struct btree_trans * trans , unsigned flags )
2017-03-17 09:18:50 +03:00
{
2019-09-27 05:21:39 +03:00
struct btree_iter * iter ;
2017-03-17 09:18:50 +03:00
2019-09-27 05:21:39 +03:00
trans_for_each_iter ( trans , iter )
2020-01-01 03:37:10 +03:00
iter - > flags & = ~ ( BTREE_ITER_KEEP_UNTIL_COMMIT |
BTREE_ITER_SET_POS_AFTER_COMMIT ) ;
2019-03-08 07:13:39 +03:00
2019-09-27 05:21:39 +03:00
bch2_trans_unlink_iters ( trans ) ;
2019-03-08 07:13:39 +03:00
2019-09-27 05:21:39 +03:00
trans - > iters_touched & = trans - > iters_live ;
2017-03-17 09:18:50 +03:00
2020-01-01 03:37:10 +03:00
trans - > need_reset = 0 ;
2019-03-25 22:34:48 +03:00
trans - > nr_updates = 0 ;
2019-12-30 22:37:25 +03:00
trans - > nr_updates2 = 0 ;
2020-02-26 23:39:46 +03:00
trans - > mem_top = 0 ;
2019-03-28 07:07:24 +03:00
2020-05-26 02:29:48 +03:00
trans - > extra_journal_entries = NULL ;
trans - > extra_journal_entry_u64s = 0 ;
2019-12-30 20:43:19 +03:00
if ( trans - > fs_usage_deltas ) {
trans - > fs_usage_deltas - > used = 0 ;
memset ( ( void * ) trans - > fs_usage_deltas +
offsetof ( struct replicas_delta_list , memset_start ) , 0 ,
( void * ) & trans - > fs_usage_deltas - > memset_end -
( void * ) & trans - > fs_usage_deltas - > memset_start ) ;
}
if ( ! ( flags & TRANS_RESET_NOTRAVERSE ) )
bch2_btree_iter_traverse_all ( trans ) ;
2017-03-17 09:18:50 +03:00
}
2019-05-15 17:54:43 +03:00
void bch2_trans_init ( struct btree_trans * trans , struct bch_fs * c ,
unsigned expected_nr_iters ,
size_t expected_mem_bytes )
2017-03-17 09:18:50 +03:00
{
2019-03-08 07:13:39 +03:00
memset ( trans , 0 , offsetof ( struct btree_trans , iters_onstack ) ) ;
2020-02-26 23:39:46 +03:00
/*
* reallocating iterators currently completely breaks
* bch2_trans_iter_put ( ) :
*/
expected_nr_iters = BTREE_ITER_MAX ;
2017-03-17 09:18:50 +03:00
trans - > c = c ;
2019-04-23 07:10:08 +03:00
trans - > ip = _RET_IP_ ;
2019-03-08 07:13:39 +03:00
trans - > size = ARRAY_SIZE ( trans - > iters_onstack ) ;
2017-03-17 09:18:50 +03:00
trans - > iters = trans - > iters_onstack ;
2019-03-08 07:13:39 +03:00
trans - > updates = trans - > updates_onstack ;
2019-12-30 22:37:25 +03:00
trans - > updates2 = trans - > updates2_onstack ;
2019-05-15 22:47:43 +03:00
trans - > fs_usage_deltas = NULL ;
2019-05-15 17:54:43 +03:00
if ( expected_nr_iters > trans - > size )
bch2_trans_realloc_iters ( trans , expected_nr_iters ) ;
if ( expected_mem_bytes )
bch2_trans_preload_mem ( trans , expected_mem_bytes ) ;
2020-06-02 23:36:11 +03:00
# ifdef CONFIG_BCACHEFS_DEBUG
2020-06-03 02:41:47 +03:00
trans - > pid = current - > pid ;
2020-06-02 23:36:11 +03:00
mutex_lock ( & c - > btree_trans_lock ) ;
list_add ( & trans - > list , & c - > btree_trans_list ) ;
mutex_unlock ( & c - > btree_trans_lock ) ;
# endif
2017-03-17 09:18:50 +03:00
}
int bch2_trans_exit ( struct btree_trans * trans )
{
2019-04-05 04:28:16 +03:00
bch2_trans_unlock ( trans ) ;
2017-03-17 09:18:50 +03:00
2020-06-02 23:36:11 +03:00
# ifdef CONFIG_BCACHEFS_DEBUG
mutex_lock ( & trans - > c - > btree_trans_lock ) ;
list_del ( & trans - > list ) ;
mutex_unlock ( & trans - > c - > btree_trans_lock ) ;
# endif
2019-03-08 03:46:10 +03:00
bch2_journal_preres_put ( & trans - > c - > journal , & trans - > journal_preres ) ;
2019-05-15 22:47:43 +03:00
kfree ( trans - > fs_usage_deltas ) ;
2017-03-17 09:18:50 +03:00
kfree ( trans - > mem ) ;
2019-03-08 07:13:39 +03:00
if ( trans - > used_mempool )
2018-08-09 04:22:46 +03:00
mempool_free ( trans - > iters , & trans - > c - > btree_iters_pool ) ;
2019-03-08 07:13:39 +03:00
else if ( trans - > iters ! = trans - > iters_onstack )
kfree ( trans - > iters ) ;
2017-03-17 09:18:50 +03:00
trans - > mem = ( void * ) 0x1 ;
trans - > iters = ( void * ) 0x1 ;
2019-04-05 04:28:16 +03:00
return trans - > error ? - EIO : 0 ;
2017-03-17 09:18:50 +03:00
}
2019-09-07 21:16:00 +03:00
2020-06-16 02:53:46 +03:00
static void bch2_btree_iter_node_to_text ( struct printbuf * out ,
struct btree_bkey_cached_common * _b ,
enum btree_iter_type type )
{
pr_buf ( out , " %px l=%u %s: " ,
_b , _b - > level , bch2_btree_ids [ _b - > btree_id ] ) ;
bch2_bpos_to_text ( out , btree_node_pos ( _b , type ) ) ;
}
2020-06-02 23:36:11 +03:00
void bch2_btree_trans_to_text ( struct printbuf * out , struct bch_fs * c )
{
# ifdef CONFIG_BCACHEFS_DEBUG
struct btree_trans * trans ;
struct btree_iter * iter ;
struct btree * b ;
unsigned l ;
mutex_lock ( & c - > btree_trans_lock ) ;
list_for_each_entry ( trans , & c - > btree_trans_list , list ) {
2020-06-12 21:58:07 +03:00
pr_buf ( out , " %i %px %ps \n " , trans - > pid , trans , ( void * ) trans - > ip ) ;
2020-06-02 23:36:11 +03:00
trans_for_each_iter ( trans , iter ) {
if ( ! iter - > nodes_locked )
continue ;
2020-06-12 21:58:07 +03:00
pr_buf ( out , " iter %u %s: " ,
iter - > idx ,
bch2_btree_ids [ iter - > btree_id ] ) ;
2020-06-02 23:36:11 +03:00
bch2_bpos_to_text ( out , iter - > pos ) ;
pr_buf ( out , " \n " ) ;
for ( l = 0 ; l < BTREE_MAX_DEPTH ; l + + ) {
if ( btree_node_locked ( iter , l ) ) {
2020-06-16 02:53:46 +03:00
pr_buf ( out , " %s l=%u " ,
btree_node_intent_locked ( iter , l ) ? " i " : " r " , l ) ;
bch2_btree_iter_node_to_text ( out ,
( void * ) iter - > l [ l ] . b ,
btree_iter_type ( iter ) ) ;
2020-06-02 23:36:11 +03:00
pr_buf ( out , " \n " ) ;
}
}
}
b = READ_ONCE ( trans - > locking ) ;
if ( b ) {
2020-06-12 21:58:07 +03:00
pr_buf ( out , " locking iter %u l=%u %s: " ,
trans - > locking_iter_idx ,
trans - > locking_level ,
bch2_btree_ids [ trans - > locking_btree_id ] ) ;
bch2_bpos_to_text ( out , trans - > locking_pos ) ;
2020-06-16 02:53:46 +03:00
pr_buf ( out , " node " ) ;
bch2_btree_iter_node_to_text ( out ,
( void * ) b ,
btree_iter_type ( & trans - > iters [ trans - > locking_iter_idx ] ) ) ;
2020-06-02 23:36:11 +03:00
pr_buf ( out , " \n " ) ;
}
}
mutex_unlock ( & c - > btree_trans_lock ) ;
# endif
}
2019-09-07 21:16:00 +03:00
void bch2_fs_btree_iter_exit ( struct bch_fs * c )
{
mempool_exit ( & c - > btree_iters_pool ) ;
}
int bch2_fs_btree_iter_init ( struct bch_fs * c )
{
unsigned nr = BTREE_ITER_MAX ;
2020-06-02 23:36:11 +03:00
INIT_LIST_HEAD ( & c - > btree_trans_list ) ;
mutex_init ( & c - > btree_trans_lock ) ;
2019-09-07 21:16:00 +03:00
return mempool_init_kmalloc_pool ( & c - > btree_iters_pool , 1 ,
sizeof ( struct btree_iter ) * nr +
2019-12-30 21:08:26 +03:00
sizeof ( struct btree_insert_entry ) * nr +
2019-12-30 22:37:25 +03:00
sizeof ( struct btree_insert_entry ) * nr ) ;
2019-09-07 21:16:00 +03:00
}