blob: 23b6c1c244f76d19c8572af8f469d2692f6c7412 [file] [log] [blame]
[email protected]b77576f52011-11-24 04:12:041// Copyright (c) 2011 The Chromium Authors. All rights reserved.
2// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
5#include "base/cancelable_callback.h"
6
kuznetsovsb020bcd82016-04-13 13:01:477#include <memory>
8
[email protected]b77576f52011-11-24 04:12:049#include "base/bind.h"
10#include "base/bind_helpers.h"
skyostil054861d2015-04-30 19:06:1511#include "base/location.h"
kuznetsovsb020bcd82016-04-13 13:01:4712#include "base/memory/ptr_util.h"
[email protected]b77576f52011-11-24 04:12:0413#include "base/memory/ref_counted.h"
[email protected]f7b98b32013-02-05 08:14:1514#include "base/run_loop.h"
skyostil054861d2015-04-30 19:06:1515#include "base/single_thread_task_runner.h"
gab6cce2f32016-05-11 19:53:4316#include "base/threading/thread_task_runner_handle.h"
[email protected]b77576f52011-11-24 04:12:0417#include "testing/gtest/include/gtest/gtest.h"
18
19namespace base {
20namespace {
21
22class TestRefCounted : public RefCountedThreadSafe<TestRefCounted> {
23 private:
24 friend class RefCountedThreadSafe<TestRefCounted>;
25 ~TestRefCounted() {};
26};
27
28void Increment(int* count) { (*count)++; }
29void IncrementBy(int* count, int n) { (*count) += n; }
30void RefCountedParam(const scoped_refptr<TestRefCounted>& ref_counted) {}
31
kuznetsovsb020bcd82016-04-13 13:01:4732void OnMoveOnlyReceived(int* value, std::unique_ptr<int> result) {
33 *value = *result;
34}
35
[email protected]b77576f52011-11-24 04:12:0436// Cancel().
37// - Callback can be run multiple times.
38// - After Cancel(), Run() completes but has no effect.
39TEST(CancelableCallbackTest, Cancel) {
40 int count = 0;
[email protected]d4d57df2011-11-30 20:33:5241 CancelableClosure cancelable(
[email protected]b77576f52011-11-24 04:12:0442 base::Bind(&Increment, base::Unretained(&count)));
43
44 base::Closure callback = cancelable.callback();
45 callback.Run();
46 EXPECT_EQ(1, count);
47
48 callback.Run();
49 EXPECT_EQ(2, count);
50
51 cancelable.Cancel();
52 callback.Run();
53 EXPECT_EQ(2, count);
54}
55
56// Cancel() called multiple times.
57// - Cancel() cancels all copies of the wrapped callback.
[email protected]d4d57df2011-11-30 20:33:5258// - Calling Cancel() more than once has no effect.
59// - After Cancel(), callback() returns a null callback.
[email protected]b77576f52011-11-24 04:12:0460TEST(CancelableCallbackTest, MultipleCancel) {
61 int count = 0;
[email protected]d4d57df2011-11-30 20:33:5262 CancelableClosure cancelable(
[email protected]b77576f52011-11-24 04:12:0463 base::Bind(&Increment, base::Unretained(&count)));
64
65 base::Closure callback1 = cancelable.callback();
[email protected]d4d57df2011-11-30 20:33:5266 base::Closure callback2 = cancelable.callback();
[email protected]b77576f52011-11-24 04:12:0467 cancelable.Cancel();
68
69 callback1.Run();
70 EXPECT_EQ(0, count);
71
[email protected]b77576f52011-11-24 04:12:0472 callback2.Run();
73 EXPECT_EQ(0, count);
74
[email protected]d4d57df2011-11-30 20:33:5275 // Calling Cancel() again has no effect.
[email protected]b77576f52011-11-24 04:12:0476 cancelable.Cancel();
[email protected]d4d57df2011-11-30 20:33:5277
78 // callback() of a cancelled callback is null.
79 base::Closure callback3 = cancelable.callback();
80 EXPECT_TRUE(callback3.is_null());
[email protected]b77576f52011-11-24 04:12:0481}
82
83// CancelableCallback destroyed before callback is run.
84// - Destruction of CancelableCallback cancels outstanding callbacks.
85TEST(CancelableCallbackTest, CallbackCanceledOnDestruction) {
86 int count = 0;
87 base::Closure callback;
88
89 {
[email protected]d4d57df2011-11-30 20:33:5290 CancelableClosure cancelable(
[email protected]b77576f52011-11-24 04:12:0491 base::Bind(&Increment, base::Unretained(&count)));
92
93 callback = cancelable.callback();
94 callback.Run();
95 EXPECT_EQ(1, count);
96 }
97
98 callback.Run();
99 EXPECT_EQ(1, count);
100}
101
102// Cancel() called on bound closure with a RefCounted parameter.
103// - Cancel drops wrapped callback (and, implicitly, its bound arguments).
104TEST(CancelableCallbackTest, CancelDropsCallback) {
105 scoped_refptr<TestRefCounted> ref_counted = new TestRefCounted;
106 EXPECT_TRUE(ref_counted->HasOneRef());
107
[email protected]d4d57df2011-11-30 20:33:52108 CancelableClosure cancelable(base::Bind(RefCountedParam, ref_counted));
[email protected]b77576f52011-11-24 04:12:04109 EXPECT_FALSE(cancelable.IsCancelled());
110 EXPECT_TRUE(ref_counted.get());
111 EXPECT_FALSE(ref_counted->HasOneRef());
112
113 // There is only one reference to |ref_counted| after the Cancel().
114 cancelable.Cancel();
115 EXPECT_TRUE(cancelable.IsCancelled());
116 EXPECT_TRUE(ref_counted.get());
117 EXPECT_TRUE(ref_counted->HasOneRef());
118}
119
120// Reset().
121// - Reset() replaces the existing wrapped callback with a new callback.
122// - Reset() deactivates outstanding callbacks.
123TEST(CancelableCallbackTest, Reset) {
124 int count = 0;
[email protected]d4d57df2011-11-30 20:33:52125 CancelableClosure cancelable(
[email protected]b77576f52011-11-24 04:12:04126 base::Bind(&Increment, base::Unretained(&count)));
127
128 base::Closure callback = cancelable.callback();
129 callback.Run();
130 EXPECT_EQ(1, count);
131
132 callback.Run();
133 EXPECT_EQ(2, count);
134
135 cancelable.Reset(
136 base::Bind(&IncrementBy, base::Unretained(&count), 3));
137 EXPECT_FALSE(cancelable.IsCancelled());
138
139 // The stale copy of the cancelable callback is non-null.
140 ASSERT_FALSE(callback.is_null());
141
142 // The stale copy of the cancelable callback is no longer active.
143 callback.Run();
144 EXPECT_EQ(2, count);
145
146 base::Closure callback2 = cancelable.callback();
147 ASSERT_FALSE(callback2.is_null());
148
149 callback2.Run();
150 EXPECT_EQ(5, count);
151}
152
153// IsCanceled().
154// - Cancel() transforms the CancelableCallback into a cancelled state.
155TEST(CancelableCallbackTest, IsNull) {
[email protected]d4d57df2011-11-30 20:33:52156 CancelableClosure cancelable;
[email protected]b77576f52011-11-24 04:12:04157 EXPECT_TRUE(cancelable.IsCancelled());
158
159 int count = 0;
160 cancelable.Reset(base::Bind(&Increment,
161 base::Unretained(&count)));
162 EXPECT_FALSE(cancelable.IsCancelled());
163
164 cancelable.Cancel();
165 EXPECT_TRUE(cancelable.IsCancelled());
166}
167
168// CancelableCallback posted to a MessageLoop with PostTask.
169// - Callbacks posted to a MessageLoop can be cancelled.
170TEST(CancelableCallbackTest, PostTask) {
[email protected]1ef90012014-01-15 22:24:33171 MessageLoop loop;
[email protected]b77576f52011-11-24 04:12:04172
173 int count = 0;
[email protected]d4d57df2011-11-30 20:33:52174 CancelableClosure cancelable(base::Bind(&Increment,
[email protected]b77576f52011-11-24 04:12:04175 base::Unretained(&count)));
176
skyostil054861d2015-04-30 19:06:15177 ThreadTaskRunnerHandle::Get()->PostTask(FROM_HERE, cancelable.callback());
[email protected]f7b98b32013-02-05 08:14:15178 RunLoop().RunUntilIdle();
[email protected]b77576f52011-11-24 04:12:04179
180 EXPECT_EQ(1, count);
181
skyostil054861d2015-04-30 19:06:15182 ThreadTaskRunnerHandle::Get()->PostTask(FROM_HERE, cancelable.callback());
[email protected]b77576f52011-11-24 04:12:04183
184 // Cancel before running the message loop.
185 cancelable.Cancel();
[email protected]f7b98b32013-02-05 08:14:15186 RunLoop().RunUntilIdle();
[email protected]b77576f52011-11-24 04:12:04187
188 // Callback never ran due to cancellation; count is the same.
189 EXPECT_EQ(1, count);
190}
191
kuznetsovsb020bcd82016-04-13 13:01:47192// CancelableCallback can be used with move-only types.
193TEST(CancelableCallbackTest, MoveOnlyType) {
194 const int kExpectedResult = 42;
195
196 int result = 0;
197 CancelableCallback<void(std::unique_ptr<int>)> cb(
198 base::Bind(&OnMoveOnlyReceived, base::Unretained(&result)));
199 cb.callback().Run(base::WrapUnique(new int(kExpectedResult)));
200
201 EXPECT_EQ(kExpectedResult, result);
202}
203
[email protected]b77576f52011-11-24 04:12:04204} // namespace
205} // namespace base