2021-12-05 00:31:54 -05:00
// SPDX-License-Identifier: GPL-2.0
# include "bcachefs.h"
2022-02-17 03:11:39 -05:00
# include "alloc_background.h"
2021-12-05 00:31:54 -05:00
# include "btree_iter.h"
# include "btree_update.h"
2023-02-06 18:51:42 -05:00
# include "btree_write_buffer.h"
2021-12-05 00:31:54 -05:00
# include "error.h"
# include "lru.h"
2022-02-17 03:11:39 -05:00
# include "recovery.h"
2021-12-05 00:31:54 -05:00
2022-12-05 16:49:13 -05:00
/* KEY_TYPE_lru is obsolete: */
2023-10-24 20:44:36 -04:00
int bch2_lru_invalid ( struct bch_fs * c , struct bkey_s_c k ,
2023-07-06 21:16:10 -04:00
enum bkey_invalid_flags flags ,
struct printbuf * err )
2021-12-05 00:31:54 -05:00
{
2023-10-24 20:44:36 -04:00
int ret = 0 ;
2023-01-04 00:00:55 -05:00
2023-10-24 20:44:36 -04:00
bkey_fsck_err_on ( ! lru_pos_time ( k . k - > p ) , c , err ,
lru_entry_at_time_0 ,
" lru entry at time=0 " ) ;
fsck_err :
return ret ;
2021-12-05 00:31:54 -05:00
}
void bch2_lru_to_text ( struct printbuf * out , struct bch_fs * c ,
struct bkey_s_c k )
{
const struct bch_lru * lru = bkey_s_c_to_lru ( k ) . v ;
2023-02-03 21:01:40 -05:00
prt_printf ( out , " idx %llu " , le64_to_cpu ( lru - > idx ) ) ;
2021-12-05 00:31:54 -05:00
}
2023-01-03 23:54:10 -05:00
void bch2_lru_pos_to_text ( struct printbuf * out , struct bpos lru )
{
prt_printf ( out , " %llu:%llu -> %llu:%llu " ,
lru_pos_id ( lru ) ,
lru_pos_time ( lru ) ,
u64_to_bucket ( lru . offset ) . inode ,
u64_to_bucket ( lru . offset ) . offset ) ;
}
2022-12-05 16:49:13 -05:00
static int __bch2_lru_set ( struct btree_trans * trans , u16 lru_id ,
2023-07-17 00:56:07 -04:00
u64 dev_bucket , u64 time , bool set )
2021-12-05 00:31:54 -05:00
{
2023-07-17 00:56:07 -04:00
return time
2024-02-08 19:10:19 -05:00
? bch2_btree_bit_mod_buffered ( trans , BTREE_ID_lru ,
lru_pos ( lru_id , dev_bucket , time ) , set )
2023-07-17 00:56:07 -04:00
: 0 ;
2021-12-05 00:31:54 -05:00
}
2022-12-05 16:49:13 -05:00
int bch2_lru_del ( struct btree_trans * trans , u16 lru_id , u64 dev_bucket , u64 time )
2021-12-05 00:31:54 -05:00
{
2022-12-05 16:49:13 -05:00
return __bch2_lru_set ( trans , lru_id , dev_bucket , time , KEY_TYPE_deleted ) ;
}
2021-12-05 00:31:54 -05:00
2022-12-05 16:49:13 -05:00
int bch2_lru_set ( struct btree_trans * trans , u16 lru_id , u64 dev_bucket , u64 time )
{
return __bch2_lru_set ( trans , lru_id , dev_bucket , time , KEY_TYPE_set ) ;
2021-12-05 00:31:54 -05:00
}
2022-12-05 16:49:13 -05:00
int bch2_lru_change ( struct btree_trans * trans ,
u16 lru_id , u64 dev_bucket ,
u64 old_time , u64 new_time )
2021-12-05 00:31:54 -05:00
{
2022-12-05 16:49:13 -05:00
if ( old_time = = new_time )
2021-12-05 00:31:54 -05:00
return 0 ;
2022-12-05 16:49:13 -05:00
return bch2_lru_del ( trans , lru_id , dev_bucket , old_time ) ? :
bch2_lru_set ( trans , lru_id , dev_bucket , new_time ) ;
2021-12-05 00:31:54 -05:00
}
2022-02-17 03:11:39 -05:00
2022-12-05 10:24:19 -05:00
static const char * const bch2_lru_types [ ] = {
# define x(n) #n,
BCH_LRU_TYPES ( )
# undef x
NULL
} ;
2022-02-17 03:11:39 -05:00
static int bch2_check_lru_key ( struct btree_trans * trans ,
2022-07-17 00:44:19 -04:00
struct btree_iter * lru_iter ,
2023-02-06 18:51:42 -05:00
struct bkey_s_c lru_k ,
struct bpos * last_flushed_pos )
2022-02-17 03:11:39 -05:00
{
struct bch_fs * c = trans - > c ;
struct btree_iter iter ;
2022-07-17 00:44:19 -04:00
struct bkey_s_c k ;
2023-01-30 20:58:43 -05:00
struct bch_alloc_v4 a_convert ;
const struct bch_alloc_v4 * a ;
2022-02-17 03:11:39 -05:00
struct printbuf buf1 = PRINTBUF ;
struct printbuf buf2 = PRINTBUF ;
2022-12-05 10:24:19 -05:00
enum bch_lru_type type = lru_type ( lru_k ) ;
2022-12-05 16:49:13 -05:00
struct bpos alloc_pos = u64_to_bucket ( lru_k . k - > p . offset ) ;
2022-12-05 10:24:19 -05:00
u64 idx ;
2022-02-17 03:11:39 -05:00
int ret ;
2022-04-09 15:15:36 -04:00
if ( fsck_err_on ( ! bch2_dev_bucket_exists ( c , alloc_pos ) , c ,
2023-10-24 20:44:36 -04:00
lru_entry_to_invalid_bucket ,
2022-04-09 15:15:36 -04:00
" lru key points to nonexistent device:bucket %llu:%llu " ,
alloc_pos . inode , alloc_pos . offset ) )
return bch2_btree_delete_at ( trans , lru_iter , 0 ) ;
2023-04-29 19:33:09 -04:00
k = bch2_bkey_get_iter ( trans , & iter , BTREE_ID_alloc , alloc_pos , 0 ) ;
2022-02-17 03:11:39 -05:00
ret = bkey_err ( k ) ;
if ( ret )
goto err ;
2023-01-30 20:58:43 -05:00
a = bch2_alloc_to_v4 ( k , & a_convert ) ;
2022-02-17 03:11:39 -05:00
2022-12-05 10:24:19 -05:00
switch ( type ) {
case BCH_LRU_read :
idx = alloc_lru_idx_read ( * a ) ;
break ;
case BCH_LRU_fragmentation :
idx = a - > fragmentation_lru ;
break ;
}
2023-02-06 18:51:42 -05:00
if ( lru_k . k - > type ! = KEY_TYPE_set | |
2022-12-05 10:24:19 -05:00
lru_pos_time ( lru_k . k - > p ) ! = idx ) {
2023-02-06 18:51:42 -05:00
if ( ! bpos_eq ( * last_flushed_pos , lru_k . k - > p ) ) {
* last_flushed_pos = lru_k . k - > p ;
ret = bch2_btree_write_buffer_flush_sync ( trans ) ? :
- BCH_ERR_transaction_restart_write_buffer_flush ;
goto out ;
}
2024-03-11 21:15:26 -04:00
if ( fsck_err ( c , lru_entry_bad ,
2023-10-24 20:44:36 -04:00
" incorrect lru entry: lru %s time %llu \n "
2022-12-05 10:24:19 -05:00
" %s \n "
" for %s " ,
bch2_lru_types [ type ] ,
lru_pos_time ( lru_k . k - > p ) ,
( bch2_bkey_val_to_text ( & buf1 , c , lru_k ) , buf1 . buf ) ,
( bch2_bkey_val_to_text ( & buf2 , c , k ) , buf2 . buf ) ) )
2023-02-06 18:51:42 -05:00
ret = bch2_btree_delete_at ( trans , lru_iter , 0 ) ;
2022-02-17 03:11:39 -05:00
}
2023-02-06 18:51:42 -05:00
out :
2022-02-17 03:11:39 -05:00
err :
fsck_err :
bch2_trans_iter_exit ( trans , & iter ) ;
printbuf_exit ( & buf2 ) ;
printbuf_exit ( & buf1 ) ;
return ret ;
}
2022-05-14 06:58:51 -04:00
int bch2_check_lrus ( struct bch_fs * c )
2022-02-17 03:11:39 -05:00
{
2023-02-06 18:51:42 -05:00
struct bpos last_flushed_pos = POS_MIN ;
2023-12-16 22:30:09 -05:00
int ret = bch2_trans_run ( c ,
2023-09-12 17:16:02 -04:00
for_each_btree_key_commit ( trans , iter ,
2023-06-20 13:49:25 -04:00
BTREE_ID_lru , POS_MIN , BTREE_ITER_PREFETCH , k ,
2023-11-11 16:31:50 -05:00
NULL , NULL , BCH_TRANS_COMMIT_no_enospc | BCH_TRANS_COMMIT_lazy_rw ,
2023-09-12 17:16:02 -04:00
bch2_check_lru_key ( trans , & iter , k , & last_flushed_pos ) ) ) ;
2023-12-16 22:43:41 -05:00
bch_err_fn ( c , ret ) ;
2022-02-17 03:11:39 -05:00
return ret ;
}