301 lines
10 KiB
C++
301 lines
10 KiB
C++
|
|
// Copyright 2013 The Chromium Authors
|
||
|
|
// Use of this source code is governed by a BSD-style license that can be
|
||
|
|
// found in the LICENSE file.
|
||
|
|
|
||
|
|
#include "base/functional/callback_helpers.h"
|
||
|
|
|
||
|
|
#include <functional>
|
||
|
|
#include <type_traits>
|
||
|
|
|
||
|
|
#include "base/functional/bind.h"
|
||
|
|
#include "base/functional/callback.h"
|
||
|
|
#include "base/test/gtest_util.h"
|
||
|
|
#include "testing/gtest/include/gtest/gtest.h"
|
||
|
|
|
||
|
|
namespace {
|
||
|
|
|
||
|
|
TEST(CallbackHelpersTest, IsBaseCallback) {
|
||
|
|
// Check that base::{Once,Repeating}Closures and references to them are
|
||
|
|
// considered base::{Once,Repeating}Callbacks.
|
||
|
|
static_assert(base::IsBaseCallback<base::OnceClosure>::value, "");
|
||
|
|
static_assert(base::IsBaseCallback<base::RepeatingClosure>::value, "");
|
||
|
|
static_assert(base::IsBaseCallback<base::OnceClosure&&>::value, "");
|
||
|
|
static_assert(base::IsBaseCallback<const base::RepeatingClosure&>::value, "");
|
||
|
|
|
||
|
|
// Check that base::{Once, Repeating}Callbacks with a given RunType and
|
||
|
|
// references to them are considered base::{Once, Repeating}Callbacks.
|
||
|
|
static_assert(base::IsBaseCallback<base::OnceCallback<int(int)>>::value, "");
|
||
|
|
static_assert(base::IsBaseCallback<base::RepeatingCallback<int(int)>>::value,
|
||
|
|
"");
|
||
|
|
static_assert(base::IsBaseCallback<base::OnceCallback<int(int)>&&>::value,
|
||
|
|
"");
|
||
|
|
static_assert(
|
||
|
|
base::IsBaseCallback<const base::RepeatingCallback<int(int)>&>::value,
|
||
|
|
"");
|
||
|
|
|
||
|
|
// Check that POD types are not considered base::{Once, Repeating}Callbacks.
|
||
|
|
static_assert(!base::IsBaseCallback<bool>::value, "");
|
||
|
|
static_assert(!base::IsBaseCallback<int>::value, "");
|
||
|
|
static_assert(!base::IsBaseCallback<double>::value, "");
|
||
|
|
|
||
|
|
// Check that the closely related std::function is not considered a
|
||
|
|
// base::{Once, Repeating}Callback.
|
||
|
|
static_assert(!base::IsBaseCallback<std::function<void()>>::value, "");
|
||
|
|
static_assert(!base::IsBaseCallback<const std::function<void()>&>::value, "");
|
||
|
|
static_assert(!base::IsBaseCallback<std::function<void()>&&>::value, "");
|
||
|
|
}
|
||
|
|
|
||
|
|
TEST(CallbackHelpersTest, IsOnceCallback) {
|
||
|
|
// Check that base::OnceClosures and references to them are considered
|
||
|
|
// base::OnceCallbacks, but base::RepeatingClosures are not.
|
||
|
|
static_assert(base::IsOnceCallback<base::OnceClosure>::value, "");
|
||
|
|
static_assert(!base::IsOnceCallback<base::RepeatingClosure>::value, "");
|
||
|
|
static_assert(base::IsOnceCallback<base::OnceClosure&&>::value, "");
|
||
|
|
static_assert(!base::IsOnceCallback<const base::RepeatingClosure&>::value,
|
||
|
|
"");
|
||
|
|
|
||
|
|
// Check that base::OnceCallbacks with a given RunType and references to them
|
||
|
|
// are considered base::OnceCallbacks, but base::RepeatingCallbacks are not.
|
||
|
|
static_assert(base::IsOnceCallback<base::OnceCallback<int(int)>>::value, "");
|
||
|
|
static_assert(!base::IsOnceCallback<base::RepeatingCallback<int(int)>>::value,
|
||
|
|
"");
|
||
|
|
static_assert(base::IsOnceCallback<base::OnceCallback<int(int)>&&>::value,
|
||
|
|
"");
|
||
|
|
static_assert(
|
||
|
|
!base::IsOnceCallback<const base::RepeatingCallback<int(int)>&>::value,
|
||
|
|
"");
|
||
|
|
|
||
|
|
// Check that POD types are not considered base::OnceCallbacks.
|
||
|
|
static_assert(!base::IsOnceCallback<bool>::value, "");
|
||
|
|
static_assert(!base::IsOnceCallback<int>::value, "");
|
||
|
|
static_assert(!base::IsOnceCallback<double>::value, "");
|
||
|
|
|
||
|
|
// Check that the closely related std::function is not considered a
|
||
|
|
// base::OnceCallback.
|
||
|
|
static_assert(!base::IsOnceCallback<std::function<void()>>::value, "");
|
||
|
|
static_assert(!base::IsOnceCallback<const std::function<void()>&>::value, "");
|
||
|
|
static_assert(!base::IsOnceCallback<std::function<void()>&&>::value, "");
|
||
|
|
|
||
|
|
// Check that the result of BindOnce is a OnceCallback.
|
||
|
|
auto cb = base::BindOnce([](int* count) { ++*count; });
|
||
|
|
static_assert(base::IsOnceCallback<decltype(cb)>::value, "");
|
||
|
|
}
|
||
|
|
|
||
|
|
void Increment(int* value) {
|
||
|
|
(*value)++;
|
||
|
|
}
|
||
|
|
|
||
|
|
TEST(CallbackHelpersTest, ScopedClosureRunnerHasClosure) {
|
||
|
|
base::ScopedClosureRunner runner1;
|
||
|
|
EXPECT_FALSE(runner1);
|
||
|
|
|
||
|
|
base::ScopedClosureRunner runner2{base::DoNothing()};
|
||
|
|
EXPECT_TRUE(runner2);
|
||
|
|
}
|
||
|
|
|
||
|
|
TEST(CallbackHelpersTest, ScopedClosureRunnerExitScope) {
|
||
|
|
int run_count = 0;
|
||
|
|
{
|
||
|
|
base::ScopedClosureRunner runner(base::BindOnce(&Increment, &run_count));
|
||
|
|
EXPECT_EQ(0, run_count);
|
||
|
|
}
|
||
|
|
EXPECT_EQ(1, run_count);
|
||
|
|
}
|
||
|
|
|
||
|
|
TEST(CallbackHelpersTest, ScopedClosureRunnerRelease) {
|
||
|
|
int run_count = 0;
|
||
|
|
base::OnceClosure c;
|
||
|
|
{
|
||
|
|
base::ScopedClosureRunner runner(base::BindOnce(&Increment, &run_count));
|
||
|
|
c = runner.Release();
|
||
|
|
EXPECT_EQ(0, run_count);
|
||
|
|
}
|
||
|
|
EXPECT_EQ(0, run_count);
|
||
|
|
std::move(c).Run();
|
||
|
|
EXPECT_EQ(1, run_count);
|
||
|
|
}
|
||
|
|
|
||
|
|
TEST(CallbackHelpersTest, ScopedClosureRunnerReplaceClosure) {
|
||
|
|
int run_count_1 = 0;
|
||
|
|
int run_count_2 = 0;
|
||
|
|
{
|
||
|
|
base::ScopedClosureRunner runner;
|
||
|
|
runner.ReplaceClosure(base::BindOnce(&Increment, &run_count_1));
|
||
|
|
runner.ReplaceClosure(base::BindOnce(&Increment, &run_count_2));
|
||
|
|
EXPECT_EQ(0, run_count_1);
|
||
|
|
EXPECT_EQ(0, run_count_2);
|
||
|
|
}
|
||
|
|
EXPECT_EQ(0, run_count_1);
|
||
|
|
EXPECT_EQ(1, run_count_2);
|
||
|
|
}
|
||
|
|
|
||
|
|
TEST(CallbackHelpersTest, ScopedClosureRunnerRunAndResetNonNull) {
|
||
|
|
int run_count_3 = 0;
|
||
|
|
{
|
||
|
|
base::ScopedClosureRunner runner(base::BindOnce(&Increment, &run_count_3));
|
||
|
|
EXPECT_EQ(0, run_count_3);
|
||
|
|
runner.RunAndReset();
|
||
|
|
EXPECT_EQ(1, run_count_3);
|
||
|
|
}
|
||
|
|
EXPECT_EQ(1, run_count_3);
|
||
|
|
}
|
||
|
|
|
||
|
|
TEST(CallbackHelpersTest, ScopedClosureRunnerRunAndResetNull) {
|
||
|
|
base::ScopedClosureRunner runner;
|
||
|
|
runner.RunAndReset(); // Should not crash.
|
||
|
|
}
|
||
|
|
|
||
|
|
TEST(CallbackHelpersTest, ScopedClosureRunnerMoveConstructor) {
|
||
|
|
int run_count = 0;
|
||
|
|
{
|
||
|
|
std::unique_ptr<base::ScopedClosureRunner> runner(
|
||
|
|
new base::ScopedClosureRunner(base::BindOnce(&Increment, &run_count)));
|
||
|
|
base::ScopedClosureRunner runner2(std::move(*runner));
|
||
|
|
runner.reset();
|
||
|
|
EXPECT_EQ(0, run_count);
|
||
|
|
}
|
||
|
|
EXPECT_EQ(1, run_count);
|
||
|
|
}
|
||
|
|
|
||
|
|
TEST(CallbackHelpersTest, ScopedClosureRunnerMoveAssignment) {
|
||
|
|
int run_count_1 = 0;
|
||
|
|
int run_count_2 = 0;
|
||
|
|
{
|
||
|
|
base::ScopedClosureRunner runner(base::BindOnce(&Increment, &run_count_1));
|
||
|
|
{
|
||
|
|
base::ScopedClosureRunner runner2(
|
||
|
|
base::BindOnce(&Increment, &run_count_2));
|
||
|
|
runner = std::move(runner2);
|
||
|
|
EXPECT_EQ(1, run_count_1);
|
||
|
|
EXPECT_EQ(0, run_count_2);
|
||
|
|
}
|
||
|
|
EXPECT_EQ(1, run_count_1);
|
||
|
|
EXPECT_EQ(0, run_count_2);
|
||
|
|
}
|
||
|
|
EXPECT_EQ(1, run_count_1);
|
||
|
|
EXPECT_EQ(1, run_count_2);
|
||
|
|
}
|
||
|
|
|
||
|
|
TEST(CallbackHelpersTest, SplitOnceCallback_EmptyCallback) {
|
||
|
|
base::OnceCallback<void(int*)> cb = base::NullCallback();
|
||
|
|
EXPECT_FALSE(cb);
|
||
|
|
|
||
|
|
auto split = base::SplitOnceCallback(std::move(cb));
|
||
|
|
|
||
|
|
static_assert(std::is_same<decltype(split),
|
||
|
|
std::pair<base::OnceCallback<void(int*)>,
|
||
|
|
base::OnceCallback<void(int*)>>>::value,
|
||
|
|
"");
|
||
|
|
EXPECT_FALSE(split.first);
|
||
|
|
EXPECT_FALSE(split.second);
|
||
|
|
}
|
||
|
|
|
||
|
|
TEST(CallbackHelpersTest, SplitOnceCallback_FirstCallback) {
|
||
|
|
int count = 0;
|
||
|
|
base::OnceCallback<void(int*)> cb =
|
||
|
|
base::BindOnce([](int* count) { ++*count; });
|
||
|
|
|
||
|
|
auto split = base::SplitOnceCallback(std::move(cb));
|
||
|
|
|
||
|
|
static_assert(std::is_same<decltype(split),
|
||
|
|
std::pair<base::OnceCallback<void(int*)>,
|
||
|
|
base::OnceCallback<void(int*)>>>::value,
|
||
|
|
"");
|
||
|
|
|
||
|
|
EXPECT_EQ(0, count);
|
||
|
|
std::move(split.first).Run(&count);
|
||
|
|
EXPECT_EQ(1, count);
|
||
|
|
|
||
|
|
#if GTEST_HAS_DEATH_TEST
|
||
|
|
EXPECT_CHECK_DEATH(std::move(split.second).Run(&count));
|
||
|
|
#endif // GTEST_HAS_DEATH_TEST
|
||
|
|
}
|
||
|
|
|
||
|
|
TEST(CallbackHelpersTest, SplitOnceCallback_SecondCallback) {
|
||
|
|
int count = 0;
|
||
|
|
base::OnceCallback<void(int*)> cb =
|
||
|
|
base::BindOnce([](int* count) { ++*count; });
|
||
|
|
|
||
|
|
auto split = base::SplitOnceCallback(std::move(cb));
|
||
|
|
|
||
|
|
static_assert(std::is_same<decltype(split),
|
||
|
|
std::pair<base::OnceCallback<void(int*)>,
|
||
|
|
base::OnceCallback<void(int*)>>>::value,
|
||
|
|
"");
|
||
|
|
|
||
|
|
EXPECT_EQ(0, count);
|
||
|
|
std::move(split.second).Run(&count);
|
||
|
|
EXPECT_EQ(1, count);
|
||
|
|
|
||
|
|
EXPECT_CHECK_DEATH(std::move(split.first).Run(&count));
|
||
|
|
}
|
||
|
|
|
||
|
|
TEST(CallbackHelpersTest, SplitSplitOnceCallback_FirstSplit) {
|
||
|
|
int count = 0;
|
||
|
|
base::OnceCallback<void(int*)> cb =
|
||
|
|
base::BindOnce([](int* count) { ++*count; });
|
||
|
|
|
||
|
|
auto split = base::SplitOnceCallback(std::move(cb));
|
||
|
|
base::OnceCallback<void(int*)> cb1 = std::move(split.first);
|
||
|
|
split = base::SplitOnceCallback(std::move(split.second));
|
||
|
|
base::OnceCallback<void(int*)> cb2 = std::move(split.first);
|
||
|
|
base::OnceCallback<void(int*)> cb3 = std::move(split.second);
|
||
|
|
|
||
|
|
EXPECT_EQ(0, count);
|
||
|
|
std::move(cb1).Run(&count);
|
||
|
|
EXPECT_EQ(1, count);
|
||
|
|
|
||
|
|
EXPECT_CHECK_DEATH(std::move(cb3).Run(&count));
|
||
|
|
}
|
||
|
|
|
||
|
|
TEST(CallbackHelpersTest, SplitSplitOnceCallback_SecondSplit) {
|
||
|
|
int count = 0;
|
||
|
|
base::OnceCallback<void(int*)> cb =
|
||
|
|
base::BindOnce([](int* count) { ++*count; });
|
||
|
|
|
||
|
|
auto split = base::SplitOnceCallback(std::move(cb));
|
||
|
|
base::OnceCallback<void(int*)> cb1 = std::move(split.first);
|
||
|
|
split = base::SplitOnceCallback(std::move(split.second));
|
||
|
|
base::OnceCallback<void(int*)> cb2 = std::move(split.first);
|
||
|
|
base::OnceCallback<void(int*)> cb3 = std::move(split.second);
|
||
|
|
|
||
|
|
EXPECT_EQ(0, count);
|
||
|
|
std::move(cb2).Run(&count);
|
||
|
|
EXPECT_EQ(1, count);
|
||
|
|
|
||
|
|
EXPECT_CHECK_DEATH(std::move(cb1).Run(&count));
|
||
|
|
}
|
||
|
|
|
||
|
|
TEST(CallbackHelpersTest, IgnoreArgs) {
|
||
|
|
int count = 0;
|
||
|
|
base::RepeatingClosure repeating_closure =
|
||
|
|
base::BindRepeating(&Increment, &count);
|
||
|
|
base::OnceClosure once_closure = base::BindOnce(&Increment, &count);
|
||
|
|
|
||
|
|
base::RepeatingCallback<void(int)> repeating_int_cb =
|
||
|
|
base::IgnoreArgs<int>(repeating_closure);
|
||
|
|
EXPECT_EQ(0, count);
|
||
|
|
repeating_int_cb.Run(42);
|
||
|
|
EXPECT_EQ(1, count);
|
||
|
|
repeating_int_cb.Run(42);
|
||
|
|
EXPECT_EQ(2, count);
|
||
|
|
|
||
|
|
base::OnceCallback<void(int)> once_int_cb =
|
||
|
|
base::IgnoreArgs<int>(std::move(once_closure));
|
||
|
|
EXPECT_EQ(2, count);
|
||
|
|
std::move(once_int_cb).Run(42);
|
||
|
|
EXPECT_EQ(3, count);
|
||
|
|
}
|
||
|
|
|
||
|
|
TEST(CallbackHelpersTest, IgnoreArgs_EmptyCallback) {
|
||
|
|
base::RepeatingCallback<void(int)> repeating_int_cb =
|
||
|
|
base::IgnoreArgs<int>(base::RepeatingClosure());
|
||
|
|
EXPECT_FALSE(repeating_int_cb);
|
||
|
|
|
||
|
|
base::OnceCallback<void(int)> once_int_cb =
|
||
|
|
base::IgnoreArgs<int>(base::OnceClosure());
|
||
|
|
EXPECT_FALSE(once_int_cb);
|
||
|
|
}
|
||
|
|
|
||
|
|
} // namespace
|