2010-04-07 02:14:15 +04:00
# include <linux/ceph/ceph_debug.h>
# include <linux/device.h>
# include <linux/slab.h>
# include <linux/module.h>
# include <linux/ctype.h>
# include <linux/debugfs.h>
# include <linux/seq_file.h>
# include <linux/ceph/libceph.h>
# include <linux/ceph/mon_client.h>
# include <linux/ceph/auth.h>
# include <linux/ceph/debugfs.h>
# ifdef CONFIG_DEBUG_FS
/*
* Implement / sys / kernel / debug / ceph fun
*
* / sys / kernel / debug / ceph / client * - an instance of the ceph client
* . . . / osdmap - current osdmap
* . . . / monmap - current monmap
* . . . / osdc - active osd requests
* . . . / monc - mon client state
2015-03-25 21:07:41 +03:00
* . . . / client_options - libceph - only ( i . e . not rbd or cephfs ) options
2010-04-07 02:14:15 +04:00
* . . . / dentry_lru - dump contents of dentry lru
* . . . / caps - expose cap ( reservation ) stats
* . . . / bdi - symlink to . . / . . / bdi / something
*/
static struct dentry * ceph_debugfs_dir ;
static int monmap_show ( struct seq_file * s , void * p )
{
int i ;
struct ceph_client * client = s - > private ;
if ( client - > monc . monmap = = NULL )
return 0 ;
seq_printf ( s , " epoch %d \n " , client - > monc . monmap - > epoch ) ;
for ( i = 0 ; i < client - > monc . monmap - > num_mon ; i + + ) {
struct ceph_entity_inst * inst =
& client - > monc . monmap - > mon_inst [ i ] ;
seq_printf ( s , " \t %s%lld \t %s \n " ,
ENTITY_NAME ( inst - > name ) ,
ceph_pr_addr ( & inst - > addr . in_addr ) ) ;
}
return 0 ;
}
static int osdmap_show ( struct seq_file * s , void * p )
{
int i ;
struct ceph_client * client = s - > private ;
2014-03-13 18:36:12 +04:00
struct ceph_osdmap * map = client - > osdc . osdmap ;
2010-04-07 02:14:15 +04:00
struct rb_node * n ;
2014-03-13 18:36:12 +04:00
if ( map = = NULL )
2010-04-07 02:14:15 +04:00
return 0 ;
2014-03-13 18:36:12 +04:00
seq_printf ( s , " epoch %d \n " , map - > epoch ) ;
2010-04-07 02:14:15 +04:00
seq_printf ( s , " flags%s%s \n " ,
2014-03-13 18:36:12 +04:00
( map - > flags & CEPH_OSDMAP_NEARFULL ) ? " NEARFULL " : " " ,
( map - > flags & CEPH_OSDMAP_FULL ) ? " FULL " : " " ) ;
for ( n = rb_first ( & map - > pg_pools ) ; n ; n = rb_next ( n ) ) {
2010-04-07 02:14:15 +04:00
struct ceph_pg_pool_info * pool =
rb_entry ( n , struct ceph_pg_pool_info , node ) ;
2014-03-13 18:36:12 +04:00
2014-04-04 18:21:32 +04:00
seq_printf ( s , " pool %lld pg_num %u (%d) read_tier %lld write_tier %lld \n " ,
pool - > id , pool - > pg_num , pool - > pg_num_mask ,
pool - > read_tier , pool - > write_tier ) ;
2010-04-07 02:14:15 +04:00
}
2014-03-13 18:36:12 +04:00
for ( i = 0 ; i < map - > max_osd ; i + + ) {
struct ceph_entity_addr * addr = & map - > osd_addr [ i ] ;
int state = map - > osd_state [ i ] ;
2010-04-07 02:14:15 +04:00
char sb [ 64 ] ;
2014-03-21 21:05:30 +04:00
seq_printf ( s , " osd%d \t %s \t %3d%% \t (%s) \t %3d%% \n " ,
2010-04-07 02:14:15 +04:00
i , ceph_pr_addr ( & addr - > in_addr ) ,
2014-03-13 18:36:12 +04:00
( ( map - > osd_weight [ i ] * 100 ) > > 16 ) ,
2014-03-21 21:05:30 +04:00
ceph_osdmap_state_str ( sb , sizeof ( sb ) , state ) ,
( ( ceph_get_primary_affinity ( map , i ) * 100 ) > > 16 ) ) ;
2010-04-07 02:14:15 +04:00
}
2014-03-13 18:36:13 +04:00
for ( n = rb_first ( & map - > pg_temp ) ; n ; n = rb_next ( n ) ) {
struct ceph_pg_mapping * pg =
rb_entry ( n , struct ceph_pg_mapping , node ) ;
seq_printf ( s , " pg_temp %llu.%x [ " , pg - > pgid . pool ,
pg - > pgid . seed ) ;
2014-03-21 21:05:29 +04:00
for ( i = 0 ; i < pg - > pg_temp . len ; i + + )
2014-03-13 18:36:13 +04:00
seq_printf ( s , " %s%d " , ( i = = 0 ? " " : " , " ) ,
2014-03-21 21:05:29 +04:00
pg - > pg_temp . osds [ i ] ) ;
2014-03-13 18:36:13 +04:00
seq_printf ( s , " ] \n " ) ;
}
2014-03-21 21:05:29 +04:00
for ( n = rb_first ( & map - > primary_temp ) ; n ; n = rb_next ( n ) ) {
struct ceph_pg_mapping * pg =
rb_entry ( n , struct ceph_pg_mapping , node ) ;
seq_printf ( s , " primary_temp %llu.%x %d \n " , pg - > pgid . pool ,
pg - > pgid . seed , pg - > primary_temp . osd ) ;
}
2014-03-13 18:36:12 +04:00
2010-04-07 02:14:15 +04:00
return 0 ;
}
static int monc_show ( struct seq_file * s , void * p )
{
struct ceph_client * client = s - > private ;
struct ceph_mon_generic_request * req ;
struct ceph_mon_client * monc = & client - > monc ;
struct rb_node * rp ;
2016-01-19 18:19:06 +03:00
int i ;
2010-04-07 02:14:15 +04:00
mutex_lock ( & monc - > mutex ) ;
2016-01-19 18:19:06 +03:00
for ( i = 0 ; i < ARRAY_SIZE ( monc - > subs ) ; i + + ) {
seq_printf ( s , " have %s %u " , ceph_sub_str [ i ] ,
monc - > subs [ i ] . have ) ;
if ( monc - > subs [ i ] . want )
seq_printf ( s , " want %llu%s " ,
le64_to_cpu ( monc - > subs [ i ] . item . start ) ,
( monc - > subs [ i ] . item . flags &
CEPH_SUBSCRIBE_ONETIME ? " " : " + " ) ) ;
seq_putc ( s , ' \n ' ) ;
}
2010-04-07 02:14:15 +04:00
for ( rp = rb_first ( & monc - > generic_request_tree ) ; rp ; rp = rb_next ( rp ) ) {
__u16 op ;
req = rb_entry ( rp , struct ceph_mon_generic_request , node ) ;
op = le16_to_cpu ( req - > request - > hdr . type ) ;
if ( op = = CEPH_MSG_STATFS )
2014-05-12 13:05:47 +04:00
seq_printf ( s , " %llu statfs \n " , req - > tid ) ;
2014-05-13 11:19:26 +04:00
else if ( op = = CEPH_MSG_MON_GET_VERSION )
seq_printf ( s , " %llu mon_get_version " , req - > tid ) ;
2010-04-07 02:14:15 +04:00
else
2014-05-12 13:05:47 +04:00
seq_printf ( s , " %llu unknown \n " , req - > tid ) ;
2010-04-07 02:14:15 +04:00
}
mutex_unlock ( & monc - > mutex ) ;
return 0 ;
}
static int osdc_show ( struct seq_file * s , void * pp )
{
struct ceph_client * client = s - > private ;
struct ceph_osd_client * osdc = & client - > osdc ;
struct rb_node * p ;
mutex_lock ( & osdc - > request_mutex ) ;
for ( p = rb_first ( & osdc - > requests ) ; p ; p = rb_next ( p ) ) {
struct ceph_osd_request * req ;
2013-04-04 06:32:51 +04:00
unsigned int i ;
2013-02-26 04:11:12 +04:00
int opcode ;
2010-04-07 02:14:15 +04:00
req = rb_entry ( p , struct ceph_osd_request , r_node ) ;
2013-02-23 22:38:16 +04:00
seq_printf ( s , " %lld \t osd%d \t %lld.%x \t " , req - > r_tid ,
2010-04-07 02:14:15 +04:00
req - > r_osd ? req - > r_osd - > o_osd : - 1 ,
2013-02-23 22:38:16 +04:00
req - > r_pgid . pool , req - > r_pgid . seed ) ;
2010-04-07 02:14:15 +04:00
2016-04-29 20:54:20 +03:00
seq_printf ( s , " %*pE " , req - > r_base_oid . name_len ,
2014-01-27 19:40:20 +04:00
req - > r_base_oid . name ) ;
2010-04-07 02:14:15 +04:00
if ( req - > r_reassert_version . epoch )
seq_printf ( s , " \t %u'%llu " ,
2012-04-15 09:58:06 +04:00
( unsigned int ) le32_to_cpu ( req - > r_reassert_version . epoch ) ,
2010-04-07 02:14:15 +04:00
le64_to_cpu ( req - > r_reassert_version . version ) ) ;
else
seq_printf ( s , " \t " ) ;
2013-02-26 04:11:12 +04:00
for ( i = 0 ; i < req - > r_num_ops ; i + + ) {
2013-04-04 06:32:51 +04:00
opcode = req - > r_ops [ i ] . op ;
2014-10-06 18:40:27 +04:00
seq_printf ( s , " %s%s " , ( i = = 0 ? " \t " : " , " ) ,
ceph_osd_op_name ( opcode ) ) ;
2010-04-07 02:14:15 +04:00
}
seq_printf ( s , " \n " ) ;
}
mutex_unlock ( & osdc - > request_mutex ) ;
return 0 ;
}
2015-03-25 21:07:41 +03:00
static int client_options_show ( struct seq_file * s , void * p )
{
struct ceph_client * client = s - > private ;
int ret ;
ret = ceph_print_client_options ( s , client ) ;
if ( ret )
return ret ;
seq_putc ( s , ' \n ' ) ;
return 0 ;
}
2010-04-07 02:14:15 +04:00
CEPH_DEFINE_SHOW_FUNC ( monmap_show )
CEPH_DEFINE_SHOW_FUNC ( osdmap_show )
CEPH_DEFINE_SHOW_FUNC ( monc_show )
CEPH_DEFINE_SHOW_FUNC ( osdc_show )
2015-03-25 21:07:41 +03:00
CEPH_DEFINE_SHOW_FUNC ( client_options_show )
2010-04-07 02:14:15 +04:00
int ceph_debugfs_init ( void )
{
ceph_debugfs_dir = debugfs_create_dir ( " ceph " , NULL ) ;
if ( ! ceph_debugfs_dir )
return - ENOMEM ;
return 0 ;
}
void ceph_debugfs_cleanup ( void )
{
debugfs_remove ( ceph_debugfs_dir ) ;
}
int ceph_debugfs_client_init ( struct ceph_client * client )
{
int ret = - ENOMEM ;
char name [ 80 ] ;
snprintf ( name , sizeof ( name ) , " %pU.client%lld " , & client - > fsid ,
client - > monc . auth - > global_id ) ;
2012-08-19 23:29:16 +04:00
dout ( " ceph_debugfs_client_init %p %s \n " , client , name ) ;
BUG_ON ( client - > debugfs_dir ) ;
2010-04-07 02:14:15 +04:00
client - > debugfs_dir = debugfs_create_dir ( name , ceph_debugfs_dir ) ;
if ( ! client - > debugfs_dir )
goto out ;
client - > monc . debugfs_file = debugfs_create_file ( " monc " ,
0600 ,
client - > debugfs_dir ,
client ,
& monc_show_fops ) ;
if ( ! client - > monc . debugfs_file )
goto out ;
client - > osdc . debugfs_file = debugfs_create_file ( " osdc " ,
0600 ,
client - > debugfs_dir ,
client ,
& osdc_show_fops ) ;
if ( ! client - > osdc . debugfs_file )
goto out ;
client - > debugfs_monmap = debugfs_create_file ( " monmap " ,
0600 ,
client - > debugfs_dir ,
client ,
& monmap_show_fops ) ;
if ( ! client - > debugfs_monmap )
goto out ;
client - > debugfs_osdmap = debugfs_create_file ( " osdmap " ,
0600 ,
client - > debugfs_dir ,
client ,
& osdmap_show_fops ) ;
if ( ! client - > debugfs_osdmap )
goto out ;
2015-03-25 21:07:41 +03:00
client - > debugfs_options = debugfs_create_file ( " client_options " ,
0600 ,
client - > debugfs_dir ,
client ,
& client_options_show_fops ) ;
if ( ! client - > debugfs_options )
goto out ;
2010-04-07 02:14:15 +04:00
return 0 ;
out :
ceph_debugfs_client_cleanup ( client ) ;
return ret ;
}
void ceph_debugfs_client_cleanup ( struct ceph_client * client )
{
2012-08-19 23:29:16 +04:00
dout ( " ceph_debugfs_client_cleanup %p \n " , client ) ;
2015-03-25 21:07:41 +03:00
debugfs_remove ( client - > debugfs_options ) ;
2010-04-07 02:14:15 +04:00
debugfs_remove ( client - > debugfs_osdmap ) ;
debugfs_remove ( client - > debugfs_monmap ) ;
debugfs_remove ( client - > osdc . debugfs_file ) ;
debugfs_remove ( client - > monc . debugfs_file ) ;
debugfs_remove ( client - > debugfs_dir ) ;
}
# else /* CONFIG_DEBUG_FS */
int ceph_debugfs_init ( void )
{
return 0 ;
}
void ceph_debugfs_cleanup ( void )
{
}
2010-08-13 03:11:25 +04:00
int ceph_debugfs_client_init ( struct ceph_client * client )
2010-04-07 02:14:15 +04:00
{
return 0 ;
}
void ceph_debugfs_client_cleanup ( struct ceph_client * client )
{
}
# endif /* CONFIG_DEBUG_FS */
EXPORT_SYMBOL ( ceph_debugfs_init ) ;
EXPORT_SYMBOL ( ceph_debugfs_cleanup ) ;