2015-03-16 09:00:32 +00:00
|
|
|
// vim:ts=4:sw=4:expandtab
|
2019-01-23 07:56:40 +00:00
|
|
|
#include <config.h>
|
2019-01-23 09:44:48 +00:00
|
|
|
#if defined(__linux__)
|
2018-09-22 16:55:53 +00:00
|
|
|
#include <sys/sysinfo.h>
|
2019-01-23 09:44:48 +00:00
|
|
|
#endif
|
2011-02-26 22:45:12 +00:00
|
|
|
#include <stdlib.h>
|
|
|
|
#include <limits.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <string.h>
|
2012-03-25 18:55:55 +00:00
|
|
|
#include <yajl/yajl_gen.h>
|
2012-04-08 12:05:47 +00:00
|
|
|
#include <yajl/yajl_version.h>
|
2019-07-29 19:25:32 +00:00
|
|
|
#include <errno.h>
|
2011-02-26 22:45:12 +00:00
|
|
|
|
2012-04-27 15:32:46 +00:00
|
|
|
#if defined(__FreeBSD__) || defined(__OpenBSD__)
|
|
|
|
#include <sys/param.h>
|
2011-07-24 01:36:33 +00:00
|
|
|
#include <sys/types.h>
|
|
|
|
#include <sys/sysctl.h>
|
2015-04-02 18:31:17 +00:00
|
|
|
#if defined(__OpenBSD__)
|
|
|
|
#include <sys/sched.h>
|
|
|
|
#else
|
2011-07-24 01:36:33 +00:00
|
|
|
#include <sys/dkstat.h>
|
|
|
|
#endif
|
2015-04-02 18:31:17 +00:00
|
|
|
#endif
|
2011-07-24 01:36:33 +00:00
|
|
|
|
2012-11-14 01:29:55 +00:00
|
|
|
#if defined(__DragonFly__)
|
|
|
|
#include <sys/param.h>
|
|
|
|
#include <sys/types.h>
|
|
|
|
#include <sys/sysctl.h>
|
|
|
|
#include <sys/resource.h>
|
|
|
|
#endif
|
|
|
|
|
2013-10-06 18:13:29 +00:00
|
|
|
#if defined(__NetBSD__)
|
|
|
|
#include <sys/param.h>
|
|
|
|
#include <sys/resource.h>
|
|
|
|
#include <sys/sysctl.h>
|
|
|
|
#include <sys/sched.h>
|
|
|
|
#endif
|
|
|
|
|
2011-02-26 22:45:12 +00:00
|
|
|
#include "i3status.h"
|
|
|
|
|
2017-03-26 10:54:07 +00:00
|
|
|
struct cpu_usage {
|
|
|
|
int user;
|
|
|
|
int nice;
|
|
|
|
int system;
|
|
|
|
int idle;
|
|
|
|
int total;
|
|
|
|
};
|
|
|
|
|
2019-07-29 20:46:20 +00:00
|
|
|
#if defined(__linux__)
|
2017-03-26 10:54:07 +00:00
|
|
|
static int cpu_count = 0;
|
2019-07-29 20:46:20 +00:00
|
|
|
#endif
|
2017-03-26 10:54:07 +00:00
|
|
|
static struct cpu_usage prev_all = {0, 0, 0, 0, 0};
|
|
|
|
static struct cpu_usage *prev_cpus = NULL;
|
|
|
|
static struct cpu_usage *curr_cpus = NULL;
|
2011-02-26 22:45:12 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Reads the CPU utilization from /proc/stat and returns the usage as a
|
|
|
|
* percentage.
|
|
|
|
*
|
|
|
|
*/
|
2017-03-26 10:54:07 +00:00
|
|
|
void print_cpu_usage(yajl_gen json_gen, char *buffer, const char *format, const char *format_above_threshold, const char *format_above_degraded_threshold, const char *path, const float max_threshold, const float degraded_threshold) {
|
2016-11-14 20:10:19 +00:00
|
|
|
const char *selected_format = format;
|
2015-03-16 09:00:32 +00:00
|
|
|
const char *walk;
|
|
|
|
char *outwalk = buffer;
|
2017-03-26 10:54:07 +00:00
|
|
|
struct cpu_usage curr_all = {0, 0, 0, 0, 0};
|
2015-03-16 09:00:32 +00:00
|
|
|
int diff_idle, diff_total, diff_usage;
|
2016-04-13 16:07:12 +00:00
|
|
|
bool colorful_output = false;
|
2011-02-26 22:45:12 +00:00
|
|
|
|
2019-01-23 07:56:40 +00:00
|
|
|
#if defined(__linux__)
|
2017-03-26 10:54:07 +00:00
|
|
|
|
|
|
|
// Detecting if CPU count has changed
|
2018-09-22 16:55:53 +00:00
|
|
|
int curr_cpu_count = get_nprocs_conf();
|
2017-03-26 10:54:07 +00:00
|
|
|
if (curr_cpu_count != cpu_count) {
|
|
|
|
cpu_count = curr_cpu_count;
|
|
|
|
free(prev_cpus);
|
|
|
|
prev_cpus = (struct cpu_usage *)calloc(cpu_count, sizeof(struct cpu_usage));
|
|
|
|
free(curr_cpus);
|
|
|
|
curr_cpus = (struct cpu_usage *)calloc(cpu_count, sizeof(struct cpu_usage));
|
|
|
|
}
|
|
|
|
|
2018-09-22 16:55:53 +00:00
|
|
|
memcpy(curr_cpus, prev_cpus, cpu_count * sizeof(struct cpu_usage));
|
2019-07-29 19:25:32 +00:00
|
|
|
FILE *f = fopen(path, "r");
|
|
|
|
if (f == NULL) {
|
|
|
|
fprintf(stderr, "i3status: open %s: %s\n", path, strerror(errno));
|
2017-03-26 10:54:07 +00:00
|
|
|
goto error;
|
2019-07-29 19:25:32 +00:00
|
|
|
}
|
|
|
|
curr_cpu_count = get_nprocs();
|
|
|
|
char line[4096];
|
|
|
|
|
|
|
|
/* Discard first line (cpu ), start at second line (cpu0) */
|
|
|
|
if (fgets(line, sizeof(line), f) == NULL) {
|
|
|
|
fclose(f);
|
|
|
|
goto error; /* unexpected EOF or read error */
|
|
|
|
}
|
|
|
|
|
2018-09-22 16:55:53 +00:00
|
|
|
for (int idx = 0; idx < curr_cpu_count; ++idx) {
|
2019-07-29 19:25:32 +00:00
|
|
|
if (fgets(line, sizeof(line), f) == NULL) {
|
|
|
|
fclose(f);
|
|
|
|
goto error; /* unexpected EOF or read error */
|
|
|
|
}
|
2018-09-22 16:55:53 +00:00
|
|
|
int cpu_idx, user, nice, system, idle;
|
2019-07-29 19:25:32 +00:00
|
|
|
if (sscanf(line, "cpu%d %d %d %d %d", &cpu_idx, &user, &nice, &system, &idle) != 5) {
|
|
|
|
fclose(f);
|
2018-09-22 16:55:53 +00:00
|
|
|
goto error;
|
|
|
|
}
|
2019-07-29 19:25:32 +00:00
|
|
|
if (cpu_idx < 0 || cpu_idx >= cpu_count) {
|
|
|
|
fclose(f);
|
2017-03-26 10:54:07 +00:00
|
|
|
goto error;
|
2019-07-29 19:25:32 +00:00
|
|
|
}
|
2018-09-22 16:55:53 +00:00
|
|
|
curr_cpus[cpu_idx].user = user;
|
|
|
|
curr_cpus[cpu_idx].nice = nice;
|
|
|
|
curr_cpus[cpu_idx].system = system;
|
|
|
|
curr_cpus[cpu_idx].idle = idle;
|
|
|
|
curr_cpus[cpu_idx].total = user + nice + system + idle;
|
|
|
|
}
|
2019-07-29 19:25:32 +00:00
|
|
|
fclose(f);
|
2018-09-22 16:55:53 +00:00
|
|
|
for (int cpu_idx = 0; cpu_idx < cpu_count; cpu_idx++) {
|
2017-03-26 10:54:07 +00:00
|
|
|
curr_all.user += curr_cpus[cpu_idx].user;
|
|
|
|
curr_all.nice += curr_cpus[cpu_idx].nice;
|
|
|
|
curr_all.system += curr_cpus[cpu_idx].system;
|
|
|
|
curr_all.idle += curr_cpus[cpu_idx].idle;
|
|
|
|
curr_all.total += curr_cpus[cpu_idx].total;
|
|
|
|
}
|
2011-02-26 22:45:12 +00:00
|
|
|
|
2017-03-26 10:54:07 +00:00
|
|
|
diff_idle = curr_all.idle - prev_all.idle;
|
|
|
|
diff_total = curr_all.total - prev_all.total;
|
2015-03-16 09:00:32 +00:00
|
|
|
diff_usage = (diff_total ? (1000 * (diff_total - diff_idle) / diff_total + 5) / 10 : 0);
|
2017-03-26 10:54:07 +00:00
|
|
|
prev_all = curr_all;
|
2013-10-06 18:13:29 +00:00
|
|
|
#elif defined(__FreeBSD__) || defined(__NetBSD__) || defined(__OpenBSD__) || defined(__DragonFly__)
|
2012-04-27 15:32:46 +00:00
|
|
|
|
2013-10-06 18:13:29 +00:00
|
|
|
#if defined(__FreeBSD__) || defined(__DragonFly__) || defined(__NetBSD__)
|
2015-03-16 09:00:32 +00:00
|
|
|
size_t size;
|
|
|
|
long cp_time[CPUSTATES];
|
|
|
|
size = sizeof cp_time;
|
|
|
|
if (sysctlbyname("kern.cp_time", &cp_time, &size, NULL, 0) < 0)
|
|
|
|
goto error;
|
2012-04-27 15:32:46 +00:00
|
|
|
#else
|
2015-03-16 09:00:32 +00:00
|
|
|
/* This information is taken from the boot cpu, any other cpus are currently ignored. */
|
|
|
|
long cp_time[CPUSTATES];
|
|
|
|
int mib[2];
|
|
|
|
size_t size = sizeof(cp_time);
|
2012-04-27 15:32:46 +00:00
|
|
|
|
2015-03-16 09:00:32 +00:00
|
|
|
mib[0] = CTL_KERN;
|
|
|
|
mib[1] = KERN_CPTIME;
|
2012-04-27 15:32:46 +00:00
|
|
|
|
2015-03-16 09:00:32 +00:00
|
|
|
if (sysctl(mib, 2, cp_time, &size, NULL, 0))
|
|
|
|
goto error;
|
2012-04-27 15:32:46 +00:00
|
|
|
#endif
|
2011-08-25 21:24:06 +00:00
|
|
|
|
2017-03-26 10:54:07 +00:00
|
|
|
curr_all.user = cp_time[CP_USER];
|
|
|
|
curr_all.nice = cp_time[CP_NICE];
|
|
|
|
curr_all.system = cp_time[CP_SYS];
|
|
|
|
curr_all.idle = cp_time[CP_IDLE];
|
|
|
|
curr_all.total = curr_all.user + curr_all.nice + curr_all.system + curr_all.idle;
|
|
|
|
diff_idle = curr_all.idle - prev_all.idle;
|
|
|
|
diff_total = curr_all.total - prev_all.total;
|
2015-03-16 09:00:32 +00:00
|
|
|
diff_usage = (diff_total ? (1000 * (diff_total - diff_idle) / diff_total + 5) / 10 : 0);
|
2017-03-26 10:54:07 +00:00
|
|
|
prev_all = curr_all;
|
2011-08-25 21:24:06 +00:00
|
|
|
#else
|
2015-03-16 09:00:32 +00:00
|
|
|
goto error;
|
2011-02-26 22:45:12 +00:00
|
|
|
#endif
|
2016-11-14 20:10:19 +00:00
|
|
|
|
|
|
|
if (diff_usage >= max_threshold) {
|
|
|
|
START_COLOR("color_bad");
|
|
|
|
colorful_output = true;
|
|
|
|
if (format_above_threshold != NULL)
|
|
|
|
selected_format = format_above_threshold;
|
|
|
|
} else if (diff_usage >= degraded_threshold) {
|
|
|
|
START_COLOR("color_degraded");
|
|
|
|
colorful_output = true;
|
|
|
|
if (format_above_degraded_threshold != NULL)
|
|
|
|
selected_format = format_above_degraded_threshold;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (walk = selected_format; *walk != '\0'; walk++) {
|
2015-03-16 09:00:32 +00:00
|
|
|
if (*walk != '%') {
|
|
|
|
*(outwalk++) = *walk;
|
2011-02-26 22:45:12 +00:00
|
|
|
|
2018-06-02 00:32:25 +00:00
|
|
|
} else if (BEGINS_WITH(walk + 1, "usage")) {
|
2015-10-02 06:28:53 +00:00
|
|
|
outwalk += sprintf(outwalk, "%02d%s", diff_usage, pct_mark);
|
2015-03-16 09:00:32 +00:00
|
|
|
walk += strlen("usage");
|
2011-02-26 22:45:12 +00:00
|
|
|
}
|
2019-01-23 07:56:40 +00:00
|
|
|
#if defined(__linux__)
|
2018-06-02 00:32:25 +00:00
|
|
|
else if (BEGINS_WITH(walk + 1, "cpu")) {
|
2018-10-21 22:37:45 +00:00
|
|
|
int number = -1;
|
2020-03-29 01:58:34 +00:00
|
|
|
int length = strlen("cpu");
|
|
|
|
sscanf(walk + 1, "cpu%d%n", &number, &length);
|
2018-11-07 22:00:22 +00:00
|
|
|
if (number == -1) {
|
|
|
|
fprintf(stderr, "i3status: provided CPU number cannot be parsed\n");
|
|
|
|
} else if (number >= cpu_count) {
|
|
|
|
fprintf(stderr, "i3status: provided CPU number '%d' above detected number of CPU %d\n", number, cpu_count);
|
2017-03-26 10:54:07 +00:00
|
|
|
} else {
|
|
|
|
int cpu_diff_idle = curr_cpus[number].idle - prev_cpus[number].idle;
|
|
|
|
int cpu_diff_total = curr_cpus[number].total - prev_cpus[number].total;
|
|
|
|
int cpu_diff_usage = (cpu_diff_total ? (1000 * (cpu_diff_total - cpu_diff_idle) / cpu_diff_total + 5) / 10 : 0);
|
|
|
|
outwalk += sprintf(outwalk, "%02d%s", cpu_diff_usage, pct_mark);
|
|
|
|
}
|
2020-03-29 01:58:34 +00:00
|
|
|
walk += length;
|
2017-03-26 10:54:07 +00:00
|
|
|
}
|
|
|
|
#endif
|
2018-06-02 00:32:25 +00:00
|
|
|
else {
|
|
|
|
*(outwalk++) = '%';
|
|
|
|
}
|
2015-03-16 09:00:32 +00:00
|
|
|
}
|
2012-02-16 23:29:29 +00:00
|
|
|
|
2018-09-22 16:55:53 +00:00
|
|
|
struct cpu_usage *temp_cpus = prev_cpus;
|
|
|
|
prev_cpus = curr_cpus;
|
|
|
|
curr_cpus = temp_cpus;
|
2017-03-26 10:54:07 +00:00
|
|
|
|
2016-11-14 20:10:19 +00:00
|
|
|
if (colorful_output)
|
|
|
|
END_COLOR;
|
|
|
|
|
2015-03-16 09:00:32 +00:00
|
|
|
OUTPUT_FULL_TEXT(buffer);
|
|
|
|
return;
|
2011-08-25 21:24:06 +00:00
|
|
|
error:
|
2015-03-16 09:00:32 +00:00
|
|
|
OUTPUT_FULL_TEXT("cant read cpu usage");
|
|
|
|
(void)fputs("i3status: Cannot read CPU usage\n", stderr);
|
2011-02-26 22:45:12 +00:00
|
|
|
}
|