1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
|
$$ This is a pump file for generating file templates. Pump is a python
$$ script that is part of the Google Test suite of utilities. Description
$$ can be found here:
$$
$$ http://code.google.com/p/googletest/wiki/PumpManual
$$
$$ See comment for MAX_ARITY in base/bind.h.pump.
$var MAX_ARITY = 7
// Copyright (c) 2012 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#ifndef MEDIA_BASE_BIND_TO_LOOP_H_
#define MEDIA_BASE_BIND_TO_LOOP_H_
#include "base/bind.h"
#include "base/location.h"
#include "base/message_loop/message_loop_proxy.h"
// This is a helper utility for base::Bind()ing callbacks on to particular
// MessageLoops. A typical use is when |a| (of class |A|) wants to hand a
// callback such as base::Bind(&A::AMethod, a) to |b|, but needs to ensure that
// when |b| executes the callback, it does so on a particular MessageLoop.
//
// Typical usage: request to be called back on the current thread:
// other->StartAsyncProcessAndCallMeBack(
// media::BindToLoop(MessageLoopProxy::current(),
// base::Bind(&MyClass::MyMethod, this)));
//
// Note that like base::Bind(), BindToLoop() can't bind non-constant references,
// and that *unlike* base::Bind(), BindToLoop() makes copies of its arguments,
// and thus can't be used with arrays.
namespace media {
// Mimic base::internal::CallbackForward, replacing p.Pass() with
// base::Passed(&p) to account for the extra layer of indirection.
namespace internal {
template <typename T>
T& TrampolineForward(T& t) { return t; }
template <typename T>
base::internal::PassedWrapper<scoped_ptr<T> > TrampolineForward(
scoped_ptr<T>& p) { return base::Passed(&p); }
template <typename T>
base::internal::PassedWrapper<scoped_array<T> > TrampolineForward(
scoped_array<T>& p) { return base::Passed(&p); }
template <typename T, typename R>
base::internal::PassedWrapper<scoped_ptr_malloc<T, R> > TrampolineForward(
scoped_ptr_malloc<T, R>& p) { return base::Passed(&p); }
template <typename T>
base::internal::PassedWrapper<ScopedVector<T> > TrampolineForward(
ScopedVector<T>& p) { return base::Passed(&p); }
template <typename T> struct TrampolineHelper;
$range ARITY 0..MAX_ARITY
$for ARITY [[
$range ARG 1..ARITY
template <$for ARG , [[typename A$(ARG)]]>
struct TrampolineHelper<void($for ARG , [[A$(ARG)]])> {
static void Run(
const scoped_refptr<base::MessageLoopProxy>& loop,
const base::Callback<void($for ARG , [[A$(ARG)]])>& cb
$if ARITY != 0 [[, ]]
$for ARG , [[A$(ARG) a$(ARG)]]
) {
loop->PostTask(FROM_HERE, base::Bind(cb
$if ARITY != 0 [[, ]]
$for ARG , [[internal::TrampolineForward(a$(ARG))]]));
}
};
]] $$ for ARITY
} // namespace internal
template<typename T>
static base::Callback<T> BindToLoop(
const scoped_refptr<base::MessageLoopProxy>& loop,
const base::Callback<T>& cb) {
return base::Bind(&internal::TrampolineHelper<T>::Run, loop, cb);
}
template<typename T>
static base::Callback<T> BindToCurrentLoop(
const base::Callback<T>& cb) {
return BindToLoop(base::MessageLoopProxy::current(), cb);
}
} // namespace media
#endif // MEDIA_BASE_BIND_TO_LOOP_H_
|