2017-03-17 09:18:50 +03:00
// SPDX-License-Identifier: GPL-2.0
/*
* bcache sysfs interfaces
*
* Copyright 2010 , 2011 Kent Overstreet < kent . overstreet @ gmail . com >
* Copyright 2012 Google , Inc .
*/
# ifndef NO_BCACHEFS_SYSFS
# include "bcachefs.h"
2018-10-06 07:46:55 +03:00
# include "alloc_background.h"
2017-03-17 09:18:50 +03:00
# include "sysfs.h"
# include "btree_cache.h"
# include "btree_io.h"
# include "btree_iter.h"
2020-06-16 02:53:46 +03:00
# include "btree_key_cache.h"
2017-03-17 09:18:50 +03:00
# include "btree_update.h"
# include "btree_update_interior.h"
# include "btree_gc.h"
# include "buckets.h"
2019-12-19 23:07:51 +03:00
# include "clock.h"
2017-03-17 09:18:50 +03:00
# include "disk_groups.h"
2018-11-01 22:13:19 +03:00
# include "ec.h"
2017-03-17 09:18:50 +03:00
# include "inode.h"
# include "journal.h"
# include "keylist.h"
# include "move.h"
# include "opts.h"
# include "rebalance.h"
# include "replicas.h"
# include "super-io.h"
# include "tests.h"
# include <linux/blkdev.h>
# include <linux/sort.h>
# include <linux/sched/clock.h>
# include "util.h"
# define SYSFS_OPS(type) \
struct sysfs_ops type # # _sysfs_ops = { \
. show = type # # _show , \
. store = type # # _store \
}
# define SHOW(fn) \
static ssize_t fn # # _show ( struct kobject * kobj , struct attribute * attr , \
char * buf ) \
# define STORE(fn) \
static ssize_t fn # # _store ( struct kobject * kobj , struct attribute * attr , \
const char * buf , size_t size ) \
# define __sysfs_attribute(_name, _mode) \
static struct attribute sysfs_ # # _name = \
{ . name = # _name , . mode = _mode }
# define write_attribute(n) __sysfs_attribute(n, S_IWUSR)
# define read_attribute(n) __sysfs_attribute(n, S_IRUGO)
# define rw_attribute(n) __sysfs_attribute(n, S_IRUGO|S_IWUSR)
# define sysfs_printf(file, fmt, ...) \
do { \
if ( attr = = & sysfs_ # # file ) \
return scnprintf ( buf , PAGE_SIZE , fmt " \n " , __VA_ARGS__ ) ; \
} while ( 0 )
# define sysfs_print(file, var) \
do { \
if ( attr = = & sysfs_ # # file ) \
return snprint ( buf , PAGE_SIZE , var ) ; \
} while ( 0 )
# define sysfs_hprint(file, val) \
do { \
if ( attr = = & sysfs_ # # file ) { \
2018-12-19 20:58:56 +03:00
bch2_hprint ( & out , val ) ; \
pr_buf ( & out , " \n " ) ; \
return out . pos - buf ; \
2017-03-17 09:18:50 +03:00
} \
} while ( 0 )
# define var_printf(_var, fmt) sysfs_printf(_var, fmt, var(_var))
# define var_print(_var) sysfs_print(_var, var(_var))
# define var_hprint(_var) sysfs_hprint(_var, var(_var))
# define sysfs_strtoul(file, var) \
do { \
if ( attr = = & sysfs_ # # file ) \
return strtoul_safe ( buf , var ) ? : ( ssize_t ) size ; \
} while ( 0 )
# define sysfs_strtoul_clamp(file, var, min, max) \
do { \
if ( attr = = & sysfs_ # # file ) \
return strtoul_safe_clamp ( buf , var , min , max ) \
? : ( ssize_t ) size ; \
} while ( 0 )
# define strtoul_or_return(cp) \
( { \
unsigned long _v ; \
int _r = kstrtoul ( cp , 10 , & _v ) ; \
if ( _r ) \
return _r ; \
_v ; \
} )
# define strtoul_restrict_or_return(cp, min, max) \
( { \
unsigned long __v = 0 ; \
int _r = strtoul_safe_restrict ( cp , __v , min , max ) ; \
if ( _r ) \
return _r ; \
__v ; \
} )
# define strtoi_h_or_return(cp) \
( { \
u64 _v ; \
int _r = strtoi_h ( cp , & _v ) ; \
if ( _r ) \
return _r ; \
_v ; \
} )
# define sysfs_hatoi(file, var) \
do { \
if ( attr = = & sysfs_ # # file ) \
return strtoi_h ( buf , & var ) ? : ( ssize_t ) size ; \
} while ( 0 )
write_attribute ( trigger_journal_flush ) ;
write_attribute ( trigger_btree_coalesce ) ;
write_attribute ( trigger_gc ) ;
write_attribute ( prune_cache ) ;
rw_attribute ( btree_gc_periodic ) ;
read_attribute ( uuid ) ;
read_attribute ( minor ) ;
read_attribute ( bucket_size ) ;
read_attribute ( block_size ) ;
read_attribute ( btree_node_size ) ;
read_attribute ( first_bucket ) ;
read_attribute ( nbuckets ) ;
read_attribute ( durability ) ;
read_attribute ( iodone ) ;
read_attribute ( io_latency_read ) ;
read_attribute ( io_latency_write ) ;
read_attribute ( io_latency_stats_read ) ;
read_attribute ( io_latency_stats_write ) ;
read_attribute ( congested ) ;
read_attribute ( bucket_quantiles_last_read ) ;
read_attribute ( bucket_quantiles_last_write ) ;
read_attribute ( bucket_quantiles_fragmentation ) ;
read_attribute ( bucket_quantiles_oldest_gen ) ;
read_attribute ( reserve_stats ) ;
read_attribute ( btree_cache_size ) ;
read_attribute ( compression_stats ) ;
read_attribute ( journal_debug ) ;
read_attribute ( journal_pins ) ;
read_attribute ( btree_updates ) ;
read_attribute ( dirty_btree_nodes ) ;
2020-11-20 04:13:30 +03:00
read_attribute ( btree_cache ) ;
2020-06-16 02:53:46 +03:00
read_attribute ( btree_key_cache ) ;
2020-06-02 23:36:11 +03:00
read_attribute ( btree_transactions ) ;
2020-07-07 03:18:13 +03:00
read_attribute ( stripes_heap ) ;
2017-03-17 09:18:50 +03:00
read_attribute ( internal_uuid ) ;
read_attribute ( has_data ) ;
read_attribute ( alloc_debug ) ;
write_attribute ( wake_allocator ) ;
read_attribute ( read_realloc_races ) ;
read_attribute ( extent_migrate_done ) ;
read_attribute ( extent_migrate_raced ) ;
rw_attribute ( journal_write_delay_ms ) ;
rw_attribute ( journal_reclaim_delay_ms ) ;
rw_attribute ( discard ) ;
rw_attribute ( cache_replacement_policy ) ;
rw_attribute ( label ) ;
rw_attribute ( copy_gc_enabled ) ;
sysfs_pd_controller_attribute ( copy_gc ) ;
rw_attribute ( rebalance_enabled ) ;
sysfs_pd_controller_attribute ( rebalance ) ;
read_attribute ( rebalance_work ) ;
rw_attribute ( promote_whole_extents ) ;
2018-11-01 22:13:19 +03:00
read_attribute ( new_stripes ) ;
2017-03-17 09:18:50 +03:00
rw_attribute ( pd_controllers_update_seconds ) ;
read_attribute ( meta_replicas_have ) ;
read_attribute ( data_replicas_have ) ;
2019-12-19 23:07:51 +03:00
read_attribute ( io_timers_read ) ;
read_attribute ( io_timers_write ) ;
2017-03-17 09:18:50 +03:00
# ifdef CONFIG_BCACHEFS_TESTS
write_attribute ( perf_test ) ;
# endif /* CONFIG_BCACHEFS_TESTS */
# define x(_name) \
static struct attribute sysfs_time_stat_ # # _name = \
{ . name = # _name , . mode = S_IRUGO } ;
BCH_TIME_STATS ( )
# undef x
static struct attribute sysfs_state_rw = {
. name = " state " ,
. mode = S_IRUGO
} ;
static size_t bch2_btree_cache_size ( struct bch_fs * c )
{
size_t ret = 0 ;
struct btree * b ;
mutex_lock ( & c - > btree_cache . lock ) ;
list_for_each_entry ( b , & c - > btree_cache . live , list )
ret + = btree_bytes ( c ) ;
mutex_unlock ( & c - > btree_cache . lock ) ;
return ret ;
}
2020-07-26 00:06:11 +03:00
static int fs_alloc_debug_to_text ( struct printbuf * out , struct bch_fs * c )
2017-03-17 09:18:50 +03:00
{
2019-02-11 03:34:47 +03:00
struct bch_fs_usage_online * fs_usage = bch2_fs_usage_read ( c ) ;
2018-07-24 20:33:07 +03:00
2019-01-21 23:32:13 +03:00
if ( ! fs_usage )
return - ENOMEM ;
2018-07-24 20:33:07 +03:00
2020-07-26 00:06:11 +03:00
bch2_fs_usage_to_text ( out , c , fs_usage ) ;
2019-01-21 23:32:13 +03:00
percpu_up_read ( & c - > mark_lock ) ;
kfree ( fs_usage ) ;
2020-07-26 00:06:11 +03:00
return 0 ;
2017-03-17 09:18:50 +03:00
}
2020-07-26 00:06:11 +03:00
static int bch2_compression_stats_to_text ( struct printbuf * out , struct bch_fs * c )
2017-03-17 09:18:50 +03:00
{
2019-03-25 22:10:15 +03:00
struct btree_trans trans ;
struct btree_iter * iter ;
2017-03-17 09:18:50 +03:00
struct bkey_s_c k ;
u64 nr_uncompressed_extents = 0 , uncompressed_sectors = 0 ,
nr_compressed_extents = 0 ,
compressed_sectors_compressed = 0 ,
compressed_sectors_uncompressed = 0 ;
2019-04-17 22:49:28 +03:00
int ret ;
2017-03-17 09:18:50 +03:00
2019-03-22 05:19:57 +03:00
if ( ! test_bit ( BCH_FS_STARTED , & c - > flags ) )
2017-03-17 09:18:50 +03:00
return - EPERM ;
2019-05-15 17:54:43 +03:00
bch2_trans_init ( & trans , c , 0 , 0 ) ;
2019-03-25 22:10:15 +03:00
2019-04-17 22:49:28 +03:00
for_each_btree_key ( & trans , iter , BTREE_ID_EXTENTS , POS_MIN , 0 , k , ret )
2018-11-01 22:10:01 +03:00
if ( k . k - > type = = KEY_TYPE_extent ) {
2017-03-17 09:18:50 +03:00
struct bkey_s_c_extent e = bkey_s_c_to_extent ( k ) ;
2018-09-28 04:08:39 +03:00
const union bch_extent_entry * entry ;
struct extent_ptr_decoded p ;
2017-03-17 09:18:50 +03:00
2018-09-28 04:08:39 +03:00
extent_for_each_ptr_decode ( e , p , entry ) {
2018-02-24 00:26:10 +03:00
if ( ! crc_is_compressed ( p . crc ) ) {
2017-03-17 09:18:50 +03:00
nr_uncompressed_extents + + ;
uncompressed_sectors + = e . k - > size ;
} else {
nr_compressed_extents + + ;
compressed_sectors_compressed + =
2018-09-28 04:08:39 +03:00
p . crc . compressed_size ;
2017-03-17 09:18:50 +03:00
compressed_sectors_uncompressed + =
2018-09-28 04:08:39 +03:00
p . crc . uncompressed_size ;
2017-03-17 09:18:50 +03:00
}
/* only looking at the first ptr */
break ;
}
}
2019-04-17 22:49:28 +03:00
ret = bch2_trans_exit ( & trans ) ? : ret ;
if ( ret )
return ret ;
2017-03-17 09:18:50 +03:00
2020-07-26 00:06:11 +03:00
pr_buf ( out ,
" uncompressed data: \n "
" nr extents: %llu \n "
" size (bytes): %llu \n "
" compressed data: \n "
" nr extents: %llu \n "
" compressed size (bytes): %llu \n "
" uncompressed size (bytes): %llu \n " ,
nr_uncompressed_extents ,
uncompressed_sectors < < 9 ,
nr_compressed_extents ,
compressed_sectors_compressed < < 9 ,
compressed_sectors_uncompressed < < 9 ) ;
return 0 ;
2018-11-01 22:13:19 +03:00
}
2017-03-17 09:18:50 +03:00
SHOW ( bch2_fs )
{
struct bch_fs * c = container_of ( kobj , struct bch_fs , kobj ) ;
2020-07-26 00:06:11 +03:00
struct printbuf out = _PBUF ( buf , PAGE_SIZE ) ;
2017-03-17 09:18:50 +03:00
sysfs_print ( minor , c - > minor ) ;
sysfs_printf ( internal_uuid , " %pU " , c - > sb . uuid . b ) ;
sysfs_print ( journal_write_delay_ms , c - > journal . write_delay_ms ) ;
sysfs_print ( journal_reclaim_delay_ms , c - > journal . reclaim_delay_ms ) ;
sysfs_print ( block_size , block_bytes ( c ) ) ;
sysfs_print ( btree_node_size , btree_bytes ( c ) ) ;
sysfs_hprint ( btree_cache_size , bch2_btree_cache_size ( c ) ) ;
sysfs_print ( read_realloc_races ,
atomic_long_read ( & c - > read_realloc_races ) ) ;
sysfs_print ( extent_migrate_done ,
atomic_long_read ( & c - > extent_migrate_done ) ) ;
sysfs_print ( extent_migrate_raced ,
atomic_long_read ( & c - > extent_migrate_raced ) ) ;
sysfs_printf ( btree_gc_periodic , " %u " , ( int ) c - > btree_gc_periodic ) ;
sysfs_printf ( copy_gc_enabled , " %i " , c - > copy_gc_enabled ) ;
sysfs_print ( pd_controllers_update_seconds ,
c - > pd_controllers_update_seconds ) ;
sysfs_printf ( rebalance_enabled , " %i " , c - > rebalance . enabled ) ;
sysfs_pd_controller_show ( rebalance , & c - > rebalance . pd ) ; /* XXX */
2020-07-11 23:28:54 +03:00
sysfs_pd_controller_show ( copy_gc , & c - > copygc_pd ) ;
2017-03-17 09:18:50 +03:00
2020-07-26 00:06:11 +03:00
if ( attr = = & sysfs_rebalance_work ) {
bch2_rebalance_work_to_text ( & out , c ) ;
return out . pos - buf ;
}
2017-03-17 09:18:50 +03:00
sysfs_print ( promote_whole_extents , c - > promote_whole_extents ) ;
2018-10-30 21:32:47 +03:00
sysfs_printf ( meta_replicas_have , " %i " , bch2_replicas_online ( c , true ) ) ;
sysfs_printf ( data_replicas_have , " %i " , bch2_replicas_online ( c , false ) ) ;
2017-03-17 09:18:50 +03:00
/* Debugging: */
if ( attr = = & sysfs_alloc_debug )
2020-07-26 00:06:11 +03:00
return fs_alloc_debug_to_text ( & out , c ) ? : out . pos - buf ;
2017-03-17 09:18:50 +03:00
2020-07-26 00:06:11 +03:00
if ( attr = = & sysfs_journal_debug ) {
bch2_journal_debug_to_text ( & out , & c - > journal ) ;
return out . pos - buf ;
}
2017-03-17 09:18:50 +03:00
2020-07-26 00:06:11 +03:00
if ( attr = = & sysfs_journal_pins ) {
bch2_journal_pins_to_text ( & out , & c - > journal ) ;
return out . pos - buf ;
}
2017-03-17 09:18:50 +03:00
2020-07-26 00:06:11 +03:00
if ( attr = = & sysfs_btree_updates ) {
bch2_btree_updates_to_text ( & out , c ) ;
return out . pos - buf ;
}
2017-03-17 09:18:50 +03:00
2020-07-26 00:06:11 +03:00
if ( attr = = & sysfs_dirty_btree_nodes ) {
bch2_dirty_btree_nodes_to_text ( & out , c ) ;
return out . pos - buf ;
}
2020-06-16 02:53:46 +03:00
2020-11-20 04:13:30 +03:00
if ( attr = = & sysfs_btree_cache ) {
bch2_btree_cache_to_text ( & out , c ) ;
return out . pos - buf ;
}
2020-06-16 02:53:46 +03:00
if ( attr = = & sysfs_btree_key_cache ) {
bch2_btree_key_cache_to_text ( & out , & c - > btree_key_cache ) ;
return out . pos - buf ;
}
2020-06-02 23:36:11 +03:00
if ( attr = = & sysfs_btree_transactions ) {
bch2_btree_trans_to_text ( & out , c ) ;
return out . pos - buf ;
}
2017-03-17 09:18:50 +03:00
2020-07-07 03:18:13 +03:00
if ( attr = = & sysfs_stripes_heap ) {
bch2_stripes_heap_to_text ( & out , c ) ;
return out . pos - buf ;
}
2020-07-26 00:06:11 +03:00
if ( attr = = & sysfs_compression_stats ) {
bch2_compression_stats_to_text ( & out , c ) ;
return out . pos - buf ;
}
2017-03-17 09:18:50 +03:00
2020-07-26 00:06:11 +03:00
if ( attr = = & sysfs_new_stripes ) {
bch2_new_stripes_to_text ( & out , c ) ;
return out . pos - buf ;
}
2018-11-01 22:13:19 +03:00
2020-07-26 00:06:11 +03:00
if ( attr = = & sysfs_io_timers_read ) {
bch2_io_timers_to_text ( & out , & c - > io_clock [ READ ] ) ;
return out . pos - buf ;
}
if ( attr = = & sysfs_io_timers_write ) {
bch2_io_timers_to_text ( & out , & c - > io_clock [ WRITE ] ) ;
return out . pos - buf ;
}
2019-12-19 23:07:51 +03:00
2017-03-17 09:18:50 +03:00
return 0 ;
}
2020-06-15 21:58:47 +03:00
STORE ( bch2_fs )
2017-03-17 09:18:50 +03:00
{
struct bch_fs * c = container_of ( kobj , struct bch_fs , kobj ) ;
sysfs_strtoul ( journal_write_delay_ms , c - > journal . write_delay_ms ) ;
sysfs_strtoul ( journal_reclaim_delay_ms , c - > journal . reclaim_delay_ms ) ;
if ( attr = = & sysfs_btree_gc_periodic ) {
ssize_t ret = strtoul_safe ( buf , c - > btree_gc_periodic )
? : ( ssize_t ) size ;
wake_up_process ( c - > gc_thread ) ;
return ret ;
}
if ( attr = = & sysfs_copy_gc_enabled ) {
ssize_t ret = strtoul_safe ( buf , c - > copy_gc_enabled )
? : ( ssize_t ) size ;
2020-07-11 23:28:54 +03:00
if ( c - > copygc_thread )
wake_up_process ( c - > copygc_thread ) ;
2017-03-17 09:18:50 +03:00
return ret ;
}
if ( attr = = & sysfs_rebalance_enabled ) {
ssize_t ret = strtoul_safe ( buf , c - > rebalance . enabled )
? : ( ssize_t ) size ;
rebalance_wakeup ( c ) ;
return ret ;
}
sysfs_strtoul ( pd_controllers_update_seconds ,
c - > pd_controllers_update_seconds ) ;
sysfs_pd_controller_store ( rebalance , & c - > rebalance . pd ) ;
2020-07-11 23:28:54 +03:00
sysfs_pd_controller_store ( copy_gc , & c - > copygc_pd ) ;
2017-03-17 09:18:50 +03:00
sysfs_strtoul ( promote_whole_extents , c - > promote_whole_extents ) ;
/* Debugging: */
2019-03-22 05:19:57 +03:00
if ( ! test_bit ( BCH_FS_STARTED , & c - > flags ) )
2017-03-17 09:18:50 +03:00
return - EPERM ;
/* Debugging: */
if ( attr = = & sysfs_trigger_journal_flush )
2020-11-14 21:12:50 +03:00
bch2_journal_meta ( & c - > journal ) ;
2017-03-17 09:18:50 +03:00
if ( attr = = & sysfs_trigger_btree_coalesce )
bch2_coalesce ( c ) ;
2020-06-15 21:58:47 +03:00
if ( attr = = & sysfs_trigger_gc ) {
2020-06-15 22:10:54 +03:00
/*
* Full gc is currently incompatible with btree key cache :
*/
#if 0
2020-06-15 21:58:47 +03:00
down_read ( & c - > state_lock ) ;
2019-03-30 05:22:45 +03:00
bch2_gc ( c , NULL , false , false ) ;
2020-06-15 21:58:47 +03:00
up_read ( & c - > state_lock ) ;
2020-06-15 22:10:54 +03:00
# else
bch2_gc_gens ( c ) ;
# endif
2020-06-15 21:58:47 +03:00
}
2017-03-17 09:18:50 +03:00
if ( attr = = & sysfs_prune_cache ) {
struct shrink_control sc ;
sc . gfp_mask = GFP_KERNEL ;
sc . nr_to_scan = strtoul_or_return ( buf ) ;
c - > btree_cache . shrink . scan_objects ( & c - > btree_cache . shrink , & sc ) ;
}
2020-06-15 21:58:47 +03:00
2017-03-17 09:18:50 +03:00
# ifdef CONFIG_BCACHEFS_TESTS
if ( attr = = & sysfs_perf_test ) {
char * tmp = kstrdup ( buf , GFP_KERNEL ) , * p = tmp ;
char * test = strsep ( & p , " \t \n " ) ;
char * nr_str = strsep ( & p , " \t \n " ) ;
char * threads_str = strsep ( & p , " \t \n " ) ;
unsigned threads ;
u64 nr ;
int ret = - EINVAL ;
if ( threads_str & &
! ( ret = kstrtouint ( threads_str , 10 , & threads ) ) & &
! ( ret = bch2_strtoull_h ( nr_str , & nr ) ) )
2020-12-01 20:23:55 +03:00
ret = bch2_btree_perf_test ( c , test , nr , threads ) ;
2017-03-17 09:18:50 +03:00
kfree ( tmp ) ;
2020-12-01 20:23:55 +03:00
if ( ret )
size = ret ;
2017-03-17 09:18:50 +03:00
}
# endif
return size ;
}
SYSFS_OPS ( bch2_fs ) ;
struct attribute * bch2_fs_files [ ] = {
& sysfs_minor ,
& sysfs_block_size ,
& sysfs_btree_node_size ,
& sysfs_btree_cache_size ,
& sysfs_meta_replicas_have ,
& sysfs_data_replicas_have ,
& sysfs_journal_write_delay_ms ,
& sysfs_journal_reclaim_delay_ms ,
& sysfs_promote_whole_extents ,
& sysfs_compression_stats ,
# ifdef CONFIG_BCACHEFS_TESTS
& sysfs_perf_test ,
# endif
NULL
} ;
/* internal dir - just a wrapper */
SHOW ( bch2_fs_internal )
{
struct bch_fs * c = container_of ( kobj , struct bch_fs , internal ) ;
return bch2_fs_show ( & c - > kobj , attr , buf ) ;
}
STORE ( bch2_fs_internal )
{
struct bch_fs * c = container_of ( kobj , struct bch_fs , internal ) ;
return bch2_fs_store ( & c - > kobj , attr , buf , size ) ;
}
SYSFS_OPS ( bch2_fs_internal ) ;
struct attribute * bch2_fs_internal_files [ ] = {
& sysfs_alloc_debug ,
& sysfs_journal_debug ,
& sysfs_journal_pins ,
& sysfs_btree_updates ,
& sysfs_dirty_btree_nodes ,
2020-11-20 04:13:30 +03:00
& sysfs_btree_cache ,
2020-06-16 02:53:46 +03:00
& sysfs_btree_key_cache ,
2020-06-02 23:36:11 +03:00
& sysfs_btree_transactions ,
2020-07-07 03:18:13 +03:00
& sysfs_stripes_heap ,
2017-03-17 09:18:50 +03:00
& sysfs_read_realloc_races ,
& sysfs_extent_migrate_done ,
& sysfs_extent_migrate_raced ,
& sysfs_trigger_journal_flush ,
& sysfs_trigger_btree_coalesce ,
& sysfs_trigger_gc ,
& sysfs_prune_cache ,
& sysfs_copy_gc_enabled ,
& sysfs_rebalance_enabled ,
& sysfs_rebalance_work ,
sysfs_pd_controller_files ( rebalance ) ,
2020-07-11 23:28:54 +03:00
sysfs_pd_controller_files ( copy_gc ) ,
2017-03-17 09:18:50 +03:00
2018-11-01 22:13:19 +03:00
& sysfs_new_stripes ,
2019-12-19 23:07:51 +03:00
& sysfs_io_timers_read ,
& sysfs_io_timers_write ,
2017-03-17 09:18:50 +03:00
& sysfs_internal_uuid ,
NULL
} ;
/* options */
SHOW ( bch2_fs_opts_dir )
{
2018-11-09 09:24:07 +03:00
struct printbuf out = _PBUF ( buf , PAGE_SIZE ) ;
2017-03-17 09:18:50 +03:00
struct bch_fs * c = container_of ( kobj , struct bch_fs , opts_dir ) ;
const struct bch_option * opt = container_of ( attr , struct bch_option , attr ) ;
int id = opt - bch2_opt_table ;
u64 v = bch2_opt_get_by_id ( & c - > opts , id ) ;
2018-11-09 09:24:07 +03:00
bch2_opt_to_text ( & out , c , opt , v , OPT_SHOW_FULL_LIST ) ;
pr_buf ( & out , " \n " ) ;
2017-03-17 09:18:50 +03:00
2018-11-09 09:24:07 +03:00
return out . pos - buf ;
2017-03-17 09:18:50 +03:00
}
STORE ( bch2_fs_opts_dir )
{
struct bch_fs * c = container_of ( kobj , struct bch_fs , opts_dir ) ;
const struct bch_option * opt = container_of ( attr , struct bch_option , attr ) ;
int ret , id = opt - bch2_opt_table ;
char * tmp ;
u64 v ;
tmp = kstrdup ( buf , GFP_KERNEL ) ;
if ( ! tmp )
return - ENOMEM ;
ret = bch2_opt_parse ( c , opt , strim ( tmp ) , & v ) ;
kfree ( tmp ) ;
if ( ret < 0 )
return ret ;
2018-11-13 02:30:55 +03:00
ret = bch2_opt_check_may_set ( c , id , v ) ;
if ( ret < 0 )
return ret ;
2017-03-17 09:18:50 +03:00
if ( opt - > set_sb ! = SET_NO_SB_OPT ) {
mutex_lock ( & c - > sb_lock ) ;
opt - > set_sb ( c - > disk_sb . sb , v ) ;
bch2_write_super ( c ) ;
mutex_unlock ( & c - > sb_lock ) ;
}
bch2_opt_set_by_id ( & c - > opts , id , v ) ;
if ( ( id = = Opt_background_target | |
id = = Opt_background_compression ) & & v ) {
bch2_rebalance_add_work ( c , S64_MAX ) ;
rebalance_wakeup ( c ) ;
}
return size ;
}
SYSFS_OPS ( bch2_fs_opts_dir ) ;
struct attribute * bch2_fs_opts_dir_files [ ] = { NULL } ;
int bch2_opts_create_sysfs_files ( struct kobject * kobj )
{
const struct bch_option * i ;
int ret ;
for ( i = bch2_opt_table ;
i < bch2_opt_table + bch2_opts_nr ;
i + + ) {
2018-12-19 20:58:56 +03:00
if ( ! ( i - > mode & ( OPT_FORMAT | OPT_MOUNT | OPT_RUNTIME ) ) )
2017-03-17 09:18:50 +03:00
continue ;
ret = sysfs_create_file ( kobj , & i - > attr ) ;
if ( ret )
return ret ;
}
return 0 ;
}
/* time stats */
SHOW ( bch2_fs_time_stats )
{
struct bch_fs * c = container_of ( kobj , struct bch_fs , time_stats ) ;
2020-07-26 00:06:11 +03:00
struct printbuf out = _PBUF ( buf , PAGE_SIZE ) ;
2017-03-17 09:18:50 +03:00
2020-07-26 00:06:11 +03:00
# define x(name) \
if ( attr = = & sysfs_time_stat_ # # name ) { \
bch2_time_stats_to_text ( & out , & c - > times [ BCH_TIME_ # # name ] ) ; \
return out . pos - buf ; \
}
2017-03-17 09:18:50 +03:00
BCH_TIME_STATS ( )
# undef x
return 0 ;
}
STORE ( bch2_fs_time_stats )
{
return size ;
}
SYSFS_OPS ( bch2_fs_time_stats ) ;
struct attribute * bch2_fs_time_stats_files [ ] = {
# define x(name) \
& sysfs_time_stat_ # # name ,
BCH_TIME_STATS ( )
# undef x
NULL
} ;
typedef unsigned ( bucket_map_fn ) ( struct bch_fs * , struct bch_dev * ,
size_t , void * ) ;
static unsigned bucket_last_io_fn ( struct bch_fs * c , struct bch_dev * ca ,
size_t b , void * private )
{
int rw = ( private ? 1 : 0 ) ;
return bucket_last_io ( c , bucket ( ca , b ) , rw ) ;
}
static unsigned bucket_sectors_used_fn ( struct bch_fs * c , struct bch_dev * ca ,
size_t b , void * private )
{
struct bucket * g = bucket ( ca , b ) ;
return bucket_sectors_used ( g - > mark ) ;
}
static unsigned bucket_oldest_gen_fn ( struct bch_fs * c , struct bch_dev * ca ,
size_t b , void * private )
{
return bucket_gc_gen ( ca , b ) ;
}
static int unsigned_cmp ( const void * _l , const void * _r )
{
2019-04-12 11:54:12 +03:00
const unsigned * l = _l ;
const unsigned * r = _r ;
2017-03-17 09:18:50 +03:00
2019-04-12 11:54:12 +03:00
return cmp_int ( * l , * r ) ;
2017-03-17 09:18:50 +03:00
}
2020-07-26 00:06:11 +03:00
static int quantiles_to_text ( struct printbuf * out ,
struct bch_fs * c , struct bch_dev * ca ,
bucket_map_fn * fn , void * private )
2017-03-17 09:18:50 +03:00
{
size_t i , n ;
/* Compute 31 quantiles */
unsigned q [ 31 ] , * p ;
down_read ( & ca - > bucket_lock ) ;
n = ca - > mi . nbuckets ;
p = vzalloc ( n * sizeof ( unsigned ) ) ;
if ( ! p ) {
up_read ( & ca - > bucket_lock ) ;
return - ENOMEM ;
}
for ( i = ca - > mi . first_bucket ; i < n ; i + + )
p [ i ] = fn ( c , ca , i , private ) ;
sort ( p , n , sizeof ( unsigned ) , unsigned_cmp , NULL ) ;
up_read ( & ca - > bucket_lock ) ;
while ( n & &
! p [ n - 1 ] )
- - n ;
for ( i = 0 ; i < ARRAY_SIZE ( q ) ; i + + )
q [ i ] = p [ n * ( i + 1 ) / ( ARRAY_SIZE ( q ) + 1 ) ] ;
vfree ( p ) ;
for ( i = 0 ; i < ARRAY_SIZE ( q ) ; i + + )
2020-07-26 00:06:11 +03:00
pr_buf ( out , " %u " , q [ i ] ) ;
pr_buf ( out , " \n " ) ;
return 0 ;
2017-03-17 09:18:50 +03:00
}
2020-07-26 00:06:11 +03:00
static void reserve_stats_to_text ( struct printbuf * out , struct bch_dev * ca )
2017-03-17 09:18:50 +03:00
{
enum alloc_reserve i ;
2019-11-21 00:16:57 +03:00
spin_lock ( & ca - > fs - > freelist_lock ) ;
2017-03-17 09:18:50 +03:00
2020-07-26 00:06:11 +03:00
pr_buf ( out , " free_inc: \t %zu \t %zu \n " ,
2018-11-09 09:24:07 +03:00
fifo_used ( & ca - > free_inc ) ,
ca - > free_inc . size ) ;
2017-03-17 09:18:50 +03:00
for ( i = 0 ; i < RESERVE_NR ; i + + )
2020-07-26 00:06:11 +03:00
pr_buf ( out , " free[%u]: \t %zu \t %zu \n " , i ,
2018-11-09 09:24:07 +03:00
fifo_used ( & ca - > free [ i ] ) ,
ca - > free [ i ] . size ) ;
2017-03-17 09:18:50 +03:00
2019-11-21 00:16:57 +03:00
spin_unlock ( & ca - > fs - > freelist_lock ) ;
2017-03-17 09:18:50 +03:00
}
2020-07-26 00:06:11 +03:00
static void dev_alloc_debug_to_text ( struct printbuf * out , struct bch_dev * ca )
2017-03-17 09:18:50 +03:00
{
struct bch_fs * c = ca - > fs ;
2020-07-22 20:27:00 +03:00
struct bch_dev_usage stats = bch2_dev_usage_read ( ca ) ;
2019-01-19 21:13:29 +03:00
unsigned i , nr [ BCH_DATA_NR ] ;
memset ( nr , 0 , sizeof ( nr ) ) ;
for ( i = 0 ; i < ARRAY_SIZE ( c - > open_buckets ) ; i + + )
nr [ c - > open_buckets [ i ] . type ] + + ;
2017-03-17 09:18:50 +03:00
2020-07-26 00:06:11 +03:00
pr_buf ( out ,
2017-03-17 09:18:50 +03:00
" free_inc: %zu/%zu \n "
" free[RESERVE_MOVINGGC]: %zu/%zu \n "
" free[RESERVE_NONE]: %zu/%zu \n "
" buckets: \n "
" capacity: %llu \n "
" sb: %llu \n "
" journal: %llu \n "
" meta: %llu \n "
" user: %llu \n "
" cached: %llu \n "
2018-11-01 22:13:19 +03:00
" erasure coded: %llu \n "
2018-07-24 23:42:49 +03:00
" available: %lli \n "
2017-03-17 09:18:50 +03:00
" sectors: \n "
" sb: %llu \n "
" journal: %llu \n "
" meta: %llu \n "
" user: %llu \n "
" cached: %llu \n "
2020-06-19 04:06:42 +03:00
" erasure coded: %llu \n "
2018-08-01 21:26:55 +03:00
" fragmented: %llu \n "
" copygc threshold: %llu \n "
2017-03-17 09:18:50 +03:00
" freelist_wait: %s \n "
" open buckets: %u/%u (reserved %u) \n "
2019-01-19 21:13:29 +03:00
" open_buckets_wait: %s \n "
" open_buckets_btree: %u \n "
" open_buckets_user: %u \n "
" btree reserve cache: %u \n " ,
2017-03-17 09:18:50 +03:00
fifo_used ( & ca - > free_inc ) , ca - > free_inc . size ,
fifo_used ( & ca - > free [ RESERVE_MOVINGGC ] ) , ca - > free [ RESERVE_MOVINGGC ] . size ,
fifo_used ( & ca - > free [ RESERVE_NONE ] ) , ca - > free [ RESERVE_NONE ] . size ,
ca - > mi . nbuckets - ca - > mi . first_bucket ,
2020-07-10 01:28:11 +03:00
stats . buckets [ BCH_DATA_sb ] ,
stats . buckets [ BCH_DATA_journal ] ,
stats . buckets [ BCH_DATA_btree ] ,
stats . buckets [ BCH_DATA_user ] ,
stats . buckets [ BCH_DATA_cached ] ,
2018-11-01 22:13:19 +03:00
stats . buckets_ec ,
2020-07-22 20:27:00 +03:00
__dev_buckets_available ( ca , stats ) ,
2020-07-10 01:28:11 +03:00
stats . sectors [ BCH_DATA_sb ] ,
stats . sectors [ BCH_DATA_journal ] ,
stats . sectors [ BCH_DATA_btree ] ,
stats . sectors [ BCH_DATA_user ] ,
stats . sectors [ BCH_DATA_cached ] ,
2020-06-19 04:06:42 +03:00
stats . sectors_ec ,
2018-08-01 21:26:55 +03:00
stats . sectors_fragmented ,
2020-07-11 23:28:54 +03:00
c - > copygc_threshold ,
2017-03-17 09:18:50 +03:00
c - > freelist_wait . list . first ? " waiting " : " empty " ,
2019-01-19 21:13:29 +03:00
c - > open_buckets_nr_free , OPEN_BUCKETS_COUNT ,
BTREE_NODE_OPEN_BUCKET_RESERVE ,
c - > open_buckets_wait . list . first ? " waiting " : " empty " ,
2020-07-10 01:28:11 +03:00
nr [ BCH_DATA_btree ] ,
nr [ BCH_DATA_user ] ,
2019-01-19 21:13:29 +03:00
c - > btree_reserve_cache_nr ) ;
2017-03-17 09:18:50 +03:00
}
static const char * const bch2_rw [ ] = {
" read " ,
" write " ,
NULL
} ;
2020-07-26 00:06:11 +03:00
static void dev_iodone_to_text ( struct printbuf * out , struct bch_dev * ca )
2017-03-17 09:18:50 +03:00
{
2019-02-06 19:42:13 +03:00
int rw , i ;
2017-03-17 09:18:50 +03:00
for ( rw = 0 ; rw < 2 ; rw + + ) {
2020-07-26 00:06:11 +03:00
pr_buf ( out , " %s: \n " , bch2_rw [ rw ] ) ;
2017-03-17 09:18:50 +03:00
2019-02-06 19:42:13 +03:00
for ( i = 1 ; i < BCH_DATA_NR ; i + + )
2020-07-26 00:06:11 +03:00
pr_buf ( out , " %-12s:%12llu \n " ,
2019-02-06 19:42:13 +03:00
bch2_data_types [ i ] ,
percpu_u64_get ( & ca - > io_done - > sectors [ rw ] [ i ] ) < < 9 ) ;
2017-03-17 09:18:50 +03:00
}
}
SHOW ( bch2_dev )
{
struct bch_dev * ca = container_of ( kobj , struct bch_dev , kobj ) ;
struct bch_fs * c = ca - > fs ;
2018-11-09 09:24:07 +03:00
struct printbuf out = _PBUF ( buf , PAGE_SIZE ) ;
2017-03-17 09:18:50 +03:00
sysfs_printf ( uuid , " %pU \n " , ca - > uuid . b ) ;
sysfs_print ( bucket_size , bucket_bytes ( ca ) ) ;
sysfs_print ( block_size , block_bytes ( c ) ) ;
sysfs_print ( first_bucket , ca - > mi . first_bucket ) ;
sysfs_print ( nbuckets , ca - > mi . nbuckets ) ;
sysfs_print ( durability , ca - > mi . durability ) ;
sysfs_print ( discard , ca - > mi . discard ) ;
if ( attr = = & sysfs_label ) {
if ( ca - > mi . group ) {
mutex_lock ( & c - > sb_lock ) ;
2018-11-09 09:24:07 +03:00
bch2_disk_path_to_text ( & out , & c - > disk_sb ,
ca - > mi . group - 1 ) ;
2017-03-17 09:18:50 +03:00
mutex_unlock ( & c - > sb_lock ) ;
}
2018-11-09 09:24:07 +03:00
pr_buf ( & out , " \n " ) ;
return out . pos - buf ;
2017-03-17 09:18:50 +03:00
}
if ( attr = = & sysfs_has_data ) {
2018-11-09 09:24:07 +03:00
bch2_flags_to_text ( & out , bch2_data_types ,
bch2_dev_has_data ( c , ca ) ) ;
pr_buf ( & out , " \n " ) ;
return out . pos - buf ;
2017-03-17 09:18:50 +03:00
}
if ( attr = = & sysfs_cache_replacement_policy ) {
2018-11-09 09:24:07 +03:00
bch2_string_opt_to_text ( & out ,
bch2_cache_replacement_policies ,
ca - > mi . replacement ) ;
pr_buf ( & out , " \n " ) ;
return out . pos - buf ;
2017-03-17 09:18:50 +03:00
}
if ( attr = = & sysfs_state_rw ) {
2018-11-09 09:24:07 +03:00
bch2_string_opt_to_text ( & out , bch2_dev_state ,
ca - > mi . state ) ;
pr_buf ( & out , " \n " ) ;
return out . pos - buf ;
2017-03-17 09:18:50 +03:00
}
2020-07-26 00:06:11 +03:00
if ( attr = = & sysfs_iodone ) {
dev_iodone_to_text ( & out , ca ) ;
return out . pos - buf ;
}
2017-03-17 09:18:50 +03:00
sysfs_print ( io_latency_read , atomic64_read ( & ca - > cur_latency [ READ ] ) ) ;
sysfs_print ( io_latency_write , atomic64_read ( & ca - > cur_latency [ WRITE ] ) ) ;
2020-07-26 00:06:11 +03:00
if ( attr = = & sysfs_io_latency_stats_read ) {
bch2_time_stats_to_text ( & out , & ca - > io_latency [ READ ] ) ;
return out . pos - buf ;
}
if ( attr = = & sysfs_io_latency_stats_write ) {
bch2_time_stats_to_text ( & out , & ca - > io_latency [ WRITE ] ) ;
return out . pos - buf ;
}
2017-03-17 09:18:50 +03:00
sysfs_printf ( congested , " %u%% " ,
clamp ( atomic_read ( & ca - > congested ) , 0 , CONGESTED_MAX )
* 100 / CONGESTED_MAX ) ;
if ( attr = = & sysfs_bucket_quantiles_last_read )
2020-07-26 00:06:11 +03:00
return quantiles_to_text ( & out , c , ca , bucket_last_io_fn , ( void * ) 0 ) ? : out . pos - buf ;
2017-03-17 09:18:50 +03:00
if ( attr = = & sysfs_bucket_quantiles_last_write )
2020-07-26 00:06:11 +03:00
return quantiles_to_text ( & out , c , ca , bucket_last_io_fn , ( void * ) 1 ) ? : out . pos - buf ;
2017-03-17 09:18:50 +03:00
if ( attr = = & sysfs_bucket_quantiles_fragmentation )
2020-07-26 00:06:11 +03:00
return quantiles_to_text ( & out , c , ca , bucket_sectors_used_fn , NULL ) ? : out . pos - buf ;
2017-03-17 09:18:50 +03:00
if ( attr = = & sysfs_bucket_quantiles_oldest_gen )
2020-07-26 00:06:11 +03:00
return quantiles_to_text ( & out , c , ca , bucket_oldest_gen_fn , NULL ) ? : out . pos - buf ;
2017-03-17 09:18:50 +03:00
2020-07-26 00:06:11 +03:00
if ( attr = = & sysfs_reserve_stats ) {
reserve_stats_to_text ( & out , ca ) ;
return out . pos - buf ;
}
if ( attr = = & sysfs_alloc_debug ) {
dev_alloc_debug_to_text ( & out , ca ) ;
return out . pos - buf ;
}
2017-03-17 09:18:50 +03:00
return 0 ;
}
STORE ( bch2_dev )
{
struct bch_dev * ca = container_of ( kobj , struct bch_dev , kobj ) ;
struct bch_fs * c = ca - > fs ;
struct bch_member * mi ;
if ( attr = = & sysfs_discard ) {
bool v = strtoul_or_return ( buf ) ;
mutex_lock ( & c - > sb_lock ) ;
mi = & bch2_sb_get_members ( c - > disk_sb . sb ) - > members [ ca - > dev_idx ] ;
if ( v ! = BCH_MEMBER_DISCARD ( mi ) ) {
SET_BCH_MEMBER_DISCARD ( mi , v ) ;
bch2_write_super ( c ) ;
}
mutex_unlock ( & c - > sb_lock ) ;
}
if ( attr = = & sysfs_cache_replacement_policy ) {
ssize_t v = __sysfs_match_string ( bch2_cache_replacement_policies , - 1 , buf ) ;
if ( v < 0 )
return v ;
mutex_lock ( & c - > sb_lock ) ;
mi = & bch2_sb_get_members ( c - > disk_sb . sb ) - > members [ ca - > dev_idx ] ;
if ( ( unsigned ) v ! = BCH_MEMBER_REPLACEMENT ( mi ) ) {
SET_BCH_MEMBER_REPLACEMENT ( mi , v ) ;
bch2_write_super ( c ) ;
}
mutex_unlock ( & c - > sb_lock ) ;
}
if ( attr = = & sysfs_label ) {
char * tmp ;
int ret ;
tmp = kstrdup ( buf , GFP_KERNEL ) ;
if ( ! tmp )
return - ENOMEM ;
ret = bch2_dev_group_set ( c , ca , strim ( tmp ) ) ;
kfree ( tmp ) ;
if ( ret )
return ret ;
}
if ( attr = = & sysfs_wake_allocator )
bch2_wake_allocator ( ca ) ;
return size ;
}
SYSFS_OPS ( bch2_dev ) ;
struct attribute * bch2_dev_files [ ] = {
& sysfs_uuid ,
& sysfs_bucket_size ,
& sysfs_block_size ,
& sysfs_first_bucket ,
& sysfs_nbuckets ,
& sysfs_durability ,
/* settings: */
& sysfs_discard ,
& sysfs_cache_replacement_policy ,
& sysfs_state_rw ,
& sysfs_label ,
& sysfs_has_data ,
& sysfs_iodone ,
& sysfs_io_latency_read ,
& sysfs_io_latency_write ,
& sysfs_io_latency_stats_read ,
& sysfs_io_latency_stats_write ,
& sysfs_congested ,
/* alloc info - other stats: */
& sysfs_bucket_quantiles_last_read ,
& sysfs_bucket_quantiles_last_write ,
& sysfs_bucket_quantiles_fragmentation ,
& sysfs_bucket_quantiles_oldest_gen ,
& sysfs_reserve_stats ,
/* debug: */
& sysfs_alloc_debug ,
& sysfs_wake_allocator ,
NULL
} ;
# endif /* _BCACHEFS_SYSFS_H_ */