1
0
mirror of git://git.suckless.org/ubase synced 2025-01-03 13:02:01 +00:00

Support -f for ps(1)

Fix missing break in tty.c as well.
This commit is contained in:
sin 2013-08-16 14:08:20 +01:00
parent 74146c54a7
commit e84531958e
4 changed files with 107 additions and 12 deletions

2
proc.h
View File

@ -27,6 +27,8 @@ struct procstat {
long rsslim; long rsslim;
}; };
int parsecmdline(pid_t pid, char *buf, size_t siz);
int parsestat(pid_t pid, struct procstat *ps); int parsestat(pid_t pid, struct procstat *ps);
int proceuid(pid_t pid, uid_t *euid); int proceuid(pid_t pid, uid_t *euid);
int procuid(pid_t pid, uid_t *euid);
int validps(const char *path); int validps(const char *path);

54
ps.c
View File

@ -1,10 +1,13 @@
/* See LICENSE file for copyright and license details. */ /* See LICENSE file for copyright and license details. */
#include <sys/sysinfo.h>
#include <errno.h> #include <errno.h>
#include <unistd.h> #include <unistd.h>
#include <stdio.h> #include <stdio.h>
#include <stdlib.h> #include <stdlib.h>
#include <string.h> #include <string.h>
#include <limits.h> #include <limits.h>
#include <time.h>
#include <pwd.h>
#include "proc.h" #include "proc.h"
#include "util.h" #include "util.h"
@ -15,7 +18,8 @@ static void psr(const char *path);
enum { enum {
PS_aflag = 1 << 0, PS_aflag = 1 << 0,
PS_Aflag = 1 << 1, PS_Aflag = 1 << 1,
PS_dflag = 1 << 2 PS_dflag = 1 << 2,
PS_fflag = 1 << 3
}; };
static int flags; static int flags;
@ -37,14 +41,17 @@ main(int argc, char *argv[])
flags |= PS_Aflag; flags |= PS_Aflag;
break; break;
case 'f': case 'f':
eprintf("not implemented\n"); flags |= PS_fflag;
break;
default: default:
usage(); usage();
} ARGEND; } ARGEND;
if (!(flags & PS_fflag))
printf(" PID TTY TIME CMD\n"); printf(" PID TTY TIME CMD\n");
else
printf("UID PID PPID C STIME TTY TIME CMD\n");
recurse("/proc", psr); recurse("/proc", psr);
return 0; return 0;
} }
@ -57,10 +64,16 @@ usage(void)
static void static void
psout(struct procstat *ps) psout(struct procstat *ps)
{ {
char cmdline[BUFSIZ], *cmd;
char stimestr[6];
char *ttystr, *myttystr; char *ttystr, *myttystr;
int tty_maj, tty_min; int tty_maj, tty_min;
uid_t myeuid, peuid; uid_t myeuid, peuid, puid;
unsigned sut; unsigned sut;
struct sysinfo info;
struct tm *tm;
time_t start;
struct passwd *pw;
/* Ignore session leaders */ /* Ignore session leaders */
if (flags & PS_dflag) if (flags & PS_dflag)
@ -72,13 +85,15 @@ psout(struct procstat *ps)
devtotty(ps->tty_nr, &tty_maj, &tty_min); devtotty(ps->tty_nr, &tty_maj, &tty_min);
ttystr = ttytostr(tty_maj, tty_min); ttystr = ttytostr(tty_maj, tty_min);
/* Only print processes that are associated with /* Only print processes that are associated with
* a terminal */ * a terminal and they are not session leaders */
if (ttystr[0] == '?' && (flags & PS_aflag)) { if (flags & PS_aflag) {
if (ps->pid == ps->sid || ttystr[0] == '?') {
free(ttystr); free(ttystr);
return; return;
} }
}
if (!flags) { if (!(flags & (PS_aflag | PS_Aflag | PS_dflag))) {
myttystr = ttyname(STDIN_FILENO); myttystr = ttyname(STDIN_FILENO);
if (myttystr) { if (myttystr) {
if (strcmp(myttystr + strlen("/dev/"), ttystr)) { if (strcmp(myttystr + strlen("/dev/"), ttystr)) {
@ -97,8 +112,33 @@ psout(struct procstat *ps)
} }
} }
procuid(ps->pid, &puid);
errno = 0;
pw = getpwuid(puid);
if (errno || !pw)
eprintf("getpwuid %d:", puid);
sysinfo(&info);
start = time(NULL) - (info.uptime - (ps->starttime / 100));
tm = localtime(&start);
strftime(stimestr, sizeof(stimestr),
"%H:%M", tm);
if (parsecmdline(ps->pid, cmdline, sizeof(cmdline)) < 0)
cmd = ps->comm;
else
cmd = cmdline;
if (!(flags & PS_fflag))
printf("%5d %-6s %02u:%02u:%02u %s\n", ps->pid, ttystr, printf("%5d %-6s %02u:%02u:%02u %s\n", ps->pid, ttystr,
sut / 3600, (sut % 3600) / 60, sut % 60, ps->comm); sut / 3600, (sut % 3600) / 60, sut % 60, ps->comm);
else {
printf("%-8s %5d %5d ? %5s %-5s %02u:%02u:%02u %s%s%s\n",
pw->pw_name, ps->pid,
ps->ppid, stimestr, ttystr,
sut / 3600, (sut % 3600) / 60, sut % 60,
(cmd == ps->comm) ? "[" : "", cmd,
(cmd == ps->comm) ? "]" : "");
}
free(ttystr); free(ttystr);
} }

View File

@ -1,4 +1,6 @@
/* See LICENSE file for copyright and license details. */ /* See LICENSE file for copyright and license details. */
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h> #include <unistd.h>
#include <errno.h> #include <errno.h>
#include <stdio.h> #include <stdio.h>
@ -32,6 +34,56 @@ proceuid(pid_t pid, uid_t *euid)
return -1; 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 int
parsestat(pid_t pid, struct procstat *ps) parsestat(pid_t pid, struct procstat *ps)
{ {

View File

@ -30,6 +30,7 @@ ttytostr(int tty_maj, int tty_min)
break; break;
case 4: case 4:
snprintf(ttystr, len, "%s%d", tty, tty_min); snprintf(ttystr, len, "%s%d", tty, tty_min);
break;
default: default:
ttystr[0] = '?'; ttystr[0] = '?';
ttystr[1] = '\0'; ttystr[1] = '\0';