// Copyright 2015 The Chromium OS Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #include <brillo/process_reaper.h> #include <signal.h> #include <sys/wait.h> #include <unistd.h> #include <base/bind.h> #include <base/location.h> #include <base/message_loop/message_loop.h> #include <brillo/asynchronous_signal_handler.h> #include <brillo/bind_lambda.h> #include <brillo/message_loops/base_message_loop.h> #include <gtest/gtest.h> namespace { pid_t ForkChildAndExit(int exit_code) { pid_t pid = fork(); PCHECK(pid != -1); if (pid == 0) { _exit(exit_code); } return pid; } pid_t ForkChildAndKill(int sig) { pid_t pid = fork(); PCHECK(pid != -1); if (pid == 0) { if (raise(sig) != 0) { PLOG(ERROR) << "raise(" << sig << ")"; } _exit(0); // Not reached. This value will cause the test to fail. } return pid; } } // namespace namespace brillo { class ProcessReaperTest : public ::testing::Test { public: void SetUp() override { brillo_loop_.SetAsCurrent(); async_signal_handler_.Init(); process_reaper_.Register(&async_signal_handler_); } protected: base::MessageLoopForIO base_loop_; brillo::BaseMessageLoop brillo_loop_{&base_loop_}; brillo::AsynchronousSignalHandler async_signal_handler_; // ProcessReaper under test. ProcessReaper process_reaper_; }; TEST_F(ProcessReaperTest, UnregisterWhenNotRegistered) { ProcessReaper another_process_reaper_; another_process_reaper_.Unregister(); } TEST_F(ProcessReaperTest, UnregisterAndReregister) { process_reaper_.Unregister(); process_reaper_.Register(&async_signal_handler_); // This checks that we can unregister the ProcessReaper and then destroy it. process_reaper_.Unregister(); } TEST_F(ProcessReaperTest, ReapExitedChild) { pid_t pid = ForkChildAndExit(123); EXPECT_TRUE(process_reaper_.WatchForChild(FROM_HERE, pid, base::Bind( [](decltype(this) test, const siginfo_t& info) { EXPECT_EQ(CLD_EXITED, info.si_code); EXPECT_EQ(123, info.si_status); test->brillo_loop_.BreakLoop(); }, base::Unretained(this)))); brillo_loop_.Run(); } // Test that simultaneous child processes fire their respective callbacks when // exiting. TEST_F(ProcessReaperTest, ReapedChildsMatchCallbacks) { int running_childs = 10; for (int i = 0; i < running_childs; ++i) { // Different processes will have different exit values. int exit_value = 1 + i; pid_t pid = ForkChildAndExit(exit_value); EXPECT_TRUE(process_reaper_.WatchForChild( FROM_HERE, pid, base::Bind( [](decltype(this) test, int exit_value, int* running_childs, const siginfo_t& info) { EXPECT_EQ(CLD_EXITED, info.si_code); EXPECT_EQ(exit_value, info.si_status); (*running_childs)--; if (*running_childs == 0) test->brillo_loop_.BreakLoop(); }, base::Unretained(this), exit_value, base::Unretained(&running_childs)))); } // This sleep is optional. It helps to have more processes exit before we // start watching for them in the message loop. usleep(10 * 1000); brillo_loop_.Run(); EXPECT_EQ(0, running_childs); } TEST_F(ProcessReaperTest, ReapKilledChild) { pid_t pid = ForkChildAndKill(SIGKILL); EXPECT_TRUE(process_reaper_.WatchForChild(FROM_HERE, pid, base::Bind( [](decltype(this) test, const siginfo_t& info) { EXPECT_EQ(CLD_KILLED, info.si_code); EXPECT_EQ(SIGKILL, info.si_status); test->brillo_loop_.BreakLoop(); }, base::Unretained(this)))); brillo_loop_.Run(); } TEST_F(ProcessReaperTest, ReapKilledAndForgottenChild) { pid_t pid = ForkChildAndExit(0); EXPECT_TRUE(process_reaper_.WatchForChild(FROM_HERE, pid, base::Bind( [](decltype(this) test, const siginfo_t& /* info */) { ADD_FAILURE() << "Child process was still tracked."; test->brillo_loop_.BreakLoop(); }, base::Unretained(this)))); EXPECT_TRUE(process_reaper_.ForgetChild(pid)); // A second call should return failure. EXPECT_FALSE(process_reaper_.ForgetChild(pid)); // Run the loop with a timeout, as the BreakLoop() above is not expected. brillo_loop_.PostDelayedTask(FROM_HERE, base::Bind(&MessageLoop::BreakLoop, base::Unretained(&brillo_loop_)), base::TimeDelta::FromMilliseconds(100)); brillo_loop_.Run(); } } // namespace brillo