mirror of
https://gitlab.torproject.org/tpo/core/tor.git
synced 2025-02-25 07:07:52 +01:00
Merge branch 'postloop_callbacks_2'
This commit is contained in:
commit
61d87dfa15
7 changed files with 227 additions and 56 deletions
6
changes/ticket25374
Normal file
6
changes/ticket25374
Normal file
|
@ -0,0 +1,6 @@
|
|||
o Code simplification and refactoring:
|
||||
- Our main loop has been simplified so that all important operations
|
||||
happen inside events. Previously, some operations had to happen
|
||||
outside the event loop, to prevent infinite sequences of event
|
||||
activations. Closes ticket 25374.
|
||||
|
|
@ -79,6 +79,43 @@ tor_event_free_(struct event *ev)
|
|||
/** Global event base for use by the main thread. */
|
||||
static struct event_base *the_event_base = NULL;
|
||||
|
||||
/**
|
||||
* @defgroup postloop post-loop event helpers
|
||||
*
|
||||
* If we're not careful, Libevent can susceptible to infinite event chains:
|
||||
* one event can activate another, whose callback activates another, whose
|
||||
* callback activates another, ad infinitum. While this is happening,
|
||||
* Libevent won't be checking timeouts, socket-based events, signals, and so
|
||||
* on.
|
||||
*
|
||||
* We solve this problem by marking some events as "post-loop". A post-loop
|
||||
* event behaves like any ordinary event, but any events that _it_ activates
|
||||
* cannot run until Libevent has checked for other events at least once.
|
||||
*
|
||||
* @{ */
|
||||
|
||||
/**
|
||||
* An event that stops Libevent from running any more events on the current
|
||||
* iteration of its loop, until it has re-checked for socket events, signal
|
||||
* events, timeouts, etc.
|
||||
*/
|
||||
static struct event *rescan_mainloop_ev = NULL;
|
||||
|
||||
/**
|
||||
* Callback to implement rescan_mainloop_ev: it simply exits the mainloop,
|
||||
* and relies on Tor to re-enter the mainloop since no error has occurred.
|
||||
*/
|
||||
static void
|
||||
rescan_mainloop_cb(evutil_socket_t fd, short events, void *arg)
|
||||
{
|
||||
(void)fd;
|
||||
(void)events;
|
||||
struct event_base *the_base = arg;
|
||||
event_base_loopbreak(the_base);
|
||||
}
|
||||
|
||||
/** @} */
|
||||
|
||||
/* This is what passes for version detection on OSX. We set
|
||||
* MACOSX_KQUEUE_IS_BROKEN to true iff we're on a version of OSX before
|
||||
* 10.4.0 (aka 1040). */
|
||||
|
@ -130,6 +167,15 @@ tor_libevent_initialize(tor_libevent_cfg *torcfg)
|
|||
/* LCOV_EXCL_STOP */
|
||||
}
|
||||
|
||||
rescan_mainloop_ev = event_new(the_event_base, -1, 0,
|
||||
rescan_mainloop_cb, the_event_base);
|
||||
if (!rescan_mainloop_ev) {
|
||||
/* LCOV_EXCL_START */
|
||||
log_err(LD_GENERAL, "Unable to create rescan event: cannot continue.");
|
||||
exit(1); // exit ok: libevent is broken.
|
||||
/* LCOV_EXCL_STOP */
|
||||
}
|
||||
|
||||
log_info(LD_GENERAL,
|
||||
"Initialized libevent version %s using method %s. Good.",
|
||||
event_get_version(), tor_libevent_get_method());
|
||||
|
@ -250,6 +296,52 @@ mainloop_event_cb(evutil_socket_t fd, short what, void *arg)
|
|||
mev->cb(mev, mev->userdata);
|
||||
}
|
||||
|
||||
/**
|
||||
* As mainloop_event_cb, but implements a post-loop event.
|
||||
*/
|
||||
static void
|
||||
mainloop_event_postloop_cb(evutil_socket_t fd, short what, void *arg)
|
||||
{
|
||||
(void)fd;
|
||||
(void)what;
|
||||
|
||||
/* Note that if rescan_mainloop_ev is already activated,
|
||||
* event_active() will do nothing: only the first post-loop event that
|
||||
* happens each time through the event loop will cause it to be
|
||||
* activated.
|
||||
*
|
||||
* Because event_active() puts events on a FIFO queue, every event
|
||||
* that is made active _after_ rescan_mainloop_ev will get its
|
||||
* callback run after rescan_mainloop_cb is called -- that is, on the
|
||||
* next iteration of the loop.
|
||||
*/
|
||||
event_active(rescan_mainloop_ev, EV_READ, 1);
|
||||
|
||||
mainloop_event_t *mev = arg;
|
||||
mev->cb(mev, mev->userdata);
|
||||
}
|
||||
|
||||
/**
|
||||
* Helper for mainloop_event_new() and mainloop_event_postloop_new().
|
||||
*/
|
||||
static mainloop_event_t *
|
||||
mainloop_event_new_impl(int postloop,
|
||||
void (*cb)(mainloop_event_t *, void *),
|
||||
void *userdata)
|
||||
{
|
||||
tor_assert(cb);
|
||||
|
||||
struct event_base *base = tor_libevent_get_base();
|
||||
mainloop_event_t *mev = tor_malloc_zero(sizeof(mainloop_event_t));
|
||||
mev->ev = tor_event_new(base, -1, 0,
|
||||
postloop ? mainloop_event_postloop_cb : mainloop_event_cb,
|
||||
mev);
|
||||
tor_assert(mev->ev);
|
||||
mev->cb = cb;
|
||||
mev->userdata = userdata;
|
||||
return mev;
|
||||
}
|
||||
|
||||
/**
|
||||
* Create and return a new mainloop_event_t to run the function <b>cb</b>.
|
||||
*
|
||||
|
@ -265,15 +357,21 @@ mainloop_event_t *
|
|||
mainloop_event_new(void (*cb)(mainloop_event_t *, void *),
|
||||
void *userdata)
|
||||
{
|
||||
tor_assert(cb);
|
||||
return mainloop_event_new_impl(0, cb, userdata);
|
||||
}
|
||||
|
||||
struct event_base *base = tor_libevent_get_base();
|
||||
mainloop_event_t *mev = tor_malloc_zero(sizeof(mainloop_event_t));
|
||||
mev->ev = tor_event_new(base, -1, 0, mainloop_event_cb, mev);
|
||||
tor_assert(mev->ev);
|
||||
mev->cb = cb;
|
||||
mev->userdata = userdata;
|
||||
return mev;
|
||||
/**
|
||||
* As mainloop_event_new(), but create a post-loop event.
|
||||
*
|
||||
* A post-loop event behaves like any ordinary event, but any events
|
||||
* that _it_ activates cannot run until Libevent has checked for other
|
||||
* events at least once.
|
||||
*/
|
||||
mainloop_event_t *
|
||||
mainloop_event_postloop_new(void (*cb)(mainloop_event_t *, void *),
|
||||
void *userdata)
|
||||
{
|
||||
return mainloop_event_new_impl(1, cb, userdata);
|
||||
}
|
||||
|
||||
/**
|
||||
|
@ -358,6 +456,7 @@ tor_init_libevent_rng(void)
|
|||
void
|
||||
tor_libevent_free_all(void)
|
||||
{
|
||||
tor_event_free(rescan_mainloop_ev);
|
||||
if (the_event_base)
|
||||
event_base_free(the_event_base);
|
||||
the_event_base = NULL;
|
||||
|
|
|
@ -37,6 +37,9 @@ void periodic_timer_free_(periodic_timer_t *);
|
|||
typedef struct mainloop_event_t mainloop_event_t;
|
||||
mainloop_event_t *mainloop_event_new(void (*cb)(mainloop_event_t *, void *),
|
||||
void *userdata);
|
||||
mainloop_event_t * mainloop_event_postloop_new(
|
||||
void (*cb)(mainloop_event_t *, void *),
|
||||
void *userdata);
|
||||
void mainloop_event_activate(mainloop_event_t *event);
|
||||
int mainloop_event_schedule(mainloop_event_t *event,
|
||||
const struct timeval *delay);
|
||||
|
|
|
@ -611,6 +611,12 @@ static smartlist_t *pending_entry_connections = NULL;
|
|||
|
||||
static int untried_pending_connections = 0;
|
||||
|
||||
/**
|
||||
* Mainloop event to tell us to scan for pending connections that can
|
||||
* be attached.
|
||||
*/
|
||||
static mainloop_event_t *attach_pending_entry_connections_ev = NULL;
|
||||
|
||||
/** Common code to connection_(ap|exit)_about_to_close. */
|
||||
static void
|
||||
connection_edge_about_to_close(edge_connection_t *edge_conn)
|
||||
|
@ -956,6 +962,14 @@ connection_ap_attach_pending(int retry)
|
|||
untried_pending_connections = 0;
|
||||
}
|
||||
|
||||
static void
|
||||
attach_pending_entry_connections_cb(mainloop_event_t *ev, void *arg)
|
||||
{
|
||||
(void)ev;
|
||||
(void)arg;
|
||||
connection_ap_attach_pending(0);
|
||||
}
|
||||
|
||||
/** Mark <b>entry_conn</b> as needing to get attached to a circuit.
|
||||
*
|
||||
* And <b>entry_conn</b> must be in AP_CONN_STATE_CIRCUIT_WAIT,
|
||||
|
@ -973,9 +987,13 @@ connection_ap_mark_as_pending_circuit_(entry_connection_t *entry_conn,
|
|||
if (conn->marked_for_close)
|
||||
return;
|
||||
|
||||
if (PREDICT_UNLIKELY(NULL == pending_entry_connections))
|
||||
if (PREDICT_UNLIKELY(NULL == pending_entry_connections)) {
|
||||
pending_entry_connections = smartlist_new();
|
||||
|
||||
}
|
||||
if (PREDICT_UNLIKELY(NULL == attach_pending_entry_connections_ev)) {
|
||||
attach_pending_entry_connections_ev = mainloop_event_postloop_new(
|
||||
attach_pending_entry_connections_cb, NULL);
|
||||
}
|
||||
if (PREDICT_UNLIKELY(smartlist_contains(pending_entry_connections,
|
||||
entry_conn))) {
|
||||
log_warn(LD_BUG, "What?? pending_entry_connections already contains %p! "
|
||||
|
@ -999,14 +1017,7 @@ connection_ap_mark_as_pending_circuit_(entry_connection_t *entry_conn,
|
|||
untried_pending_connections = 1;
|
||||
smartlist_add(pending_entry_connections, entry_conn);
|
||||
|
||||
/* Work-around for bug 19969: we handle pending_entry_connections at
|
||||
* the end of run_main_loop_once(), but in many cases that function will
|
||||
* take a very long time, if ever, to finish its call to event_base_loop().
|
||||
*
|
||||
* So the fix is to tell it right now that it ought to finish its loop at
|
||||
* its next available opportunity.
|
||||
*/
|
||||
tell_event_loop_to_run_external_code();
|
||||
mainloop_event_activate(attach_pending_entry_connections_ev);
|
||||
}
|
||||
|
||||
/** Mark <b>entry_conn</b> as no longer waiting for a circuit. */
|
||||
|
@ -4165,5 +4176,6 @@ connection_edge_free_all(void)
|
|||
untried_pending_connections = 0;
|
||||
smartlist_free(pending_entry_connections);
|
||||
pending_entry_connections = NULL;
|
||||
mainloop_event_free(attach_pending_entry_connections_ev);
|
||||
}
|
||||
|
||||
|
|
|
@ -410,6 +410,27 @@ connection_unlink(connection_t *conn)
|
|||
connection_free(conn);
|
||||
}
|
||||
|
||||
/**
|
||||
* Callback: used to activate read events for all linked connections, so
|
||||
* libevent knows to call their read callbacks. This callback run as a
|
||||
* postloop event, so that the events _it_ activates don't happen until
|
||||
* Libevent has a chance to check for other events.
|
||||
*/
|
||||
static void
|
||||
schedule_active_linked_connections_cb(mainloop_event_t *event, void *arg)
|
||||
{
|
||||
(void)event;
|
||||
(void)arg;
|
||||
|
||||
/* All active linked conns should get their read events activated,
|
||||
* so that libevent knows to run their callbacks. */
|
||||
SMARTLIST_FOREACH(active_linked_connection_lst, connection_t *, conn,
|
||||
event_active(conn->read_event, EV_READ, 1));
|
||||
}
|
||||
|
||||
/** Event that invokes schedule_active_linked_connections_cb. */
|
||||
static mainloop_event_t *schedule_active_linked_connections_event = NULL;
|
||||
|
||||
/** Initialize the global connection list, closeable connection list,
|
||||
* and active connection list. */
|
||||
STATIC void
|
||||
|
@ -710,20 +731,6 @@ connection_should_read_from_linked_conn(connection_t *conn)
|
|||
return 0;
|
||||
}
|
||||
|
||||
/** If we called event_base_loop() and told it to never stop until it
|
||||
* runs out of events, now we've changed our mind: tell it we want it to
|
||||
* exit once the current round of callbacks is done, so that we can
|
||||
* run external code, and then return to the main loop. */
|
||||
void
|
||||
tell_event_loop_to_run_external_code(void)
|
||||
{
|
||||
if (!called_loop_once) {
|
||||
struct timeval tv = { 0, 0 };
|
||||
tor_libevent_exit_loop_after_delay(tor_libevent_get_base(), &tv);
|
||||
called_loop_once = 1; /* hack to avoid adding more exit events */
|
||||
}
|
||||
}
|
||||
|
||||
/** Event to run 'shutdown did not work callback'. */
|
||||
static struct event *shutdown_did_not_work_event = NULL;
|
||||
|
||||
|
@ -803,10 +810,7 @@ connection_start_reading_from_linked_conn(connection_t *conn)
|
|||
if (!conn->active_on_link) {
|
||||
conn->active_on_link = 1;
|
||||
smartlist_add(active_linked_connection_lst, conn);
|
||||
/* make sure that the event_base_loop() function exits at
|
||||
* the end of its run through the current connections, so we can
|
||||
* activate read events for linked connections. */
|
||||
tell_event_loop_to_run_external_code();
|
||||
mainloop_event_activate(schedule_active_linked_connections_event);
|
||||
} else {
|
||||
tor_assert(smartlist_contains(active_linked_connection_lst, conn));
|
||||
}
|
||||
|
@ -2583,6 +2587,11 @@ do_main_loop(void)
|
|||
initialize_periodic_events();
|
||||
}
|
||||
|
||||
if (!schedule_active_linked_connections_event) {
|
||||
schedule_active_linked_connections_event =
|
||||
mainloop_event_postloop_new(schedule_active_linked_connections_cb, NULL);
|
||||
}
|
||||
|
||||
/* initialize dns resolve map, spawn workers if needed */
|
||||
if (dns_init() < 0) {
|
||||
if (get_options()->ServerDNSAllowBrokenConfig)
|
||||
|
@ -2787,17 +2796,12 @@ run_main_loop_once(void)
|
|||
errno = 0;
|
||||
#endif
|
||||
|
||||
/* All active linked conns should get their read events activated,
|
||||
* so that libevent knows to run their callbacks. */
|
||||
SMARTLIST_FOREACH(active_linked_connection_lst, connection_t *, conn,
|
||||
event_active(conn->read_event, EV_READ, 1));
|
||||
|
||||
if (get_options()->MainloopStats) {
|
||||
/* We always enforce that EVLOOP_ONCE is passed to event_base_loop() if we
|
||||
* are collecting main loop statistics. */
|
||||
called_loop_once = 1;
|
||||
} else {
|
||||
called_loop_once = smartlist_len(active_linked_connection_lst) ? 1 : 0;
|
||||
called_loop_once = 0;
|
||||
}
|
||||
|
||||
/* Make sure we know (about) what time it is. */
|
||||
|
@ -2853,19 +2857,6 @@ run_main_loop_once(void)
|
|||
if (main_loop_should_exit)
|
||||
return 0;
|
||||
|
||||
/* And here is where we put callbacks that happen "every time the event loop
|
||||
* runs." They must be very fast, or else the whole Tor process will get
|
||||
* slowed down.
|
||||
*
|
||||
* Note that this gets called once per libevent loop, which will make it
|
||||
* happen once per group of events that fire, or once per second. */
|
||||
|
||||
/* If there are any pending client connections, try attaching them to
|
||||
* circuits (if we can.) This will be pretty fast if nothing new is
|
||||
* pending.
|
||||
*/
|
||||
connection_ap_attach_pending(0);
|
||||
|
||||
return 1;
|
||||
}
|
||||
|
||||
|
@ -3500,6 +3491,7 @@ tor_free_all(int postfork)
|
|||
tor_event_free(shutdown_did_not_work_event);
|
||||
tor_event_free(initialize_periodic_events_event);
|
||||
mainloop_event_free(directory_all_unreachable_cb_event);
|
||||
mainloop_event_free(schedule_active_linked_connections_event);
|
||||
|
||||
#ifdef HAVE_SYSTEMD_209
|
||||
periodic_timer_free(systemd_watchdog_timer);
|
||||
|
|
|
@ -45,7 +45,6 @@ int connection_is_writing(connection_t *conn);
|
|||
MOCK_DECL(void,connection_stop_writing,(connection_t *conn));
|
||||
MOCK_DECL(void,connection_start_writing,(connection_t *conn));
|
||||
|
||||
void tell_event_loop_to_run_external_code(void);
|
||||
void tor_shutdown_event_loop_and_exit(int exitcode);
|
||||
int tor_event_loop_shutdown_is_pending(void);
|
||||
|
||||
|
|
|
@ -121,10 +121,70 @@ test_compat_libevent_header_version(void *ignored)
|
|||
(void)0;
|
||||
}
|
||||
|
||||
/* Test for postloop events */
|
||||
|
||||
/* Event callback to increment a counter. */
|
||||
static void
|
||||
increment_int_counter_cb(periodic_timer_t *timer, void *arg)
|
||||
{
|
||||
(void)timer;
|
||||
int *ctr = arg;
|
||||
++*ctr;
|
||||
}
|
||||
|
||||
static int activated_counter = 0;
|
||||
|
||||
/* Mainloop event callback to activate another mainloop event */
|
||||
static void
|
||||
activate_event_cb(mainloop_event_t *ev, void *arg)
|
||||
{
|
||||
(void)ev;
|
||||
mainloop_event_t **other_event = arg;
|
||||
mainloop_event_activate(*other_event);
|
||||
++activated_counter;
|
||||
}
|
||||
|
||||
static void
|
||||
test_compat_libevent_postloop_events(void *arg)
|
||||
{
|
||||
(void)arg;
|
||||
mainloop_event_t *a = NULL, *b = NULL;
|
||||
periodic_timer_t *timed = NULL;
|
||||
|
||||
tor_libevent_postfork();
|
||||
|
||||
/* If postloop events don't work, then these events will activate one
|
||||
* another ad infinitum and, and the periodic event will never occur. */
|
||||
b = mainloop_event_postloop_new(activate_event_cb, &a);
|
||||
a = mainloop_event_postloop_new(activate_event_cb, &b);
|
||||
|
||||
int counter = 0;
|
||||
struct timeval fifty_ms = { 0, 10 * 1000 };
|
||||
timed = periodic_timer_new(tor_libevent_get_base(), &fifty_ms,
|
||||
increment_int_counter_cb, &counter);
|
||||
|
||||
mainloop_event_activate(a);
|
||||
int r;
|
||||
do {
|
||||
r = tor_libevent_run_event_loop(tor_libevent_get_base(), 0);
|
||||
if (r == -1)
|
||||
break;
|
||||
} while (counter < 5);
|
||||
|
||||
tt_int_op(activated_counter, OP_GE, 2);
|
||||
|
||||
done:
|
||||
mainloop_event_free(a);
|
||||
mainloop_event_free(b);
|
||||
periodic_timer_free(timed);
|
||||
}
|
||||
|
||||
struct testcase_t compat_libevent_tests[] = {
|
||||
{ "logging_callback", test_compat_libevent_logging_callback,
|
||||
TT_FORK, NULL, NULL },
|
||||
{ "header_version", test_compat_libevent_header_version, 0, NULL, NULL },
|
||||
{ "postloop_events", test_compat_libevent_postloop_events,
|
||||
TT_FORK, NULL, NULL },
|
||||
END_OF_TESTCASES
|
||||
};
|
||||
|
||||
|
|
Loading…
Add table
Reference in a new issue