1 /*
2  * Copyright (C) 2014 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 #ifndef ART_LIBARTBASE_BASE_DEBUG_STACK_H_
18 #define ART_LIBARTBASE_BASE_DEBUG_STACK_H_
19 
20 #include <android-base/logging.h>
21 #include <android-base/macros.h>
22 
23 #include "globals.h"
24 
25 namespace art {
26 
27 // Helper classes for reference counting to enforce construction/destruction order and
28 // usage of the top element of a stack in debug mode with no overhead in release mode.
29 
30 // Reference counter. No references allowed in destructor or in explicitly called CheckNoRefs().
31 template <bool kIsDebug>
32 class DebugStackRefCounterImpl;
33 // Reference. Allows an explicit check that it's the top reference.
34 template <bool kIsDebug>
35 class DebugStackReferenceImpl;
36 // Indirect top reference. Checks that the reference is the top reference when used.
37 template <bool kIsDebug>
38 class DebugStackIndirectTopRefImpl;
39 
40 using DebugStackRefCounter = DebugStackRefCounterImpl<kIsDebugBuild>;
41 using DebugStackReference = DebugStackReferenceImpl<kIsDebugBuild>;
42 using DebugStackIndirectTopRef = DebugStackIndirectTopRefImpl<kIsDebugBuild>;
43 
44 // Non-debug mode specializations. This should be optimized away.
45 
46 template <>
47 class DebugStackRefCounterImpl<false> {
48  public:
IncrementRefCount()49   size_t IncrementRefCount() { return 0u; }
DecrementRefCount()50   void DecrementRefCount() { }
GetRefCount()51   size_t GetRefCount() const { return 0u; }
CheckNoRefs()52   void CheckNoRefs() const { }
53 };
54 
55 template <>
56 class DebugStackReferenceImpl<false> {
57  public:
DebugStackReferenceImpl(DebugStackRefCounterImpl<false> * counter)58   explicit DebugStackReferenceImpl([[maybe_unused]] DebugStackRefCounterImpl<false>* counter) {}
59   DebugStackReferenceImpl(const DebugStackReferenceImpl& other) = default;
60   DebugStackReferenceImpl& operator=(const DebugStackReferenceImpl& other) = default;
CheckTop()61   void CheckTop() { }
62 };
63 
64 template <>
65 class DebugStackIndirectTopRefImpl<false> {
66  public:
DebugStackIndirectTopRefImpl(DebugStackReferenceImpl<false> * ref)67   explicit DebugStackIndirectTopRefImpl([[maybe_unused]] DebugStackReferenceImpl<false>* ref) {}
68   DebugStackIndirectTopRefImpl(const DebugStackIndirectTopRefImpl& other) = default;
69   DebugStackIndirectTopRefImpl& operator=(const DebugStackIndirectTopRefImpl& other) = default;
CheckTop()70   void CheckTop() { }
71 };
72 
73 // Debug mode versions.
74 
75 template <bool kIsDebug>
76 class DebugStackRefCounterImpl {
77  public:
DebugStackRefCounterImpl()78   DebugStackRefCounterImpl() : ref_count_(0u) { }
~DebugStackRefCounterImpl()79   ~DebugStackRefCounterImpl() { CheckNoRefs(); }
IncrementRefCount()80   size_t IncrementRefCount() { return ++ref_count_; }
DecrementRefCount()81   void DecrementRefCount() { --ref_count_; }
GetRefCount()82   size_t GetRefCount() const { return ref_count_; }
CheckNoRefs()83   void CheckNoRefs() const { CHECK_EQ(ref_count_, 0u); }
84 
85  private:
86   size_t ref_count_;
87 };
88 
89 template <bool kIsDebug>
90 class DebugStackReferenceImpl {
91  public:
DebugStackReferenceImpl(DebugStackRefCounterImpl<kIsDebug> * counter)92   explicit DebugStackReferenceImpl(DebugStackRefCounterImpl<kIsDebug>* counter)
93     : counter_(counter), ref_count_(counter->IncrementRefCount()) {
94   }
DebugStackReferenceImpl(const DebugStackReferenceImpl & other)95   DebugStackReferenceImpl(const DebugStackReferenceImpl& other)
96     : counter_(other.counter_), ref_count_(counter_->IncrementRefCount()) {
97   }
DebugStackReferenceImpl(DebugStackReferenceImpl && other)98   DebugStackReferenceImpl(DebugStackReferenceImpl&& other) noexcept
99     : counter_(other.counter_), ref_count_(other.ref_count_) {
100     other.counter_ = nullptr;
101   }
102   DebugStackReferenceImpl& operator=(const DebugStackReferenceImpl& other) {
103     CHECK(counter_ == other.counter_);
104     return *this;
105   }
~DebugStackReferenceImpl()106   ~DebugStackReferenceImpl() {
107     if (counter_ != nullptr) {
108       counter_->DecrementRefCount();
109     }
110   }
CheckTop()111   void CheckTop() { CHECK_EQ(counter_->GetRefCount(), ref_count_); }
112 
113  private:
114   DebugStackRefCounterImpl<true>* counter_;
115   size_t ref_count_;
116 };
117 
118 template <bool kIsDebug>
119 class DebugStackIndirectTopRefImpl {
120  public:
DebugStackIndirectTopRefImpl(DebugStackReferenceImpl<kIsDebug> * ref)121   explicit DebugStackIndirectTopRefImpl(DebugStackReferenceImpl<kIsDebug>* ref)
122       : ref_(ref) {
123     CheckTop();
124   }
DebugStackIndirectTopRefImpl(const DebugStackIndirectTopRefImpl & other)125   DebugStackIndirectTopRefImpl(const DebugStackIndirectTopRefImpl& other)
126       : ref_(other.ref_) {
127     CheckTop();
128   }
129   DebugStackIndirectTopRefImpl& operator=(const DebugStackIndirectTopRefImpl& other) {
130     CHECK(ref_ == other.ref_);
131     CheckTop();
132     return *this;
133   }
~DebugStackIndirectTopRefImpl()134   ~DebugStackIndirectTopRefImpl() {
135     CheckTop();
136   }
CheckTop()137   void CheckTop() {
138     ref_->CheckTop();
139   }
140 
141  private:
142   DebugStackReferenceImpl<kIsDebug>* ref_;
143 };
144 
145 }  // namespace art
146 
147 #endif  // ART_LIBARTBASE_BASE_DEBUG_STACK_H_
148