1 /*
2 * Copyright (c) 2021 Huawei Device Co., Ltd.
3 * Licensed under the Apache License, Version 2.0 (the "License");
4 * you may not use this file except in compliance with the License.
5 * You may obtain a copy of the License at
6 *
7 * http://www.apache.org/licenses/LICENSE-2.0
8 *
9 * Unless required by applicable law or agreed to in writing, software
10 * distributed under the License is distributed on an "AS IS" BASIS,
11 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12 * See the License for the specific language governing permissions and
13 * limitations under the License.
14 */
15
16 #include "refbase.h"
17 #ifdef DEBUG_REFBASE
18 #include "utils_log.h"
19 #include <unistd.h>
20 #endif
21
22 namespace OHOS {
23
WeakRefCounter(RefCounter * counter,void * cookie)24 WeakRefCounter::WeakRefCounter(RefCounter *counter, void *cookie)
25 : atomicWeak_(0), refCounter_(counter), cookie_(cookie)
26 {
27 if (refCounter_ != nullptr) {
28 refCounter_->IncRefCount();
29 }
30 }
31
~WeakRefCounter()32 WeakRefCounter::~WeakRefCounter()
33 {
34 if (refCounter_ != nullptr) {
35 refCounter_->DecRefCount();
36 }
37 }
38
GetRefPtr()39 void* WeakRefCounter::GetRefPtr()
40 {
41 if ((cookie_ != nullptr) && (!refCounter_->IsRefPtrValid())) {
42 cookie_ = nullptr;
43 }
44 return cookie_;
45 }
46
IncWeakRefCount(const void * objectId)47 void WeakRefCounter::IncWeakRefCount(const void *objectId)
48 {
49 if (atomicWeak_.fetch_add(1, std::memory_order_relaxed) == 0) {
50 refCounter_->IncWeakRefCount(objectId);
51 }
52 }
53
DecWeakRefCount(const void * objectId)54 void WeakRefCounter::DecWeakRefCount(const void *objectId)
55 {
56 if (atomicWeak_.fetch_sub(1, std::memory_order_release) == 1) {
57 refCounter_->DecWeakRefCount(objectId);
58 delete this;
59 }
60 }
61
AttemptIncStrongRef(const void * objectId)62 bool WeakRefCounter::AttemptIncStrongRef(const void *objectId)
63 {
64 int unuse = 0;
65 return refCounter_->AttemptIncStrongRef(objectId, unuse);
66 }
67
68 #ifdef DEBUG_REFBASE
RefTracker(RefTracker * exTracker,const void * id,int strong,int weak,int ref,int pid,int tid)69 RefTracker::RefTracker(RefTracker* exTracker, const void* id, int strong, int weak, int ref, int pid, int tid)
70 : ptrID (id), strongRefCNT (strong), weakRefCNT (weak), refCNT (ref), PID (pid), TID (tid), exTrace (exTracker)
71 {
72 }
73
GetTrace(RefTracker * exTracker,const void * id,int strong,int weak,int ref,int pid,int tid)74 void RefTracker::GetTrace(RefTracker* exTracker, const void* id, int strong, int weak, int ref, int pid, int tid)
75 {
76 ptrID = id;
77 strongRefCNT = strong;
78 weakRefCNT = weak;
79 refCNT = ref;
80 PID = pid;
81 TID = tid;
82 exTrace = exTracker;
83 }
84
GetStrongTrace(RefTracker * exTracker,const void * id,int strong,int pid,int tid)85 void RefTracker::GetStrongTrace(RefTracker* exTracker, const void* id, int strong, int pid, int tid)
86 {
87 ptrID = id;
88 strongRefCNT = strong;
89 weakRefCNT = -(INITIAL_PRIMARY_VALUE);
90 refCNT = -(INITIAL_PRIMARY_VALUE);
91 PID = pid;
92 TID = tid;
93 exTrace = exTracker;
94 }
95
GetWeakTrace(RefTracker * exTracker,const void * id,int weak,int pid,int tid)96 void RefTracker::GetWeakTrace(RefTracker* exTracker, const void* id, int weak, int pid, int tid)
97 {
98 ptrID = id;
99 strongRefCNT = -(INITIAL_PRIMARY_VALUE);
100 weakRefCNT = weak;
101 refCNT = -(INITIAL_PRIMARY_VALUE);
102 PID = pid;
103 TID = tid;
104 exTrace = exTracker;
105 }
106
PrintTrace(const void * root)107 void RefTracker::PrintTrace(const void* root)
108 {
109 UTILS_LOGI("ptrID(%{public}lx) call %{public}lx. strongcnt: %{public}d weakcnt: %{public}d " \
110 "refcnt: %{public}d PID: %{public}d TID: %{public}d",
111 reinterpret_cast<size_t>(ptrID), reinterpret_cast<size_t>(root), strongRefCNT, weakRefCNT, refCNT, PID, TID);
112 }
113
PrintStrongTrace(const void * root)114 void RefTracker::PrintStrongTrace(const void* root)
115 {
116 UTILS_LOGI("ptrID(%{public}lx) call %{public}lx. strongcnt: %{public}d PID: %{public}d TID: %{public}d",
117 reinterpret_cast<size_t>(ptrID), reinterpret_cast<size_t>(root), strongRefCNT, PID, TID);
118 }
119
PrintWeakTrace(const void * root)120 void RefTracker::PrintWeakTrace(const void* root)
121 {
122 UTILS_LOGI("ptrID(%{public}lx) call %{public}lx. weakcnt: %{public}d PID: %{public}d TID: %{public}d",
123 reinterpret_cast<size_t>(ptrID), reinterpret_cast<size_t>(root), weakRefCNT, PID, TID);
124 }
125
GetexTrace()126 RefTracker* RefTracker::GetexTrace()
127 {
128 return exTrace;
129 }
130
PopTrace(const void * root)131 RefTracker* RefTracker::PopTrace(const void* root)
132 {
133 RefTracker* ref = exTrace;
134 PrintTrace(root);
135 delete this;
136 return ref;
137 }
138
GetNewTrace(const void * object)139 void RefCounter::GetNewTrace(const void* object)
140 {
141 std::lock_guard<std::mutex> lock(trackerMutex);
142 RefTracker* newTracker = new RefTracker(refTracker, object, atomicStrong_.load(std::memory_order_relaxed),
143 atomicWeak_.load(std::memory_order_relaxed), atomicRefCount_.load(std::memory_order_relaxed),
144 getpid(), gettid());
145 refTracker = newTracker;
146 }
147
PrintTracker()148 void RefCounter::PrintTracker()
149 {
150 std::lock_guard<std::mutex> lock(trackerMutex);
151 while (refTracker) {
152 refTracker = refTracker->PopTrace(this);
153 }
154 }
155 #endif
156
RefCounter()157 RefCounter::RefCounter()
158 : atomicStrong_(INITIAL_PRIMARY_VALUE), atomicWeak_(0), atomicRefCount_(0), atomicFlags_(0), atomicAttempt_(0)
159 {
160 }
161
GetRefCount()162 int RefCounter::GetRefCount()
163 {
164 return atomicRefCount_.load(std::memory_order_relaxed);
165 }
166
IncRefCount()167 void RefCounter::IncRefCount()
168 {
169 atomicRefCount_.fetch_add(1, std::memory_order_relaxed);
170 }
171
DecRefCount()172 void RefCounter::DecRefCount()
173 {
174 if (atomicRefCount_.load(std::memory_order_relaxed) > 0) {
175 if (atomicRefCount_.fetch_sub(1, std::memory_order_release) == 1) {
176 delete (this);
177 }
178 }
179 }
180
SetCallback(const RefPtrCallback & callback)181 void RefCounter::SetCallback(const RefPtrCallback& callback)
182 {
183 callback_ = callback;
184 }
185
RemoveCallback()186 void RefCounter::RemoveCallback()
187 {
188 callback_ = nullptr;
189 }
190
IsRefPtrValid()191 bool RefCounter::IsRefPtrValid()
192 {
193 return callback_ != nullptr;
194 }
195
~RefCounter()196 RefCounter::~RefCounter()
197 {
198 #ifdef DEBUG_REFBASE
199 PrintTracker();
200 #endif
201 }
202
IncStrongRefCount(const void * object)203 int RefCounter::IncStrongRefCount(const void* object)
204 {
205 #ifdef DEBUG_REFBASE
206 GetNewTrace(object);
207 #endif
208 int curCount = atomicStrong_.load(std::memory_order_relaxed);
209 if (curCount >= 0) {
210 curCount = atomicStrong_.fetch_add(1, std::memory_order_relaxed);
211 if (curCount == INITIAL_PRIMARY_VALUE) {
212 atomicStrong_.fetch_sub(INITIAL_PRIMARY_VALUE, std::memory_order_release);
213 }
214 }
215
216 return curCount;
217 }
218
DecStrongRefCount(const void * object)219 int RefCounter::DecStrongRefCount(const void* object)
220 {
221 #ifdef DEBUG_REFBASE
222 GetNewTrace(object);
223 #endif
224 int curCount = GetStrongRefCount();
225 if (curCount == INITIAL_PRIMARY_VALUE) {
226 // unexpected case: there had never a strong reference.
227 } else if (curCount > 0) {
228 // we should update the current count here.
229 // it may be changed after last operation.
230 curCount = atomicStrong_.fetch_sub(1, std::memory_order_release);
231 }
232
233 return curCount;
234 }
235
GetStrongRefCount()236 int RefCounter::GetStrongRefCount()
237 {
238 return atomicStrong_.load(std::memory_order_relaxed);
239 }
240
IncWeakRefCount(const void * object)241 int RefCounter::IncWeakRefCount(const void* object)
242 {
243 #ifdef DEBUG_REFBASE
244 GetNewTrace(object);
245 #endif
246 return atomicWeak_.fetch_add(1, std::memory_order_relaxed);
247 }
248
DecWeakRefCount(const void * object)249 int RefCounter::DecWeakRefCount(const void* object)
250 {
251 #ifdef DEBUG_REFBASE
252 GetNewTrace(object);
253 #endif
254 int curCount = GetWeakRefCount();
255 if (curCount > 0) {
256 curCount = atomicWeak_.fetch_sub(1, std::memory_order_release);
257 }
258
259 if (curCount != 1) {
260 return curCount;
261 }
262
263 if (IsLifeTimeExtended() && GetStrongRefCount() == 0) {
264 if (callback_) {
265 callback_();
266 }
267 } else {
268 // only weak ptr case: no strong reference, delete the object
269 if (GetStrongRefCount() == INITIAL_PRIMARY_VALUE) {
270 if (callback_) {
271 callback_();
272 }
273 } else {
274 // free RefCounter
275 DecRefCount();
276 }
277 }
278
279 return curCount;
280 }
281
GetWeakRefCount()282 int RefCounter::GetWeakRefCount()
283 {
284 return atomicWeak_.load(std::memory_order_relaxed);
285 }
286
SetAttemptAcquire()287 void RefCounter::SetAttemptAcquire()
288 {
289 (void)atomicAttempt_.fetch_add(1, std::memory_order_relaxed);
290 }
291
IsAttemptAcquireSet()292 bool RefCounter::IsAttemptAcquireSet()
293 {
294 return static_cast<bool>(atomicAttempt_.load(std::memory_order_relaxed) > 0);
295 }
296
ClearAttemptAcquire()297 void RefCounter::ClearAttemptAcquire()
298 {
299 atomicAttempt_.fetch_sub(1, std::memory_order_relaxed);
300 }
301
ExtendObjectLifetime()302 void RefCounter::ExtendObjectLifetime()
303 {
304 atomicFlags_.fetch_or(FLAG_EXTEND_LIFE_TIME, std::memory_order_relaxed);
305 }
306
IsLifeTimeExtended()307 bool RefCounter::IsLifeTimeExtended()
308 {
309 return static_cast<bool>(atomicFlags_.load(std::memory_order_relaxed) & FLAG_EXTEND_LIFE_TIME);
310 }
311
AttemptIncStrongRef(const void * objectId,int & outCount)312 bool RefCounter::AttemptIncStrongRef(const void *objectId, int &outCount)
313 {
314 int curCount = GetStrongRefCount();
315 IncWeakRefCount(objectId);
316
317 // if the object already had strong references.just promoting it.
318 while ((curCount > 0) && (curCount != INITIAL_PRIMARY_VALUE)) {
319 if (atomicStrong_.compare_exchange_weak(curCount, curCount + 1, std::memory_order_relaxed)) {
320 goto ATTEMPT_SUCCESS;
321 }
322 // someone else changed the counter.re-acquire the counter value.
323 curCount = atomicStrong_.load(std::memory_order_relaxed);
324 }
325
326 if ((curCount == INITIAL_PRIMARY_VALUE) && !IsLifeTimeExtended()) {
327 // this object has a "normal" life-time,
328 while (curCount > 0) {
329 if (atomicStrong_.compare_exchange_weak(curCount, curCount + 1, std::memory_order_relaxed)) {
330 goto ATTEMPT_SUCCESS;
331 }
332 curCount = atomicStrong_.load(std::memory_order_relaxed);
333 }
334 }
335
336 if (IsLifeTimeExtended()) {
337 curCount = atomicStrong_.fetch_add(1, std::memory_order_relaxed);
338 }
339
340 ATTEMPT_SUCCESS:
341 if (curCount >= INITIAL_PRIMARY_VALUE) {
342 outCount = curCount;
343 atomicStrong_.fetch_sub(INITIAL_PRIMARY_VALUE, std::memory_order_release);
344 return true;
345 }
346
347 if (curCount < 0 || (!IsLifeTimeExtended() && curCount == 0)) {
348 // the object destroyed on strong reference count reduce to zero.
349 DecWeakRefCount(objectId);
350 return false;
351 }
352
353 return true;
354 }
355
AttemptIncStrong(const void * objectId)356 bool RefCounter::AttemptIncStrong(const void *objectId)
357 {
358 IncWeakRefCount(objectId);
359 int curCount = GetStrongRefCount();
360 while (curCount > 0) {
361 if (atomicStrong_.compare_exchange_weak(curCount, curCount + 1, std::memory_order_relaxed)) {
362 break;
363 }
364 curCount = atomicStrong_.load(std::memory_order_relaxed);
365 }
366 if (curCount <= 0) {
367 DecWeakRefCount(objectId);
368 }
369 return curCount > 0;
370 }
371
RefBase()372 RefBase::RefBase() : refs_(new RefCounter())
373 {
374 refs_->IncRefCount();
375 refs_->SetCallback(std::bind(&RefBase::RefPtrCallback, this));
376 }
377
RefBase(const RefBase &)378 RefBase::RefBase(const RefBase &)
379 {
380 refs_ = new (std::nothrow) RefCounter();
381 if (refs_ != nullptr) {
382 refs_->IncRefCount();
383 refs_->SetCallback(std::bind(&RefBase::RefPtrCallback, this));
384 }
385 }
386
RefPtrCallback()387 void RefBase::RefPtrCallback()
388 {
389 delete this;
390 }
391
392 /*
393 * The two ends of the assignment are two independent and exclusive,
394 * and the application should not share the reference counter.
395 * RISK: If there is a reference count on the left of the equal sign,
396 * it may cause a reference count exception
397 */
operator =(const RefBase &)398 RefBase &RefBase::operator=(const RefBase &)
399 {
400 if (refs_ != nullptr) {
401 refs_->RemoveCallback();
402 refs_->DecRefCount();
403 }
404
405 refs_ = new (std::nothrow) RefCounter();
406 if (refs_ != nullptr) {
407 refs_->IncRefCount();
408 refs_->SetCallback(std::bind(&RefBase::RefPtrCallback, this));
409 }
410
411 return *this;
412 }
413
RefBase(RefBase && other)414 RefBase::RefBase(RefBase &&other) noexcept
415 {
416 refs_ = other.refs_;
417 other.refs_ = nullptr;
418 }
419
operator =(RefBase && other)420 RefBase &RefBase::operator=(RefBase &&other) noexcept
421 {
422 if (refs_ == other.refs_) {
423 return *this;
424 }
425
426 if (refs_ != nullptr) {
427 refs_->RemoveCallback();
428 refs_->DecRefCount();
429 }
430
431 refs_ = other.refs_;
432 other.refs_ = nullptr;
433 return *this;
434 }
435
~RefBase()436 RefBase::~RefBase()
437 {
438 if (refs_ != nullptr) {
439 refs_->RemoveCallback();
440 if ((refs_->IsLifeTimeExtended() && refs_->GetWeakRefCount() == 0)
441 || refs_->GetStrongRefCount() == INITIAL_PRIMARY_VALUE) {
442 refs_->DecRefCount();
443 }
444 refs_ = nullptr;
445 }
446 }
447
ExtendObjectLifetime()448 void RefBase::ExtendObjectLifetime()
449 {
450 refs_->ExtendObjectLifetime();
451 }
452
IncStrongRef(const void * objectId)453 void RefBase::IncStrongRef(const void *objectId)
454 {
455 if (refs_ == nullptr) {
456 return;
457 }
458
459 const int curCount = refs_->IncStrongRefCount(objectId);
460 IncWeakRef(objectId);
461 if (curCount == INITIAL_PRIMARY_VALUE) {
462 OnFirstStrongRef(objectId);
463 }
464 if (refs_->IsAttemptAcquireSet()) {
465 refs_->ClearAttemptAcquire();
466 refs_->DecStrongRefCount(objectId);
467 refs_->DecWeakRefCount(objectId);
468 }
469 }
470
DecStrongRef(const void * objectId)471 void RefBase::DecStrongRef(const void *objectId)
472 {
473 if (refs_ == nullptr) {
474 return;
475 }
476
477 RefCounter * const refs = refs_;
478 const int curCount = refs->DecStrongRefCount(objectId);
479 if (curCount == 1) {
480 OnLastStrongRef(objectId);
481 if (!refs->IsLifeTimeExtended()) {
482 if (refs->callback_) {
483 refs->callback_();
484 }
485 }
486 }
487
488 refs->DecWeakRefCount(objectId);
489 }
490
GetSptrRefCount()491 int RefBase::GetSptrRefCount()
492 {
493 if (refs_ == nullptr) {
494 return 0;
495 }
496 return refs_->GetStrongRefCount();
497 }
498
CreateWeakRef(void * cookie)499 WeakRefCounter *RefBase::CreateWeakRef(void *cookie)
500 {
501 if (refs_ != nullptr) {
502 return new WeakRefCounter(refs_, cookie);
503 }
504 return nullptr;
505 }
506
IncWeakRef(const void * objectId)507 void RefBase::IncWeakRef(const void *objectId)
508 {
509 if (refs_ != nullptr) {
510 refs_->IncWeakRefCount(objectId);
511 }
512 }
513
DecWeakRef(const void * objectId)514 void RefBase::DecWeakRef(const void *objectId)
515 {
516 if (refs_ != nullptr) {
517 refs_->DecWeakRefCount(objectId);
518 }
519 }
520
GetWptrRefCount()521 int RefBase::GetWptrRefCount()
522 {
523 if (refs_ == nullptr) {
524 return 0;
525 }
526 return refs_->GetWeakRefCount();
527 }
528
AttemptAcquire(const void * objectId)529 bool RefBase::AttemptAcquire(const void *objectId)
530 {
531 if (refs_ == nullptr) {
532 return false;
533 }
534
535 int count = 0;
536 if (refs_->AttemptIncStrongRef(objectId, count)) {
537 refs_->SetAttemptAcquire();
538 if (count == INITIAL_PRIMARY_VALUE) {
539 OnFirstStrongRef(objectId);
540 }
541
542 return true;
543 }
544 return false;
545 }
546
AttemptIncStrongRef(const void * objectId)547 bool RefBase::AttemptIncStrongRef(const void *objectId)
548 {
549 if ((refs_ != nullptr) && (OnAttemptPromoted(objectId))) {
550 int count = 0;
551 bool ret = refs_->AttemptIncStrongRef(objectId, count);
552 if (count == INITIAL_PRIMARY_VALUE) {
553 OnFirstStrongRef(objectId);
554 }
555 return ret;
556 }
557
558 return false;
559 }
560
AttemptIncStrong(const void * objectId)561 bool RefBase::AttemptIncStrong(const void *objectId)
562 {
563 if (refs_ == nullptr) {
564 return false;
565 }
566 if (refs_->AttemptIncStrong(objectId)) {
567 refs_->SetAttemptAcquire();
568 return true;
569 }
570 return false;
571 }
572
IsAttemptAcquireSet()573 bool RefBase::IsAttemptAcquireSet()
574 {
575 if (refs_ == nullptr) {
576 return false;
577 }
578 return refs_->IsAttemptAcquireSet();
579 }
580
IsExtendLifeTimeSet()581 bool RefBase::IsExtendLifeTimeSet()
582 {
583 if (refs_ == nullptr) {
584 return false;
585 }
586 return refs_->IsLifeTimeExtended();
587 }
588
OnFirstStrongRef(const void *)589 void RefBase::OnFirstStrongRef(const void*)
590 {}
591
OnLastStrongRef(const void *)592 void RefBase::OnLastStrongRef(const void*)
593 {}
594
OnLastWeakRef(const void *)595 void RefBase::OnLastWeakRef(const void*)
596 {}
597
OnAttemptPromoted(const void *)598 bool RefBase::OnAttemptPromoted(const void*)
599 {
600 return true;
601 }
602
603 } // namespace OHOS
604