2005-11-08 19:33:45 +03:00
/*
* Unix SMB / CIFS implementation .
* Performance Counter Daemon
*
* Copyright ( C ) Marcin Krzysztof Porwit 2005
*
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License as published by
2007-07-09 23:25:36 +04:00
* the Free Software Foundation ; either version 3 of the License , or
2005-11-08 19:33:45 +03:00
* ( at your option ) any later version .
*
* This program is distributed in the hope that it will 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 .
*
* You should have received a copy of the GNU General Public License
* along with this program ; if not , write to the Free Software
* Foundation , Inc . , 675 Mass Ave , Cambridge , MA 0213 9 , USA .
*/
# include "perf.h"
void init_diskdata_desc ( PERF_DATA_BLOCK * data )
{
init_perf_counter ( & ( data - > diskInfo . diskObjDesc ) ,
& ( data - > diskInfo . diskObjDesc ) ,
get_counter_id ( data ) ,
" Logical Disk " ,
" The Logical Disk object consists of counters that show information about disks. " ,
0 ,
PERF_OBJECT ) ;
init_perf_counter ( & ( data - > diskInfo . freeMegs ) ,
& ( data - > diskInfo . diskObjDesc ) ,
get_counter_id ( data ) ,
" Megabytes Free " ,
" The amount of available disk space, in megabytes. " ,
PERF_SIZE_LARGE | PERF_TYPE_NUMBER | PERF_NUMBER_DECIMAL | PERF_DISPLAY_NO_SUFFIX ,
PERF_COUNTER ) ;
init_perf_counter ( & ( data - > diskInfo . writesPerSec ) ,
& ( data - > diskInfo . diskObjDesc ) ,
get_counter_id ( data ) ,
" Writes/sec " ,
" The number of writes per second to that disk. " ,
PERF_SIZE_DWORD | PERF_TYPE_COUNTER | PERF_COUNTER_RATE | PERF_DELTA_COUNTER | PERF_DISPLAY_PER_SEC ,
PERF_COUNTER ) ;
init_perf_counter ( & ( data - > diskInfo . readsPerSec ) ,
& ( data - > diskInfo . diskObjDesc ) ,
get_counter_id ( data ) ,
" Reads/sec " ,
" The number of reads of that disk per second. " ,
PERF_SIZE_DWORD | PERF_TYPE_COUNTER | PERF_COUNTER_RATE | PERF_DELTA_COUNTER | PERF_DISPLAY_PER_SEC ,
PERF_COUNTER ) ;
return ;
}
void init_num_disks ( PERF_DATA_BLOCK * data )
{
FILE * mtab ;
char buf [ PROC_BUF ] ;
char * start , * stop ;
int i = 0 , num ;
if ( ! ( mtab = fopen ( " /etc/mtab " , " r " ) ) )
{
perror ( " init_disk_names: fopen " ) ;
exit ( 1 ) ;
}
rewind ( mtab ) ;
fflush ( mtab ) ;
while ( fgets ( buf , sizeof ( buf ) , mtab ) )
{
if ( start = strstr ( buf , " /dev/ " ) )
{
if ( start = strstr ( start , " da " ) )
{
i + + ;
}
}
}
data - > diskInfo . numDisks = i ;
fclose ( mtab ) ;
return ;
}
void init_disk_names ( PERF_DATA_BLOCK * data )
{
FILE * mtab ;
char buf [ PROC_BUF ] ;
char * start , * stop ;
int i = 0 , num ;
if ( ! ( mtab = fopen ( " /etc/mtab " , " r " ) ) )
{
perror ( " init_disk_names: fopen " ) ;
exit ( 1 ) ;
}
rewind ( mtab ) ;
fflush ( mtab ) ;
while ( fgets ( buf , sizeof ( buf ) , mtab ) )
{
if ( start = strstr ( buf , " /dev/ " ) )
{
if ( start = strstr ( start , " da " ) )
{
start - = 1 ;
stop = strstr ( start , " " ) ;
memcpy ( data - > diskInfo . mdata [ i ] . name , start , stop - start ) ;
start = stop + 1 ;
stop = strstr ( start , " " ) ;
memcpy ( data - > diskInfo . mdata [ i ] . mountpoint , start , stop - start ) ;
i + + ;
}
}
}
fclose ( mtab ) ;
return ;
}
void get_diskinfo ( PERF_DATA_BLOCK * data )
{
int i ;
DiskData * p ;
struct statfs statfsbuf ;
int status , num ;
char buf [ LARGE_BUF ] , * start ;
FILE * diskstats ;
long reads , writes , discard ;
diskstats = fopen ( " /proc/diskstats " , " r " ) ;
rewind ( diskstats ) ;
fflush ( diskstats ) ;
status = fread ( buf , sizeof ( char ) , LARGE_BUF , diskstats ) ;
fclose ( diskstats ) ;
for ( i = 0 ; i < data - > diskInfo . numDisks ; i + + )
{
p = & ( data - > diskInfo . data [ i ] ) ;
status = statfs ( data - > diskInfo . mdata [ i ] . mountpoint , & statfsbuf ) ;
p - > freeMegs = ( statfsbuf . f_bfree * statfsbuf . f_bsize ) / 1048576 ;
start = strstr ( buf , data - > diskInfo . mdata [ i ] . name ) ;
start + = strlen ( data - > diskInfo . mdata [ i ] . name ) + 1 ;
num = sscanf ( start , " %u %u %u %u " ,
& reads ,
& discard ,
& writes ,
& discard ) ;
p - > writesPerSec = writes ;
p - > readsPerSec = reads ;
fprintf ( stderr , " %s: \t %u \t %u \n " ,
data - > diskInfo . mdata [ i ] . mountpoint ,
reads , writes ) ;
}
return ;
}
void init_disk_data ( PERF_DATA_BLOCK * data )
{
init_diskdata_desc ( data ) ;
init_num_disks ( data ) ;
data - > diskInfo . mdata = calloc ( data - > diskInfo . numDisks , sizeof ( DiskMetaData ) ) ;
if ( ! data - > diskInfo . mdata )
{
fatal ( " init_disk_data: out of memory " ) ;
}
init_disk_names ( data ) ;
data - > diskInfo . data = calloc ( data - > diskInfo . numDisks , sizeof ( DiskData ) ) ;
if ( ! data - > diskInfo . data )
{
fatal ( " init_disk_data: out of memory " ) ;
}
get_diskinfo ( data ) ;
return ;
}
void output_disk_desc ( PERF_DATA_BLOCK * data , RuntimeSettings rt )
{
output_perf_desc ( data - > diskInfo . diskObjDesc , rt ) ;
output_perf_desc ( data - > diskInfo . freeMegs , rt ) ;
output_perf_desc ( data - > diskInfo . writesPerSec , rt ) ;
output_perf_desc ( data - > diskInfo . readsPerSec , rt ) ;
output_num_instances ( data - > diskInfo . diskObjDesc , data - > diskInfo . numDisks , rt ) ;
return ;
}
void output_diskinfo ( PERF_DATA_BLOCK * data , RuntimeSettings rt , int tdb_flags )
{
int i ;
output_perf_counter ( data - > diskInfo . freeMegs ,
data - > diskInfo . data [ 0 ] . freeMegs ,
rt , tdb_flags ) ;
output_perf_counter ( data - > diskInfo . writesPerSec ,
( unsigned long long ) data - > diskInfo . data [ 0 ] . writesPerSec ,
rt , tdb_flags ) ;
output_perf_counter ( data - > diskInfo . readsPerSec ,
( unsigned long long ) data - > diskInfo . data [ 0 ] . readsPerSec ,
rt , tdb_flags ) ;
for ( i = 0 ; i < data - > diskInfo . numDisks ; i + + )
{
output_perf_instance ( data - > diskInfo . diskObjDesc . index ,
i ,
( void * ) & ( data - > diskInfo . data [ i ] ) ,
sizeof ( DiskData ) ,
data - > diskInfo . mdata [ i ] . mountpoint ,
rt , tdb_flags ) ;
}
return ;
}