2016-04-29 10:19:50 +08:00
|
|
|
// Functions for handling event triggers.
|
2016-05-19 06:30:21 +08:00
|
|
|
#include "config.h" // IWYU pragma: keep
|
|
|
|
|
2016-04-29 10:19:50 +08:00
|
|
|
#include <signal.h>
|
2017-02-13 12:24:22 +08:00
|
|
|
#include <stddef.h>
|
2016-04-21 14:00:54 +08:00
|
|
|
#include <unistd.h>
|
2017-02-11 10:47:02 +08:00
|
|
|
|
2016-04-21 14:00:54 +08:00
|
|
|
#include <algorithm>
|
2017-02-11 10:47:02 +08:00
|
|
|
#include <functional>
|
2016-04-29 10:19:50 +08:00
|
|
|
#include <memory>
|
2015-07-25 23:14:25 +08:00
|
|
|
#include <string>
|
2017-02-11 10:47:02 +08:00
|
|
|
#include <type_traits>
|
2005-10-06 06:37:08 +08:00
|
|
|
|
|
|
|
#include "common.h"
|
|
|
|
#include "event.h"
|
2016-04-29 10:19:50 +08:00
|
|
|
#include "fallback.h" // IWYU pragma: keep
|
|
|
|
#include "input_common.h"
|
2015-07-25 23:14:25 +08:00
|
|
|
#include "io.h"
|
2016-04-29 10:19:50 +08:00
|
|
|
#include "parser.h"
|
|
|
|
#include "proc.h"
|
|
|
|
#include "signal.h"
|
|
|
|
#include "wutil.h" // IWYU pragma: keep
|
2006-07-20 06:55:49 +08:00
|
|
|
|
2016-04-29 10:19:50 +08:00
|
|
|
/// Number of signals that can be queued before an overflow occurs.
|
2005-10-06 06:37:08 +08:00
|
|
|
#define SIG_UNHANDLED_MAX 64
|
|
|
|
|
2016-04-29 10:19:50 +08:00
|
|
|
/// This struct contains a list of generated signals waiting to be dispatched.
|
|
|
|
typedef struct {
|
|
|
|
/// Number of delivered signals.
|
2012-12-23 23:37:43 +08:00
|
|
|
volatile int count;
|
2016-04-29 10:19:50 +08:00
|
|
|
/// Whether signals have been skipped.
|
2012-12-23 23:37:43 +08:00
|
|
|
volatile int overflow;
|
2016-04-29 10:19:50 +08:00
|
|
|
/// Array of signal events.
|
2012-12-23 23:37:43 +08:00
|
|
|
volatile int signal[SIG_UNHANDLED_MAX];
|
2016-04-29 10:19:50 +08:00
|
|
|
} signal_list_t;
|
2005-10-06 06:37:08 +08:00
|
|
|
|
2016-04-29 10:19:50 +08:00
|
|
|
/// The signal event list. Actually two separate lists. One which is active, which is the one that
|
|
|
|
/// new events is written to. The inactive one contains the events that are currently beeing
|
|
|
|
/// performed.
|
|
|
|
static signal_list_t sig_list[2] = {{}, {}};
|
2005-10-06 06:37:08 +08:00
|
|
|
|
2016-04-29 10:19:50 +08:00
|
|
|
/// The index of sig_list that is the list of signals currently written to.
|
|
|
|
static volatile int active_list = 0;
|
2005-10-06 06:37:08 +08:00
|
|
|
|
2017-01-22 08:48:07 +08:00
|
|
|
typedef std::vector<shared_ptr<event_t>> event_list_t;
|
2011-12-27 14:27:58 +08:00
|
|
|
|
2016-04-29 10:19:50 +08:00
|
|
|
/// List of event handlers.
|
2014-12-29 16:34:36 +08:00
|
|
|
static event_list_t s_event_handlers;
|
2005-10-06 06:37:08 +08:00
|
|
|
|
2016-04-29 10:19:50 +08:00
|
|
|
/// List of events that have been sent but have not yet been delivered because they are blocked.
|
2011-12-27 14:27:58 +08:00
|
|
|
static event_list_t blocked;
|
2005-12-12 06:21:01 +08:00
|
|
|
|
2016-04-29 10:19:50 +08:00
|
|
|
/// Variables (one per signal) set when a signal is observed. This is inspected by a signal handler.
|
2014-12-29 16:34:36 +08:00
|
|
|
static volatile bool s_observed_signals[NSIG] = {};
|
2016-04-29 10:19:50 +08:00
|
|
|
static void set_signal_observed(int sig, bool val) {
|
2014-12-29 16:34:36 +08:00
|
|
|
ASSERT_IS_MAIN_THREAD();
|
2016-04-29 10:19:50 +08:00
|
|
|
if (sig >= 0 && (size_t)sig < sizeof s_observed_signals / sizeof *s_observed_signals) {
|
2014-12-29 16:34:36 +08:00
|
|
|
s_observed_signals[sig] = val;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-04-29 10:19:50 +08:00
|
|
|
/// Tests if one event instance matches the definition of a event class. If both the class and the
|
|
|
|
/// instance name a function, they must name the same function.
|
|
|
|
static int event_match(const event_t &classv, const event_t &instance) {
|
|
|
|
// If the function names are both non-empty and different, then it's not a match.
|
|
|
|
if (!classv.function_name.empty() && !instance.function_name.empty() &&
|
|
|
|
classv.function_name != instance.function_name) {
|
2012-03-06 06:18:16 +08:00
|
|
|
return 0;
|
2012-11-19 08:30:30 +08:00
|
|
|
}
|
2012-11-18 18:23:22 +08:00
|
|
|
|
2016-04-29 10:19:50 +08:00
|
|
|
if (classv.type == EVENT_ANY) return 1;
|
|
|
|
if (classv.type != instance.type) return 0;
|
2012-11-18 18:23:22 +08:00
|
|
|
|
2016-04-29 10:19:50 +08:00
|
|
|
switch (classv.type) {
|
|
|
|
case EVENT_SIGNAL: {
|
|
|
|
if (classv.param1.signal == EVENT_ANY_SIGNAL) return 1;
|
2012-12-20 18:48:36 +08:00
|
|
|
return classv.param1.signal == instance.param1.signal;
|
2016-04-29 10:19:50 +08:00
|
|
|
}
|
|
|
|
case EVENT_VARIABLE: {
|
2012-12-20 18:48:36 +08:00
|
|
|
return instance.str_param1 == classv.str_param1;
|
2016-04-29 10:19:50 +08:00
|
|
|
}
|
|
|
|
case EVENT_EXIT: {
|
|
|
|
if (classv.param1.pid == EVENT_ANY_PID) return 1;
|
2012-12-20 18:48:36 +08:00
|
|
|
return classv.param1.pid == instance.param1.pid;
|
2016-04-29 10:19:50 +08:00
|
|
|
}
|
|
|
|
case EVENT_JOB_ID: {
|
2012-12-20 18:48:36 +08:00
|
|
|
return classv.param1.job_id == instance.param1.job_id;
|
2016-04-29 10:19:50 +08:00
|
|
|
}
|
|
|
|
case EVENT_GENERIC: {
|
2012-12-20 18:48:36 +08:00
|
|
|
return instance.str_param1 == classv.str_param1;
|
2016-04-29 10:19:50 +08:00
|
|
|
}
|
2016-10-30 08:25:48 +08:00
|
|
|
default: {
|
|
|
|
DIE("unexpected classv.type");
|
|
|
|
break;
|
|
|
|
}
|
2012-11-19 08:30:30 +08:00
|
|
|
}
|
2012-11-18 18:23:22 +08:00
|
|
|
|
2016-04-29 10:19:50 +08:00
|
|
|
// This should never be reached.
|
2012-12-19 22:56:19 +08:00
|
|
|
debug(0, "Warning: Unreachable code reached in event_match in event.cpp\n");
|
2012-11-19 08:30:30 +08:00
|
|
|
return 0;
|
2005-10-06 06:37:08 +08:00
|
|
|
}
|
|
|
|
|
2016-04-29 10:19:50 +08:00
|
|
|
/// Test if specified event is blocked.
|
|
|
|
static int event_is_blocked(const event_t &e) {
|
2013-12-21 09:41:21 +08:00
|
|
|
const block_t *block;
|
2012-11-19 08:30:30 +08:00
|
|
|
parser_t &parser = parser_t::principal_parser();
|
2014-01-15 17:40:40 +08:00
|
|
|
|
2013-12-21 09:41:21 +08:00
|
|
|
size_t idx = 0;
|
2016-04-29 10:19:50 +08:00
|
|
|
while ((block = parser.block_at_index(idx++))) {
|
|
|
|
if (event_block_list_blocks_type(block->event_blocks, e.type)) return true;
|
2012-11-19 08:30:30 +08:00
|
|
|
}
|
2012-12-20 18:48:36 +08:00
|
|
|
return event_block_list_blocks_type(parser.global_event_blocks, e.type);
|
2005-12-12 06:21:01 +08:00
|
|
|
}
|
|
|
|
|
2016-04-29 10:19:50 +08:00
|
|
|
wcstring event_get_desc(const event_t &e) {
|
2012-11-19 08:30:30 +08:00
|
|
|
wcstring result;
|
2016-04-29 10:19:50 +08:00
|
|
|
switch (e.type) {
|
|
|
|
case EVENT_SIGNAL: {
|
|
|
|
result = format_string(_(L"signal handler for %ls (%ls)"), sig2wcs(e.param1.signal),
|
|
|
|
signal_get_desc(e.param1.signal));
|
2012-11-19 16:31:03 +08:00
|
|
|
break;
|
2016-04-29 10:19:50 +08:00
|
|
|
}
|
|
|
|
case EVENT_VARIABLE: {
|
2012-12-20 18:48:36 +08:00
|
|
|
result = format_string(_(L"handler for variable '%ls'"), e.str_param1.c_str());
|
2012-11-19 16:31:03 +08:00
|
|
|
break;
|
2016-04-29 10:19:50 +08:00
|
|
|
}
|
|
|
|
case EVENT_EXIT: {
|
|
|
|
if (e.param1.pid > 0) {
|
2012-12-20 18:48:36 +08:00
|
|
|
result = format_string(_(L"exit handler for process %d"), e.param1.pid);
|
2016-04-29 10:19:50 +08:00
|
|
|
} else {
|
2017-04-23 21:58:26 +08:00
|
|
|
// In events, PGIDs are stored as negative PIDs
|
2012-12-20 18:48:36 +08:00
|
|
|
job_t *j = job_get_from_pid(-e.param1.pid);
|
2012-11-19 16:31:03 +08:00
|
|
|
if (j)
|
2016-04-29 10:19:50 +08:00
|
|
|
result = format_string(_(L"exit handler for job %d, '%ls'"), j->job_id,
|
|
|
|
j->command_wcstr());
|
2012-11-19 16:31:03 +08:00
|
|
|
else
|
2016-04-29 10:19:50 +08:00
|
|
|
result = format_string(_(L"exit handler for job with process group %d"),
|
|
|
|
-e.param1.pid);
|
2012-11-19 16:31:03 +08:00
|
|
|
}
|
|
|
|
break;
|
2016-04-29 10:19:50 +08:00
|
|
|
}
|
|
|
|
case EVENT_JOB_ID: {
|
2012-12-20 18:48:36 +08:00
|
|
|
job_t *j = job_get(e.param1.job_id);
|
2016-04-29 10:19:50 +08:00
|
|
|
if (j) {
|
|
|
|
result = format_string(_(L"exit handler for job %d, '%ls'"), j->job_id,
|
|
|
|
j->command_wcstr());
|
|
|
|
} else {
|
2012-12-20 18:48:36 +08:00
|
|
|
result = format_string(_(L"exit handler for job with job id %d"), e.param1.job_id);
|
2016-04-29 10:19:50 +08:00
|
|
|
}
|
2012-11-19 16:31:03 +08:00
|
|
|
break;
|
|
|
|
}
|
2016-04-29 10:19:50 +08:00
|
|
|
case EVENT_GENERIC: {
|
2012-12-20 18:48:36 +08:00
|
|
|
result = format_string(_(L"handler for generic event '%ls'"), e.str_param1.c_str());
|
2012-11-19 16:31:03 +08:00
|
|
|
break;
|
2016-04-29 10:19:50 +08:00
|
|
|
}
|
|
|
|
default: {
|
2012-12-23 04:21:31 +08:00
|
|
|
result = format_string(_(L"Unknown event type '0x%x'"), e.type);
|
2012-11-19 16:31:03 +08:00
|
|
|
break;
|
2016-04-29 10:19:50 +08:00
|
|
|
}
|
2012-11-19 08:30:30 +08:00
|
|
|
}
|
2012-11-18 18:23:22 +08:00
|
|
|
|
2012-11-19 08:30:30 +08:00
|
|
|
return result;
|
2006-02-01 23:49:11 +08:00
|
|
|
}
|
|
|
|
|
2012-02-18 07:58:02 +08:00
|
|
|
#if 0
|
2016-04-29 10:19:50 +08:00
|
|
|
static void show_all_handlers(void) {
|
2017-01-14 12:34:15 +08:00
|
|
|
fwprintf(stdout, L"event handlers:\n");
|
2016-04-29 10:19:50 +08:00
|
|
|
for (event_list_t::const_iterator iter = events.begin(); iter != events.end(); ++iter) {
|
2012-02-18 03:36:49 +08:00
|
|
|
const event_t *foo = *iter;
|
|
|
|
wcstring tmp = event_get_desc(foo);
|
2017-01-14 12:34:15 +08:00
|
|
|
fwprintf(stdout, L" handler now %ls\n", tmp.c_str());
|
2012-02-18 03:36:49 +08:00
|
|
|
}
|
|
|
|
}
|
2012-02-18 07:58:02 +08:00
|
|
|
#endif
|
2005-12-12 06:21:01 +08:00
|
|
|
|
2016-04-29 10:19:50 +08:00
|
|
|
/// Give a more condensed description of \c event compared to \c event_get_desc. It includes what
|
|
|
|
/// function will fire if the \c event is an event handler.
|
|
|
|
static wcstring event_desc_compact(const event_t &event) {
|
2012-12-23 04:21:31 +08:00
|
|
|
wcstring res;
|
|
|
|
wchar_t const *temp;
|
|
|
|
int sig;
|
2016-04-29 10:19:50 +08:00
|
|
|
switch (event.type) {
|
|
|
|
case EVENT_ANY: {
|
2012-12-23 04:21:31 +08:00
|
|
|
res = L"EVENT_ANY";
|
|
|
|
break;
|
2016-04-29 10:19:50 +08:00
|
|
|
}
|
|
|
|
case EVENT_VARIABLE: {
|
|
|
|
if (event.str_param1.c_str()) {
|
2012-12-23 04:21:31 +08:00
|
|
|
res = format_string(L"EVENT_VARIABLE($%ls)", event.str_param1.c_str());
|
2016-04-29 10:19:50 +08:00
|
|
|
} else {
|
2012-12-23 04:21:31 +08:00
|
|
|
res = L"EVENT_VARIABLE([any])";
|
|
|
|
}
|
|
|
|
break;
|
2016-04-29 10:19:50 +08:00
|
|
|
}
|
|
|
|
case EVENT_SIGNAL: {
|
2012-12-23 04:21:31 +08:00
|
|
|
sig = event.param1.signal;
|
2016-04-29 10:19:50 +08:00
|
|
|
if (sig == EVENT_ANY_SIGNAL) {
|
2012-12-23 04:21:31 +08:00
|
|
|
temp = L"[all signals]";
|
2016-04-29 10:19:50 +08:00
|
|
|
} else if (sig == 0) {
|
2012-12-23 04:21:31 +08:00
|
|
|
temp = L"not set";
|
2016-04-29 10:19:50 +08:00
|
|
|
} else {
|
2012-12-23 04:21:31 +08:00
|
|
|
temp = sig2wcs(sig);
|
|
|
|
}
|
|
|
|
res = format_string(L"EVENT_SIGNAL(%ls)", temp);
|
|
|
|
break;
|
2016-04-29 10:19:50 +08:00
|
|
|
}
|
|
|
|
case EVENT_EXIT: {
|
|
|
|
if (event.param1.pid == EVENT_ANY_PID) {
|
2012-12-23 04:21:31 +08:00
|
|
|
res = wcstring(L"EVENT_EXIT([all child processes])");
|
2016-04-29 10:19:50 +08:00
|
|
|
} else if (event.param1.pid > 0) {
|
2012-12-23 04:21:31 +08:00
|
|
|
res = format_string(L"EVENT_EXIT(pid %d)", event.param1.pid);
|
2016-04-29 10:19:50 +08:00
|
|
|
} else {
|
2017-04-23 21:58:26 +08:00
|
|
|
// In events, PGIDs are stored as negative PIDs
|
2012-12-23 04:21:31 +08:00
|
|
|
job_t *j = job_get_from_pid(-event.param1.pid);
|
|
|
|
if (j)
|
2016-04-29 10:19:50 +08:00
|
|
|
res = format_string(L"EVENT_EXIT(jobid %d: \"%ls\")", j->job_id,
|
|
|
|
j->command_wcstr());
|
2012-12-23 04:21:31 +08:00
|
|
|
else
|
|
|
|
res = format_string(L"EVENT_EXIT(pgid %d)", -event.param1.pid);
|
|
|
|
}
|
|
|
|
break;
|
2016-04-29 10:19:50 +08:00
|
|
|
}
|
|
|
|
case EVENT_JOB_ID: {
|
2012-12-23 04:21:31 +08:00
|
|
|
job_t *j = job_get(event.param1.job_id);
|
|
|
|
if (j)
|
2016-04-29 10:19:50 +08:00
|
|
|
res =
|
|
|
|
format_string(L"EVENT_JOB_ID(job %d: \"%ls\")", j->job_id, j->command_wcstr());
|
2012-12-23 04:21:31 +08:00
|
|
|
else
|
|
|
|
res = format_string(L"EVENT_JOB_ID(jobid %d)", event.param1.job_id);
|
|
|
|
break;
|
|
|
|
}
|
2016-04-29 10:19:50 +08:00
|
|
|
case EVENT_GENERIC: {
|
2012-12-23 04:21:31 +08:00
|
|
|
res = format_string(L"EVENT_GENERIC(%ls)", event.str_param1.c_str());
|
|
|
|
break;
|
2016-04-29 10:19:50 +08:00
|
|
|
}
|
2016-11-02 13:36:30 +08:00
|
|
|
default: {
|
|
|
|
res = format_string(L"unknown/illegal event(%x)", event.type);
|
|
|
|
break;
|
|
|
|
}
|
2012-12-23 04:21:31 +08:00
|
|
|
}
|
2016-04-29 10:19:50 +08:00
|
|
|
if (event.function_name.size()) {
|
2012-12-23 04:21:31 +08:00
|
|
|
return format_string(L"%ls: \"%ls\"", res.c_str(), event.function_name.c_str());
|
|
|
|
}
|
2016-05-05 06:19:47 +08:00
|
|
|
return res;
|
2012-12-19 07:32:03 +08:00
|
|
|
}
|
|
|
|
|
2016-04-29 10:19:50 +08:00
|
|
|
void event_add_handler(const event_t &event) {
|
|
|
|
if (debug_level >= 3) {
|
2012-12-23 01:38:28 +08:00
|
|
|
wcstring desc = event_desc_compact(event);
|
2017-06-19 13:07:48 +08:00
|
|
|
debug(3, "register: %ls", desc.c_str());
|
2012-12-23 01:38:28 +08:00
|
|
|
}
|
2012-12-20 17:52:44 +08:00
|
|
|
|
2017-01-22 08:48:07 +08:00
|
|
|
shared_ptr<event_t> e = std::make_shared<event_t>(event);
|
2016-04-29 10:19:50 +08:00
|
|
|
if (e->type == EVENT_SIGNAL) {
|
2012-11-19 08:30:30 +08:00
|
|
|
signal_handle(e->param1.signal, 1);
|
2014-12-29 16:34:36 +08:00
|
|
|
set_signal_observed(e->param1.signal, true);
|
2012-11-19 08:30:30 +08:00
|
|
|
}
|
2005-10-06 06:37:08 +08:00
|
|
|
|
2017-01-22 08:48:07 +08:00
|
|
|
s_event_handlers.push_back(std::move(e));
|
2005-10-06 06:37:08 +08:00
|
|
|
}
|
|
|
|
|
2016-04-29 10:19:50 +08:00
|
|
|
void event_remove(const event_t &criterion) {
|
|
|
|
if (debug_level >= 3) {
|
2012-12-23 01:38:28 +08:00
|
|
|
wcstring desc = event_desc_compact(criterion);
|
2017-06-19 13:07:48 +08:00
|
|
|
debug(3, "unregister: %ls", desc.c_str());
|
2012-12-23 01:38:28 +08:00
|
|
|
}
|
2012-11-18 18:23:22 +08:00
|
|
|
|
2017-01-22 08:48:07 +08:00
|
|
|
event_list_t::iterator iter = s_event_handlers.begin();
|
|
|
|
while (iter != s_event_handlers.end()) {
|
|
|
|
const event_t *n = iter->get();
|
2017-01-27 12:00:43 +08:00
|
|
|
if (!event_match(criterion, *n)) {
|
2017-01-22 08:48:07 +08:00
|
|
|
++iter;
|
|
|
|
continue;
|
|
|
|
}
|
2012-11-18 18:23:22 +08:00
|
|
|
|
2017-01-22 08:48:07 +08:00
|
|
|
// If this event was a signal handler and no other handler handles the specified signal
|
|
|
|
// type, do not handle that type of signal any more.
|
|
|
|
if (n->type == EVENT_SIGNAL) {
|
|
|
|
event_t e = event_t::signal_event(n->param1.signal);
|
|
|
|
if (event_get(e, NULL) == 1) {
|
|
|
|
signal_handle(e.param1.signal, 0);
|
|
|
|
set_signal_observed(e.param1.signal, 0);
|
2012-11-19 08:30:30 +08:00
|
|
|
}
|
|
|
|
}
|
2017-01-22 08:48:07 +08:00
|
|
|
iter = s_event_handlers.erase(iter);
|
2012-11-18 18:23:22 +08:00
|
|
|
}
|
2005-10-06 06:37:08 +08:00
|
|
|
}
|
|
|
|
|
2017-01-22 08:48:07 +08:00
|
|
|
int event_get(const event_t &criterion, event_list_t *out) {
|
|
|
|
ASSERT_IS_MAIN_THREAD();
|
2012-11-19 08:30:30 +08:00
|
|
|
int found = 0;
|
2017-01-22 08:48:07 +08:00
|
|
|
for (const shared_ptr<event_t> &n : s_event_handlers) {
|
2016-04-29 10:19:50 +08:00
|
|
|
if (event_match(criterion, *n)) {
|
2012-11-19 08:30:30 +08:00
|
|
|
found++;
|
2016-04-29 10:19:50 +08:00
|
|
|
if (out) out->push_back(n);
|
2012-11-19 08:30:30 +08:00
|
|
|
}
|
2012-11-18 18:23:22 +08:00
|
|
|
}
|
2012-11-19 08:30:30 +08:00
|
|
|
return found;
|
2005-10-06 06:37:08 +08:00
|
|
|
}
|
|
|
|
|
2016-04-29 10:19:50 +08:00
|
|
|
bool event_is_signal_observed(int sig) {
|
|
|
|
// We are in a signal handler! Don't allocate memory, etc.
|
2014-12-29 16:34:36 +08:00
|
|
|
bool result = false;
|
2016-10-10 05:36:08 +08:00
|
|
|
if (sig >= 0 && (unsigned long)sig < sizeof(s_observed_signals) / sizeof(*s_observed_signals)) {
|
2014-12-29 16:34:36 +08:00
|
|
|
result = s_observed_signals[sig];
|
2012-06-05 05:20:01 +08:00
|
|
|
}
|
2014-12-29 16:34:36 +08:00
|
|
|
return result;
|
2012-06-05 05:20:01 +08:00
|
|
|
}
|
|
|
|
|
2016-04-29 10:19:50 +08:00
|
|
|
/// Perform the specified event. Since almost all event firings will not be matched by even a single
|
|
|
|
/// event handler, we make sure to optimize the 'no matches' path. This means that nothing is
|
|
|
|
/// allocated/initialized unless needed.
|
|
|
|
static void event_fire_internal(const event_t &event) {
|
2017-01-22 08:48:07 +08:00
|
|
|
// Iterate over all events, adding events that should be fired to a second list. We need
|
2016-04-29 10:19:50 +08:00
|
|
|
// to do this in a separate step since an event handler might call event_remove or
|
|
|
|
// event_add_handler, which will change the contents of the \c events list.
|
2017-01-22 08:48:07 +08:00
|
|
|
event_list_t fire;
|
|
|
|
for (shared_ptr<event_t> &criterion : s_event_handlers) {
|
2016-04-29 10:19:50 +08:00
|
|
|
// Check if this event is a match.
|
|
|
|
if (event_match(*criterion, event)) {
|
2011-12-27 14:51:34 +08:00
|
|
|
fire.push_back(criterion);
|
2012-11-19 08:30:30 +08:00
|
|
|
}
|
2012-11-18 18:23:22 +08:00
|
|
|
}
|
|
|
|
|
2016-04-29 10:19:50 +08:00
|
|
|
// No matches. Time to return.
|
|
|
|
if (fire.empty()) return;
|
2012-11-18 18:23:22 +08:00
|
|
|
|
2016-04-29 10:19:50 +08:00
|
|
|
if (signal_is_blocked()) {
|
|
|
|
// Fix for https://github.com/fish-shell/fish-shell/issues/608. Don't run event handlers
|
|
|
|
// while signals are blocked.
|
2017-01-27 12:00:43 +08:00
|
|
|
input_common_add_callback([event]() {
|
2017-01-22 09:15:45 +08:00
|
|
|
ASSERT_IS_MAIN_THREAD();
|
|
|
|
event_fire(&event);
|
|
|
|
});
|
2013-04-04 08:26:02 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2016-04-29 10:19:50 +08:00
|
|
|
// Iterate over our list of matching events.
|
2017-01-22 08:48:07 +08:00
|
|
|
for (shared_ptr<event_t> &criterion : fire) {
|
|
|
|
// Only fire if this event is still present
|
2018-08-12 10:55:06 +08:00
|
|
|
if (!contains(s_event_handlers, criterion)) {
|
2017-01-22 08:48:07 +08:00
|
|
|
continue;
|
|
|
|
}
|
2012-11-19 08:30:30 +08:00
|
|
|
|
2016-04-29 10:19:50 +08:00
|
|
|
// Fire event.
|
2012-11-19 08:30:30 +08:00
|
|
|
wcstring buffer = criterion->function_name;
|
2012-11-18 18:23:22 +08:00
|
|
|
|
2016-04-29 10:19:50 +08:00
|
|
|
for (size_t j = 0; j < event.arguments.size(); j++) {
|
2012-12-23 04:40:34 +08:00
|
|
|
wcstring arg_esc = escape_string(event.arguments.at(j), 1);
|
|
|
|
buffer += L" ";
|
|
|
|
buffer += arg_esc;
|
2011-12-27 16:06:07 +08:00
|
|
|
}
|
2005-10-12 03:23:43 +08:00
|
|
|
|
2012-12-23 04:21:31 +08:00
|
|
|
// debug( 1, L"Event handler fires command '%ls'", buffer.c_str() );
|
2012-11-18 18:23:22 +08:00
|
|
|
|
2016-04-29 10:19:50 +08:00
|
|
|
// Event handlers are not part of the main flow of code, so they are marked as
|
|
|
|
// non-interactive.
|
2012-11-19 08:30:30 +08:00
|
|
|
proc_push_interactive(0);
|
2017-01-22 08:48:07 +08:00
|
|
|
int prev_status = proc_get_last_status();
|
2012-01-23 13:40:08 +08:00
|
|
|
parser_t &parser = parser_t::principal_parser();
|
2012-11-18 18:23:22 +08:00
|
|
|
|
2017-01-22 07:35:35 +08:00
|
|
|
event_block_t *b = parser.push_block<event_block_t>(event);
|
2012-11-19 08:30:30 +08:00
|
|
|
parser.eval(buffer, io_chain_t(), TOP);
|
2017-01-22 07:35:35 +08:00
|
|
|
parser.pop_block(b);
|
2012-11-19 08:30:30 +08:00
|
|
|
proc_pop_interactive();
|
|
|
|
proc_set_last_status(prev_status);
|
|
|
|
}
|
2005-10-06 06:37:08 +08:00
|
|
|
}
|
|
|
|
|
2016-04-29 10:19:50 +08:00
|
|
|
/// Handle all pending signal events.
|
|
|
|
static void event_fire_delayed() {
|
|
|
|
// If is_event is one, we are running the event-handler non-recursively.
|
|
|
|
//
|
|
|
|
// When the event handler has called a piece of code that triggers another event, we do not want
|
|
|
|
// to fire delayed events because of concurrency problems.
|
|
|
|
if (!blocked.empty() && is_event == 1) {
|
2017-01-22 08:48:07 +08:00
|
|
|
event_list_t local_blocked;
|
|
|
|
local_blocked.swap(blocked);
|
|
|
|
for (const shared_ptr<event_t> &e : local_blocked) {
|
2016-04-29 10:19:50 +08:00
|
|
|
if (event_is_blocked(*e)) {
|
2017-01-22 08:48:07 +08:00
|
|
|
blocked.push_back(e);
|
2016-04-29 10:19:50 +08:00
|
|
|
} else {
|
2012-12-20 18:48:36 +08:00
|
|
|
event_fire_internal(*e);
|
2012-11-19 08:30:30 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2012-11-18 18:23:22 +08:00
|
|
|
|
2012-12-23 23:37:43 +08:00
|
|
|
int al = active_list;
|
|
|
|
|
2016-04-29 10:19:50 +08:00
|
|
|
while (sig_list[al].count > 0) {
|
2012-11-19 08:30:30 +08:00
|
|
|
signal_list_t *lst;
|
|
|
|
|
2016-04-29 10:19:50 +08:00
|
|
|
// Switch signal lists.
|
|
|
|
sig_list[1 - al].count = 0;
|
|
|
|
sig_list[1 - al].overflow = 0;
|
|
|
|
al = 1 - al;
|
|
|
|
active_list = al;
|
|
|
|
|
|
|
|
// Set up.
|
|
|
|
lst = &sig_list[1 - al];
|
|
|
|
if (lst->overflow) {
|
2012-11-19 08:30:30 +08:00
|
|
|
debug(0, _(L"Signal list overflow. Signals have been ignored."));
|
|
|
|
}
|
2012-11-18 18:23:22 +08:00
|
|
|
|
2016-04-29 10:19:50 +08:00
|
|
|
// Send all signals in our private list.
|
|
|
|
for (int i = 0; i < lst->count; i++) {
|
2017-01-22 08:48:07 +08:00
|
|
|
shared_ptr<event_t> e = std::make_shared<event_t>(EVENT_SIGNAL);
|
|
|
|
int signal = lst->signal[i];
|
|
|
|
e->param1.signal = signal;
|
|
|
|
e->arguments.push_back(sig2wcs(signal));
|
|
|
|
if (event_is_blocked(*e)) {
|
|
|
|
blocked.push_back(e);
|
2016-04-29 10:19:50 +08:00
|
|
|
} else {
|
2017-01-22 08:48:07 +08:00
|
|
|
event_fire_internal(*e);
|
2012-11-19 08:30:30 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2005-10-06 06:37:08 +08:00
|
|
|
}
|
|
|
|
|
2016-04-29 10:19:50 +08:00
|
|
|
void event_fire_signal(int signal) {
|
|
|
|
// This means we are in a signal handler. We must be very careful not do do anything that could
|
|
|
|
// cause a memory allocation or something else that might be bad when in a signal handler.
|
2012-11-19 08:30:30 +08:00
|
|
|
if (sig_list[active_list].count < SIG_UNHANDLED_MAX)
|
2016-04-29 10:19:50 +08:00
|
|
|
sig_list[active_list].signal[sig_list[active_list].count++] = signal;
|
2012-06-05 05:20:01 +08:00
|
|
|
else
|
2016-04-29 10:19:50 +08:00
|
|
|
sig_list[active_list].overflow = 1;
|
2012-06-05 05:20:01 +08:00
|
|
|
}
|
|
|
|
|
2016-04-29 10:19:50 +08:00
|
|
|
void event_fire(const event_t *event) {
|
|
|
|
if (event && event->type == EVENT_SIGNAL) {
|
2012-06-05 05:20:01 +08:00
|
|
|
event_fire_signal(event->param1.signal);
|
2016-04-29 10:19:50 +08:00
|
|
|
} else {
|
2012-06-05 05:20:01 +08:00
|
|
|
is_event++;
|
|
|
|
|
2016-04-29 10:19:50 +08:00
|
|
|
// Fire events triggered by signals.
|
2012-11-19 08:30:30 +08:00
|
|
|
event_fire_delayed();
|
2012-11-18 18:23:22 +08:00
|
|
|
|
2016-04-29 10:19:50 +08:00
|
|
|
if (event) {
|
|
|
|
if (event_is_blocked(*event)) {
|
2017-01-22 08:48:07 +08:00
|
|
|
blocked.push_back(std::make_shared<event_t>(*event));
|
2016-04-29 10:19:50 +08:00
|
|
|
} else {
|
2012-12-20 18:48:36 +08:00
|
|
|
event_fire_internal(*event);
|
2012-11-19 08:30:30 +08:00
|
|
|
}
|
|
|
|
}
|
2012-06-05 05:20:01 +08:00
|
|
|
is_event--;
|
2017-06-20 12:05:34 +08:00
|
|
|
assert(is_event >= 0);
|
2012-11-19 08:30:30 +08:00
|
|
|
}
|
2005-10-06 06:37:08 +08:00
|
|
|
}
|
|
|
|
|
2018-03-10 18:08:33 +08:00
|
|
|
/// Mapping between event type to name.
|
|
|
|
/// Note we don't bother to sort this.
|
|
|
|
struct event_type_name_t {
|
|
|
|
event_type_t type;
|
|
|
|
const wchar_t *name;
|
|
|
|
};
|
2018-02-01 01:47:23 +08:00
|
|
|
|
2018-03-10 18:08:33 +08:00
|
|
|
static const event_type_name_t events_mapping[] = {
|
|
|
|
{EVENT_SIGNAL, L"signal"},
|
|
|
|
{EVENT_VARIABLE, L"variable"},
|
|
|
|
{EVENT_EXIT, L"exit"},
|
|
|
|
{EVENT_JOB_ID, L"job-id"},
|
|
|
|
{EVENT_GENERIC, L"generic"}
|
|
|
|
};
|
|
|
|
|
|
|
|
maybe_t<event_type_t> event_type_for_name(const wcstring &name) {
|
|
|
|
for (const auto &em : events_mapping) {
|
|
|
|
if (name == em.name) {
|
|
|
|
return em.type;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return none();
|
2018-02-01 01:47:23 +08:00
|
|
|
}
|
|
|
|
|
2018-03-10 18:08:33 +08:00
|
|
|
static const wchar_t *event_name_for_type(event_type_t type) {
|
|
|
|
for (const auto &em : events_mapping) {
|
|
|
|
if (type == em.type) {
|
|
|
|
return em.name;
|
|
|
|
}
|
2018-02-01 01:47:23 +08:00
|
|
|
}
|
2018-03-10 18:08:33 +08:00
|
|
|
return L"";
|
2018-02-01 01:47:23 +08:00
|
|
|
}
|
|
|
|
|
2018-03-10 18:08:33 +08:00
|
|
|
|
|
|
|
void event_print(io_streams_t &streams, maybe_t<event_type_t> type_filter) {
|
2018-02-01 01:47:23 +08:00
|
|
|
std::vector<shared_ptr<event_t>> tmp = s_event_handlers;
|
|
|
|
std::sort(tmp.begin(), tmp.end(),
|
|
|
|
[](const shared_ptr<event_t> &e1, const shared_ptr<event_t> &e2) {
|
|
|
|
if (e1->type == e2->type) {
|
|
|
|
switch (e1->type) {
|
|
|
|
case EVENT_SIGNAL:
|
|
|
|
return e1->param1.signal < e2->param1.signal;
|
|
|
|
case EVENT_JOB_ID:
|
|
|
|
return e1->param1.job_id < e2->param1.job_id;
|
|
|
|
case EVENT_VARIABLE:
|
|
|
|
case EVENT_ANY:
|
|
|
|
case EVENT_GENERIC:
|
|
|
|
return e1->str_param1 < e2->str_param1;
|
|
|
|
}
|
|
|
|
}
|
2018-03-10 18:08:33 +08:00
|
|
|
return e1->type < e2->type;
|
2018-02-01 01:47:23 +08:00
|
|
|
});
|
|
|
|
|
2018-03-10 18:08:33 +08:00
|
|
|
maybe_t<event_type_t> last_type{};
|
2018-02-01 01:47:23 +08:00
|
|
|
for (const shared_ptr<event_t> &evt : tmp) {
|
2018-03-10 18:08:33 +08:00
|
|
|
// If we have a filter, skip events that don't match.
|
|
|
|
if (type_filter && *type_filter != evt->type) {
|
|
|
|
continue;
|
|
|
|
}
|
2018-02-01 01:47:23 +08:00
|
|
|
|
2018-03-10 18:08:33 +08:00
|
|
|
if (!last_type || *last_type != evt->type) {
|
|
|
|
if (last_type)
|
|
|
|
streams.out.append(L"\n");
|
|
|
|
last_type = static_cast<event_type_t>(evt->type);
|
|
|
|
streams.out.append_format(L"Event %ls\n", event_name_for_type(*last_type));
|
|
|
|
}
|
|
|
|
switch (evt->type) {
|
|
|
|
case EVENT_SIGNAL:
|
|
|
|
streams.out.append_format(L"%ls %ls\n", sig2wcs(evt->param1.signal),
|
|
|
|
evt->function_name.c_str());
|
|
|
|
break;
|
|
|
|
case EVENT_JOB_ID:
|
|
|
|
streams.out.append_format(L"%d %ls\n", evt->param1,
|
|
|
|
evt->function_name.c_str());
|
|
|
|
break;
|
|
|
|
case EVENT_VARIABLE:
|
|
|
|
case EVENT_GENERIC:
|
|
|
|
streams.out.append_format(L"%ls %ls\n", evt->str_param1.c_str(),
|
|
|
|
evt->function_name.c_str());
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
streams.out.append_format(L"%ls\n", evt->function_name.c_str());
|
|
|
|
break;
|
2018-02-01 01:47:23 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-04-29 10:19:50 +08:00
|
|
|
void event_fire_generic(const wchar_t *name, wcstring_list_t *args) {
|
|
|
|
CHECK(name, );
|
2012-02-09 11:02:25 +08:00
|
|
|
|
2012-11-19 08:30:30 +08:00
|
|
|
event_t ev(EVENT_GENERIC);
|
|
|
|
ev.str_param1 = name;
|
2016-04-29 10:19:50 +08:00
|
|
|
if (args) ev.arguments = *args;
|
2012-11-19 08:30:30 +08:00
|
|
|
event_fire(&ev);
|
2007-08-20 00:42:30 +08:00
|
|
|
}
|
|
|
|
|
2016-04-29 10:19:50 +08:00
|
|
|
event_t::event_t(int t) : type(t), param1(), str_param1(), function_name(), arguments() {}
|
2013-04-13 16:32:07 +08:00
|
|
|
|
2018-02-19 10:44:58 +08:00
|
|
|
event_t::~event_t() = default;
|
2013-04-13 16:32:07 +08:00
|
|
|
|
2016-04-29 10:19:50 +08:00
|
|
|
event_t event_t::signal_event(int sig) {
|
2012-02-09 11:02:25 +08:00
|
|
|
event_t event(EVENT_SIGNAL);
|
|
|
|
event.param1.signal = sig;
|
|
|
|
return event;
|
|
|
|
}
|
2007-08-20 00:42:30 +08:00
|
|
|
|
2016-04-29 10:19:50 +08:00
|
|
|
event_t event_t::variable_event(const wcstring &str) {
|
2012-02-09 11:02:25 +08:00
|
|
|
event_t event(EVENT_VARIABLE);
|
|
|
|
event.str_param1 = str;
|
|
|
|
return event;
|
|
|
|
}
|
|
|
|
|
2016-04-29 10:19:50 +08:00
|
|
|
event_t event_t::generic_event(const wcstring &str) {
|
2012-02-09 11:02:25 +08:00
|
|
|
event_t event(EVENT_GENERIC);
|
|
|
|
event.str_param1 = str;
|
|
|
|
return event;
|
|
|
|
}
|