-
Notifications
You must be signed in to change notification settings - Fork 9
/
Copy pathIOPerformanceLinux.cpp
619 lines (561 loc) · 24.9 KB
/
IOPerformanceLinux.cpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
/* ######################################################################### */
/* ## ## */
/* ## Dynamo / IOPerformanceLinux.cpp ## */
/* ## ## */
/* ## ------------------------------------------------------------------- ## */
/* ## ## */
/* ## Job .......: The GNU/Linux variant of the Performance class. ## */
/* ## This class does two jobs: ## */
/* ## 1.) Collects basic informations like the number ## */
/* ## of CPU's. ## */
/* ## 2.) Collection of system-wide performance ## */
/* ## informations. ## */
/* ## ## */
/* ## ------------------------------------------------------------------- ## */
/* ## ## */
/* ## Intel Open Source License ## */
/* ## ## */
/* ## Copyright (c) 2001 Intel Corporation ## */
/* ## All rights reserved. ## */
/* ## Redistribution and use in source and binary forms, with or ## */
/* ## without modification, are permitted provided that the following ## */
/* ## conditions are met: ## */
/* ## ## */
/* ## Redistributions of source code must retain the above copyright ## */
/* ## notice, this list of conditions and the following disclaimer. ## */
/* ## ## */
/* ## Redistributions in binary form must reproduce the above copyright ## */
/* ## notice, this list of conditions and the following disclaimer in ## */
/* ## the documentation and/or other materials provided with the ## */
/* ## distribution. ## */
/* ## ## */
/* ## Neither the name of the Intel Corporation nor the names of its ## */
/* ## contributors may be used to endorse or promote products derived ## */
/* ## from this software without specific prior written permission. ## */
/* ## ## */
/* ## THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND ## */
/* ## CONTRIBUTORS ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, ## */
/* ## INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF ## */
/* ## MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE ## */
/* ## DISCLAIMED. IN NO EVENT SHALL THE INTEL OR ITS CONTRIBUTORS BE ## */
/* ## LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, ## */
/* ## OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, ## */
/* ## PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, ## */
/* ## OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY ## */
/* ## THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR ## */
/* ## TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT ## */
/* ## OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY ## */
/* ## OF SUCH DAMAGE. ## */
/* ## ## */
/* ## ------------------------------------------------------------------- ## */
/* ## ## */
/* ## Remarks ...: - This content of this file should be merged into ## */
/* ## the generic IOPerformance.cpp file. ## */
/* ## ## */
/* ## ------------------------------------------------------------------- ## */
/* ## ## */
/* ## Changes ...: 2005-01-12 ([email protected]) ## */
/* ## - Ia64 (Itanium) can get its CPU data the same as ## */
/* ## i386. ## */
/* ## 2004-09-01 ([email protected]) ## */
/* ## - X86_64 can get its CPU data the same as i368. ## */
/* ## 2004-03-27 ([email protected]) ## */
/* ## - Applied Dan Bar Dov's patch for adding ## */
/* ## Linux on PPC support. ## */
/* ## 2004-02-15 ([email protected]) ## */
/* ## - Bugfix for Get_NI_Counters to parse /proc/net/dev ## */
/* ## with correct format. ## */
/* ## 2004-02-07 ([email protected]) ## */
/* ## - Changed all im_kstat to iomtr_kstat with Daniel ## */
/* ## suggestion. A better name. ## */
/* ## 2004-02-06 ([email protected]) ## */
/* ## - Added im_kstat ioctl call in Get_Processor_Count. ## */
/* ## - Added im_kstat ioctl call in Get_Processor_Speed. ## */
/* ## - Added im_kstat ioctl call in Get_CPU_Counters. ## */
/* ## - Handle two /proc/stat style in Get_CPU_Counters. ## */
/* ## - Implement Get_TCP_Counters. ## */
/* ## - Added im_kstat ioctl call in Get_NI_Counters. ## */
/* ## - Fixed a time difference computation error in ## */
/* ## Calculate_NI_Stats() since time_counter unit is ## */
/* ## not nanosecond but clock_tick ## */
/* ## 2003-10-15 ([email protected]) ## */
/* ## - Moved to the use of the IOMTR_[OSFAMILY|OS|CPU]_* ## */
/* ## global defines. ## */
/* ## - Integrated the License Statement into this header. ## */
/* ## 2003-03-02 ([email protected]) ## */
/* ## - Cleaned the code from Windows related stuff ## */
/* ## (doesn't make sense in a pure Linux context). ## */
/* ## 2003-03-02 ([email protected]) ## */
/* ## - Bugfix for Get_NI_Counters to ensure, that ## */
/* ## interface names with no leading blanks (for ## */
/* ## example dummy0) are handled as well. ## */
/* ## 2003-02-26 ([email protected]) ## */
/* ## - Added some more GHz processor stuff. ## */
/* ## - Changed Get_NI_Counters so interfaces that do not ## */
/* ## contain statistics in /proc/net/dev do not cause ## */
/* ## dynamo to die. Right now the only interface that ## */
/* ## does this is for Intel Link Aggregation (ians) ## */
/* ## 2003-02-09 ([email protected]) ## */
/* ## - Modified Get_CPU_Counters to add the Jiffies for ## */
/* ## user mode with low priority (nice) to the user ## */
/* ## mode utilization as well as the total utilization. ## */
/* ## 2003-02-02 ([email protected]) ## */
/* ## - Added new header holding the changelog. ## */
/* ## - Applied the iometer-initial-datatypes.patch file ## */
/* ## (brings some minor changes to main() function). ## */
/* ## ## */
/* ######################################################################### */
#define PERFORMANCE_DETAILS 0 // Turn on to display additional performance messages.
#if defined(IOMTR_OS_LINUX)
#include <sys/time.h>
#include <assert.h>
#include "IOPerformance.h"
#include "iomtr_kstat/iomtr_kstat.h"
extern int kstatfd;
//
// Initializing system performance data.
//
Performance::Performance()
{
// Obtaining the number of CPUs in the system and their speed.
processor_count = Get_Processor_Count();
timer_resolution = Get_Timer_Resolution();
// Network performance statistics are disabled by default. Assume this
// unless later performance calls succeed. Then, set the correct number of
// interfaces.
network_interfaces = 0;
if (!timer_resolution || !processor_count) {
cout << "*** Unable to initialize needed performance data.\n"
<< "This error may indicate that you are trying to run on an\n"
<< "unsupported processor or OS. See the Iometer User's Guide for\n"
<< "information on supported platforms.\n";
exit(1);
}
// Set the clock ticks per second
// TODO: try to find a correct way to get this number
// DF: Base on ''man sysconf'', this macro is obsolete now. so things become tricky
clock_tick = sysconf(_SC_CLK_TCK);
// Initialize all the arrays to 0.
memset(raw_cpu_data, 0, (MAX_CPUS * CPU_RESULTS * MAX_SNAPSHOTS * sizeof(_int64)));
memset(raw_ni_data, 0, (MAX_NUM_INTERFACES * NI_RESULTS * MAX_SNAPSHOTS * sizeof(_int64)));
memset(raw_tcp_data, 0, (TCP_RESULTS * MAX_SNAPSHOTS * sizeof(_int64)));
}
//
// Freeing memory used by class object.
//
Performance::~Performance()
{
}
int Performance::Get_Processor_Count(void)
{
char buf[4096];
const char *p = "processor\t:";
int i = 0;
size_t res, j;
FILE *cpuinfo;
cpuinfo = fopen("/proc/cpuinfo", "r");
if (cpuinfo) {
j = strlen(p);
do {
memset(buf, 0, 4096);
res = fread(buf, 1, 4095, cpuinfo);
if (res >= j) {
char *c = buf;
while ((c = strstr(c, p))) {
i++;
c += j;
};
(void)fseek(cpuinfo, -j + 1, SEEK_CUR);
}
} while (res >= j);
(void)fclose(cpuinfo);
}
if (!i) {
/* fall through to the glib way, though sometime it is buggy. */
return sysconf(_SC_NPROCESSORS_ONLN);
}
return i;
}
//
// Getting the speed of the processors in Hz. We get this from /proc/cpuinfo.
// Example output of /proc/cpuinfo:
//////////////////////////////////////////////////////////////////////
// processor : 0
// vendor_id : GenuineIntel
// cpu family : 6
// model : 7
// model name : Pentium III (Katmai)
// stepping : 3
// cpu MHz : 497.845315
// cache size : 512 KB
// ... file continues on ...
//////////////////////////////////////////////////////////////////////
// Note: We just take the first CPU we find and return its CPU speed.
double Performance::Get_Timer_Resolution()
{
#if defined(IOMTR_CPU_I386) || defined(IOMTR_CPU_XSCALE) || defined(IOMTR_CPU_X86_64) || defined(IOMTR_CPU_IA64)
int c;
char label[40];
int scanDecodes;
double result;
FILE *cpuInfo;
int khz;
if (kstatfd > 0 && ioctl(kstatfd, IM_IOC_GETCPUKHZ, &khz) >= 0) {
cout << "CPU KHZ: " << khz << endl;
return (double)khz *1000.0;
}
cpuInfo = fopen("/proc/cpuinfo", "r");
if (!cpuInfo) {
cerr << "Error determining CPU speed.\n";
return (0.0);
}
do {
fscanf(cpuInfo, "%7c", label);
if (!strncmp(label, "cpu MHz", 7)) {
scanDecodes = fscanf(cpuInfo, "%*s %lf", &result);
result *= 1000000.0;
fclose(cpuInfo);
if (scanDecodes == 1) {
return (result);
} else {
cerr << "Error determining CPU speed.\n";
return (0.0);
}
} else if (!strncmp(label, "cpu GHz", 7)) {
scanDecodes = fscanf(cpuInfo, "%*s %lf", &result);
result *= 1000000000.0;
fclose(cpuInfo);
if (scanDecodes == 1) {
return (result);
} else {
cerr << "Error determining CPU speed.\n";
return (0.0);
}
}
// Skip to the next line.
do {
c = getc(cpuInfo);
} while ((c != '\n') && (c != EOF));
} while (c != EOF);
fclose(cpuInfo);
cerr << "Error determining CPU speed.\n";
return (0.0);
#elif defined(IOMTR_CPU_PPC)
#define USEC(tv) (tv.tv_sec*1000000+tv.tv_usec)
struct timeval tv1, tv2;
DWORD t1, t2;
double result;
gettimeofday(&tv1, NULL);
t1 = get_tbl();
sleep(1);
t2 = get_tbl();
gettimeofday(&tv2, NULL);
result = ((double)(t2 - t1)) * (1000000.0 / (double)(USEC(tv2) - USEC(tv1)));
return (result);
#else
#warning ===> WARNING: You have to do some coding here to get the port done!
#endif
}
void Performance::Get_Perf_Data(DWORD perf_data_type, int snapshot)
{
// Get the performance data stored by the system.
#ifdef _DEBUG
cout << " Getting system performance data." << endl << flush;
#endif
time_counter[snapshot] = jiffies();
if (snapshot == LAST_SNAPSHOT) {
// calculate time diff in clock ticks..
timediff = (time_counter[LAST_SNAPSHOT] - time_counter[FIRST_SNAPSHOT]);
}
switch (perf_data_type) {
case PERF_PROCESSOR:
Get_CPU_Counters(snapshot);
break;
case PERF_NETWORK_TCP:
Get_TCP_Counters(snapshot);
break;
case PERF_NETWORK_INTERFACE:
Get_NI_Counters(snapshot);
break;
default:
break;
}
}
void Performance::Get_CPU_Counters(int snapshot)
{
int numScans, i;
char tmpBuf[SMLBUFSIZE], *b;
__int64 jiffiesCpuNiceUtilization, jiffiesCpuidle, jiffiesCpuiowait;
__int64 jiffiesCpuirq, jiffiesCpusoftirq, jiffiesCpusteal;
__int64 jiffiesUnknown;
FILE *cpuStat;
struct cpu_data_type raw_cpu;
if (kstatfd > 0 && ioctl(kstatfd, IM_IOC_GETCPUDATA, &raw_cpu) >= 0) {
for (i = 0; i < processor_count; i++) {
raw_cpu_data[i][CPU_USER_UTILIZATION][snapshot] = raw_cpu.user_time[i];
raw_cpu_data[i][CPU_PRIVILEGED_UTILIZATION][snapshot] = raw_cpu.system_time[i];
raw_cpu_data[i][CPU_TOTAL_UTILIZATION][snapshot] =
raw_cpu_data[i][CPU_USER_UTILIZATION][snapshot] +
raw_cpu_data[i][CPU_PRIVILEGED_UTILIZATION][snapshot];
}
raw_cpu_data[0][CPU_IRQ][snapshot] = raw_cpu.intr;
return;
}
cpuStat = fopen("/proc/stat", "r");
if (!cpuStat)
goto err_out;
// different kernel has quite different /proc/stat output.
// First few rows of the /proc/stat output in 2.4 style.
// --------------------------------------------------------------------
// cpu 20969 2260 50042 1069377
// cpu0 20969 2260 50042 1069377
// page 1279455 540189
// swap 30 483
// intr 1896409 1142648 2 0 0 ...
// ...
// --------------------------------------------------------------------
// First few rows of the /proc/stat output in early 2.6 style
// cpu 31483 0 87972 461075 9775 1319 200
// cpu0 31483 0 87972 461075 9775 1319 200
// intr 6473496 5918277 10963 0 0 0 0 0 7 1 2 0 365055 154381 0 24810 0 0
//...
// --------------------------------------------------------------------
// First few rows of the /proc/stat output in 2.6.11 or later style
// cpu 152192 5922 18433 1509327 67908 3064 5273 0
// cpu0 152192 5922 18433 1509327 67908 3064 5273 0
// intr 644038....
//
// followed code gain idea from procps package.
memset(tmpBuf, 0, SMLBUFSIZE);
b = fgets(tmpBuf, SMLBUFSIZE, cpuStat);
if (!b)
goto err_out;
for (i = 0; i < processor_count; i++) {
int cpuid;
b = fgets(tmpBuf, SMLBUFSIZE, cpuStat);
if (!b)
goto err_out;
jiffiesCpuiowait = jiffiesCpuirq = jiffiesCpusoftirq = jiffiesCpusteal = 0;
numScans = sscanf(tmpBuf, "cpu%d %lld %lld %lld %lld %lld %lld %lld %lld %lld\n",
&cpuid,
&raw_cpu_data[i][CPU_USER_UTILIZATION][snapshot],
&jiffiesCpuNiceUtilization,
&raw_cpu_data[i][CPU_PRIVILEGED_UTILIZATION][snapshot],
&jiffiesCpuidle, &jiffiesCpuiowait, &jiffiesCpuirq,
&jiffiesCpusoftirq, &jiffiesCpusteal, &jiffiesUnknown);
if (numScans < 4)
goto err_out;
raw_cpu_data[i][CPU_USER_UTILIZATION][snapshot] += jiffiesCpuNiceUtilization;
raw_cpu_data[i][CPU_PRIVILEGED_UTILIZATION][snapshot] +=
jiffiesCpuirq + jiffiesCpusoftirq;
raw_cpu_data[i][CPU_TOTAL_UTILIZATION][snapshot] =
raw_cpu_data[i][CPU_USER_UTILIZATION][snapshot] +
raw_cpu_data[i][CPU_PRIVILEGED_UTILIZATION][snapshot];
}
fread(tmpBuf, sizeof(char), SMLBUFSIZE, cpuStat);
b = strstr(tmpBuf, "intr");
if (b)
sscanf(b, "intr %Lu", &raw_cpu_data[0][CPU_IRQ][snapshot]);
goto out;
err_out:
cerr << "Can not get CPU performance data" << endl;
out:
fclose(cpuStat);
}
void Performance::Get_TCP_Counters(int snapshot)
{
FILE *fp;
__int64 insegs, outsegs, retranssegs;
struct tcp_data_type raw_tcp;
char tmpBuf[SMLBUFSIZE];
int i;
if (kstatfd > 0 && ioctl(kstatfd, IM_IOC_GETTCPDATA, &raw_tcp) >= 0) {
raw_tcp_data[TCP_SEGMENTS_RESENT][snapshot] = raw_tcp.retranssegs;
return;
}
fp = fopen("/proc/net/snmp", "r");
for (i = 0; i < 5; i++)
fgets(tmpBuf, SMLBUFSIZE, fp);
retranssegs = 0;
fscanf(fp, "Tcp: %*s %*s %*s %*s %*s %*s %*s %*s %*s %lld %lld %lld %*s %*s\n", &insegs, &outsegs,
&retranssegs);
raw_tcp_data[TCP_SEGMENTS_RESENT][snapshot] = retranssegs;
fclose(fp);
}
//
// Calculating CPU statistics based on snapshots of performance counters.
//
void Performance::Calculate_CPU_Stats(CPU_Results * cpu_results)
{
int cpu, stat; // Loop control variables.
// Loop though all CPUs and determine various utilization statistics.
cpu_results->count = processor_count;
for (cpu = 0; cpu < processor_count; cpu++) {
// Loop through the counters and calculate performance.
for (stat = 0; stat < CPU_RESULTS; stat++) {
#if PERFORMANCE_DETAILS
cout << "Calculating stat " << stat << " for CPU " << cpu << endl;
#endif
double result;
if (stat == CPU_IRQ) {
// we have to calculate Interrupts/sec.
// This is similar to calculating Network packets per second
// but we are more fortunate here.
// See the corresponding Notes at the end of this file for a description.
//
result =
((double)raw_cpu_data[cpu][stat][LAST_SNAPSHOT] -
raw_cpu_data[cpu][stat][FIRST_SNAPSHOT]) * clock_tick / timediff;
cpu_results->CPU_utilization[cpu][stat] = result;
} else {
// All other CPU statistics.
result =
((double)raw_cpu_data[cpu][stat][LAST_SNAPSHOT] -
raw_cpu_data[cpu][stat][FIRST_SNAPSHOT]) / timediff;
if (result < 0.0) {
result = 0.0;
//
// CPU Utilization figures are outside valid range far too often.
// Ok, not in every cycle but frequent still.
// So, it is better to comment it out rather than have the message
// pop up on the screen at regular intervals.
//
// cout << "***** Error : CPU utilization outside valid timerange 0% - 100% *****" << endl;
}
if (result > 1.0) {
result = 1.0;
}
cpu_results->CPU_utilization[cpu][stat] = (result * 100);
}
#if PERFORMANCE_DETAILS || _DETAILS
cout << "CPU " << cpu << " recorded stat " << stat << " = "
<< cpu_results->CPU_utilization[cpu][stat] << endl;
#endif
}
}
}
//
// Calculate network performance statistics based on snapshots of performance counters.
//
void Performance::Calculate_TCP_Stats(Net_Results * net_results)
{
int stat; // Loop control variable.
// Loop through the counters and calculate performance.
for (stat = 0; stat < TCP_RESULTS; stat++) {
// If we've never set the counter offsets, then we've never successfully retrieved
// the performance data. Set all of the values to 0.
double result;
result = ((double)raw_tcp_data[stat][LAST_SNAPSHOT] - raw_tcp_data[stat][FIRST_SNAPSHOT]) / timediff;
result *= clock_tick; // note that timediff is in CLK_TCKs and not seconds
net_results->tcp_stats[stat] = result;
#if PERFORMANCE_DETAILS || _DETAILS
cout << "TCP recorded stat " << stat << " = " << net_results->tcp_stats[stat] << endl;
#endif
}
}
//
// Extracting counters for NT network interface performance data. This is done
// by reading the file "/proc/net/dev". WARNING: This file changes from one
// version of Linux to the next. This one will work on linux 2.2.12; no
// guarantees can be made for other versions!
//
// Example output from /proc/net/dev:
//////////////////////////////////////////////////////////////////////
// Inter-| Receive | Transmit
// face |bytes packets errs drop fifo frame compressed multicast|bytes packets errs drop fifo colls carrier compressed
// lo: 4632 76 0 0 0 0 0 0 4632 76 0 0 0 0 0 0
// eth0:30165814 176832 0 0 0 0 0 0 3700725 27219 0 0 0 288 0 0
//////////////////////////////////////////////////////////////////////
// Note: "lo" is the loopback device, and "eth0" is an ethernet card.
//
// Updated: This file is same from 2.4 to 2.6 kernel. so should works at most of the time.
#define NET_IF_TO_IGNORE "ians"
void Performance::Get_NI_Counters(int snapshot)
{
int c, scanCount, packetIn, packetOut;
char ifname[32];
FILE *netInfo;
struct ni_data_type raw_net;
if (kstatfd > 0 && ioctl(kstatfd, IM_IOC_GETNIDATA, &raw_net) >= 0) {
for (network_interfaces = 0; network_interfaces < MAX_NUM_INTERFACES; network_interfaces++) {
raw_ni_data[network_interfaces][NI_IN_ERRORS][snapshot] = raw_net.in_err[network_interfaces];
raw_ni_data[network_interfaces][NI_OUT_ERRORS][snapshot] = raw_net.out_err[network_interfaces];
raw_ni_data[network_interfaces][NI_PACKETS][snapshot] = raw_net.nr_pkt[network_interfaces];
}
return;
}
netInfo = fopen("/proc/net/dev", "r");
assert(netInfo != NULL);
// Pull out the first two lines of the file. These two lines contain
// labels for the columns.
for (int i = 0; i < 2; ++i) {
do {
c = getc(netInfo);
} while ((c != '\n') && (c != EOF));
}
for (network_interfaces = 0; network_interfaces < MAX_NUM_INTERFACES; ++network_interfaces) {
// grab the interface names (if there are leading blanks,
// then they are removed using the Strip() function)
scanCount = fscanf(netInfo, "%[^:]: %*d %d %lld %*d %*d %*d %*d %*d %*d %d %lld", ifname,
&packetIn, &raw_ni_data[network_interfaces]
[NI_IN_ERRORS][snapshot], &packetOut, &raw_ni_data[network_interfaces]
[NI_OUT_ERRORS][snapshot]);
if (scanCount == EOF) {
fclose(netInfo);
return;
}
Strip(ifname);
if (strstr(NET_IF_TO_IGNORE, ifname) != NULL) {
#ifdef _DEBUG
cout << "Ignoring network interface: " << ifname << endl;
#endif
// We are supposed to ignore this interface so...
network_interfaces--;
} else {
assert(scanCount == 5);
raw_ni_data[network_interfaces][NI_PACKETS][snapshot] = packetIn + packetOut;
}
// Skip to the next line.
do {
c = getc(netInfo);
} while ((c != '\n') && (c != EOF));
}
fclose(netInfo);
}
//
// Calculate network performance statistics based on snapshots of performance counters.
//
void Performance::Calculate_NI_Stats(Net_Results * net_results)
{
int net, stat; // Loop control variables.
// Loop through the counters and calculate performance.
net_results->ni_count = network_interfaces;
for (net = 0; net < network_interfaces; net++) {
for (stat = 0; stat < NI_RESULTS; stat++) {
// If we've never set the counter offsets, then we've never successfully retrieved
// the performance data. Set all of the values to 0.
double result;
//
// Note:
// The array time_counter[] stores time in nanoseconds.
// Earlier, we used to divide by the calculated value of timediff and then
// multiply the result by clock_ticks per second to get the NI_data per
// second which was theoretically correct (and mathematically same as what
// we are doing now) but reported wrong values while working with such
// large numbers.
//
//result = ((double) raw_ni_data[net][stat][LAST_SNAPSHOT] -
// raw_ni_data[net][stat][FIRST_SNAPSHOT]) * 1000000000.0 /
// ((double) time_counter[LAST_SNAPSHOT] - time_counter[FIRST_SNAPSHOT]);
// DF: I think the time_counter do not use nanoseconds but clock_tick
result = ((double)raw_ni_data[net][stat][LAST_SNAPSHOT] -
raw_ni_data[net][stat][FIRST_SNAPSHOT]) * clock_tick / timediff;
net_results->ni_stats[net][stat] = result;
#if PERFORMANCE_DETAILS || _DETAILS
cout << " Network interface " << net << " recorded stat " << stat << " = "
<< net_results->ni_stats[net][stat] << endl;
#endif
}
}
}
#endif /* LINUX */