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,
};
|