2018-06-06 05:42:14 +03:00
// SPDX-License-Identifier: GPL-2.0
2005-04-17 02:20:36 +04:00
/*
2005-11-02 06:58:39 +03:00
* Copyright ( c ) 2000 - 2003 , 2005 Silicon Graphics , Inc .
* All Rights Reserved .
2005-04-17 02:20:36 +04:00
*/
# include "xfs.h"
# include <linux/proc_fs.h>
2015-10-11 21:19:45 +03:00
struct xstats xfsstats ;
2005-04-17 02:20:36 +04:00
2015-10-11 21:19:45 +03:00
static int counter_val ( struct xfsstats __percpu * stats , int idx )
2012-03-13 12:52:33 +04:00
{
int val = 0 , cpu ;
for_each_possible_cpu ( cpu )
2015-10-11 21:19:45 +03:00
val + = * ( ( ( __u32 * ) per_cpu_ptr ( stats , cpu ) + idx ) ) ;
2012-03-13 12:52:33 +04:00
return val ;
}
2015-10-11 21:19:45 +03:00
int xfs_stats_format ( struct xfsstats __percpu * stats , char * buf )
2015-10-11 21:15:45 +03:00
{
int i , j ;
int len = 0 ;
2017-06-16 21:00:05 +03:00
uint64_t xs_xstrat_bytes = 0 ;
uint64_t xs_write_bytes = 0 ;
uint64_t xs_read_bytes = 0 ;
2015-10-11 21:15:45 +03:00
static const struct xstats_entry {
char * desc ;
int endpoint ;
} xstats [ ] = {
{ " extent_alloc " , XFSSTAT_END_EXTENT_ALLOC } ,
{ " abt " , XFSSTAT_END_ALLOC_BTREE } ,
{ " blk_map " , XFSSTAT_END_BLOCK_MAPPING } ,
{ " bmbt " , XFSSTAT_END_BLOCK_MAP_BTREE } ,
{ " dir " , XFSSTAT_END_DIRECTORY_OPS } ,
{ " trans " , XFSSTAT_END_TRANSACTIONS } ,
{ " ig " , XFSSTAT_END_INODE_OPS } ,
{ " log " , XFSSTAT_END_LOG_OPS } ,
{ " push_ail " , XFSSTAT_END_TAIL_PUSHING } ,
{ " xstrat " , XFSSTAT_END_WRITE_CONVERT } ,
{ " rw " , XFSSTAT_END_READ_WRITE_OPS } ,
{ " attr " , XFSSTAT_END_ATTRIBUTE_OPS } ,
{ " icluster " , XFSSTAT_END_INODE_CLUSTER } ,
{ " vnodes " , XFSSTAT_END_VNODE_OPS } ,
{ " buf " , XFSSTAT_END_BUF } ,
{ " abtb2 " , XFSSTAT_END_ABTB_V2 } ,
{ " abtc2 " , XFSSTAT_END_ABTC_V2 } ,
{ " bmbt2 " , XFSSTAT_END_BMBT_V2 } ,
{ " ibt2 " , XFSSTAT_END_IBT_V2 } ,
{ " fibt2 " , XFSSTAT_END_FIBT_V2 } ,
2016-08-03 04:31:11 +03:00
{ " rmapbt " , XFSSTAT_END_RMAP_V2 } ,
2016-10-03 19:11:16 +03:00
{ " refcntbt " , XFSSTAT_END_REFCOUNT } ,
2015-10-11 21:15:45 +03:00
/* we print both series of quota information together */
{ " qm " , XFSSTAT_END_QM } ,
} ;
/* Loop over all stats groups */
for ( i = j = 0 ; i < ARRAY_SIZE ( xstats ) ; i + + ) {
len + = snprintf ( buf + len , PATH_MAX - len , " %s " ,
xstats [ i ] . desc ) ;
/* inner loop does each group */
for ( ; j < xstats [ i ] . endpoint ; j + + )
len + = snprintf ( buf + len , PATH_MAX - len , " %u " ,
2015-10-11 21:19:45 +03:00
counter_val ( stats , j ) ) ;
2015-10-11 21:15:45 +03:00
len + = snprintf ( buf + len , PATH_MAX - len , " \n " ) ;
}
/* extra precision counters */
for_each_possible_cpu ( i ) {
2016-12-05 06:38:58 +03:00
xs_xstrat_bytes + = per_cpu_ptr ( stats , i ) - > s . xs_xstrat_bytes ;
xs_write_bytes + = per_cpu_ptr ( stats , i ) - > s . xs_write_bytes ;
xs_read_bytes + = per_cpu_ptr ( stats , i ) - > s . xs_read_bytes ;
2015-10-11 21:15:45 +03:00
}
len + = snprintf ( buf + len , PATH_MAX - len , " xpc %Lu %Lu %Lu \n " ,
xs_xstrat_bytes , xs_write_bytes , xs_read_bytes ) ;
len + = snprintf ( buf + len , PATH_MAX - len , " debug %u \n " ,
# if defined(DEBUG)
1 ) ;
# else
0 ) ;
# endif
return len ;
}
2015-10-11 21:19:45 +03:00
void xfs_stats_clearall ( struct xfsstats __percpu * stats )
2015-10-11 21:15:45 +03:00
{
int c ;
2017-06-16 21:00:05 +03:00
uint32_t vn_active ;
2015-10-11 21:15:45 +03:00
xfs_notice ( NULL , " Clearing xfsstats " ) ;
for_each_possible_cpu ( c ) {
preempt_disable ( ) ;
/* save vn_active, it's a universal truth! */
2016-12-05 06:38:58 +03:00
vn_active = per_cpu_ptr ( stats , c ) - > s . vn_active ;
2015-10-11 21:19:45 +03:00
memset ( per_cpu_ptr ( stats , c ) , 0 , sizeof ( * stats ) ) ;
2016-12-05 06:38:58 +03:00
per_cpu_ptr ( stats , c ) - > s . vn_active = vn_active ;
2015-10-11 21:15:45 +03:00
preempt_enable ( ) ;
}
}
2018-05-25 18:16:04 +03:00
# ifdef CONFIG_PROC_FS
2012-03-13 12:52:33 +04:00
/* legacy quota interfaces */
# ifdef CONFIG_XFS_QUOTA
static int xqm_proc_show ( struct seq_file * m , void * v )
{
/* maximum; incore; ratio free to inuse; freelist */
seq_printf ( m , " %d \t %d \t %d \t %u \n " ,
2015-10-11 21:19:45 +03:00
0 , counter_val ( xfsstats . xs_stats , XFSSTAT_END_XQMSTAT ) ,
0 , counter_val ( xfsstats . xs_stats , XFSSTAT_END_XQMSTAT + 1 ) ) ;
2012-03-13 12:52:33 +04:00
return 0 ;
}
/* legacy quota stats interface no 2 */
static int xqmstat_proc_show ( struct seq_file * m , void * v )
{
int j ;
seq_printf ( m , " qm " ) ;
for ( j = XFSSTAT_END_IBT_V2 ; j < XFSSTAT_END_XQMSTAT ; j + + )
2015-10-11 21:19:45 +03:00
seq_printf ( m , " %u " , counter_val ( xfsstats . xs_stats , j ) ) ;
2012-03-13 12:52:33 +04:00
seq_putc ( m , ' \n ' ) ;
return 0 ;
}
# endif /* CONFIG_XFS_QUOTA */
2008-07-18 11:11:46 +04:00
int
2005-04-17 02:20:36 +04:00
xfs_init_procfs ( void )
{
if ( ! proc_mkdir ( " fs/xfs " , NULL ) )
2015-10-12 10:21:22 +03:00
return - ENOMEM ;
2008-07-18 11:11:46 +04:00
2015-10-11 21:16:45 +03:00
if ( ! proc_symlink ( " fs/xfs/stat " , NULL ,
" /sys/fs/xfs/stats/stats " ) )
2015-10-12 10:21:22 +03:00
goto out ;
2015-10-11 21:16:45 +03:00
2012-03-13 12:52:33 +04:00
# ifdef CONFIG_XFS_QUOTA
2018-05-15 16:57:23 +03:00
if ( ! proc_create_single ( " fs/xfs/xqmstat " , 0 , NULL , xqmstat_proc_show ) )
2015-10-12 10:21:22 +03:00
goto out ;
2018-05-15 16:57:23 +03:00
if ( ! proc_create_single ( " fs/xfs/xqm " , 0 , NULL , xqm_proc_show ) )
2015-10-12 10:21:22 +03:00
goto out ;
2012-03-13 12:52:33 +04:00
# endif
2008-07-18 11:11:46 +04:00
return 0 ;
2015-10-12 10:21:22 +03:00
out :
remove_proc_subtree ( " fs/xfs " , NULL ) ;
2008-07-18 11:11:46 +04:00
return - ENOMEM ;
2005-04-17 02:20:36 +04:00
}
void
xfs_cleanup_procfs ( void )
{
2015-10-12 10:21:22 +03:00
remove_proc_subtree ( " fs/xfs " , NULL ) ;
2005-04-17 02:20:36 +04:00
}
2015-10-19 00:42:46 +03:00
# endif /* CONFIG_PROC_FS */