mirror of
https://github.com/fish-shell/fish-shell
synced 2024-12-31 23:28:45 +00:00
Optimize exit event generation
Watching for exit events is rare, so check if we have any exit events before actually emitting them. This saves about 2% of time in external_cmds benchmark.
This commit is contained in:
parent
7bd9f1bb23
commit
1b6ef6670f
3 changed files with 44 additions and 12 deletions
|
@ -224,6 +224,15 @@ event_handler_list_t event_get_function_handlers(const wcstring &name) {
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
enum_set_t<event_type_t> event_get_handled_types() {
|
||||||
|
enum_set_t<event_type_t> result{};
|
||||||
|
auto handlers = s_event_handlers.acquire();
|
||||||
|
for (const shared_ptr<event_handler_t> &eh : *handlers) {
|
||||||
|
result.set(eh->desc.type);
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
bool event_is_signal_observed(int sig) {
|
bool event_is_signal_observed(int sig) {
|
||||||
// We are in a signal handler! Don't allocate memory, etc.
|
// We are in a signal handler! Don't allocate memory, etc.
|
||||||
bool result = false;
|
bool result = false;
|
||||||
|
|
10
src/event.h
10
src/event.h
|
@ -13,6 +13,7 @@
|
||||||
#include <vector>
|
#include <vector>
|
||||||
|
|
||||||
#include "common.h"
|
#include "common.h"
|
||||||
|
#include "enum_set.h"
|
||||||
#include "io.h"
|
#include "io.h"
|
||||||
|
|
||||||
/// The process id that is used to match any process id.
|
/// The process id that is used to match any process id.
|
||||||
|
@ -37,6 +38,11 @@ enum class event_type_t {
|
||||||
generic,
|
generic,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
template <>
|
||||||
|
struct enum_info_t<event_type_t> {
|
||||||
|
static constexpr size_t count = static_cast<size_t>(event_type_t::generic) + 1;
|
||||||
|
};
|
||||||
|
|
||||||
/// Null-terminated list of valid event filter names.
|
/// Null-terminated list of valid event filter names.
|
||||||
/// These are what are valid to pass to 'functions --handlers-type'
|
/// These are what are valid to pass to 'functions --handlers-type'
|
||||||
extern const wchar_t *const event_filter_names[];
|
extern const wchar_t *const event_filter_names[];
|
||||||
|
@ -134,6 +140,10 @@ void event_remove_function_handlers(const wcstring &name);
|
||||||
/// Return all event handlers for the given function.
|
/// Return all event handlers for the given function.
|
||||||
event_handler_list_t event_get_function_handlers(const wcstring &name);
|
event_handler_list_t event_get_function_handlers(const wcstring &name);
|
||||||
|
|
||||||
|
/// \return the event types for which handlers are registered.
|
||||||
|
/// This can be a performance optimization to avoid emitting events.
|
||||||
|
enum_set_t<event_type_t> event_get_handled_types();
|
||||||
|
|
||||||
/// Returns whether an event listener is registered for the given signal. This is safe to call from
|
/// Returns whether an event listener is registered for the given signal. This is safe to call from
|
||||||
/// a signal handler.
|
/// a signal handler.
|
||||||
bool event_is_signal_observed(int signal);
|
bool event_is_signal_observed(int signal);
|
||||||
|
|
37
src/proc.cpp
37
src/proc.cpp
|
@ -482,25 +482,28 @@ static void process_mark_finished_children(parser_t &parser, bool block_ok) {
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Given a job that has completed, generate job_exit, process_exit, and caller_exit events.
|
/// Given a job that has completed, generate job_exit, process_exit, and caller_exit events.
|
||||||
static void generate_exit_events(const job_ref_t &j, std::vector<event_t> *out_evts) {
|
static void generate_exit_events(const job_ref_t &j, const enum_set_t<event_type_t> handled,
|
||||||
|
std::vector<event_t> *out_evts) {
|
||||||
// Generate proc and job exit events, except for jobs originating in event handlers.
|
// Generate proc and job exit events, except for jobs originating in event handlers.
|
||||||
if (!j->from_event_handler()) {
|
if (handled.get(event_type_t::process_exit) && !j->from_event_handler()) {
|
||||||
// process_exit events.
|
|
||||||
for (const auto &proc : j->processes) {
|
for (const auto &proc : j->processes) {
|
||||||
if (proc->pid > 0) {
|
if (proc->pid > 0) {
|
||||||
out_evts->push_back(event_t::process_exit(proc->pid, proc->status.status_value()));
|
out_evts->push_back(event_t::process_exit(proc->pid, proc->status.status_value()));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// job_exit events.
|
if (handled.get(event_type_t::job_exit) && !j->from_event_handler() &&
|
||||||
if (j->posts_job_exit_events()) {
|
j->posts_job_exit_events()) {
|
||||||
if (auto last_pid = j->get_last_pid()) {
|
if (auto last_pid = j->get_last_pid()) {
|
||||||
out_evts->push_back(event_t::job_exit(*last_pid, j->internal_job_id));
|
out_evts->push_back(event_t::job_exit(*last_pid, j->internal_job_id));
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Generate caller_exit events.
|
// Generate caller_exit events.
|
||||||
out_evts->push_back(event_t::caller_exit(j->internal_job_id, j->job_id()));
|
if (handled.get(event_type_t::caller_exit)) {
|
||||||
|
out_evts->push_back(event_t::caller_exit(j->internal_job_id, j->job_id()));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// \return whether to emit a fish_job_summary call for a process.
|
/// \return whether to emit a fish_job_summary call for a process.
|
||||||
|
@ -680,6 +683,11 @@ static bool process_clean_after_marking(parser_t &parser, bool allow_interactive
|
||||||
// complete.
|
// complete.
|
||||||
std::vector<event_t> exit_events;
|
std::vector<event_t> exit_events;
|
||||||
|
|
||||||
|
// Figure out which event types have handlers, so we can avoid the cost of creating events that
|
||||||
|
// are unhandled. In principle, an event handler could add a handler for another event type,
|
||||||
|
// which won't receive it. We ignore that possibility.
|
||||||
|
const enum_set_t<event_type_t> handled_event_types = event_get_handled_types();
|
||||||
|
|
||||||
// Defer processing under-construction jobs or jobs that want a message when we are not
|
// Defer processing under-construction jobs or jobs that want a message when we are not
|
||||||
// interactive.
|
// interactive.
|
||||||
auto should_process_job = [=](const shared_ptr<job_t> &j) {
|
auto should_process_job = [=](const shared_ptr<job_t> &j) {
|
||||||
|
@ -713,7 +721,7 @@ static bool process_clean_after_marking(parser_t &parser, bool allow_interactive
|
||||||
// Remember it for summary later, generate exit events, maybe save its wait handle if it
|
// Remember it for summary later, generate exit events, maybe save its wait handle if it
|
||||||
// finished in the background.
|
// finished in the background.
|
||||||
if (job_or_proc_wants_summary(j)) jobs_to_summarize.push_back(j);
|
if (job_or_proc_wants_summary(j)) jobs_to_summarize.push_back(j);
|
||||||
generate_exit_events(j, &exit_events);
|
generate_exit_events(j, handled_event_types, &exit_events);
|
||||||
save_wait_handle_for_completed_job(j, parser.get_wait_handles());
|
save_wait_handle_for_completed_job(j, parser.get_wait_handles());
|
||||||
|
|
||||||
// Remove it.
|
// Remove it.
|
||||||
|
@ -724,8 +732,13 @@ static bool process_clean_after_marking(parser_t &parser, bool allow_interactive
|
||||||
bool printed = summarize_jobs(parser, jobs_to_summarize);
|
bool printed = summarize_jobs(parser, jobs_to_summarize);
|
||||||
|
|
||||||
// Post pending exit events.
|
// Post pending exit events.
|
||||||
for (const auto &evt : exit_events) {
|
// If we have none, still give any other pending events (e.g. from signals) a chance to run.
|
||||||
event_fire(parser, evt);
|
if (exit_events.empty()) {
|
||||||
|
event_fire_delayed(parser);
|
||||||
|
} else {
|
||||||
|
for (const auto &evt : exit_events) {
|
||||||
|
event_fire(parser, evt);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (printed) {
|
if (printed) {
|
||||||
|
|
Loading…
Reference in a new issue