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
*
2005-11-02 06:58:39 +03:00
* This program is free software ; you can redistribute it and / or
* modify it under the terms of the GNU General Public License as
2005-04-17 02:20:36 +04:00
* published by the Free Software Foundation .
*
2005-11-02 06:58:39 +03:00
* This program is distributed in the hope that it would be useful ,
* but WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
* GNU General Public License for more details .
2005-04-17 02:20:36 +04:00
*
2005-11-02 06:58:39 +03:00
* You should have received a copy of the GNU General Public License
* along with this program ; if not , write the Free Software Foundation ,
* Inc . , 51 Franklin St , Fifth Floor , Boston , MA 02110 - 1301 USA
2005-04-17 02:20:36 +04:00
*/
# include "xfs.h"
# include <linux/proc_fs.h>
DEFINE_PER_CPU ( struct xfsstats , xfsstats ) ;
STATIC int
xfs_read_xfsstats (
char * buffer ,
char * * start ,
off_t offset ,
int count ,
int * eof ,
void * data )
{
int c , i , j , len , val ;
__uint64_t xs_xstrat_bytes = 0 ;
__uint64_t xs_write_bytes = 0 ;
__uint64_t xs_read_bytes = 0 ;
static 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 } ,
} ;
/* Loop over all stats groups */
for ( i = j = len = 0 ; i < sizeof ( xstats ) / sizeof ( struct xstats_entry ) ; i + + ) {
len + = sprintf ( buffer + len , xstats [ i ] . desc ) ;
/* inner loop does each group */
while ( j < xstats [ i ] . endpoint ) {
val = 0 ;
/* sum over all cpus */
for ( c = 0 ; c < NR_CPUS ; c + + ) {
if ( ! cpu_possible ( c ) ) continue ;
val + = * ( ( ( __u32 * ) & per_cpu ( xfsstats , c ) + j ) ) ;
}
len + = sprintf ( buffer + len , " %u " , val ) ;
j + + ;
}
buffer [ len + + ] = ' \n ' ;
}
/* extra precision counters */
for ( i = 0 ; i < NR_CPUS ; i + + ) {
if ( ! cpu_possible ( i ) ) continue ;
xs_xstrat_bytes + = per_cpu ( xfsstats , i ) . xs_xstrat_bytes ;
xs_write_bytes + = per_cpu ( xfsstats , i ) . xs_write_bytes ;
xs_read_bytes + = per_cpu ( xfsstats , i ) . xs_read_bytes ;
}
len + = sprintf ( buffer + len , " xpc %Lu %Lu %Lu \n " ,
xs_xstrat_bytes , xs_write_bytes , xs_read_bytes ) ;
len + = sprintf ( buffer + len , " debug %u \n " ,
# if defined(DEBUG)
1 ) ;
# else
0 ) ;
# endif
if ( offset > = len ) {
* start = buffer ;
* eof = 1 ;
return 0 ;
}
* start = buffer + offset ;
if ( ( len - = offset ) > count )
return count ;
* eof = 1 ;
return len ;
}
void
xfs_init_procfs ( void )
{
if ( ! proc_mkdir ( " fs/xfs " , NULL ) )
return ;
create_proc_read_entry ( " fs/xfs/stat " , 0 , NULL , xfs_read_xfsstats , NULL ) ;
}
void
xfs_cleanup_procfs ( void )
{
remove_proc_entry ( " fs/xfs/stat " , NULL ) ;
remove_proc_entry ( " fs/xfs " , NULL ) ;
}