From 12ffa2437e4c7dc5b94287bf30101d3f74c1e7fd Mon Sep 17 00:00:00 2001 From: Dillon Skaggs Date: Sun, 5 Jan 2025 02:35:32 -0600 Subject: [PATCH] tweak(mumble): swap to completely using TCP ping for checking if UDP is available - The previous implementation would constantly swap between UDP/TCP constantly causing the players mic to stutter - The old implementation should've been completely removed, this is an oversight in https://github.com/citizenfx/fivem/commit/f182bb5176e37ed996d27fdb87170a4097d6025b --- .../voip-mumble/include/MumbleClientImpl.h | 7 ++++- .../voip-mumble/src/MumbleClient.cpp | 26 ++++++++++--------- 2 files changed, 20 insertions(+), 13 deletions(-) diff --git a/code/components/voip-mumble/include/MumbleClientImpl.h b/code/components/voip-mumble/include/MumbleClientImpl.h index a204e1dcd8..6d49dfeae7 100644 --- a/code/components/voip-mumble/include/MumbleClientImpl.h +++ b/code/components/voip-mumble/include/MumbleClientImpl.h @@ -239,7 +239,12 @@ class MumbleClient : public IMumbleClient, public Botan::TLS::Callbacks bool m_hasUdp = false; - std::chrono::milliseconds m_lastUdp; + bool m_udpTimedOut = false; + + // the time in milliseconds since the player joined the mumble server + // This is used in for `Ping` packets to determine if we should allow the client to swap from UDP -> TCP + // By default we wont swap back to TCP for the first 20 seconds of the clients connection (only after we have UDP) + std::chrono::milliseconds m_timeSinceJoin; std::chrono::milliseconds m_nextPing; diff --git a/code/components/voip-mumble/src/MumbleClient.cpp b/code/components/voip-mumble/src/MumbleClient.cpp index bf804324f5..d4d5044e5a 100644 --- a/code/components/voip-mumble/src/MumbleClient.cpp +++ b/code/components/voip-mumble/src/MumbleClient.cpp @@ -24,7 +24,6 @@ static __declspec(thread) MumbleClient* g_currentMumbleClient; using namespace std::chrono_literals; -constexpr auto kUDPTimeout = 5000ms; constexpr auto kUDPPingInterval = 1000ms; constexpr uint16_t kMaxUdpPacket = 1024; @@ -39,7 +38,6 @@ void MumbleClient::Initialize() m_voiceTarget = 0; - m_lastUdp = {}; m_nextPing = {}; m_loop = Instance::Get()->GetOrCreate("mumble"); @@ -116,6 +114,7 @@ void MumbleClient::Initialize() // don't start idle timer here - it should only start after TLS handshake is done! + m_timeSinceJoin = msec(); m_connectionInfo.isConnected = true; }); @@ -166,11 +165,6 @@ void MumbleClient::Initialize() m_idleTimer = m_loop->Get()->resource(); m_idleTimer->on([this](const uvw::TimerEvent& ev, uvw::TimerHandle& t) { - if (m_hasUdp && (msec() - m_lastUdp) > kUDPTimeout) - { - m_hasUdp = false; - console::PrintWarning("mumble", "Server isn't responding to UDP packets, swapping to TCP.\n"); - } auto lockedIsActive = [this]() { @@ -428,7 +422,6 @@ concurrency::task MumbleClient::ConnectAsync(const net::P m_tcpPingCount = 0; - m_lastUdp = {}; memset(m_tcpPings, 0, sizeof(m_tcpPings)); @@ -782,8 +775,6 @@ void MumbleClient::HandleUDP(const uint8_t* buf, size_t size) return; } - // update UDP timestamp - m_lastUdp = msec(); // handle voice packet HandleVoice(outBuf, size - 4); @@ -971,10 +962,21 @@ void MumbleClient::HandlePing(const MumbleProto::Ping& ping) m_crypto.m_remoteLost = ping.lost(); m_crypto.m_remoteResync = ping.resync(); - if (m_hasUdp && (m_crypto.m_remoteGood == 0 || m_crypto.m_localGood == 0) && (msec() - m_lastUdp) > 2s) + if (m_hasUdp && (m_crypto.m_remoteGood == 0 || m_crypto.m_localGood == 0) && (msec() - m_timeSinceJoin) > 20s) { - console::PrintWarning("mumble", "UDP packets can *not* be received. Switching to TCP tunnel mode.\n"); m_hasUdp = false; + if (m_crypto.m_remoteGood == 0 && m_crypto.m_localGood == 0) + { + console::PrintWarning("mumble", "The server couldn't send or receive the clients UDP packets. Switching to TCP mode."); + } + else if (m_crypto.m_remoteGood == 0) + { + console::PrintWarning("mumble", "The clients UDP packets are not being received by the server. Switching to TCP mode."); + } + else + { + console::PrintWarning("mumble", "The client isn't receiving UDP packets. Switching to TCP mode."); + } } else if (!m_hasUdp && m_crypto.m_remoteGood > 3 && m_crypto.m_localGood > 3) {