123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109 |
- /*
- * Copyright (c) 2018-2020, Andreas Kling <kling@serenityos.org>
- * Copyright (c) 2019-2020, Shannon Booth <shannon.ml.booth@gmail.com>
- * Copyright (c) 2021, Brian Gianforcaro <bgianf@serenityos.org>
- *
- * SPDX-License-Identifier: BSD-2-Clause
- */
- #include <AK/Assertions.h>
- #include <AK/Platform.h>
- #include <LibTest/CrashTest.h>
- #include <sys/wait.h>
- #include <unistd.h>
- #if !defined(AK_OS_MACOS) && !defined(AK_OS_EMSCRIPTEN)
- # include <sys/prctl.h>
- #endif
- namespace Test {
- Crash::Crash(String test_type, Function<Crash::Failure()> crash_function, int crash_signal)
- : m_type(move(test_type))
- , m_crash_function(move(crash_function))
- , m_crash_signal(crash_signal)
- {
- }
- bool Crash::run(RunType run_type)
- {
- outln("\x1B[33mTesting\x1B[0m: \"{}\"", m_type);
- if (run_type == RunType::UsingCurrentProcess) {
- return do_report(m_crash_function());
- } else {
- // Run the test in a child process so that we do not crash the crash program :^)
- pid_t pid = fork();
- if (pid < 0) {
- perror("fork");
- VERIFY_NOT_REACHED();
- } else if (pid == 0) {
- #if !defined(AK_OS_MACOS) && !defined(AK_OS_EMSCRIPTEN)
- if (prctl(PR_SET_DUMPABLE, 0, 0) < 0)
- perror("prctl(PR_SET_DUMPABLE)");
- #endif
- exit((int)m_crash_function());
- }
- int status;
- waitpid(pid, &status, 0);
- if (WIFEXITED(status)) {
- return do_report(Failure(WEXITSTATUS(status)));
- }
- if (WIFSIGNALED(status)) {
- int signal = WTERMSIG(status);
- VERIFY(signal > 0);
- return do_report(signal);
- }
- VERIFY_NOT_REACHED();
- }
- }
- bool Crash::do_report(Report report)
- {
- bool pass = false;
- if (m_crash_signal == ANY_SIGNAL) {
- pass = report.has<int>();
- } else if (m_crash_signal == 0) {
- pass = report.has<Failure>() && report.get<Failure>() == Failure::DidNotCrash;
- } else if (m_crash_signal > 0) {
- pass = report.has<int>() && report.get<int>() == m_crash_signal;
- } else {
- VERIFY_NOT_REACHED();
- }
- if (pass)
- out("\x1B[32mPASS\x1B[0m: ");
- else
- out("\x1B[31mFAIL\x1B[0m: ");
- report.visit(
- [&](Failure const& failure) {
- switch (failure) {
- case Failure::DidNotCrash:
- out("Did not crash");
- break;
- case Failure::UnexpectedError:
- out("Unexpected error");
- break;
- default:
- VERIFY_NOT_REACHED();
- }
- },
- [&](int const& signal) {
- out("Terminated with signal {}", signal);
- });
- if (!pass) {
- if (m_crash_signal == ANY_SIGNAL) {
- out(" while expecting any signal");
- } else if (m_crash_signal > 0) {
- out(" while expecting signal {}", m_crash_signal);
- }
- }
- outln();
- return pass;
- }
- }
|