Source-Changes-HG archive

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index][Old Index]

[src/trunk]: src/tests/lib/libc/sys Add new ATF ptrace(2) tests verifying cra...



details:   https://anonhg.NetBSD.org/src/rev/0d241dd20a4d
branches:  trunk
changeset: 362168:0d241dd20a4d
user:      kamil <kamil%NetBSD.org@localhost>
date:      Tue May 29 10:40:54 2018 +0000

description:
Add new ATF ptrace(2) tests verifying crash signal handling

Add new tests:
 - traceme_crash_trap
 - traceme_crash_segv
 - traceme_crash_ill (temporarily disabled)
 - traceme_crash_fpe
 - traceme_crash_bus
 - unrelated_tracer_sees_crash_trap
 - unrelated_tracer_sees_crash_segv
 - unrelated_tracer_sees_crash_ill (temporarily disabled)
 - unrelated_tracer_sees_crash_fpe
 - unrelated_tracer_sees_crash_bus

These tests verify two subtle kernel paths that differ internally:
 - for a tracer that is a parent of a traced program (and not vforked)
   real parent == tracer
 - for a tracer that is an unrelated process of a traced program
   real parent != tracer

Regressions in the signal code can cause to break one of the scenarios, and
keep working the other one. It might be propagated to a scenario like:
 - attaching to a process with a debugger (over pid)
 - starting a process inside a debugger

We can assert on the ATF level that both will be functional.

While there, cover all the crash signals as testing is done practically
for free. While the Machine Independent signal routing code for crash
signals is mostly the same in the kernel, we can verify whether Machine
Dependent parts works as expected emitting the proper signal number and
signal code (si_code).

There have been already caught an unexpected behavior on Alpha, that
instead of SIGBUS we receive SIGSEGV.

Another benefit is that it's easier to catch races, as they pop up more
frequently. (This already helped to catch bugs more quickly)

These tests will help assert correctness of future kernel changes in the
signal routines.

All tests pass.

Sponsored by <The NetBSD Foundation>

diffstat:

 tests/lib/libc/sys/t_ptrace_wait.c |  307 ++++++++++++++++++++++++++++++++++++-
 1 files changed, 303 insertions(+), 4 deletions(-)

diffs (truncated from 358 to 300 lines):

diff -r fce3e77f0bff -r 0d241dd20a4d tests/lib/libc/sys/t_ptrace_wait.c
--- a/tests/lib/libc/sys/t_ptrace_wait.c        Tue May 29 09:25:44 2018 +0000
+++ b/tests/lib/libc/sys/t_ptrace_wait.c        Tue May 29 10:40:54 2018 +0000
@@ -1,4 +1,4 @@
-/*     $NetBSD: t_ptrace_wait.c,v 1.58 2018/05/28 11:35:50 kamil Exp $ */
+/*     $NetBSD: t_ptrace_wait.c,v 1.59 2018/05/29 10:40:54 kamil Exp $ */
 
 /*-
  * Copyright (c) 2016 The NetBSD Foundation, Inc.
@@ -27,7 +27,7 @@
  */
 
 #include <sys/cdefs.h>
-__RCSID("$NetBSD: t_ptrace_wait.c,v 1.58 2018/05/28 11:35:50 kamil Exp $");
+__RCSID("$NetBSD: t_ptrace_wait.c,v 1.59 2018/05/29 10:40:54 kamil Exp $");
 
 #include <sys/param.h>
 #include <sys/types.h>
@@ -177,6 +177,114 @@
 /// ----------------------------------------------------------------------------
 
 static void
