You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
tdebase/ksysguard/ksysguardd/OpenBSD/ProcessList.c

501 lines
11 KiB

/*
KSysGuard, the KDE System Guard
Copyright (c) 1999-2000 Hans Petter Bieker<bieker@kde.org>
Copyright (c) 1999 Chris Schlaeger <cs@kde.org>
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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
*/
#include <config.h>
#include <ctype.h>
#include <dirent.h>
#include <pwd.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/param.h>
#include <sys/sysctl.h>
#include <sys/time.h>
#include <sys/types.h>
#include <sys/user.h>
#include <unistd.h>
#include <signal.h>
#include "../../gui/SignalIDs.h"
#include "Command.h"
#include "ProcessList.h"
#include "ccont.h"
#include "ksysguardd.h"
CONTAINER ProcessList = 0;
#define BUFSIZE 1024
typedef struct
{
/* This flag is set for all found processes at the beginning of the
* process list update. Processes that do not have this flag set will
* be assumed dead and removed from the list. The flag is cleared after
* each list update. */
int alive;
/* the process ID */
pid_t pid;
/* the parent process ID */
pid_t ppid;
/* the real user ID */
uid_t uid;
/* the real group ID */
gid_t gid;
/* a character description of the process status */
char status[16];
/* the number of the tty the process owns */
int ttyNo;
/*
* The nice level. The range should be -20 to 20. I'm not sure
* whether this is true for all platforms.
*/
int niceLevel;
/*
* The scheduling priority.
*/
int priority;
/*
* The total amount of memory the process uses. This includes shared and
* swapped memory.
*/
unsigned int vmSize;
/*
* The amount of physical memory the process currently uses.
*/
unsigned int vmRss;
/*
* The amount of memory (shared/swapped/etc) the process shares with
* other processes.
*/
unsigned int vmLib;
/*
* The number of 1/100 of a second the process has spend in user space.
* If a machine has an uptime of 1 1/2 years or longer this is not a
* good idea. I never thought that the stability of UNIX could get me
* into trouble! ;)
*/
unsigned int userTime;
/*
* The number of 1/100 of a second the process has spend in system space.
* If a machine has an uptime of 1 1/2 years or longer this is not a
* good idea. I never thought that the stability of UNIX could get me
* into trouble! ;)
*/
unsigned int sysTime;
/* system time as multime of 100ms */
int centStamp;
/* the current CPU load (in %) from user space */
double userLoad;
/* the current CPU load (in %) from system space */
double sysLoad;
/* the name of the process */
char name[64];
/* the command used to start the process */
char cmdline[256];
/* the login name of the user that owns this process */
char userName[32];
} ProcessInfo;
static unsigned ProcessCount;
static int
processCmp(void* p1, void* p2)
{
return (((ProcessInfo*) p1)->pid - ((ProcessInfo*) p2)->pid);
}
static ProcessInfo*
findProcessInList(int pid)
{
ProcessInfo key;
long index;
key.pid = pid;
if ((index = search_ctnr(ProcessList, processCmp, &key)) < 0)
return (0);
return (get_ctnr(ProcessList, index));
}
static void
fillProcessCmdline(char *cmdline, struct kinfo_proc *p, size_t maxlen)
{
int mib[4];
int ret = -1;
static char *argbuf = NULL;
static size_t arglen = 0;
strlcpy(cmdline, p->p_comm, maxlen);
if (!argbuf) {
arglen = 1024;
argbuf = malloc(arglen);
}
mib[0] = CTL_KERN;
mib[1] = KERN_PROC_ARGS;
mib[2] = p->p_pid;
mib[3] = KERN_PROC_ARGV;
while (argbuf) {
ret = sysctl(mib, 4, argbuf, &arglen, NULL, 0);
if (ret == -1 && errno == ENOMEM) {
char *n;
n = realloc(argbuf, arglen * 2);
if (n != 0) {
argbuf = n;
arglen *= 2;
continue;
}
}
break;
}
if (ret != 1) {
char **argv;
int argc;
argv = (char **)argbuf;
if (argv[0] != NULL)
strlcpy(cmdline, argv[0], maxlen);
for (argc = 1; argv[argc] != NULL; argc++) {
strlcat(cmdline, " ", maxlen);
strlcat(cmdline, argv[argc], maxlen);
}
} else {
strlcpy(cmdline, p->p_comm, maxlen);
}
}
static int
updateProcess(struct kinfo_proc *p)
{
static char *statuses[] = { "idle","run","sleep","stop","zombie" };
ProcessInfo* ps;
struct passwd* pwent;
pid_t pid = p->p_pid;
if ((ps = findProcessInList(pid)) == 0)
{
ps = (ProcessInfo*) malloc(sizeof(ProcessInfo));
ps->pid = pid;
ps->centStamp = 0;
push_ctnr(ProcessList, ps);
bsort_ctnr(ProcessList, processCmp);
}
ps->alive = 1;
ps->pid = p->p_pid;
ps->ppid = p->p_ppid;
ps->uid = p->p_uid;
ps->gid = p->p_gid;
ps->priority = p->p_priority;
ps->niceLevel = p->p_nice;
/* this isn't usertime -- it's total time (??) */
ps->userTime = p->p_uutime_sec*100+p->p_uutime_usec/100;
ps->sysTime = 0;
ps->sysLoad = 0;
/* memory, process name, process uid */
/* find out user name with process uid */
pwent = getpwuid(ps->uid);
strlcpy(ps->userName,pwent&&pwent->pw_name? pwent->pw_name:"????",sizeof(ps->userName));
ps->userName[sizeof(ps->userName)-1]='\0';
ps->userLoad = p->p_pctcpu / 100;
ps->vmSize = (p->p_vm_tsize +
p->p_vm_dsize +
p->p_vm_ssize) * getpagesize();
ps->vmRss = p->p_vm_rssize * getpagesize();
strlcpy(ps->name,p->p_comm ? p->p_comm : "????", sizeof(ps->name));
strlcpy(ps->status,(p->p_stat>=1)&&(p->p_stat<=5)? statuses[p->p_stat-1]:"????", sizeof(ps->status));
fillProcessCmdline(ps->cmdline, p, sizeof(ps->cmdline));
/* process command line */
return (0);
}
static void
cleanupProcessList(void)
{
ProcessInfo* ps;
ProcessCount = 0;
/* All processes that do not have the active flag set are assumed dead
* and will be removed from the list. The alive flag is cleared. */
for (ps = first_ctnr(ProcessList); ps; ps = next_ctnr(ProcessList))
{
if (ps->alive)
{
/* Process is still alive. Just clear flag. */
ps->alive = 0;
ProcessCount++;
}
else
{
/* Process has probably died. We remove it from the list and
* destruct the data structure. i needs to be decremented so
* that after i++ the next list element will be inspected. */
free(remove_ctnr(ProcessList));
}
}
}
/*
================================ public part ==================================
*/
void
initProcessList(struct SensorModul* sm)
{
ProcessList = new_ctnr();
registerMonitor("ps", "table", printProcessList, printProcessListInfo, sm);
registerMonitor("pscount", "integer", printProcessCount, printProcessCountInfo, sm);
if (!RunAsDaemon)
{
registerCommand("kill", killProcess);
registerCommand("setpriority", setPriority);
}
updateProcessList();
}
void
exitProcessList(void)
{
removeMonitor("ps");
removeMonitor("pscount");
if (ProcessList)
free (ProcessList);
}
int
updateProcessList(void)
{
int mib[6];
size_t len;
size_t num;
struct kinfo_proc *p;
mib[0] = CTL_KERN;
mib[1] = KERN_PROC;
mib[2] = KERN_PROC_ALL;
mib[3] = 0;
mib[4] = sizeof(struct kinfo_proc);
mib[5] = 0;
if (sysctl(mib, 6, NULL, &len, NULL, 0) == -1)
return 0;
len = 5 * len / 4;
p = malloc(len);
if (!p)
return 0;
mib[5] = len/ sizeof(struct kinfo_proc);
if (sysctl(mib, 6, p, &len, NULL, 0) == -1)
return 0;
for (num = 0; num < len / sizeof(struct kinfo_proc); num++)
updateProcess(&p[num]);
free(p);
cleanupProcessList();
return (0);
}
void
printProcessListInfo(const char* cmd)
{
fprintf(CurrentClient, "Name\tPID\tPPID\tUID\tGID\tStatus\tUser%%\tSystem%%\tNice\tVmSize\tVmRss\tLogin\tCommand\n");
fprintf(CurrentClient, "s\td\td\td\td\tS\tf\tf\td\tD\tD\ts\ts\n");
}
void
printProcessList(const char* cmd)
{
ProcessInfo* ps;
for (ps = first_ctnr(ProcessList); ps; ps = next_ctnr(ProcessList))
{
fprintf(CurrentClient, "%s\t%ld\t%ld\t%ld\t%ld\t%s\t%.2f\t%.2f\t%d\t%d\t%d\t%s\t%s\n",
ps->name, (long)ps->pid, (long)ps->ppid,
(long)ps->uid, (long)ps->gid, ps->status,
ps->userLoad, ps->sysLoad, ps->niceLevel,
ps->vmSize / 1024, ps->vmRss / 1024, ps->userName, ps->cmdline);
}
}
void
printProcessCount(const char* cmd)
{
fprintf(CurrentClient, "%d\n", ProcessCount);
}
void
printProcessCountInfo(const char* cmd)
{
fprintf(CurrentClient, "Number of Processes\t1\t65535\t\n");
}
void
killProcess(const char* cmd)
{
int sig, pid;
sscanf(cmd, "%*s %d %d", &pid, &sig);
switch(sig)
{
case MENU_ID_SIGABRT:
sig = SIGABRT;
break;
case MENU_ID_SIGALRM:
sig = SIGALRM;
break;
case MENU_ID_SIGCHLD:
sig = SIGCHLD;
break;
case MENU_ID_SIGCONT:
sig = SIGCONT;
break;
case MENU_ID_SIGFPE:
sig = SIGFPE;
break;
case MENU_ID_SIGHUP:
sig = SIGHUP;
break;
case MENU_ID_SIGILL:
sig = SIGILL;
break;
case MENU_ID_SIGINT:
sig = SIGINT;
break;
case MENU_ID_SIGKILL:
sig = SIGKILL;
break;
case MENU_ID_SIGPIPE:
sig = SIGPIPE;
break;
case MENU_ID_SIGQUIT:
sig = SIGQUIT;
break;
case MENU_ID_SIGSEGV:
sig = SIGSEGV;
break;
case MENU_ID_SIGSTOP:
sig = SIGSTOP;
break;
case MENU_ID_SIGTERM:
sig = SIGTERM;
break;
case MENU_ID_SIGTSTP:
sig = SIGTSTP;
break;
case MENU_ID_SIGTTIN:
sig = SIGTTIN;
break;
case MENU_ID_SIGTTOU:
sig = SIGTTOU;
break;
case MENU_ID_SIGUSR1:
sig = SIGUSR1;
break;
case MENU_ID_SIGUSR2:
sig = SIGUSR2;
break;
}
if (kill((pid_t) pid, sig))
{
switch(errno)
{
case EINVAL:
fprintf(CurrentClient, "4\t%d\n", pid);
break;
case ESRCH:
fprintf(CurrentClient, "3\t%d\n", pid);
break;
case EPERM:
fprintf(CurrentClient, "2\t%d\n", pid);
break;
default:
fprintf(CurrentClient, "1\t%d\n", pid); /* unknown error */
break;
}
}
else
fprintf(CurrentClient, "0\t%d\n", pid);
}
void
setPriority(const char* cmd)
{
int pid, prio;
sscanf(cmd, "%*s %d %d", &pid, &prio);
if (setpriority(PRIO_PROCESS, pid, prio))
{
switch(errno)
{
case EINVAL:
fprintf(CurrentClient, "4\n");
break;
case ESRCH:
fprintf(CurrentClient, "3\n");
break;
case EPERM:
case EACCES:
fprintf(CurrentClient, "2\n");
break;
default:
fprintf(CurrentClient, "1\n"); /* unknown error */
break;
}
}
else
fprintf(CurrentClient, "0\n");
}