blob: b8ec529f66d864f1d3d124d42037fd7dd075edd4 [file] [log] [blame]
Benjamin Lerman21c7e372014-07-10 14:20:27 +02001// Copyright 2014 The Chromium OS Authors. All rights reserved.
2// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
Alex Vakulenko9ed0cab2015-10-12 15:21:28 -07005#include "brillo/asynchronous_signal_handler.h"
Benjamin Lerman21c7e372014-07-10 14:20:27 +02006
7#include <signal.h>
8#include <sys/types.h>
9#include <unistd.h>
10
Alex Deymofd626d72015-07-20 17:27:51 -070011#include <base/bind.h>
Ben Chancb104642014-09-05 08:21:06 -070012#include <base/files/file_util.h>
Benjamin Lerman21c7e372014-07-10 14:20:27 +020013#include <base/logging.h>
14#include <base/message_loop/message_loop.h>
Alex Deymofd626d72015-07-20 17:27:51 -070015#include <base/posix/eintr_wrapper.h>
Benjamin Lerman21c7e372014-07-10 14:20:27 +020016
Benjamin Lerman21c7e372014-07-10 14:20:27 +020017namespace {
18const int kInvalidDescriptor = -1;
19} // namespace
20
Alex Vakulenko9ed0cab2015-10-12 15:21:28 -070021namespace brillo {
Benjamin Lerman21c7e372014-07-10 14:20:27 +020022
23AsynchronousSignalHandler::AsynchronousSignalHandler()
Alex Deymofd626d72015-07-20 17:27:51 -070024 : descriptor_(kInvalidDescriptor) {
Benjamin Lerman21c7e372014-07-10 14:20:27 +020025 CHECK_EQ(sigemptyset(&signal_mask_), 0) << "Failed to initialize signal mask";
26 CHECK_EQ(sigemptyset(&saved_signal_mask_), 0)
27 << "Failed to initialize signal mask";
28}
29
30AsynchronousSignalHandler::~AsynchronousSignalHandler() {
31 if (descriptor_ != kInvalidDescriptor) {
Alex Deymofd626d72015-07-20 17:27:51 -070032 MessageLoop::current()->CancelTask(fd_watcher_task_);
Benjamin Lerman21c7e372014-07-10 14:20:27 +020033
Alex Deymofd626d72015-07-20 17:27:51 -070034 if (IGNORE_EINTR(close(descriptor_)) != 0)
Benjamin Lerman21c7e372014-07-10 14:20:27 +020035 PLOG(WARNING) << "Failed to close file descriptor";
36
37 descriptor_ = kInvalidDescriptor;
Alex Vakulenko9d99bc62014-08-28 13:42:01 -070038 CHECK_EQ(0, sigprocmask(SIG_SETMASK, &saved_signal_mask_, nullptr));
Benjamin Lerman21c7e372014-07-10 14:20:27 +020039 }
40}
41
42void AsynchronousSignalHandler::Init() {
43 CHECK_EQ(kInvalidDescriptor, descriptor_);
44 CHECK_EQ(0, sigprocmask(SIG_BLOCK, &signal_mask_, &saved_signal_mask_));
45 descriptor_ =
46 signalfd(descriptor_, &signal_mask_, SFD_CLOEXEC | SFD_NONBLOCK);
47 CHECK_NE(kInvalidDescriptor, descriptor_);
Alex Deymofd626d72015-07-20 17:27:51 -070048 fd_watcher_task_ = MessageLoop::current()->WatchFileDescriptor(
49 FROM_HERE,
Alex Vakulenko05d29042015-01-13 09:39:25 -080050 descriptor_,
Alex Deymofd626d72015-07-20 17:27:51 -070051 MessageLoop::WatchMode::kWatchRead,
Alex Vakulenko05d29042015-01-13 09:39:25 -080052 true,
Alex Deymofd626d72015-07-20 17:27:51 -070053 base::Bind(&AsynchronousSignalHandler::OnFileCanReadWithoutBlocking,
54 base::Unretained(this)));
55 CHECK(fd_watcher_task_ != MessageLoop::kTaskIdNull)
Benjamin Lerman21c7e372014-07-10 14:20:27 +020056 << "Watching shutdown pipe failed.";
57}
58
59void AsynchronousSignalHandler::RegisterHandler(int signal,
60 const SignalHandler& callback) {
61 registered_callbacks_[signal] = callback;
62 CHECK_EQ(0, sigaddset(&signal_mask_, signal));
63 UpdateSignals();
64}
65
66void AsynchronousSignalHandler::UnregisterHandler(int signal) {
67 Callbacks::iterator callback_it = registered_callbacks_.find(signal);
68 if (callback_it != registered_callbacks_.end()) {
69 registered_callbacks_.erase(callback_it);
70 ResetSignal(signal);
71 }
72}
73
Alex Deymofd626d72015-07-20 17:27:51 -070074void AsynchronousSignalHandler::OnFileCanReadWithoutBlocking() {
Benjamin Lerman21c7e372014-07-10 14:20:27 +020075 struct signalfd_siginfo info;
Alex Deymofd626d72015-07-20 17:27:51 -070076 while (base::ReadFromFD(descriptor_,
77 reinterpret_cast<char*>(&info), sizeof(info))) {
Benjamin Lerman21c7e372014-07-10 14:20:27 +020078 int signal = info.ssi_signo;
79 Callbacks::iterator callback_it = registered_callbacks_.find(signal);
80 if (callback_it == registered_callbacks_.end()) {
81 LOG(WARNING) << "Unable to find a signal handler for signal: " << signal;
82 // Can happen if a signal has been called multiple time, and the callback
83 // asked to be unregistered the first time.
84 continue;
85 }
86 const SignalHandler& callback = callback_it->second;
87 bool must_unregister = callback.Run(info);
88 if (must_unregister) {
89 UnregisterHandler(signal);
90 }
91 }
92}
93
Benjamin Lerman21c7e372014-07-10 14:20:27 +020094void AsynchronousSignalHandler::ResetSignal(int signal) {
95 CHECK_EQ(0, sigdelset(&signal_mask_, signal));
96 UpdateSignals();
97}
98
99void AsynchronousSignalHandler::UpdateSignals() {
100 if (descriptor_ != kInvalidDescriptor) {
Alex Vakulenko9d99bc62014-08-28 13:42:01 -0700101 CHECK_EQ(0, sigprocmask(SIG_SETMASK, &saved_signal_mask_, nullptr));
102 CHECK_EQ(0, sigprocmask(SIG_BLOCK, &signal_mask_, nullptr));
Benjamin Lerman21c7e372014-07-10 14:20:27 +0200103 CHECK_EQ(descriptor_,
104 signalfd(descriptor_, &signal_mask_, SFD_CLOEXEC | SFD_NONBLOCK));
105 }
106}
107
Alex Vakulenko9ed0cab2015-10-12 15:21:28 -0700108} // namespace brillo