dockapps/wmmemload/src/mem_linux.c
Doug Torrance 52b638616c wmmemload: Read memory correctly with Linux kernel version >= 3.14.
Beginning with Linux kernel version 3.14, a new MemAvailable line appears
in /proc/meminfo[1]. This caused wmmemload to look for Cached and Buffers one
line off of their correct location.  The problem was reported in [2].

[1] https://git.kernel.org/cgit/linux/kernel/git/torvalds/linux.git/commit/?id=34e431b0ae398fc54ea69ff85ec700722c9da773
[2] https://bugs.debian.org/cgi-bin/bugreport.cgi?bug=749216
2014-10-10 10:09:53 +01:00

195 lines
4.6 KiB
C

/*
* mem_linux.c - module to get memory/swap usages, for GNU/Linxu
*
* Copyright(C) 2001,2002 Seiichi SATO <ssato@sh.rim.or.jp>
* Copyright(C) 2001 John McCutchan <ttb@tentacle.dhs.org>
*
* licensed under the GPL
*/
#ifdef HAVE_CONFIG_H
#include "config.h"
#endif
#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#if defined(HAVE_STRING_H)
#include <string.h>
#elif defined(HAVE_STRINGS_H)
#include <strings.h>
#endif
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <ctype.h>
#include <sys/utsname.h>
#include "mem.h"
static enum {
BEFORE_2_5_1,
BETWEEN_2_5_1_AND_3_14,
AFTER_3_14
} format;
#ifdef DEBUG
# define INLINE_STATIC static
#else
# define INLINE_STATIC static inline
#endif
/* initialize function */
void mem_init(void)
{
struct utsname un;
int version, patchlevel, sublevel;
/* get kernel version */
if (uname(&un) == -1)
perror("uname()");
sscanf(un.release, "%d.%d.%d", &version, &patchlevel, &sublevel);
/* new format ? (kernel >= 3.14 or 2.5.1pre?) */
/* see linux/fs/proc/proc_misc.c */
/* or linux/fs/proc/meminfo.c */
if ((version == 3 && patchlevel >= 14) || version > 3)
format = AFTER_3_14;
else if ((version == 2 && patchlevel >= 5 && sublevel >= 1) ||
(version == 2 && patchlevel >= 6 && sublevel >= 0) ||
version > 2)
format = BETWEEN_2_5_1_AND_3_14;
else
format = BEFORE_2_5_1;
}
INLINE_STATIC char *skip_line(const char *p)
{
while (*p != '\n')
p++;
return (char *) ++p;
}
INLINE_STATIC char *skip_token(const char *p)
{
while (isspace(*p))
p++;
while (*p && !isspace(*p))
p++;
return (char *)p;
}
INLINE_STATIC char *skip_multiple_token(const char *p, int count)
{
int i;
for (i = 0; i < count; i++)
p = skip_token(p);
return (char *)p;
}
/* return mem/swap usage in percent 0 to 100 */
void mem_getusage(int *per_mem, int *per_swap, const struct mem_options *opts)
{
char buffer[BUFSIZ], *p;
int fd, len, i;
u_int64_t mtotal, mused, mfree, mbuffer, mcached;
u_int64_t stotal, sused, sfree, scached = 0;
/* read /proc/meminfo */
fd = open("/proc/meminfo", O_RDONLY);
if (fd < 0) {
perror("can't open /proc/meminfo");
exit(1);
}
len = read(fd, buffer, BUFSIZ - 1);
if (len < 0) {
perror("can't read /proc/meminfo");
exit(1);
}
close(fd);
buffer[len] = '\0';
p = buffer;
switch (format) {
case (BEFORE_2_5_1):
/* skip 3 lines */
for (i = 0; i < 3; i++)
p = skip_line(p);
p = skip_token(p);
/* examine each line of file */
mtotal = strtoul(p, &p, 0); p = skip_multiple_token(p, 2);
mfree = strtoul(p, &p, 0); p = skip_multiple_token(p, 5);
mbuffer = strtoul(p, &p, 0); p = skip_multiple_token(p, 2);
mcached = strtoul(p, &p, 0); p = skip_multiple_token(p, 2);
scached = strtoul(p, &p, 0);
break;
case (BETWEEN_2_5_1_AND_3_14):
p = skip_token(p);
/* examine each line of file */
mtotal = strtoul(p, &p, 0); p = skip_multiple_token(p, 2);
mfree = strtoul(p, &p, 0); p = skip_multiple_token(p, 2);
mbuffer = strtoul(p, &p, 0); p = skip_multiple_token(p, 2);
mcached = strtoul(p, &p, 0); p = skip_multiple_token(p, 2);
scached = strtoul(p, &p, 0);
break;
case (AFTER_3_14):
p = skip_token(p);
/* examine each line of file */
mtotal = strtoul(p, &p, 0); p = skip_multiple_token(p, 2);
mfree = strtoul(p, &p, 0);
p = skip_multiple_token(p, 5); /* skip MemAvailable line */
mbuffer = strtoul(p, &p, 0); p = skip_multiple_token(p, 2);
mcached = strtoul(p, &p, 0); p = skip_multiple_token(p, 2);
scached = strtoul(p, &p, 0);
break;
}
/* skip N lines and examine info about swap */
/* kernel-2.4.2:N=8 2.4.16:N=7 */
while (isprint(p[0])) {
p = skip_line(p);
if (strncmp(p, "SwapTotal", 9) == 0)
break;
}
p = skip_token(p);
stotal = strtoul(p, &p, 0); p = skip_multiple_token(p, 2);
sfree = strtoul(p, &p, 0);
/* calculate memory usage in percent */
mused = mtotal - mfree;
if (opts->ignore_buffers)
mused -= mbuffer;
if (opts->ignore_cached)
mused -= mcached;
*per_mem = 100 * (double) mused / (double) mtotal;
/* calculate swap usage in percent */
sused = stotal - sfree;
if (opts->ignore_cached)
sused -= scached;
if (!stotal)
*per_swap = 0;
else
*per_swap = 100 * (double) sused / (double) stotal;
#if DEBUG
printf("-----------------------\n");
printf("MemTotal: %12ld\n", (unsigned long)mtotal);
printf("MemFree: %12ld\n", (unsigned long)mfree);
printf("Buffers: %12ld\n", (unsigned long)mbuffer);
printf("Cached: %12ld\n", (unsigned long)mcached);
printf("SwapTotal: %12ld\n", (unsigned long)stotal);
printf("SwapFree: %12ld\n", (unsigned long)sfree);
printf("SwapCached:%12ld\n", (unsigned long)scached);
printf("-----------------------\n\n");
#endif
}