blob: 373498cbded8975a0bf8da191d47529f7203051a [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>;
Chris Watkinsbb7211c2017-11-29 07:16:3826 ~TestRefCounted() = default;
27 ;
[email protected]b77576f52011-11-24 04:12:0428};
29
30void Increment(int* count) { (*count)++; }
31void IncrementBy(int* count, int n) { (*count) += n; }
32void RefCountedParam(const scoped_refptr<TestRefCounted>& ref_counted) {}
33
kuznetsovsb020bcd82016-04-13 13:01:4734void OnMoveOnlyReceived(int* value, std::unique_ptr<int> result) {
35 *value = *result;
36}
37
[email protected]b77576f52011-11-24 04:12:0438// Cancel().
39// - Callback can be run multiple times.
40// - After Cancel(), Run() completes but has no effect.
41TEST(CancelableCallbackTest, Cancel) {
42 int count = 0;
[email protected]d4d57df2011-11-30 20:33:5243 CancelableClosure cancelable(
[email protected]b77576f52011-11-24 04:12:0444 base::Bind(&Increment, base::Unretained(&count)));
45
46 base::Closure callback = cancelable.callback();
47 callback.Run();
48 EXPECT_EQ(1, count);
49
50 callback.Run();
51 EXPECT_EQ(2, count);
52
53 cancelable.Cancel();
54 callback.Run();
55 EXPECT_EQ(2, count);
56}
57
58// Cancel() called multiple times.
59// - Cancel() cancels all copies of the wrapped callback.
[email protected]d4d57df2011-11-30 20:33:5260// - Calling Cancel() more than once has no effect.
61// - After Cancel(), callback() returns a null callback.
[email protected]b77576f52011-11-24 04:12:0462TEST(CancelableCallbackTest, MultipleCancel) {
63 int count = 0;
[email protected]d4d57df2011-11-30 20:33:5264 CancelableClosure cancelable(
[email protected]b77576f52011-11-24 04:12:0465 base::Bind(&Increment, base::Unretained(&count)));
66
67 base::Closure callback1 = cancelable.callback();
[email protected]d4d57df2011-11-30 20:33:5268 base::Closure callback2 = cancelable.callback();
[email protected]b77576f52011-11-24 04:12:0469 cancelable.Cancel();
70
71 callback1.Run();
72 EXPECT_EQ(0, count);
73
[email protected]b77576f52011-11-24 04:12:0474 callback2.Run();
75 EXPECT_EQ(0, count);
76
[email protected]d4d57df2011-11-30 20:33:5277 // Calling Cancel() again has no effect.
[email protected]b77576f52011-11-24 04:12:0478 cancelable.Cancel();
[email protected]d4d57df2011-11-30 20:33:5279
80 // callback() of a cancelled callback is null.
81 base::Closure callback3 = cancelable.callback();
82 EXPECT_TRUE(callback3.is_null());
[email protected]b77576f52011-11-24 04:12:0483}
84
85// CancelableCallback destroyed before callback is run.
86// - Destruction of CancelableCallback cancels outstanding callbacks.
87TEST(CancelableCallbackTest, CallbackCanceledOnDestruction) {
88 int count = 0;
89 base::Closure callback;
90
91 {
[email protected]d4d57df2011-11-30 20:33:5292 CancelableClosure cancelable(
[email protected]b77576f52011-11-24 04:12:0493 base::Bind(&Increment, base::Unretained(&count)));
94
95 callback = cancelable.callback();
96 callback.Run();
97 EXPECT_EQ(1, count);
98 }
99
100 callback.Run();
101 EXPECT_EQ(1, count);
102}
103
104// Cancel() called on bound closure with a RefCounted parameter.
105// - Cancel drops wrapped callback (and, implicitly, its bound arguments).
106TEST(CancelableCallbackTest, CancelDropsCallback) {
107 scoped_refptr<TestRefCounted> ref_counted = new TestRefCounted;
108 EXPECT_TRUE(ref_counted->HasOneRef());
109
[email protected]d4d57df2011-11-30 20:33:52110 CancelableClosure cancelable(base::Bind(RefCountedParam, ref_counted));
[email protected]b77576f52011-11-24 04:12:04111 EXPECT_FALSE(cancelable.IsCancelled());
112 EXPECT_TRUE(ref_counted.get());
113 EXPECT_FALSE(ref_counted->HasOneRef());
114
115 // There is only one reference to |ref_counted| after the Cancel().
116 cancelable.Cancel();
117 EXPECT_TRUE(cancelable.IsCancelled());
118 EXPECT_TRUE(ref_counted.get());
119 EXPECT_TRUE(ref_counted->HasOneRef());
120}
121
122// Reset().
123// - Reset() replaces the existing wrapped callback with a new callback.
124// - Reset() deactivates outstanding callbacks.
125TEST(CancelableCallbackTest, Reset) {
126 int count = 0;
[email protected]d4d57df2011-11-30 20:33:52127 CancelableClosure cancelable(
[email protected]b77576f52011-11-24 04:12:04128 base::Bind(&Increment, base::Unretained(&count)));
129
130 base::Closure callback = cancelable.callback();
131 callback.Run();
132 EXPECT_EQ(1, count);
133
134 callback.Run();
135 EXPECT_EQ(2, count);
136
137 cancelable.Reset(
138 base::Bind(&IncrementBy, base::Unretained(&count), 3));
139 EXPECT_FALSE(cancelable.IsCancelled());
140
141 // The stale copy of the cancelable callback is non-null.
142 ASSERT_FALSE(callback.is_null());
143
144 // The stale copy of the cancelable callback is no longer active.
145 callback.Run();
146 EXPECT_EQ(2, count);
147
148 base::Closure callback2 = cancelable.callback();
149 ASSERT_FALSE(callback2.is_null());
150
151 callback2.Run();
152 EXPECT_EQ(5, count);
153}
154
155// IsCanceled().
156// - Cancel() transforms the CancelableCallback into a cancelled state.
157TEST(CancelableCallbackTest, IsNull) {
[email protected]d4d57df2011-11-30 20:33:52158 CancelableClosure cancelable;
[email protected]b77576f52011-11-24 04:12:04159 EXPECT_TRUE(cancelable.IsCancelled());
160
161 int count = 0;
162 cancelable.Reset(base::Bind(&Increment,
163 base::Unretained(&count)));
164 EXPECT_FALSE(cancelable.IsCancelled());
165
166 cancelable.Cancel();
167 EXPECT_TRUE(cancelable.IsCancelled());
168}
169
170// CancelableCallback posted to a MessageLoop with PostTask.
171// - Callbacks posted to a MessageLoop can be cancelled.
172TEST(CancelableCallbackTest, PostTask) {
[email protected]1ef90012014-01-15 22:24:33173 MessageLoop loop;
[email protected]b77576f52011-11-24 04:12:04174
175 int count = 0;
[email protected]d4d57df2011-11-30 20:33:52176 CancelableClosure cancelable(base::Bind(&Increment,
[email protected]b77576f52011-11-24 04:12:04177 base::Unretained(&count)));
178
skyostil054861d2015-04-30 19:06:15179 ThreadTaskRunnerHandle::Get()->PostTask(FROM_HERE, cancelable.callback());
[email protected]f7b98b32013-02-05 08:14:15180 RunLoop().RunUntilIdle();
[email protected]b77576f52011-11-24 04:12:04181
182 EXPECT_EQ(1, count);
183
skyostil054861d2015-04-30 19:06:15184 ThreadTaskRunnerHandle::Get()->PostTask(FROM_HERE, cancelable.callback());
[email protected]b77576f52011-11-24 04:12:04185
186 // Cancel before running the message loop.
187 cancelable.Cancel();
[email protected]f7b98b32013-02-05 08:14:15188 RunLoop().RunUntilIdle();
[email protected]b77576f52011-11-24 04:12:04189
190 // Callback never ran due to cancellation; count is the same.
191 EXPECT_EQ(1, count);
192}
193
kuznetsovsb020bcd82016-04-13 13:01:47194// CancelableCallback can be used with move-only types.
195TEST(CancelableCallbackTest, MoveOnlyType) {
196 const int kExpectedResult = 42;
197
198 int result = 0;
199 CancelableCallback<void(std::unique_ptr<int>)> cb(
200 base::Bind(&OnMoveOnlyReceived, base::Unretained(&result)));
201 cb.callback().Run(base::WrapUnique(new int(kExpectedResult)));
202
203 EXPECT_EQ(kExpectedResult, result);
204}
205
[email protected]b77576f52011-11-24 04:12:04206} // namespace
207} // namespace base