From b88147d0cdb523d177b9cf0116bce463d8a7358b Mon Sep 17 00:00:00 2001 From: kaetemi Date: Sat, 4 Mar 2023 17:06:52 +0800 Subject: [PATCH] Shutdown and close remaining QUIC connections on exit, ref ryzom/ryzomcore#628 --- .../src/frontend_service/quic_transceiver.cpp | 63 ++++++++++++++++++- .../src/frontend_service/quic_transceiver.h | 32 ++++++++++ 2 files changed, 93 insertions(+), 2 deletions(-) diff --git a/ryzom/server/src/frontend_service/quic_transceiver.cpp b/ryzom/server/src/frontend_service/quic_transceiver.cpp index cbb5c886a6..63787fb6bd 100644 --- a/ryzom/server/src/frontend_service/quic_transceiver.cpp +++ b/ryzom/server/src/frontend_service/quic_transceiver.cpp @@ -141,6 +141,9 @@ class CQuicTransceiverImpl std::mutex ListenerStopMutex; std::condition_variable ListenerStopCondition; + // Connections that have not shutdown yet + NLMISC::CSynchronized> Connections; + static QUIC_STATUS #ifdef _Function_class_ _Function_class_(QUIC_LISTENER_CALLBACK) @@ -403,6 +406,49 @@ void CQuicTransceiver::stop() m->Listener = null; nldebug("Listener closed"); } + + // Shutdown all connections + for (;;) // Doesn't need to loop, but just in case... + { + std::set connectionsCopy; + { + NLMISC::CSynchronized>::CAccessor connections(&m->Connections); + connectionsCopy = connections.value(); + } + if (connectionsCopy.empty()) + break; + + nldebug("Shutdown %i connections", (int)connectionsCopy.size()); + for (const CQuicUserContextPtr &user : connectionsCopy) + { + if (user->Connection) + { + MsQuic->ConnectionShutdown((HQUIC)user->Connection, QUIC_CONNECTION_SHUTDOWN_FLAG_NONE, 0); + } + } + // Wait for connections to be empty, lousy spin loop + nldebug("Wait for shutdown"); + for (;;) + { + NLMISC::CSynchronized>::CAccessor connections(&m->Connections); + if (connections.value().empty()) + break; + nlSleep(1); + } + nldebug("Shutdown ok"); + // Forcefully close all connections + for (const CQuicUserContextPtr &user : connectionsCopy) + { + if (user->Connection) + { + MsQuic->ConnectionClose((HQUIC)user->Connection); + user->Connection = nullptr; + } + } + nldebug("Closed all connections"); + connectionsCopy.clear(); + nldebug("Cleared references. There may be leftovers in CClientHost, etc"); + } // Clear queue nldebug("Clear current write queue"); @@ -477,7 +523,12 @@ _Function_class_(QUIC_LISTENER_CALLBACK) user->Transceiver = self; user->TokenAddr = self->generateTokenAddr(); // ev->NEW_CONNECTION.Info->RemoteAddress // Could change on migration, so don't expose it for now (OK in the future) user->Connection = ev->NEW_CONNECTION.Connection; - user->increaseRef(); + CQuicUserContextPtr userPtr = user; + { + NLMISC::CSynchronized>::CAccessor connections(&m->Connections); + connections.value().insert(userPtr); + } + // user->increaseRef(); // They're in. MsQuic->SetCallbackHandler(ev->NEW_CONNECTION.Connection, (void *)CQuicTransceiverImpl::connectionCallback, (void *)user); status = MsQuic->ConnectionSetConfiguration(ev->NEW_CONNECTION.Connection, m->Configuration); @@ -535,7 +586,12 @@ _Function_class_(QUIC_CONNECTION_CALLBACK) status = QUIC_STATUS_SUCCESS; break; case QUIC_CONNECTION_EVENT_SHUTDOWN_COMPLETE: { - CQuicUserContextRelease releaseUser(user); // Hopefully we only get QUIC_CONNECTION_EVENT_SHUTDOWN_COMPLETE once! + // CQuicUserContextRelease releaseUser(user); // Hopefully we only get QUIC_CONNECTION_EVENT_SHUTDOWN_COMPLETE once! + CQuicUserContextPtr userPtr = user; + { + NLMISC::CSynchronized>::CAccessor connections(&m->Connections); + connections.value().erase(userPtr); + } #ifdef FE_DEBUG_QUIC user->DebugRefCount = true; #endif @@ -659,6 +715,7 @@ NLMISC::CBufFIFO *CQuicTransceiver::swapWriteQueue(NLMISC::CBufFIFO *writeQueue) void CQuicTransceiver::clearQueue(NLMISC::CBufFIFO *queue) { CAtomicFlagLockYield lock(m->BufferMutex); + int count = 0; while (!queue->empty()) { // Data, don't care @@ -674,7 +731,9 @@ void CQuicTransceiver::clearQueue(NLMISC::CBufFIFO *queue) // Decrease ref count after pop CQuicUserContextRelease releaseUser(user); queue->pop(); + ++count; } + nldebug("Cleared %i messages from queue [%p]", count, queue); } // void CQuicTransceiver::sendDatagram(CQuicUserContext *user, const uint8 *buffer, uint32 size) diff --git a/ryzom/server/src/frontend_service/quic_transceiver.h b/ryzom/server/src/frontend_service/quic_transceiver.h index 878a8e5a9a..2b1f0bac24 100644 --- a/ryzom/server/src/frontend_service/quic_transceiver.h +++ b/ryzom/server/src/frontend_service/quic_transceiver.h @@ -217,6 +217,38 @@ class CQuicUserContextPtr return m_User; } + // compare + bool operator==(const CQuicUserContextPtr &other) const + { + return m_User == other.m_User; + } + + bool operator!=(const CQuicUserContextPtr &other) const + { + return m_User != other.m_User; + } + + bool operator==(const CQuicUserContext *other) const + { + return m_User == other; + } + + bool operator!=(const CQuicUserContext *other) const + { + return m_User != other; + } + + // compare less + bool operator<(const CQuicUserContextPtr &other) const + { + return m_User < other.m_User; + } + + bool operator<(const CQuicUserContext *other) const + { + return m_User < other; + } + private: CQuicUserContext *m_User; };