energymech/src/hostinfo.c
2025-09-21 16:22:13 +02:00

444 lines
10 KiB
C

/*
EnergyMech, IRC bot software
Copyright (c) 2020-2024 proton
This program is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 2 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program; if not, write to the Free Software
Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
*/
#define HOSTINFO_C
#include "config.h"
#ifdef HOSTINFO
#include "defines.h"
#include "structs.h"
#include "global.h"
#include "h.h"
#include <sys/utsname.h>
#include <sys/inotify.h>
/*
Emulate this, but use the same memory space:
char vmpeak[32];
char vmsize[32];
char vmrss[32];
char vmdata[32];
char vmstk[32];
char vmexe[32];
char vmlib[32];
*/
char omni[224]; /* 32*7 */
#define vmpeak &omni[0]
#define vmsize &omni[32]
#define vmrss &omni[64]
#define vmdata &omni[96]
#define vmstk &omni[128]
#define vmexe &omni[160]
#define vmlib &omni[192]
struct /* statusvalues */
{
const char *key;
const int klen;
char *valbuf;
} sv[] =
{
{ "VmPeak:", 7, vmpeak },
{ "VmSize:", 7, vmsize },
{ "VmRSS:", 6, vmrss },
{ "VmData:", 7, vmdata },
{ "VmStk:", 6, vmstk },
{ "VmExe:", 6, vmexe },
{ "VmLib:", 6, vmlib },
{ NULL, 0, NULL }
};
#ifdef DEBUG
struct
{
int value;
char *str;
} in2str[] =
{
{ IN_ACCESS, "IN_ACCESS" }, /* File was accessed (read) */
{ IN_ATTRIB, "IN_ATTRIB" }, /* Metadata changed, e.g., permissions, timestamps, extended attributes, link count, UID, GID, etc. */
{ IN_CLOSE_WRITE, "IN_CLOSE_WRITE" }, /* File opened for writing was closed */
{ IN_CLOSE_NOWRITE, "IN_CLOSE_NOWRITE" }, /* File not opened for writing was closed */
{ IN_CREATE, "IN_CREATE" }, /* File/directory created in watched directory */
{ IN_DELETE, "IN_DELETE" }, /* File/directory deleted from watched directory */
{ IN_DELETE_SELF, "IN_DELETE_SELF" }, /* Watched file/directory was itself deleted */
{ IN_MODIFY, "IN_MODIFY" }, /* File was modified */
{ IN_MOVE_SELF, "IN_MOVE_SELF" }, /* Watched file/directory was itself moved */
{ IN_MOVED_FROM, "IN_MOVED_FROM" }, /* Generated for the directory containing the old filename when a file is renamed */
{ IN_MOVED_TO, "IN_MOVED_TO" }, /* Generated for the directory containing the new filename when a file is renamed */
{ IN_OPEN, "IN_OPEN" }, /* File was opened */
{ 0, NULL }
};
char *inomask2str(uint32_t mask, char *dst)
{
const char *src;
int i,n = 0;
for(i=0;in2str[i].str;i++)
{
if ((mask & in2str[i].value) == in2str[i].value)
{
if (n)
dst[n++] = '|';
src = in2str[i].str;
for(;src[n];n++)
dst[n] = src[n];
}
}
dst[n] = 0;
return(dst);
}
#endif /* DEBUG */
int monitor_fs(const char *file)
{
FileMon *fmon,*fnew;
int ino;
if ((ino = inotify_init()) < 0)
return(-1);
inotify_add_watch(ino,file,IN_ALL_EVENTS);
#ifdef DEBUG
debug("(monitor_fs) added notifier on %s [fd %i]\n",file,ino);
#endif
set_mallocdoer(monitor_fs);
fnew = Calloc(sizeof(FileMon) + strlen(file));
fnew->fd = ino;
fnew->nospam = now;
stringcpy(fnew->filename,file);
fnew->next = filemonlist;
filemonlist = fnew;
return(0);
}
int parse_proc_status(char *line)
{
const char *key;
char *dest,*limit;
int i;
key = chop(&line);
#ifdef DEBUG
debug("(parse_proc_status) pps key = %s (%s)\n",key,line);
#endif
if (key == NULL)
return(FALSE);
for(i=0;sv[i].key;i++)
{
if (strncmp(key,sv[i].key,sv[i].klen) == 0)
{
#ifdef DEBUG
debug("(parse_proc_status) key %s -> %s\n",key,line);
#endif /* DEBUG */
dest = sv[i].valbuf;
limit = sv[i].valbuf + 31;
while(*line == ' ')
line++;
while(*line && dest <= limit)
*(dest++) = *(line++);
*dest = 0;
}
}
return(FALSE);
}
int havemodel,bogo,procct,physid,cpus,cores,siblings;
/*
proton@endemic:~/energymech/src> cat /proc/loadavg
0.00 0.00 0.00 1/178 6759
processor : 0
model name : Intel(R) Core(TM)2 Quad CPU Q8200 @ 2.33GHz
physical id : 0
siblings : 4 <-- total number of cores in all cpus
core id : 0
cpu cores : 4
bogomips : 4533.39
*/
const char *cfind(const char *s1, const char *s2)
{
while(*s2)
{
while(*s1 == ' ' || *s1 == '\t')
s1++;
if (tolowertab[(uchar)(*s1)] != tolowertab[(uchar)(*s2)])
return(NULL);
s1++;
s2++;
}
while(*s1 == ' ' || *s1 == '\t' || *s1 == ':')
s1++;
return(s1);
}
int parse_proc_cpuinfo(char *line)
{
const char *src;
char *dst = omni,*end = omni+sizeof(omni)-1;
int v;
if ((src = cfind(line,"cpumodel")) || (src = cfind(line,"modelname")))
{
if (havemodel == 1)
return(FALSE);
*(dst++) = ' '; /* prime with a leading space */
while(*src && dst < end)
{
if (*src != ' ' || dst[-1] != ' ')
*(dst++) = *src;
src++;
}
*dst = 0;
#ifdef DEBUG
debug("(parse_proc_cpuinfo) model name = %s\n",omni);
#endif
havemodel = 1;
}
if ((src = cfind(line,"physicalid")) || (src = cfind(line,"core")))
{
v = asc2int(src);
if (errno == 0)
{
physid += (v+1);
}
}
else
if (siblings == 0 && (src = cfind(line,"siblings")))
{
cores = siblings = asc2int(src);
}
else
if ((src = cfind(line,"bogomips")))
{
bogo++;
if (bogo != 1)
return(FALSE);
dst = vmlib;
while(*src && dst < end)
{
if (*src != ' ' || dst[-1] != ' ')
*(dst++) = *src;
src++;
}
*dst = 0;
}
return(FALSE); /* return false to continue reading lines */
}
void select_monitor()
{
FileMon *fmon;
for(fmon=filemonlist;fmon;fmon=fmon->next)
if (fmon->fd >= 0)
{
FD_SET(fmon->fd,&read_fds);
chkhigh(fmon->fd);
}
}
void process_monitor()
{
FileMon *fmon;
struct inotify_event *ivent;
char tmp[256];
int n,m;
for(fmon=filemonlist;fmon;fmon=fmon->next)
{
if (fmon->fd >= 0 && FD_ISSET(fmon->fd,&read_fds))
{
ivent = (struct inotify_event *)&globaldata;
n = read(fmon->fd,globaldata,sizeof(struct inotify_event));
if (ivent->len > 0)
m = read(fmon->fd,ivent->name,ivent->len);
else
*ivent->name = 0;
#ifdef DEBUG
debug("(process_monitor) ino %i, n %i, sz %i\n",fmon->fd,n,sizeof(in2str));
debug("(process_monitor) wd %i, mask %lu, cookie %lu, len %lu, name %s\n",
ivent->wd,ivent->mask,ivent->cookie,ivent->len,ivent->name);
debug("(process_monitor) %s\n",inomask2str(ivent->mask,tmp));
debug("(process_monitor) ino %i bytes read, int wd = %i, uint32_t mask = %s (%lu), "
"uint32_t cookie = %lu, uint32_t len = %lu, char name = %s\n",
n,ivent->wd,inomask2str(ivent->mask,tmp),ivent->mask,ivent->cookie,ivent->len,ivent->name);
#endif
if ((ivent->mask & IN_CLOSE_WRITE) == IN_CLOSE_WRITE)
return;
if (fmon->nospam > now-30)
return;
fmon->nospam = now;
send_global(SPYSTR_SYSMON,"Alert: file ``%s'' was touched",fmon->filename);
}
}
}
/*---------------------------------------------------------------------------------------------------------------------------------------------------*/
/*
help:SYSINFO:(no arguments)
Equivalent to ``uname -orm''
See also: meminfo, cpuinfo
*/
void do_sysinfo(COMMAND_ARGS)
{
char *h,hostname[256];
struct utsname un;
hostname[255] = 0;
if (gethostname(hostname,250) < 0)
h = "(hostname error)";
else
h = hostname;
if (uname(&un) == 0)
to_user_q(from,"%s %s %s %s",h,un.sysname,un.release,un.machine);
}
/*
help:MEMINFO:(no arguments)
Will display memory usage of the energymech process.
VM Virtual size, size if everything was loaded into memory)
RSS Resident set size, physical memory actually in use right now.
Code Memory allocated for code
Data Memory allocated for data
Libs Memory used by shared libraries
Stack Memory allocated for stack
See also: hostinfo, cpuinfo
*/
void do_meminfo(COMMAND_ARGS)
{
char fn[64];
pid_t p;
int i,fd;
p = getpid();
snprintf(fn,sizeof(fn),"/proc/%i/status",p);
for(i=0;sv[i].key;i++)
*(sv[i].valbuf) = 0;
if ((fd = open(fn,O_RDONLY)) < 0)
return;
readline(fd,&parse_proc_status); /* readline closes fd */
to_user_q(from,"VM %s (Max %s), RSS %s [ Code %s, Data %s, Libs %s, Stack %s ]",
vmsize,vmpeak,vmrss,vmexe,vmdata,vmlib,vmstk);
}
/*
help:CPUINFO:(no arguments)
See also: hostinfo, meminfo
*/
void do_cpuinfo(COMMAND_ARGS)
{
char bogostr[256],cpustr[64];
char *a1,*a2,*a3,*dst;
int fd,n;
double loads[3];
#ifdef DEVELOPING
a1 = chop(&rest);
if (a1)
sprintf(bogostr,"/home/git/cpuinfo/%s",a1);
else
stringcpy(bogostr,"/proc/cpuinfo");
if ((fd = open(bogostr,O_RDONLY)) < 0)
#endif
if ((fd = open("/proc/cpuinfo",O_RDONLY)) < 0)
#ifdef DEBUG
{
debug("(do_cpuinfo) /proc/cpuinfo: %s\n",strerror(errno));
return;
}
#else
return;
#endif
global_from = from;
havemodel = bogo = siblings = procct = cpus = cores = physid = 0;
omni[1] = 0;
readline(fd,&parse_proc_cpuinfo); /* readline closes fd */
if (getloadavg(loads,3) < 3)
return;
#ifdef DEBUG
debug("(do_cpuinfo) procct %i, physid %i, cores %i, bogo %i\n",procct,physid,cores,bogo);
#endif
if (cores == 0)
cores = bogo;
if (cores && physid && (physid % cores) == 0)
cpus = (physid / cores)-1;
if (cores && (cpus == 0 || physid == cores))
cpus = 1;
*bogostr = 0;
*cpustr = 0;
if (bogo)
sprintf(bogostr,", %s BogoMips",vmlib);
if (cpus > 1 || (cores > cpus))
{
sprintf(cpustr,", %i physical cpu%s",cpus,(cpus == 1) ? "" : "s");
if (cores)
sprintf(STREND(cpustr),", %i core%s",cores,(cores == 1) ? "" : "s");
}
#ifdef DEBUG
debug("(do_cpuinfo) %s%s%s, loadavg: %.2f(1m) %.2f(5m) %.2f(15m)",
omni+1,bogostr,cpustr,(loads[0]),(loads[1]),(loads[2]));
#endif
to_user_q(from,"%s%s%s, loadavg: %.2f(1m) %.2f(5m) %.2f(15m)",
omni+1,bogostr,cpustr,(loads[0]),(loads[1]),(loads[2]));
}
void do_filemon(COMMAND_ARGS)
{
struct stat st;
#ifdef DEBUG
debug("(do_filemon) rest = '%s'\n",rest);
#endif
if (stat(rest,&st) < 0 || monitor_fs(rest) < 0)
{
to_user_q(from,"Unable to add file monitor for: ``%s''",rest);
return;
}
to_user_q(from,"File monitor added.");
}
#endif /* HOSTINFO */