diff options
Diffstat (limited to 'src/netcpu_perfstat.c')
-rw-r--r-- | src/netcpu_perfstat.c | 355 |
1 files changed, 355 insertions, 0 deletions
diff --git a/src/netcpu_perfstat.c b/src/netcpu_perfstat.c new file mode 100644 index 0000000..e0e3399 --- /dev/null +++ b/src/netcpu_perfstat.c @@ -0,0 +1,355 @@ +char netcpu_perfstat_id[]="\ +@(#)netcpu_perfstat.c Version 2.6.0"; + +#if HAVE_CONFIG_H +# include <config.h> +#endif + +#include <stdio.h> + +#if HAVE_INTTYPES_H +# include <inttypes.h> +#else +# if HAVE_STDINT_H +# include <stdint.h> +# endif +#endif + +#if TIME_WITH_SYS_TIME +# include <sys/time.h> +# include <time.h> +#else +# if HAVE_SYS_TIME_H +# include <sys/time.h> +# else +# include <time.h> +# endif +#endif + +#if HAVE_LIMITS_H +# include <limits.h> +# ifndef LONG_LONG_MAX +# define LONG_LONG_MAX LLONG_MAX +# endif /* LONG_LONG_MAX */ +#endif + +#include <errno.h> + +#include "netsh.h" +#include "netlib.h" + +/* the lib_start_count and lib_end_count arrays hold the starting + and ending values of whatever is counting when the system is + idle. The rate at which this increments during a test is compared + with a previous calibration to arrive at a CPU utilization + percentage. raj 2005-01-26 */ +static uint64_t lib_start_count[MAXCPUS]; +static uint64_t lib_end_count[MAXCPUS]; + + +void +cpu_util_init(void) +{ + return; +} + +void +cpu_util_terminate(void) +{ + return; +} + +int +get_cpu_method(void) +{ + return PERFSTAT; +} + +static void +get_cpu_idle(uint64_t *res) +{ + perfstat_cpu_t *perfstat_buffer; + perfstat_cpu_t *per_cpu_pointer; + perfstat_id_t name; + int i,ret; + + /* a name of "" will cause us to start from the beginning */ + strcpy(name.name,""); + perfstat_buffer = (perfstat_cpu_t *)malloc(lib_num_loc_cpus * + sizeof(perfstat_cpu_t)); + if (perfstat_buffer == NULL) { + fprintf(where, + "cpu_start: malloc failed errno %d\n", + errno); + fflush(where); + exit(-1); + } + + /* happiness and joy, keep going */ + ret = perfstat_cpu(&name, + perfstat_buffer, + sizeof(perfstat_cpu_t), + lib_num_loc_cpus); + + if ((ret == -1) || + (ret != lib_num_loc_cpus)) { + fprintf(where, + "cpu_start: perfstat_cpu failed/count off; errno %d cpus %d count %d\n", + errno, + lib_num_loc_cpus, + ret); + fflush(where); + exit(-1); + } + + per_cpu_pointer = perfstat_buffer; + for (i = 0; i < lib_num_loc_cpus; i++){ + res[i] = per_cpu_pointer->idle; + per_cpu_pointer++; + } + free(perfstat_buffer); + + return; +} + +float +calibrate_idle_rate(int iterations, int interval) +{ + unsigned long long + firstcnt[MAXCPUS], + secondcnt[MAXCPUS]; + + float + elapsed, + temp_rate, + rate[MAXTIMES], + local_maxrate; + + long + sec, + usec; + + int + i, + j; + + struct timeval time1, time2 ; + struct timezone tz; + + perfstat_cpu_t *perfstat_buffer; + perfstat_cpu_t *per_cpu_pointer; + perfstat_id_t name; + int ret; + + if (debug) { + fprintf(where,"enter calibrate_perfstat\n"); + fflush(where); + } + + if (iterations > MAXTIMES) { + iterations = MAXTIMES; + } + + local_maxrate = (float)-1.0; + + perfstat_buffer = (perfstat_cpu_t *)malloc(lib_num_loc_cpus * + sizeof(perfstat_cpu_t)); + if (perfstat_buffer == NULL) { + fprintf(where, + "calibrate_perfstat: malloc failed errno %d\n", + errno); + fflush(where); + exit(-1); + } + + for(i = 0; i < iterations; i++) { + rate[i] = (float)0.0; + /* a name of "" will cause us to start from the beginning */ + strcpy(name.name,""); + + /* happiness and joy, keep going */ + ret = perfstat_cpu(&name, + perfstat_buffer, + sizeof(perfstat_cpu_t), + lib_num_loc_cpus); + + if ((ret == -1) || + (ret != lib_num_loc_cpus)) { + fprintf(where, + "calibrate_perfstat: perfstat_cpu failed/count off; errno %d cpus %d count %d\n", + errno, + lib_num_loc_cpus, + ret); + fflush(where); + exit(-1); + } + + per_cpu_pointer = perfstat_buffer; + for (j = 0; j < lib_num_loc_cpus; j++) { + firstcnt[j] = per_cpu_pointer->idle; + per_cpu_pointer++; + } + gettimeofday (&time1, &tz); + sleep(interval); + gettimeofday (&time2, &tz); + + if (time2.tv_usec < time1.tv_usec) + { + time2.tv_usec += 1000000; + time2.tv_sec -=1; + } + sec = time2.tv_sec - time1.tv_sec; + usec = time2.tv_usec - time1.tv_usec; + elapsed = (float)sec + ((float)usec/(float)1000000.0); + + /* happiness and joy, keep going */ + ret = perfstat_cpu(&name, + perfstat_buffer, + sizeof(perfstat_cpu_t), + lib_num_loc_cpus); + + if ((ret == -1) || + (ret != lib_num_loc_cpus)) { + fprintf(where, + "calibrate_perfstat: perfstat_cpu failed/count off; errno %d cpus %d count %d\n", + errno, + lib_num_loc_cpus, + ret); + fflush(where); + exit(-1); + } + + per_cpu_pointer = perfstat_buffer; + + if(debug) { + fprintf(where, + "Calibration for perfstat counter run: %d\n" + "\tsec = %ld usec = %ld\n" + "\telapsed time = %g\n", + i, + sec,usec, + elapsed); + } + + for (j = 0; j < lib_num_loc_cpus; j++) { + secondcnt[j] = per_cpu_pointer->idle; + per_cpu_pointer++; + if(debug) { + /* I know that there are situations where compilers know about + long long, but the library functions do not... raj 4/95 */ + fprintf(where, + "\tfirstcnt[%d] = 0x%8.8lx%8.8lx secondcnt[%d] = 0x%8.8lx%8.8lx\n", + j, + firstcnt[j], + firstcnt[j], + j, + secondcnt[j], + secondcnt[j]); + } + /* we assume that it would wrap no more than once. we also + assume that the result of subtracting will "fit" raj 4/95 */ + temp_rate = (secondcnt[j] >= firstcnt[j]) ? + (float)(secondcnt[j] - firstcnt[j])/elapsed : + (float)(secondcnt[j]-firstcnt[j]+MAXLONG)/elapsed; + if (temp_rate > rate[i]) rate[i] = temp_rate; + if(debug) { + fprintf(where,"\trate[%d] = %g\n",i,rate[i]); + fflush(where); + } + if (local_maxrate < rate[i]) local_maxrate = rate[i]; + } + } + if(debug) { + fprintf(where,"\tlocal maxrate = %g per sec. \n",local_maxrate); + fflush(where); + } + free(perfstat_buffer); + return local_maxrate; +} + +float +calc_cpu_util_internal(float elapsed_time) +{ + int i; + + float actual_rate; + float correction_factor; + + memset(&lib_local_cpu_stats, 0, sizeof(lib_local_cpu_stats)); + + /* It is possible that the library measured a time other than the + one that the user want for the cpu utilization calculations - for + example, tests that were ended by watchdog timers such as the udp + stream test. We let these tests tell up what the elapsed time + should be. */ + + if (elapsed_time != 0.0) { + correction_factor = (float) 1.0 + + ((lib_elapsed - elapsed_time) / elapsed_time); + } + else { + correction_factor = (float) 1.0; + } + + /* this looks just like the looper case. at least I think it should + :) raj 4/95 */ + for (i = 0; i < lib_num_loc_cpus; i++) { + + /* we assume that the two are not more than a long apart. I know + that this is bad, but trying to go from long longs to a float + (perhaps a double) is boggling my mind right now. raj 4/95 */ + + long long + diff; + + if (lib_end_count[i] >= lib_start_count[i]) { + diff = lib_end_count[i] - lib_start_count[i]; + } + else { + diff = lib_end_count[i] - lib_start_count[i] + LONG_LONG_MAX; + } + actual_rate = (float) diff / lib_elapsed; + lib_local_per_cpu_util[i] = (lib_local_maxrate - actual_rate) / + lib_local_maxrate * 100; + lib_local_cpu_stats.cpu_util += lib_local_per_cpu_util[i]; + if (debug) { + fprintf(where, + "calc_cpu_util: actual_rate on cpu %d is %g max_rate %g cpu %6.2f\n", + i, + actual_rate, + lib_local_maxrate, + lib_local_per_cpu_util[i]); + } + } + + /* we want the average across all n processors */ + lib_local_cpu_stats.cpu_util /= (float)lib_num_loc_cpus; + + if (debug) { + fprintf(where, + "calc_cpu_util: average across CPUs is %g\n", + lib_local_cpu_stats.cpu_util); + } + + lib_local_cpu_stats.cpu_util *= correction_factor; + + if (debug) { + fprintf(where, + "calc_cpu_util: returning %g\n",lib_local_cpu_stats.cpu_util); + } + + return lib_local_cpu_stats.cpu_util; + +} +void +cpu_start_internal(void) +{ + get_cpu_idle(lib_start_count); + return; +} + +void +cpu_stop_internal(void) +{ + get_cpu_idle(lib_end_count); +} |