2016-05-03 20:02:26 +00:00
|
|
|
// The library for various signal related issues.
|
2016-05-18 22:30:21 +00:00
|
|
|
#include "config.h" // IWYU pragma: keep
|
|
|
|
|
2005-10-06 11:54:16 +00:00
|
|
|
#include <errno.h>
|
2006-07-30 20:26:59 +00:00
|
|
|
#ifdef HAVE_SIGINFO_H
|
|
|
|
#include <siginfo.h>
|
|
|
|
#endif
|
2022-08-21 06:14:48 +00:00
|
|
|
#include <unistd.h>
|
2006-07-30 20:26:59 +00:00
|
|
|
|
2021-09-22 00:57:25 +00:00
|
|
|
#include <csignal>
|
2022-08-21 06:14:48 +00:00
|
|
|
#include <cwchar>
|
|
|
|
#include <mutex>
|
2021-09-22 00:57:25 +00:00
|
|
|
|
2005-10-06 07:30:50 +00:00
|
|
|
#include "common.h"
|
|
|
|
#include "event.h"
|
2016-05-03 20:02:26 +00:00
|
|
|
#include "fallback.h" // IWYU pragma: keep
|
2022-08-21 06:14:48 +00:00
|
|
|
#include "global_safety.h"
|
2016-05-03 20:02:26 +00:00
|
|
|
#include "reader.h"
|
2023-01-14 22:56:24 +00:00
|
|
|
#include "signals.h"
|
2020-06-07 23:05:52 +00:00
|
|
|
#include "termsize.h"
|
Introduce topic monitoring
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.
2019-02-02 23:39:04 +00:00
|
|
|
#include "topic_monitor.h"
|
2016-05-03 20:02:26 +00:00
|
|
|
#include "wutil.h" // IWYU pragma: keep
|
2006-07-19 22:55:49 +00:00
|
|
|
|
2023-04-30 22:40:06 +00:00
|
|
|
extern "C" {
|
|
|
|
void get_signals_with_handlers_ffi(sigset_t *set);
|
2017-05-03 03:55:41 +00:00
|
|
|
}
|
2023-04-30 22:40:06 +00:00
|
|
|
void get_signals_with_handlers(sigset_t *set) { get_signals_with_handlers_ffi(set); }
|
2019-05-26 02:05:24 +00:00
|
|
|
|
2020-08-07 02:00:53 +00:00
|
|
|
sigchecker_t::sigchecker_t(topic_t signal) : topic_(signal) {
|
2019-05-26 02:05:24 +00:00
|
|
|
// Call check() to update our generation.
|
|
|
|
check();
|
|
|
|
}
|
|
|
|
|
2020-06-20 16:23:36 +00:00
|
|
|
bool sigchecker_t::check() {
|
2023-01-14 22:56:24 +00:00
|
|
|
auto &tm = topic_monitor_principal();
|
2020-06-20 16:23:36 +00:00
|
|
|
generation_t gen = tm.generation_for_topic(topic_);
|
2019-05-26 02:05:24 +00:00
|
|
|
bool changed = this->gen_ != gen;
|
|
|
|
this->gen_ = gen;
|
|
|
|
return changed;
|
|
|
|
}
|
2019-05-26 02:19:03 +00:00
|
|
|
|
2020-06-20 16:23:36 +00:00
|
|
|
void sigchecker_t::wait() const {
|
2023-01-14 22:56:24 +00:00
|
|
|
auto &tm = topic_monitor_principal();
|
|
|
|
generation_list_t gens = invalid_generations();
|
|
|
|
gens.at_mut(topic_) = this->gen_;
|
2020-08-07 02:00:53 +00:00
|
|
|
tm.check(&gens, true /* wait */);
|
2019-05-26 02:19:03 +00:00
|
|
|
}
|