|  | // Copyright (c) 2012 The Chromium Authors. All rights reserved. | 
|  | // Use of this source code is governed by a BSD-style license that can be | 
|  | // found in the LICENSE file. | 
|  |  | 
|  | #include <vector> | 
|  |  | 
|  | #import <CoreFoundation/CoreFoundation.h> | 
|  |  | 
|  | #include "base/logging.h" | 
|  | #import "base/mac/scoped_nsobject.h" | 
|  | #include "testing/gtest/include/gtest/gtest.h" | 
|  |  | 
|  | #if !defined(__has_feature) || !__has_feature(objc_arc) | 
|  | #error "This file requires ARC support." | 
|  | #endif | 
|  |  | 
|  | namespace { | 
|  |  | 
|  | template <typename NST> | 
|  | CFIndex GetRetainCount(const base::scoped_nsobject<NST>& nst) { | 
|  | @autoreleasepool { | 
|  | return CFGetRetainCount((__bridge CFTypeRef)nst.get()) - 1; | 
|  | } | 
|  | } | 
|  |  | 
|  | #if __has_feature(objc_arc_weak) | 
|  | TEST(ScopedNSObjectTestARC, DefaultPolicyIsRetain) { | 
|  | __weak id o; | 
|  | @autoreleasepool { | 
|  | base::scoped_nsprotocol<id> p([[NSObject alloc] init]); | 
|  | o = p.get(); | 
|  | DCHECK_EQ(o, p.get()); | 
|  | } | 
|  | DCHECK_EQ(o, nil); | 
|  | } | 
|  | #endif | 
|  |  | 
|  | TEST(ScopedNSObjectTestARC, ScopedNSObject) { | 
|  | base::scoped_nsobject<NSObject> p1([[NSObject alloc] init]); | 
|  | @autoreleasepool { | 
|  | EXPECT_TRUE(p1.get()); | 
|  | EXPECT_TRUE(p1.get()); | 
|  | } | 
|  | EXPECT_EQ(1, GetRetainCount(p1)); | 
|  | EXPECT_EQ(1, GetRetainCount(p1)); | 
|  | base::scoped_nsobject<NSObject> p2(p1); | 
|  | @autoreleasepool { | 
|  | EXPECT_EQ(p1.get(), p2.get()); | 
|  | } | 
|  | EXPECT_EQ(2, GetRetainCount(p1)); | 
|  | p2.reset(); | 
|  | EXPECT_EQ(nil, p2.get()); | 
|  | EXPECT_EQ(1, GetRetainCount(p1)); | 
|  | { | 
|  | base::scoped_nsobject<NSObject> p3 = p1; | 
|  | @autoreleasepool { | 
|  | EXPECT_EQ(p1.get(), p3.get()); | 
|  | } | 
|  | EXPECT_EQ(2, GetRetainCount(p1)); | 
|  | @autoreleasepool { | 
|  | p3 = p1; | 
|  | EXPECT_EQ(p1.get(), p3.get()); | 
|  | } | 
|  | EXPECT_EQ(2, GetRetainCount(p1)); | 
|  | } | 
|  | EXPECT_EQ(1, GetRetainCount(p1)); | 
|  | base::scoped_nsobject<NSObject> p4; | 
|  | @autoreleasepool { | 
|  | p4 = base::scoped_nsobject<NSObject>(p1.get()); | 
|  | } | 
|  | EXPECT_EQ(2, GetRetainCount(p1)); | 
|  | @autoreleasepool { | 
|  | EXPECT_TRUE(p1 == p1.get()); | 
|  | EXPECT_TRUE(p1 == p1); | 
|  | EXPECT_FALSE(p1 != p1); | 
|  | EXPECT_FALSE(p1 != p1.get()); | 
|  | } | 
|  | base::scoped_nsobject<NSObject> p5([[NSObject alloc] init]); | 
|  | @autoreleasepool { | 
|  | EXPECT_TRUE(p1 != p5); | 
|  | EXPECT_TRUE(p1 != p5.get()); | 
|  | EXPECT_FALSE(p1 == p5); | 
|  | EXPECT_FALSE(p1 == p5.get()); | 
|  | } | 
|  |  | 
|  | base::scoped_nsobject<NSObject> p6 = p1; | 
|  | EXPECT_EQ(3, GetRetainCount(p6)); | 
|  | @autoreleasepool { | 
|  | p6.autorelease(); | 
|  | EXPECT_EQ(nil, p6.get()); | 
|  | } | 
|  | EXPECT_EQ(2, GetRetainCount(p1)); | 
|  | } | 
|  |  | 
|  | TEST(ScopedNSObjectTestARC, ScopedNSObjectInContainer) { | 
|  | base::scoped_nsobject<id> p([[NSObject alloc] init]); | 
|  | @autoreleasepool { | 
|  | EXPECT_TRUE(p.get()); | 
|  | } | 
|  | EXPECT_EQ(1, GetRetainCount(p)); | 
|  | @autoreleasepool { | 
|  | std::vector<base::scoped_nsobject<id>> objects; | 
|  | objects.push_back(p); | 
|  | EXPECT_EQ(2, GetRetainCount(p)); | 
|  | @autoreleasepool { | 
|  | EXPECT_EQ(p.get(), objects[0].get()); | 
|  | } | 
|  | objects.push_back(base::scoped_nsobject<id>([[NSObject alloc] init])); | 
|  | @autoreleasepool { | 
|  | EXPECT_TRUE(objects[1].get()); | 
|  | } | 
|  | EXPECT_EQ(1, GetRetainCount(objects[1])); | 
|  | } | 
|  | EXPECT_EQ(1, GetRetainCount(p)); | 
|  | } | 
|  |  | 
|  | TEST(ScopedNSObjectTestARC, ScopedNSObjectFreeFunctions) { | 
|  | base::scoped_nsobject<id> p1([[NSObject alloc] init]); | 
|  | id o1 = p1.get(); | 
|  | EXPECT_TRUE(o1 == p1); | 
|  | EXPECT_FALSE(o1 != p1); | 
|  | base::scoped_nsobject<id> p2([[NSObject alloc] init]); | 
|  | EXPECT_TRUE(o1 != p2); | 
|  | EXPECT_FALSE(o1 == p2); | 
|  | id o2 = p2.get(); | 
|  | swap(p1, p2); | 
|  | EXPECT_EQ(o2, p1.get()); | 
|  | EXPECT_EQ(o1, p2.get()); | 
|  | } | 
|  |  | 
|  | }  // namespace |