chromium/base/apple/bind_objc_block_unittest.mm

// 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 <string>

#include "base/functional/bind.h"
#include "base/functional/callback.h"
#include "base/functional/callback_helpers.h"
#include "testing/gtest/include/gtest/gtest.h"
#include "testing/gtest_mac.h"

@interface CounterForBindObjcBlockTest : NSObject
@property(nonatomic, assign) NSUInteger counter;
@end

@implementation CounterForBindObjcBlockTest
@synthesize counter = _counter;
@end

@interface HelperForBindObjcBlockTest : NSObject

- (instancetype)initWithCounter:(CounterForBindObjcBlockTest*)counter;
- (void)incrementCounter;

@end

@implementation HelperForBindObjcBlockTest {
  CounterForBindObjcBlockTest* _counter;
}

- (instancetype)initWithCounter:(CounterForBindObjcBlockTest*)counter {
  if ((self = [super init])) {
    _counter = counter;
    DCHECK(_counter);
  }
  return self;
}

- (void)incrementCounter {
  ++_counter.counter;
}

@end

namespace {

TEST(BindObjcBlockTest, TestScopedClosureRunnerExitScope) {
  int run_count = 0;
  int* ptr = &run_count;
  {
    base::ScopedClosureRunner runner(base::BindOnce(^{
      (*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(^{
      (*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<int(void)> c = base::BindOnce(^{
    return kReturnValue;
  });
  EXPECT_EQ(kReturnValue, std::move(c).Run());
}

TEST(BindObjcBlockTest, TestArgument) {
  const int kArgument = 42;
  base::OnceCallback<int(int)> c = base::BindOnce(^(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<void(const std::string&, const std::string&)> c =
      base::BindOnce(^(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<void(const std::string&, const std::string&,
                          const std::string&)>
      cb = base::BindOnce(
          ^(const std::string& a, const std::string& b, const std::string& c) {
            *ptr = a + b + c;
          });
  std::move(cb).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<void(int, int, const std::string&, const std::string&, int,
                          const std::string&)>
      cb = base::BindOnce(^(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(cb).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(
        ^(std::unique_ptr<BOOL> v) {
          invoked_block = *v;
        },
        std::make_unique<BOOL>(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 https://crbug.com/845687).
TEST(BindObjcBlockTest, TestBlockDeallocation) {
  base::RepeatingClosure closure;
  __block BOOL invoked_block = NO;
  closure = base::BindRepeating(
      ^(base::RepeatingClosure* this_closure) {
        *this_closure = base::RepeatingClosure();
        invoked_block = YES;
      },
      &closure);
  closure.Run();
  EXPECT_TRUE(invoked_block);
}

TEST(BindObjcBlockTest, TestBlockReleased) {
  __weak NSObject* weak_nsobject;
  @autoreleasepool {
    NSObject* nsobject = [[NSObject alloc] init];
    weak_nsobject = nsobject;

    auto callback = base::BindOnce(^{
      [nsobject description];
    });
  }
  EXPECT_NSEQ(nil, weak_nsobject);
}

// Tests that base::BindOnce(..., __strong NSObject*, ...) strongly captures
// the Objective-C object.
TEST(BindObjcBlockTest, TestBindOnceBoundStrongPointer) {
  CounterForBindObjcBlockTest* counter =
      [[CounterForBindObjcBlockTest alloc] init];
  ASSERT_EQ(counter.counter, 0u);

  base::OnceClosure closure;
  @autoreleasepool {
    HelperForBindObjcBlockTest* helper =
        [[HelperForBindObjcBlockTest alloc] initWithCounter:counter];

    // Creates a closure with a lambda taking the parameter as a __strong
    // pointer and bound with a __strong pointer. This should retain the
    // object.
    closure = base::BindOnce(
        [](HelperForBindObjcBlockTest* helper) { [helper incrementCounter]; },
        helper);
  }

  // Check that calling the closure increments the counter since the helper
  // object was captured strongly and thus is retained by the closure.
  std::move(closure).Run();
  EXPECT_EQ(counter.counter, 1u);
}

// Tests that base::BindOnce(..., __weak NSObject*, ...) weakly captures
// the Objective-C object.
TEST(BindObjcBlockTest, TestBindOnceBoundWeakPointer) {
  CounterForBindObjcBlockTest* counter =
      [[CounterForBindObjcBlockTest alloc] init];
  ASSERT_EQ(counter.counter, 0u);

  base::OnceClosure closure;
  @autoreleasepool {
    HelperForBindObjcBlockTest* helper =
        [[HelperForBindObjcBlockTest alloc] initWithCounter:counter];

    // Creates a closure with a lambda taking the parameter as a __strong
    // pointer and bound with a __weak pointer. This should not retain the
    // object.
    __weak HelperForBindObjcBlockTest* weak_helper = helper;
    closure = base::BindOnce(
        [](HelperForBindObjcBlockTest* helper) { [helper incrementCounter]; },
        weak_helper);
  }

  // Check that calling the closure does not increment the counter since
  // the helper object was captured weakly and thus is not retained by
  // the closure.
  std::move(closure).Run();
  EXPECT_EQ(counter.counter, 0u);
}

// Tests that base::BindRepeating(..., __strong NSObject*, ...) strongly
// captures the Objective-C object.
TEST(BindObjcBlockTest, TestBindRepeatingBoundStrongPointer) {
  CounterForBindObjcBlockTest* counter =
      [[CounterForBindObjcBlockTest alloc] init];
  ASSERT_EQ(counter.counter, 0u);

  base::RepeatingClosure closure;
  @autoreleasepool {
    HelperForBindObjcBlockTest* helper =
        [[HelperForBindObjcBlockTest alloc] initWithCounter:counter];

    // Creates a closure with a lambda taking the parameter as a __strong
    // pointer and bound with a __strong pointer. This should retain the
    // object.
    closure = base::BindRepeating(
        [](HelperForBindObjcBlockTest* helper) { [helper incrementCounter]; },
        helper);
  }

  // Check that calling the closure increments the counter since the helper
  // object was captured strongly and thus is retained by the closure.
  closure.Run();
  closure.Run();
  closure.Run();
  EXPECT_EQ(counter.counter, 3u);
}

// Tests that base::BindRepeating(..., __weak NSObject*, ...) weakly captures
// the Objective-C object.
TEST(BindObjcBlockTest, TestBindRepeatingBoundWeakPointer) {
  CounterForBindObjcBlockTest* counter =
      [[CounterForBindObjcBlockTest alloc] init];
  ASSERT_EQ(counter.counter, 0u);

  base::RepeatingClosure closure;
  @autoreleasepool {
    HelperForBindObjcBlockTest* helper =
        [[HelperForBindObjcBlockTest alloc] initWithCounter:counter];

    // Creates a closure with a lambda taking the parameter as a __strong
    // pointer and bound with a __weak pointer. This should not retain the
    // object.
    __weak HelperForBindObjcBlockTest* weak_helper = helper;
    closure = base::BindRepeating(
        [](HelperForBindObjcBlockTest* helper) { [helper incrementCounter]; },
        weak_helper);
  }

  // Check that calling the closure does not increment the counter since
  // the helper object was captured weakly and thus is not retained by
  // the closure.
  closure.Run();
  closure.Run();
  closure.Run();
  EXPECT_EQ(counter.counter, 0u);
}

}  // namespace