mirror of
https://github.com/fish-shell/fish-shell
synced 2025-01-16 23:14:04 +00:00
265 lines
10 KiB
C++
265 lines
10 KiB
C++
#ifndef FISH_TOPIC_MONITOR_H
|
|
#define FISH_TOPIC_MONITOR_H
|
|
|
|
#include <semaphore.h>
|
|
|
|
#include <array>
|
|
#include <atomic>
|
|
#include <bitset>
|
|
#include <condition_variable>
|
|
#include <limits>
|
|
#include <numeric>
|
|
|
|
#include "common.h"
|
|
#include "io.h"
|
|
|
|
/** 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.
|
|
*/
|
|
|
|
/// A generation is a counter incremented every time the value of a topic changes.
|
|
/// It is 64 bit so it will never wrap.
|
|
using generation_t = uint64_t;
|
|
|
|
/// A generation value which indicates the topic is not of interest.
|
|
constexpr generation_t invalid_generation = std::numeric_limits<generation_t>::max();
|
|
|
|
/// The list of topics which may be observed.
|
|
enum class topic_t : uint8_t {
|
|
sighupint, // Corresponds to both SIGHUP and SIGINT signals.
|
|
sigchld, // Corresponds to SIGCHLD signal.
|
|
internal_exit, // Corresponds to an internal process exit.
|
|
};
|
|
|
|
/// Helper to return all topics, allowing easy iteration.
|
|
inline std::array<topic_t, 3> all_topics() {
|
|
return {{topic_t::sighupint, topic_t::sigchld, topic_t::internal_exit}};
|
|
}
|
|
|
|
/// Simple value type containing the values for a topic.
|
|
/// This should be kept in sync with topic_t.
|
|
class generation_list_t {
|
|
public:
|
|
generation_list_t() = default;
|
|
|
|
generation_t sighupint{0};
|
|
generation_t sigchld{0};
|
|
generation_t internal_exit{0};
|
|
|
|
/// \return the value for a topic.
|
|
generation_t &at(topic_t topic) {
|
|
switch (topic) {
|
|
case topic_t::sigchld:
|
|
return sigchld;
|
|
case topic_t::sighupint:
|
|
return sighupint;
|
|
case topic_t::internal_exit:
|
|
return internal_exit;
|
|
}
|
|
DIE("Unreachable");
|
|
}
|
|
|
|
generation_t at(topic_t topic) const {
|
|
switch (topic) {
|
|
case topic_t::sighupint:
|
|
return sighupint;
|
|
case topic_t::sigchld:
|
|
return sigchld;
|
|
case topic_t::internal_exit:
|
|
return internal_exit;
|
|
}
|
|
DIE("Unreachable");
|
|
}
|
|
|
|
/// \return ourselves as an array.
|
|
std::array<generation_t, 3> as_array() const { return {{sighupint, sigchld, internal_exit}}; }
|
|
|
|
/// Set the value of \p topic to the smaller of our value and the value in \p other.
|
|
void set_min_from(topic_t topic, const generation_list_t &other) {
|
|
if (this->at(topic) > other.at(topic)) {
|
|
this->at(topic) = other.at(topic);
|
|
}
|
|
}
|
|
|
|
/// \return whether a topic is valid.
|
|
bool is_valid(topic_t topic) const { return this->at(topic) != invalid_generation; }
|
|
|
|
/// \return whether any topic is valid.
|
|
bool any_valid() const {
|
|
bool valid = false;
|
|
for (auto gen : as_array()) {
|
|
if (gen != invalid_generation) valid = true;
|
|
}
|
|
return valid;
|
|
}
|
|
|
|
bool operator==(const generation_list_t &rhs) const {
|
|
return sighupint == rhs.sighupint && sigchld == rhs.sigchld &&
|
|
internal_exit == rhs.internal_exit;
|
|
}
|
|
|
|
bool operator!=(const generation_list_t &rhs) const { return !(*this == rhs); }
|
|
|
|
/// return a string representation for debugging.
|
|
wcstring describe() const;
|
|
|
|
/// Generation list containing invalid generations only.
|
|
static generation_list_t invalids() {
|
|
return generation_list_t(invalid_generation, invalid_generation, invalid_generation);
|
|
}
|
|
|
|
private:
|
|
generation_list_t(generation_t sighupint, generation_t sigchld, generation_t internal_exit)
|
|
: sighupint(sighupint), sigchld(sigchld), internal_exit(internal_exit) {}
|
|
};
|
|
|
|
/// A simple binary semaphore.
|
|
/// On systems that do not support unnamed semaphores (macOS in particular) this is built on top of
|
|
/// a self-pipe. Note that post() must be async-signal safe.
|
|
class binary_semaphore_t {
|
|
public:
|
|
binary_semaphore_t();
|
|
~binary_semaphore_t();
|
|
|
|
/// Release a waiting thread.
|
|
void post();
|
|
|
|
/// Wait for a post.
|
|
/// This loops on EINTR.
|
|
void wait();
|
|
|
|
private:
|
|
// Print a message and exit.
|
|
void die(const wchar_t *msg) const;
|
|
|
|
// Whether our semaphore was successfully initialized.
|
|
bool sem_ok_{};
|
|
|
|
// The semaphore, if initalized.
|
|
sem_t sem_{};
|
|
|
|
// Pipes used to emulate a semaphore, if not initialized.
|
|
autoclose_pipes_t pipes_{};
|
|
};
|
|
|
|
/// The topic monitor class. This permits querying the current generation values for topics,
|
|
/// optionally blocking until they increase.
|
|
/// What we would like to write is that we have a set of topics, and threads wait for changes on a
|
|
/// condition variable which is tickled in post(). But this can't work because post() may be called
|
|
/// from a signal handler and condition variables are not async-signal safe.
|
|
/// So instead the signal handler announces changes via a binary semaphore.
|
|
/// In the wait case, what generally happens is:
|
|
/// A thread fetches the generations, see they have not changed, and then decides to try to wait.
|
|
/// It does so by atomically swapping in STATUS_NEEDS_WAKEUP to the status bits.
|
|
/// If that succeeds, it waits on the binary semaphore. The post() call will then wake the thread
|
|
/// up. If if failed, then either a post() call updated the status values (so perhaps there is a
|
|
/// new topic post) or some other thread won the race and called wait() on the semaphore. Here our
|
|
/// thread will wait on the data_notifier_ queue.
|
|
class topic_monitor_t {
|
|
private:
|
|
using topic_bitmask_t = uint8_t;
|
|
|
|
// Some stuff that needs to be protected by the same lock.
|
|
struct data_t {
|
|
/// The current values.
|
|
generation_list_t current{};
|
|
|
|
/// A flag indicating that there is a current reader.
|
|
/// The 'reader' is responsible for calling sema_.wait().
|
|
bool has_reader{false};
|
|
};
|
|
owning_lock<data_t> data_{};
|
|
|
|
/// Condition variable for broadcasting notifications.
|
|
/// This is associated with data_'s mutex.
|
|
std::condition_variable data_notifier_{};
|
|
|
|
/// A status value which describes our current state, managed via atomics.
|
|
/// Three possibilities:
|
|
/// 0: no changed topics, no thread is waiting.
|
|
/// 128: no changed topics, some thread is waiting and needs wakeup.
|
|
/// anything else: some changed topic, no thread is waiting.
|
|
/// Note that if the msb is set (status == 128) no other bit may be set.
|
|
using status_bits_t = uint8_t;
|
|
std::atomic<uint8_t> status_{};
|
|
|
|
/// Sentinel status value indicating that a thread is waiting and needs a wakeup.
|
|
/// Note it is an error for this bit to be set and also any topic bit.
|
|
static constexpr uint8_t STATUS_NEEDS_WAKEUP = 128;
|
|
|
|
/// Binary semaphore used to communicate changes.
|
|
/// If status_ is STATUS_NEEDS_WAKEUP, then a thread has commited to call wait() on our sema and
|
|
/// this must be balanced by the next call to post(). Note only one thread may wait at a time.
|
|
binary_semaphore_t sema_{};
|
|
|
|
/// Apply any pending updates to the data.
|
|
/// This accepts data because it must be locked.
|
|
/// \return the updated generation list.
|
|
generation_list_t updated_gens_in_data(acquired_lock<data_t> &data);
|
|
|
|
/// Given a list of input generations, attempt to update them to something newer.
|
|
/// If \p gens is older, then just return those by reference, and directly return false (not
|
|
/// becoming the reader).
|
|
/// If \p gens is current and there is not a reader, then do not update \p gens and return true,
|
|
/// indicating we should become the reader. Now it is our responsibility to wait on the
|
|
/// semaphore and notify on a change via the condition variable. If \p gens is current, and
|
|
/// there is already a reader, then wait until the reader notifies us and try again.
|
|
bool try_update_gens_maybe_becoming_reader(generation_list_t *gens);
|
|
|
|
/// Wait for some entry in the list of generations to change.
|
|
/// \return the new gens.
|
|
generation_list_t await_gens(const generation_list_t &input_gens);
|
|
|
|
/// \return the current generation list, opportunistically applying any pending updates.
|
|
generation_list_t updated_gens();
|
|
|
|
/// Helper to convert a topic to a bitmask containing just that topic.
|
|
static topic_bitmask_t topic_to_bit(topic_t t) { return 1 << static_cast<topic_bitmask_t>(t); }
|
|
|
|
public:
|
|
topic_monitor_t();
|
|
~topic_monitor_t();
|
|
|
|
/// topic_monitors should not be copied, and there should be no reason to move one.
|
|
void operator=(const topic_monitor_t &) = delete;
|
|
topic_monitor_t(const topic_monitor_t &) = delete;
|
|
void operator=(topic_monitor_t &&) = delete;
|
|
topic_monitor_t(topic_monitor_t &&) = delete;
|
|
|
|
/// The principal topic_monitor. This may be fetched from a signal handler.
|
|
static topic_monitor_t &principal();
|
|
|
|
/// Post to a topic, potentially from a signal handler.
|
|
void post(topic_t topic);
|
|
|
|
/// Access the current generations.
|
|
generation_list_t current_generations() { return updated_gens(); }
|
|
|
|
/// Access the generation for a topic.
|
|
generation_t generation_for_topic(topic_t topic) { return current_generations().at(topic); }
|
|
|
|
/// For each valid topic in \p gens, check to see if the current topic is larger than
|
|
/// the value in \p gens.
|
|
/// If \p wait is set, then wait if there are no changes; otherwise return immediately.
|
|
/// \return true if some topic changed, false if none did.
|
|
/// On a true return, this updates the generation list \p gens.
|
|
bool check(generation_list_t *gens, bool wait);
|
|
};
|
|
|
|
#endif
|