blob: 9c9055e4631c85659a38ad9ffbd447d8dfd61fd0 [file] [log] [blame]
// Copyright 2014 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 "chromeos/asynchronous_signal_handler.h"
#include <signal.h>
#include <sys/types.h>
#include <unistd.h>
#include <vector>
#include <base/at_exit.h>
#include <base/bind.h>
#include <base/macros.h>
#include <base/message_loop/message_loop.h>
#include <base/run_loop.h>
#include <gtest/gtest.h>
namespace chromeos {
class AsynchronousSignalHandlerTest : public ::testing::Test {
public:
AsynchronousSignalHandlerTest() {}
virtual ~AsynchronousSignalHandlerTest() {}
virtual void SetUp() { handler_.Init(); }
virtual void TearDown() {}
bool RecordInfoAndQuit(bool response, const struct signalfd_siginfo& info) {
infos_.push_back(info);
loop_.PostTask(FROM_HERE, base::MessageLoop::QuitClosure());
return response;
}
void Run() {
base::RunLoop run_loop;
run_loop.Run();
}
protected:
base::AtExitManager at_exit_;
base::MessageLoopForIO loop_;
std::vector<struct signalfd_siginfo> infos_;
AsynchronousSignalHandler handler_;
private:
DISALLOW_COPY_AND_ASSIGN(AsynchronousSignalHandlerTest);
};
TEST_F(AsynchronousSignalHandlerTest, CheckTerm) {
handler_.RegisterHandler(
SIGTERM,
base::Bind(&AsynchronousSignalHandlerTest::RecordInfoAndQuit,
base::Unretained(this),
true));
EXPECT_EQ(0, infos_.size());
EXPECT_EQ(0, kill(getpid(), SIGTERM));
// Spin the message loop.
Run();
ASSERT_EQ(1, infos_.size());
EXPECT_EQ(SIGTERM, infos_[0].ssi_signo);
}
TEST_F(AsynchronousSignalHandlerTest, CheckSignalUnregistration) {
handler_.RegisterHandler(
SIGCHLD,
base::Bind(&AsynchronousSignalHandlerTest::RecordInfoAndQuit,
base::Unretained(this),
true));
EXPECT_EQ(0, infos_.size());
EXPECT_EQ(0, kill(getpid(), SIGCHLD));
// Spin the message loop.
Run();
ASSERT_EQ(1, infos_.size());
EXPECT_EQ(SIGCHLD, infos_[0].ssi_signo);
EXPECT_EQ(0, kill(getpid(), SIGCHLD));
// Run the loop with a timeout, as no message are expected.
loop_.PostDelayedTask(FROM_HERE,
base::MessageLoop::QuitClosure(),
base::TimeDelta::FromMilliseconds(10));
Run();
// The signal handle should have been unregistered. No new message are
// expected.
EXPECT_EQ(1, infos_.size());
}
TEST_F(AsynchronousSignalHandlerTest, CheckMultipleSignal) {
const uint8_t NB_SIGNALS = 5;
handler_.RegisterHandler(
SIGCHLD,
base::Bind(&AsynchronousSignalHandlerTest::RecordInfoAndQuit,
base::Unretained(this),
false));
EXPECT_EQ(0, infos_.size());
for (int i = 0; i < NB_SIGNALS; ++i) {
EXPECT_EQ(0, kill(getpid(), SIGCHLD));
// Spin the message loop.
Run();
}
ASSERT_EQ(NB_SIGNALS, infos_.size());
for (int i = 0; i < NB_SIGNALS; ++i) {
EXPECT_EQ(SIGCHLD, infos_[i].ssi_signo);
}
}
TEST_F(AsynchronousSignalHandlerTest, CheckChld) {
handler_.RegisterHandler(
SIGCHLD,
base::Bind(&AsynchronousSignalHandlerTest::RecordInfoAndQuit,
base::Unretained(this),
false));
pid_t child_pid = fork();
if (child_pid == 0) {
_Exit(EXIT_SUCCESS);
}
EXPECT_EQ(0, infos_.size());
// Spin the message loop.
Run();
ASSERT_EQ(1, infos_.size());
EXPECT_EQ(SIGCHLD, infos_[0].ssi_signo);
EXPECT_EQ(child_pid, infos_[0].ssi_pid);
EXPECT_EQ(static_cast<int>(CLD_EXITED), infos_[0].ssi_code);
EXPECT_EQ(EXIT_SUCCESS, infos_[0].ssi_status);
}
} // namespace chromeos