2020-09-13 20:46:34 +03:00
|
|
|
/*
|
|
|
|
htop - DiskIOMeter.c
|
2020-12-04 15:42:00 +03:00
|
|
|
(C) 2020 htop dev team
|
2021-09-22 12:33:00 +03:00
|
|
|
Released under the GNU GPLv2+, see the COPYING file
|
2020-09-13 20:46:34 +03:00
|
|
|
in the source distribution for its full text.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "DiskIOMeter.h"
|
|
|
|
|
2020-11-18 16:26:30 +03:00
|
|
|
#include <stdbool.h>
|
2020-09-19 14:55:23 +03:00
|
|
|
#include <stdio.h>
|
2020-09-13 20:46:34 +03:00
|
|
|
|
|
|
|
#include "CRT.h"
|
2020-09-19 14:55:23 +03:00
|
|
|
#include "Macros.h"
|
2021-12-04 20:08:59 +03:00
|
|
|
#include "Meter.h"
|
2020-09-19 14:55:23 +03:00
|
|
|
#include "Object.h"
|
2020-09-13 20:46:34 +03:00
|
|
|
#include "Platform.h"
|
2021-04-29 18:12:43 +03:00
|
|
|
#include "ProcessList.h"
|
2020-09-19 14:55:23 +03:00
|
|
|
#include "RichString.h"
|
2020-10-14 21:21:09 +03:00
|
|
|
#include "XUtils.h"
|
2020-09-13 20:46:34 +03:00
|
|
|
|
|
|
|
|
|
|
|
static const int DiskIOMeter_attributes[] = {
|
2020-10-16 20:44:53 +03:00
|
|
|
METER_VALUE_NOTICE,
|
|
|
|
METER_VALUE_IOREAD,
|
|
|
|
METER_VALUE_IOWRITE,
|
2020-09-13 20:46:34 +03:00
|
|
|
};
|
|
|
|
|
2021-12-04 20:08:59 +03:00
|
|
|
static MeterRateStatus status = RATESTATUS_INIT;
|
2021-03-02 04:09:29 +03:00
|
|
|
static uint32_t cached_read_diff;
|
|
|
|
static uint32_t cached_write_diff;
|
2021-03-01 04:10:18 +03:00
|
|
|
static double cached_utilisation_diff;
|
2020-09-13 20:46:34 +03:00
|
|
|
|
2020-10-06 14:13:16 +03:00
|
|
|
static void DiskIOMeter_updateValues(Meter* this) {
|
2021-03-23 09:27:05 +03:00
|
|
|
const ProcessList* pl = this->pl;
|
2020-09-13 20:46:34 +03:00
|
|
|
|
2021-03-23 09:27:05 +03:00
|
|
|
static uint64_t cached_last_update;
|
2021-03-30 07:55:48 +03:00
|
|
|
uint64_t passedTimeInMs = pl->realtimeMs - cached_last_update;
|
2020-09-13 20:46:34 +03:00
|
|
|
|
2021-12-04 20:08:59 +03:00
|
|
|
/* update only every 500ms to have a sane span for rate calculation */
|
2020-10-20 22:40:51 +03:00
|
|
|
if (passedTimeInMs > 500) {
|
2021-03-02 04:09:29 +03:00
|
|
|
static uint64_t cached_read_total;
|
|
|
|
static uint64_t cached_write_total;
|
|
|
|
static uint64_t cached_msTimeSpend_total;
|
|
|
|
uint64_t diff;
|
2020-11-22 02:53:12 +03:00
|
|
|
|
2020-10-21 18:06:32 +03:00
|
|
|
DiskIOData data;
|
2021-12-04 20:08:59 +03:00
|
|
|
if (!Platform_getDiskIO(&data)) {
|
|
|
|
status = RATESTATUS_NODATA;
|
|
|
|
} else if (cached_last_update == 0) {
|
|
|
|
status = RATESTATUS_INIT;
|
|
|
|
} else if (passedTimeInMs > 30000) {
|
|
|
|
status = RATESTATUS_STALE;
|
|
|
|
} else {
|
|
|
|
status = RATESTATUS_DATA;
|
|
|
|
}
|
|
|
|
|
|
|
|
cached_last_update = pl->realtimeMs;
|
2020-09-13 20:46:34 +03:00
|
|
|
|
2021-12-04 20:08:59 +03:00
|
|
|
if (status == RATESTATUS_NODATA) {
|
2020-10-06 14:13:16 +03:00
|
|
|
xSnprintf(this->txtBuffer, sizeof(this->txtBuffer), "no data");
|
2020-10-20 22:40:51 +03:00
|
|
|
return;
|
|
|
|
}
|
2020-09-13 20:46:34 +03:00
|
|
|
|
2020-11-02 16:46:42 +03:00
|
|
|
if (data.totalBytesRead > cached_read_total) {
|
2021-03-01 04:10:18 +03:00
|
|
|
diff = data.totalBytesRead - cached_read_total;
|
|
|
|
diff /= 1024; /* Meter_humanUnit() expects unit in kilo */
|
2021-03-02 04:09:29 +03:00
|
|
|
cached_read_diff = (uint32_t)diff;
|
2020-11-02 16:46:42 +03:00
|
|
|
} else {
|
2021-03-02 04:09:29 +03:00
|
|
|
cached_read_diff = 0;
|
2020-11-02 16:46:42 +03:00
|
|
|
}
|
2020-10-21 18:06:32 +03:00
|
|
|
cached_read_total = data.totalBytesRead;
|
2020-09-13 20:46:34 +03:00
|
|
|
|
2020-11-02 16:46:42 +03:00
|
|
|
if (data.totalBytesWritten > cached_write_total) {
|
2021-03-01 04:10:18 +03:00
|
|
|
diff = data.totalBytesWritten - cached_write_total;
|
|
|
|
diff /= 1024; /* Meter_humanUnit() expects unit in kilo */
|
2021-03-02 04:09:29 +03:00
|
|
|
cached_write_diff = (uint32_t)diff;
|
2020-11-02 16:46:42 +03:00
|
|
|
} else {
|
2021-03-02 04:09:29 +03:00
|
|
|
cached_write_diff = 0;
|
2020-11-02 16:46:42 +03:00
|
|
|
}
|
2020-10-21 18:06:32 +03:00
|
|
|
cached_write_total = data.totalBytesWritten;
|
2020-09-13 20:46:34 +03:00
|
|
|
|
2020-11-02 16:46:42 +03:00
|
|
|
if (data.totalMsTimeSpend > cached_msTimeSpend_total) {
|
2021-03-01 04:10:18 +03:00
|
|
|
diff = data.totalMsTimeSpend - cached_msTimeSpend_total;
|
|
|
|
cached_utilisation_diff = 100.0 * (double)diff / passedTimeInMs;
|
2020-11-02 16:46:42 +03:00
|
|
|
} else {
|
|
|
|
cached_utilisation_diff = 0.0;
|
|
|
|
}
|
2020-10-21 18:06:32 +03:00
|
|
|
cached_msTimeSpend_total = data.totalMsTimeSpend;
|
2020-09-13 20:46:34 +03:00
|
|
|
}
|
|
|
|
|
2021-12-04 20:08:59 +03:00
|
|
|
if (status == RATESTATUS_INIT) {
|
|
|
|
xSnprintf(this->txtBuffer, sizeof(this->txtBuffer), "init");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (status == RATESTATUS_STALE) {
|
|
|
|
xSnprintf(this->txtBuffer, sizeof(this->txtBuffer), "stale");
|
|
|
|
return;
|
2021-10-30 11:57:14 +03:00
|
|
|
}
|
|
|
|
|
2020-09-13 20:46:34 +03:00
|
|
|
this->values[0] = cached_utilisation_diff;
|
|
|
|
this->total = MAXIMUM(this->values[0], 100.0); /* fix total after (initial) spike */
|
|
|
|
|
|
|
|
char bufferRead[12], bufferWrite[12];
|
|
|
|
Meter_humanUnit(bufferRead, cached_read_diff, sizeof(bufferRead));
|
|
|
|
Meter_humanUnit(bufferWrite, cached_write_diff, sizeof(bufferWrite));
|
2020-10-06 14:13:16 +03:00
|
|
|
snprintf(this->txtBuffer, sizeof(this->txtBuffer), "%sB %sB %.1f%%", bufferRead, bufferWrite, cached_utilisation_diff);
|
2020-09-13 20:46:34 +03:00
|
|
|
}
|
|
|
|
|
2021-02-15 19:42:22 +03:00
|
|
|
static void DiskIOMeter_display(ATTR_UNUSED const Object* cast, RichString* out) {
|
2021-12-04 20:08:59 +03:00
|
|
|
switch (status) {
|
|
|
|
case RATESTATUS_NODATA:
|
2020-12-04 16:44:57 +03:00
|
|
|
RichString_writeAscii(out, CRT_colors[METER_VALUE_ERROR], "no data");
|
2020-10-20 22:40:51 +03:00
|
|
|
return;
|
2021-12-04 20:08:59 +03:00
|
|
|
case RATESTATUS_INIT:
|
|
|
|
RichString_writeAscii(out, CRT_colors[METER_VALUE], "initializing...");
|
|
|
|
return;
|
|
|
|
case RATESTATUS_STALE:
|
|
|
|
RichString_writeAscii(out, CRT_colors[METER_VALUE_WARN], "stale data");
|
|
|
|
return;
|
|
|
|
case RATESTATUS_DATA:
|
|
|
|
break;
|
2020-10-20 22:40:51 +03:00
|
|
|
}
|
|
|
|
|
2020-09-13 20:46:34 +03:00
|
|
|
char buffer[16];
|
2021-04-14 21:47:42 +03:00
|
|
|
int len;
|
2020-09-13 20:46:34 +03:00
|
|
|
|
2020-10-16 20:44:53 +03:00
|
|
|
int color = cached_utilisation_diff > 40.0 ? METER_VALUE_NOTICE : METER_VALUE;
|
2021-04-14 21:47:42 +03:00
|
|
|
len = xSnprintf(buffer, sizeof(buffer), "%.1f%%", cached_utilisation_diff);
|
|
|
|
RichString_appendnAscii(out, CRT_colors[color], buffer, len);
|
2020-09-13 20:46:34 +03:00
|
|
|
|
2020-12-04 16:44:57 +03:00
|
|
|
RichString_appendAscii(out, CRT_colors[METER_TEXT], " read: ");
|
2020-09-13 20:46:34 +03:00
|
|
|
Meter_humanUnit(buffer, cached_read_diff, sizeof(buffer));
|
2020-12-04 16:44:57 +03:00
|
|
|
RichString_appendAscii(out, CRT_colors[METER_VALUE_IOREAD], buffer);
|
2020-09-13 20:46:34 +03:00
|
|
|
|
2020-12-04 16:44:57 +03:00
|
|
|
RichString_appendAscii(out, CRT_colors[METER_TEXT], " write: ");
|
2020-09-13 20:46:34 +03:00
|
|
|
Meter_humanUnit(buffer, cached_write_diff, sizeof(buffer));
|
2020-12-04 16:44:57 +03:00
|
|
|
RichString_appendAscii(out, CRT_colors[METER_VALUE_IOWRITE], buffer);
|
2020-09-13 20:46:34 +03:00
|
|
|
}
|
|
|
|
|
2020-10-05 14:19:50 +03:00
|
|
|
const MeterClass DiskIOMeter_class = {
|
2020-09-13 20:46:34 +03:00
|
|
|
.super = {
|
|
|
|
.extends = Class(Meter),
|
|
|
|
.delete = Meter_delete,
|
2021-02-15 19:42:22 +03:00
|
|
|
.display = DiskIOMeter_display
|
2020-09-13 20:46:34 +03:00
|
|
|
},
|
|
|
|
.updateValues = DiskIOMeter_updateValues,
|
|
|
|
.defaultMode = TEXT_METERMODE,
|
|
|
|
.maxItems = 1,
|
|
|
|
.total = 100.0,
|
|
|
|
.attributes = DiskIOMeter_attributes,
|
|
|
|
.name = "DiskIO",
|
|
|
|
.uiName = "Disk IO",
|
|
|
|
.caption = "Disk IO: "
|
|
|
|
};
|