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 13 matching lines...) Expand all Loading... | |
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::EnsureForCurrentThread() { |
33 if (JingleThreadWrapper::current() == NULL) { | 33 if (JingleThreadWrapper::current() == NULL) { |
34 g_jingle_thread_wrapper.Get().Set( | 34 MessageLoop* message_loop = MessageLoop::current(); |
Wez
2012/08/09 23:30:25
This will only work if called from a MessageLoop;
Sergey Ulanov
2012/08/13 21:15:34
Yes. That's intentional. We need it to add destruc
Wez
2012/08/14 00:06:04
Would it be useful to have a LeakyEnsureForCurrent
Sergey Ulanov
2012/08/14 01:02:14
Don't think it's necessary. It's always possible t
| |
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 DCHECK(task_runner->BelongsToCurrentThread()); |
53 | 56 DCHECK(!talk_base::Thread::Current()); |
54 talk_base::ThreadManager::Instance()->UnwrapCurrentThread(); | |
55 talk_base::ThreadManager::Instance()->SetCurrentThread(this); | |
56 talk_base::MessageQueueManager::Instance()->Add(this); | 57 talk_base::MessageQueueManager::Instance()->Add(this); |
57 message_loop_->AddDestructionObserver(this); | |
58 | |
59 WrapCurrent(); | 58 WrapCurrent(); |
60 } | 59 } |
61 | 60 |
62 JingleThreadWrapper::~JingleThreadWrapper() { | 61 JingleThreadWrapper::~JingleThreadWrapper() { |
63 Clear(NULL, talk_base::MQID_ANY, NULL); | 62 Clear(NULL, talk_base::MQID_ANY, NULL); |
64 } | 63 } |
65 | 64 |
66 void JingleThreadWrapper::WillDestroyCurrentMessageLoop() { | 65 void JingleThreadWrapper::WillDestroyCurrentMessageLoop() { |
67 DCHECK_EQ(talk_base::Thread::Current(), current()); | 66 DCHECK_EQ(talk_base::Thread::Current(), current()); |
68 UnwrapCurrent(); | 67 UnwrapCurrent(); |
69 g_jingle_thread_wrapper.Get().Set(NULL); | 68 g_jingle_thread_wrapper.Get().Set(NULL); |
70 talk_base::ThreadManager::Instance()->SetCurrentThread(NULL); | 69 talk_base::ThreadManager::Instance()->SetCurrentThread(NULL); |
71 talk_base::MessageQueueManager::Instance()->Remove(this); | 70 talk_base::MessageQueueManager::Instance()->Remove(this); |
72 message_loop_->RemoveDestructionObserver(this); | |
73 talk_base::SocketServer* ss = socketserver(); | 71 talk_base::SocketServer* ss = socketserver(); |
74 delete this; | 72 delete this; |
75 delete ss; | 73 delete ss; |
76 } | 74 } |
77 | 75 |
78 void JingleThreadWrapper::Post( | 76 void JingleThreadWrapper::Post( |
79 talk_base::MessageHandler* handler, uint32 message_id, | 77 talk_base::MessageHandler* handler, uint32 message_id, |
80 talk_base::MessageData* data, bool time_sensitive) { | 78 talk_base::MessageData* data, bool time_sensitive) { |
81 PostTaskInternal(0, handler, message_id, data); | 79 PostTaskInternal(0, handler, message_id, data); |
82 } | 80 } |
(...skipping 72 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
155 | 153 |
156 PendingSend pending_send(message); | 154 PendingSend pending_send(message); |
157 { | 155 { |
158 base::AutoLock auto_lock(lock_); | 156 base::AutoLock auto_lock(lock_); |
159 pending_send_messages_.push_back(&pending_send); | 157 pending_send_messages_.push_back(&pending_send); |
160 } | 158 } |
161 | 159 |
162 // Need to signal |pending_send_event_| here in case the thread is | 160 // Need to signal |pending_send_event_| here in case the thread is |
163 // sending message to another thread. | 161 // sending message to another thread. |
164 pending_send_event_.Signal(); | 162 pending_send_event_.Signal(); |
165 message_loop_->PostTask(FROM_HERE, | 163 task_runner_->PostTask(FROM_HERE, |
166 base::Bind(&JingleThreadWrapper::ProcessPendingSends, | 164 base::Bind(&JingleThreadWrapper::ProcessPendingSends, |
167 base::Unretained(this))); | 165 base::Unretained(this))); |
Wez
2012/08/09 23:30:25
Should this be a WeakPtr?
Sergey Ulanov
2012/08/13 21:15:34
Yes. Thanks for catching this. Fixed.
| |
168 | 166 |
169 | 167 |
170 while (!pending_send.done_event.IsSignaled()) { | 168 while (!pending_send.done_event.IsSignaled()) { |
171 base::WaitableEvent* events[] = {&pending_send.done_event, | 169 base::WaitableEvent* events[] = {&pending_send.done_event, |
172 ¤t_thread->pending_send_event_}; | 170 ¤t_thread->pending_send_event_}; |
173 size_t event = base::WaitableEvent::WaitMany(events, arraysize(events)); | 171 size_t event = base::WaitableEvent::WaitMany(events, arraysize(events)); |
174 DCHECK(event == 0 || event == 1); | 172 DCHECK(event == 0 || event == 1); |
175 | 173 |
176 if (event == 1) | 174 if (event == 1) |
177 current_thread->ProcessPendingSends(); | 175 current_thread->ProcessPendingSends(); |
(...skipping 29 matching lines...) Expand all Loading... | |
207 message.phandler = handler; | 205 message.phandler = handler; |
208 message.message_id = message_id; | 206 message.message_id = message_id; |
209 message.pdata = data; | 207 message.pdata = data; |
210 { | 208 { |
211 base::AutoLock auto_lock(lock_); | 209 base::AutoLock auto_lock(lock_); |
212 task_id = ++last_task_id_; | 210 task_id = ++last_task_id_; |
213 messages_.insert(std::pair<int, talk_base::Message>(task_id, message)); | 211 messages_.insert(std::pair<int, talk_base::Message>(task_id, message)); |
214 } | 212 } |
215 | 213 |
216 if (delay_ms <= 0) { | 214 if (delay_ms <= 0) { |
217 message_loop_->PostTask(FROM_HERE, | 215 task_runner_->PostTask(FROM_HERE, |
218 base::Bind(&JingleThreadWrapper::RunTask, | 216 base::Bind(&JingleThreadWrapper::RunTask, |
219 base::Unretained(this), task_id)); | 217 base::Unretained(this), task_id)); |
Wez
2012/08/09 23:30:25
WeakPtr?
Sergey Ulanov
2012/08/13 21:15:34
Done.
| |
220 } else { | 218 } else { |
221 message_loop_->PostDelayedTask(FROM_HERE, | 219 task_runner_->PostDelayedTask(FROM_HERE, |
222 base::Bind(&JingleThreadWrapper::RunTask, | 220 base::Bind(&JingleThreadWrapper::RunTask, |
223 base::Unretained(this), task_id), | 221 base::Unretained(this), task_id), |
Wez
2012/08/09 23:30:25
WeakPtr?
Sergey Ulanov
2012/08/13 21:15:34
Done.
| |
224 base::TimeDelta::FromMilliseconds(delay_ms)); | 222 base::TimeDelta::FromMilliseconds(delay_ms)); |
225 } | 223 } |
226 } | 224 } |
227 | 225 |
228 void JingleThreadWrapper::RunTask(int task_id) { | 226 void JingleThreadWrapper::RunTask(int task_id) { |
229 bool have_message = false; | 227 bool have_message = false; |
230 talk_base::Message message; | 228 talk_base::Message message; |
231 { | 229 { |
232 base::AutoLock auto_lock(lock_); | 230 base::AutoLock auto_lock(lock_); |
233 MessagesQueue::iterator it = messages_.find(task_id); | 231 MessagesQueue::iterator it = messages_.find(task_id); |
234 if (it != messages_.end()) { | 232 if (it != messages_.end()) { |
(...skipping 58 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
293 | 291 |
294 void JingleThreadWrapper::Stop() { | 292 void JingleThreadWrapper::Stop() { |
295 NOTREACHED(); | 293 NOTREACHED(); |
296 } | 294 } |
297 | 295 |
298 void JingleThreadWrapper::Run() { | 296 void JingleThreadWrapper::Run() { |
299 NOTREACHED(); | 297 NOTREACHED(); |
300 } | 298 } |
301 | 299 |
302 } // namespace jingle_glue | 300 } // namespace jingle_glue |
OLD | NEW |