mirror of
https://github.com/fish-shell/fish-shell
synced 2025-01-16 06:54:03 +00:00
Use fd_event_signaller_t in fd_monitor_t
fd_monitor_t allows observing a collection of fds. It also has its own fd, which it uses to awaken itself when there are changes. Switch to using fd_event_signaller_t instead of a pipe; this reduces the number of file descriptors and is more efficient under Linux.
This commit is contained in:
parent
e004930947
commit
a942df3886
2 changed files with 37 additions and 69 deletions
|
@ -11,45 +11,14 @@
|
||||||
static constexpr uint64_t kUsecPerMsec = 1000;
|
static constexpr uint64_t kUsecPerMsec = 1000;
|
||||||
static constexpr uint64_t kUsecPerSec = 1000 * kUsecPerMsec;
|
static constexpr uint64_t kUsecPerSec = 1000 * kUsecPerMsec;
|
||||||
|
|
||||||
fd_monitor_t::fd_monitor_t() {
|
fd_monitor_t::fd_monitor_t() = default;
|
||||||
auto self_pipe = make_autoclose_pipes();
|
|
||||||
if (!self_pipe) {
|
|
||||||
DIE("Unable to create pipes");
|
|
||||||
}
|
|
||||||
|
|
||||||
// Ensure the write side is nonblocking to avoid deadlock.
|
|
||||||
notify_write_fd_ = std::move(self_pipe->write);
|
|
||||||
if (make_fd_nonblocking(notify_write_fd_.fd())) {
|
|
||||||
wperror(L"fcntl");
|
|
||||||
}
|
|
||||||
|
|
||||||
// Add an item for ourselves.
|
|
||||||
// We don't need to go through 'pending' because we have not yet launched the thread, and don't
|
|
||||||
// want to yet.
|
|
||||||
auto callback = [this](const autoclose_fd_t &fd, item_wake_reason_t reason) {
|
|
||||||
ASSERT_IS_BACKGROUND_THREAD();
|
|
||||||
assert(reason == item_wake_reason_t::readable &&
|
|
||||||
"Should not be poked, or time out with kNoTimeout");
|
|
||||||
(void)reason;
|
|
||||||
// Read some to take data off of the notifier.
|
|
||||||
char buff[4096];
|
|
||||||
ssize_t amt = read(fd.fd(), buff, sizeof buff);
|
|
||||||
if (amt > 0) {
|
|
||||||
this->has_pending_or_pokes_ = true;
|
|
||||||
} else if (amt == 0) {
|
|
||||||
this->terminate_ = true;
|
|
||||||
} else {
|
|
||||||
wperror(L"read");
|
|
||||||
}
|
|
||||||
};
|
|
||||||
items_.push_back(fd_monitor_item_t(std::move(self_pipe->read), std::move(callback)));
|
|
||||||
}
|
|
||||||
|
|
||||||
// Extremely hacky destructor to clean up.
|
|
||||||
// This is used in the tests to not leave stale fds around.
|
|
||||||
// In fish shell, we never invoke the dtor so it doesn't matter that this is very dumb.
|
|
||||||
fd_monitor_t::~fd_monitor_t() {
|
fd_monitor_t::~fd_monitor_t() {
|
||||||
notify_write_fd_.close();
|
// In orindary usage, we never invoke the dtor.
|
||||||
|
// This is used in the tests to not leave stale fds around.
|
||||||
|
// That is why this is very hacky!
|
||||||
|
data_.acquire()->terminate = true;
|
||||||
|
change_signaller_.post();
|
||||||
while (data_.acquire()->running) {
|
while (data_.acquire()->running) {
|
||||||
std::this_thread::sleep_for(std::chrono::milliseconds(5));
|
std::this_thread::sleep_for(std::chrono::milliseconds(5));
|
||||||
}
|
}
|
||||||
|
@ -87,26 +56,23 @@ fd_monitor_item_id_t fd_monitor_t::add(fd_monitor_item_t &&item) {
|
||||||
DIE("Unable to create a new pthread");
|
DIE("Unable to create a new pthread");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
// Tickle our notifier.
|
// Tickle our signaller.
|
||||||
char byte = 0;
|
change_signaller_.post();
|
||||||
(void)write_loop(notify_write_fd_.fd(), &byte, 1);
|
|
||||||
return item_id;
|
return item_id;
|
||||||
}
|
}
|
||||||
|
|
||||||
void fd_monitor_t::poke_item(fd_monitor_item_id_t item_id) {
|
void fd_monitor_t::poke_item(fd_monitor_item_id_t item_id) {
|
||||||
assert(item_id > 0 && "Invalid item ID");
|
assert(item_id > 0 && "Invalid item ID");
|
||||||
bool needs_notifier_byte = false;
|
bool needs_notification = false;
|
||||||
{
|
{
|
||||||
auto data = data_.acquire();
|
auto data = data_.acquire();
|
||||||
needs_notifier_byte = data->pokelist.empty();
|
needs_notification = data->pokelist.empty();
|
||||||
// Insert it, sorted.
|
// Insert it, sorted.
|
||||||
auto where = std::lower_bound(data->pokelist.begin(), data->pokelist.end(), item_id);
|
auto where = std::lower_bound(data->pokelist.begin(), data->pokelist.end(), item_id);
|
||||||
data->pokelist.insert(where, item_id);
|
data->pokelist.insert(where, item_id);
|
||||||
}
|
}
|
||||||
if (needs_notifier_byte) {
|
if (needs_notification) {
|
||||||
// Tickle our notifier.
|
change_signaller_.post();
|
||||||
char byte = 0;
|
|
||||||
(void)write_loop(notify_write_fd_.fd(), &byte, 1);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -161,11 +127,16 @@ void fd_monitor_t::run_in_background() {
|
||||||
pokelist.clear();
|
pokelist.clear();
|
||||||
}
|
}
|
||||||
|
|
||||||
uint64_t timeout_usec = fd_monitor_item_t::kNoTimeout;
|
|
||||||
int max_fd = -1;
|
|
||||||
fd_set fds;
|
fd_set fds;
|
||||||
FD_ZERO(&fds);
|
FD_ZERO(&fds);
|
||||||
|
|
||||||
|
// Our change_signaller is special cased.
|
||||||
|
int change_signal_fd = change_signaller_.read_fd();
|
||||||
|
FD_SET(change_signal_fd, &fds);
|
||||||
|
int max_fd = change_signal_fd;
|
||||||
|
|
||||||
auto now = std::chrono::steady_clock::now();
|
auto now = std::chrono::steady_clock::now();
|
||||||
|
uint64_t timeout_usec = fd_monitor_item_t::kNoTimeout;
|
||||||
|
|
||||||
for (auto &item : items_) {
|
for (auto &item : items_) {
|
||||||
FD_SET(item.fd.fd(), &fds);
|
FD_SET(item.fd.fd(), &fds);
|
||||||
|
@ -207,16 +178,13 @@ void fd_monitor_t::run_in_background() {
|
||||||
now = std::chrono::steady_clock::now();
|
now = std::chrono::steady_clock::now();
|
||||||
items_.erase(std::remove_if(items_.begin(), items_.end(), servicer), items_.end());
|
items_.erase(std::remove_if(items_.begin(), items_.end(), servicer), items_.end());
|
||||||
|
|
||||||
if (terminate_) {
|
// Handle any changes if the change signaller was set. Alternatively this may be the wait
|
||||||
// Time to go.
|
|
||||||
data_.acquire()->running = false;
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Maybe we got some new items. Check if our callback says so, or if this is the wait
|
|
||||||
// lap, in which case we might want to commit to exiting.
|
// lap, in which case we might want to commit to exiting.
|
||||||
if (has_pending_or_pokes_ || is_wait_lap) {
|
if (FD_ISSET(change_signal_fd, &fds) || is_wait_lap) {
|
||||||
|
// Clear the change signaller before processing incoming changes.
|
||||||
|
change_signaller_.try_consume();
|
||||||
auto data = data_.acquire();
|
auto data = data_.acquire();
|
||||||
|
|
||||||
// Move from 'pending' to 'items'.
|
// Move from 'pending' to 'items'.
|
||||||
items_.insert(items_.end(), std::make_move_iterator(data->pending.begin()),
|
items_.insert(items_.end(), std::make_move_iterator(data->pending.begin()),
|
||||||
std::make_move_iterator(data->pending.end()));
|
std::make_move_iterator(data->pending.end()));
|
||||||
|
@ -227,10 +195,9 @@ void fd_monitor_t::run_in_background() {
|
||||||
pokelist = std::move(data->pokelist);
|
pokelist = std::move(data->pokelist);
|
||||||
data->pokelist.clear();
|
data->pokelist.clear();
|
||||||
|
|
||||||
has_pending_or_pokes_ = false;
|
if (data->terminate || (is_wait_lap && items_.empty())) {
|
||||||
|
// Maybe terminate is set.
|
||||||
if (is_wait_lap && items_.size() == 1) {
|
// Alternatively, maybe we had no items, waited a bit, and still have no items.
|
||||||
// We had no items, waited a bit, and still have no items. We're going to shut down.
|
|
||||||
// It's important to do this while holding the lock, otherwise we race with new
|
// It's important to do this while holding the lock, otherwise we race with new
|
||||||
// items being added.
|
// items being added.
|
||||||
assert(data->running && "Thread should be running because we're that thread");
|
assert(data->running && "Thread should be running because we're that thread");
|
||||||
|
|
|
@ -103,6 +103,10 @@ class fd_monitor_t {
|
||||||
// The background thread runner.
|
// The background thread runner.
|
||||||
void run_in_background();
|
void run_in_background();
|
||||||
|
|
||||||
|
// If our self-signaller is reported as ready, this reads from it and handles any changes.
|
||||||
|
// Called in the background thread.
|
||||||
|
void handle_self_signal_in_background();
|
||||||
|
|
||||||
// Poke items in the pokelist, removing any items that close their FD.
|
// Poke items in the pokelist, removing any items that close their FD.
|
||||||
// The pokelist is consumed after this.
|
// The pokelist is consumed after this.
|
||||||
// This is only called in the background thread.
|
// This is only called in the background thread.
|
||||||
|
@ -111,13 +115,6 @@ class fd_monitor_t {
|
||||||
// The list of items to monitor. This is only accessed on the background thread.
|
// The list of items to monitor. This is only accessed on the background thread.
|
||||||
item_list_t items_{};
|
item_list_t items_{};
|
||||||
|
|
||||||
// Set to true by the background thread when our self-pipe becomes readable.
|
|
||||||
bool has_pending_or_pokes_{false};
|
|
||||||
|
|
||||||
// Latched to true by the background thread if our self-pipe is closed, which indicates we are
|
|
||||||
// in the destructor and so should terminate.
|
|
||||||
bool terminate_{false};
|
|
||||||
|
|
||||||
struct data_t {
|
struct data_t {
|
||||||
/// Pending items. This is set under the lock, then the background thread grabs them.
|
/// Pending items. This is set under the lock, then the background thread grabs them.
|
||||||
item_list_t pending{};
|
item_list_t pending{};
|
||||||
|
@ -130,11 +127,15 @@ class fd_monitor_t {
|
||||||
|
|
||||||
/// Whether the thread is running.
|
/// Whether the thread is running.
|
||||||
bool running{false};
|
bool running{false};
|
||||||
|
|
||||||
|
// Set if we should terminate.
|
||||||
|
bool terminate{false};
|
||||||
};
|
};
|
||||||
owning_lock<data_t> data_;
|
owning_lock<data_t> data_;
|
||||||
|
|
||||||
/// The write end of our self-pipe.
|
/// Our self-signaller. When this is written to, it means there are new items pending, or new
|
||||||
autoclose_fd_t notify_write_fd_{};
|
/// items in the pokelist, or terminate is set.
|
||||||
|
fd_event_signaller_t change_signaller_;
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
Loading…
Reference in a new issue