blob: e9d582d809e28766b33d2a4e1bff2aeddb8d6d97 [file] [log] [blame]
bungeman@google.com55487522012-05-14 14:09:24 +00001/*
2 * Copyright 2012 Google Inc.
3 *
4 * Use of this source code is governed by a BSD-style license that can be
5 * found in the LICENSE file.
6 */
7
bungeman@google.com55487522012-05-14 14:09:24 +00008#include "SkRefCnt.h"
tfarina@chromium.orge4fafb12013-12-12 21:11:12 +00009#include "SkThreadUtils.h"
10#include "SkTypes.h"
11#include "SkWeakRefCnt.h"
tfarina@chromium.org8f6884a2014-01-24 20:56:26 +000012#include "Test.h"
bungeman@google.com55487522012-05-14 14:09:24 +000013
14static void bounce_ref(void* data) {
15 SkRefCnt* ref = static_cast<SkRefCnt*>(data);
16 for (int i = 0; i < 100000; ++i) {
17 ref->ref();
18 ref->unref();
19 }
20}
21
22static void test_refCnt(skiatest::Reporter* reporter) {
23 SkRefCnt* ref = new SkRefCnt();
24
25 SkThread thing1(bounce_ref, ref);
26 SkThread thing2(bounce_ref, ref);
27
bungeman@google.com55487522012-05-14 14:09:24 +000028 SkASSERT(thing1.start());
29 SkASSERT(thing2.start());
30
31 thing1.join();
32 thing2.join();
33
mtkleinbbb61d72014-11-24 13:09:39 -080034 REPORTER_ASSERT(reporter, ref->unique());
bungeman@google.com55487522012-05-14 14:09:24 +000035 ref->unref();
36}
37
bungeman@google.coma02bc152012-05-16 18:21:56 +000038static void bounce_weak_ref(void* data) {
39 SkWeakRefCnt* ref = static_cast<SkWeakRefCnt*>(data);
40 for (int i = 0; i < 100000; ++i) {
41 if (ref->try_ref()) {
42 ref->unref();
43 }
44 }
45}
46
47static void bounce_weak_weak_ref(void* data) {
48 SkWeakRefCnt* ref = static_cast<SkWeakRefCnt*>(data);
49 for (int i = 0; i < 100000; ++i) {
50 ref->weak_ref();
51 ref->weak_unref();
52 }
53}
54
55static void test_weakRefCnt(skiatest::Reporter* reporter) {
56 SkWeakRefCnt* ref = new SkWeakRefCnt();
57
58 SkThread thing1(bounce_ref, ref);
59 SkThread thing2(bounce_ref, ref);
60 SkThread thing3(bounce_weak_ref, ref);
61 SkThread thing4(bounce_weak_weak_ref, ref);
62
bungeman@google.coma02bc152012-05-16 18:21:56 +000063 SkASSERT(thing1.start());
64 SkASSERT(thing2.start());
65 SkASSERT(thing3.start());
66 SkASSERT(thing4.start());
67
68 thing1.join();
69 thing2.join();
70 thing3.join();
71 thing4.join();
72
mtkleinbbb61d72014-11-24 13:09:39 -080073 REPORTER_ASSERT(reporter, ref->unique());
Mike Klein874a62a2014-07-09 09:04:07 -040074 REPORTER_ASSERT(reporter, ref->getWeakCnt() == 1);
bungeman@google.coma02bc152012-05-16 18:21:56 +000075 ref->unref();
76}
77
tfarina@chromium.orge4fafb12013-12-12 21:11:12 +000078DEF_TEST(RefCnt, reporter) {
bungeman@google.coma02bc152012-05-16 18:21:56 +000079 test_refCnt(reporter);
80 test_weakRefCnt(reporter);
81}
reedbb7b0432016-03-01 07:28:51 -080082
83///////////////////////////////////////////////////////////////////////////////////////////////////
84
85static int gRefCounter;
86static int gUnrefCounter;
87static int gNewCounter;
88static int gDeleteCounter;
89
90#define check(reporter, ref, unref, make, kill) \
91 REPORTER_ASSERT(reporter, gRefCounter == ref); \
92 REPORTER_ASSERT(reporter, gUnrefCounter == unref); \
93 REPORTER_ASSERT(reporter, gNewCounter == make); \
94 REPORTER_ASSERT(reporter, gDeleteCounter == kill);
95
96
97class Effect {
98public:
99 Effect() : fRefCnt(1) {
100 gNewCounter += 1;
101 }
102
103 int fRefCnt;
104
105 void ref() {
106 gRefCounter += 1;
107 fRefCnt += 1;
108 }
109 void unref() {
110 gUnrefCounter += 1;
111
112 SkASSERT(fRefCnt > 0);
113 if (0 == --fRefCnt) {
114 gDeleteCounter += 1;
115 delete this;
116 }
117 }
118
119 int* method() const { return new int; }
120};
121
122static sk_sp<Effect> Create() {
halcanary217c0b32016-03-02 08:06:20 -0800123 return sk_make_sp<Effect>();
reedbb7b0432016-03-01 07:28:51 -0800124}
125
126class Paint {
127public:
128 sk_sp<Effect> fEffect;
129
130 const sk_sp<Effect>& get() const { return fEffect; }
131
132 void set(sk_sp<Effect> value) {
133 fEffect = std::move(value);
134 }
135};
136
137DEF_TEST(sk_sp, reporter) {
138 gRefCounter = 0;
139 gUnrefCounter = 0;
140 gNewCounter = 0;
141 gDeleteCounter = 0;
142
143 Paint paint;
144 REPORTER_ASSERT(reporter, paint.fEffect.get() == nullptr);
145 REPORTER_ASSERT(reporter, !paint.get());
146 check(reporter, 0, 0, 0, 0);
147
148 paint.set(Create());
149 check(reporter, 0, 0, 1, 0);
150 REPORTER_ASSERT(reporter, paint.fEffect.get()->fRefCnt == 1);
151
152 paint.set(nullptr);
153 check(reporter, 0, 1, 1, 1);
154
155 auto e = Create();
156 REPORTER_ASSERT(reporter, sizeof(e) == sizeof(void*));
157
158 check(reporter, 0, 1, 2, 1);
159 paint.set(e);
160 check(reporter, 1, 1, 2, 1);
161 REPORTER_ASSERT(reporter, paint.fEffect.get()->fRefCnt == 2);
162
163 Paint paint2;
164 paint2.set(paint.get());
165 check(reporter, 2, 1, 2, 1);
166 REPORTER_ASSERT(reporter, paint.fEffect.get()->fRefCnt == 3);
167
168 delete paint.get()->method();
169 check(reporter, 2, 1, 2, 1);
170}
171