264 lines
8.8 KiB
C
264 lines
8.8 KiB
C
// vim:ts=4:sw=4:expandtab
|
|
#include <config.h>
|
|
#include <ctype.h>
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <yajl/yajl_gen.h>
|
|
#include <yajl/yajl_version.h>
|
|
#include <sys/param.h>
|
|
#include <sys/sysctl.h>
|
|
#include "i3status.h"
|
|
|
|
#define MAX_DECIMALS 4
|
|
#define STRING_SIZE ((sizeof "1023. TiB") + MAX_DECIMALS)
|
|
|
|
#define BINARY_BASE 1024UL
|
|
|
|
static const char *const iec_symbols[] = {"B", "KiB", "MiB", "GiB", "TiB"};
|
|
#define MAX_EXPONENT ((sizeof iec_symbols / sizeof *iec_symbols) - 1)
|
|
|
|
struct print_mem_info {
|
|
unsigned long ram_total;
|
|
unsigned long ram_free;
|
|
unsigned long ram_used;
|
|
unsigned long ram_available;
|
|
unsigned long ram_buffers;
|
|
unsigned long ram_cached;
|
|
unsigned long ram_shared;
|
|
char *output_color;
|
|
};
|
|
|
|
/*
|
|
* Prints the given amount of bytes in a human readable manner.
|
|
*
|
|
*/
|
|
static int print_bytes_human(char *outwalk, unsigned long bytes, const char *unit, const int decimals) {
|
|
double base = bytes;
|
|
size_t exponent = 0;
|
|
while (base >= BINARY_BASE && exponent < MAX_EXPONENT) {
|
|
if (strcasecmp(unit, iec_symbols[exponent]) == 0) {
|
|
break;
|
|
}
|
|
|
|
base /= BINARY_BASE;
|
|
exponent += 1;
|
|
}
|
|
const int prec = decimals > MAX_DECIMALS ? MAX_DECIMALS : decimals;
|
|
return sprintf(outwalk, "%.*f %s", prec, base, iec_symbols[exponent]);
|
|
}
|
|
|
|
static int print_percentage(char *outwalk, float percent) {
|
|
return snprintf(outwalk, STRING_SIZE, "%.1f%s", percent, pct_mark);
|
|
}
|
|
|
|
static unsigned long page2byte (unsigned pageCnt) {
|
|
return pageCnt * getpagesize ();
|
|
}
|
|
|
|
/*
|
|
* Convert a string to its absolute representation based on the total
|
|
* memory of `mem_total`.
|
|
*
|
|
* The string can contain any percentage values, which then return a
|
|
* the value of `mem_amount` in relation to `mem_total`.
|
|
* Alternatively an absolute value can be given, suffixed with an iec
|
|
* symbol.
|
|
*
|
|
*/
|
|
static unsigned long memory_absolute(const char *mem_amount, const unsigned long mem_total) {
|
|
char *endptr;
|
|
unsigned long amount = strtoul(mem_amount, &endptr, 10);
|
|
|
|
while (endptr[0] != '\0' && isspace(endptr[0]))
|
|
endptr++;
|
|
|
|
switch (endptr[0]) {
|
|
case 'T':
|
|
case 't':
|
|
amount *= BINARY_BASE;
|
|
/* fall-through */
|
|
case 'G':
|
|
case 'g':
|
|
amount *= BINARY_BASE;
|
|
/* fall-through */
|
|
case 'M':
|
|
case 'm':
|
|
amount *= BINARY_BASE;
|
|
/* fall-through */
|
|
case 'K':
|
|
case 'k':
|
|
amount *= BINARY_BASE;
|
|
break;
|
|
case '%':
|
|
amount = mem_total * amount / 100;
|
|
break;
|
|
}
|
|
|
|
return amount;
|
|
}
|
|
|
|
static void handle_used_method(struct print_mem_info *minfo, memory_ctx_t *ctx) {
|
|
if (BEGINS_WITH(ctx->memory_used_method, "memavailable")) {
|
|
minfo->ram_used = minfo->ram_total - minfo->ram_available;
|
|
} else if (BEGINS_WITH(ctx->memory_used_method, "classical")) {
|
|
minfo->ram_used = minfo->ram_total - minfo->ram_free - minfo->ram_buffers - minfo->ram_cached;
|
|
} else {
|
|
die("Unexpected value: memory_used_method = %s", ctx->memory_used_method);
|
|
}
|
|
}
|
|
|
|
static void handle_threshold(struct print_mem_info *minfo, memory_ctx_t *ctx) {
|
|
if (ctx->threshold_degraded) {
|
|
const unsigned long threshold = memory_absolute(ctx->threshold_degraded, minfo->ram_total);
|
|
if (minfo->ram_available < threshold) {
|
|
minfo->output_color = "color_degraded";
|
|
}
|
|
}
|
|
|
|
if (ctx->threshold_critical) {
|
|
const unsigned long threshold = memory_absolute(ctx->threshold_critical, minfo->ram_total);
|
|
if (minfo->ram_available < threshold) {
|
|
minfo->output_color = "color_bad";
|
|
}
|
|
}
|
|
}
|
|
|
|
static void get_memory_info(struct print_mem_info *minfo, memory_ctx_t *ctx) {
|
|
char *outwalk = ctx->buf;
|
|
#if defined(__linux__)
|
|
int unread_fields = 6;
|
|
|
|
minfo->output_color = NULL
|
|
|
|
FILE *file = fopen("/proc/meminfo", "r");
|
|
if (!file) {
|
|
goto error;
|
|
}
|
|
for (char line[128]; fgets(line, sizeof line, file);) {
|
|
if (BEGINS_WITH(line, "MemTotal:")) {
|
|
minfo->ram_total = strtoul(line + strlen("MemTotal:"), NULL, 10);
|
|
} else if (BEGINS_WITH(line, "MemFree:")) {
|
|
minfo->ram_free = strtoul(line + strlen("MemFree:"), NULL, 10);
|
|
} else if (BEGINS_WITH(line, "MemAvailable:")) {
|
|
minfo->ram_available = strtoul(line + strlen("MemAvailable:"), NULL, 10);
|
|
} else if (BEGINS_WITH(line, "Buffers:")) {
|
|
minfo->ram_buffers = strtoul(line + strlen("Buffers:"), NULL, 10);
|
|
} else if (BEGINS_WITH(line, "Cached:")) {
|
|
minfo->ram_cached = strtoul(line + strlen("Cached:"), NULL, 10);
|
|
} else if (BEGINS_WITH(line, "Shmem:")) {
|
|
minfo->ram_shared = strtoul(line + strlen("Shmem:"), NULL, 10);
|
|
} else {
|
|
continue;
|
|
}
|
|
if (--unread_fields == 0) {
|
|
break;
|
|
}
|
|
}
|
|
fclose(file);
|
|
|
|
if (unread_fields > 0) {
|
|
goto error;
|
|
}
|
|
|
|
// Values are in kB, convert them to B.
|
|
minfo->ram_total *= 1024UL;
|
|
minfo->ram_free *= 1024UL;
|
|
minfo->ram_available *= 1024UL;
|
|
minfo->ram_buffers *= 1024UL;
|
|
minfo->ram_cached *= 1024UL;
|
|
minfo->ram_shared *= 1024UL;
|
|
|
|
handle_used_method(minfo, ctx);
|
|
handle_threshold(minfo, ctx);
|
|
error:
|
|
OUTPUT_FULL_TEXT("can't read memory");
|
|
fputs("i3status: Cannot read system memory using /proc/meminfo\n", stderr);
|
|
|
|
#elif defined(__OpenBSD__)
|
|
int mib [] = { CTL_VM, VM_UVMEXP };
|
|
struct uvmexp our_uvmexp;
|
|
size_t size = sizeof (our_uvmexp);
|
|
|
|
if (sysctl (mib, 2, &our_uvmexp, &size, NULL, 0) < 0) {
|
|
OUTPUT_FULL_TEXT("");
|
|
fputs("i3status: cannot get memory info!\n", stderr);
|
|
return;
|
|
}
|
|
|
|
minfo->ram_total = page2byte(our_uvmexp.npages);
|
|
minfo->ram_free = page2byte(our_uvmexp.free + our_uvmexp.inactive);
|
|
minfo->ram_used = page2byte(our_uvmexp.npages - our_uvmexp.free - our_uvmexp.inactive);
|
|
minfo->ram_available = minfo->ram_free;
|
|
minfo->ram_buffers = 0;
|
|
minfo->ram_cached = 0;
|
|
minfo->ram_shared = 0;
|
|
minfo->output_color = NULL;
|
|
|
|
handle_used_method(minfo, ctx);
|
|
handle_threshold(minfo, ctx);
|
|
#else
|
|
// For FreeBSD implementation see: https://bal0n.es/git/slstatus/file/components/ram.c.html
|
|
OUTPUT_FULL_TEXT("");
|
|
fputs("i3status: Memory status information is not supported on this system\n", stderr);
|
|
#endif
|
|
}
|
|
|
|
static void print_formatted_memory(struct print_mem_info *minfo, memory_ctx_t *ctx) {
|
|
char *outwalk = ctx->buf;
|
|
const char *selected_format = ctx->format;
|
|
char string_ram_total[STRING_SIZE];
|
|
char string_ram_used[STRING_SIZE];
|
|
char string_ram_free[STRING_SIZE];
|
|
char string_ram_available[STRING_SIZE];
|
|
char string_ram_shared[STRING_SIZE];
|
|
char string_percentage_free[STRING_SIZE];
|
|
char string_percentage_available[STRING_SIZE];
|
|
char string_percentage_used[STRING_SIZE];
|
|
char string_percentage_shared[STRING_SIZE];
|
|
|
|
if (minfo->output_color) {
|
|
START_COLOR(minfo->output_color);
|
|
|
|
if (ctx->format_degraded)
|
|
selected_format = ctx->format_degraded;
|
|
}
|
|
|
|
print_bytes_human(string_ram_total, minfo->ram_total, ctx->unit, ctx->decimals);
|
|
print_bytes_human(string_ram_used, minfo->ram_used, ctx->unit, ctx->decimals);
|
|
print_bytes_human(string_ram_free, minfo->ram_free, ctx->unit, ctx->decimals);
|
|
print_bytes_human(string_ram_available, minfo->ram_available, ctx->unit, ctx->decimals);
|
|
print_bytes_human(string_ram_shared, minfo->ram_shared, ctx->unit, ctx->decimals);
|
|
print_percentage(string_percentage_free, 100.0 * minfo->ram_free / minfo->ram_total);
|
|
print_percentage(string_percentage_available, 100.0 * minfo->ram_available / minfo->ram_total);
|
|
print_percentage(string_percentage_used, 100.0 * minfo->ram_used / minfo->ram_total);
|
|
print_percentage(string_percentage_shared, 100.0 * minfo->ram_shared / minfo->ram_total);
|
|
|
|
placeholder_t placeholders[] = {
|
|
{.name = "%total", .value = string_ram_total},
|
|
{.name = "%used", .value = string_ram_used},
|
|
{.name = "%free", .value = string_ram_free},
|
|
{.name = "%available", .value = string_ram_available},
|
|
{.name = "%shared", .value = string_ram_shared},
|
|
{.name = "%percentage_free", .value = string_percentage_free},
|
|
{.name = "%percentage_available", .value = string_percentage_available},
|
|
{.name = "%percentage_used", .value = string_percentage_used},
|
|
{.name = "%percentage_shared", .value = string_percentage_shared}};
|
|
|
|
const size_t num = sizeof(placeholders) / sizeof(placeholder_t);
|
|
char *formatted = format_placeholders(selected_format, &placeholders[0], num);
|
|
OUTPUT_FORMATTED;
|
|
free(formatted);
|
|
|
|
if (minfo->output_color)
|
|
END_COLOR;
|
|
|
|
OUTPUT_FULL_TEXT(ctx->buf);
|
|
}
|
|
|
|
void print_memory(memory_ctx_t *ctx) {
|
|
struct print_mem_info minfo;
|
|
|
|
get_memory_info(&minfo, ctx);
|
|
print_formatted_memory(&minfo, ctx);
|
|
}
|