aboutsummaryrefslogtreecommitdiff
path: root/src/blocks/ram.c
blob: 44dd1a3d969f1c43f3bb9d9751c8f3d7572b5ccd (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
#include <string.h>
#include <stdio.h>
#include <assert.h>
#include <stdlib.h>

#include "scheme.h"

#include "../any_log.h"

char *str_replace(char *orig, char *rep, char *with) {
    char *result; // the return string
    char *ins;    // the next insert point
    char *tmp;    // varies
    int len_rep;  // length of rep (the string to remove)
    int len_with; // length of with (the string to replace rep with)
    int len_front; // distance between rep and end of last rep
    int count;    // number of replacements

    // sanity checks and initialization
    if (!orig || !rep)
        return NULL;
    len_rep = strlen(rep);
    if (len_rep == 0)
        return NULL; // empty rep causes infinite loop during count
    if (!with)
        with = "";
    len_with = strlen(with);

    // count the number of replacements needed
    ins = orig;
    for (count = 0; (tmp = strstr(ins, rep)); ++count) {
        ins = tmp + len_rep;
    }

    tmp = result = malloc(strlen(orig) + (len_with - len_rep) * count + 1);

    if (!result)
        return NULL;

    // first time through the loop, all the variable are set correctly
    // from here on,
    //    tmp points to the end of the result string
    //    ins points to the next occurrence of rep in orig
    //    orig points to the remainder of orig after "end of rep"
    while (count--) {
        ins = strstr(orig, rep);
        len_front = ins - orig;
        tmp = strncpy(tmp, orig, len_front) + len_front;
        tmp = strcpy(tmp, with) + len_with;
        orig += len_front + len_rep; // move to next "end of rep"
    }
    strcpy(tmp, orig);
    return result;
}

static void block_ram_update(block_t *block)
{
    FILE *meminfo = fopen("/proc/meminfo", "rb");
    assert(meminfo != NULL);

    uintmax_t total, unused, buffers, cached;
    assert(5 == fscanf(meminfo,
                       "MemTotal: %ju kB\n"
                       "MemFree: %ju kB\n"
                       "MemAvailable: %ju kB\n"
                       "Buffers: %ju kB\n"
                       "Cached: %ju kB\n",
                       &total, &unused, &buffers, &buffers, &cached));

    int usage = 100 * (total - unused - buffers - cached) / total;
    fclose(meminfo);

    free(block->text.text);

    char str[100];
    sprintf(str, "%d", usage);

    block->text.text = str_replace(block->state, "%{ram}", str);
    assert(block->text.text != NULL);
}

static void block_ram_finalize(block_t *block)
{
    free(block->state);
}

static bool block_ram_validate(block_t *block, const block_scheme_t *scheme)
{
    block->state = block->text.text;
    block->text.text = NULL;

    if (block->state == NULL) {
        log_error("Block '%s' requires key '%s'", block->label, "text");
        return false;
    }

    if (strstr(block->state, "%{ram}") != NULL)
        log_warn("Block '%s' does not have a ram format", block->label);

    return true;
}

const block_scheme_t block_ram_scheme = {
    .name = "ram",
    .block = {
        .type = BLOCK_TEXT,
        .update_interval = {
            .tv_sec = 1,
            .tv_nsec = 0,
        },
        .update_cb = block_ram_update,
        .finalize_cb = block_ram_finalize,
    },
    .size = sizeof(char *),
    .entries = NULL,
    .validate = block_ram_validate,
};