diff --git a/src/net.cpp b/src/net.cpp index 1fa303ff50e91d..cb2fddd6b7d803 100644 --- a/src/net.cpp +++ b/src/net.cpp @@ -1438,11 +1438,11 @@ bool CConnman::GenerateSelectSet(std::set &recv_set, std::set &s } #ifdef USE_POLL -void CConnman::SocketEvents(std::set &recv_set, std::set &send_set, std::set &error_set) +void CConnman::SocketEvents(std::set &recv_set, std::set &send_set, std::set &error_set, bool fOnlyPoll) { std::set recv_select_set, send_select_set, error_select_set; if (!GenerateSelectSet(recv_select_set, send_select_set, error_select_set)) { - interruptNet.sleep_for(std::chrono::milliseconds(SELECT_TIMEOUT_MILLISECONDS)); + if (!fOnlyPoll) interruptNet.sleep_for(std::chrono::milliseconds(SELECT_TIMEOUT_MILLISECONDS)); return; } @@ -1470,7 +1470,7 @@ void CConnman::SocketEvents(std::set &recv_set, std::set &send_s } wakeupSelectNeeded = true; - int r = poll(vpollfds.data(), vpollfds.size(), SELECT_TIMEOUT_MILLISECONDS); + int r = poll(vpollfds.data(), vpollfds.size(), fOnlyPoll ? 0 : SELECT_TIMEOUT_MILLISECONDS); wakeupSelectNeeded = false; if (r < 0) { return; @@ -1485,7 +1485,7 @@ void CConnman::SocketEvents(std::set &recv_set, std::set &send_s } } #else -void CConnman::SocketEvents(std::set &recv_set, std::set &send_set, std::set &error_set) +void CConnman::SocketEvents(std::set &recv_set, std::set &send_set, std::set &error_set, bool fOnlyPoll) { std::set recv_select_set, send_select_set, error_select_set; if (!GenerateSelectSet(recv_select_set, send_select_set, error_select_set)) { @@ -1498,7 +1498,7 @@ void CConnman::SocketEvents(std::set &recv_set, std::set &send_s // struct timeval timeout; timeout.tv_sec = 0; - timeout.tv_usec = SELECT_TIMEOUT_MILLISECONDS * 1000; // frequency to poll pnode->vSend + timeout.tv_usec = fOnlyPoll ? 0 : SELECT_TIMEOUT_MILLISECONDS * 1000; // frequency to poll pnode->vSend fd_set fdsetRecv; fd_set fdsetSend; @@ -1563,8 +1563,26 @@ void CConnman::SocketEvents(std::set &recv_set, std::set &send_s void CConnman::SocketHandler() { + bool fOnlyPoll = false; + { + // check if we have work to do and thus should avoid waiting for events + LOCK2(cs_vNodes, cs_mapNodesWithDataToSend); + if (!mapReceivableNodes.empty()) { + fOnlyPoll = true; + } else if (!mapSendableNodes.empty() && !mapNodesWithDataToSend.empty()) { + // we must check if at least one of the nodes with pending messages is also sendable, as otherwise a single + // node would be able to make the network thread busy with polling + for (auto& p : mapNodesWithDataToSend) { + if (mapSendableNodes.count(p.first)) { + fOnlyPoll = true; + break; + } + } + } + } + std::set recv_set, send_set, error_set; - SocketEvents(recv_set, send_set, error_set); + SocketEvents(recv_set, send_set, error_set, fOnlyPoll); #ifdef USE_WAKEUP_PIPE // drain the wakeup pipe diff --git a/src/net.h b/src/net.h index 094a7d4d9754c4..f66962ddcdbc22 100644 --- a/src/net.h +++ b/src/net.h @@ -471,7 +471,7 @@ friend class CNode; void NotifyNumConnectionsChanged(); void InactivityCheck(CNode *pnode); bool GenerateSelectSet(std::set &recv_set, std::set &send_set, std::set &error_set); - void SocketEvents(std::set &recv_set, std::set &send_set, std::set &error_set); + void SocketEvents(std::set &recv_set, std::set &send_set, std::set &error_set, bool fOnlyPoll); void SocketHandler(); void ThreadSocketHandler(); void ThreadDNSAddressSeed();