mirror of
git://git.suckless.org/ubase
synced 2024-12-26 00:32:58 +00:00
127 lines
2.6 KiB
C
127 lines
2.6 KiB
C
/* See LICENSE file for copyright and license details. */
|
|
#include <sys/stat.h>
|
|
#include <fcntl.h>
|
|
#include <unistd.h>
|
|
#include <errno.h>
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <limits.h>
|
|
#include "../proc.h"
|
|
#include "../util.h"
|
|
|
|
/* TODO: Unify proc{euid,uid}() into parsestatus() */
|
|
int
|
|
proceuid(pid_t pid, uid_t *euid)
|
|
{
|
|
FILE *fp;
|
|
char buf[BUFSIZ], *p;
|
|
char path[PATH_MAX];
|
|
uid_t procuid, proceuid;
|
|
|
|
snprintf(path, sizeof(path), "/proc/%d/status", pid);
|
|
if (!(fp = fopen(path, "r")))
|
|
eprintf("%s fopen:", path);
|
|
while (fgets(buf, sizeof(buf), fp)) {
|
|
if (!strncmp(buf, "Uid:", 4)) {
|
|
p = buf + strlen("Uid:");
|
|
sscanf(p, "%u %u", &procuid, &proceuid);
|
|
*euid = proceuid;
|
|
fclose(fp);
|
|
return 0;
|
|
}
|
|
}
|
|
fclose(fp);
|
|
return -1;
|
|
}
|
|
|
|
int
|
|
procuid(pid_t pid, uid_t *uid)
|
|
{
|
|
FILE *fp;
|
|
char buf[BUFSIZ], *p;
|
|
char path[PATH_MAX];
|
|
uid_t procuid;
|
|
|
|
snprintf(path, sizeof(path), "/proc/%d/status", pid);
|
|
if (!(fp = fopen(path, "r")))
|
|
eprintf("%s fopen:", path);
|
|
while (fgets(buf, sizeof(buf), fp)) {
|
|
if (!strncmp(buf, "Uid:", 4)) {
|
|
p = buf + strlen("Uid:");
|
|
sscanf(p, "%u", &procuid);
|
|
*uid = procuid;
|
|
fclose(fp);
|
|
return 0;
|
|
}
|
|
}
|
|
fclose(fp);
|
|
return -1;
|
|
}
|
|
|
|
int
|
|
parsecmdline(pid_t pid, char *buf, size_t siz)
|
|
{
|
|
int fd;
|
|
char path[PATH_MAX];
|
|
ssize_t n, i;
|
|
|
|
snprintf(path, sizeof(path), "/proc/%d/cmdline", pid);
|
|
fd = open(path, O_RDONLY);
|
|
if (fd < 0)
|
|
eprintf("open %s:", path);
|
|
n = read(fd, buf, siz - 1);
|
|
if (n < 0)
|
|
eprintf("read %s:", path);
|
|
if (!n) {
|
|
close(fd);
|
|
return -1;
|
|
}
|
|
buf[n] = '\0';
|
|
for (i = 0; i < n; i++)
|
|
if (buf[i] == '\0')
|
|
buf[i] = ' ';
|
|
close(fd);
|
|
return 0;
|
|
}
|
|
|
|
int
|
|
parsestat(pid_t pid, struct procstat *ps)
|
|
{
|
|
char path[PATH_MAX];
|
|
FILE *fp;
|
|
|
|
snprintf(path, sizeof(path), "/proc/%d/stat", pid);
|
|
if (!(fp = fopen(path, "r")))
|
|
eprintf("fopen %s:", path);
|
|
fscanf(fp, "%d %s %c %d %d %d %d %d %u %lu %lu %lu %lu %lu %lu",
|
|
&ps->pid, ps->comm,
|
|
&ps->state, &ps->ppid, &ps->pgrp,
|
|
&ps->sid, &ps->tty_nr, &ps->tpgid, &ps->flags,
|
|
&ps->minflt, &ps->cminflt, &ps->majflt, &ps->cmajflt,
|
|
&ps->utime, &ps->stime);
|
|
fscanf(fp, "%ld %ld %ld %ld %ld %ld %lld %lu %ld %ld",
|
|
&ps->cutime, &ps->cstime, &ps->priority, &ps->nice,
|
|
&ps->num_threads, &ps->itrealvalue, &ps->starttime,
|
|
&ps->vsize, &ps->rss, &ps->rsslim);
|
|
/* Filter out '(' and ')' from comm */
|
|
ps->comm[strlen(ps->comm) - 1] = '\0';
|
|
memmove(ps->comm, ps->comm + 1, strlen(ps->comm));
|
|
fclose(fp);
|
|
return 0;
|
|
}
|
|
|
|
int
|
|
validps(const char *path)
|
|
{
|
|
char *end;
|
|
|
|
errno = 0;
|
|
strtol(path, &end, 10);
|
|
if (*end != '\0')
|
|
return 0;
|
|
if (errno != 0)
|
|
return 0;
|
|
return 1;
|
|
}
|