blob: c4afd3a194d79d00ac2ebceb460d6ac05c8a636f [file] [log] [blame]
sergeyu@chromium.org9cf037b2014-02-07 19:03:26 +00001/*
2 * libjingle
3 * Copyright 2014 Google Inc.
4 *
5 * Redistribution and use in source and binary forms, with or without
6 * modification, are permitted provided that the following conditions are met:
7 *
8 * 1. Redistributions of source code must retain the above copyright notice,
9 * this list of conditions and the following disclaimer.
10 * 2. Redistributions in binary form must reproduce the above copyright notice,
11 * this list of conditions and the following disclaimer in the documentation
12 * and/or other materials provided with the distribution.
13 * 3. The name of the author may not be used to endorse or promote products
14 * derived from this software without specific prior written permission.
15 *
16 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED
17 * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
18 * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO
19 * EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
20 * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
21 * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS;
22 * OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
23 * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
24 * OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
25 * ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
26 */
27
28#ifndef TALK_BASE_ASYNCINVOKER_INL_H_
29#define TALK_BASE_ASYNCINVOKER_INL_H_
30
31#include "talk/base/criticalsection.h"
32#include "talk/base/messagehandler.h"
33#include "talk/base/sigslot.h"
34#include "talk/base/thread.h"
35
36namespace talk_base {
37
38// Helper class for AsyncInvoker. Runs a functor on a message queue or thread
39// and doesn't execute the callback when finished if the calling thread ends.
40template <class ReturnT, class FunctorT>
41class AsyncFunctorMessageHandler
42 : public FunctorMessageHandler<ReturnT, FunctorT>,
43 public sigslot::has_slots<> {
44 typedef AsyncFunctorMessageHandler<ReturnT, FunctorT> ThisT;
45 public:
46 explicit AsyncFunctorMessageHandler(const FunctorT& functor)
47 : FunctorMessageHandler<ReturnT, FunctorT>(functor),
48 thread_(Thread::Current()),
49 shutting_down_(false) {
50 thread_->SignalQueueDestroyed.connect(this, &ThisT::OnThreadDestroyed);
51 }
52
53 virtual ~AsyncFunctorMessageHandler() {
54 CritScope cs(&running_crit_);
55 shutting_down_ = true;
56 }
57
58 virtual void OnMessage(Message* msg) {
59 CritScope cs(&running_crit_);
60 if (!shutting_down_) {
61 FunctorMessageHandler<ReturnT, FunctorT>::OnMessage(msg);
62 }
63 }
64
65 // Returns the thread that initiated the async call.
66 Thread* thread() const { return thread_; }
67
68 // Wraps a callback so that it won't execute if |thread_| goes away.
69 void WrapCallback(Callback0<void> cb) {
70 this->SetCallback(
71 Callback0<void>(Bind(&ThisT::MaybeRunCallback, this, cb)));
72 }
73
74 private:
75 void OnThreadDestroyed() {
76 CritScope cs(&thread_crit_);
77 thread_ = NULL;
78 this->SetCallback(Callback0<void>()); // Clear out the callback.
79 }
80
81 void MaybeRunCallback(Callback0<void> cb) {
82#ifdef _DEBUG
83 ASSERT(running_crit_.CurrentThreadIsOwner());
84#endif
85 CritScope cs(&thread_crit_);
86 if (thread_ && !shutting_down_) {
87 cb();
88 }
89 }
90
91 FunctorT functor_;
92 Thread* thread_;
93 CriticalSection thread_crit_;
94 CriticalSection running_crit_;
95 bool shutting_down_;
96};
97
98} // namespace talk_base
99
100
101#endif // TALK_BASE_ASYNCINVOKER_INL_H_