2018-05-16 22:31:43 -07:00
|
|
|
#include <gmock/gmock.h>
|
2018-05-18 22:29:41 -07:00
|
|
|
#include <csignal>
|
2018-05-16 22:31:43 -07:00
|
|
|
#include "cpp-utils/assert/backtrace.h"
|
2018-08-01 16:29:14 -07:00
|
|
|
#include "cpp-utils/process/subprocess.h"
|
2019-03-23 22:01:26 -07:00
|
|
|
#include <boost/filesystem.hpp>
|
|
|
|
#include "my-gtest-main.h"
|
2018-05-16 22:31:43 -07:00
|
|
|
|
|
|
|
using std::string;
|
|
|
|
using testing::HasSubstr;
|
2019-03-23 22:01:26 -07:00
|
|
|
namespace bf = boost::filesystem;
|
2018-05-16 22:31:43 -07:00
|
|
|
|
2018-08-01 16:29:14 -07:00
|
|
|
namespace {
|
|
|
|
std::string call_process_exiting_with(const std::string& kind, const std::string& signal = "") {
|
|
|
|
#if defined(_MSC_VER)
|
2019-03-23 22:01:26 -07:00
|
|
|
auto executable = get_executable().parent_path() / "cpp-utils-test_exit_signal.exe";
|
2018-08-01 16:29:14 -07:00
|
|
|
#else
|
2019-03-23 22:01:26 -07:00
|
|
|
auto executable = get_executable().parent_path() / "cpp-utils-test_exit_signal";
|
2018-08-01 16:29:14 -07:00
|
|
|
#endif
|
2019-03-23 22:01:26 -07:00
|
|
|
if (!bf::exists(executable)) {
|
|
|
|
throw std::runtime_error(executable.string() + " not found.");
|
|
|
|
}
|
|
|
|
const std::string command = executable.string() + " \"" + kind + "\" \"" + signal + "\" 2>&1";
|
2018-08-01 16:29:14 -07:00
|
|
|
auto result = cpputils::Subprocess::call(command);
|
|
|
|
return result.output;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-01-23 16:45:30 -08:00
|
|
|
#if !(defined(_MSC_VER) && defined(NDEBUG))
|
2019-04-01 19:18:49 -07:00
|
|
|
|
2018-05-16 22:31:43 -07:00
|
|
|
TEST(BacktraceTest, ContainsTopLevelLine) {
|
|
|
|
string backtrace = cpputils::backtrace();
|
|
|
|
EXPECT_THAT(backtrace, HasSubstr("BacktraceTest"));
|
|
|
|
EXPECT_THAT(backtrace, HasSubstr("ContainsTopLevelLine"));
|
|
|
|
}
|
2019-01-23 16:45:30 -08:00
|
|
|
#endif
|
2018-08-01 16:29:14 -07:00
|
|
|
|
2018-05-16 22:42:03 -07:00
|
|
|
namespace {
|
2018-08-01 16:29:14 -07:00
|
|
|
std::string call_process_exiting_with_nullptr_violation() {
|
|
|
|
return call_process_exiting_with("nullptr");
|
|
|
|
}
|
|
|
|
std::string call_process_exiting_with_exception(const std::string& message) {
|
|
|
|
return call_process_exiting_with("exception", message);
|
2018-07-08 19:34:08 -07:00
|
|
|
}
|
2018-05-18 22:29:41 -07:00
|
|
|
}
|
2018-07-08 19:34:08 -07:00
|
|
|
#if defined(_MSC_VER)
|
|
|
|
#include <Windows.h>
|
|
|
|
namespace {
|
2018-08-01 16:29:14 -07:00
|
|
|
std::string call_process_exiting_with_sigsegv() {
|
|
|
|
return call_process_exiting_with("signal", std::to_string(EXCEPTION_ACCESS_VIOLATION));
|
2018-07-08 19:34:08 -07:00
|
|
|
}
|
2018-08-01 16:29:14 -07:00
|
|
|
std::string call_process_exiting_with_sigill() {
|
|
|
|
return call_process_exiting_with("signal", std::to_string(EXCEPTION_ILLEGAL_INSTRUCTION));
|
2018-07-08 19:34:08 -07:00
|
|
|
}
|
2018-08-01 16:29:14 -07:00
|
|
|
std::string call_process_exiting_with_code(DWORD code) {
|
|
|
|
return call_process_exiting_with("signal", std::to_string(code));
|
2018-07-08 19:34:08 -07:00
|
|
|
}
|
2018-05-16 22:42:03 -07:00
|
|
|
}
|
2018-07-08 19:34:08 -07:00
|
|
|
#else
|
|
|
|
namespace {
|
2018-08-01 16:29:14 -07:00
|
|
|
std::string call_process_exiting_with_sigsegv() {
|
|
|
|
return call_process_exiting_with("signal", std::to_string(SIGSEGV));
|
2018-07-08 19:34:08 -07:00
|
|
|
}
|
2018-08-01 16:29:14 -07:00
|
|
|
std::string call_process_exiting_with_sigabrt() {
|
|
|
|
return call_process_exiting_with("signal", std::to_string(SIGABRT));
|
2018-07-08 19:34:08 -07:00
|
|
|
}
|
2018-08-01 16:29:14 -07:00
|
|
|
std::string call_process_exiting_with_sigill() {
|
|
|
|
return call_process_exiting_with("signal", std::to_string(SIGILL));
|
2018-07-08 19:34:08 -07:00
|
|
|
}
|
2018-05-16 22:42:03 -07:00
|
|
|
}
|
2018-07-08 19:34:08 -07:00
|
|
|
#endif
|
2018-05-16 22:42:03 -07:00
|
|
|
|
2018-08-01 16:29:14 -07:00
|
|
|
TEST(BacktraceTest, DoesntCrashOnCaughtException) {
|
|
|
|
// This is needed to make sure we don't use some kind of vectored exception handler on Windows
|
|
|
|
// that ignores the call stack and always jumps on when an exception happens.
|
|
|
|
cpputils::showBacktraceOnCrash();
|
|
|
|
try {
|
|
|
|
throw std::logic_error("exception");
|
|
|
|
} catch (const std::logic_error& e) {
|
|
|
|
// intentionally empty
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-01-23 16:45:30 -08:00
|
|
|
#if !(defined(_MSC_VER) && defined(NDEBUG))
|
2019-04-01 19:18:49 -07:00
|
|
|
TEST(BacktraceTest, ContainsBacktrace) {
|
|
|
|
string backtrace = cpputils::backtrace();
|
|
|
|
#if defined(_MSC_VER)
|
|
|
|
EXPECT_THAT(backtrace, HasSubstr("testing::Test::Run"));
|
|
|
|
#else
|
|
|
|
EXPECT_THAT(backtrace, HasSubstr("BacktraceTest_ContainsBacktrace_Test::TestBody"));
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2018-05-18 22:29:41 -07:00
|
|
|
TEST(BacktraceTest, ShowBacktraceOnNullptrAccess) {
|
2018-08-01 16:29:14 -07:00
|
|
|
auto output = call_process_exiting_with_nullptr_violation();
|
2019-04-01 19:18:49 -07:00
|
|
|
#if defined(_MSC_VER)
|
2019-02-27 23:34:49 -08:00
|
|
|
EXPECT_THAT(output, HasSubstr("handle_exit_signal"));
|
2019-04-01 19:18:49 -07:00
|
|
|
#else
|
|
|
|
EXPECT_THAT(output, HasSubstr("cpputils::backtrace"));
|
2019-02-27 23:34:49 -08:00
|
|
|
#endif
|
2018-05-16 22:42:03 -07:00
|
|
|
}
|
2018-05-18 22:29:41 -07:00
|
|
|
|
|
|
|
TEST(BacktraceTest, ShowBacktraceOnSigSegv) {
|
2018-08-01 16:29:14 -07:00
|
|
|
auto output = call_process_exiting_with_sigsegv();
|
2019-02-27 23:34:49 -08:00
|
|
|
#if defined(_MSC_VER)
|
|
|
|
EXPECT_THAT(output, HasSubstr("handle_exit_signal"));
|
|
|
|
#else
|
2019-04-01 19:18:49 -07:00
|
|
|
EXPECT_THAT(output, HasSubstr("cpputils::backtrace"));
|
2019-02-27 23:34:49 -08:00
|
|
|
#endif
|
2018-08-01 16:29:14 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
TEST(BacktraceTest, ShowBacktraceOnUnhandledException) {
|
|
|
|
auto output = call_process_exiting_with_exception("my_exception_message");
|
2019-02-27 23:34:49 -08:00
|
|
|
#if defined(_MSC_VER)
|
|
|
|
EXPECT_THAT(output, HasSubstr("handle_exit_signal"));
|
|
|
|
#else
|
2019-04-01 19:18:49 -07:00
|
|
|
EXPECT_THAT(output, HasSubstr("cpputils::backtrace"));
|
2019-02-27 23:34:49 -08:00
|
|
|
#endif
|
2018-05-18 22:29:41 -07:00
|
|
|
}
|
|
|
|
|
2018-07-08 19:34:08 -07:00
|
|
|
TEST(BacktraceTest, ShowBacktraceOnSigIll) {
|
2018-08-01 16:29:14 -07:00
|
|
|
auto output = call_process_exiting_with_sigill();
|
2019-02-27 23:34:49 -08:00
|
|
|
#if defined(_MSC_VER)
|
2019-04-01 19:18:49 -07:00
|
|
|
EXPECT_THAT(output, HasSubstr("handle_exit_signal"));
|
2019-02-27 23:34:49 -08:00
|
|
|
#else
|
2019-04-01 19:18:49 -07:00
|
|
|
EXPECT_THAT(output, HasSubstr("cpputils::backtrace"));
|
2019-02-27 23:34:49 -08:00
|
|
|
#endif
|
2018-07-08 19:34:08 -07:00
|
|
|
}
|
2019-01-23 16:45:30 -08:00
|
|
|
#else
|
2019-04-01 19:18:49 -07:00
|
|
|
TEST(BacktraceTest, ContainsBacktrace) {
|
|
|
|
string backtrace = cpputils::backtrace();
|
|
|
|
EXPECT_THAT(backtrace, HasSubstr("#1"));
|
|
|
|
}
|
2019-01-23 16:45:30 -08:00
|
|
|
TEST(BacktraceTest, ShowBacktraceOnNullptrAccess) {
|
|
|
|
auto output = call_process_exiting_with_nullptr_violation();
|
|
|
|
EXPECT_THAT(output, HasSubstr("#1"));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(BacktraceTest, ShowBacktraceOnSigSegv) {
|
|
|
|
auto output = call_process_exiting_with_sigsegv();
|
|
|
|
EXPECT_THAT(output, HasSubstr("#1"));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(BacktraceTest, ShowBacktraceOnUnhandledException) {
|
|
|
|
auto output = call_process_exiting_with_exception("my_exception_message");
|
|
|
|
EXPECT_THAT(output, HasSubstr("#1"));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(BacktraceTest, ShowBacktraceOnSigIll) {
|
|
|
|
auto output = call_process_exiting_with_sigill();
|
|
|
|
EXPECT_THAT(output, HasSubstr("#1"));
|
|
|
|
}
|
|
|
|
#endif
|
2018-07-08 19:34:08 -07:00
|
|
|
|
|
|
|
#if !defined(_MSC_VER)
|
2018-05-18 22:29:41 -07:00
|
|
|
TEST(BacktraceTest, ShowBacktraceOnSigAbrt) {
|
2018-08-01 16:29:14 -07:00
|
|
|
auto output = call_process_exiting_with_sigabrt();
|
2019-04-01 19:18:49 -07:00
|
|
|
EXPECT_THAT(output, HasSubstr("cpputils::backtrace"));
|
2018-05-18 22:29:41 -07:00
|
|
|
}
|
|
|
|
|
2018-07-08 19:34:08 -07:00
|
|
|
TEST(BacktraceTest, ShowBacktraceOnSigAbrt_ShowsCorrectSignalName) {
|
2018-08-01 16:29:14 -07:00
|
|
|
auto output = call_process_exiting_with_sigabrt();
|
|
|
|
EXPECT_THAT(output, HasSubstr("SIGABRT"));
|
2018-05-18 22:29:41 -07:00
|
|
|
}
|
2018-07-08 19:34:08 -07:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#if !defined(_MSC_VER)
|
|
|
|
constexpr const char* sigsegv_message = "SIGSEGV";
|
|
|
|
constexpr const char* sigill_message = "SIGILL";
|
|
|
|
#else
|
|
|
|
constexpr const char* sigsegv_message = "EXCEPTION_ACCESS_VIOLATION";
|
|
|
|
constexpr const char* sigill_message = "EXCEPTION_ILLEGAL_INSTRUCTION";
|
|
|
|
#endif
|
2018-05-18 22:29:41 -07:00
|
|
|
|
|
|
|
TEST(BacktraceTest, ShowBacktraceOnSigSegv_ShowsCorrectSignalName) {
|
2018-08-01 16:29:14 -07:00
|
|
|
auto output = call_process_exiting_with_sigsegv();
|
|
|
|
EXPECT_THAT(output, HasSubstr(sigsegv_message));
|
2018-05-18 22:29:41 -07:00
|
|
|
}
|
|
|
|
|
2018-07-08 19:34:08 -07:00
|
|
|
TEST(BacktraceTest, ShowBacktraceOnSigIll_ShowsCorrectSignalName) {
|
2018-08-01 16:29:14 -07:00
|
|
|
auto output = call_process_exiting_with_sigill();
|
|
|
|
EXPECT_THAT(output, HasSubstr(sigill_message));
|
2018-05-18 22:29:41 -07:00
|
|
|
}
|
|
|
|
|
2018-08-01 16:29:14 -07:00
|
|
|
#if !defined(_MSC_VER)
|
|
|
|
TEST(BacktraceTest, ShowBacktraceOnUnhandledException_ShowsCorrectExceptionMessage) {
|
|
|
|
auto output = call_process_exiting_with_exception("my_exception_message");
|
|
|
|
EXPECT_THAT(output, HasSubstr("my_exception_message"));
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2018-07-08 19:34:08 -07:00
|
|
|
#if defined(_MSC_VER)
|
|
|
|
TEST(BacktraceTest, UnknownCode_ShowsCorrectSignalName) {
|
2018-09-19 03:01:31 -07:00
|
|
|
auto output = call_process_exiting_with_code(0x1234567);
|
|
|
|
EXPECT_THAT(output, HasSubstr("UNKNOWN_CODE(0x1234567)"));
|
2018-05-18 22:29:41 -07:00
|
|
|
}
|
2018-07-08 19:34:08 -07:00
|
|
|
#endif
|