1cb93a386Sopenharmony_ci/*
2cb93a386Sopenharmony_ci * Copyright 2012 Google Inc.
3cb93a386Sopenharmony_ci *
4cb93a386Sopenharmony_ci * Use of this source code is governed by a BSD-style license that can be
5cb93a386Sopenharmony_ci * found in the LICENSE file.
6cb93a386Sopenharmony_ci */
7cb93a386Sopenharmony_ci
8cb93a386Sopenharmony_ci#include "include/core/SkRefCnt.h"
9cb93a386Sopenharmony_ci#include "include/core/SkTypes.h"
10cb93a386Sopenharmony_ci#include "include/private/SkWeakRefCnt.h"
11cb93a386Sopenharmony_ci#include "tests/Test.h"
12cb93a386Sopenharmony_ci
13cb93a386Sopenharmony_ci#include <thread>
14cb93a386Sopenharmony_ci
15cb93a386Sopenharmony_cistatic void bounce_ref(void* data) {
16cb93a386Sopenharmony_ci    SkRefCnt* ref = static_cast<SkRefCnt*>(data);
17cb93a386Sopenharmony_ci    for (int i = 0; i < 100000; ++i) {
18cb93a386Sopenharmony_ci        ref->ref();
19cb93a386Sopenharmony_ci        ref->unref();
20cb93a386Sopenharmony_ci    }
21cb93a386Sopenharmony_ci}
22cb93a386Sopenharmony_ci
23cb93a386Sopenharmony_cistatic void test_refCnt(skiatest::Reporter* reporter) {
24cb93a386Sopenharmony_ci    SkRefCnt* ref = new SkRefCnt();
25cb93a386Sopenharmony_ci
26cb93a386Sopenharmony_ci    std::thread thing1(bounce_ref, ref);
27cb93a386Sopenharmony_ci    std::thread thing2(bounce_ref, ref);
28cb93a386Sopenharmony_ci
29cb93a386Sopenharmony_ci    thing1.join();
30cb93a386Sopenharmony_ci    thing2.join();
31cb93a386Sopenharmony_ci
32cb93a386Sopenharmony_ci    REPORTER_ASSERT(reporter, ref->unique());
33cb93a386Sopenharmony_ci    ref->unref();
34cb93a386Sopenharmony_ci}
35cb93a386Sopenharmony_ci
36cb93a386Sopenharmony_cistatic void bounce_weak_ref(void* data) {
37cb93a386Sopenharmony_ci    SkWeakRefCnt* ref = static_cast<SkWeakRefCnt*>(data);
38cb93a386Sopenharmony_ci    for (int i = 0; i < 100000; ++i) {
39cb93a386Sopenharmony_ci        if (ref->try_ref()) {
40cb93a386Sopenharmony_ci            ref->unref();
41cb93a386Sopenharmony_ci        }
42cb93a386Sopenharmony_ci    }
43cb93a386Sopenharmony_ci}
44cb93a386Sopenharmony_ci
45cb93a386Sopenharmony_cistatic void bounce_weak_weak_ref(void* data) {
46cb93a386Sopenharmony_ci    SkWeakRefCnt* ref = static_cast<SkWeakRefCnt*>(data);
47cb93a386Sopenharmony_ci    for (int i = 0; i < 100000; ++i) {
48cb93a386Sopenharmony_ci        ref->weak_ref();
49cb93a386Sopenharmony_ci        ref->weak_unref();
50cb93a386Sopenharmony_ci    }
51cb93a386Sopenharmony_ci}
52cb93a386Sopenharmony_ci
53cb93a386Sopenharmony_cistatic void test_weakRefCnt(skiatest::Reporter* reporter) {
54cb93a386Sopenharmony_ci    SkWeakRefCnt* ref = new SkWeakRefCnt();
55cb93a386Sopenharmony_ci
56cb93a386Sopenharmony_ci    std::thread thing1(bounce_ref, ref);
57cb93a386Sopenharmony_ci    std::thread thing2(bounce_ref, ref);
58cb93a386Sopenharmony_ci    std::thread thing3(bounce_weak_ref, ref);
59cb93a386Sopenharmony_ci    std::thread thing4(bounce_weak_weak_ref, ref);
60cb93a386Sopenharmony_ci
61cb93a386Sopenharmony_ci    thing1.join();
62cb93a386Sopenharmony_ci    thing2.join();
63cb93a386Sopenharmony_ci    thing3.join();
64cb93a386Sopenharmony_ci    thing4.join();
65cb93a386Sopenharmony_ci
66cb93a386Sopenharmony_ci    REPORTER_ASSERT(reporter, ref->unique());
67cb93a386Sopenharmony_ci    SkDEBUGCODE(REPORTER_ASSERT(reporter, ref->getWeakCnt() == 1));
68cb93a386Sopenharmony_ci    ref->unref();
69cb93a386Sopenharmony_ci}
70cb93a386Sopenharmony_ci
71cb93a386Sopenharmony_ciDEF_TEST(RefCnt, reporter) {
72cb93a386Sopenharmony_ci    test_refCnt(reporter);
73cb93a386Sopenharmony_ci    test_weakRefCnt(reporter);
74cb93a386Sopenharmony_ci}
75cb93a386Sopenharmony_ci
76cb93a386Sopenharmony_ci///////////////////////////////////////////////////////////////////////////////////////////////////
77cb93a386Sopenharmony_ci
78cb93a386Sopenharmony_cistatic int gRefCounter;
79cb93a386Sopenharmony_cistatic int gUnrefCounter;
80cb93a386Sopenharmony_cistatic int gNewCounter;
81cb93a386Sopenharmony_cistatic int gDeleteCounter;
82cb93a386Sopenharmony_ci
83cb93a386Sopenharmony_ci#define check(reporter, ref, unref, make, kill)        \
84cb93a386Sopenharmony_ci    REPORTER_ASSERT(reporter, gRefCounter == ref);     \
85cb93a386Sopenharmony_ci    REPORTER_ASSERT(reporter, gUnrefCounter == unref); \
86cb93a386Sopenharmony_ci    REPORTER_ASSERT(reporter, gNewCounter == make);    \
87cb93a386Sopenharmony_ci    REPORTER_ASSERT(reporter, gDeleteCounter == kill)
88cb93a386Sopenharmony_ci
89cb93a386Sopenharmony_ciclass Effect {
90cb93a386Sopenharmony_cipublic:
91cb93a386Sopenharmony_ci    Effect() : fRefCnt(1) {
92cb93a386Sopenharmony_ci        gNewCounter += 1;
93cb93a386Sopenharmony_ci    }
94cb93a386Sopenharmony_ci    virtual ~Effect() {}
95cb93a386Sopenharmony_ci
96cb93a386Sopenharmony_ci    int fRefCnt;
97cb93a386Sopenharmony_ci
98cb93a386Sopenharmony_ci    void ref() {
99cb93a386Sopenharmony_ci        gRefCounter += 1;
100cb93a386Sopenharmony_ci        fRefCnt += 1;
101cb93a386Sopenharmony_ci    }
102cb93a386Sopenharmony_ci    void unref() {
103cb93a386Sopenharmony_ci        gUnrefCounter += 1;
104cb93a386Sopenharmony_ci
105cb93a386Sopenharmony_ci        SkASSERT(fRefCnt > 0);
106cb93a386Sopenharmony_ci        if (0 == --fRefCnt) {
107cb93a386Sopenharmony_ci            gDeleteCounter += 1;
108cb93a386Sopenharmony_ci            delete this;
109cb93a386Sopenharmony_ci        }
110cb93a386Sopenharmony_ci    }
111cb93a386Sopenharmony_ci
112cb93a386Sopenharmony_ci    int* method() const { return new int; }
113cb93a386Sopenharmony_ci};
114cb93a386Sopenharmony_ci
115cb93a386Sopenharmony_cistatic sk_sp<Effect> Create() {
116cb93a386Sopenharmony_ci    return sk_make_sp<Effect>();
117cb93a386Sopenharmony_ci}
118cb93a386Sopenharmony_ci
119cb93a386Sopenharmony_ciclass Paint {
120cb93a386Sopenharmony_cipublic:
121cb93a386Sopenharmony_ci    sk_sp<Effect> fEffect;
122cb93a386Sopenharmony_ci
123cb93a386Sopenharmony_ci    const sk_sp<Effect>& get() const { return fEffect; }
124cb93a386Sopenharmony_ci
125cb93a386Sopenharmony_ci    void set(sk_sp<Effect> value) {
126cb93a386Sopenharmony_ci        fEffect = std::move(value);
127cb93a386Sopenharmony_ci    }
128cb93a386Sopenharmony_ci};
129cb93a386Sopenharmony_ci
130cb93a386Sopenharmony_cistruct EffectImpl : public Effect {
131cb93a386Sopenharmony_ci    ~EffectImpl() override {}
132cb93a386Sopenharmony_ci
133cb93a386Sopenharmony_ci    static sk_sp<EffectImpl> Create() {
134cb93a386Sopenharmony_ci        return sk_sp<EffectImpl>(new EffectImpl);
135cb93a386Sopenharmony_ci    }
136cb93a386Sopenharmony_ci    int fValue;
137cb93a386Sopenharmony_ci};
138cb93a386Sopenharmony_cistatic sk_sp<Effect> make_effect() {
139cb93a386Sopenharmony_ci    auto foo = EffectImpl::Create();
140cb93a386Sopenharmony_ci    foo->fValue = 42;
141cb93a386Sopenharmony_ci    return std::move(foo);
142cb93a386Sopenharmony_ci}
143cb93a386Sopenharmony_ci
144cb93a386Sopenharmony_cistatic void reset_counters() {
145cb93a386Sopenharmony_ci    gRefCounter = 0;
146cb93a386Sopenharmony_ci    gUnrefCounter = 0;
147cb93a386Sopenharmony_ci    gNewCounter = 0;
148cb93a386Sopenharmony_ci    gDeleteCounter = 0;
149cb93a386Sopenharmony_ci}
150cb93a386Sopenharmony_ciDEF_TEST(sk_sp, reporter) {
151cb93a386Sopenharmony_ci    reset_counters();
152cb93a386Sopenharmony_ci
153cb93a386Sopenharmony_ci    Paint paint;
154cb93a386Sopenharmony_ci    REPORTER_ASSERT(reporter, paint.fEffect.get() == nullptr);
155cb93a386Sopenharmony_ci    REPORTER_ASSERT(reporter, !paint.get());
156cb93a386Sopenharmony_ci    check(reporter, 0, 0, 0, 0);
157cb93a386Sopenharmony_ci
158cb93a386Sopenharmony_ci    paint.set(Create());
159cb93a386Sopenharmony_ci    check(reporter, 0, 0, 1, 0);
160cb93a386Sopenharmony_ci    REPORTER_ASSERT(reporter, paint.fEffect.get()->fRefCnt == 1);
161cb93a386Sopenharmony_ci
162cb93a386Sopenharmony_ci    if (paint.get()) {
163cb93a386Sopenharmony_ci        REPORTER_ASSERT(reporter, true);
164cb93a386Sopenharmony_ci    } else {
165cb93a386Sopenharmony_ci        REPORTER_ASSERT(reporter, false);
166cb93a386Sopenharmony_ci    }
167cb93a386Sopenharmony_ci    if (!paint.get()) {
168cb93a386Sopenharmony_ci        REPORTER_ASSERT(reporter, false);
169cb93a386Sopenharmony_ci    } else {
170cb93a386Sopenharmony_ci        REPORTER_ASSERT(reporter, true);
171cb93a386Sopenharmony_ci    }
172cb93a386Sopenharmony_ci
173cb93a386Sopenharmony_ci    paint.set(nullptr);
174cb93a386Sopenharmony_ci    check(reporter, 0, 1, 1, 1);
175cb93a386Sopenharmony_ci
176cb93a386Sopenharmony_ci    if (paint.get()) {
177cb93a386Sopenharmony_ci        REPORTER_ASSERT(reporter, false);
178cb93a386Sopenharmony_ci    } else {
179cb93a386Sopenharmony_ci        REPORTER_ASSERT(reporter, true);
180cb93a386Sopenharmony_ci    }
181cb93a386Sopenharmony_ci    if (!paint.get()) {
182cb93a386Sopenharmony_ci        REPORTER_ASSERT(reporter, true);
183cb93a386Sopenharmony_ci    } else {
184cb93a386Sopenharmony_ci        REPORTER_ASSERT(reporter, false);
185cb93a386Sopenharmony_ci    }
186cb93a386Sopenharmony_ci
187cb93a386Sopenharmony_ci    auto e = Create();
188cb93a386Sopenharmony_ci    REPORTER_ASSERT(reporter, sizeof(e) == sizeof(void*));
189cb93a386Sopenharmony_ci
190cb93a386Sopenharmony_ci    check(reporter, 0, 1, 2, 1);
191cb93a386Sopenharmony_ci    paint.set(e);
192cb93a386Sopenharmony_ci    check(reporter, 1, 1, 2, 1);
193cb93a386Sopenharmony_ci    REPORTER_ASSERT(reporter, paint.fEffect.get()->fRefCnt == 2);
194cb93a386Sopenharmony_ci
195cb93a386Sopenharmony_ci    Paint paint2;
196cb93a386Sopenharmony_ci    paint2.set(paint.get());
197cb93a386Sopenharmony_ci    check(reporter, 2, 1, 2, 1);
198cb93a386Sopenharmony_ci    REPORTER_ASSERT(reporter, paint.fEffect.get()->fRefCnt == 3);
199cb93a386Sopenharmony_ci
200cb93a386Sopenharmony_ci    // Test sk_sp::operator->
201cb93a386Sopenharmony_ci    delete paint.get()->method();
202cb93a386Sopenharmony_ci    check(reporter, 2, 1, 2, 1);
203cb93a386Sopenharmony_ci
204cb93a386Sopenharmony_ci    // Test sk_sp::operator*
205cb93a386Sopenharmony_ci    delete (*paint.get()).method();
206cb93a386Sopenharmony_ci    check(reporter, 2, 1, 2, 1);
207cb93a386Sopenharmony_ci
208cb93a386Sopenharmony_ci    paint.set(nullptr);
209cb93a386Sopenharmony_ci    e = nullptr;
210cb93a386Sopenharmony_ci    paint2.set(nullptr);
211cb93a386Sopenharmony_ci    check(reporter, 2, 4, 2, 2);
212cb93a386Sopenharmony_ci
213cb93a386Sopenharmony_ci    reset_counters();
214cb93a386Sopenharmony_ci    {
215cb93a386Sopenharmony_ci        // Test convertible sk_sp assignment.
216cb93a386Sopenharmony_ci        check(reporter, 0, 0, 0, 0);
217cb93a386Sopenharmony_ci        sk_sp<Effect> foo(nullptr);
218cb93a386Sopenharmony_ci        REPORTER_ASSERT(reporter, !foo);
219cb93a386Sopenharmony_ci        foo = make_effect();
220cb93a386Sopenharmony_ci        REPORTER_ASSERT(reporter, foo);
221cb93a386Sopenharmony_ci        check(reporter, 0, 0, 1, 0);
222cb93a386Sopenharmony_ci    }
223cb93a386Sopenharmony_ci    check(reporter, 0, 1, 1, 1);
224cb93a386Sopenharmony_ci
225cb93a386Sopenharmony_ci    // Test passing convertible rvalue into funtion.
226cb93a386Sopenharmony_ci    reset_counters();
227cb93a386Sopenharmony_ci    paint.set(EffectImpl::Create());
228cb93a386Sopenharmony_ci    check(reporter, 0, 0, 1, 0);
229cb93a386Sopenharmony_ci    paint.set(nullptr);
230cb93a386Sopenharmony_ci    check(reporter, 0, 1, 1, 1);
231cb93a386Sopenharmony_ci
232cb93a386Sopenharmony_ci    reset_counters();
233cb93a386Sopenharmony_ci    auto baz = EffectImpl::Create();
234cb93a386Sopenharmony_ci    check(reporter, 0, 0, 1, 0);
235cb93a386Sopenharmony_ci    paint.set(std::move(baz));
236cb93a386Sopenharmony_ci    check(reporter, 0, 0, 1, 0);
237cb93a386Sopenharmony_ci    REPORTER_ASSERT(reporter, !baz);  // NOLINT(bugprone-use-after-move)
238cb93a386Sopenharmony_ci    paint.set(nullptr);
239cb93a386Sopenharmony_ci    check(reporter, 0, 1, 1, 1);
240cb93a386Sopenharmony_ci
241cb93a386Sopenharmony_ci    reset_counters();
242cb93a386Sopenharmony_ci    {
243cb93a386Sopenharmony_ci        // test comparison operator with convertible type.
244cb93a386Sopenharmony_ci        sk_sp<EffectImpl> bar1 = EffectImpl::Create();
245cb93a386Sopenharmony_ci        sk_sp<Effect> bar2(bar1);  // convertible copy constructor
246cb93a386Sopenharmony_ci        check(reporter, 1, 0, 1, 0);
247cb93a386Sopenharmony_ci        REPORTER_ASSERT(reporter, bar1);
248cb93a386Sopenharmony_ci        REPORTER_ASSERT(reporter, bar2);
249cb93a386Sopenharmony_ci        REPORTER_ASSERT(reporter, bar1 == bar2);
250cb93a386Sopenharmony_ci        REPORTER_ASSERT(reporter, bar2 == bar1);
251cb93a386Sopenharmony_ci        REPORTER_ASSERT(reporter, !(bar1 != bar2));
252cb93a386Sopenharmony_ci        REPORTER_ASSERT(reporter, !(bar2 != bar1));
253cb93a386Sopenharmony_ci        sk_sp<Effect> bar3(nullptr);
254cb93a386Sopenharmony_ci        bar3 = bar1;  // convertible copy assignment
255cb93a386Sopenharmony_ci        check(reporter, 2, 0, 1, 0);
256cb93a386Sopenharmony_ci
257cb93a386Sopenharmony_ci    }
258cb93a386Sopenharmony_ci    check(reporter, 2, 3, 1, 1);
259cb93a386Sopenharmony_ci
260cb93a386Sopenharmony_ci    // test passing convertible copy into funtion.
261cb93a386Sopenharmony_ci    reset_counters();
262cb93a386Sopenharmony_ci    baz = EffectImpl::Create();
263cb93a386Sopenharmony_ci    check(reporter, 0, 0, 1, 0);
264cb93a386Sopenharmony_ci    paint.set(baz);
265cb93a386Sopenharmony_ci    check(reporter, 1, 0, 1, 0);
266cb93a386Sopenharmony_ci    baz = nullptr;
267cb93a386Sopenharmony_ci    check(reporter, 1, 1, 1, 0);
268cb93a386Sopenharmony_ci    paint.set(nullptr);
269cb93a386Sopenharmony_ci    check(reporter, 1, 2, 1, 1);
270cb93a386Sopenharmony_ci
271cb93a386Sopenharmony_ci    {
272cb93a386Sopenharmony_ci        sk_sp<SkRefCnt> empty;
273cb93a386Sopenharmony_ci        sk_sp<SkRefCnt> notEmpty = sk_make_sp<SkRefCnt>();
274cb93a386Sopenharmony_ci        REPORTER_ASSERT(reporter, empty == sk_sp<SkRefCnt>());
275cb93a386Sopenharmony_ci
276cb93a386Sopenharmony_ci        REPORTER_ASSERT(reporter, notEmpty != empty);
277cb93a386Sopenharmony_ci        REPORTER_ASSERT(reporter, empty != notEmpty);
278cb93a386Sopenharmony_ci
279cb93a386Sopenharmony_ci        REPORTER_ASSERT(reporter, nullptr == empty);
280cb93a386Sopenharmony_ci        REPORTER_ASSERT(reporter, empty == nullptr);
281cb93a386Sopenharmony_ci        REPORTER_ASSERT(reporter, empty == empty);
282cb93a386Sopenharmony_ci    }
283cb93a386Sopenharmony_ci
284cb93a386Sopenharmony_ci    {
285cb93a386Sopenharmony_ci        sk_sp<SkRefCnt> a = sk_make_sp<SkRefCnt>();
286cb93a386Sopenharmony_ci        sk_sp<SkRefCnt> b = sk_make_sp<SkRefCnt>();
287cb93a386Sopenharmony_ci        REPORTER_ASSERT(reporter, a != b);
288cb93a386Sopenharmony_ci        REPORTER_ASSERT(reporter, a == a);
289cb93a386Sopenharmony_ci    }
290cb93a386Sopenharmony_ci
291cb93a386Sopenharmony_ci    // http://wg21.cmeerw.net/lwg/issue998
292cb93a386Sopenharmony_ci    {
293cb93a386Sopenharmony_ci        class foo : public SkRefCnt {
294cb93a386Sopenharmony_ci        public:
295cb93a386Sopenharmony_ci            foo() : bar(this) {}
296cb93a386Sopenharmony_ci            void reset() { bar.reset(); }
297cb93a386Sopenharmony_ci        private:
298cb93a386Sopenharmony_ci            sk_sp<foo> bar;
299cb93a386Sopenharmony_ci        };
300cb93a386Sopenharmony_ci        // The following should properly delete the object and not cause undefined behavior.
301cb93a386Sopenharmony_ci        // This is an ugly example, but the same issue can arise in more subtle ways.
302cb93a386Sopenharmony_ci        (new foo)->reset();
303cb93a386Sopenharmony_ci    }
304cb93a386Sopenharmony_ci
305cb93a386Sopenharmony_ci    // https://crrev.com/0d4ef2583a6f19c3e61be04d36eb1a60b133832c
306cb93a386Sopenharmony_ci    {
307cb93a386Sopenharmony_ci        struct StructB;
308cb93a386Sopenharmony_ci        struct StructA : public SkRefCnt {
309cb93a386Sopenharmony_ci            sk_sp<StructB> b;
310cb93a386Sopenharmony_ci        };
311cb93a386Sopenharmony_ci
312cb93a386Sopenharmony_ci        struct StructB : public SkRefCnt {
313cb93a386Sopenharmony_ci            sk_sp<StructA> a;
314cb93a386Sopenharmony_ci            ~StructB() override {} // Some clang versions don't emit this implicitly.
315cb93a386Sopenharmony_ci        };
316cb93a386Sopenharmony_ci
317cb93a386Sopenharmony_ci        // Create a reference cycle.
318cb93a386Sopenharmony_ci        StructA* a = new StructA;
319cb93a386Sopenharmony_ci        a->b.reset(new StructB);
320cb93a386Sopenharmony_ci        a->b->a.reset(a);
321cb93a386Sopenharmony_ci
322cb93a386Sopenharmony_ci        // Break the cycle by calling reset(). This will cause |a| (and hence, |a.b|)
323cb93a386Sopenharmony_ci        // to be deleted before the call to reset() returns. This tests that the
324cb93a386Sopenharmony_ci        // implementation of sk_sp::reset() doesn't access |this| after it
325cb93a386Sopenharmony_ci        // deletes the underlying pointer. This behaviour is consistent with the
326cb93a386Sopenharmony_ci        // definition of unique_ptr::reset in C++11.
327cb93a386Sopenharmony_ci        a->b.reset();
328cb93a386Sopenharmony_ci    }
329cb93a386Sopenharmony_ci}
330cb93a386Sopenharmony_ci
331cb93a386Sopenharmony_cinamespace {
332cb93a386Sopenharmony_cistruct FooAbstract : public SkRefCnt {
333cb93a386Sopenharmony_ci    virtual void f() = 0;
334cb93a386Sopenharmony_ci};
335cb93a386Sopenharmony_cistruct FooConcrete : public FooAbstract {
336cb93a386Sopenharmony_ci    void f() override {}
337cb93a386Sopenharmony_ci};
338cb93a386Sopenharmony_ci}  // namespace
339cb93a386Sopenharmony_cistatic sk_sp<FooAbstract> make_foo() {
340cb93a386Sopenharmony_ci    // can not cast FooConcrete to FooAbstract.
341cb93a386Sopenharmony_ci    // can cast FooConcrete* to FooAbstract*.
342cb93a386Sopenharmony_ci    return sk_make_sp<FooConcrete>();
343cb93a386Sopenharmony_ci}
344cb93a386Sopenharmony_ciDEF_TEST(sk_make_sp, r) {
345cb93a386Sopenharmony_ci    auto x = make_foo();
346cb93a386Sopenharmony_ci}
347cb93a386Sopenharmony_ci
348cb93a386Sopenharmony_ci// Test that reset() "adopts" ownership from the caller, even if we are given the same ptr twice
349cb93a386Sopenharmony_ci//
350cb93a386Sopenharmony_ciDEF_TEST(sk_sp_reset, r) {
351cb93a386Sopenharmony_ci    SkRefCnt* rc = new SkRefCnt;
352cb93a386Sopenharmony_ci    REPORTER_ASSERT(r, rc->unique());
353cb93a386Sopenharmony_ci
354cb93a386Sopenharmony_ci    sk_sp<SkRefCnt> sp;
355cb93a386Sopenharmony_ci    sp.reset(rc);
356cb93a386Sopenharmony_ci    // We have transfered our ownership over to sp
357cb93a386Sopenharmony_ci    REPORTER_ASSERT(r, rc->unique());
358cb93a386Sopenharmony_ci
359cb93a386Sopenharmony_ci    rc->ref();  // now "rc" is also an owner
360cb93a386Sopenharmony_ci    REPORTER_ASSERT(r, !rc->unique());
361cb93a386Sopenharmony_ci
362cb93a386Sopenharmony_ci    sp.reset(rc);   // this should transfer our ownership over to sp
363cb93a386Sopenharmony_ci    REPORTER_ASSERT(r, rc->unique());
364cb93a386Sopenharmony_ci}
365cb93a386Sopenharmony_ci
366cb93a386Sopenharmony_ciDEF_TEST(sk_sp_ref, r) {
367cb93a386Sopenharmony_ci    SkRefCnt* rc = new SkRefCnt;
368cb93a386Sopenharmony_ci    REPORTER_ASSERT(r, rc->unique());
369cb93a386Sopenharmony_ci
370cb93a386Sopenharmony_ci    {
371cb93a386Sopenharmony_ci        sk_sp<SkRefCnt> sp = sk_ref_sp(rc);
372cb93a386Sopenharmony_ci        REPORTER_ASSERT(r, !rc->unique());
373cb93a386Sopenharmony_ci    }
374cb93a386Sopenharmony_ci
375cb93a386Sopenharmony_ci    REPORTER_ASSERT(r, rc->unique());
376cb93a386Sopenharmony_ci    rc->unref();
377cb93a386Sopenharmony_ci}
378