ps,top: add an option to show threads. +260 bytes of code
Signed-off-by: Denys Vlasenko <vda.linux@googlemail.com>
This commit is contained in:
parent
f00cfdfae5
commit
b410d4ada7
@ -1275,6 +1275,7 @@ enum { COMM_LEN = 16 };
|
|||||||
#endif
|
#endif
|
||||||
typedef struct procps_status_t {
|
typedef struct procps_status_t {
|
||||||
DIR *dir;
|
DIR *dir;
|
||||||
|
IF_FEATURE_SHOW_THREADS(DIR *task_dir;)
|
||||||
uint8_t shift_pages_to_bytes;
|
uint8_t shift_pages_to_bytes;
|
||||||
uint8_t shift_pages_to_kb;
|
uint8_t shift_pages_to_kb;
|
||||||
/* Fields are set to 0/NULL if failed to determine (or not requested) */
|
/* Fields are set to 0/NULL if failed to determine (or not requested) */
|
||||||
@ -1348,6 +1349,7 @@ enum {
|
|||||||
PSSCAN_CPU = (1 << 19) * ENABLE_FEATURE_TOP_SMP_PROCESS,
|
PSSCAN_CPU = (1 << 19) * ENABLE_FEATURE_TOP_SMP_PROCESS,
|
||||||
PSSCAN_NICE = (1 << 20) * ENABLE_FEATURE_PS_ADDITIONAL_COLUMNS,
|
PSSCAN_NICE = (1 << 20) * ENABLE_FEATURE_PS_ADDITIONAL_COLUMNS,
|
||||||
PSSCAN_RUIDGID = (1 << 21) * ENABLE_FEATURE_PS_ADDITIONAL_COLUMNS,
|
PSSCAN_RUIDGID = (1 << 21) * ENABLE_FEATURE_PS_ADDITIONAL_COLUMNS,
|
||||||
|
PSSCAN_TASKS = (1 << 22) * ENABLE_FEATURE_SHOW_THREADS,
|
||||||
/* These are all retrieved from proc/NN/stat in one go: */
|
/* These are all retrieved from proc/NN/stat in one go: */
|
||||||
PSSCAN_STAT = PSSCAN_PPID | PSSCAN_PGID | PSSCAN_SID
|
PSSCAN_STAT = PSSCAN_PPID | PSSCAN_PGID | PSSCAN_SID
|
||||||
/**/ | PSSCAN_COMM | PSSCAN_STATE
|
/**/ | PSSCAN_COMM | PSSCAN_STATE
|
||||||
|
@ -110,6 +110,10 @@ static procps_status_t* FAST_FUNC alloc_procps_scan(void)
|
|||||||
void FAST_FUNC free_procps_scan(procps_status_t* sp)
|
void FAST_FUNC free_procps_scan(procps_status_t* sp)
|
||||||
{
|
{
|
||||||
closedir(sp->dir);
|
closedir(sp->dir);
|
||||||
|
#if ENABLE_FEATURE_SHOW_THREADS
|
||||||
|
if (sp->task_dir)
|
||||||
|
closedir(sp->task_dir);
|
||||||
|
#endif
|
||||||
free(sp->argv0);
|
free(sp->argv0);
|
||||||
free(sp->exe);
|
free(sp->exe);
|
||||||
IF_SELINUX(free(sp->context);)
|
IF_SELINUX(free(sp->context);)
|
||||||
@ -189,14 +193,35 @@ procps_status_t* FAST_FUNC procps_scan(procps_status_t* sp, int flags)
|
|||||||
sp = alloc_procps_scan();
|
sp = alloc_procps_scan();
|
||||||
|
|
||||||
for (;;) {
|
for (;;) {
|
||||||
|
#if ENABLE_FEATURE_SHOW_THREADS
|
||||||
|
if ((flags & PSSCAN_TASKS) && sp->task_dir) {
|
||||||
|
entry = readdir(sp->task_dir);
|
||||||
|
if (entry)
|
||||||
|
goto got_entry;
|
||||||
|
closedir(sp->task_dir);
|
||||||
|
sp->task_dir = NULL;
|
||||||
|
}
|
||||||
|
#endif
|
||||||
entry = readdir(sp->dir);
|
entry = readdir(sp->dir);
|
||||||
if (entry == NULL) {
|
if (entry == NULL) {
|
||||||
free_procps_scan(sp);
|
free_procps_scan(sp);
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
IF_FEATURE_SHOW_THREADS(got_entry:)
|
||||||
pid = bb_strtou(entry->d_name, NULL, 10);
|
pid = bb_strtou(entry->d_name, NULL, 10);
|
||||||
if (errno)
|
if (errno)
|
||||||
continue;
|
continue;
|
||||||
|
#if ENABLE_FEATURE_SHOW_THREADS
|
||||||
|
if ((flags & PSSCAN_TASKS) && !sp->task_dir) {
|
||||||
|
/* We found another /proc/PID. Do not use it,
|
||||||
|
* there will be /proc/PID/task/PID (same PID!),
|
||||||
|
* so just go ahead and dive into /proc/PID/task. */
|
||||||
|
char task_dir[sizeof("/proc/%u/task") + sizeof(int)*3];
|
||||||
|
sprintf(task_dir, "/proc/%u/task", pid);
|
||||||
|
sp->task_dir = xopendir(task_dir);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
/* After this point we have to break, not continue
|
/* After this point we have to break, not continue
|
||||||
* ("continue" would mean that current /proc/NNN
|
* ("continue" would mean that current /proc/NNN
|
||||||
|
@ -188,6 +188,13 @@ config FEATURE_TOPMEM
|
|||||||
help
|
help
|
||||||
Enable 's' in top (gives lots of memory info).
|
Enable 's' in top (gives lots of memory info).
|
||||||
|
|
||||||
|
config FEATURE_SHOW_THREADS
|
||||||
|
bool "Support for showing threads in ps/top"
|
||||||
|
default n
|
||||||
|
depends on PS || TOP
|
||||||
|
help
|
||||||
|
Enables ps -T option and 'h' command in top
|
||||||
|
|
||||||
config UPTIME
|
config UPTIME
|
||||||
bool "uptime"
|
bool "uptime"
|
||||||
default n
|
default n
|
||||||
@ -203,5 +210,4 @@ config WATCH
|
|||||||
watch is used to execute a program periodically, showing
|
watch is used to execute a program periodically, showing
|
||||||
output to the screen.
|
output to the screen.
|
||||||
|
|
||||||
|
|
||||||
endmenu
|
endmenu
|
||||||
|
31
procps/ps.c
31
procps/ps.c
@ -17,7 +17,6 @@ enum { MAX_WIDTH = 2*1024 };
|
|||||||
#if ENABLE_DESKTOP
|
#if ENABLE_DESKTOP
|
||||||
|
|
||||||
#include <sys/times.h> /* for times() */
|
#include <sys/times.h> /* for times() */
|
||||||
//#include <sys/sysinfo.h> /* for sysinfo() */
|
|
||||||
#ifndef AT_CLKTCK
|
#ifndef AT_CLKTCK
|
||||||
#define AT_CLKTCK 17
|
#define AT_CLKTCK 17
|
||||||
#endif
|
#endif
|
||||||
@ -61,6 +60,7 @@ struct globals {
|
|||||||
#define kernel_HZ (G.kernel_HZ )
|
#define kernel_HZ (G.kernel_HZ )
|
||||||
#define seconds_since_boot (G.seconds_since_boot)
|
#define seconds_since_boot (G.seconds_since_boot)
|
||||||
#define default_o (G.default_o )
|
#define default_o (G.default_o )
|
||||||
|
#define INIT_G() do { } while (0)
|
||||||
|
|
||||||
#if ENABLE_FEATURE_PS_TIME
|
#if ENABLE_FEATURE_PS_TIME
|
||||||
/* for ELF executables, notes are pushed before environment and args */
|
/* for ELF executables, notes are pushed before environment and args */
|
||||||
@ -452,21 +452,34 @@ int ps_main(int argc UNUSED_PARAM, char **argv)
|
|||||||
{
|
{
|
||||||
procps_status_t *p;
|
procps_status_t *p;
|
||||||
llist_t* opt_o = NULL;
|
llist_t* opt_o = NULL;
|
||||||
IF_SELINUX(int opt;)
|
int opt;
|
||||||
|
enum {
|
||||||
|
OPT_Z = (1 << 0),
|
||||||
|
OPT_o = (1 << 1),
|
||||||
|
OPT_a = (1 << 2),
|
||||||
|
OPT_A = (1 << 3),
|
||||||
|
OPT_d = (1 << 4),
|
||||||
|
OPT_e = (1 << 5),
|
||||||
|
OPT_f = (1 << 6),
|
||||||
|
OPT_l = (1 << 7),
|
||||||
|
OPT_T = (1 << 8) * ENABLE_FEATURE_SHOW_THREADS,
|
||||||
|
};
|
||||||
|
|
||||||
|
INIT_G();
|
||||||
|
|
||||||
// POSIX:
|
// POSIX:
|
||||||
// -a Write information for all processes associated with terminals
|
// -a Write information for all processes associated with terminals
|
||||||
// Implementations may omit session leaders from this list
|
// Implementations may omit session leaders from this list
|
||||||
// -A Write information for all processes
|
// -A Write information for all processes
|
||||||
// -d Write information for all processes, except session leaders
|
// -d Write information for all processes, except session leaders
|
||||||
// -e Write information for all processes (equivalent to -A.)
|
// -e Write information for all processes (equivalent to -A)
|
||||||
// -f Generate a full listing
|
// -f Generate a full listing
|
||||||
// -l Generate a long listing
|
// -l Generate a long listing
|
||||||
// -o col1,col2,col3=header
|
// -o col1,col2,col3=header
|
||||||
// Select which columns to display
|
// Select which columns to display
|
||||||
/* We allow (and ignore) most of the above. FIXME */
|
/* We allow (and ignore) most of the above. FIXME */
|
||||||
opt_complementary = "o::";
|
opt_complementary = "o::";
|
||||||
IF_SELINUX(opt =) getopt32(argv, "Zo:aAdefl", &opt_o);
|
opt = getopt32(argv, "Zo:aAdefl" IF_FEATURE_SHOW_THREADS("T"), &opt_o);
|
||||||
if (opt_o) {
|
if (opt_o) {
|
||||||
do {
|
do {
|
||||||
parse_o(llist_pop(&opt_o));
|
parse_o(llist_pop(&opt_o));
|
||||||
@ -474,7 +487,7 @@ int ps_main(int argc UNUSED_PARAM, char **argv)
|
|||||||
} else {
|
} else {
|
||||||
/* Below: parse_o() needs char*, NOT const char*... */
|
/* Below: parse_o() needs char*, NOT const char*... */
|
||||||
#if ENABLE_SELINUX
|
#if ENABLE_SELINUX
|
||||||
if (!(opt & 1) || !is_selinux_enabled()) {
|
if (!(opt & OPT_Z) || !is_selinux_enabled()) {
|
||||||
/* no -Z or no SELinux: do not show LABEL */
|
/* no -Z or no SELinux: do not show LABEL */
|
||||||
strcpy(default_o, DEFAULT_O_STR + sizeof(SELINUX_O_PREFIX)-1);
|
strcpy(default_o, DEFAULT_O_STR + sizeof(SELINUX_O_PREFIX)-1);
|
||||||
} else
|
} else
|
||||||
@ -485,6 +498,10 @@ int ps_main(int argc UNUSED_PARAM, char **argv)
|
|||||||
parse_o(default_o);
|
parse_o(default_o);
|
||||||
}
|
}
|
||||||
post_process();
|
post_process();
|
||||||
|
#if ENABLE_FEATURE_SHOW_THREADS
|
||||||
|
if (opt & OPT_T)
|
||||||
|
need_flags |= PSSCAN_TASKS;
|
||||||
|
#endif
|
||||||
|
|
||||||
/* Was INT_MAX, but some libc's go belly up with printf("%.*s")
|
/* Was INT_MAX, but some libc's go belly up with printf("%.*s")
|
||||||
* and such large widths */
|
* and such large widths */
|
||||||
@ -497,7 +514,7 @@ int ps_main(int argc UNUSED_PARAM, char **argv)
|
|||||||
format_header();
|
format_header();
|
||||||
|
|
||||||
p = NULL;
|
p = NULL;
|
||||||
while ((p = procps_scan(p, need_flags))) {
|
while ((p = procps_scan(p, need_flags)) != NULL) {
|
||||||
format_process(p);
|
format_process(p);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -558,7 +575,7 @@ int ps_main(int argc UNUSED_PARAM, char **argv UNUSED_PARAM)
|
|||||||
| PSSCAN_VSZ
|
| PSSCAN_VSZ
|
||||||
| PSSCAN_COMM
|
| PSSCAN_COMM
|
||||||
| use_selinux
|
| use_selinux
|
||||||
))) {
|
)) != NULL) {
|
||||||
#if ENABLE_SELINUX
|
#if ENABLE_SELINUX
|
||||||
if (use_selinux) {
|
if (use_selinux) {
|
||||||
len = printf("%5u %-32.32s %s ",
|
len = printf("%5u %-32.32s %s ",
|
||||||
|
21
procps/top.c
21
procps/top.c
@ -976,7 +976,10 @@ int top_main(int argc UNUSED_PARAM, char **argv)
|
|||||||
/* read process IDs & status for all the processes */
|
/* read process IDs & status for all the processes */
|
||||||
while ((p = procps_scan(p, scan_mask)) != NULL) {
|
while ((p = procps_scan(p, scan_mask)) != NULL) {
|
||||||
int n;
|
int n;
|
||||||
if (scan_mask == TOP_MASK) {
|
#if ENABLE_FEATURE_TOPMEM
|
||||||
|
if (scan_mask != TOPMEM_MASK)
|
||||||
|
#endif
|
||||||
|
{
|
||||||
n = ntop;
|
n = ntop;
|
||||||
top = xrealloc_vector(top, 6, ntop++);
|
top = xrealloc_vector(top, 6, ntop++);
|
||||||
top[n].pid = p->pid;
|
top[n].pid = p->pid;
|
||||||
@ -991,8 +994,9 @@ int top_main(int argc UNUSED_PARAM, char **argv)
|
|||||||
#if ENABLE_FEATURE_TOP_SMP_PROCESS
|
#if ENABLE_FEATURE_TOP_SMP_PROCESS
|
||||||
top[n].last_seen_on_cpu = p->last_seen_on_cpu;
|
top[n].last_seen_on_cpu = p->last_seen_on_cpu;
|
||||||
#endif
|
#endif
|
||||||
} else { /* TOPMEM */
|
}
|
||||||
#if ENABLE_FEATURE_TOPMEM
|
#if ENABLE_FEATURE_TOPMEM
|
||||||
|
else { /* TOPMEM */
|
||||||
if (!(p->mapped_ro | p->mapped_rw))
|
if (!(p->mapped_ro | p->mapped_rw))
|
||||||
continue; /* kernel threads are ignored */
|
continue; /* kernel threads are ignored */
|
||||||
n = ntop;
|
n = ntop;
|
||||||
@ -1007,15 +1011,15 @@ int top_main(int argc UNUSED_PARAM, char **argv)
|
|||||||
topmem[n].dirty = p->private_dirty + p->shared_dirty;
|
topmem[n].dirty = p->private_dirty + p->shared_dirty;
|
||||||
topmem[n].dirty_sh = p->shared_dirty;
|
topmem[n].dirty_sh = p->shared_dirty;
|
||||||
topmem[n].stack = p->stack;
|
topmem[n].stack = p->stack;
|
||||||
#endif
|
|
||||||
}
|
}
|
||||||
|
#endif
|
||||||
} /* end of "while we read /proc" */
|
} /* end of "while we read /proc" */
|
||||||
if (ntop == 0) {
|
if (ntop == 0) {
|
||||||
bb_error_msg("no process info in /proc");
|
bb_error_msg("no process info in /proc");
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (scan_mask == TOP_MASK) {
|
if (scan_mask != TOPMEM_MASK) {
|
||||||
#if ENABLE_FEATURE_TOP_CPU_USAGE_PERCENTAGE
|
#if ENABLE_FEATURE_TOP_CPU_USAGE_PERCENTAGE
|
||||||
if (!prev_hist_count) {
|
if (!prev_hist_count) {
|
||||||
do_stats();
|
do_stats();
|
||||||
@ -1039,7 +1043,7 @@ int top_main(int argc UNUSED_PARAM, char **argv)
|
|||||||
if (OPT_BATCH_MODE) {
|
if (OPT_BATCH_MODE) {
|
||||||
lines_rem = INT_MAX;
|
lines_rem = INT_MAX;
|
||||||
}
|
}
|
||||||
if (scan_mask == TOP_MASK)
|
if (scan_mask != TOPMEM_MASK)
|
||||||
display_process_list(lines_rem, col);
|
display_process_list(lines_rem, col);
|
||||||
#if ENABLE_FEATURE_TOPMEM
|
#if ENABLE_FEATURE_TOPMEM
|
||||||
else
|
else
|
||||||
@ -1076,6 +1080,13 @@ int top_main(int argc UNUSED_PARAM, char **argv)
|
|||||||
sort_function[2] = time_sort;
|
sort_function[2] = time_sort;
|
||||||
# endif
|
# endif
|
||||||
}
|
}
|
||||||
|
#if ENABLE_FEATURE_SHOW_THREADS
|
||||||
|
if (c == 'h'
|
||||||
|
IF_FEATURE_TOPMEM(&& scan_mask != TOPMEM_MASK)
|
||||||
|
) {
|
||||||
|
scan_mask ^= PSSCAN_TASKS;
|
||||||
|
}
|
||||||
|
#endif
|
||||||
# if ENABLE_FEATURE_TOP_CPU_USAGE_PERCENTAGE
|
# if ENABLE_FEATURE_TOP_CPU_USAGE_PERCENTAGE
|
||||||
if (c == 'p') {
|
if (c == 'p') {
|
||||||
IF_FEATURE_TOPMEM(scan_mask = TOP_MASK;)
|
IF_FEATURE_TOPMEM(scan_mask = TOP_MASK;)
|
||||||
|
Loading…
x
Reference in New Issue
Block a user