From c800ab2d04bb6c6d5e6a5617137f358a8f3d91ea Mon Sep 17 00:00:00 2001 From: daniuk-microsoft Date: Tue, 2 Aug 2016 11:38:02 -0700 Subject: [PATCH] Fixed race condition in Casablanca ws_client_winrt.cpp Fixed race condition in Casablanca ws_client_winrt.cpp - Simplified message queuing logic to always use the outgoing queue - Removed m_num_sends member variable --- .../src/websockets/client/ws_client_winrt.cpp | 46 ++++++++++++------- 1 file changed, 29 insertions(+), 17 deletions(-) diff --git a/Release/src/websockets/client/ws_client_winrt.cpp b/Release/src/websockets/client/ws_client_winrt.cpp index a35f269adf..035f9b02cd 100644 --- a/Release/src/websockets/client/ws_client_winrt.cpp +++ b/Release/src/websockets/client/ws_client_winrt.cpp @@ -85,8 +85,7 @@ class winrt_callback_client : public websocket_client_callback_impl, public std: public: winrt_callback_client(websocket_client_config config) : websocket_client_callback_impl(std::move(config)), - m_connected(false), - m_num_sends(0) + m_connected(false) { m_msg_websocket = ref new MessageWebSocket(); @@ -243,17 +242,24 @@ class winrt_callback_client : public websocket_client_callback_impl, public std: return pplx::task_from_exception(websocket_exception("Message size too large. Ensure message length is less than UINT_MAX.")); } - if (++m_num_sends == 1) // No sends in progress + bool msg_pending = false; { - // Start sending the message - send_msg(msg); - } - else - { - // Only actually have to take the lock if touching the queue. std::lock_guard lock(m_send_lock); + if (m_outgoing_msg_queue.size() > 0) + { + msg_pending = true; + } + m_outgoing_msg_queue.push(msg); } + + // No sends in progress + if (msg_pending == false) + { + // Start sending the message + send_msg(msg); + } + return pplx::create_task(msg.body_sent()); } @@ -385,15 +391,24 @@ class winrt_callback_client : public websocket_client_callback_impl, public std: msg.signal_body_sent(); } - if (--this_client->m_num_sends > 0) + bool msg_pending = false; + websocket_outgoing_message next_msg; { // Only hold the lock when actually touching the queue. - websocket_outgoing_message next_msg; + std::lock_guard lock(this_client->m_send_lock); + + // First message in queue has been sent + this_client->m_outgoing_msg_queue.pop(); + + if (this_client->m_outgoing_msg_queue.size() > 0) { - std::lock_guard lock(this_client->m_send_lock); next_msg = this_client->m_outgoing_msg_queue.front(); - this_client->m_outgoing_msg_queue.pop(); + msg_pending = true; } + } + + if (msg_pending) + { this_client->send_msg(next_msg); } }); @@ -443,11 +458,8 @@ class winrt_callback_client : public websocket_client_callback_impl, public std: // The implementation has to ensure ordering of send requests std::mutex m_send_lock; - // Queue to order the sends + // Queue to track pending sends std::queue m_outgoing_msg_queue; - - // Number of sends in progress and queued up. - std::atomic m_num_sends; }; void ReceiveContext::OnReceive(MessageWebSocket^ sender, MessageWebSocketMessageReceivedEventArgs^ args)