[tcmalloc_unittest] don't disable threaded test for fork torturing

We now detect "SIGTRAP is blocked" case and handle it properly. This
enables us to run pthread_create. So there is no more need to disable
ManyThreads test.

Note, we're not propagating single-stepping mode into child threads,
so we're still only fork-testing main thread, but that should be
plenty good for now. I.e. merely not crashing on pthread_create
anymore when fork torturing is enabled.
This commit is contained in:
Aliaksei Kandratsenka 2024-10-15 20:39:34 -04:00
parent 2e6061d07a
commit 6f7f81b699

View File

@ -82,11 +82,11 @@
#if __linux__ && __x86_64__ #if __linux__ && __x86_64__
// for fork testing // for fork testing
#include <dlfcn.h>
#include <errno.h> #include <errno.h>
#include <sched.h> #include <sched.h>
#include <semaphore.h> #include <semaphore.h>
#include <signal.h> #include <signal.h>
#include <sys/syscall.h>
#include <sys/wait.h> #include <sys/wait.h>
#include <ucontext.h> #include <ucontext.h>
#include <unistd.h> #include <unistd.h>
@ -601,8 +601,6 @@ class TesterThread {
}; };
TEST(TCMallocTest, ManyThreads) { TEST(TCMallocTest, ManyThreads) {
if (running_fork_testing) return;
printf("Testing threaded allocation/deallocation (%d threads)\n", printf("Testing threaded allocation/deallocation (%d threads)\n",
FLAGS_numthreads); FLAGS_numthreads);
@ -1894,6 +1892,8 @@ std::function<void()> SetupExec(int argc, char** argv) {
} }
#ifdef HAVE_FORK_TESTING_SUPPORT #ifdef HAVE_FORK_TESTING_SUPPORT
namespace fork_torture {
// Fork torture testing. // Fork torture testing.
// //
// Basic idea is to enable x86 single-stepping mode. And have signal // Basic idea is to enable x86 single-stepping mode. And have signal
@ -1937,12 +1937,16 @@ void xsem_wait(sem_t* sem) {
} }
} }
constexpr uintptr_t kTF = 0x100; // Trace flag in x86 FLAGS register.
bool try_handle_sigtrap_blocking(uint8_t* at_rip, ucontext_t* uc);
void step_handler(int signo, siginfo_t* si, void* _uc) { void step_handler(int signo, siginfo_t* si, void* _uc) {
constexpr uintptr_t TF_FLAG = 0x100;
ucontext_t* uc = static_cast<ucontext_t*>(_uc); ucontext_t* uc = static_cast<ucontext_t*>(_uc);
auto at_rip = reinterpret_cast<uint8_t*>(uc->uc_mcontext.gregs[REG_RIP]);
if (stepping_stop_requested) { if (stepping_stop_requested) {
uc->uc_mcontext.gregs[REG_EFL] &= ~TF_FLAG; uc->uc_mcontext.gregs[REG_EFL] &= ~kTF;
while (in_fork) { while (in_fork) {
(void)*const_cast<volatile bool*>(&in_fork); (void)*const_cast<volatile bool*>(&in_fork);
} }
@ -1950,6 +1954,10 @@ void step_handler(int signo, siginfo_t* si, void* _uc) {
return; return;
} }
if (try_handle_sigtrap_blocking(at_rip, uc)) {
return;
}
if (in_fork) { if (in_fork) {
return; return;
} }
@ -1957,12 +1965,11 @@ void step_handler(int signo, siginfo_t* si, void* _uc) {
// Add TF to flags and request SIGTRAP on every instruction in this // Add TF to flags and request SIGTRAP on every instruction in this
// thread. We could do it only once, but it is harmless to do it // thread. We could do it only once, but it is harmless to do it
// always. // always.
uc->uc_mcontext.gregs[REG_EFL] |= TF_FLAG; uc->uc_mcontext.gregs[REG_EFL] |= kTF;
static bool last_was_lock; static bool last_was_lock;
if (!last_was_lock) { if (!last_was_lock) {
auto at_rip = reinterpret_cast<uint8_t*>(uc->uc_mcontext.gregs[REG_RIP]);
if (*at_rip == 0xf0) { // lock prefix. if (*at_rip == 0xf0) { // lock prefix.
last_was_lock = true; last_was_lock = true;
} }
@ -1979,6 +1986,45 @@ void step_handler(int signo, siginfo_t* si, void* _uc) {
errno = errno_save; errno = errno_save;
} }
bool try_handle_sigtrap_blocking(uint8_t* at_rip, ucontext_t* uc) {
if (at_rip[0] != 0x0f || at_rip[1] != 0x05) {
return false;
}
// syscall instruction. Lets check if someone is about to block
// SIGTRAP. If so we must turn off single-stepping, because
// otherwise blocked SIGTRAP and pending single-stepping will kill
// the process.
auto& regs = uc->uc_mcontext.gregs;
if (regs[REG_RAX] != SYS_rt_sigprocmask) {
return false;
}
if (regs[REG_RDI] != SIG_SETMASK && regs[REG_RDI] != SIG_BLOCK) {
return false;
}
sigset_t* newmask = reinterpret_cast<sigset_t*>(regs[REG_RSI]);
if (!newmask || !sigismember(newmask, SIGTRAP)) {
return false;
}
// okay, once we detected this case, we drop single-stepping
// flag, block SIGTRAP and raise it. So that when SIGTRAP is
// eventually unblocked, we'll get back to signal hander and
// re-set single-stepping back.
regs[REG_EFL] &= ~kTF;
raise(SIGTRAP);
sigset_t* oldmask = reinterpret_cast<sigset_t*>(regs[REG_RDX]);
if (oldmask) {
*oldmask = uc->uc_sigmask;
regs[REG_RDX] = 0; // handle "get old mask" part, so we can block
// our signal
}
sigaddset(&uc->uc_sigmask, SIGTRAP);
return true;
}
tcmalloc::Cleanup<std::function<void()>> setup_fork_testing(int* argc, char *** argv) { tcmalloc::Cleanup<std::function<void()>> setup_fork_testing(int* argc, char *** argv) {
if (*argc < 2 || (*argv)[1] != std::string("--with-fork-torture")) { if (*argc < 2 || (*argv)[1] != std::string("--with-fork-torture")) {
printf("Not enabling fork torture\n"); printf("Not enabling fork torture\n");
@ -2088,6 +2134,9 @@ tcmalloc::Cleanup<std::function<void()>> setup_fork_testing(int* argc, char ***
printf("Done with fork torturing!\n"); printf("Done with fork torturing!\n");
})); }));
} }
} // namespace fork_torture
using fork_torture::setup_fork_testing;
#else // HAVE_FORK_TESTING_SUPPORT #else // HAVE_FORK_TESTING_SUPPORT