NetworkIOMeter.c 5.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179
  1. /*
  2. htop - NetworkIOMeter.c
  3. (C) 2020-2023 htop dev team
  4. Released under the GNU GPLv2+, see the COPYING file
  5. in the source distribution for its full text.
  6. */
  7. #include "config.h" // IWYU pragma: keep
  8. #include "NetworkIOMeter.h"
  9. #include <stdbool.h>
  10. #include "CRT.h"
  11. #include "Machine.h"
  12. #include "Macros.h"
  13. #include "Meter.h"
  14. #include "Object.h"
  15. #include "Platform.h"
  16. #include "RichString.h"
  17. #include "Row.h"
  18. #include "XUtils.h"
  19. static const int NetworkIOMeter_attributes[] = {
  20. METER_VALUE_IOREAD,
  21. METER_VALUE_IOWRITE,
  22. };
  23. static MeterRateStatus status = RATESTATUS_INIT;
  24. static double cached_rxb_diff;
  25. static char cached_rxb_diff_str[6];
  26. static uint32_t cached_rxp_diff;
  27. static double cached_txb_diff;
  28. static char cached_txb_diff_str[6];
  29. static uint32_t cached_txp_diff;
  30. static void NetworkIOMeter_updateValues(Meter* this) {
  31. const Machine* host = this->host;
  32. static uint64_t cached_last_update = 0;
  33. uint64_t passedTimeInMs = host->realtimeMs - cached_last_update;
  34. bool hasNewData = false;
  35. NetworkIOData data;
  36. /* update only every 500ms to have a sane span for rate calculation */
  37. if (passedTimeInMs > 500) {
  38. hasNewData = Platform_getNetworkIO(&data);
  39. if (!hasNewData) {
  40. status = RATESTATUS_NODATA;
  41. } else if (cached_last_update == 0) {
  42. status = RATESTATUS_INIT;
  43. } else if (passedTimeInMs > 30000) {
  44. status = RATESTATUS_STALE;
  45. } else {
  46. status = RATESTATUS_DATA;
  47. }
  48. cached_last_update = host->realtimeMs;
  49. }
  50. if (hasNewData) {
  51. static uint64_t cached_rxb_total;
  52. static uint64_t cached_rxp_total;
  53. static uint64_t cached_txb_total;
  54. static uint64_t cached_txp_total;
  55. if (status != RATESTATUS_INIT) {
  56. uint64_t diff;
  57. if (data.bytesReceived > cached_rxb_total) {
  58. diff = data.bytesReceived - cached_rxb_total;
  59. diff = (1000 * diff) / passedTimeInMs; /* convert to B/s */
  60. cached_rxb_diff = diff;
  61. } else {
  62. cached_rxb_diff = 0;
  63. }
  64. Meter_humanUnit(cached_rxb_diff_str, cached_rxb_diff / ONE_K, sizeof(cached_rxb_diff_str));
  65. if (data.packetsReceived > cached_rxp_total) {
  66. diff = data.packetsReceived - cached_rxp_total;
  67. diff = (1000 * diff) / passedTimeInMs; /* convert to pkts/s */
  68. cached_rxp_diff = (uint32_t)diff;
  69. } else {
  70. cached_rxp_diff = 0;
  71. }
  72. if (data.bytesTransmitted > cached_txb_total) {
  73. diff = data.bytesTransmitted - cached_txb_total;
  74. diff = (1000 * diff) / passedTimeInMs; /* convert to B/s */
  75. cached_txb_diff = diff;
  76. } else {
  77. cached_txb_diff = 0;
  78. }
  79. Meter_humanUnit(cached_txb_diff_str, cached_txb_diff / ONE_K, sizeof(cached_txb_diff_str));
  80. if (data.packetsTransmitted > cached_txp_total) {
  81. diff = data.packetsTransmitted - cached_txp_total;
  82. diff = (1000 * diff) / passedTimeInMs; /* convert to pkts/s */
  83. cached_txp_diff = (uint32_t)diff;
  84. } else {
  85. cached_txp_diff = 0;
  86. }
  87. }
  88. cached_rxb_total = data.bytesReceived;
  89. cached_rxp_total = data.packetsReceived;
  90. cached_txb_total = data.bytesTransmitted;
  91. cached_txp_total = data.packetsTransmitted;
  92. }
  93. this->values[0] = cached_rxb_diff;
  94. this->values[1] = cached_txb_diff;
  95. if (cached_rxb_diff + cached_txb_diff > this->total) {
  96. this->total = cached_rxb_diff + cached_txb_diff;
  97. }
  98. if (status == RATESTATUS_NODATA) {
  99. xSnprintf(this->txtBuffer, sizeof(this->txtBuffer), "no data");
  100. return;
  101. }
  102. if (status == RATESTATUS_INIT) {
  103. xSnprintf(this->txtBuffer, sizeof(this->txtBuffer), "init");
  104. return;
  105. }
  106. if (status == RATESTATUS_STALE) {
  107. xSnprintf(this->txtBuffer, sizeof(this->txtBuffer), "stale");
  108. return;
  109. }
  110. xSnprintf(this->txtBuffer, sizeof(this->txtBuffer), "rx:%siB/s tx:%siB/s %u/%upkts/s",
  111. cached_rxb_diff_str, cached_txb_diff_str, cached_rxp_diff, cached_txp_diff);
  112. }
  113. static void NetworkIOMeter_display(ATTR_UNUSED const Object* cast, RichString* out) {
  114. switch (status) {
  115. case RATESTATUS_NODATA:
  116. RichString_writeAscii(out, CRT_colors[METER_VALUE_ERROR], "no data");
  117. return;
  118. case RATESTATUS_INIT:
  119. RichString_writeAscii(out, CRT_colors[METER_VALUE], "initializing...");
  120. return;
  121. case RATESTATUS_STALE:
  122. RichString_writeAscii(out, CRT_colors[METER_VALUE_WARN], "stale data");
  123. return;
  124. case RATESTATUS_DATA:
  125. break;
  126. }
  127. char buffer[64];
  128. RichString_writeAscii(out, CRT_colors[METER_TEXT], "rx: ");
  129. RichString_appendAscii(out, CRT_colors[METER_VALUE_IOREAD], cached_rxb_diff_str);
  130. RichString_appendAscii(out, CRT_colors[METER_VALUE_IOREAD], "iB/s");
  131. RichString_appendAscii(out, CRT_colors[METER_TEXT], " tx: ");
  132. RichString_appendAscii(out, CRT_colors[METER_VALUE_IOWRITE], cached_txb_diff_str);
  133. RichString_appendAscii(out, CRT_colors[METER_VALUE_IOWRITE], "iB/s");
  134. int len = xSnprintf(buffer, sizeof(buffer), " (%u/%u pkts/s) ", cached_rxp_diff, cached_txp_diff);
  135. RichString_appendnAscii(out, CRT_colors[METER_TEXT], buffer, len);
  136. }
  137. const MeterClass NetworkIOMeter_class = {
  138. .super = {
  139. .extends = Class(Meter),
  140. .delete = Meter_delete,
  141. .display = NetworkIOMeter_display
  142. },
  143. .updateValues = NetworkIOMeter_updateValues,
  144. .defaultMode = TEXT_METERMODE,
  145. .supportedModes = METERMODE_DEFAULT_SUPPORTED,
  146. .maxItems = 2,
  147. .total = 100.0,
  148. .attributes = NetworkIOMeter_attributes,
  149. .name = "NetworkIO",
  150. .uiName = "Network IO",
  151. .caption = "Network: "
  152. };