| OLD | NEW |
| 1 // Copyright 2015 The Chromium Authors. All rights reserved. | 1 // Copyright 2015 The Chromium Authors. All rights reserved. |
| 2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
| 3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
| 4 | 4 |
| 5 #include "ipc/mojo/scoped_ipc_support.h" | 5 #include "ipc/mojo/scoped_ipc_support.h" |
| 6 | 6 |
| 7 #include "base/bind.h" | 7 #include "base/bind.h" |
| 8 #include "base/lazy_instance.h" | 8 #include "base/lazy_instance.h" |
| 9 #include "base/logging.h" | 9 #include "base/logging.h" |
| 10 #include "base/memory/weak_ptr.h" | |
| 11 #include "base/message_loop/message_loop.h" | 10 #include "base/message_loop/message_loop.h" |
| 12 #include "base/synchronization/condition_variable.h" | 11 #include "base/synchronization/condition_variable.h" |
| 13 #include "base/synchronization/lock.h" | 12 #include "base/synchronization/lock.h" |
| 14 #include "base/synchronization/waitable_event.h" | 13 #include "base/synchronization/waitable_event.h" |
| 15 #include "base/thread_task_runner_handle.h" | 14 #include "base/thread_task_runner_handle.h" |
| 16 #include "third_party/mojo/src/mojo/edk/embedder/embedder.h" | 15 #include "third_party/mojo/src/mojo/edk/embedder/embedder.h" |
| 17 #include "third_party/mojo/src/mojo/edk/embedder/process_delegate.h" | 16 #include "third_party/mojo/src/mojo/edk/embedder/process_delegate.h" |
| 18 | 17 |
| 19 namespace IPC { | 18 namespace IPC { |
| 20 | 19 |
| 21 namespace { | 20 namespace { |
| 22 | 21 |
| 23 class IPCSupportInitializer : public mojo::embedder::ProcessDelegate { | 22 class IPCSupportInitializer : public mojo::embedder::ProcessDelegate { |
| 24 public: | 23 public: |
| 25 IPCSupportInitializer() | 24 IPCSupportInitializer() |
| 26 : init_count_(0), | 25 : init_count_(0), |
| 27 shutting_down_(false), | 26 shutting_down_(false), |
| 28 was_shut_down_(false), | 27 was_shut_down_(false), |
| 29 observer_(nullptr), | 28 observer_(nullptr) {} |
| 30 weak_factory_(this) {} | |
| 31 | 29 |
| 32 ~IPCSupportInitializer() override { DCHECK(!observer_); } | 30 ~IPCSupportInitializer() override { DCHECK(!observer_); } |
| 33 | 31 |
| 34 void Init(scoped_refptr<base::TaskRunner> io_thread_task_runner); | 32 void Init(scoped_refptr<base::TaskRunner> io_thread_task_runner); |
| 35 void ShutDown(); | 33 void ShutDown(); |
| 36 | 34 |
| 37 // Forces the initializer to shut down even if scopers are still holding it. | 35 // Forces the initializer to shut down even if scopers are still holding it. |
| 38 void ForceShutdown(); | 36 void ForceShutdown(); |
| 39 | 37 |
| 40 private: | 38 private: |
| 41 // This watches for destruction of the MessageLoop that IPCSupportInitializer | 39 // This watches for destruction of the MessageLoop that IPCSupportInitializer |
| 42 // uses for IO, and guarantees that the initializer is shut down if it still | 40 // uses for IO, and guarantees that the initializer is shut down if it still |
| 43 // exists when the loop is being destroyed. | 41 // exists when the loop is being destroyed. |
| 44 class MessageLoopObserver : public base::MessageLoop::DestructionObserver { | 42 class MessageLoopObserver : public base::MessageLoop::DestructionObserver { |
| 45 public: | 43 public: |
| 46 MessageLoopObserver( | 44 MessageLoopObserver(IPCSupportInitializer* initializer) |
| 47 scoped_refptr<base::TaskRunner> initializer_task_runner, | 45 : initializer_(initializer) {} |
| 48 base::WeakPtr<IPCSupportInitializer> weak_initializer) | |
| 49 : initializer_task_runner_(initializer_task_runner), | |
| 50 weak_initializer_(weak_initializer) {} | |
| 51 | 46 |
| 52 ~MessageLoopObserver() override { | 47 ~MessageLoopObserver() override { |
| 53 base::MessageLoop::current()->RemoveDestructionObserver(this); | 48 base::MessageLoop::current()->RemoveDestructionObserver(this); |
| 54 } | 49 } |
| 55 | 50 |
| 56 private: | 51 private: |
| 57 // base::MessageLoop::DestructionObserver: | 52 // base::MessageLoop::DestructionObserver: |
| 58 void WillDestroyCurrentMessageLoop() override { | 53 void WillDestroyCurrentMessageLoop() override { |
| 59 initializer_task_runner_->PostTask( | 54 initializer_->ForceShutdown(); |
| 60 FROM_HERE, | |
| 61 base::Bind(&IPCSupportInitializer::ForceShutdown, weak_initializer_)); | |
| 62 } | 55 } |
| 63 | 56 |
| 64 scoped_refptr<base::TaskRunner> initializer_task_runner_; | 57 IPCSupportInitializer* initializer_; |
| 65 base::WeakPtr<IPCSupportInitializer> weak_initializer_; | |
| 66 | 58 |
| 67 DISALLOW_COPY_AND_ASSIGN(MessageLoopObserver); | 59 DISALLOW_COPY_AND_ASSIGN(MessageLoopObserver); |
| 68 }; | 60 }; |
| 69 | 61 |
| 70 void ShutDownOnIOThread(); | 62 void ShutDownOnIOThread(); |
| 71 | 63 |
| 72 // mojo::embedder::ProcessDelegate: | 64 // mojo::embedder::ProcessDelegate: |
| 73 void OnShutdownComplete() override {} | 65 void OnShutdownComplete() override {} |
| 74 | 66 |
| 75 static void WatchMessageLoopOnIOThread(MessageLoopObserver* observer); | 67 static void WatchMessageLoopOnIOThread(MessageLoopObserver* observer); |
| 76 | 68 |
| 77 base::Lock lock_; | 69 base::Lock lock_; |
| 78 size_t init_count_; | 70 size_t init_count_; |
| 79 bool shutting_down_; | 71 bool shutting_down_; |
| 80 | 72 |
| 81 // This is used to track whether shutdown has occurred yet, since we can be | 73 // This is used to track whether shutdown has occurred yet, since we can be |
| 82 // shut down by either the scoper or IO MessageLoop destruction. | 74 // shut down by either the scoper or IO MessageLoop destruction. |
| 83 bool was_shut_down_; | 75 bool was_shut_down_; |
| 84 | 76 |
| 85 // The message loop destruction observer we have watching our IO loop. This | 77 // The message loop destruction observer we have watching our IO loop. This |
| 86 // is created on the initializer's own thread but is used and destroyed on the | 78 // is created on the initializer's own thread but is used and destroyed on the |
| 87 // IO thread. | 79 // IO thread. |
| 88 MessageLoopObserver* observer_; | 80 MessageLoopObserver* observer_; |
| 89 | 81 |
| 90 scoped_refptr<base::TaskRunner> io_thread_task_runner_; | 82 scoped_refptr<base::TaskRunner> io_thread_task_runner_; |
| 91 | 83 |
| 92 base::WeakPtrFactory<IPCSupportInitializer> weak_factory_; | |
| 93 | |
| 94 DISALLOW_COPY_AND_ASSIGN(IPCSupportInitializer); | 84 DISALLOW_COPY_AND_ASSIGN(IPCSupportInitializer); |
| 95 }; | 85 }; |
| 96 | 86 |
| 97 void IPCSupportInitializer::Init( | 87 void IPCSupportInitializer::Init( |
| 98 scoped_refptr<base::TaskRunner> io_thread_task_runner) { | 88 scoped_refptr<base::TaskRunner> io_thread_task_runner) { |
| 99 base::AutoLock locker(lock_); | 89 base::AutoLock locker(lock_); |
| 100 DCHECK((init_count_ == 0 && !io_thread_task_runner_) || | 90 DCHECK((init_count_ == 0 && !io_thread_task_runner_) || |
| 101 io_thread_task_runner_ == io_thread_task_runner); | 91 io_thread_task_runner_ == io_thread_task_runner); |
| 102 | 92 |
| 103 if (shutting_down_) { | 93 if (shutting_down_) { |
| 104 // If reinitialized before a pending shutdown task is executed, we | 94 // If reinitialized before a pending shutdown task is executed, we |
| 105 // effectively cancel the shutdown task. | 95 // effectively cancel the shutdown task. |
| 106 DCHECK(init_count_ == 1); | 96 DCHECK(init_count_ == 1); |
| 107 shutting_down_ = false; | 97 shutting_down_ = false; |
| 108 return; | 98 return; |
| 109 } | 99 } |
| 110 | 100 |
| 111 init_count_++; | 101 init_count_++; |
| 112 if (init_count_ == 1) { | 102 if (init_count_ == 1) { |
| 113 was_shut_down_ = false; | 103 was_shut_down_ = false; |
| 114 observer_ = new MessageLoopObserver(base::ThreadTaskRunnerHandle::Get(), | 104 observer_ = new MessageLoopObserver(this); |
| 115 weak_factory_.GetWeakPtr()); | |
| 116 io_thread_task_runner_ = io_thread_task_runner; | 105 io_thread_task_runner_ = io_thread_task_runner; |
| 117 io_thread_task_runner_->PostTask( | 106 io_thread_task_runner_->PostTask( |
| 118 FROM_HERE, base::Bind(&WatchMessageLoopOnIOThread, observer_)); | 107 FROM_HERE, base::Bind(&WatchMessageLoopOnIOThread, observer_)); |
| 119 mojo::embedder::InitIPCSupport( | 108 mojo::embedder::InitIPCSupport( |
| 120 mojo::embedder::ProcessType::NONE, io_thread_task_runner_, this, | 109 mojo::embedder::ProcessType::NONE, io_thread_task_runner_, this, |
| 121 io_thread_task_runner_, mojo::embedder::ScopedPlatformHandle()); | 110 io_thread_task_runner_, mojo::embedder::ScopedPlatformHandle()); |
| 122 } | 111 } |
| 123 } | 112 } |
| 124 | 113 |
| 125 void IPCSupportInitializer::ShutDown() { | 114 void IPCSupportInitializer::ShutDown() { |
| (...skipping 54 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 180 ScopedIPCSupport::ScopedIPCSupport( | 169 ScopedIPCSupport::ScopedIPCSupport( |
| 181 scoped_refptr<base::TaskRunner> io_thread_task_runner) { | 170 scoped_refptr<base::TaskRunner> io_thread_task_runner) { |
| 182 ipc_support_initializer.Get().Init(io_thread_task_runner); | 171 ipc_support_initializer.Get().Init(io_thread_task_runner); |
| 183 } | 172 } |
| 184 | 173 |
| 185 ScopedIPCSupport::~ScopedIPCSupport() { | 174 ScopedIPCSupport::~ScopedIPCSupport() { |
| 186 ipc_support_initializer.Get().ShutDown(); | 175 ipc_support_initializer.Get().ShutDown(); |
| 187 } | 176 } |
| 188 | 177 |
| 189 } // namespace IPC | 178 } // namespace IPC |
| OLD | NEW |