diff options
author | jam@chromium.org <jam@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98> | 2008-12-05 20:27:20 +0000 |
---|---|---|
committer | jam@chromium.org <jam@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98> | 2008-12-05 20:27:20 +0000 |
commit | d3ae7a0705a7f1a11831731a799f337055d54a85 (patch) | |
tree | 6b9fdb5767392191d6e53993e1e9bba6efa769f2 | |
parent | 2ea7075326a997cc81f3d84b044c2ca39be8af19 (diff) | |
download | chromium_src-d3ae7a0705a7f1a11831731a799f337055d54a85.zip chromium_src-d3ae7a0705a7f1a11831731a799f337055d54a85.tar.gz chromium_src-d3ae7a0705a7f1a11831731a799f337055d54a85.tar.bz2 |
Fix a race condition in the SyncChannel code.
The problem is that QueueMessage would be called with listener() on the IPC thread, and then the function would wait on message_lock_ before adding the message and its associated listener. However, in between the time that listener() was called and the lock was taken control of, the listener thread may be calling RemoveListener.
The fix is to not send the listener, but instead just send a pointer to the context. It'll be used on the listener thread, at which point listener() can be called to check if the listener is available.
BUG=1305036
Review URL: http://codereview.chromium.org/12952
git-svn-id: svn://svn.chromium.org/chrome/trunk/src@6448 0039d316-1c4b-4281-b951-d872f2087c98
-rw-r--r-- | chrome/common/ipc_channel_proxy.h | 7 | ||||
-rw-r--r-- | chrome/common/ipc_sync_channel.cc | 91 |
2 files changed, 37 insertions, 61 deletions
diff --git a/chrome/common/ipc_channel_proxy.h b/chrome/common/ipc_channel_proxy.h index 6b9d777..2b08c09 100644 --- a/chrome/common/ipc_channel_proxy.h +++ b/chrome/common/ipc_channel_proxy.h @@ -128,8 +128,10 @@ class ChannelProxy : public Message::Sender { MessageLoop* ipc_thread); virtual ~Context() { } MessageLoop* ipc_message_loop() const { return ipc_message_loop_; } + Channel::Listener* listener() const { return listener_; } + const std::wstring& channel_id() const { return channel_id_; } - protected: + protected: // IPC::Channel::Listener methods: virtual void OnMessageReceived(const Message& message); virtual void OnChannelConnected(int32 peer_pid); @@ -138,9 +140,6 @@ class ChannelProxy : public Message::Sender { // Like OnMessageReceived but doesn't try the filters. void OnMessageReceivedNoFilter(const Message& message); - Channel::Listener* listener() const { return listener_; } - const std::wstring& channel_id() const { return channel_id_; } - // Gives the filters a chance at processing |message|. // Returns true if the message was processed, false otherwise. bool TryFilters(const Message& message); diff --git a/chrome/common/ipc_sync_channel.cc b/chrome/common/ipc_sync_channel.cc index 68402b8..45e42d6 100644 --- a/chrome/common/ipc_sync_channel.cc +++ b/chrome/common/ipc_sync_channel.cc @@ -40,8 +40,8 @@ class SyncChannel::ReceivedSyncMsgQueue : public base::RefCountedThreadSafe<ReceivedSyncMsgQueue> { public: // Returns the ReceivedSyncMsgQueue instance for this thread, creating one - // if necessary. Call RemoveListener on the same thread when done. - static ReceivedSyncMsgQueue* AddListener() { + // if necessary. Call RemoveContext on the same thread when done. + static ReceivedSyncMsgQueue* AddContext() { // We want one ReceivedSyncMsgQueue per listener thread (i.e. since multiple // SyncChannel objects can block the same thread). ReceivedSyncMsgQueue* rv = lazy_tls_ptr_.Pointer()->Get(); @@ -57,8 +57,7 @@ class SyncChannel::ReceivedSyncMsgQueue : } // Called on IPC thread when a synchronous message or reply arrives. - void QueueMessage(const Message& msg, Channel::Listener* listener, - const std::wstring& channel_id) { + void QueueMessage(const Message& msg, SyncChannel::SyncContext* context) { bool was_task_pending; { AutoLock auto_lock(message_lock_); @@ -68,8 +67,7 @@ class SyncChannel::ReceivedSyncMsgQueue : // We set the event in case the listener thread is blocked (or is about // to). In case it's not, the PostTask dispatches the messages. - message_queue_.push(ReceivedMessage(new Message(msg), listener, - channel_id)); + message_queue_.push_back(QueuedMessage(new Message(msg), context)); } SetEvent(dispatch_event_); @@ -80,10 +78,10 @@ class SyncChannel::ReceivedSyncMsgQueue : } void QueueReply(const Message &msg, SyncChannel::SyncContext* context) { - received_replies_.push_back(Reply(new Message(msg), context)); + received_replies_.push_back(QueuedMessage(new Message(msg), context)); } - // Called on the listerner's thread to process any queues synchronous + // Called on the listener's thread to process any queues synchronous // messages. void DispatchMessagesTask() { { @@ -95,19 +93,16 @@ class SyncChannel::ReceivedSyncMsgQueue : void DispatchMessages() { while (true) { - Message* message = NULL; - std::wstring channel_id; - Channel::Listener* listener = NULL; + Message* message; + scoped_refptr<SyncChannel::SyncContext> context; { AutoLock auto_lock(message_lock_); if (message_queue_.empty()) break; - ReceivedMessage& blocking_msg = message_queue_.front(); - message = blocking_msg.message; - listener = blocking_msg.listener; - channel_id = blocking_msg.channel_id; - message_queue_.pop(); + message = message_queue_.front().message; + context = message_queue_.front().context; + message_queue_.pop_front(); } #ifdef IPC_MESSAGE_LOG_ENABLED @@ -116,12 +111,12 @@ class SyncChannel::ReceivedSyncMsgQueue : logger->OnPreDispatchMessage(*message); #endif - if (listener) - listener->OnMessageReceived(*message); + if (context->listener()) + context->listener()->OnMessageReceived(*message); #ifdef IPC_MESSAGE_LOG_ENABLED if (logger->Enabled()) - logger->OnPostDispatchMessage(*message, channel_id); + logger->OnPostDispatchMessage(*message, context->channel_id()); #endif delete message; @@ -129,23 +124,17 @@ class SyncChannel::ReceivedSyncMsgQueue : } // SyncChannel calls this in its destructor. - void RemoveListener(Channel::Listener* listener) { + void RemoveContext(SyncContext* context) { AutoLock auto_lock(message_lock_); - SyncMessageQueue temp_queue; - while (!message_queue_.empty()) { - if (message_queue_.front().listener != listener) { - temp_queue.push(message_queue_.front()); + SyncMessageQueue::iterator iter = message_queue_.begin(); + while (iter != message_queue_.end()) { + if (iter->context == context) { + delete iter->message; + iter = message_queue_.erase(iter); } else { - delete message_queue_.front().message; + iter++; } - - message_queue_.pop(); - } - - while (!temp_queue.empty()) { - message_queue_.push(temp_queue.front()); - temp_queue.pop(); } if (--listener_count_ == 0) { @@ -184,29 +173,17 @@ class SyncChannel::ReceivedSyncMsgQueue : listener_count_(0) { } - // Holds information about a queued synchronous message. - struct ReceivedMessage { - ReceivedMessage(Message* m, Channel::Listener* l, const std::wstring& i) - : message(m), listener(l), channel_id(i) { } - Message* message; - Channel::Listener* listener; - std::wstring channel_id; - }; - - typedef std::queue<ReceivedMessage> SyncMessageQueue; - SyncMessageQueue message_queue_; - - // Holds information about a queued reply message. - struct Reply { - Reply(Message* m, SyncChannel::SyncContext* c) - : message(m), - context(c) { } - + // Holds information about a queued synchronous message or reply. + struct QueuedMessage { + QueuedMessage(Message* m, SyncContext* c) : message(m), context(c) { } Message* message; scoped_refptr<SyncChannel::SyncContext> context; }; - std::vector<Reply> received_replies_; + typedef std::deque<QueuedMessage> SyncMessageQueue; + SyncMessageQueue message_queue_; + + std::vector<QueuedMessage> received_replies_; // Set when we got a synchronous message that we must respond to as the // sender needs its reply before it can reply to our original synchronous @@ -228,7 +205,7 @@ SyncChannel::SyncContext::SyncContext( HANDLE shutdown_event) : ChannelProxy::Context(listener, filter, ipc_thread), shutdown_event_(shutdown_event), - received_sync_msgs_(ReceivedSyncMsgQueue::AddListener()){ + received_sync_msgs_(ReceivedSyncMsgQueue::AddContext()){ } SyncChannel::SyncContext::~SyncContext() { @@ -304,7 +281,7 @@ bool SyncChannel::SyncContext::TryToUnblockListener(const Message* msg) { void SyncChannel::SyncContext::Clear() { CancelPendingSends(); - received_sync_msgs_->RemoveListener(listener()); + received_sync_msgs_->RemoveContext(this); Context::Clear(); } @@ -318,7 +295,7 @@ void SyncChannel::SyncContext::OnMessageReceived(const Message& msg) { return; if (msg.should_unblock()) { - received_sync_msgs_->QueueMessage(msg, listener(), channel_id()); + received_sync_msgs_->QueueMessage(msg, this); return; } @@ -349,8 +326,8 @@ void SyncChannel::SyncContext::OnSendTimeout(int message_id) { AutoLock auto_lock(deserializers_lock_); PendingSyncMessageQueue::iterator iter; for (iter = deserializers_.begin(); iter != deserializers_.end(); iter++) { - if ((*iter).id == message_id) { - SetEvent((*iter).done_event); + if (iter->id == message_id) { + SetEvent(iter->done_event); break; } } @@ -360,7 +337,7 @@ void SyncChannel::SyncContext::CancelPendingSends() { AutoLock auto_lock(deserializers_lock_); PendingSyncMessageQueue::iterator iter; for (iter = deserializers_.begin(); iter != deserializers_.end(); iter++) - SetEvent((*iter).done_event); + SetEvent(iter->done_event); } void SyncChannel::SyncContext::OnObjectSignaled(HANDLE object) { |