2017-03-16 22:18:50 -08:00
// SPDX-License-Identifier: GPL-2.0
# include "bcachefs.h"
# include "error.h"
# include "io.h"
# include "super.h"
2019-09-07 12:42:27 -04:00
# define FSCK_ERR_RATELIMIT_NR 10
2017-03-16 22:18:50 -08:00
bool bch2_inconsistent_error ( struct bch_fs * c )
{
set_bit ( BCH_FS_ERROR , & c - > flags ) ;
switch ( c - > opts . errors ) {
case BCH_ON_ERROR_CONTINUE :
return false ;
case BCH_ON_ERROR_RO :
if ( bch2_fs_emergency_read_only ( c ) )
bch_err ( c , " emergency read only " ) ;
return true ;
case BCH_ON_ERROR_PANIC :
panic ( bch2_fmt ( c , " panic after error " ) ) ;
return true ;
default :
BUG ( ) ;
}
}
void bch2_fatal_error ( struct bch_fs * c )
{
if ( bch2_fs_emergency_read_only ( c ) )
bch_err ( c , " emergency read only " ) ;
}
void bch2_io_error_work ( struct work_struct * work )
{
struct bch_dev * ca = container_of ( work , struct bch_dev , io_error_work ) ;
struct bch_fs * c = ca - > fs ;
bool dev ;
2020-06-15 14:58:47 -04:00
down_write ( & c - > state_lock ) ;
2017-03-16 22:18:50 -08:00
dev = bch2_dev_state_allowed ( c , ca , BCH_MEMBER_STATE_RO ,
BCH_FORCE_IF_DEGRADED ) ;
if ( dev
? __bch2_dev_set_state ( c , ca , BCH_MEMBER_STATE_RO ,
BCH_FORCE_IF_DEGRADED )
: bch2_fs_emergency_read_only ( c ) )
bch_err ( ca ,
" too many IO errors, setting %s RO " ,
dev ? " device " : " filesystem " ) ;
2020-06-15 14:58:47 -04:00
up_write ( & c - > state_lock ) ;
2017-03-16 22:18:50 -08:00
}
void bch2_io_error ( struct bch_dev * ca )
{
//queue_work(system_long_wq, &ca->io_error_work);
}
# ifdef __KERNEL__
# define ask_yn() false
# else
# include "tools-util.h"
# endif
enum fsck_err_ret bch2_fsck_err ( struct bch_fs * c , unsigned flags ,
const char * fmt , . . . )
{
2019-11-06 15:32:11 -05:00
struct fsck_err_state * s = NULL ;
2017-03-16 22:18:50 -08:00
va_list args ;
bool fix = false , print = true , suppressing = false ;
char _buf [ sizeof ( s - > buf ) ] , * buf = _buf ;
2019-03-28 03:28:59 -04:00
if ( test_bit ( BCH_FS_FSCK_DONE , & c - > flags ) ) {
va_start ( args , fmt ) ;
vprintk ( fmt , args ) ;
va_end ( args ) ;
2017-03-16 22:18:50 -08:00
2019-03-28 09:34:55 -04:00
return bch2_inconsistent_error ( c )
? FSCK_ERR_EXIT
: FSCK_ERR_FIX ;
2019-03-28 03:28:59 -04:00
}
mutex_lock ( & c - > fsck_error_lock ) ;
2017-03-16 22:18:50 -08:00
list_for_each_entry ( s , & c - > fsck_errors , list )
if ( s - > fmt = = fmt )
goto found ;
2020-05-28 17:15:41 -04:00
s = kzalloc ( sizeof ( * s ) , GFP_NOFS ) ;
2017-03-16 22:18:50 -08:00
if ( ! s ) {
if ( ! c - > fsck_alloc_err )
bch_err ( c , " kmalloc err, cannot ratelimit fsck errs " ) ;
c - > fsck_alloc_err = true ;
buf = _buf ;
goto print ;
}
INIT_LIST_HEAD ( & s - > list ) ;
s - > fmt = fmt ;
found :
list_move ( & s - > list , & c - > fsck_errors ) ;
s - > nr + + ;
2019-11-06 15:32:11 -05:00
if ( c - > opts . ratelimit_errors & &
s - > nr > = FSCK_ERR_RATELIMIT_NR ) {
if ( s - > nr = = FSCK_ERR_RATELIMIT_NR )
suppressing = true ;
else
print = false ;
}
2017-03-16 22:18:50 -08:00
buf = s - > buf ;
print :
va_start ( args , fmt ) ;
vscnprintf ( buf , sizeof ( _buf ) , fmt , args ) ;
va_end ( args ) ;
if ( c - > opts . fix_errors = = FSCK_OPT_EXIT ) {
bch_err ( c , " %s, exiting " , buf ) ;
2019-03-28 09:34:55 -04:00
} else if ( flags & FSCK_CAN_FIX ) {
2017-03-16 22:18:50 -08:00
if ( c - > opts . fix_errors = = FSCK_OPT_ASK ) {
printk ( KERN_ERR " %s: fix? " , buf ) ;
fix = ask_yn ( ) ;
} else if ( c - > opts . fix_errors = = FSCK_OPT_YES | |
( c - > opts . nochanges & &
! ( flags & FSCK_CAN_IGNORE ) ) ) {
if ( print )
bch_err ( c , " %s, fixing " , buf ) ;
fix = true ;
} else {
if ( print )
bch_err ( c , " %s, not fixing " , buf ) ;
fix = false ;
}
} else if ( flags & FSCK_NEED_FSCK ) {
if ( print )
bch_err ( c , " %s (run fsck to correct) " , buf ) ;
} else {
if ( print )
bch_err ( c , " %s (repair unimplemented) " , buf ) ;
}
if ( suppressing )
bch_err ( c , " Ratelimiting new instances of previous error " ) ;
mutex_unlock ( & c - > fsck_error_lock ) ;
2019-03-28 09:34:55 -04:00
if ( fix ) {
set_bit ( BCH_FS_ERRORS_FIXED , & c - > flags ) ;
return FSCK_ERR_FIX ;
} else {
set_bit ( BCH_FS_ERROR , & c - > flags ) ;
return c - > opts . fix_errors = = FSCK_OPT_EXIT | |
! ( flags & FSCK_CAN_IGNORE )
? FSCK_ERR_EXIT
: FSCK_ERR_IGNORE ;
}
2017-03-16 22:18:50 -08:00
}
void bch2_flush_fsck_errs ( struct bch_fs * c )
{
struct fsck_err_state * s , * n ;
mutex_lock ( & c - > fsck_error_lock ) ;
list_for_each_entry_safe ( s , n , & c - > fsck_errors , list ) {
2019-11-06 15:32:11 -05:00
if ( s - > ratelimited )
2017-03-16 22:18:50 -08:00
bch_err ( c , " Saw %llu errors like: \n %s " , s - > nr , s - > buf ) ;
list_del ( & s - > list ) ;
kfree ( s ) ;
}
mutex_unlock ( & c - > fsck_error_lock ) ;
}