proc_net_sockstat.c 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538
  1. // SPDX-License-Identifier: GPL-3.0-or-later
  2. #include "plugin_proc.h"
  3. #define PLUGIN_PROC_MODULE_NET_SOCKSTAT_NAME "/proc/net/sockstat"
  4. static struct proc_net_sockstat {
  5. kernel_uint_t sockets_used;
  6. kernel_uint_t tcp_inuse;
  7. kernel_uint_t tcp_orphan;
  8. kernel_uint_t tcp_tw;
  9. kernel_uint_t tcp_alloc;
  10. kernel_uint_t tcp_mem;
  11. kernel_uint_t udp_inuse;
  12. kernel_uint_t udp_mem;
  13. kernel_uint_t udplite_inuse;
  14. kernel_uint_t raw_inuse;
  15. kernel_uint_t frag_inuse;
  16. kernel_uint_t frag_memory;
  17. } sockstat_root = { 0 };
  18. static int read_tcp_mem(void) {
  19. static char *filename = NULL;
  20. static RRDVAR *tcp_mem_low_threshold = NULL,
  21. *tcp_mem_pressure_threshold = NULL,
  22. *tcp_mem_high_threshold = NULL;
  23. if(unlikely(!tcp_mem_low_threshold)) {
  24. tcp_mem_low_threshold = rrdvar_custom_host_variable_create(localhost, "tcp_mem_low");
  25. tcp_mem_pressure_threshold = rrdvar_custom_host_variable_create(localhost, "tcp_mem_pressure");
  26. tcp_mem_high_threshold = rrdvar_custom_host_variable_create(localhost, "tcp_mem_high");
  27. }
  28. if(unlikely(!filename)) {
  29. char buffer[FILENAME_MAX + 1];
  30. snprintfz(buffer, FILENAME_MAX, "%s/proc/sys/net/ipv4/tcp_mem", netdata_configured_host_prefix);
  31. filename = strdupz(buffer);
  32. }
  33. char buffer[200 + 1], *start, *end;
  34. if(read_file(filename, buffer, 200) != 0) return 1;
  35. buffer[200] = '\0';
  36. unsigned long long low = 0, pressure = 0, high = 0;
  37. start = buffer;
  38. low = strtoull(start, &end, 10);
  39. start = end;
  40. pressure = strtoull(start, &end, 10);
  41. start = end;
  42. high = strtoull(start, &end, 10);
  43. // fprintf(stderr, "TCP MEM low = %llu, pressure = %llu, high = %llu\n", low, pressure, high);
  44. rrdvar_custom_host_variable_set(localhost, tcp_mem_low_threshold, low * sysconf(_SC_PAGESIZE) / 1024.0);
  45. rrdvar_custom_host_variable_set(localhost, tcp_mem_pressure_threshold, pressure * sysconf(_SC_PAGESIZE) / 1024.0);
  46. rrdvar_custom_host_variable_set(localhost, tcp_mem_high_threshold, high * sysconf(_SC_PAGESIZE) / 1024.0);
  47. return 0;
  48. }
  49. static kernel_uint_t read_tcp_max_orphans(void) {
  50. static char *filename = NULL;
  51. static RRDVAR *tcp_max_orphans_var = NULL;
  52. if(unlikely(!filename)) {
  53. char buffer[FILENAME_MAX + 1];
  54. snprintfz(buffer, FILENAME_MAX, "%s/proc/sys/net/ipv4/tcp_max_orphans", netdata_configured_host_prefix);
  55. filename = strdupz(buffer);
  56. }
  57. unsigned long long tcp_max_orphans = 0;
  58. if(read_single_number_file(filename, &tcp_max_orphans) == 0) {
  59. if(unlikely(!tcp_max_orphans_var))
  60. tcp_max_orphans_var = rrdvar_custom_host_variable_create(localhost, "tcp_max_orphans");
  61. rrdvar_custom_host_variable_set(localhost, tcp_max_orphans_var, tcp_max_orphans);
  62. return tcp_max_orphans;
  63. }
  64. return 0;
  65. }
  66. int do_proc_net_sockstat(int update_every, usec_t dt) {
  67. (void)dt;
  68. static procfile *ff = NULL;
  69. static uint32_t hash_sockets = 0,
  70. hash_raw = 0,
  71. hash_frag = 0,
  72. hash_tcp = 0,
  73. hash_udp = 0,
  74. hash_udplite = 0;
  75. static long long update_constants_every = 60, update_constants_count = 0;
  76. static ARL_BASE *arl_sockets = NULL;
  77. static ARL_BASE *arl_tcp = NULL;
  78. static ARL_BASE *arl_udp = NULL;
  79. static ARL_BASE *arl_udplite = NULL;
  80. static ARL_BASE *arl_raw = NULL;
  81. static ARL_BASE *arl_frag = NULL;
  82. static int do_sockets = -1, do_tcp_sockets = -1, do_tcp_mem = -1, do_udp_sockets = -1, do_udp_mem = -1, do_udplite_sockets = -1, do_raw_sockets = -1, do_frag_sockets = -1, do_frag_mem = -1;
  83. static char *keys[7] = { NULL };
  84. static uint32_t hashes[7] = { 0 };
  85. static ARL_BASE *bases[7] = { NULL };
  86. if(unlikely(!arl_sockets)) {
  87. do_sockets = config_get_boolean_ondemand("plugin:proc:/proc/net/sockstat", "ipv4 sockets", CONFIG_BOOLEAN_AUTO);
  88. do_tcp_sockets = config_get_boolean_ondemand("plugin:proc:/proc/net/sockstat", "ipv4 TCP sockets", CONFIG_BOOLEAN_AUTO);
  89. do_tcp_mem = config_get_boolean_ondemand("plugin:proc:/proc/net/sockstat", "ipv4 TCP memory", CONFIG_BOOLEAN_AUTO);
  90. do_udp_sockets = config_get_boolean_ondemand("plugin:proc:/proc/net/sockstat", "ipv4 UDP sockets", CONFIG_BOOLEAN_AUTO);
  91. do_udp_mem = config_get_boolean_ondemand("plugin:proc:/proc/net/sockstat", "ipv4 UDP memory", CONFIG_BOOLEAN_AUTO);
  92. do_udplite_sockets = config_get_boolean_ondemand("plugin:proc:/proc/net/sockstat", "ipv4 UDPLITE sockets", CONFIG_BOOLEAN_AUTO);
  93. do_raw_sockets = config_get_boolean_ondemand("plugin:proc:/proc/net/sockstat", "ipv4 RAW sockets", CONFIG_BOOLEAN_AUTO);
  94. do_frag_sockets = config_get_boolean_ondemand("plugin:proc:/proc/net/sockstat", "ipv4 FRAG sockets", CONFIG_BOOLEAN_AUTO);
  95. do_frag_mem = config_get_boolean_ondemand("plugin:proc:/proc/net/sockstat", "ipv4 FRAG memory", CONFIG_BOOLEAN_AUTO);
  96. update_constants_every = config_get_number("plugin:proc:/proc/net/sockstat", "update constants every", update_constants_every);
  97. update_constants_count = update_constants_every;
  98. arl_sockets = arl_create("sockstat/sockets", arl_callback_str2kernel_uint_t, 60);
  99. arl_expect(arl_sockets, "used", &sockstat_root.sockets_used);
  100. arl_tcp = arl_create("sockstat/TCP", arl_callback_str2kernel_uint_t, 60);
  101. arl_expect(arl_tcp, "inuse", &sockstat_root.tcp_inuse);
  102. arl_expect(arl_tcp, "orphan", &sockstat_root.tcp_orphan);
  103. arl_expect(arl_tcp, "tw", &sockstat_root.tcp_tw);
  104. arl_expect(arl_tcp, "alloc", &sockstat_root.tcp_alloc);
  105. arl_expect(arl_tcp, "mem", &sockstat_root.tcp_mem);
  106. arl_udp = arl_create("sockstat/UDP", arl_callback_str2kernel_uint_t, 60);
  107. arl_expect(arl_udp, "inuse", &sockstat_root.udp_inuse);
  108. arl_expect(arl_udp, "mem", &sockstat_root.udp_mem);
  109. arl_udplite = arl_create("sockstat/UDPLITE", arl_callback_str2kernel_uint_t, 60);
  110. arl_expect(arl_udplite, "inuse", &sockstat_root.udplite_inuse);
  111. arl_raw = arl_create("sockstat/RAW", arl_callback_str2kernel_uint_t, 60);
  112. arl_expect(arl_raw, "inuse", &sockstat_root.raw_inuse);
  113. arl_frag = arl_create("sockstat/FRAG", arl_callback_str2kernel_uint_t, 60);
  114. arl_expect(arl_frag, "inuse", &sockstat_root.frag_inuse);
  115. arl_expect(arl_frag, "memory", &sockstat_root.frag_memory);
  116. hash_sockets = simple_hash("sockets");
  117. hash_tcp = simple_hash("TCP");
  118. hash_udp = simple_hash("UDP");
  119. hash_udplite = simple_hash("UDPLITE");
  120. hash_raw = simple_hash("RAW");
  121. hash_frag = simple_hash("FRAG");
  122. keys[0] = "sockets"; hashes[0] = hash_sockets; bases[0] = arl_sockets;
  123. keys[1] = "TCP"; hashes[1] = hash_tcp; bases[1] = arl_tcp;
  124. keys[2] = "UDP"; hashes[2] = hash_udp; bases[2] = arl_udp;
  125. keys[3] = "UDPLITE"; hashes[3] = hash_udplite; bases[3] = arl_udplite;
  126. keys[4] = "RAW"; hashes[4] = hash_raw; bases[4] = arl_raw;
  127. keys[5] = "FRAG"; hashes[5] = hash_frag; bases[5] = arl_frag;
  128. keys[6] = NULL; // terminator
  129. }
  130. update_constants_count += update_every;
  131. if(unlikely(update_constants_count > update_constants_every)) {
  132. read_tcp_max_orphans();
  133. read_tcp_mem();
  134. update_constants_count = 0;
  135. }
  136. if(unlikely(!ff)) {
  137. char filename[FILENAME_MAX + 1];
  138. snprintfz(filename, FILENAME_MAX, "%s%s", netdata_configured_host_prefix, "/proc/net/sockstat");
  139. ff = procfile_open(config_get("plugin:proc:/proc/net/sockstat", "filename to monitor", filename), " \t:", PROCFILE_FLAG_DEFAULT);
  140. if(unlikely(!ff)) return 1;
  141. }
  142. ff = procfile_readall(ff);
  143. if(unlikely(!ff)) return 0; // we return 0, so that we will retry to open it next time
  144. size_t lines = procfile_lines(ff), l;
  145. for(l = 0; l < lines ;l++) {
  146. size_t words = procfile_linewords(ff, l);
  147. char *key = procfile_lineword(ff, l, 0);
  148. uint32_t hash = simple_hash(key);
  149. int k;
  150. for(k = 0; keys[k] ; k++) {
  151. if(unlikely(hash == hashes[k] && strcmp(key, keys[k]) == 0)) {
  152. // fprintf(stderr, "KEY: '%s', l=%zu, w=1, words=%zu\n", key, l, words);
  153. ARL_BASE *arl = bases[k];
  154. arl_begin(arl);
  155. size_t w = 1;
  156. while(w + 1 < words) {
  157. char *name = procfile_lineword(ff, l, w); w++;
  158. char *value = procfile_lineword(ff, l, w); w++;
  159. // fprintf(stderr, " > NAME '%s', VALUE '%s', l=%zu, w=%zu, words=%zu\n", name, value, l, w, words);
  160. if(unlikely(arl_check(arl, name, value) != 0))
  161. break;
  162. }
  163. break;
  164. }
  165. }
  166. }
  167. // ------------------------------------------------------------------------
  168. if(do_sockets == CONFIG_BOOLEAN_YES || (do_sockets == CONFIG_BOOLEAN_AUTO &&
  169. (sockstat_root.sockets_used ||
  170. netdata_zero_metrics_enabled == CONFIG_BOOLEAN_YES))) {
  171. do_sockets = CONFIG_BOOLEAN_YES;
  172. static RRDSET *st = NULL;
  173. static RRDDIM *rd_used = NULL;
  174. if(unlikely(!st)) {
  175. st = rrdset_create_localhost(
  176. "ipv4"
  177. , "sockstat_sockets"
  178. , NULL
  179. , "sockets"
  180. , NULL
  181. , "IPv4 Sockets Used"
  182. , "sockets"
  183. , PLUGIN_PROC_NAME
  184. , PLUGIN_PROC_MODULE_NET_SOCKSTAT_NAME
  185. , NETDATA_CHART_PRIO_IPV4_SOCKETS
  186. , update_every
  187. , RRDSET_TYPE_LINE
  188. );
  189. rd_used = rrddim_add(st, "used", NULL, 1, 1, RRD_ALGORITHM_ABSOLUTE);
  190. }
  191. else rrdset_next(st);
  192. rrddim_set_by_pointer(st, rd_used, (collected_number)sockstat_root.sockets_used);
  193. rrdset_done(st);
  194. }
  195. // ------------------------------------------------------------------------
  196. if(do_tcp_sockets == CONFIG_BOOLEAN_YES || (do_tcp_sockets == CONFIG_BOOLEAN_AUTO &&
  197. (sockstat_root.tcp_inuse ||
  198. sockstat_root.tcp_orphan ||
  199. sockstat_root.tcp_tw ||
  200. sockstat_root.tcp_alloc ||
  201. netdata_zero_metrics_enabled == CONFIG_BOOLEAN_YES))) {
  202. do_tcp_sockets = CONFIG_BOOLEAN_YES;
  203. static RRDSET *st = NULL;
  204. static RRDDIM *rd_inuse = NULL,
  205. *rd_orphan = NULL,
  206. *rd_timewait = NULL,
  207. *rd_alloc = NULL;
  208. if(unlikely(!st)) {
  209. st = rrdset_create_localhost(
  210. "ipv4"
  211. , "sockstat_tcp_sockets"
  212. , NULL
  213. , "tcp"
  214. , NULL
  215. , "IPv4 TCP Sockets"
  216. , "sockets"
  217. , PLUGIN_PROC_NAME
  218. , PLUGIN_PROC_MODULE_NET_SOCKSTAT_NAME
  219. , NETDATA_CHART_PRIO_IPV4_TCP_SOCKETS
  220. , update_every
  221. , RRDSET_TYPE_LINE
  222. );
  223. rd_alloc = rrddim_add(st, "alloc", NULL, 1, 1, RRD_ALGORITHM_ABSOLUTE);
  224. rd_orphan = rrddim_add(st, "orphan", NULL, 1, 1, RRD_ALGORITHM_ABSOLUTE);
  225. rd_inuse = rrddim_add(st, "inuse", NULL, 1, 1, RRD_ALGORITHM_ABSOLUTE);
  226. rd_timewait = rrddim_add(st, "timewait", NULL, 1, 1, RRD_ALGORITHM_ABSOLUTE);
  227. }
  228. else rrdset_next(st);
  229. rrddim_set_by_pointer(st, rd_inuse, (collected_number)sockstat_root.tcp_inuse);
  230. rrddim_set_by_pointer(st, rd_orphan, (collected_number)sockstat_root.tcp_orphan);
  231. rrddim_set_by_pointer(st, rd_timewait, (collected_number)sockstat_root.tcp_tw);
  232. rrddim_set_by_pointer(st, rd_alloc, (collected_number)sockstat_root.tcp_alloc);
  233. rrdset_done(st);
  234. }
  235. // ------------------------------------------------------------------------
  236. if(do_tcp_mem == CONFIG_BOOLEAN_YES || (do_tcp_mem == CONFIG_BOOLEAN_AUTO &&
  237. (sockstat_root.tcp_mem || netdata_zero_metrics_enabled == CONFIG_BOOLEAN_YES))) {
  238. do_tcp_mem = CONFIG_BOOLEAN_YES;
  239. static RRDSET *st = NULL;
  240. static RRDDIM *rd_mem = NULL;
  241. if(unlikely(!st)) {
  242. st = rrdset_create_localhost(
  243. "ipv4"
  244. , "sockstat_tcp_mem"
  245. , NULL
  246. , "tcp"
  247. , NULL
  248. , "IPv4 TCP Sockets Memory"
  249. , "KiB"
  250. , PLUGIN_PROC_NAME
  251. , PLUGIN_PROC_MODULE_NET_SOCKSTAT_NAME
  252. , NETDATA_CHART_PRIO_IPV4_TCP_MEM
  253. , update_every
  254. , RRDSET_TYPE_AREA
  255. );
  256. rd_mem = rrddim_add(st, "mem", NULL, sysconf(_SC_PAGESIZE), 1024, RRD_ALGORITHM_ABSOLUTE);
  257. }
  258. else rrdset_next(st);
  259. rrddim_set_by_pointer(st, rd_mem, (collected_number)sockstat_root.tcp_mem);
  260. rrdset_done(st);
  261. }
  262. // ------------------------------------------------------------------------
  263. if(do_udp_sockets == CONFIG_BOOLEAN_YES || (do_udp_sockets == CONFIG_BOOLEAN_AUTO &&
  264. (sockstat_root.udp_inuse ||
  265. netdata_zero_metrics_enabled == CONFIG_BOOLEAN_YES))) {
  266. do_udp_sockets = CONFIG_BOOLEAN_YES;
  267. static RRDSET *st = NULL;
  268. static RRDDIM *rd_inuse = NULL;
  269. if(unlikely(!st)) {
  270. st = rrdset_create_localhost(
  271. "ipv4"
  272. , "sockstat_udp_sockets"
  273. , NULL
  274. , "udp"
  275. , NULL
  276. , "IPv4 UDP Sockets"
  277. , "sockets"
  278. , PLUGIN_PROC_NAME
  279. , PLUGIN_PROC_MODULE_NET_SOCKSTAT_NAME
  280. , NETDATA_CHART_PRIO_IPV4_UDP
  281. , update_every
  282. , RRDSET_TYPE_LINE
  283. );
  284. rd_inuse = rrddim_add(st, "inuse", NULL, 1, 1, RRD_ALGORITHM_ABSOLUTE);
  285. }
  286. else rrdset_next(st);
  287. rrddim_set_by_pointer(st, rd_inuse, (collected_number)sockstat_root.udp_inuse);
  288. rrdset_done(st);
  289. }
  290. // ------------------------------------------------------------------------
  291. if(do_udp_mem == CONFIG_BOOLEAN_YES || (do_udp_mem == CONFIG_BOOLEAN_AUTO &&
  292. (sockstat_root.udp_mem ||
  293. netdata_zero_metrics_enabled == CONFIG_BOOLEAN_YES))) {
  294. do_udp_mem = CONFIG_BOOLEAN_YES;
  295. static RRDSET *st = NULL;
  296. static RRDDIM *rd_mem = NULL;
  297. if(unlikely(!st)) {
  298. st = rrdset_create_localhost(
  299. "ipv4"
  300. , "sockstat_udp_mem"
  301. , NULL
  302. , "udp"
  303. , NULL
  304. , "IPv4 UDP Sockets Memory"
  305. , "KiB"
  306. , PLUGIN_PROC_NAME
  307. , PLUGIN_PROC_MODULE_NET_SOCKSTAT_NAME
  308. , NETDATA_CHART_PRIO_IPV4_UDP_MEM
  309. , update_every
  310. , RRDSET_TYPE_AREA
  311. );
  312. rd_mem = rrddim_add(st, "mem", NULL, sysconf(_SC_PAGESIZE), 1024, RRD_ALGORITHM_ABSOLUTE);
  313. }
  314. else rrdset_next(st);
  315. rrddim_set_by_pointer(st, rd_mem, (collected_number)sockstat_root.udp_mem);
  316. rrdset_done(st);
  317. }
  318. // ------------------------------------------------------------------------
  319. if(do_udplite_sockets == CONFIG_BOOLEAN_YES || (do_udplite_sockets == CONFIG_BOOLEAN_AUTO &&
  320. (sockstat_root.udplite_inuse ||
  321. netdata_zero_metrics_enabled == CONFIG_BOOLEAN_YES))) {
  322. do_udplite_sockets = CONFIG_BOOLEAN_YES;
  323. static RRDSET *st = NULL;
  324. static RRDDIM *rd_inuse = NULL;
  325. if(unlikely(!st)) {
  326. st = rrdset_create_localhost(
  327. "ipv4"
  328. , "sockstat_udplite_sockets"
  329. , NULL
  330. , "udplite"
  331. , NULL
  332. , "IPv4 UDPLITE Sockets"
  333. , "sockets"
  334. , PLUGIN_PROC_NAME
  335. , PLUGIN_PROC_MODULE_NET_SOCKSTAT_NAME
  336. , NETDATA_CHART_PRIO_IPV4_UDPLITE
  337. , update_every
  338. , RRDSET_TYPE_LINE
  339. );
  340. rd_inuse = rrddim_add(st, "inuse", NULL, 1, 1, RRD_ALGORITHM_ABSOLUTE);
  341. }
  342. else rrdset_next(st);
  343. rrddim_set_by_pointer(st, rd_inuse, (collected_number)sockstat_root.udplite_inuse);
  344. rrdset_done(st);
  345. }
  346. // ------------------------------------------------------------------------
  347. if(do_raw_sockets == CONFIG_BOOLEAN_YES || (do_raw_sockets == CONFIG_BOOLEAN_AUTO &&
  348. (sockstat_root.raw_inuse ||
  349. netdata_zero_metrics_enabled == CONFIG_BOOLEAN_YES))) {
  350. do_raw_sockets = CONFIG_BOOLEAN_YES;
  351. static RRDSET *st = NULL;
  352. static RRDDIM *rd_inuse = NULL;
  353. if(unlikely(!st)) {
  354. st = rrdset_create_localhost(
  355. "ipv4"
  356. , "sockstat_raw_sockets"
  357. , NULL
  358. , "raw"
  359. , NULL
  360. , "IPv4 RAW Sockets"
  361. , "sockets"
  362. , PLUGIN_PROC_NAME
  363. , PLUGIN_PROC_MODULE_NET_SOCKSTAT_NAME
  364. , NETDATA_CHART_PRIO_IPV4_RAW
  365. , update_every
  366. , RRDSET_TYPE_LINE
  367. );
  368. rd_inuse = rrddim_add(st, "inuse", NULL, 1, 1, RRD_ALGORITHM_ABSOLUTE);
  369. }
  370. else rrdset_next(st);
  371. rrddim_set_by_pointer(st, rd_inuse, (collected_number)sockstat_root.raw_inuse);
  372. rrdset_done(st);
  373. }
  374. // ------------------------------------------------------------------------
  375. if(do_frag_sockets == CONFIG_BOOLEAN_YES || (do_frag_sockets == CONFIG_BOOLEAN_AUTO &&
  376. (sockstat_root.frag_inuse ||
  377. netdata_zero_metrics_enabled == CONFIG_BOOLEAN_YES))) {
  378. do_frag_sockets = CONFIG_BOOLEAN_YES;
  379. static RRDSET *st = NULL;
  380. static RRDDIM *rd_inuse = NULL;
  381. if(unlikely(!st)) {
  382. st = rrdset_create_localhost(
  383. "ipv4"
  384. , "sockstat_frag_sockets"
  385. , NULL
  386. , "fragments"
  387. , NULL
  388. , "IPv4 FRAG Sockets"
  389. , "fragments"
  390. , PLUGIN_PROC_NAME
  391. , PLUGIN_PROC_MODULE_NET_SOCKSTAT_NAME
  392. , NETDATA_CHART_PRIO_IPV4_FRAGMENTS
  393. , update_every
  394. , RRDSET_TYPE_LINE
  395. );
  396. rd_inuse = rrddim_add(st, "inuse", NULL, 1, 1, RRD_ALGORITHM_ABSOLUTE);
  397. }
  398. else rrdset_next(st);
  399. rrddim_set_by_pointer(st, rd_inuse, (collected_number)sockstat_root.frag_inuse);
  400. rrdset_done(st);
  401. }
  402. // ------------------------------------------------------------------------
  403. if(do_frag_mem == CONFIG_BOOLEAN_YES || (do_frag_mem == CONFIG_BOOLEAN_AUTO &&
  404. (sockstat_root.frag_memory ||
  405. netdata_zero_metrics_enabled == CONFIG_BOOLEAN_YES))) {
  406. do_frag_mem = CONFIG_BOOLEAN_YES;
  407. static RRDSET *st = NULL;
  408. static RRDDIM *rd_mem = NULL;
  409. if(unlikely(!st)) {
  410. st = rrdset_create_localhost(
  411. "ipv4"
  412. , "sockstat_frag_mem"
  413. , NULL
  414. , "fragments"
  415. , NULL
  416. , "IPv4 FRAG Sockets Memory"
  417. , "KiB"
  418. , PLUGIN_PROC_NAME
  419. , PLUGIN_PROC_MODULE_NET_SOCKSTAT_NAME
  420. , NETDATA_CHART_PRIO_IPV4_FRAGMENTS_MEM
  421. , update_every
  422. , RRDSET_TYPE_AREA
  423. );
  424. rd_mem = rrddim_add(st, "mem", NULL, 1, 1024, RRD_ALGORITHM_ABSOLUTE);
  425. }
  426. else rrdset_next(st);
  427. rrddim_set_by_pointer(st, rd_mem, (collected_number)sockstat_root.frag_memory);
  428. rrdset_done(st);
  429. }
  430. return 0;
  431. }