Probe/src/probe.c

392 lines
7.2 KiB
C

/* probe.c */
#include <System.h>
#include <utmpx.h>
#include <stdlib.h>
#include <stdio.h>
#ifdef _GNU_SOURCE
# include <sys/sysinfo.h>
#else
# include <sys/param.h>
# include <sys/sysctl.h>
# include <sys/resource.h>
# include <string.h>
#endif
#define PROBE_REFRESH 10
/* globals */
/* sysinfo */
#ifndef _GNU_SOURCE
struct sysinfo
{
long uptime;
unsigned long loads[3];
unsigned long totalram;
unsigned long freeram;
unsigned long sharedram;
unsigned long bufferram;
unsigned long totalswap;
unsigned long freeswap;
unsigned short procs;
};
static int sysinfo(struct sysinfo * info)
{
struct timeval now;
struct timeval tv;
struct loadavg la;
struct uvmexp ue;
int mib[3];
int len;
int ret = 0;
/* uptime */
mib[0] = CTL_KERN;
mib[1] = KERN_BOOTTIME;
len = sizeof(tv);
if(gettimeofday(&now, NULL) != 0
|| sysctl(mib, 2, &tv, &len, NULL, 0) == -1)
{
info->uptime = 0;
ret++;
}
else
info->uptime = now.tv_sec - tv.tv_sec;
/* loads */
/* FIXME getloadavg() looks portable */
mib[0] = CTL_VM;
mib[1] = VM_LOADAVG;
len = sizeof(la);
if(sysctl(mib, 2, &la, &len, NULL, 0) == -1)
{
memset(info->loads, 0, sizeof(info->loads));
ret++;
}
else
{
info->loads[0] = la.ldavg[0];
info->loads[1] = la.ldavg[1];
info->loads[2] = la.ldavg[2];
}
/* ram */
mib[0] = CTL_VM;
mib[1] = VM_UVMEXP;
len = sizeof(ue);
if(sysctl(mib, 2, &ue, &len, NULL, 0) == -1)
{
info->totalram = 0;
info->freeram = 0;
info->sharedram = 0;
info->bufferram = 0;
info->totalswap = 0;
info->freeswap = 0;
ret++;
}
else
{
info->totalram = ue.pagesize * ue.npages;
info->freeram = ue.pagesize * ue.free;
info->sharedram = ue.pagesize * ue.execpages;
info->bufferram = ue.pagesize * ue.filepages;
info->totalswap = ue.pagesize * ue.swpages;
info->freeswap = info->totalswap - (ue.pagesize * ue.swpgonly);
}
/* procs */
mib[0] = CTL_KERN;
mib[1] = KERN_PROC;
mib[2] = KERN_PROC_ALL;
len = 0;
if(sysctl(mib, 3, NULL, &len, NULL, 0) == -1)
{
info->procs = 0;
ret++;
}
else
info->procs = len / sizeof(struct kinfo_proc);
return ret;
}
#endif
/* ifinfo */
enum InterfaceInfo
{
IF_RX_BYTES = 0, IF_RX_PACKETS, IF_RX_ERRS, IF_RX_DROP, IF_RX_FIFO,
IF_RX_FRAME, IF_RX_COMPRESSED, IF_RX_MULTICAST,
IF_TX_BYTES, IF_TX_PACKETS, IF_TX_ERRS, IF_TX_DROP, IF_TX_FIFO,
IF_TX_FRAME, IF_TX_COMPRESSED
};
#define IF_LAST IF_TX_COMPRESSED
struct ifinfo
{
char name[6];
int stats[IF_LAST+1];
};
#ifdef _GNU_SOURCE
static int ifinfo_append(struct ifinfo ** dev, char * buf, int nb);
#endif
static int ifinfo(struct ifinfo ** dev)
{
int ret = 0;
#ifdef _GNU_SOURCE
FILE * fp;
char buf[200];
int i;
int len;
if((fp = fopen("/proc/net/dev", "r")) == NULL)
return -1;
for(i = 0; fgets(buf, sizeof(buf), fp) != NULL; i++)
{
len = string_length(buf);
if(buf[len-1] != '\n')
{
ret = 1;
break;
}
if(i < 2)
continue;
if(ifinfo_append(dev, buf, i - 2) != 0)
{
ret = -1;
break;
}
ret++;
}
fclose(fp);
#endif
return ret;
}
#ifdef _GNU_SOURCE
static int ifinfo_append(struct ifinfo ** dev, char * buf, int nb)
{
struct ifinfo * p;
int i;
char * q;
int j = 0;
if((p = realloc(*dev, sizeof(struct ifinfo) * (nb + 1))) == NULL)
return 1;
*dev = p;
for(i = 0; i < 6 && buf[i] != '\0'; i++);
if(i != 6)
return 1;
buf[6] = '\0';
for(q = buf; q[0] == ' '; q++);
strcpy((*dev)[nb].name, q);
for(i++; buf[i] != '\0'; i++)
{
if(j > IF_LAST)
break;
if(buf[i] == ' ')
continue;
q = &buf[i];
for(; buf[i] >= '0' && buf[i] <= '9'; i++);
buf[i] = '\0';
(*dev)[nb].stats[j++] = strtol(q, &q, 10);
if(*q != '\0')
return 1;
}
return 0;
}
#endif
/* Probe */
/* types */
typedef struct _Probe
{
struct sysinfo sysinfo;
unsigned int users;
struct ifinfo * ifinfo;
unsigned int ifinfo_cnt;
} Probe;
/* variables */
Probe probe;
/* functions */
static int _probe_error(char * message, int ret);
static int _probe_timeout(Probe * probe);
static int _probe(void)
{
AppServer * appserver;
Event * event;
struct timeval tv;
probe.ifinfo = NULL;
probe.ifinfo_cnt = 0;
if(_probe_timeout(&probe) != 0)
return _probe_error("sysinfo", 1);
if((event = event_new()) == NULL)
return _probe_error("Event", 2);
if((appserver = appserver_new_event("Probe", ASO_REMOTE, event))
== NULL)
{
event_delete(event);
return _probe_error("AppServer", 1);
}
tv.tv_sec = PROBE_REFRESH;
tv.tv_usec = 0;
if(event_register_timeout(event, tv, (EventTimeoutFunc)_probe_timeout,
&probe) != 0)
_probe_error("timeout", 0);
else
event_loop(event);
appserver_delete(appserver);
event_delete(event);
return 2;
}
static int _probe_error(char * message, int ret)
{
fprintf(stderr, "%s", "Probe: ");
perror(message);
return ret;
}
static int _probe_timeout(Probe * probe)
{
struct utmpx * ut;
int i;
#ifdef DEBUG
static unsigned int count = 0;
fprintf(stderr, "%s%d%s", "_probe_timeout(", count++, ")\n");
#endif
if(sysinfo(&probe->sysinfo) != 0)
return _probe_error("sysinfo", 1);
for(probe->users = 0; (ut = getutxent()) != NULL;)
if(ut->ut_type == USER_PROCESS)
probe->users++;
endutxent();
if((i = ifinfo(&probe->ifinfo)) < 0)
return _probe_error("ifinfo", 1);
probe->ifinfo_cnt = i;
return 0;
}
/* AppInterface */
int uptime(void)
{
printf("%s%ld%s", "Uptime: ", probe.sysinfo.uptime, "\n");
return probe.sysinfo.uptime;
}
int load_1(void)
{
printf("%s%lu%s", "Load 1: ", probe.sysinfo.loads[0], "\n");
return probe.sysinfo.loads[0];
}
int load_5(void)
{
printf("%s%lu%s", "Load 5: ", probe.sysinfo.loads[1], "\n");
return probe.sysinfo.loads[1];
}
int load_15(void)
{
printf("%s%lu%s", "Load 15: ", probe.sysinfo.loads[2], "\n");
return probe.sysinfo.loads[2];
}
int ram_total(void)
{
printf("%s%lu%s", "Total RAM: ", probe.sysinfo.totalram, "\n");
return probe.sysinfo.totalram;
}
int ram_free(void)
{
printf("%s%lu%s", "Free RAM: ", probe.sysinfo.freeram, "\n");
return probe.sysinfo.freeram;
}
int ram_shared(void)
{
printf("%s%lu%s", "Shared RAM: ", probe.sysinfo.sharedram, "\n");
return probe.sysinfo.sharedram;
}
int ram_buffer(void)
{
printf("%s%lu%s", "Buffered RAM: ", probe.sysinfo.bufferram, "\n");
return probe.sysinfo.bufferram;
}
int swap_total(void)
{
printf("%s%lu%s", "Total swap: ", probe.sysinfo.totalswap, "\n");
return probe.sysinfo.totalswap;
}
int swap_free(void)
{
printf("%s%lu%s", "Free swap: ", probe.sysinfo.freeswap, "\n");
return probe.sysinfo.freeswap;
}
int procs(void)
{
printf("%s%u%s", "Procs: ", probe.sysinfo.procs, "\n");
return probe.sysinfo.procs;
}
int users(void)
{
printf("%s%u%s", "Users: ", probe.users, "\n");
return probe.users;
}
int ifrxbytes(char * dev)
{
unsigned int i;
for(i = 0; i < probe.ifinfo_cnt
&& string_compare(probe.ifinfo[i].name, dev) != 0; i++);
if(i == probe.ifinfo_cnt)
return -1;
printf("%s%s%s%u%s", "Interface ", probe.ifinfo[i].name, ": ",
probe.ifinfo[i].stats[IF_RX_BYTES], "\n");
return 0;
}
int iftxbytes(char * dev)
{
unsigned int i;
for(i = 0; i < probe.ifinfo_cnt
&& string_compare(probe.ifinfo[i].name, dev) != 0; i++);
if(i == probe.ifinfo_cnt)
return -1;
printf("%s%s%s%u%s", "Interface ", probe.ifinfo[i].name, ": ",
probe.ifinfo[i].stats[IF_TX_BYTES], "\n");
return 0;
}
/* main */
int main(void)
{
return _probe() == 0 ? 0 : 2;
}