blob: e793a836db7a3f6391ecd6cdbaaae8d54822f648 [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"
gabf64a25e2017-05-12 19:42:5614#include "base/message_loop/message_loop.h"
[email protected]f7b98b32013-02-05 08:14:1515#include "base/run_loop.h"
skyostil054861d2015-04-30 19:06:1516#include "base/single_thread_task_runner.h"
gab6cce2f32016-05-11 19:53:4317#include "base/threading/thread_task_runner_handle.h"
[email protected]b77576f52011-11-24 04:12:0418#include "testing/gtest/include/gtest/gtest.h"
19
20namespace base {
21namespace {
22
23class TestRefCounted : public RefCountedThreadSafe<TestRefCounted> {
24 private:
25 friend class RefCountedThreadSafe<TestRefCounted>;
26 ~TestRefCounted() {};
27};
28
29void Increment(int* count) { (*count)++; }
30void IncrementBy(int* count, int n) { (*count) += n; }
31void RefCountedParam(const scoped_refptr<TestRefCounted>& ref_counted) {}
32
kuznetsovsb020bcd82016-04-13 13:01:4733void OnMoveOnlyReceived(int* value, std::unique_ptr<int> result) {
34 *value = *result;
35}
36
[email protected]b77576f52011-11-24 04:12:0437// Cancel().
38// - Callback can be run multiple times.
39// - After Cancel(), Run() completes but has no effect.
40TEST(CancelableCallbackTest, Cancel) {
41 int count = 0;
[email protected]d4d57df2011-11-30 20:33:5242 CancelableClosure cancelable(
[email protected]b77576f52011-11-24 04:12:0443 base::Bind(&Increment, base::Unretained(&count)));
44
45 base::Closure callback = cancelable.callback();
46 callback.Run();
47 EXPECT_EQ(1, count);
48
49 callback.Run();
50 EXPECT_EQ(2, count);
51
52 cancelable.Cancel();
53 callback.Run();
54 EXPECT_EQ(2, count);
55}
56
57// Cancel() called multiple times.
58// - Cancel() cancels all copies of the wrapped callback.
[email protected]d4d57df2011-11-30 20:33:5259// - Calling Cancel() more than once has no effect.
60// - After Cancel(), callback() returns a null callback.
[email protected]b77576f52011-11-24 04:12:0461TEST(CancelableCallbackTest, MultipleCancel) {
62 int count = 0;
[email protected]d4d57df2011-11-30 20:33:5263 CancelableClosure cancelable(
[email protected]b77576f52011-11-24 04:12:0464 base::Bind(&Increment, base::Unretained(&count)));
65
66 base::Closure callback1 = cancelable.callback();
[email protected]d4d57df2011-11-30 20:33:5267 base::Closure callback2 = cancelable.callback();
[email protected]b77576f52011-11-24 04:12:0468 cancelable.Cancel();
69
70 callback1.Run();
71 EXPECT_EQ(0, count);
72
[email protected]b77576f52011-11-24 04:12:0473 callback2.Run();
74 EXPECT_EQ(0, count);
75
[email protected]d4d57df2011-11-30 20:33:5276 // Calling Cancel() again has no effect.
[email protected]b77576f52011-11-24 04:12:0477 cancelable.Cancel();
[email protected]d4d57df2011-11-30 20:33:5278
79 // callback() of a cancelled callback is null.
80 base::Closure callback3 = cancelable.callback();
81 EXPECT_TRUE(callback3.is_null());
[email protected]b77576f52011-11-24 04:12:0482}
83
84// CancelableCallback destroyed before callback is run.
85// - Destruction of CancelableCallback cancels outstanding callbacks.
86TEST(CancelableCallbackTest, CallbackCanceledOnDestruction) {
87 int count = 0;
88 base::Closure callback;
89
90 {
[email protected]d4d57df2011-11-30 20:33:5291 CancelableClosure cancelable(
[email protected]b77576f52011-11-24 04:12:0492 base::Bind(&Increment, base::Unretained(&count)));
93
94 callback = cancelable.callback();
95 callback.Run();
96 EXPECT_EQ(1, count);
97 }
98
99 callback.Run();
100 EXPECT_EQ(1, count);
101}
102
103// Cancel() called on bound closure with a RefCounted parameter.
104// - Cancel drops wrapped callback (and, implicitly, its bound arguments).
105TEST(CancelableCallbackTest, CancelDropsCallback) {
106 scoped_refptr<TestRefCounted> ref_counted = new TestRefCounted;
107 EXPECT_TRUE(ref_counted->HasOneRef());
108
[email protected]d4d57df2011-11-30 20:33:52109 CancelableClosure cancelable(base::Bind(RefCountedParam, ref_counted));
[email protected]b77576f52011-11-24 04:12:04110 EXPECT_FALSE(cancelable.IsCancelled());
111 EXPECT_TRUE(ref_counted.get());
112 EXPECT_FALSE(ref_counted->HasOneRef());
113
114 // There is only one reference to |ref_counted| after the Cancel().
115 cancelable.Cancel();
116 EXPECT_TRUE(cancelable.IsCancelled());
117 EXPECT_TRUE(ref_counted.get());
118 EXPECT_TRUE(ref_counted->HasOneRef());
119}
120
121// Reset().
122// - Reset() replaces the existing wrapped callback with a new callback.
123// - Reset() deactivates outstanding callbacks.
124TEST(CancelableCallbackTest, Reset) {
125 int count = 0;
[email protected]d4d57df2011-11-30 20:33:52126 CancelableClosure cancelable(
[email protected]b77576f52011-11-24 04:12:04127 base::Bind(&Increment, base::Unretained(&count)));
128
129 base::Closure callback = cancelable.callback();
130 callback.Run();
131 EXPECT_EQ(1, count);
132
133 callback.Run();
134 EXPECT_EQ(2, count);
135
136 cancelable.Reset(
137 base::Bind(&IncrementBy, base::Unretained(&count), 3));
138 EXPECT_FALSE(cancelable.IsCancelled());
139
140 // The stale copy of the cancelable callback is non-null.
141 ASSERT_FALSE(callback.is_null());
142
143 // The stale copy of the cancelable callback is no longer active.
144 callback.Run();
145 EXPECT_EQ(2, count);
146
147 base::Closure callback2 = cancelable.callback();
148 ASSERT_FALSE(callback2.is_null());
149
150 callback2.Run();
151 EXPECT_EQ(5, count);
152}
153
154// IsCanceled().
155// - Cancel() transforms the CancelableCallback into a cancelled state.
156TEST(CancelableCallbackTest, IsNull) {
[email protected]d4d57df2011-11-30 20:33:52157 CancelableClosure cancelable;
[email protected]b77576f52011-11-24 04:12:04158 EXPECT_TRUE(cancelable.IsCancelled());
159
160 int count = 0;
161 cancelable.Reset(base::Bind(&Increment,
162 base::Unretained(&count)));
163 EXPECT_FALSE(cancelable.IsCancelled());
164
165 cancelable.Cancel();
166 EXPECT_TRUE(cancelable.IsCancelled());
167}
168
169// CancelableCallback posted to a MessageLoop with PostTask.
170// - Callbacks posted to a MessageLoop can be cancelled.
171TEST(CancelableCallbackTest, PostTask) {
[email protected]1ef90012014-01-15 22:24:33172 MessageLoop loop;
[email protected]b77576f52011-11-24 04:12:04173
174 int count = 0;
[email protected]d4d57df2011-11-30 20:33:52175 CancelableClosure cancelable(base::Bind(&Increment,
[email protected]b77576f52011-11-24 04:12:04176 base::Unretained(&count)));
177
skyostil054861d2015-04-30 19:06:15178 ThreadTaskRunnerHandle::Get()->PostTask(FROM_HERE, cancelable.callback());
[email protected]f7b98b32013-02-05 08:14:15179 RunLoop().RunUntilIdle();
[email protected]b77576f52011-11-24 04:12:04180
181 EXPECT_EQ(1, count);
182
skyostil054861d2015-04-30 19:06:15183 ThreadTaskRunnerHandle::Get()->PostTask(FROM_HERE, cancelable.callback());
[email protected]b77576f52011-11-24 04:12:04184
185 // Cancel before running the message loop.
186 cancelable.Cancel();
[email protected]f7b98b32013-02-05 08:14:15187 RunLoop().RunUntilIdle();
[email protected]b77576f52011-11-24 04:12:04188
189 // Callback never ran due to cancellation; count is the same.
190 EXPECT_EQ(1, count);
191}
192
kuznetsovsb020bcd82016-04-13 13:01:47193// CancelableCallback can be used with move-only types.
194TEST(CancelableCallbackTest, MoveOnlyType) {
195 const int kExpectedResult = 42;
196
197 int result = 0;
198 CancelableCallback<void(std::unique_ptr<int>)> cb(
199 base::Bind(&OnMoveOnlyReceived, base::Unretained(&result)));
200 cb.callback().Run(base::WrapUnique(new int(kExpectedResult)));
201
202 EXPECT_EQ(kExpectedResult, result);
203}
204
[email protected]b77576f52011-11-24 04:12:04205} // namespace
206} // namespace base