forked from OSchip/llvm-project
188 lines
5.7 KiB
C++
188 lines
5.7 KiB
C++
//===- llvm/unittest/Support/CrashRecoveryTest.cpp ------------------------===//
|
|
//
|
|
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
|
|
// See https://llvm.org/LICENSE.txt for license information.
|
|
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#include "llvm/ADT/Triple.h"
|
|
#include "llvm/Config/config.h"
|
|
#include "llvm/Support/CommandLine.h"
|
|
#include "llvm/Support/Compiler.h"
|
|
#include "llvm/Support/CrashRecoveryContext.h"
|
|
#include "llvm/Support/FileSystem.h"
|
|
#include "llvm/Support/Host.h"
|
|
#include "llvm/Support/Program.h"
|
|
#include "llvm/Support/Signals.h"
|
|
#include "llvm/Support/raw_ostream.h"
|
|
#include "gtest/gtest.h"
|
|
|
|
#ifdef _WIN32
|
|
#define WIN32_LEAN_AND_MEAN
|
|
#define NOGDI
|
|
#include <windows.h>
|
|
#endif
|
|
|
|
#ifdef LLVM_ON_UNIX
|
|
#ifdef HAVE_SIGNAL_H
|
|
#include <signal.h>
|
|
#endif
|
|
#endif
|
|
|
|
using namespace llvm;
|
|
using namespace llvm::sys;
|
|
|
|
static int GlobalInt = 0;
|
|
static void nullDeref() { *(volatile int *)0x10 = 0; }
|
|
static void incrementGlobal() { ++GlobalInt; }
|
|
static void llvmTrap() { LLVM_BUILTIN_TRAP; }
|
|
static void incrementGlobalWithParam(void *) { ++GlobalInt; }
|
|
|
|
TEST(CrashRecoveryTest, Basic) {
|
|
llvm::CrashRecoveryContext::Enable();
|
|
GlobalInt = 0;
|
|
EXPECT_TRUE(CrashRecoveryContext().RunSafely(incrementGlobal));
|
|
EXPECT_EQ(1, GlobalInt);
|
|
EXPECT_FALSE(CrashRecoveryContext().RunSafely(nullDeref));
|
|
EXPECT_FALSE(CrashRecoveryContext().RunSafely(llvmTrap));
|
|
}
|
|
|
|
struct IncrementGlobalCleanup : CrashRecoveryContextCleanup {
|
|
IncrementGlobalCleanup(CrashRecoveryContext *CRC)
|
|
: CrashRecoveryContextCleanup(CRC) {}
|
|
void recoverResources() override { ++GlobalInt; }
|
|
};
|
|
|
|
static void noop() {}
|
|
|
|
TEST(CrashRecoveryTest, Cleanup) {
|
|
llvm::CrashRecoveryContext::Enable();
|
|
GlobalInt = 0;
|
|
{
|
|
CrashRecoveryContext CRC;
|
|
CRC.registerCleanup(new IncrementGlobalCleanup(&CRC));
|
|
EXPECT_TRUE(CRC.RunSafely(noop));
|
|
} // run cleanups
|
|
EXPECT_EQ(1, GlobalInt);
|
|
|
|
GlobalInt = 0;
|
|
{
|
|
CrashRecoveryContext CRC;
|
|
CRC.registerCleanup(new IncrementGlobalCleanup(&CRC));
|
|
EXPECT_FALSE(CRC.RunSafely(nullDeref));
|
|
} // run cleanups
|
|
EXPECT_EQ(1, GlobalInt);
|
|
llvm::CrashRecoveryContext::Disable();
|
|
}
|
|
|
|
TEST(CrashRecoveryTest, DumpStackCleanup) {
|
|
SmallString<128> Filename;
|
|
std::error_code EC = sys::fs::createTemporaryFile("crash", "test", Filename);
|
|
EXPECT_FALSE(EC);
|
|
sys::RemoveFileOnSignal(Filename);
|
|
llvm::sys::AddSignalHandler(incrementGlobalWithParam, nullptr);
|
|
GlobalInt = 0;
|
|
llvm::CrashRecoveryContext::Enable();
|
|
{
|
|
CrashRecoveryContext CRC;
|
|
CRC.DumpStackAndCleanupOnFailure = true;
|
|
EXPECT_TRUE(CRC.RunSafely(noop));
|
|
}
|
|
EXPECT_TRUE(sys::fs::exists(Filename));
|
|
EXPECT_EQ(GlobalInt, 0);
|
|
{
|
|
CrashRecoveryContext CRC;
|
|
CRC.DumpStackAndCleanupOnFailure = true;
|
|
EXPECT_FALSE(CRC.RunSafely(nullDeref));
|
|
EXPECT_NE(CRC.RetCode, 0);
|
|
}
|
|
EXPECT_FALSE(sys::fs::exists(Filename));
|
|
EXPECT_EQ(GlobalInt, 1);
|
|
llvm::CrashRecoveryContext::Disable();
|
|
}
|
|
|
|
TEST(CrashRecoveryTest, LimitedStackTrace) {
|
|
std::string Res;
|
|
llvm::raw_string_ostream RawStream(Res);
|
|
PrintStackTrace(RawStream, 1);
|
|
std::string Str = RawStream.str();
|
|
// FIXME: Handle "Depth" parameter in PrintStackTrace() function
|
|
// to print stack trace upto a specified Depth.
|
|
if (!Triple(sys::getProcessTriple()).isOSWindows()) {
|
|
EXPECT_EQ(std::string::npos, Str.find("#1"));
|
|
}
|
|
}
|
|
|
|
#ifdef _WIN32
|
|
static void raiseIt() {
|
|
RaiseException(123, EXCEPTION_NONCONTINUABLE, 0, NULL);
|
|
}
|
|
|
|
TEST(CrashRecoveryTest, RaiseException) {
|
|
llvm::CrashRecoveryContext::Enable();
|
|
EXPECT_FALSE(CrashRecoveryContext().RunSafely(raiseIt));
|
|
}
|
|
|
|
static void outputString() {
|
|
OutputDebugStringA("output for debugger\n");
|
|
}
|
|
|
|
TEST(CrashRecoveryTest, CallOutputDebugString) {
|
|
llvm::CrashRecoveryContext::Enable();
|
|
EXPECT_TRUE(CrashRecoveryContext().RunSafely(outputString));
|
|
}
|
|
|
|
TEST(CrashRecoveryTest, Abort) {
|
|
llvm::CrashRecoveryContext::Enable();
|
|
auto A = []() { abort(); };
|
|
EXPECT_FALSE(CrashRecoveryContext().RunSafely(A));
|
|
// Test a second time to ensure we reinstall the abort signal handler.
|
|
EXPECT_FALSE(CrashRecoveryContext().RunSafely(A));
|
|
}
|
|
#endif
|
|
|
|
// Specifically ensure that programs that signal() or abort() through the
|
|
// CrashRecoveryContext can re-throw again their signal, so that `not --crash`
|
|
// succeeds.
|
|
#ifdef LLVM_ON_UNIX
|
|
// See llvm/utils/unittest/UnitTestMain/TestMain.cpp
|
|
extern const char *TestMainArgv0;
|
|
|
|
// Just a reachable symbol to ease resolving of the executable's path.
|
|
static cl::opt<std::string> CrashTestStringArg1("crash-test-string-arg1");
|
|
|
|
TEST(CrashRecoveryTest, UnixCRCReturnCode) {
|
|
using namespace llvm::sys;
|
|
if (getenv("LLVM_CRC_UNIXCRCRETURNCODE")) {
|
|
llvm::CrashRecoveryContext::Enable();
|
|
CrashRecoveryContext CRC;
|
|
// This path runs in a subprocess that exits by signalling, so don't use
|
|
// the googletest macros to verify things as they won't report properly.
|
|
if (CRC.RunSafely(abort))
|
|
llvm_unreachable("RunSafely returned true!");
|
|
if (CRC.RetCode != 128 + SIGABRT)
|
|
llvm_unreachable("Unexpected RetCode!");
|
|
// re-throw signal
|
|
llvm::sys::unregisterHandlers();
|
|
raise(CRC.RetCode - 128);
|
|
llvm_unreachable("Should have exited already!");
|
|
}
|
|
|
|
std::string Executable =
|
|
sys::fs::getMainExecutable(TestMainArgv0, &CrashTestStringArg1);
|
|
StringRef argv[] = {
|
|
Executable, "--gtest_filter=CrashRecoveryTest.UnixCRCReturnCode"};
|
|
|
|
// Add LLVM_CRC_UNIXCRCRETURNCODE to the environment of the child process.
|
|
int Res = setenv("LLVM_CRC_UNIXCRCRETURNCODE", "1", 0);
|
|
ASSERT_EQ(Res, 0);
|
|
|
|
std::string Error;
|
|
bool ExecutionFailed;
|
|
int RetCode = ExecuteAndWait(Executable, argv, {}, {}, 0, 0, &Error,
|
|
&ExecutionFailed);
|
|
ASSERT_EQ(-2, RetCode);
|
|
}
|
|
#endif
|