Commit 31e3977a authored by Luca Boccassi's avatar Luca Boccassi Committed by GitHub

Merge pull request #2731 from sigiesec/fix-issue-2723

Problem: assertion in src\select.cpp:111 on Windows or hang on zmq_ctx_destroy
parents 17b95683 23e018f3
...@@ -99,22 +99,19 @@ zmq::select_t::find_fd_entry_by_handle (fd_entries_t &fd_entries, ...@@ -99,22 +99,19 @@ zmq::select_t::find_fd_entry_by_handle (fd_entries_t &fd_entries,
++fd_entry_it) ++fd_entry_it)
if (fd_entry_it->fd == handle_) if (fd_entry_it->fd == handle_)
break; break;
zmq_assert (fd_entry_it != fd_entries.end ());
return fd_entry_it; return fd_entry_it;
} }
void zmq::select_t::rm_fd (handle_t handle_) bool zmq::select_t::try_remove_fd_entry (
family_entries_t::iterator family_entry_it, zmq::fd_t &handle_)
{ {
#if defined ZMQ_HAVE_WINDOWS family_entry_t &family_entry = family_entry_it->second;
u_short family = get_fd_family (handle_);
wsa_assert (family != AF_UNSPEC);
family_entries_t::iterator family_entry_it = family_entries.find (family);
family_entry_t& family_entry = family_entry_it->second;
fd_entries_t::iterator fd_entry_it = fd_entries_t::iterator fd_entry_it =
find_fd_entry_by_handle (family_entry.fd_entries, handle_); find_fd_entry_by_handle (family_entry.fd_entries, handle_);
if (fd_entry_it == family_entry.fd_entries.end ())
return false;
if (family_entry_it != current_family_entry_it) { if (family_entry_it != current_family_entry_it) {
// Family is not currently being iterated and can be safely // Family is not currently being iterated and can be safely
// modified in-place. So later it can be skipped without // modified in-place. So later it can be skipped without
...@@ -127,9 +124,35 @@ void zmq::select_t::rm_fd (handle_t handle_) ...@@ -127,9 +124,35 @@ void zmq::select_t::rm_fd (handle_t handle_)
family_entry.retired = true; family_entry.retired = true;
} }
family_entry.fds_set.remove_fd (handle_); family_entry.fds_set.remove_fd (handle_);
return true;
}
void zmq::select_t::rm_fd (handle_t handle_)
{
#if defined ZMQ_HAVE_WINDOWS
u_short family = get_fd_family (handle_);
if (family != AF_UNSPEC) {
family_entries_t::iterator family_entry_it =
family_entries.find (family);
int removed = try_remove_fd_entry (family_entry_it, handle_);
assert (removed);
} else {
// get_fd_family may fail and return AF_UNSPEC if the socket was not
// successfully connected. In that case, we need to look for the
// socket in all family_entries.
family_entries_t::iterator end = family_entries.end ();
for (family_entries_t::iterator family_entry_it =
family_entries.begin ();
family_entry_it != end; ++family_entry_it) {
if (try_remove_fd_entry (family_entry_it, handle_))
break;
}
}
#else #else
fd_entries_t::iterator fd_entry_it = fd_entries_t::iterator fd_entry_it =
find_fd_entry_by_handle (fd_entries, handle_); find_fd_entry_by_handle (fd_entries, handle_);
assert (fd_entry_it != fd_entries.end ());
fd_entry_it->fd = retired_fd; fd_entry_it->fd = retired_fd;
fds_set.remove_fd (handle_); fds_set.remove_fd (handle_);
......
...@@ -137,6 +137,9 @@ namespace zmq ...@@ -137,6 +137,9 @@ namespace zmq
family_entries_t family_entries; family_entries_t family_entries;
// See loop for details. // See loop for details.
family_entries_t::iterator current_family_entry_it; family_entries_t::iterator current_family_entry_it;
bool try_remove_fd_entry (family_entries_t::iterator family_entry_it,
zmq::fd_t &handle_);
#else #else
fd_entries_t fd_entries; fd_entries_t fd_entries;
fds_set_t fds_set; fds_set_t fds_set;
......
...@@ -434,13 +434,19 @@ void zmq::session_base_t::engine_error ( ...@@ -434,13 +434,19 @@ void zmq::session_base_t::engine_error (
switch (reason) { switch (reason) {
case stream_engine_t::timeout_error: case stream_engine_t::timeout_error:
case stream_engine_t::connection_error: case stream_engine_t::connection_error:
if (active) if (active) {
reconnect (); reconnect ();
else break;
terminate (); }
break;
case stream_engine_t::protocol_error: case stream_engine_t::protocol_error:
terminate (); if (pending) {
if (pipe)
pipe->terminate (0);
if (zap_pipe)
zap_pipe->terminate (0);
} else {
terminate ();
}
break; break;
} }
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment