summaryrefslogtreecommitdiff
path: root/libps/procstat.c
diff options
context:
space:
mode:
Diffstat (limited to 'libps/procstat.c')
-rw-r--r--libps/procstat.c120
1 files changed, 91 insertions, 29 deletions
diff --git a/libps/procstat.c b/libps/procstat.c
index 138612a1..fd91669c 100644
--- a/libps/procstat.c
+++ b/libps/procstat.c
@@ -31,7 +31,7 @@
/* These directly correspond to the bits in a state, starting from 0. See
ps.h for an explanation of what each of these means. */
-char *proc_stat_state_tags = "RTHDSIWN<Z+sempox";
+char *proc_stat_state_tags = "RTHDSIWN<Z+sfmpox";
/* ---------------------------------------------------------------- */
@@ -76,25 +76,11 @@ thread_state(thread_basic_info_t bi)
/* ---------------------------------------------------------------- */
-/* Add FLAGS to PS's flags, fetching information as necessary to validate
- the corresponding fields in PS. Afterwards you must still check the flags
- field before using new fields, as something might have failed. Returns
- a system error code if a fatal error occurred, or 0 if none. */
-error_t
-proc_stat_set_flags(proc_stat_t ps, ps_flags_t flags)
+/* Returns FLAGS augmented with any other flags that are necessary
+ preconditions to setting them. */
+static ps_flags_t
+add_preconditions (ps_flags_t flags)
{
- error_t err;
- ps_flags_t have = ps->flags; /* flags set in ps */
- ps_flags_t need; /* flags not set in ps, but desired to be */
- process_t server = ps_context_server(ps->context);
-
- if (flags & PSTAT_NO_MSGPORT)
- /* Ensure that we don't try to use the process's msg port. */
- {
- have |= PSTAT_NO_MSGPORT;
- have &= ~PSTAT_MSGPORT;
- }
-
/* Implement any inter-flag dependencies: if the new flags in FLAGS depend
on some other set of flags to be set, make sure those are also in
FLAGS. */
@@ -104,7 +90,7 @@ proc_stat_set_flags(proc_stat_t ps, ps_flags_t flags)
flags |= PSTAT_EXEC_FLAGS; /* For the traced bit. */
if (flags & (PSTAT_CTTYID | PSTAT_CWDIR | PSTAT_AUTH | PSTAT_UMASK
| PSTAT_EXEC_FLAGS)
- && !(have & PSTAT_NO_MSGPORT))
+ && !(flags & PSTAT_NO_MSGPORT))
{
flags |= PSTAT_MSGPORT;
flags |= PSTAT_TASK; /* for authentication */
@@ -114,6 +100,69 @@ proc_stat_set_flags(proc_stat_t ps, ps_flags_t flags)
if (flags & PSTAT_TASK_EVENTS_INFO)
flags |= PSTAT_TASK;
+ return flags;
+}
+
+/* Those flags that should be set before calling should_suppress_msgport. */
+#define SHOULD_SUPPRESS_MSGPORT_FLAGS (PSTAT_INFO | PSTAT_THREAD_INFO)
+
+/* Return true when there's some condition indicating that we shouldn't use
+ PS's msg port. For this routine to work correctly, PS's flags should
+ contain as many flags in SHOULD_SUPPRESS_MSGPORT_FLAGS as possible. */
+static int
+should_suppress_msgport (proc_stat_t ps)
+{
+ ps_flags_t have = ps->flags;
+
+ if ((have & PSTAT_INFO) && ps->info->taskinfo.suspend_count != 0)
+ /* Task is suspended. */
+ return TRUE;
+
+ if ((have & PSTAT_THREAD_INFO) && ps->thread_basic_info.suspend_count != 0)
+ /* All threads are suspended. */
+ return TRUE;
+
+ if ((have & PSTAT_INFO) && ps->info->nthreads == 0)
+ /* No threads (some bug could cause the procserver still to think there's
+ a msgport). */
+ return TRUE;
+
+ return FALSE;
+}
+
+/* Returns FLAGS with PSTAT_MSGPORT turned off and PSTAT_NO_MSGPORT on. */
+#define SUPPRESS_MSGPORT(flags) (((flags) & ~PSTAT_MSGPORT) | PSTAT_NO_MSGPORT)
+
+/* ---------------------------------------------------------------- */
+
+/* Add FLAGS to PS's flags, fetching information as necessary to validate
+ the corresponding fields in PS. Afterwards you must still check the flags
+ field before using new fields, as something might have failed. Returns
+ a system error code if a fatal error occurred, or 0 if none. */
+error_t
+proc_stat_set_flags (proc_stat_t ps, ps_flags_t flags)
+{
+ error_t err;
+ ps_flags_t have = ps->flags; /* flags set in ps */
+ ps_flags_t need; /* flags not set in ps, but desired to be */
+ ps_flags_t no_msgport_flags; /* a version of FLAGS for use when we can't
+ use the msg port. */
+ process_t server = ps_context_server(ps->context);
+
+ /* Propagate PSTAT_NO_MSGPORT. */
+ if (flags & PSTAT_NO_MSGPORT)
+ have = SUPPRESS_MSGPORT (have);
+ if (have & PSTAT_NO_MSGPORT)
+ flags = SUPPRESS_MSGPORT (flags);
+
+ no_msgport_flags = add_preconditions (SUPPRESS_MSGPORT (flags));
+ flags = add_preconditions (flags);
+
+ if (flags & PSTAT_MSGPORT)
+ /* Add in some values that we can use to determine whether the msgport
+ shouldn't be used. */
+ flags |= add_preconditions (SHOULD_SUPPRESS_MSGPORT_FLAGS);
+
need = flags & ~have;
/* MGET: If we're trying to set FLAG, and the preconditions PRECOND are set
@@ -128,13 +177,6 @@ proc_stat_set_flags(proc_stat_t ps, ps_flags_t flags)
: 0), \
err))
- /* The process's process port. */
- MGET(PSTAT_PROCESS, PSTAT_PID, proc_pid2proc(server, ps->pid, &ps->process));
- /* The process's task port. */
- MGET(PSTAT_TASK, PSTAT_PID, proc_pid2task(server, ps->pid, &ps->task));
- /* The process's libc msg port (see <hurd/msg.defs>). */
- MGET(PSTAT_MSGPORT, PSTAT_PID, proc_getmsgport(server, ps->pid, &ps->msgport));
-
/* the process's struct procinfo as returned by proc_getprocinfo. */
if ((need & PSTAT_INFO) && (have & PSTAT_PID))
{
@@ -187,6 +229,10 @@ proc_stat_set_flags(proc_stat_t ps, ps_flags_t flags)
tbi->cpu_usage += bi->cpu_usage;
tbi->sleep_time += bi->sleep_time;
+ /* The aggregate suspend count is the minimum of all threads. */
+ if (i == 0 || tbi->suspend_count > bi->suspend_count)
+ tbi->suspend_count = bi->suspend_count;
+
tbi->user_time.seconds += bi->user_time.seconds;
tbi->user_time.microseconds += bi->user_time.microseconds;
tbi->system_time.seconds += bi->system_time.seconds;
@@ -208,6 +254,22 @@ proc_stat_set_flags(proc_stat_t ps, ps_flags_t flags)
have |= PSTAT_THREAD_INFO;
}
+ ps->flags = have; /* should_suppress_msgport looks at them. */
+ if (should_suppress_msgport (ps))
+ /* Something is likely to have hosed the msg port, so don't use it. */
+ {
+ /* Turn off those things that were only good given the msg port. */
+ need &= ~(flags & ~no_msgport_flags);
+ have &= ~PSTAT_MSGPORT;
+ }
+
+ /* The process's process port. */
+ MGET(PSTAT_PROCESS, PSTAT_PID, proc_pid2proc(server, ps->pid, &ps->process));
+ /* The process's task port. */
+ MGET(PSTAT_TASK, PSTAT_PID, proc_pid2task(server, ps->pid, &ps->task));
+ /* The process's libc msg port (see <hurd/msg.defs>). */
+ MGET(PSTAT_MSGPORT, PSTAT_PID, proc_getmsgport(server, ps->pid, &ps->msgport));
+
/* VM statistics for the task. See <mach/task_info.h>. */
if ((need & PSTAT_TASK_EVENTS_INFO) && (have & PSTAT_TASK))
{
@@ -241,8 +303,8 @@ proc_stat_set_flags(proc_stat_t ps, ps_flags_t flags)
state |= PSTAT_STATE_ZOMBIE;
if (pi_flags & PI_SESSLD)
state |= PSTAT_STATE_SESSLDR;
- if (pi_flags & PI_EXECED)
- state |= PSTAT_STATE_EXECED;
+ if (!(pi_flags & PI_EXECED))
+ state |= PSTAT_STATE_FORKED;
if (pi_flags & PI_NOMSG)
state |= PSTAT_STATE_NOMSG;
if (pi_flags & PI_NOPARENT)