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_RUNTIME_READ_BARRIER_H_ 18 #define ART_RUNTIME_READ_BARRIER_H_ 19 20 #include <android-base/logging.h> 21 22 #include "base/locks.h" 23 #include "base/macros.h" 24 #include "base/runtime_debug.h" 25 #include "gc_root.h" 26 #include "jni.h" 27 #include "mirror/object_reference.h" 28 #include "offsets.h" 29 #include "read_barrier_config.h" 30 31 namespace art HIDDEN { 32 namespace mirror { 33 class Object; 34 template<typename MirrorType> class HeapReference; 35 } // namespace mirror 36 class ArtMethod; 37 38 class ReadBarrier { 39 public: 40 // Enable the to-space invariant checks. This is slow and happens very often. Do not enable in 41 // fast-debug environment. 42 DECLARE_RUNTIME_DEBUG_FLAG(kEnableToSpaceInvariantChecks); 43 44 // Enable the read barrier checks. This is slow and happens very often. Do not enable in 45 // fast-debug environment. 46 DECLARE_RUNTIME_DEBUG_FLAG(kEnableReadBarrierInvariantChecks); 47 48 // Return the reference at ref_addr, invoking read barrier as appropriate. 49 // Ref_addr is an address within obj. 50 // It's up to the implementation whether the given field gets updated whereas the return value 51 // must be an updated reference unless kAlwaysUpdateField is true. 52 template <typename MirrorType, 53 bool kIsVolatile, 54 ReadBarrierOption kReadBarrierOption = kWithReadBarrier, 55 bool kAlwaysUpdateField = false> 56 ALWAYS_INLINE static MirrorType* Barrier( 57 mirror::Object* obj, MemberOffset offset, mirror::HeapReference<MirrorType>* ref_addr) 58 REQUIRES_SHARED(Locks::mutator_lock_); 59 60 // It's up to the implementation whether the given root gets updated 61 // whereas the return value must be an updated reference. 62 template <typename MirrorType, ReadBarrierOption kReadBarrierOption = kWithReadBarrier> 63 ALWAYS_INLINE static MirrorType* BarrierForRoot(MirrorType** root, 64 GcRootSource* gc_root_source = nullptr) 65 REQUIRES_SHARED(Locks::mutator_lock_); 66 67 // It's up to the implementation whether the given root gets updated 68 // whereas the return value must be an updated reference. 69 template <typename MirrorType, ReadBarrierOption kReadBarrierOption = kWithReadBarrier> 70 ALWAYS_INLINE static MirrorType* BarrierForRoot(mirror::CompressedReference<MirrorType>* root, 71 GcRootSource* gc_root_source = nullptr) 72 REQUIRES_SHARED(Locks::mutator_lock_); 73 74 // Return the mirror Object if it is marked, or null if not. 75 template <typename MirrorType> 76 ALWAYS_INLINE static MirrorType* IsMarked(MirrorType* ref) 77 REQUIRES_SHARED(Locks::mutator_lock_); 78 79 static bool IsDuringStartup(); 80 81 // Without the holder object. AssertToSpaceInvariant(mirror::Object * ref)82 static void AssertToSpaceInvariant(mirror::Object* ref) 83 REQUIRES_SHARED(Locks::mutator_lock_) { 84 AssertToSpaceInvariant(nullptr, MemberOffset(0), ref); 85 } 86 // With the holder object. 87 static void AssertToSpaceInvariant(mirror::Object* obj, MemberOffset offset, 88 mirror::Object* ref) 89 REQUIRES_SHARED(Locks::mutator_lock_); 90 // With GcRootSource. 91 static void AssertToSpaceInvariant(GcRootSource* gc_root_source, mirror::Object* ref) 92 REQUIRES_SHARED(Locks::mutator_lock_); 93 94 // Without the holder object, and only with the read barrier configuration (no-op otherwise). MaybeAssertToSpaceInvariant(mirror::Object * ref)95 static void MaybeAssertToSpaceInvariant(mirror::Object* ref) 96 REQUIRES_SHARED(Locks::mutator_lock_) { 97 if (gUseReadBarrier) { 98 AssertToSpaceInvariant(ref); 99 } 100 } 101 102 // ALWAYS_INLINE on this caused a performance regression b/26744236. 103 static mirror::Object* Mark(mirror::Object* obj) REQUIRES_SHARED(Locks::mutator_lock_); 104 NonGrayState()105 static constexpr uint32_t NonGrayState() { 106 return kNonGrayState; 107 } GrayState()108 static constexpr uint32_t GrayState() { 109 return kGrayState; 110 } 111 112 // fake_address_dependency will be zero which should be bitwise-or'ed with the address of the 113 // subsequent load to prevent the reordering of the read barrier bit load and the subsequent 114 // object reference load (from one of `obj`'s fields). 115 // *fake_address_dependency will be set to 0. 116 ALWAYS_INLINE static bool IsGray(mirror::Object* obj, uintptr_t* fake_address_dependency) 117 REQUIRES_SHARED(Locks::mutator_lock_); 118 119 // This uses a load-acquire to load the read barrier bit internally to prevent the reordering of 120 // the read barrier bit load and the subsequent load. 121 ALWAYS_INLINE static bool IsGray(mirror::Object* obj) 122 REQUIRES_SHARED(Locks::mutator_lock_); 123 IsValidReadBarrierState(uint32_t rb_state)124 static bool IsValidReadBarrierState(uint32_t rb_state) { 125 return rb_state == kNonGrayState || rb_state == kGrayState; 126 } 127 128 private: 129 static constexpr uint32_t kNonGrayState = 0x0; // White (not marked) or black (marked through). 130 static constexpr uint32_t kGrayState = 0x1; // Marked, but not marked through. On mark stack. 131 static constexpr uint32_t kRBStateMask = 0x1; // The low bits for non-gray|gray. 132 }; 133 134 } // namespace art 135 136 #endif // ART_RUNTIME_READ_BARRIER_H_ 137