• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
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