// Copyright 2012 The Chromium Authors // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #include #include "base/functional/bind.h" #include "base/functional/callback.h" #include "base/functional/callback_helpers.h" #include "build/build_config.h" #include "testing/gtest/include/gtest/gtest.h" #include "testing/gtest_mac.h" namespace { TEST(BindObjcBlockTest, TestScopedClosureRunnerExitScope) { int run_count = 0; int* ptr = &run_count; { base::ScopedClosureRunner runner(base::BindOnce(base::RetainBlock(^{ (*ptr)++; }))); EXPECT_EQ(0, run_count); } EXPECT_EQ(1, run_count); } TEST(BindObjcBlockTest, TestScopedClosureRunnerRelease) { int run_count = 0; int* ptr = &run_count; base::OnceClosure c; { base::ScopedClosureRunner runner(base::BindOnce(base::RetainBlock(^{ (*ptr)++; }))); c = runner.Release(); EXPECT_EQ(0, run_count); } EXPECT_EQ(0, run_count); std::move(c).Run(); EXPECT_EQ(1, run_count); } TEST(BindObjcBlockTest, TestReturnValue) { const int kReturnValue = 42; base::OnceCallback c = base::BindOnce(base::RetainBlock(^{ return kReturnValue; })); EXPECT_EQ(kReturnValue, std::move(c).Run()); } TEST(BindObjcBlockTest, TestArgument) { const int kArgument = 42; base::OnceCallback c = base::BindOnce(base::RetainBlock(^(int a) { return a + 1; })); EXPECT_EQ(kArgument + 1, std::move(c).Run(kArgument)); } TEST(BindObjcBlockTest, TestTwoArguments) { std::string result; std::string* ptr = &result; base::OnceCallback c = base::BindOnce( base::RetainBlock(^(const std::string& a, const std::string& b) { *ptr = a + b; })); std::move(c).Run("forty", "two"); EXPECT_EQ(result, "fortytwo"); } TEST(BindObjcBlockTest, TestThreeArguments) { std::string result; std::string* ptr = &result; base::OnceCallback callback = base::BindOnce(base::RetainBlock( ^(const std::string& a, const std::string& b, const std::string& c) { *ptr = a + b + c; })); std::move(callback).Run("six", "times", "nine"); EXPECT_EQ(result, "sixtimesnine"); } TEST(BindObjcBlockTest, TestSixArguments) { std::string result1; std::string* ptr = &result1; int result2; int* ptr2 = &result2; base::OnceCallback callback = base::BindOnce(base::RetainBlock( ^(int a, int b, const std::string& c, const std::string& d, int e, const std::string& f) { *ptr = c + d + f; *ptr2 = a + b + e; })); std::move(callback).Run(1, 2, "infinite", "improbability", 3, "drive"); EXPECT_EQ(result1, "infiniteimprobabilitydrive"); EXPECT_EQ(result2, 6); } TEST(BindObjcBlockTest, TestBlockMoveable) { base::OnceClosure c; __block BOOL invoked_block = NO; @autoreleasepool { c = base::BindOnce(base::RetainBlock(^(std::unique_ptr v) { invoked_block = *v; }), std::make_unique(YES)); }; std::move(c).Run(); EXPECT_TRUE(invoked_block); } // Tests that the bound block is retained until the end of its execution, // even if the callback itself is destroyed during the invocation. It was // found that some code depends on this behaviour (see crbug.com/845687). TEST(BindObjcBlockTest, TestBlockDeallocation) { base::RepeatingClosure closure; __block BOOL invoked_block = NO; closure = base::BindRepeating( base::RetainBlock(^(base::RepeatingClosure* this_closure) { *this_closure = base::RepeatingClosure(); invoked_block = YES; }), &closure); closure.Run(); EXPECT_TRUE(invoked_block); } } // namespace