+traceme_crash(int sig)
+{
+       pid_t child, wpid;
+#if defined(TWAIT_HAVE_STATUS)
+       int status;
+#endif
+       struct ptrace_siginfo info;
+       memset(&info, 0, sizeof(info));
+
+       DPRINTF("Before forking process PID=%d\n", getpid());
+       SYSCALL_REQUIRE((child = fork()) != -1);
+       if (child == 0) {
+               DPRINTF("Before calling PT_TRACE_ME from child %d\n", getpid());
+               FORKEE_ASSERT(ptrace(PT_TRACE_ME, 0, NULL, 0) != -1);
+
+               DPRINTF("Before executing a trap\n");
+               switch (sig) {
+               case SIGTRAP:
+                       trigger_trap();
+                       break;
+               case SIGSEGV:
+                       trigger_segv();
+                       break;
+               case SIGILL:
+                       trigger_ill();
+                       break;
+               case SIGFPE:
+                       trigger_fpe();
+                       break;
+               case SIGBUS:
+                       trigger_bus();
+                       break;
+               default:
+                       /* NOTREACHED */
+                       FORKEE_ASSERTX(0 && "This shall not be reached");
+               }
+
+               /* NOTREACHED */
+               FORKEE_ASSERTX(0 && "This shall not be reached");
+       }
+       DPRINTF("Parent process PID=%d, child's PID=%d\n", getpid(), child);
+
+       DPRINTF("Before calling %s() for the child\n", TWAIT_FNAME);
+       TWAIT_REQUIRE_SUCCESS(wpid = TWAIT_GENERIC(child, &status, 0), child);
+
+       validate_status_stopped(status, sig);
+
+       DPRINTF("Before calling ptrace(2) with PT_GET_SIGINFO for child");
+       SYSCALL_REQUIRE(ptrace(PT_GET_SIGINFO, child, &info, sizeof(info)) != -1);
+
+       DPRINTF("Signal traced to lwpid=%d\n", info.psi_lwpid);
+       DPRINTF("Signal properties: si_signo=%#x si_code=%#x si_errno=%#x\n",
+               info.psi_siginfo.si_signo, info.psi_siginfo.si_code,
+               info.psi_siginfo.si_errno);
+
+       ATF_REQUIRE_EQ(info.psi_siginfo.si_signo, sig);
+       switch (sig) {
+       case SIGTRAP:
+               ATF_REQUIRE_EQ(info.psi_siginfo.si_code, TRAP_BRKPT);
+               break;
+       case SIGSEGV:
+               ATF_REQUIRE_EQ(info.psi_siginfo.si_code, SEGV_MAPERR);
+               break;
+//     case SIGILL:
+//             ATF_REQUIRE_EQ(info.psi_siginfo.si_code, ILL_ILLOP);
+//             break;
+       case SIGFPE:
+               ATF_REQUIRE_EQ(info.psi_siginfo.si_code, FPE_INTDIV);
+               break;
+       case SIGBUS:
+               ATF_REQUIRE_EQ(info.psi_siginfo.si_code, BUS_ADRERR);
+               break;
+       }
+
+       SYSCALL_REQUIRE(ptrace(PT_KILL, child, NULL, 0) != -1);
+
+       DPRINTF("Before calling %s() for the child\n", TWAIT_FNAME);
+       TWAIT_REQUIRE_SUCCESS(wpid = TWAIT_GENERIC(child, &status, 0), child);
+
+       validate_status_signaled(status, SIGKILL, 0);
+
+       DPRINTF("Before calling %s() for the child\n", TWAIT_FNAME);
+       TWAIT_REQUIRE_FAILURE(ECHILD, wpid = TWAIT_GENERIC(child, &status, 0));
+}
+
+#define TRACEME_CRASH(test, sig)                                               \
+ATF_TC(test);                                                                  \
+ATF_TC_HEAD(test, tc)                                                          \
+{                                                                              \
+       atf_tc_set_md_var(tc, "descr",                                          \
+           "Verify crash signal " #sig " in a child after PT_TRACE_ME");       \
+}                                                                              \
+                                                                               \
+ATF_TC_BODY(test, tc)                                                          \
+{                                                                              \
+                                                                               \
+       traceme_crash(sig);                                                     \
+}
+
+TRACEME_CRASH(traceme_crash_trap, SIGTRAP)
+TRACEME_CRASH(traceme_crash_segv, SIGSEGV)
+//TRACEME_CRASH(traceme_crash_ill, SIGILL)
+TRACEME_CRASH(traceme_crash_fpe, SIGFPE)
+TRACEME_CRASH(traceme_crash_bus, SIGBUS)
+
+/// ----------------------------------------------------------------------------
+
+static void
 traceme_sendsignal_handle(int sigsent, void (*sah)(int a), int *traceme_caught)
 {
        const int exitval = 5;
@@ -882,6 +990,185 @@
 
 #if defined(TWAIT_HAVE_PID)
 static void
+unrelated_tracer_sees_crash(int sig)
+{
+       struct msg_fds parent_tracee, parent_tracer;
+       const int exitval = 10;
+       pid_t tracee, tracer, wpid;
+       uint8_t msg = 0xde; /* dummy message for IPC based on pipe(2) */
+#if defined(TWAIT_HAVE_STATUS)
+       int status;
+#endif
+       struct ptrace_siginfo info;
+       memset(&info, 0, sizeof(info));
+
+       DPRINTF("Spawn tracee\n");
+       SYSCALL_REQUIRE(msg_open(&parent_tracee) == 0);
+       tracee = atf_utils_fork();
+       if (tracee == 0) {
+               // Wait for parent to let us crash
+               CHILD_FROM_PARENT("exit tracee", parent_tracee, msg);
+               
+               DPRINTF("Before executing a trap\n");
+               switch (sig) {
+               case SIGTRAP:
+                       trigger_trap();
+                       break;
+               case SIGSEGV:
+                       trigger_segv();
+                       break;
+               case SIGILL:
+                       trigger_ill();
+                       break;
+               case SIGFPE:
+                       trigger_fpe();
+                       break;
+               case SIGBUS:
+                       trigger_bus();
+                       break;
+               default:
+                       /* NOTREACHED */
+                       FORKEE_ASSERTX(0 && "This shall not be reached");
+               }
+
+               /* NOTREACHED */
+               FORKEE_ASSERTX(0 && "This shall not be reached");
+       }
+
+       DPRINTF("Spawn debugger\n");
+       SYSCALL_REQUIRE(msg_open(&parent_tracer) == 0);
+       tracer = atf_utils_fork();
+       if (tracer == 0) {
+               /* Fork again and drop parent to reattach to PID 1 */
+               tracer = atf_utils_fork();
+               if (tracer != 0)
+                               _exit(exitval);
+
+               DPRINTF("Before calling PT_ATTACH from tracee %d\n", getpid());
+               FORKEE_ASSERT(ptrace(PT_ATTACH, tracee, NULL, 0) != -1);
+
+               /* Wait for tracee and assert that it was stopped w/ SIGSTOP */
+               FORKEE_REQUIRE_SUCCESS(
+                   wpid = TWAIT_GENERIC(tracee, &status, 0), tracee);
+
+               forkee_status_stopped(status, SIGSTOP);
+
+               /* Resume tracee with PT_CONTINUE */
+               FORKEE_ASSERT(ptrace(PT_CONTINUE, tracee, (void *)1, 0) != -1);
+
+               /* Inform parent that tracer has attached to tracee */
+               CHILD_TO_PARENT("tracer ready", parent_tracer, msg);
+
+               /* Wait for parent to tell use that tracee should have exited */
+               CHILD_FROM_PARENT("wait for tracee exit", parent_tracer, msg);
+
+               /* Wait for tracee and assert that it exited */
+               FORKEE_REQUIRE_SUCCESS(
+                   wpid = TWAIT_GENERIC(tracee, &status, 0), tracee);
+
+               validate_status_stopped(status, sig);
+
+               DPRINTF("Before calling ptrace(2) with PT_GET_SIGINFO for the "
+                       "traced process\n");
+               SYSCALL_REQUIRE(ptrace(PT_GET_SIGINFO, tracee, &info,
+                                      sizeof(info)) != -1);
+
+               DPRINTF("Signal traced to lwpid=%d\n", info.psi_lwpid);
+               DPRINTF("Signal properties: si_signo=%#x si_code=%#x "
+                       "si_errno=%#x\n", info.psi_siginfo.si_signo,
+                       info.psi_siginfo.si_code, info.psi_siginfo.si_errno);
+
+               ATF_REQUIRE_EQ(info.psi_siginfo.si_signo, sig);
+               switch (sig) {
+               case SIGTRAP:
+                       ATF_REQUIRE_EQ(info.psi_siginfo.si_code, TRAP_BRKPT);
+                       break;
+               case SIGSEGV:
+                       ATF_REQUIRE_EQ(info.psi_siginfo.si_code, SEGV_MAPERR);
+                       break;
+//             case SIGILL:
+//                     ATF_REQUIRE_EQ(info.psi_siginfo.si_code, ILL_ILLOP);
+//                     break;
+               case SIGFPE:
+                       ATF_REQUIRE_EQ(info.psi_siginfo.si_code, FPE_INTDIV);
+                       break;
+               case SIGBUS:
+                       ATF_REQUIRE_EQ(info.psi_siginfo.si_code, BUS_ADRERR);
+                       break;
+               }
+
+               FORKEE_ASSERT(ptrace(PT_KILL, tracee, NULL, 0) != -1);
+               DPRINTF("Before calling %s() for the tracee\n", TWAIT_FNAME);
+               TWAIT_REQUIRE_SUCCESS(wpid = TWAIT_GENERIC(tracee, &status, 0),
+                                     tracee);
+
+               validate_status_signaled(status, SIGKILL, 0);
+
+               DPRINTF("Before calling %s() for tracee\n", TWAIT_FNAME);
+               TWAIT_REQUIRE_FAILURE(ECHILD,
+                   wpid = TWAIT_GENERIC(tracee, &status, 0));
+
+               DPRINTF("Before exiting of the tracer process\n");
+               _exit(0 /* collect by initproc */);
+       }
+
+       DPRINTF("Wait for the tracer process (direct child) to exit "
+           "calling %s()\n", TWAIT_FNAME);
+       TWAIT_REQUIRE_SUCCESS(
+           wpid = TWAIT_GENERIC(tracer, &status, 0), tracer);
+
+       validate_status_exited(status, exitval);
+
+       DPRINTF("Wait for the non-exited tracee process with %s()\n",
+           TWAIT_FNAME);
+       TWAIT_REQUIRE_SUCCESS(
+           wpid = TWAIT_GENERIC(tracee, NULL, WNOHANG), 0);
+
+       DPRINTF("Wait for the tracer to attach to the tracee\n");
+       PARENT_FROM_CHILD("tracer ready", parent_tracer, msg);
+
+       DPRINTF("Resume the tracee and let it crash\n");
+       PARENT_TO_CHILD("exit tracee", parent_tracee,  msg);
+
+       DPRINTF("Resume the tracer and let it detect crashed tracee\n");
+       PARENT_TO_CHILD("Message 2", parent_tracer, msg);
+
+       DPRINTF("Wait for tracee to finish its job and exit - calling %s()\n",
+           TWAIT_FNAME);
+       TWAIT_REQUIRE_SUCCESS(wpid = TWAIT_GENERIC(tracee, &status, 0), tracee);
+
+       validate_status_signaled(status, SIGKILL, 0);
+
+       msg_close(&parent_tracer);
+       msg_close(&parent_tracee);
+}
+
+#define UNRELATED_TRACER_SEES_CRASH(test, sig)                                 \
+ATF_TC(test);                                                                  \
+ATF_TC_HEAD(test, tc)                                                          \
+{                                                                              \
+       atf_tc_set_md_var(tc, "descr",                                          \
+           "Assert that an unrelated tracer sees crash signal from the "       \
+           "debuggee");                                                        \
+}                                                                              \
+                                                                               \
+ATF_TC_BODY(test, tc)                                                          \



Home | Main Index | Thread Index | Old Index