2017-03-16 22:18:50 -08: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 00:46:55 -04:00
# include "alloc_background.h"
2017-03-16 22:18:50 -08:00
# include "sysfs.h"
# include "btree_cache.h"
# include "btree_io.h"
# include "btree_iter.h"
# include "btree_update.h"
# include "btree_update_interior.h"
# include "btree_gc.h"
# include "buckets.h"
# include "disk_groups.h"
2018-11-01 15:13:19 -04:00
# include "ec.h"
2017-03-16 22:18:50 -08: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 12:58:56 -05:00
struct printbuf out = _PBUF ( buf , PAGE_SIZE ) ; \
bch2_hprint ( & out , val ) ; \
pr_buf ( & out , " \n " ) ; \
return out . pos - buf ; \
2017-03-16 22:18:50 -08: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 ) ;
2019-01-24 18:32:13 -05:00
write_attribute ( trigger_alloc_write ) ;
2017-03-16 22:18:50 -08:00
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 ) ;
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 15:13:19 -04:00
read_attribute ( new_stripes ) ;
2017-03-16 22:18:50 -08:00
rw_attribute ( pd_controllers_update_seconds ) ;
read_attribute ( meta_replicas_have ) ;
read_attribute ( data_replicas_have ) ;
# ifdef CONFIG_BCACHEFS_TESTS
write_attribute ( perf_test ) ;
# endif /* CONFIG_BCACHEFS_TESTS */
# define BCH_DEBUG_PARAM(name, description) \
rw_attribute ( name ) ;
BCH_DEBUG_PARAMS ( )
# undef BCH_DEBUG_PARAM
# 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 ;
}
static ssize_t show_fs_alloc_debug ( struct bch_fs * c , char * buf )
{
2018-11-09 01:24:07 -05:00
struct printbuf out = _PBUF ( buf , PAGE_SIZE ) ;
2019-01-21 15:32:13 -05:00
struct bch_fs_usage * fs_usage = bch2_fs_usage_read ( c ) ;
unsigned i ;
2018-07-24 13:33:07 -04:00
2019-01-21 15:32:13 -05:00
if ( ! fs_usage )
return - ENOMEM ;
2018-07-24 13:33:07 -04:00
2019-01-24 19:50:00 -05:00
pr_buf ( & out , " capacity: \t \t \t %llu \n " , c - > capacity ) ;
pr_buf ( & out , " hidden: \t \t \t \t %llu \n " ,
2019-02-14 18:38:52 -05:00
fs_usage - > hidden ) ;
2019-01-24 19:50:00 -05:00
pr_buf ( & out , " data: \t \t \t \t %llu \n " ,
2019-02-14 18:38:52 -05:00
fs_usage - > data ) ;
2019-01-24 19:50:00 -05:00
pr_buf ( & out , " cached: \t \t \t \t %llu \n " ,
2019-02-14 18:38:52 -05:00
fs_usage - > cached ) ;
2019-01-24 19:50:00 -05:00
pr_buf ( & out , " reserved: \t \t \t %llu \n " ,
2019-02-14 18:38:52 -05:00
fs_usage - > reserved ) ;
2019-01-24 19:50:00 -05:00
pr_buf ( & out , " nr_inodes: \t \t \t %llu \n " ,
2019-02-14 18:38:52 -05:00
fs_usage - > nr_inodes ) ;
2019-01-24 19:50:00 -05:00
pr_buf ( & out , " online reserved: \t \t %llu \n " ,
2019-02-14 18:38:52 -05:00
fs_usage - > online_reserved ) ;
2018-07-24 13:33:07 -04:00
2019-01-21 15:32:13 -05:00
for ( i = 0 ;
i < ARRAY_SIZE ( fs_usage - > persistent_reserved ) ;
i + + ) {
pr_buf ( & out , " %u replicas: \n " , i + 1 ) ;
2019-01-24 19:50:00 -05:00
pr_buf ( & out , " \t reserved: \t \t %llu \n " ,
2019-01-21 15:32:13 -05:00
fs_usage - > persistent_reserved [ i ] ) ;
2018-07-24 13:33:07 -04:00
}
2019-01-21 15:32:13 -05:00
for ( i = 0 ; i < c - > replicas . nr ; i + + ) {
struct bch_replicas_entry * e =
cpu_replicas_entry ( & c - > replicas , i ) ;
2018-07-24 14:54:39 -04:00
2019-01-21 15:32:13 -05:00
pr_buf ( & out , " \t " ) ;
bch2_replicas_entry_to_text ( & out , e ) ;
2019-02-14 18:38:52 -05:00
pr_buf ( & out , " : \t %llu \n " , fs_usage - > replicas [ i ] ) ;
2019-01-21 15:32:13 -05:00
}
percpu_up_read ( & c - > mark_lock ) ;
kfree ( fs_usage ) ;
2018-07-24 13:33:07 -04:00
2018-11-09 01:24:07 -05:00
return out . pos - buf ;
2017-03-16 22:18:50 -08:00
}
static ssize_t bch2_compression_stats ( struct bch_fs * c , char * buf )
{
2019-03-25 15:10:15 -04:00
struct btree_trans trans ;
struct btree_iter * iter ;
2017-03-16 22:18:50 -08: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-03-21 22:19:57 -04:00
if ( ! test_bit ( BCH_FS_STARTED , & c - > flags ) )
2017-03-16 22:18:50 -08:00
return - EPERM ;
2019-03-25 15:10:15 -04:00
bch2_trans_init ( & trans , c ) ;
for_each_btree_key ( & trans , iter , BTREE_ID_EXTENTS , POS_MIN , 0 , k )
2018-11-01 15:10:01 -04:00
if ( k . k - > type = = KEY_TYPE_extent ) {
2017-03-16 22:18:50 -08:00
struct bkey_s_c_extent e = bkey_s_c_to_extent ( k ) ;
2018-09-27 21:08:39 -04:00
const union bch_extent_entry * entry ;
struct extent_ptr_decoded p ;
2017-03-16 22:18:50 -08:00
2018-09-27 21:08:39 -04:00
extent_for_each_ptr_decode ( e , p , entry ) {
if ( p . crc . compression_type = = BCH_COMPRESSION_NONE ) {
2017-03-16 22:18:50 -08:00
nr_uncompressed_extents + + ;
uncompressed_sectors + = e . k - > size ;
} else {
nr_compressed_extents + + ;
compressed_sectors_compressed + =
2018-09-27 21:08:39 -04:00
p . crc . compressed_size ;
2017-03-16 22:18:50 -08:00
compressed_sectors_uncompressed + =
2018-09-27 21:08:39 -04:00
p . crc . uncompressed_size ;
2017-03-16 22:18:50 -08:00
}
/* only looking at the first ptr */
break ;
}
}
2019-03-25 15:10:15 -04:00
bch2_trans_exit ( & trans ) ;
2017-03-16 22:18:50 -08:00
return scnprintf ( buf , PAGE_SIZE ,
" 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 ) ;
}
2018-11-01 15:13:19 -04:00
static ssize_t bch2_new_stripes ( struct bch_fs * c , char * buf )
{
char * out = buf , * end = buf + PAGE_SIZE ;
struct ec_stripe_head * h ;
struct ec_stripe_new * s ;
mutex_lock ( & c - > ec_new_stripe_lock ) ;
list_for_each_entry ( h , & c - > ec_new_stripe_list , list ) {
out + = scnprintf ( out , end - out ,
" target %u algo %u redundancy %u: \n " ,
h - > target , h - > algo , h - > redundancy ) ;
if ( h - > s )
out + = scnprintf ( out , end - out ,
" \t pending: blocks %u allocated %u \n " ,
h - > s - > blocks . nr ,
bitmap_weight ( h - > s - > blocks_allocated ,
h - > s - > blocks . nr ) ) ;
mutex_lock ( & h - > lock ) ;
list_for_each_entry ( s , & h - > stripes , list )
out + = scnprintf ( out , end - out ,
" \t in flight: blocks %u allocated %u pin %u \n " ,
s - > blocks . nr ,
bitmap_weight ( s - > blocks_allocated ,
s - > blocks . nr ) ,
atomic_read ( & s - > pin ) ) ;
mutex_unlock ( & h - > lock ) ;
}
mutex_unlock ( & c - > ec_new_stripe_lock ) ;
return out - buf ;
}
2017-03-16 22:18:50 -08:00
SHOW ( bch2_fs )
{
struct bch_fs * c = container_of ( kobj , struct bch_fs , kobj ) ;
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 */
if ( attr = = & sysfs_rebalance_work )
return bch2_rebalance_work_show ( c , buf ) ;
sysfs_print ( promote_whole_extents , c - > promote_whole_extents ) ;
2018-10-30 14:32:47 -04: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-16 22:18:50 -08:00
/* Debugging: */
if ( attr = = & sysfs_alloc_debug )
return show_fs_alloc_debug ( c , buf ) ;
if ( attr = = & sysfs_journal_debug )
return bch2_journal_print_debug ( & c - > journal , buf ) ;
if ( attr = = & sysfs_journal_pins )
return bch2_journal_print_pins ( & c - > journal , buf ) ;
if ( attr = = & sysfs_btree_updates )
return bch2_btree_updates_print ( c , buf ) ;
if ( attr = = & sysfs_dirty_btree_nodes )
return bch2_dirty_btree_nodes_print ( c , buf ) ;
if ( attr = = & sysfs_compression_stats )
return bch2_compression_stats ( c , buf ) ;
2018-11-01 15:13:19 -04:00
if ( attr = = & sysfs_new_stripes )
return bch2_new_stripes ( c , buf ) ;
2017-03-16 22:18:50 -08:00
# define BCH_DEBUG_PARAM(name, description) sysfs_print(name, c->name);
BCH_DEBUG_PARAMS ( )
# undef BCH_DEBUG_PARAM
return 0 ;
}
STORE ( __bch2_fs )
{
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 ) {
struct bch_dev * ca ;
unsigned i ;
ssize_t ret = strtoul_safe ( buf , c - > copy_gc_enabled )
? : ( ssize_t ) size ;
for_each_member_device ( ca , c , i )
if ( ca - > copygc_thread )
wake_up_process ( ca - > copygc_thread ) ;
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 ) ;
sysfs_strtoul ( promote_whole_extents , c - > promote_whole_extents ) ;
/* Debugging: */
# define BCH_DEBUG_PARAM(name, description) sysfs_strtoul(name, c->name);
BCH_DEBUG_PARAMS ( )
# undef BCH_DEBUG_PARAM
2019-03-21 22:19:57 -04:00
if ( ! test_bit ( BCH_FS_STARTED , & c - > flags ) )
2017-03-16 22:18:50 -08:00
return - EPERM ;
/* Debugging: */
if ( attr = = & sysfs_trigger_journal_flush )
bch2_journal_meta_async ( & c - > journal , NULL ) ;
if ( attr = = & sysfs_trigger_btree_coalesce )
bch2_coalesce ( c ) ;
if ( attr = = & sysfs_trigger_gc )
2019-03-29 22:22:45 -04:00
bch2_gc ( c , NULL , false , false ) ;
2017-03-16 22:18:50 -08:00
2019-01-24 18:32:13 -05:00
if ( attr = = & sysfs_trigger_alloc_write ) {
bool wrote ;
bch2_alloc_write ( c , false , & wrote ) ;
}
2017-03-16 22:18:50 -08: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 ) ;
}
# 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 ) ) )
bch2_btree_perf_test ( c , test , nr , threads ) ;
else
size = ret ;
kfree ( tmp ) ;
}
# endif
return size ;
}
STORE ( bch2_fs )
{
struct bch_fs * c = container_of ( kobj , struct bch_fs , kobj ) ;
mutex_lock ( & c - > state_lock ) ;
size = __bch2_fs_store ( kobj , attr , buf , size ) ;
mutex_unlock ( & c - > state_lock ) ;
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 ,
& sysfs_read_realloc_races ,
& sysfs_extent_migrate_done ,
& sysfs_extent_migrate_raced ,
& sysfs_trigger_journal_flush ,
& sysfs_trigger_btree_coalesce ,
& sysfs_trigger_gc ,
2019-01-24 18:32:13 -05:00
& sysfs_trigger_alloc_write ,
2017-03-16 22:18:50 -08:00
& sysfs_prune_cache ,
& sysfs_copy_gc_enabled ,
& sysfs_rebalance_enabled ,
& sysfs_rebalance_work ,
sysfs_pd_controller_files ( rebalance ) ,
2018-11-01 15:13:19 -04:00
& sysfs_new_stripes ,
2017-03-16 22:18:50 -08:00
& sysfs_internal_uuid ,
# define BCH_DEBUG_PARAM(name, description) &sysfs_##name,
BCH_DEBUG_PARAMS ( )
# undef BCH_DEBUG_PARAM
NULL
} ;
/* options */
SHOW ( bch2_fs_opts_dir )
{
2018-11-09 01:24:07 -05:00
struct printbuf out = _PBUF ( buf , PAGE_SIZE ) ;
2017-03-16 22:18:50 -08: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 01:24:07 -05:00
bch2_opt_to_text ( & out , c , opt , v , OPT_SHOW_FULL_LIST ) ;
pr_buf ( & out , " \n " ) ;
2017-03-16 22:18:50 -08:00
2018-11-09 01:24:07 -05:00
return out . pos - buf ;
2017-03-16 22:18:50 -08: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-12 18:30:55 -05:00
ret = bch2_opt_check_may_set ( c , id , v ) ;
if ( ret < 0 )
return ret ;
2017-03-16 22:18:50 -08: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 12:58:56 -05:00
if ( ! ( i - > mode & ( OPT_FORMAT | OPT_MOUNT | OPT_RUNTIME ) ) )
2017-03-16 22:18:50 -08: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 ) ;
# define x(name) \
if ( attr = = & sysfs_time_stat_ # # name ) \
return bch2_time_stats_print ( & c - > times [ BCH_TIME_ # # name ] , \
buf , PAGE_SIZE ) ;
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 )
{
unsigned l = * ( ( unsigned * ) _l ) ;
unsigned r = * ( ( unsigned * ) _r ) ;
return ( l > r ) - ( l < r ) ;
}
static ssize_t show_quantiles ( struct bch_fs * c , struct bch_dev * ca ,
char * buf , bucket_map_fn * fn , void * private )
{
size_t i , n ;
/* Compute 31 quantiles */
unsigned q [ 31 ] , * p ;
ssize_t ret = 0 ;
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 + + )
ret + = scnprintf ( buf + ret , PAGE_SIZE - ret ,
" %u " , q [ i ] ) ;
buf [ ret - 1 ] = ' \n ' ;
return ret ;
}
static ssize_t show_reserve_stats ( struct bch_dev * ca , char * buf )
{
2018-11-09 01:24:07 -05:00
struct printbuf out = _PBUF ( buf , PAGE_SIZE ) ;
2017-03-16 22:18:50 -08:00
enum alloc_reserve i ;
spin_lock ( & ca - > freelist_lock ) ;
2018-11-09 01:24:07 -05:00
pr_buf ( & out , " free_inc: \t %zu \t %zu \n " ,
fifo_used ( & ca - > free_inc ) ,
ca - > free_inc . size ) ;
2017-03-16 22:18:50 -08:00
for ( i = 0 ; i < RESERVE_NR ; i + + )
2018-11-09 01:24:07 -05:00
pr_buf ( & out , " free[%u]: \t %zu \t %zu \n " , i ,
fifo_used ( & ca - > free [ i ] ) ,
ca - > free [ i ] . size ) ;
2017-03-16 22:18:50 -08:00
spin_unlock ( & ca - > freelist_lock ) ;
2018-11-09 01:24:07 -05:00
return out . pos - buf ;
2017-03-16 22:18:50 -08:00
}
static ssize_t show_dev_alloc_debug ( struct bch_dev * ca , char * buf )
{
struct bch_fs * c = ca - > fs ;
struct bch_dev_usage stats = bch2_dev_usage_read ( c , ca ) ;
2019-01-19 13:13:29 -05: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-16 22:18:50 -08:00
return scnprintf ( buf , PAGE_SIZE ,
" free_inc: %zu/%zu \n "
" free[RESERVE_BTREE]: %zu/%zu \n "
" free[RESERVE_MOVINGGC]: %zu/%zu \n "
" free[RESERVE_NONE]: %zu/%zu \n "
" buckets: \n "
" capacity: %llu \n "
" alloc: %llu \n "
" sb: %llu \n "
" journal: %llu \n "
" meta: %llu \n "
" user: %llu \n "
" cached: %llu \n "
2018-11-01 15:13:19 -04:00
" erasure coded: %llu \n "
2018-07-24 16:42:49 -04:00
" available: %lli \n "
2017-03-16 22:18:50 -08:00
" sectors: \n "
" sb: %llu \n "
" journal: %llu \n "
" meta: %llu \n "
" user: %llu \n "
" cached: %llu \n "
2018-08-01 14:26:55 -04:00
" fragmented: %llu \n "
" copygc threshold: %llu \n "
2017-03-16 22:18:50 -08:00
" freelist_wait: %s \n "
" open buckets: %u/%u (reserved %u) \n "
2019-01-19 13:13:29 -05:00
" open_buckets_wait: %s \n "
" open_buckets_btree: %u \n "
" open_buckets_user: %u \n "
" btree reserve cache: %u \n " ,
2017-03-16 22:18:50 -08:00
fifo_used ( & ca - > free_inc ) , ca - > free_inc . size ,
fifo_used ( & ca - > free [ RESERVE_BTREE ] ) , ca - > free [ RESERVE_BTREE ] . 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 ,
stats . buckets_alloc ,
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 15:13:19 -04:00
stats . buckets_ec ,
2018-07-24 16:42:49 -04:00
ca - > mi . nbuckets - ca - > mi . first_bucket - stats . buckets_unavailable ,
2017-03-16 22:18:50 -08: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 ] ,
2018-08-01 14:26:55 -04:00
stats . sectors_fragmented ,
ca - > copygc_threshold ,
2017-03-16 22:18:50 -08:00
c - > freelist_wait . list . first ? " waiting " : " empty " ,
2019-01-19 13:13:29 -05:00
c - > open_buckets_nr_free , OPEN_BUCKETS_COUNT ,
BTREE_NODE_OPEN_BUCKET_RESERVE ,
c - > open_buckets_wait . list . first ? " waiting " : " empty " ,
nr [ BCH_DATA_BTREE ] ,
nr [ BCH_DATA_USER ] ,
c - > btree_reserve_cache_nr ) ;
2017-03-16 22:18:50 -08:00
}
static const char * const bch2_rw [ ] = {
" read " ,
" write " ,
NULL
} ;
static ssize_t show_dev_iodone ( struct bch_dev * ca , char * buf )
{
2018-11-09 01:24:07 -05:00
struct printbuf out = _PBUF ( buf , PAGE_SIZE ) ;
2019-02-06 11:42:13 -05:00
int rw , i ;
2017-03-16 22:18:50 -08:00
for ( rw = 0 ; rw < 2 ; rw + + ) {
2018-11-09 01:24:07 -05:00
pr_buf ( & out , " %s: \n " , bch2_rw [ rw ] ) ;
2017-03-16 22:18:50 -08:00
2019-02-06 11:42:13 -05:00
for ( i = 1 ; i < BCH_DATA_NR ; i + + )
2018-11-09 01:24:07 -05:00
pr_buf ( & out , " %-12s:%12llu \n " ,
2019-02-06 11:42:13 -05:00
bch2_data_types [ i ] ,
percpu_u64_get ( & ca - > io_done - > sectors [ rw ] [ i ] ) < < 9 ) ;
2017-03-16 22:18:50 -08:00
}
2018-11-09 01:24:07 -05:00
return out . pos - buf ;
2017-03-16 22:18:50 -08:00
}
SHOW ( bch2_dev )
{
struct bch_dev * ca = container_of ( kobj , struct bch_dev , kobj ) ;
struct bch_fs * c = ca - > fs ;
2018-11-09 01:24:07 -05:00
struct printbuf out = _PBUF ( buf , PAGE_SIZE ) ;
2017-03-16 22:18:50 -08: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 01:24:07 -05:00
bch2_disk_path_to_text ( & out , & c - > disk_sb ,
ca - > mi . group - 1 ) ;
2017-03-16 22:18:50 -08:00
mutex_unlock ( & c - > sb_lock ) ;
} else {
2018-11-09 01:24:07 -05:00
pr_buf ( & out , " none " ) ;
2017-03-16 22:18:50 -08:00
}
2018-11-09 01:24:07 -05:00
pr_buf ( & out , " \n " ) ;
return out . pos - buf ;
2017-03-16 22:18:50 -08:00
}
if ( attr = = & sysfs_has_data ) {
2018-11-09 01:24:07 -05: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-16 22:18:50 -08:00
}
sysfs_pd_controller_show ( copy_gc , & ca - > copygc_pd ) ;
if ( attr = = & sysfs_cache_replacement_policy ) {
2018-11-09 01:24:07 -05:00
bch2_string_opt_to_text ( & out ,
bch2_cache_replacement_policies ,
ca - > mi . replacement ) ;
pr_buf ( & out , " \n " ) ;
return out . pos - buf ;
2017-03-16 22:18:50 -08:00
}
if ( attr = = & sysfs_state_rw ) {
2018-11-09 01:24:07 -05:00
bch2_string_opt_to_text ( & out , bch2_dev_state ,
ca - > mi . state ) ;
pr_buf ( & out , " \n " ) ;
return out . pos - buf ;
2017-03-16 22:18:50 -08:00
}
if ( attr = = & sysfs_iodone )
return show_dev_iodone ( ca , buf ) ;
sysfs_print ( io_latency_read , atomic64_read ( & ca - > cur_latency [ READ ] ) ) ;
sysfs_print ( io_latency_write , atomic64_read ( & ca - > cur_latency [ WRITE ] ) ) ;
if ( attr = = & sysfs_io_latency_stats_read )
return bch2_time_stats_print ( & ca - > io_latency [ READ ] , buf , PAGE_SIZE ) ;
if ( attr = = & sysfs_io_latency_stats_write )
return bch2_time_stats_print ( & ca - > io_latency [ WRITE ] , buf , PAGE_SIZE ) ;
sysfs_printf ( congested , " %u%% " ,
clamp ( atomic_read ( & ca - > congested ) , 0 , CONGESTED_MAX )
* 100 / CONGESTED_MAX ) ;
if ( attr = = & sysfs_bucket_quantiles_last_read )
return show_quantiles ( c , ca , buf , bucket_last_io_fn , ( void * ) 0 ) ;
if ( attr = = & sysfs_bucket_quantiles_last_write )
return show_quantiles ( c , ca , buf , bucket_last_io_fn , ( void * ) 1 ) ;
if ( attr = = & sysfs_bucket_quantiles_fragmentation )
return show_quantiles ( c , ca , buf , bucket_sectors_used_fn , NULL ) ;
if ( attr = = & sysfs_bucket_quantiles_oldest_gen )
return show_quantiles ( c , ca , buf , bucket_oldest_gen_fn , NULL ) ;
if ( attr = = & sysfs_reserve_stats )
return show_reserve_stats ( ca , buf ) ;
if ( attr = = & sysfs_alloc_debug )
return show_dev_alloc_debug ( ca , buf ) ;
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 ;
sysfs_pd_controller_store ( copy_gc , & ca - > copygc_pd ) ;
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 ,
sysfs_pd_controller_files ( copy_gc ) ,
NULL
} ;
# endif /* _BCACHEFS_SYSFS_H_ */