summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--trans/crash.c31
1 files changed, 31 insertions, 0 deletions
diff --git a/trans/crash.c b/trans/crash.c
index b2647429..812f587d 100644
--- a/trans/crash.c
+++ b/trans/crash.c
@@ -76,6 +76,35 @@ struct crasher
struct port_class *crasher_portclass;
+/* If the process referred to by proc port USERPROC is not orphaned,
+ then send SIGSTOP to all the other members of its pgrp. */
+void
+stop_pgrp (process_t userproc)
+{
+ pid_t pid, pgrp;
+ int orphaned;
+ error_t err;
+ size_t numpids = 20;
+ pid_t pids_[numpids], *pids = pids_;
+ int i;
+
+ err = proc_getpids (userproc, &pid, &pgrp, &orphaned);
+ if (err)
+ return;
+
+ /* Use userproc so that if it's just died we get an error and don't
+ do anything. */
+ err = proc_getpgrppids (userproc, pgrp, &pids, &numpids);
+ if (err)
+ return;
+
+ for (i = 0; i < numpids; i++)
+ if (pids[i] != pid)
+ kill (pids[i], SIGSTOP);
+ if (pids != pids_)
+ vm_deallocate (mach_task_self (), (vm_address_t) pids, numpids);
+}
+
kern_return_t
S_crash_dump_task (mach_port_t port,
@@ -108,6 +137,8 @@ S_crash_dump_task (mach_port_t port,
sizeof *c, &c);
if (! err)
{
+ stop_pgrp (user_proc);
+
/* Install our port as the crasher's msgport.
We will wait for signals to resume (crash) it. */
mach_port_t msgport = ports_get_right (c);