blob: fcbe23c1de34acbda3eca9cf56587a97594256c1 [file] [log] [blame]
jhawkins@chromium.org78936ff2011-11-24 13:12:04 +09001// 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
7#include "base/bind.h"
8#include "base/bind_helpers.h"
9#include "base/memory/ref_counted.h"
avi@chromium.orga043a862013-07-18 17:12:40 +090010#include "base/message_loop/message_loop.h"
tfarina@chromium.orge9e11f52013-02-05 17:14:15 +090011#include "base/run_loop.h"
jhawkins@chromium.org78936ff2011-11-24 13:12:04 +090012#include "testing/gtest/include/gtest/gtest.h"
13
14namespace base {
15namespace {
16
17class TestRefCounted : public RefCountedThreadSafe<TestRefCounted> {
18 private:
19 friend class RefCountedThreadSafe<TestRefCounted>;
20 ~TestRefCounted() {};
21};
22
23void Increment(int* count) { (*count)++; }
24void IncrementBy(int* count, int n) { (*count) += n; }
25void RefCountedParam(const scoped_refptr<TestRefCounted>& ref_counted) {}
26
27// Cancel().
28// - Callback can be run multiple times.
29// - After Cancel(), Run() completes but has no effect.
30TEST(CancelableCallbackTest, Cancel) {
31 int count = 0;
jhawkins@chromium.org3250e172011-12-01 05:33:52 +090032 CancelableClosure cancelable(
jhawkins@chromium.org78936ff2011-11-24 13:12:04 +090033 base::Bind(&Increment, base::Unretained(&count)));
34
35 base::Closure callback = cancelable.callback();
36 callback.Run();
37 EXPECT_EQ(1, count);
38
39 callback.Run();
40 EXPECT_EQ(2, count);
41
42 cancelable.Cancel();
43 callback.Run();
44 EXPECT_EQ(2, count);
45}
46
47// Cancel() called multiple times.
48// - Cancel() cancels all copies of the wrapped callback.
jhawkins@chromium.org3250e172011-12-01 05:33:52 +090049// - Calling Cancel() more than once has no effect.
50// - After Cancel(), callback() returns a null callback.
jhawkins@chromium.org78936ff2011-11-24 13:12:04 +090051TEST(CancelableCallbackTest, MultipleCancel) {
52 int count = 0;
jhawkins@chromium.org3250e172011-12-01 05:33:52 +090053 CancelableClosure cancelable(
jhawkins@chromium.org78936ff2011-11-24 13:12:04 +090054 base::Bind(&Increment, base::Unretained(&count)));
55
56 base::Closure callback1 = cancelable.callback();
jhawkins@chromium.org3250e172011-12-01 05:33:52 +090057 base::Closure callback2 = cancelable.callback();
jhawkins@chromium.org78936ff2011-11-24 13:12:04 +090058 cancelable.Cancel();
59
60 callback1.Run();
61 EXPECT_EQ(0, count);
62
jhawkins@chromium.org78936ff2011-11-24 13:12:04 +090063 callback2.Run();
64 EXPECT_EQ(0, count);
65
jhawkins@chromium.org3250e172011-12-01 05:33:52 +090066 // Calling Cancel() again has no effect.
jhawkins@chromium.org78936ff2011-11-24 13:12:04 +090067 cancelable.Cancel();
jhawkins@chromium.org3250e172011-12-01 05:33:52 +090068
69 // callback() of a cancelled callback is null.
70 base::Closure callback3 = cancelable.callback();
71 EXPECT_TRUE(callback3.is_null());
jhawkins@chromium.org78936ff2011-11-24 13:12:04 +090072}
73
74// CancelableCallback destroyed before callback is run.
75// - Destruction of CancelableCallback cancels outstanding callbacks.
76TEST(CancelableCallbackTest, CallbackCanceledOnDestruction) {
77 int count = 0;
78 base::Closure callback;
79
80 {
jhawkins@chromium.org3250e172011-12-01 05:33:52 +090081 CancelableClosure cancelable(
jhawkins@chromium.org78936ff2011-11-24 13:12:04 +090082 base::Bind(&Increment, base::Unretained(&count)));
83
84 callback = cancelable.callback();
85 callback.Run();
86 EXPECT_EQ(1, count);
87 }
88
89 callback.Run();
90 EXPECT_EQ(1, count);
91}
92
93// Cancel() called on bound closure with a RefCounted parameter.
94// - Cancel drops wrapped callback (and, implicitly, its bound arguments).
95TEST(CancelableCallbackTest, CancelDropsCallback) {
96 scoped_refptr<TestRefCounted> ref_counted = new TestRefCounted;
97 EXPECT_TRUE(ref_counted->HasOneRef());
98
jhawkins@chromium.org3250e172011-12-01 05:33:52 +090099 CancelableClosure cancelable(base::Bind(RefCountedParam, ref_counted));
jhawkins@chromium.org78936ff2011-11-24 13:12:04 +0900100 EXPECT_FALSE(cancelable.IsCancelled());
101 EXPECT_TRUE(ref_counted.get());
102 EXPECT_FALSE(ref_counted->HasOneRef());
103
104 // There is only one reference to |ref_counted| after the Cancel().
105 cancelable.Cancel();
106 EXPECT_TRUE(cancelable.IsCancelled());
107 EXPECT_TRUE(ref_counted.get());
108 EXPECT_TRUE(ref_counted->HasOneRef());
109}
110
111// Reset().
112// - Reset() replaces the existing wrapped callback with a new callback.
113// - Reset() deactivates outstanding callbacks.
114TEST(CancelableCallbackTest, Reset) {
115 int count = 0;
jhawkins@chromium.org3250e172011-12-01 05:33:52 +0900116 CancelableClosure cancelable(
jhawkins@chromium.org78936ff2011-11-24 13:12:04 +0900117 base::Bind(&Increment, base::Unretained(&count)));
118
119 base::Closure callback = cancelable.callback();
120 callback.Run();
121 EXPECT_EQ(1, count);
122
123 callback.Run();
124 EXPECT_EQ(2, count);
125
126 cancelable.Reset(
127 base::Bind(&IncrementBy, base::Unretained(&count), 3));
128 EXPECT_FALSE(cancelable.IsCancelled());
129
130 // The stale copy of the cancelable callback is non-null.
131 ASSERT_FALSE(callback.is_null());
132
133 // The stale copy of the cancelable callback is no longer active.
134 callback.Run();
135 EXPECT_EQ(2, count);
136
137 base::Closure callback2 = cancelable.callback();
138 ASSERT_FALSE(callback2.is_null());
139
140 callback2.Run();
141 EXPECT_EQ(5, count);
142}
143
144// IsCanceled().
145// - Cancel() transforms the CancelableCallback into a cancelled state.
146TEST(CancelableCallbackTest, IsNull) {
jhawkins@chromium.org3250e172011-12-01 05:33:52 +0900147 CancelableClosure cancelable;
jhawkins@chromium.org78936ff2011-11-24 13:12:04 +0900148 EXPECT_TRUE(cancelable.IsCancelled());
149
150 int count = 0;
151 cancelable.Reset(base::Bind(&Increment,
152 base::Unretained(&count)));
153 EXPECT_FALSE(cancelable.IsCancelled());
154
155 cancelable.Cancel();
156 EXPECT_TRUE(cancelable.IsCancelled());
157}
158
159// CancelableCallback posted to a MessageLoop with PostTask.
160// - Callbacks posted to a MessageLoop can be cancelled.
161TEST(CancelableCallbackTest, PostTask) {
sky@chromium.orgdb9105a2014-01-16 07:24:33 +0900162 MessageLoop loop;
jhawkins@chromium.org78936ff2011-11-24 13:12:04 +0900163
164 int count = 0;
jhawkins@chromium.org3250e172011-12-01 05:33:52 +0900165 CancelableClosure cancelable(base::Bind(&Increment,
jhawkins@chromium.org78936ff2011-11-24 13:12:04 +0900166 base::Unretained(&count)));
167
168 MessageLoop::current()->PostTask(FROM_HERE, cancelable.callback());
tfarina@chromium.orge9e11f52013-02-05 17:14:15 +0900169 RunLoop().RunUntilIdle();
jhawkins@chromium.org78936ff2011-11-24 13:12:04 +0900170
171 EXPECT_EQ(1, count);
172
173 MessageLoop::current()->PostTask(FROM_HERE, cancelable.callback());
jhawkins@chromium.org78936ff2011-11-24 13:12:04 +0900174
175 // Cancel before running the message loop.
176 cancelable.Cancel();
tfarina@chromium.orge9e11f52013-02-05 17:14:15 +0900177 RunLoop().RunUntilIdle();
jhawkins@chromium.org78936ff2011-11-24 13:12:04 +0900178
179 // Callback never ran due to cancellation; count is the same.
180 EXPECT_EQ(1, count);
181}
182
183} // namespace
184} // namespace base