OLD | NEW |
1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. | 1 // Copyright (c) 2012 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 "jingle/glue/thread_wrapper.h" | 5 #include "jingle/glue/thread_wrapper.h" |
6 | 6 |
7 #include "base/bind.h" | 7 #include "base/bind.h" |
8 #include "base/bind_helpers.h" | 8 #include "base/bind_helpers.h" |
9 #include "base/lazy_instance.h" | 9 #include "base/lazy_instance.h" |
10 #include "base/threading/thread_local.h" | 10 #include "base/threading/thread_local.h" |
(...skipping 11 matching lines...) Expand all Loading... |
22 | 22 |
23 JingleThreadWrapper* sending_thread; | 23 JingleThreadWrapper* sending_thread; |
24 talk_base::Message message; | 24 talk_base::Message message; |
25 base::WaitableEvent done_event; | 25 base::WaitableEvent done_event; |
26 }; | 26 }; |
27 | 27 |
28 base::LazyInstance<base::ThreadLocalPointer<JingleThreadWrapper> > | 28 base::LazyInstance<base::ThreadLocalPointer<JingleThreadWrapper> > |
29 g_jingle_thread_wrapper = LAZY_INSTANCE_INITIALIZER; | 29 g_jingle_thread_wrapper = LAZY_INSTANCE_INITIALIZER; |
30 | 30 |
31 // static | 31 // static |
32 void JingleThreadWrapper::EnsureForCurrentThread() { | 32 void JingleThreadWrapper::EnsureForCurrentMessageLoop() { |
33 if (JingleThreadWrapper::current() == NULL) { | 33 if (JingleThreadWrapper::current() == NULL) { |
34 g_jingle_thread_wrapper.Get().Set( | 34 MessageLoop* message_loop = MessageLoop::current(); |
35 new JingleThreadWrapper(MessageLoop::current())); | 35 g_jingle_thread_wrapper.Get().Set(new JingleThreadWrapper( |
| 36 message_loop->message_loop_proxy())); |
| 37 message_loop->AddDestructionObserver(current()); |
36 } | 38 } |
37 | 39 |
38 DCHECK_EQ(talk_base::Thread::Current(), current()); | 40 DCHECK_EQ(talk_base::Thread::Current(), current()); |
39 } | 41 } |
40 | 42 |
41 // static | 43 // static |
42 JingleThreadWrapper* JingleThreadWrapper::current() { | 44 JingleThreadWrapper* JingleThreadWrapper::current() { |
43 return g_jingle_thread_wrapper.Get().Get(); | 45 return g_jingle_thread_wrapper.Get().Get(); |
44 } | 46 } |
45 | 47 |
46 JingleThreadWrapper::JingleThreadWrapper(MessageLoop* message_loop) | 48 JingleThreadWrapper::JingleThreadWrapper( |
| 49 scoped_refptr<base::SingleThreadTaskRunner> task_runner) |
47 : talk_base::Thread(new talk_base::NullSocketServer()), | 50 : talk_base::Thread(new talk_base::NullSocketServer()), |
48 message_loop_(message_loop), | 51 task_runner_(task_runner), |
49 send_allowed_(false), | 52 send_allowed_(false), |
50 last_task_id_(0), | 53 last_task_id_(0), |
51 pending_send_event_(true, false) { | 54 pending_send_event_(true, false), |
52 DCHECK_EQ(message_loop_, MessageLoop::current()); | 55 weak_ptr_factory_(this), |
53 | 56 weak_ptr_(weak_ptr_factory_.GetWeakPtr()) { |
54 talk_base::ThreadManager::Instance()->UnwrapCurrentThread(); | 57 DCHECK(task_runner->BelongsToCurrentThread()); |
55 talk_base::ThreadManager::Instance()->SetCurrentThread(this); | 58 DCHECK(!talk_base::Thread::Current()); |
56 talk_base::MessageQueueManager::Instance()->Add(this); | 59 talk_base::MessageQueueManager::Instance()->Add(this); |
57 message_loop_->AddDestructionObserver(this); | |
58 | |
59 WrapCurrent(); | 60 WrapCurrent(); |
60 } | 61 } |
61 | 62 |
62 JingleThreadWrapper::~JingleThreadWrapper() { | 63 JingleThreadWrapper::~JingleThreadWrapper() { |
63 Clear(NULL, talk_base::MQID_ANY, NULL); | 64 Clear(NULL, talk_base::MQID_ANY, NULL); |
64 } | 65 } |
65 | 66 |
66 void JingleThreadWrapper::WillDestroyCurrentMessageLoop() { | 67 void JingleThreadWrapper::WillDestroyCurrentMessageLoop() { |
67 DCHECK_EQ(talk_base::Thread::Current(), current()); | 68 DCHECK_EQ(talk_base::Thread::Current(), current()); |
68 UnwrapCurrent(); | 69 UnwrapCurrent(); |
69 g_jingle_thread_wrapper.Get().Set(NULL); | 70 g_jingle_thread_wrapper.Get().Set(NULL); |
70 talk_base::ThreadManager::Instance()->SetCurrentThread(NULL); | 71 talk_base::ThreadManager::Instance()->SetCurrentThread(NULL); |
71 talk_base::MessageQueueManager::Instance()->Remove(this); | 72 talk_base::MessageQueueManager::Instance()->Remove(this); |
72 message_loop_->RemoveDestructionObserver(this); | |
73 talk_base::SocketServer* ss = socketserver(); | 73 talk_base::SocketServer* ss = socketserver(); |
74 delete this; | 74 delete this; |
75 delete ss; | 75 delete ss; |
76 } | 76 } |
77 | 77 |
78 void JingleThreadWrapper::Post( | 78 void JingleThreadWrapper::Post( |
79 talk_base::MessageHandler* handler, uint32 message_id, | 79 talk_base::MessageHandler* handler, uint32 message_id, |
80 talk_base::MessageData* data, bool time_sensitive) { | 80 talk_base::MessageData* data, bool time_sensitive) { |
81 PostTaskInternal(0, handler, message_id, data); | 81 PostTaskInternal(0, handler, message_id, data); |
82 } | 82 } |
(...skipping 72 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
155 | 155 |
156 PendingSend pending_send(message); | 156 PendingSend pending_send(message); |
157 { | 157 { |
158 base::AutoLock auto_lock(lock_); | 158 base::AutoLock auto_lock(lock_); |
159 pending_send_messages_.push_back(&pending_send); | 159 pending_send_messages_.push_back(&pending_send); |
160 } | 160 } |
161 | 161 |
162 // Need to signal |pending_send_event_| here in case the thread is | 162 // Need to signal |pending_send_event_| here in case the thread is |
163 // sending message to another thread. | 163 // sending message to another thread. |
164 pending_send_event_.Signal(); | 164 pending_send_event_.Signal(); |
165 message_loop_->PostTask(FROM_HERE, | 165 task_runner_->PostTask(FROM_HERE, |
166 base::Bind(&JingleThreadWrapper::ProcessPendingSends, | 166 base::Bind(&JingleThreadWrapper::ProcessPendingSends, |
167 base::Unretained(this))); | 167 weak_ptr_)); |
168 | 168 |
169 | 169 |
170 while (!pending_send.done_event.IsSignaled()) { | 170 while (!pending_send.done_event.IsSignaled()) { |
171 base::WaitableEvent* events[] = {&pending_send.done_event, | 171 base::WaitableEvent* events[] = {&pending_send.done_event, |
172 ¤t_thread->pending_send_event_}; | 172 ¤t_thread->pending_send_event_}; |
173 size_t event = base::WaitableEvent::WaitMany(events, arraysize(events)); | 173 size_t event = base::WaitableEvent::WaitMany(events, arraysize(events)); |
174 DCHECK(event == 0 || event == 1); | 174 DCHECK(event == 0 || event == 1); |
175 | 175 |
176 if (event == 1) | 176 if (event == 1) |
177 current_thread->ProcessPendingSends(); | 177 current_thread->ProcessPendingSends(); |
(...skipping 29 matching lines...) Expand all Loading... |
207 message.phandler = handler; | 207 message.phandler = handler; |
208 message.message_id = message_id; | 208 message.message_id = message_id; |
209 message.pdata = data; | 209 message.pdata = data; |
210 { | 210 { |
211 base::AutoLock auto_lock(lock_); | 211 base::AutoLock auto_lock(lock_); |
212 task_id = ++last_task_id_; | 212 task_id = ++last_task_id_; |
213 messages_.insert(std::pair<int, talk_base::Message>(task_id, message)); | 213 messages_.insert(std::pair<int, talk_base::Message>(task_id, message)); |
214 } | 214 } |
215 | 215 |
216 if (delay_ms <= 0) { | 216 if (delay_ms <= 0) { |
217 message_loop_->PostTask(FROM_HERE, | 217 task_runner_->PostTask(FROM_HERE, |
218 base::Bind(&JingleThreadWrapper::RunTask, | 218 base::Bind(&JingleThreadWrapper::RunTask, |
219 base::Unretained(this), task_id)); | 219 weak_ptr_, task_id)); |
220 } else { | 220 } else { |
221 message_loop_->PostDelayedTask(FROM_HERE, | 221 task_runner_->PostDelayedTask(FROM_HERE, |
222 base::Bind(&JingleThreadWrapper::RunTask, | 222 base::Bind(&JingleThreadWrapper::RunTask, |
223 base::Unretained(this), task_id), | 223 weak_ptr_, task_id), |
224 base::TimeDelta::FromMilliseconds(delay_ms)); | 224 base::TimeDelta::FromMilliseconds(delay_ms)); |
225 } | 225 } |
226 } | 226 } |
227 | 227 |
228 void JingleThreadWrapper::RunTask(int task_id) { | 228 void JingleThreadWrapper::RunTask(int task_id) { |
229 bool have_message = false; | 229 bool have_message = false; |
230 talk_base::Message message; | 230 talk_base::Message message; |
231 { | 231 { |
232 base::AutoLock auto_lock(lock_); | 232 base::AutoLock auto_lock(lock_); |
233 MessagesQueue::iterator it = messages_.find(task_id); | 233 MessagesQueue::iterator it = messages_.find(task_id); |
234 if (it != messages_.end()) { | 234 if (it != messages_.end()) { |
(...skipping 58 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
293 | 293 |
294 void JingleThreadWrapper::Stop() { | 294 void JingleThreadWrapper::Stop() { |
295 NOTREACHED(); | 295 NOTREACHED(); |
296 } | 296 } |
297 | 297 |
298 void JingleThreadWrapper::Run() { | 298 void JingleThreadWrapper::Run() { |
299 NOTREACHED(); | 299 NOTREACHED(); |
300 } | 300 } |
301 | 301 |
302 } // namespace jingle_glue | 302 } // namespace jingle_glue |
OLD | NEW |