proc_vmstat.c 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263
  1. // SPDX-License-Identifier: GPL-3.0-or-later
  2. #include "plugin_proc.h"
  3. #define PLUGIN_PROC_MODULE_VMSTAT_NAME "/proc/vmstat"
  4. int do_proc_vmstat(int update_every, usec_t dt) {
  5. (void)dt;
  6. static procfile *ff = NULL;
  7. static int do_swapio = -1, do_io = -1, do_pgfaults = -1, do_numa = -1;
  8. static int has_numa = -1;
  9. static ARL_BASE *arl_base = NULL;
  10. static unsigned long long numa_foreign = 0ULL;
  11. static unsigned long long numa_hint_faults = 0ULL;
  12. static unsigned long long numa_hint_faults_local = 0ULL;
  13. static unsigned long long numa_huge_pte_updates = 0ULL;
  14. static unsigned long long numa_interleave = 0ULL;
  15. static unsigned long long numa_local = 0ULL;
  16. static unsigned long long numa_other = 0ULL;
  17. static unsigned long long numa_pages_migrated = 0ULL;
  18. static unsigned long long numa_pte_updates = 0ULL;
  19. static unsigned long long pgfault = 0ULL;
  20. static unsigned long long pgmajfault = 0ULL;
  21. static unsigned long long pgpgin = 0ULL;
  22. static unsigned long long pgpgout = 0ULL;
  23. static unsigned long long pswpin = 0ULL;
  24. static unsigned long long pswpout = 0ULL;
  25. if(unlikely(!arl_base)) {
  26. do_swapio = config_get_boolean_ondemand("plugin:proc:/proc/vmstat", "swap i/o", CONFIG_BOOLEAN_AUTO);
  27. do_io = config_get_boolean("plugin:proc:/proc/vmstat", "disk i/o", 1);
  28. do_pgfaults = config_get_boolean("plugin:proc:/proc/vmstat", "memory page faults", 1);
  29. do_numa = config_get_boolean_ondemand("plugin:proc:/proc/vmstat", "system-wide numa metric summary", CONFIG_BOOLEAN_AUTO);
  30. arl_base = arl_create("vmstat", NULL, 60);
  31. arl_expect(arl_base, "pgfault", &pgfault);
  32. arl_expect(arl_base, "pgmajfault", &pgmajfault);
  33. arl_expect(arl_base, "pgpgin", &pgpgin);
  34. arl_expect(arl_base, "pgpgout", &pgpgout);
  35. arl_expect(arl_base, "pswpin", &pswpin);
  36. arl_expect(arl_base, "pswpout", &pswpout);
  37. if(do_numa == CONFIG_BOOLEAN_YES || (do_numa == CONFIG_BOOLEAN_AUTO &&
  38. (get_numa_node_count() >= 2 ||
  39. netdata_zero_metrics_enabled == CONFIG_BOOLEAN_YES))) {
  40. arl_expect(arl_base, "numa_foreign", &numa_foreign);
  41. arl_expect(arl_base, "numa_hint_faults_local", &numa_hint_faults_local);
  42. arl_expect(arl_base, "numa_hint_faults", &numa_hint_faults);
  43. arl_expect(arl_base, "numa_huge_pte_updates", &numa_huge_pte_updates);
  44. arl_expect(arl_base, "numa_interleave", &numa_interleave);
  45. arl_expect(arl_base, "numa_local", &numa_local);
  46. arl_expect(arl_base, "numa_other", &numa_other);
  47. arl_expect(arl_base, "numa_pages_migrated", &numa_pages_migrated);
  48. arl_expect(arl_base, "numa_pte_updates", &numa_pte_updates);
  49. }
  50. else {
  51. // Do not expect numa metrics when they are not needed.
  52. // By not adding them, the ARL will stop processing the file
  53. // when all the expected metrics are collected.
  54. // Also ARL will not parse their values.
  55. has_numa = 0;
  56. do_numa = CONFIG_BOOLEAN_NO;
  57. }
  58. }
  59. if(unlikely(!ff)) {
  60. char filename[FILENAME_MAX + 1];
  61. snprintfz(filename, FILENAME_MAX, "%s%s", netdata_configured_host_prefix, "/proc/vmstat");
  62. ff = procfile_open(config_get("plugin:proc:/proc/vmstat", "filename to monitor", filename), " \t:", PROCFILE_FLAG_DEFAULT);
  63. if(unlikely(!ff)) return 1;
  64. }
  65. ff = procfile_readall(ff);
  66. if(unlikely(!ff)) return 0; // we return 0, so that we will retry to open it next time
  67. size_t lines = procfile_lines(ff), l;
  68. arl_begin(arl_base);
  69. for(l = 0; l < lines ;l++) {
  70. size_t words = procfile_linewords(ff, l);
  71. if(unlikely(words < 2)) {
  72. if(unlikely(words)) error("Cannot read /proc/vmstat line %zu. Expected 2 params, read %zu.", l, words);
  73. continue;
  74. }
  75. if(unlikely(arl_check(arl_base,
  76. procfile_lineword(ff, l, 0),
  77. procfile_lineword(ff, l, 1)))) break;
  78. }
  79. // --------------------------------------------------------------------
  80. if(do_swapio == CONFIG_BOOLEAN_YES || (do_swapio == CONFIG_BOOLEAN_AUTO &&
  81. (pswpin || pswpout ||
  82. netdata_zero_metrics_enabled == CONFIG_BOOLEAN_YES))) {
  83. do_swapio = CONFIG_BOOLEAN_YES;
  84. static RRDSET *st_swapio = NULL;
  85. static RRDDIM *rd_in = NULL, *rd_out = NULL;
  86. if(unlikely(!st_swapio)) {
  87. st_swapio = rrdset_create_localhost(
  88. "system"
  89. , "swapio"
  90. , NULL
  91. , "swap"
  92. , NULL
  93. , "Swap I/O"
  94. , "KiB/s"
  95. , PLUGIN_PROC_NAME
  96. , PLUGIN_PROC_MODULE_VMSTAT_NAME
  97. , NETDATA_CHART_PRIO_SYSTEM_SWAPIO
  98. , update_every
  99. , RRDSET_TYPE_AREA
  100. );
  101. rd_in = rrddim_add(st_swapio, "in", NULL, sysconf(_SC_PAGESIZE), 1024, RRD_ALGORITHM_INCREMENTAL);
  102. rd_out = rrddim_add(st_swapio, "out", NULL, -sysconf(_SC_PAGESIZE), 1024, RRD_ALGORITHM_INCREMENTAL);
  103. }
  104. else rrdset_next(st_swapio);
  105. rrddim_set_by_pointer(st_swapio, rd_in, pswpin);
  106. rrddim_set_by_pointer(st_swapio, rd_out, pswpout);
  107. rrdset_done(st_swapio);
  108. }
  109. // --------------------------------------------------------------------
  110. if(do_io) {
  111. static RRDSET *st_io = NULL;
  112. static RRDDIM *rd_in = NULL, *rd_out = NULL;
  113. if(unlikely(!st_io)) {
  114. st_io = rrdset_create_localhost(
  115. "system"
  116. , "pgpgio"
  117. , NULL
  118. , "disk"
  119. , NULL
  120. , "Memory Paged from/to disk"
  121. , "KiB/s"
  122. , PLUGIN_PROC_NAME
  123. , PLUGIN_PROC_MODULE_VMSTAT_NAME
  124. , NETDATA_CHART_PRIO_SYSTEM_PGPGIO
  125. , update_every
  126. , RRDSET_TYPE_AREA
  127. );
  128. rd_in = rrddim_add(st_io, "in", NULL, 1, 1, RRD_ALGORITHM_INCREMENTAL);
  129. rd_out = rrddim_add(st_io, "out", NULL, -1, 1, RRD_ALGORITHM_INCREMENTAL);
  130. }
  131. else rrdset_next(st_io);
  132. rrddim_set_by_pointer(st_io, rd_in, pgpgin);
  133. rrddim_set_by_pointer(st_io, rd_out, pgpgout);
  134. rrdset_done(st_io);
  135. }
  136. // --------------------------------------------------------------------
  137. if(do_pgfaults) {
  138. static RRDSET *st_pgfaults = NULL;
  139. static RRDDIM *rd_minor = NULL, *rd_major = NULL;
  140. if(unlikely(!st_pgfaults)) {
  141. st_pgfaults = rrdset_create_localhost(
  142. "mem"
  143. , "pgfaults"
  144. , NULL
  145. , "system"
  146. , NULL
  147. , "Memory Page Faults"
  148. , "faults/s"
  149. , PLUGIN_PROC_NAME
  150. , PLUGIN_PROC_MODULE_VMSTAT_NAME
  151. , NETDATA_CHART_PRIO_MEM_SYSTEM_PGFAULTS
  152. , update_every
  153. , RRDSET_TYPE_LINE
  154. );
  155. rrdset_flag_set(st_pgfaults, RRDSET_FLAG_DETAIL);
  156. rd_minor = rrddim_add(st_pgfaults, "minor", NULL, 1, 1, RRD_ALGORITHM_INCREMENTAL);
  157. rd_major = rrddim_add(st_pgfaults, "major", NULL, -1, 1, RRD_ALGORITHM_INCREMENTAL);
  158. }
  159. else rrdset_next(st_pgfaults);
  160. rrddim_set_by_pointer(st_pgfaults, rd_minor, pgfault);
  161. rrddim_set_by_pointer(st_pgfaults, rd_major, pgmajfault);
  162. rrdset_done(st_pgfaults);
  163. }
  164. // --------------------------------------------------------------------
  165. // Ondemand criteria for NUMA. Since this won't change at run time, we
  166. // check it only once. We check whether the node count is >= 2 because
  167. // single-node systems have uninteresting statistics (since all accesses
  168. // are local).
  169. if(unlikely(has_numa == -1))
  170. has_numa = (numa_local || numa_foreign || numa_interleave || numa_other || numa_pte_updates ||
  171. numa_huge_pte_updates || numa_hint_faults || numa_hint_faults_local || numa_pages_migrated) ? 1 : 0;
  172. if(do_numa == CONFIG_BOOLEAN_YES || (do_numa == CONFIG_BOOLEAN_AUTO && has_numa)) {
  173. do_numa = CONFIG_BOOLEAN_YES;
  174. static RRDSET *st_numa = NULL;
  175. static RRDDIM *rd_local = NULL, *rd_foreign = NULL, *rd_interleave = NULL, *rd_other = NULL, *rd_pte_updates = NULL, *rd_huge_pte_updates = NULL, *rd_hint_faults = NULL, *rd_hint_faults_local = NULL, *rd_pages_migrated = NULL;
  176. if(unlikely(!st_numa)) {
  177. st_numa = rrdset_create_localhost(
  178. "mem"
  179. , "numa"
  180. , NULL
  181. , "numa"
  182. , NULL
  183. , "NUMA events"
  184. , "events/s"
  185. , PLUGIN_PROC_NAME
  186. , PLUGIN_PROC_MODULE_VMSTAT_NAME
  187. , NETDATA_CHART_PRIO_MEM_NUMA
  188. , update_every
  189. , RRDSET_TYPE_LINE
  190. );
  191. rrdset_flag_set(st_numa, RRDSET_FLAG_DETAIL);
  192. // These depend on CONFIG_NUMA in the kernel.
  193. rd_local = rrddim_add(st_numa, "local", NULL, 1, 1, RRD_ALGORITHM_INCREMENTAL);
  194. rd_foreign = rrddim_add(st_numa, "foreign", NULL, 1, 1, RRD_ALGORITHM_INCREMENTAL);
  195. rd_interleave = rrddim_add(st_numa, "interleave", NULL, 1, 1, RRD_ALGORITHM_INCREMENTAL);
  196. rd_other = rrddim_add(st_numa, "other", NULL, 1, 1, RRD_ALGORITHM_INCREMENTAL);
  197. // The following stats depend on CONFIG_NUMA_BALANCING in the
  198. // kernel.
  199. rd_pte_updates = rrddim_add(st_numa, "pte_updates", NULL, 1, 1, RRD_ALGORITHM_INCREMENTAL);
  200. rd_huge_pte_updates = rrddim_add(st_numa, "huge_pte_updates", NULL, 1, 1, RRD_ALGORITHM_INCREMENTAL);
  201. rd_hint_faults = rrddim_add(st_numa, "hint_faults", NULL, 1, 1, RRD_ALGORITHM_INCREMENTAL);
  202. rd_hint_faults_local = rrddim_add(st_numa, "hint_faults_local", NULL, 1, 1, RRD_ALGORITHM_INCREMENTAL);
  203. rd_pages_migrated = rrddim_add(st_numa, "pages_migrated", NULL, 1, 1, RRD_ALGORITHM_INCREMENTAL);
  204. }
  205. else rrdset_next(st_numa);
  206. rrddim_set_by_pointer(st_numa, rd_local, numa_local);
  207. rrddim_set_by_pointer(st_numa, rd_foreign, numa_foreign);
  208. rrddim_set_by_pointer(st_numa, rd_interleave, numa_interleave);
  209. rrddim_set_by_pointer(st_numa, rd_other, numa_other);
  210. rrddim_set_by_pointer(st_numa, rd_pte_updates, numa_pte_updates);
  211. rrddim_set_by_pointer(st_numa, rd_huge_pte_updates, numa_huge_pte_updates);
  212. rrddim_set_by_pointer(st_numa, rd_hint_faults, numa_hint_faults);
  213. rrddim_set_by_pointer(st_numa, rd_hint_faults_local, numa_hint_faults_local);
  214. rrddim_set_by_pointer(st_numa, rd_pages_migrated, numa_pages_migrated);
  215. rrdset_done(st_numa);
  216. }
  217. return 0;
  218. }