Clean up event_t handling

Use shared_ptr instead of the silly killme list
This commit is contained in:
ridiculousfish 2017-01-21 16:48:07 -08:00
parent 8a0d4854e8
commit f0065cda13
4 changed files with 51 additions and 100 deletions

View file

@ -944,7 +944,7 @@ static wcstring functions_def(const wcstring &name) {
function_get_definition(name, &def); function_get_definition(name, &def);
event_t search(EVENT_ANY); event_t search(EVENT_ANY);
search.function_name = name; search.function_name = name;
std::vector<event_t *> ev; std::vector<std::shared_ptr<event_t>> ev;
event_get(search, &ev); event_get(search, &ev);
out.append(L"function "); out.append(L"function ");
@ -967,7 +967,7 @@ static wcstring functions_def(const wcstring &name) {
} }
for (size_t i = 0; i < ev.size(); i++) { for (size_t i = 0; i < ev.size(); i++) {
event_t *next = ev.at(i); const event_t *next = ev.at(i).get();
switch (next->type) { switch (next->type) {
case EVENT_SIGNAL: { case EVENT_SIGNAL: {
append_format(out, L" --on-signal %ls", sig2wcs(next->param1.signal)); append_format(out, L" --on-signal %ls", sig2wcs(next->param1.signal));

View file

@ -479,7 +479,6 @@ void complete_remove(const wcstring &cmd, bool cmd_is_path, const wcstring &opti
bool delete_it = entry.remove_option(option, type); bool delete_it = entry.remove_option(option, type);
if (delete_it) { if (delete_it) {
// Delete this entry.
completion_set.erase(iter); completion_set.erase(iter);
} }
} }

View file

@ -39,12 +39,10 @@ static signal_list_t sig_list[2] = {{}, {}};
/// The index of sig_list that is the list of signals currently written to. /// The index of sig_list that is the list of signals currently written to.
static volatile int active_list = 0; static volatile int active_list = 0;
typedef std::vector<event_t *> event_list_t; typedef std::vector<shared_ptr<event_t>> event_list_t;
/// List of event handlers. /// List of event handlers.
static event_list_t s_event_handlers; static event_list_t s_event_handlers;
/// List of event handlers that should be removed.
static event_list_t killme;
/// List of events that have been sent but have not yet been delivered because they are blocked. /// List of events that have been sent but have not yet been delivered because they are blocked.
static event_list_t blocked; static event_list_t blocked;
@ -242,61 +240,51 @@ static wcstring event_desc_compact(const event_t &event) {
} }
void event_add_handler(const event_t &event) { void event_add_handler(const event_t &event) {
event_t *e;
if (debug_level >= 3) { if (debug_level >= 3) {
wcstring desc = event_desc_compact(event); wcstring desc = event_desc_compact(event);
debug(3, "register: %ls\n", desc.c_str()); debug(3, "register: %ls\n", desc.c_str());
} }
e = new event_t(event); shared_ptr<event_t> e = std::make_shared<event_t>(event);
if (e->type == EVENT_SIGNAL) { if (e->type == EVENT_SIGNAL) {
signal_handle(e->param1.signal, 1); signal_handle(e->param1.signal, 1);
set_signal_observed(e->param1.signal, true); set_signal_observed(e->param1.signal, true);
} }
s_event_handlers.push_back(e); s_event_handlers.push_back(std::move(e));
} }
void event_remove(const event_t &criterion) { void event_remove(const event_t &criterion) {
event_list_t new_list;
if (debug_level >= 3) { if (debug_level >= 3) {
wcstring desc = event_desc_compact(criterion); wcstring desc = event_desc_compact(criterion);
debug(3, "unregister: %ls\n", desc.c_str()); debug(3, "unregister: %ls\n", desc.c_str());
} }
// Because of concurrency issues (env_remove could remove an event that is currently being event_list_t::iterator iter = s_event_handlers.begin();
// executed), env_remove does not actually free any events - instead it simply moves all events while (iter != s_event_handlers.end()) {
// that should be removed from the event list to the killme list, and the ones that shouldn't be const event_t *n = iter->get();
// killed to new_list, and then drops the empty events-list. if (! event_match(criterion, *n)) {
if (s_event_handlers.empty()) return; ++iter;
continue;
for (size_t i = 0; i < s_event_handlers.size(); i++) { }
event_t *n = s_event_handlers.at(i);
if (event_match(criterion, *n)) {
killme.push_back(n);
// If this event was a signal handler and no other handler handles the specified signal // 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. // type, do not handle that type of signal any more.
if (n->type == EVENT_SIGNAL) { if (n->type == EVENT_SIGNAL) {
event_t e = event_t::signal_event(n->param1.signal); event_t e = event_t::signal_event(n->param1.signal);
if (event_get(e, 0) == 1) { if (event_get(e, NULL) == 1) {
signal_handle(e.param1.signal, 0); signal_handle(e.param1.signal, 0);
set_signal_observed(e.param1.signal, 0); set_signal_observed(e.param1.signal, 0);
} }
} }
} else { iter = s_event_handlers.erase(iter);
new_list.push_back(n);
} }
}
s_event_handlers.swap(new_list);
} }
int event_get(const event_t &criterion, std::vector<event_t *> *out) { int event_get(const event_t &criterion, event_list_t *out) {
ASSERT_IS_MAIN_THREAD();
int found = 0; int found = 0;
for (size_t i = 0; i < s_event_handlers.size(); i++) { for (const shared_ptr<event_t> &n : s_event_handlers) {
event_t *n = s_event_handlers.at(i);
if (event_match(criterion, *n)) { if (event_match(criterion, *n)) {
found++; found++;
if (out) out->push_back(n); if (out) out->push_back(n);
@ -314,17 +302,6 @@ bool event_is_signal_observed(int sig) {
return result; return result;
} }
/// Free all events in the kill list.
static void event_free_kills() {
for_each(killme.begin(), killme.end(), event_free);
killme.resize(0);
}
/// Test if the specified event is waiting to be killed.
static int event_is_killed(const event_t &e) {
return std::find(killme.begin(), killme.end(), &e) != killme.end();
}
/// Callback for firing (and then deleting) an event. /// Callback for firing (and then deleting) an event.
static void fire_event_callback(void *arg) { static void fire_event_callback(void *arg) {
ASSERT_IS_MAIN_THREAD(); ASSERT_IS_MAIN_THREAD();
@ -338,20 +315,11 @@ static void fire_event_callback(void *arg) {
/// event handler, we make sure to optimize the 'no matches' path. This means that nothing is /// event handler, we make sure to optimize the 'no matches' path. This means that nothing is
/// allocated/initialized unless needed. /// allocated/initialized unless needed.
static void event_fire_internal(const event_t &event) { static void event_fire_internal(const event_t &event) {
event_list_t fire; // Iterate over all events, adding events that should be fired to a second list. We need
// First we free all events that have been removed, but only if this invocation of
// event_fire_internal is not a recursive call.
if (is_event <= 1) event_free_kills();
if (s_event_handlers.empty()) return;
// Then we iterate over all events, adding events that should be fired to a second list. We need
// to do this in a separate step since an event handler might call event_remove or // 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. // event_add_handler, which will change the contents of the \c events list.
for (size_t i = 0; i < s_event_handlers.size(); i++) { event_list_t fire;
event_t *criterion = s_event_handlers.at(i); for (shared_ptr<event_t> &criterion : s_event_handlers) {
// Check if this event is a match. // Check if this event is a match.
if (event_match(*criterion, event)) { if (event_match(*criterion, event)) {
fire.push_back(criterion); fire.push_back(criterion);
@ -370,12 +338,13 @@ static void event_fire_internal(const event_t &event) {
} }
// Iterate over our list of matching events. // Iterate over our list of matching events.
for (size_t i = 0; i < fire.size(); i++) { for (shared_ptr<event_t> &criterion : fire) {
event_t *criterion = fire.at(i); // Only fire if this event is still present
int prev_status; if (std::find(s_event_handlers.begin(),
s_event_handlers.end(),
// Check if this event has been removed, if so, dont fire it. criterion) == s_event_handlers.end()) {
if (event_is_killed(*criterion)) continue; continue;
}
// Fire event. // Fire event.
wcstring buffer = criterion->function_name; wcstring buffer = criterion->function_name;
@ -391,7 +360,7 @@ static void event_fire_internal(const event_t &event) {
// Event handlers are not part of the main flow of code, so they are marked as // Event handlers are not part of the main flow of code, so they are marked as
// non-interactive. // non-interactive.
proc_push_interactive(0); proc_push_interactive(0);
prev_status = proc_get_last_status(); int prev_status = proc_get_last_status();
parser_t &parser = parser_t::principal_parser(); parser_t &parser = parser_t::principal_parser();
event_block_t *b = parser.push_block<event_block_t>(event); event_block_t *b = parser.push_block<event_block_t>(event);
@ -400,9 +369,6 @@ static void event_fire_internal(const event_t &event) {
proc_pop_interactive(); proc_pop_interactive();
proc_set_last_status(prev_status); proc_set_last_status(prev_status);
} }
// Free killed events.
if (is_event <= 1) event_free_kills();
} }
/// Handle all pending signal events. /// Handle all pending signal events.
@ -412,18 +378,15 @@ static void event_fire_delayed() {
// When the event handler has called a piece of code that triggers another event, we do not want // 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. // to fire delayed events because of concurrency problems.
if (!blocked.empty() && is_event == 1) { if (!blocked.empty() && is_event == 1) {
event_list_t new_blocked; event_list_t local_blocked;
local_blocked.swap(blocked);
for (size_t i = 0; i < blocked.size(); i++) { for (const shared_ptr<event_t> &e : local_blocked) {
event_t *e = blocked.at(i);
if (event_is_blocked(*e)) { if (event_is_blocked(*e)) {
new_blocked.push_back(new event_t(*e)); blocked.push_back(e);
} else { } else {
event_fire_internal(*e); event_fire_internal(*e);
event_free(e);
} }
} }
blocked.swap(new_blocked);
} }
int al = active_list; int al = active_list;
@ -439,21 +402,20 @@ static void event_fire_delayed() {
// Set up. // Set up.
lst = &sig_list[1 - al]; lst = &sig_list[1 - al];
event_t e = event_t::signal_event(0);
e.arguments.resize(1);
if (lst->overflow) { if (lst->overflow) {
debug(0, _(L"Signal list overflow. Signals have been ignored.")); debug(0, _(L"Signal list overflow. Signals have been ignored."));
} }
// Send all signals in our private list. // Send all signals in our private list.
for (int i = 0; i < lst->count; i++) { for (int i = 0; i < lst->count; i++) {
e.param1.signal = lst->signal[i]; shared_ptr<event_t> e = std::make_shared<event_t>(EVENT_SIGNAL);
e.arguments.at(0) = sig2wcs(e.param1.signal); int signal = lst->signal[i];
if (event_is_blocked(e)) { e->param1.signal = signal;
blocked.push_back(new event_t(e)); e->arguments.push_back(sig2wcs(signal));
if (event_is_blocked(*e)) {
blocked.push_back(e);
} else { } else {
event_fire_internal(e); event_fire_internal(*e);
} }
} }
} }
@ -479,7 +441,7 @@ void event_fire(const event_t *event) {
if (event) { if (event) {
if (event_is_blocked(*event)) { if (event_is_blocked(*event)) {
blocked.push_back(new event_t(*event)); blocked.push_back(std::make_shared<event_t>(*event));
} else { } else {
event_fire_internal(*event); event_fire_internal(*event);
} }
@ -491,16 +453,7 @@ void event_fire(const event_t *event) {
void event_init() {} void event_init() {}
void event_destroy() { void event_destroy() {
for_each(s_event_handlers.begin(), s_event_handlers.end(), event_free);
s_event_handlers.clear(); s_event_handlers.clear();
for_each(killme.begin(), killme.end(), event_free);
killme.clear();
}
void event_free(event_t *e) {
CHECK(e, );
delete e;
} }
void event_fire_generic(const wchar_t *name, wcstring_list_t *args) { void event_fire_generic(const wchar_t *name, wcstring_list_t *args) {

View file

@ -7,6 +7,7 @@
#define FISH_EVENT_H #define FISH_EVENT_H
#include <unistd.h> #include <unistd.h>
#include <memory>
#include <vector> #include <vector>
#include "common.h" #include "common.h"
@ -100,7 +101,7 @@ void event_remove(const event_t &event);
/// result count will still be valid /// result count will still be valid
/// ///
/// \return the number of found matches /// \return the number of found matches
int event_get(const event_t &criterion, std::vector<event_t *> *out); int event_get(const event_t &criterion, std::vector<std::shared_ptr<event_t>> *out);
/// 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.
@ -120,6 +121,7 @@ bool event_is_signal_observed(int signal);
void event_fire(const event_t *event); void event_fire(const event_t *event);
/// Like event_fire, but takes a signal directly. /// Like event_fire, but takes a signal directly.
/// May be called from signal handlers
void event_fire_signal(int signal); void event_fire_signal(int signal);
/// Initialize the event-handling library. /// Initialize the event-handling library.
@ -128,9 +130,6 @@ void event_init();
/// Destroy the event-handling library. /// Destroy the event-handling library.
void event_destroy(); void event_destroy();
/// Free all memory used by the specified event.
void event_free(event_t *e);
/// Returns a string describing the specified event. /// Returns a string describing the specified event.
wcstring event_get_desc(const event_t &e); wcstring event_get_desc(const event_t &e);