mirror of
https://github.com/fish-shell/fish-shell
synced 2025-01-08 11:08:53 +00:00
4e03d3c264
This fixes a race condition in the topic monitor. A thread may decide to enter the wait queue, but before it does the generation list changes, and so our thread will wait forever, resulting in a hang. It also simplifies the implementation of the topic monitor considerably; on reflection the whole "metagen" thing isn't providing any value and we should just compare generations directly. In the new design, we have a lock-protected list of current generations, along with a boolean as to whether someone is reading from the pipe. The reader (only one at a time) is responsible for broadcasting notifications via a condition variable.
150 lines
5.9 KiB
C++
150 lines
5.9 KiB
C++
#ifndef FISH_TOPIC_MONITOR_H
|
|
#define FISH_TOPIC_MONITOR_H
|
|
|
|
#include "common.h"
|
|
#include "enum_set.h"
|
|
#include "io.h"
|
|
|
|
#include <array>
|
|
#include <atomic>
|
|
#include <bitset>
|
|
#include <condition_variable>
|
|
#include <limits>
|
|
#include <numeric>
|
|
|
|
/** 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.
|
|
*/
|
|
|
|
/// The list of topics that may be observed.
|
|
enum class topic_t : uint8_t {
|
|
sigchld, // Corresponds to SIGCHLD signal.
|
|
sighupint, // Corresponds to both SIGHUP and SIGINT signals.
|
|
internal_exit, // Corresponds to an internal process exit.
|
|
COUNT
|
|
};
|
|
|
|
/// Allow enum_iter to be used.
|
|
template <>
|
|
struct enum_info_t<topic_t> {
|
|
static constexpr auto count = topic_t::COUNT;
|
|
};
|
|
|
|
/// Set of topics.
|
|
using topic_set_t = enum_set_t<topic_t>;
|
|
|
|
/// Counting iterator for topics.
|
|
using topic_iter_t = enum_iter_t<topic_t>;
|
|
|
|
/// 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 is guaranteed to never be set and be larger than any valid generation.
|
|
constexpr generation_t invalid_generation = std::numeric_limits<generation_t>::max();
|
|
|
|
/// List of generation values, indexed by topics.
|
|
/// The initial value of a generation is always 0.
|
|
using generation_list_t = enum_array_t<generation_t, topic_t>;
|
|
|
|
/// Teh topic monitor class. This permits querying the current generation values for topics,
|
|
/// optionally blocking until they increase.
|
|
class topic_monitor_t {
|
|
private:
|
|
using topic_set_raw_t = uint8_t;
|
|
|
|
static_assert(sizeof(topic_set_raw_t) * CHAR_BIT >= enum_count<topic_t>(),
|
|
"topic_set_raw is too small");
|
|
|
|
// Some stuff that needs to be protected by the same lock.
|
|
struct data_t {
|
|
/// The current generation list.
|
|
generation_list_t current_gens{};
|
|
|
|
/// Whether there is a thread currently reading from the notifier pipe.
|
|
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_{};
|
|
|
|
/// The set of topics which have pending increments.
|
|
/// This is managed via atomics.
|
|
std::atomic<topic_set_raw_t> pending_updates_{};
|
|
|
|
/// Self-pipes used to communicate changes.
|
|
/// The writer is a signal handler.
|
|
/// "The reader" refers to a thread that wants to wait for changes. Only one thread can be the
|
|
/// reader at a given time.
|
|
autoclose_pipes_t pipes_;
|
|
|
|
/// 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 read from the pipes
|
|
/// 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();
|
|
|
|
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); }
|
|
|
|
/// See if for any topic (specified in \p topics) has changed from the values in the generation
|
|
/// list \p gens. If \p wait is set, then wait if there are no changes; otherwise return
|
|
/// immediately.
|
|
/// \return the set of topics that changed, updating the generation list \p gens.
|
|
topic_set_t check(generation_list_t *gens, topic_set_t topics, bool wait);
|
|
};
|
|
|
|
#endif
|