]> arthur.barton.de Git - netdata.git/blob - src/proc_meminfo.c
build: migrate to autotools
[netdata.git] / src / proc_meminfo.c
1 #ifdef HAVE_CONFIG_H
2 #include <config.h>
3 #endif
4 #include <stdio.h>
5 #include <stdlib.h>
6 #include <string.h>
7
8 #include "common.h"
9 #include "log.h"
10 #include "appconfig.h"
11 #include "procfile.h"
12 #include "rrd.h"
13 #include "plugin_proc.h"
14
15 #define MAX_PROC_MEMINFO_LINE 4096
16 #define MAX_PROC_MEMINFO_NAME 1024
17
18 int do_proc_meminfo(int update_every, unsigned long long dt) {
19         static procfile *ff = NULL;
20
21         static int do_ram = -1, do_swap = -1, do_hwcorrupt = -1, do_committed = -1, do_writeback = -1, do_kernel = -1, do_slab = -1;
22
23         if(do_ram == -1)                do_ram                  = config_get_boolean("plugin:proc:/proc/meminfo", "system ram", 1);
24         if(do_swap == -1)               do_swap                 = config_get_boolean("plugin:proc:/proc/meminfo", "system swap", 1);
25         if(do_hwcorrupt == -1)  do_hwcorrupt    = config_get_boolean("plugin:proc:/proc/meminfo", "hardware corrupted ECC", 1);
26         if(do_committed == -1)  do_committed    = config_get_boolean("plugin:proc:/proc/meminfo", "committed memory", 1);
27         if(do_writeback == -1)  do_writeback    = config_get_boolean("plugin:proc:/proc/meminfo", "writeback memory", 1);
28         if(do_kernel == -1)             do_kernel               = config_get_boolean("plugin:proc:/proc/meminfo", "kernel memory", 1);
29         if(do_slab == -1)               do_slab                 = config_get_boolean("plugin:proc:/proc/meminfo", "slab memory", 1);
30
31         if(dt) {};
32
33         if(!ff) {
34                 char filename[FILENAME_MAX + 1];
35                 snprintf(filename, FILENAME_MAX, "%s%s", global_host_prefix, "/proc/meminfo");
36                 ff = procfile_open(config_get("plugin:proc:/proc/meminfo", "filename to monitor", filename), " \t:", PROCFILE_FLAG_DEFAULT);
37         }
38         if(!ff) return 1;
39
40         ff = procfile_readall(ff);
41         if(!ff) return 0; // we return 0, so that we will retry to open it next time
42
43         uint32_t lines = procfile_lines(ff), l;
44         uint32_t words;
45
46         int hwcorrupted = 0;
47
48         unsigned long long MemTotal = 0, MemFree = 0, Buffers = 0, Cached = 0, SwapCached = 0,
49                 Active = 0, Inactive = 0, ActiveAnon = 0, InactiveAnon = 0, ActiveFile = 0, InactiveFile = 0,
50                 Unevictable = 0, Mlocked = 0, SwapTotal = 0, SwapFree = 0, Dirty = 0, Writeback = 0, AnonPages = 0,
51                 Mapped = 0, Shmem = 0, Slab = 0, SReclaimable = 0, SUnreclaim = 0, KernelStack = 0, PageTables = 0,
52                 NFS_Unstable = 0, Bounce = 0, WritebackTmp = 0, CommitLimit = 0, Committed_AS = 0,
53                 VmallocTotal = 0, VmallocUsed = 0, VmallocChunk = 0,
54                 AnonHugePages = 0, HugePages_Total = 0, HugePages_Free = 0, HugePages_Rsvd = 0, HugePages_Surp = 0, Hugepagesize = 0,
55                 DirectMap4k = 0, DirectMap2M = 0, HardwareCorrupted = 0;
56
57         for(l = 0; l < lines ;l++) {
58                 words = procfile_linewords(ff, l);
59                 if(words < 2) continue;
60
61                 char *name = procfile_lineword(ff, l, 0);
62                 unsigned long long value = strtoull(procfile_lineword(ff, l, 1), NULL, 10);
63
64                      if(!MemTotal && strcmp(name, "MemTotal") == 0) MemTotal = value;
65                 else if(!MemFree && strcmp(name, "MemFree") == 0) MemFree = value;
66                 else if(!Buffers && strcmp(name, "Buffers") == 0) Buffers = value;
67                 else if(!Cached && strcmp(name, "Cached") == 0) Cached = value;
68                 else if(!SwapCached && strcmp(name, "SwapCached") == 0) SwapCached = value;
69                 else if(!Active && strcmp(name, "Active") == 0) Active = value;
70                 else if(!Inactive && strcmp(name, "Inactive") == 0) Inactive = value;
71                 else if(!ActiveAnon && strcmp(name, "ActiveAnon") == 0) ActiveAnon = value;
72                 else if(!InactiveAnon && strcmp(name, "InactiveAnon") == 0) InactiveAnon = value;
73                 else if(!ActiveFile && strcmp(name, "ActiveFile") == 0) ActiveFile = value;
74                 else if(!InactiveFile && strcmp(name, "InactiveFile") == 0) InactiveFile = value;
75                 else if(!Unevictable && strcmp(name, "Unevictable") == 0) Unevictable = value;
76                 else if(!Mlocked && strcmp(name, "Mlocked") == 0) Mlocked = value;
77                 else if(!SwapTotal && strcmp(name, "SwapTotal") == 0) SwapTotal = value;
78                 else if(!SwapFree && strcmp(name, "SwapFree") == 0) SwapFree = value;
79                 else if(!Dirty && strcmp(name, "Dirty") == 0) Dirty = value;
80                 else if(!Writeback && strcmp(name, "Writeback") == 0) Writeback = value;
81                 else if(!AnonPages && strcmp(name, "AnonPages") == 0) AnonPages = value;
82                 else if(!Mapped && strcmp(name, "Mapped") == 0) Mapped = value;
83                 else if(!Shmem && strcmp(name, "Shmem") == 0) Shmem = value;
84                 else if(!Slab && strcmp(name, "Slab") == 0) Slab = value;
85                 else if(!SReclaimable && strcmp(name, "SReclaimable") == 0) SReclaimable = value;
86                 else if(!SUnreclaim && strcmp(name, "SUnreclaim") == 0) SUnreclaim = value;
87                 else if(!KernelStack && strcmp(name, "KernelStack") == 0) KernelStack = value;
88                 else if(!PageTables && strcmp(name, "PageTables") == 0) PageTables = value;
89                 else if(!NFS_Unstable && strcmp(name, "NFS_Unstable") == 0) NFS_Unstable = value;
90                 else if(!Bounce && strcmp(name, "Bounce") == 0) Bounce = value;
91                 else if(!WritebackTmp && strcmp(name, "WritebackTmp") == 0) WritebackTmp = value;
92                 else if(!CommitLimit && strcmp(name, "CommitLimit") == 0) CommitLimit = value;
93                 else if(!Committed_AS && strcmp(name, "Committed_AS") == 0) Committed_AS = value;
94                 else if(!VmallocTotal && strcmp(name, "VmallocTotal") == 0) VmallocTotal = value;
95                 else if(!VmallocUsed && strcmp(name, "VmallocUsed") == 0) VmallocUsed = value;
96                 else if(!VmallocChunk && strcmp(name, "VmallocChunk") == 0) VmallocChunk = value;
97                 else if(!HardwareCorrupted && strcmp(name, "HardwareCorrupted") == 0) { HardwareCorrupted = value; hwcorrupted = 1; }
98                 else if(!AnonHugePages && strcmp(name, "AnonHugePages") == 0) AnonHugePages = value;
99                 else if(!HugePages_Total && strcmp(name, "HugePages_Total") == 0) HugePages_Total = value;
100                 else if(!HugePages_Free && strcmp(name, "HugePages_Free") == 0) HugePages_Free = value;
101                 else if(!HugePages_Rsvd && strcmp(name, "HugePages_Rsvd") == 0) HugePages_Rsvd = value;
102                 else if(!HugePages_Surp && strcmp(name, "HugePages_Surp") == 0) HugePages_Surp = value;
103                 else if(!Hugepagesize && strcmp(name, "Hugepagesize") == 0) Hugepagesize = value;
104                 else if(!DirectMap4k && strcmp(name, "DirectMap4k") == 0) DirectMap4k = value;
105                 else if(!DirectMap2M && strcmp(name, "DirectMap2M") == 0) DirectMap2M = value;
106         }
107
108         RRDSET *st;
109
110         // --------------------------------------------------------------------
111         
112         // http://stackoverflow.com/questions/3019748/how-to-reliably-measure-available-memory-in-linux
113         unsigned long long MemUsed = MemTotal - MemFree - Cached - Buffers;
114
115         if(do_ram) {
116                 st = rrdset_find("system.ram");
117                 if(!st) {
118                         st = rrdset_create("system", "ram", NULL, "mem", "System RAM", "MB", 200, update_every, RRDSET_TYPE_STACKED);
119
120                         rrddim_add(st, "buffers", NULL, 1, 1024, RRDDIM_ABSOLUTE);
121                         rrddim_add(st, "used",    NULL, 1, 1024, RRDDIM_ABSOLUTE);
122                         rrddim_add(st, "cached",  NULL, 1, 1024, RRDDIM_ABSOLUTE);
123                         rrddim_add(st, "free",    NULL, 1, 1024, RRDDIM_ABSOLUTE);
124                 }
125                 else rrdset_next(st);
126
127                 rrddim_set(st, "used", MemUsed);
128                 rrddim_set(st, "free", MemFree);
129                 rrddim_set(st, "cached", Cached);
130                 rrddim_set(st, "buffers", Buffers);
131                 rrdset_done(st);
132         }
133
134         // --------------------------------------------------------------------
135         
136         unsigned long long SwapUsed = SwapTotal - SwapFree;
137
138         if(do_swap) {
139                 st = rrdset_find("system.swap");
140                 if(!st) {
141                         st = rrdset_create("system", "swap", NULL, "mem", "System Swap", "MB", 201, update_every, RRDSET_TYPE_STACKED);
142                         st->isdetail = 1;
143
144                         rrddim_add(st, "free",    NULL, 1, 1024, RRDDIM_ABSOLUTE);
145                         rrddim_add(st, "used",    NULL, 1, 1024, RRDDIM_ABSOLUTE);
146                 }
147                 else rrdset_next(st);
148
149                 rrddim_set(st, "used", SwapUsed);
150                 rrddim_set(st, "free", SwapFree);
151                 rrdset_done(st);
152         }
153
154         // --------------------------------------------------------------------
155         
156         if(hwcorrupted && do_hwcorrupt) {
157                 st = rrdset_find("mem.hwcorrupt");
158                 if(!st) {
159                         st = rrdset_create("mem", "hwcorrupt", NULL, "mem", "Hardware Corrupted ECC", "MB", 9000, update_every, RRDSET_TYPE_LINE);
160                         st->isdetail = 1;
161
162                         rrddim_add(st, "HardwareCorrupted", NULL, 1, 1024, RRDDIM_ABSOLUTE);
163                 }
164                 else rrdset_next(st);
165
166                 rrddim_set(st, "HardwareCorrupted", HardwareCorrupted);
167                 rrdset_done(st);
168         }
169
170         // --------------------------------------------------------------------
171         
172         if(do_committed) {
173                 st = rrdset_find("mem.committed");
174                 if(!st) {
175                         st = rrdset_create("mem", "committed", NULL, "mem", "Committed (Allocated) Memory", "MB", 5000, update_every, RRDSET_TYPE_AREA);
176                         st->isdetail = 1;
177
178                         rrddim_add(st, "Committed_AS", NULL, 1, 1024, RRDDIM_ABSOLUTE);
179                 }
180                 else rrdset_next(st);
181
182                 rrddim_set(st, "Committed_AS", Committed_AS);
183                 rrdset_done(st);
184         }
185
186         // --------------------------------------------------------------------
187         
188         if(do_writeback) {
189                 st = rrdset_find("mem.writeback");
190                 if(!st) {
191                         st = rrdset_create("mem", "writeback", NULL, "mem", "Writeback Memory", "MB", 4000, update_every, RRDSET_TYPE_LINE);
192                         st->isdetail = 1;
193
194                         rrddim_add(st, "Dirty", NULL, 1, 1024, RRDDIM_ABSOLUTE);
195                         rrddim_add(st, "Writeback", NULL, 1, 1024, RRDDIM_ABSOLUTE);
196                         rrddim_add(st, "FuseWriteback", NULL, 1, 1024, RRDDIM_ABSOLUTE);
197                         rrddim_add(st, "NfsWriteback", NULL, 1, 1024, RRDDIM_ABSOLUTE);
198                         rrddim_add(st, "Bounce", NULL, 1, 1024, RRDDIM_ABSOLUTE);
199                 }
200                 else rrdset_next(st);
201
202                 rrddim_set(st, "Dirty", Dirty);
203                 rrddim_set(st, "Writeback", Writeback);
204                 rrddim_set(st, "FuseWriteback", WritebackTmp);
205                 rrddim_set(st, "NfsWriteback", NFS_Unstable);
206                 rrddim_set(st, "Bounce", Bounce);
207                 rrdset_done(st);
208         }
209
210         // --------------------------------------------------------------------
211         
212         if(do_kernel) {
213                 st = rrdset_find("mem.kernel");
214                 if(!st) {
215                         st = rrdset_create("mem", "kernel", NULL, "mem", "Memory Used by Kernel", "MB", 6000, update_every, RRDSET_TYPE_STACKED);
216                         st->isdetail = 1;
217
218                         rrddim_add(st, "Slab", NULL, 1, 1024, RRDDIM_ABSOLUTE);
219                         rrddim_add(st, "KernelStack", NULL, 1, 1024, RRDDIM_ABSOLUTE);
220                         rrddim_add(st, "PageTables", NULL, 1, 1024, RRDDIM_ABSOLUTE);
221                         rrddim_add(st, "VmallocUsed", NULL, 1, 1024, RRDDIM_ABSOLUTE);
222                 }
223                 else rrdset_next(st);
224
225                 rrddim_set(st, "KernelStack", KernelStack);
226                 rrddim_set(st, "Slab", Slab);
227                 rrddim_set(st, "PageTables", PageTables);
228                 rrddim_set(st, "VmallocUsed", VmallocUsed);
229                 rrdset_done(st);
230         }
231
232         // --------------------------------------------------------------------
233         
234         if(do_slab) {
235                 st = rrdset_find("mem.slab");
236                 if(!st) {
237                         st = rrdset_create("mem", "slab", NULL, "mem", "Reclaimable Kernel Memory", "MB", 6500, update_every, RRDSET_TYPE_STACKED);
238                         st->isdetail = 1;
239
240                         rrddim_add(st, "reclaimable", NULL, 1, 1024, RRDDIM_ABSOLUTE);
241                         rrddim_add(st, "unreclaimable", NULL, 1, 1024, RRDDIM_ABSOLUTE);
242                 }
243                 else rrdset_next(st);
244
245                 rrddim_set(st, "reclaimable", SReclaimable);
246                 rrddim_set(st, "unreclaimable", SUnreclaim);
247                 rrdset_done(st);
248         }
249
250         return 0;
251 }
252