Print stacktraces using std::stacktrace if available

If std::stacktrace is available (currently only GCC (opt-in feature) and
MSVC), use it for printing stack traces.

If both Abseil and std::stacktrace are available, prefer
std::stacktrace.

Advantages of std::stacktrace over Abseil:

* Includes file and line information
* Includes DLL information
* Better stack trace quality on Windows

Disadvantages of std::stacktrace compared to Abseil:

* Runs more slowly on Windows
* Includes information which might not be relevant such as byte offsets
* Printed lines are longer thus harder to skim

Before (Windows using Abseil):

    [ RUN      ] FooTest.Xyz
    [snip]\googletest-filter-unittest_.cc(49): error: Failed
    Expected failure.
    Stack trace:
      00007FF69EF90E8D: testing::internal::HandleSehExceptionsInMethodIfSupported<testing::Test,void>
      00007FF69EF90AC3: testing::internal::HandleExceptionsInMethodIfSupported<testing::Test,void>
      00007FF69EF5C01B: testing::Test::Run
      00007FF69EF5CCC5: testing::TestInfo::Run
      00007FF69EF5D6EC: testing::TestSuite::Run
    ... Google Test internal frames ...

    [  FAILED  ] FooTest.Xyz (21 ms)

After (Windows using MSVC STL's std::stacktrace):

    [ RUN      ] FooTest.Xyz
    [snip]\googletest-filter-unittest_.cc(49): error: Failed
    Expected failure.
    Stack trace:
      00007FF6770283A9: googletest_filter_unittest_!`anonymous namespace'::FooTest_Xyz_Test::TestBody+0x89 [snip]\googletest-filter-unittest_.cc:49
      00007FF677080E8D: googletest_filter_unittest_!testing::internal::HandleSehExceptionsInMethodIfSupported<testing::Test,void>+0x1D [snip]\gtest.cc:2609
      00007FF677080AC3: googletest_filter_unittest_!testing::internal::HandleExceptionsInMethodIfSupported<testing::Test,void>+0x73 [snip]\gtest.cc:2652
      00007FF67704C01B: googletest_filter_unittest_!testing::Test::Run+0xAB [snip]\gtest.cc:2698
    ... Google Test internal frames ...

    [  FAILED  ] FooTest.Xyz (127 ms)
This commit is contained in:
Matthew Glazar 2023-08-09 20:39:34 -07:00
parent 8aaea6c092
commit dd45329f01
3 changed files with 100 additions and 0 deletions

View file

@ -216,6 +216,7 @@
// specializations. Always defined to 0 or 1.
// GTEST_USE_OWN_FLAGFILE_FLAG_ - Always defined to 0 or 1.
// GTEST_HAS_CXXABI_H_ - Always defined to 0 or 1.
// GTEST_HAS_STD_STACKTRACE_ - Always defined to 0 or 1.
// GTEST_CAN_STREAM_RESULTS_ - Always defined to 0 or 1.
// GTEST_HAS_ALT_PATH_SEP_ - Always defined to 0 or 1.
// GTEST_WIDE_STRING_USES_UTF16_ - Always defined to 0 or 1.
@ -873,6 +874,14 @@ typedef struct _RTL_CRITICAL_SECTION GTEST_CRITICAL_SECTION;
#endif
#endif
#if !defined(GTEST_HAS_STD_STACKTRACE_)
#if defined(__cpp_lib_stacktrace) && __cpp_lib_stacktrace >= 202011L
#define GTEST_HAS_STD_STACKTRACE_ 1
#else
#define GTEST_HAS_STD_STACKTRACE_ 0
#endif
#endif
// A function level attribute to disable checking for use of uninitialized
// memory when built with MemorySanitizer.
#if GTEST_HAVE_ATTRIBUTE_(no_sanitize_memory)

View file

@ -59,6 +59,10 @@
#include <windows.h> // NOLINT
#endif // GTEST_OS_WINDOWS
#if GTEST_HAS_STD_STACKTRACE_
#include <stacktrace>
#endif
#include "gtest/gtest-spi.h"
#include "gtest/gtest.h"
@ -456,6 +460,29 @@ class AbslStackTraceGetter : public OsStackTraceGetterInterface {
};
#endif // GTEST_HAS_ABSL
#if GTEST_HAS_STD_STACKTRACE_
// A working implementation of the OsStackTraceGetterInterface interface.
class StdStackTraceGetter : public OsStackTraceGetterInterface {
public:
StdStackTraceGetter() = default;
std::string CurrentStackTrace(int max_depth, int skip_count) override;
void UponLeavingGTest() override;
private:
Mutex mutex_; // Protects all internal state.
// We save the stack frame below the frame that calls user code.
// We do this because the address of the frame immediately below
// the user code changes between the call to UponLeavingGTest()
// and any calls to the stack trace code from within the user code.
std::stacktrace_entry::native_handle_type caller_frame_ = {};
StdStackTraceGetter(const StdStackTraceGetter&) = delete;
StdStackTraceGetter& operator=(const StdStackTraceGetter&) = delete;
};
#endif // GTEST_HAS_STD_STACKTRACE_
// An implementation of OsStackTraceGetterInterface which returns no
// stack trace.
class NoopStackTraceGetter : public OsStackTraceGetterInterface {

View file

@ -153,6 +153,10 @@
#include "absl/strings/strip.h"
#endif // GTEST_HAS_ABSL
#if GTEST_HAS_STD_STACKTRACE_
#include <stacktrace>
#endif
// Checks builtin compiler feature |x| while avoiding an extra layer of #ifdefs
// at the callsite.
#if defined(__has_builtin)
@ -5025,6 +5029,64 @@ void AbslStackTraceGetter::UponLeavingGTest() GTEST_LOCK_EXCLUDED_(mutex_) {
}
#endif // GTEST_HAS_ABSL
#if GTEST_HAS_STD_STACKTRACE_
std::string StdStackTraceGetter::CurrentStackTrace(int max_depth, int skip_count)
GTEST_LOCK_EXCLUDED_(mutex_) {
if (max_depth <= 0) {
return "";
}
max_depth = std::min(max_depth, kMaxStackTraceDepth);
// Skips the frames requested by the caller, plus this function.
skip_count += 1;
std::stacktrace stack = std::stacktrace::current(skip_count, max_depth);
std::stacktrace_entry::native_handle_type caller_frame = {};
{
MutexLock lock(&mutex_);
caller_frame = caller_frame_;
}
std::ostringstream result;
for (const std::stacktrace_entry &entry : stack) {
if (entry.native_handle() == caller_frame &&
!GTEST_FLAG_GET(show_internal_stack_frames)) {
// Add a marker to the trace and stop adding frames.
result << kElidedFramesMarker << '\n';
break;
}
result << " " << entry.native_handle() << ": ";
std::string description = entry.description();
if (description.empty()) {
result << "(unknown)";
} else {
result << description;
}
std::string source_file = entry.source_file();
if (!source_file.empty()) {
result << ' ' << source_file << ':' << entry.source_line();
}
result << '\n';
}
return std::move(result).str();
}
#endif // GTEST_HAS_ABSL
#if GTEST_HAS_STD_STACKTRACE_
void StdStackTraceGetter::UponLeavingGTest() GTEST_LOCK_EXCLUDED_(mutex_) {
std::stacktrace stack = std::stacktrace::current(/*skip=*/2, /*max_depth=*/1);
std::stacktrace_entry::native_handle_type caller_frame = {};
if (!stack.empty()) {
caller_frame = stack[0].native_handle();
}
MutexLock lock(&mutex_);
caller_frame_ = caller_frame;
}
#endif // GTEST_HAS_ABSL
std::string NoopStackTraceGetter::CurrentStackTrace(int max_depth, int skip_count) {
static_cast<void>(max_depth);
static_cast<void>(skip_count);
@ -6244,6 +6306,8 @@ OsStackTraceGetterInterface* UnitTestImpl::os_stack_trace_getter() {
if (os_stack_trace_getter_ == nullptr) {
#ifdef GTEST_OS_STACK_TRACE_GETTER_
os_stack_trace_getter_ = new GTEST_OS_STACK_TRACE_GETTER_;
#elif GTEST_HAS_STD_STACKTRACE_
os_stack_trace_getter_ = new StdStackTraceGetter;
#elif defined(GTEST_HAS_ABSL)
os_stack_trace_getter_ = new AbslStackTraceGetter;
#else