mirror of
https://github.com/fish-shell/fish-shell
synced 2025-01-06 18:18:51 +00:00
fc9d238642
topic_monitor allows for querying changes posted to one or more topics, initially sigchld. This will eventually replace the waitpid logic in process_mark_finished_children(). Comment from the new header: Topic monitoring support. Topics are conceptually "a thing that can happen." For example, delivery of a SIGINT, a child process exits, etc. It is possible to post to a topic, which means that that thing happened. Associated with each topic is a current generation, which is a 64 bit value. When you query a topic, you get back a generation. If on the next query the generation has increased, then it indicates someone posted to the topic. For example, if you are monitoring a child process, you can query the sigchld topic. If it has increased since your last query, it is possible that your child process has exited. Topic postings may be coalesced. That is there may be two posts to a given topic, yet the generation only increases by 1. The only guarantee is that after a topic post, the current generation value is larger than any value previously queried. Tying this all together is the topic_monitor_t. This provides the current topic generations, and also provides the ability to perform a blocking wait for any topic to change in a particular topic set. This is the real power of topics: you can wait for a sigchld signal OR a thread exit.
114 lines
4.3 KiB
C++
114 lines
4.3 KiB
C++
#include "config.h" // IWYU pragma: keep
|
|
|
|
#include "limits.h"
|
|
#include "topic_monitor.h"
|
|
#include "wutil.h"
|
|
|
|
#include <unistd.h>
|
|
|
|
/// Implementation of the principal monitor. This uses new (and leaks) to avoid registering a
|
|
/// pointless at-exit handler for the dtor.
|
|
static topic_monitor_t *const s_principal = new topic_monitor_t();
|
|
|
|
topic_monitor_t &topic_monitor_t::principal() {
|
|
// Do not attempt to move s_principal to a function-level static, it needs to be accessed from a
|
|
// signal handler so it must not be lazily created.
|
|
return *s_principal;
|
|
}
|
|
|
|
topic_monitor_t::topic_monitor_t() {
|
|
// Set up our pipes. Assert it succeeds.
|
|
auto pipes = make_autoclose_pipes({});
|
|
assert(pipes.has_value() && "Failed to make pubsub pipes");
|
|
pipes_ = pipes.acquire();
|
|
|
|
// Make sure that our write side doesn't block, else we risk hanging in a signal handler.
|
|
// The read end must block to avoid spinning in await.
|
|
DIE_ON_FAILURE(make_fd_nonblocking(pipes_.write.fd()));
|
|
}
|
|
|
|
topic_monitor_t::~topic_monitor_t() = default;
|
|
|
|
void topic_monitor_t::post(topic_t topic) {
|
|
// Beware, we may be in a signal handler!
|
|
// Atomically update the pending topics.
|
|
auto rawtopics = topic_set_t{topic}.to_raw();
|
|
auto oldtopics = pending_updates_.fetch_or(rawtopics, std::memory_order_relaxed);
|
|
if ((oldtopics & rawtopics) == rawtopics) {
|
|
// No new bits were set.
|
|
return;
|
|
}
|
|
|
|
// Ok, we changed one or more bits. Ensure the topic change is visible, and announce the change
|
|
// by writing a byte to the pipe.
|
|
std::atomic_thread_fence(std::memory_order_release);
|
|
ssize_t ret;
|
|
do {
|
|
// write() is async signal safe.
|
|
const uint8_t v = 0;
|
|
ret = write(pipes_.write.fd(), &v, sizeof v);
|
|
} while (ret < 0 && errno == EINTR);
|
|
// Ignore EAGAIN and other errors (which conceivably could occur during shutdown).
|
|
}
|
|
|
|
void topic_monitor_t::await_metagen(generation_t mgen) {
|
|
// Fast check of the metagen before taking the lock. If it's changed we're done.
|
|
if (mgen != current_metagen()) return;
|
|
|
|
// Take the lock (which may take a long time) and then check again.
|
|
std::unique_lock<std::mutex> locker{wait_queue_lock_};
|
|
if (mgen != current_metagen()) return;
|
|
|
|
// Our metagen hasn't changed. Push our metagen onto the queue, then wait until we're the
|
|
// lowest. If multiple waiters are the lowest, then anyone can be the observer.
|
|
// Note the reason for picking the lowest metagen is to avoid a priority inversion where a lower
|
|
// metagen (therefore someone who should see changes) is blocked waiting for a higher metagen
|
|
// (who has already seen the changes).
|
|
wait_queue_.push(mgen);
|
|
while (wait_queue_.top() != mgen) {
|
|
wait_queue_notifier_.wait(locker);
|
|
}
|
|
wait_queue_.pop();
|
|
|
|
// We now have the lowest metagen in the wait queue. Notice we still hold the lock.
|
|
// Read until the metagen changes. It may already have changed.
|
|
// Note because changes are coalesced, we can read a lot, potentially draining the pipe.
|
|
while (mgen == current_metagen()) {
|
|
uint8_t ignored[PIPE_BUF];
|
|
(void)read(pipes_.read.fd(), ignored, sizeof ignored);
|
|
}
|
|
|
|
// Release the lock and wake up the remaining waiters.
|
|
locker.unlock();
|
|
wait_queue_notifier_.notify_all();
|
|
}
|
|
|
|
topic_set_t topic_monitor_t::check(generation_list_t *gens, topic_set_t topics, bool wait) {
|
|
if (topics.none()) return topics;
|
|
|
|
topic_set_t changed{};
|
|
for (;;) {
|
|
// Load the topic list and see if anything has changed.
|
|
generation_list_t current = updated_gens();
|
|
for (topic_t topic : topic_iter_t{}) {
|
|
if (topics.get(topic)) {
|
|
assert(gens->at(topic) <= current.at(topic) &&
|
|
"Incoming gen count exceeded published count");
|
|
if (gens->at(topic) < current.at(topic)) {
|
|
gens->at(topic) = current.at(topic);
|
|
changed.set(topic);
|
|
}
|
|
}
|
|
}
|
|
|
|
// If we're not waiting, or something changed, then we're done.
|
|
if (!wait || changed.any()) {
|
|
break;
|
|
}
|
|
|
|
// Try again. Note that we use the metagen corresponding to the topic list we just
|
|
// inspected, not the current one (which may have updates since we checked).
|
|
await_metagen(metagen_for(current));
|
|
}
|
|
return changed;
|
|
}
|