/* * Copyright (C) 2014 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #ifndef ART_RUNTIME_READ_BARRIER_H_ #define ART_RUNTIME_READ_BARRIER_H_ #include #include "base/locks.h" #include "base/macros.h" #include "base/runtime_debug.h" #include "gc_root.h" #include "jni.h" #include "mirror/object_reference.h" #include "offsets.h" #include "read_barrier_config.h" namespace art HIDDEN { namespace mirror { class Object; template class HeapReference; } // namespace mirror class ArtMethod; class ReadBarrier { public: // Enable the to-space invariant checks. This is slow and happens very often. Do not enable in // fast-debug environment. DECLARE_RUNTIME_DEBUG_FLAG(kEnableToSpaceInvariantChecks); // Enable the read barrier checks. This is slow and happens very often. Do not enable in // fast-debug environment. DECLARE_RUNTIME_DEBUG_FLAG(kEnableReadBarrierInvariantChecks); // Return the reference at ref_addr, invoking read barrier as appropriate. // Ref_addr is an address within obj. // It's up to the implementation whether the given field gets updated whereas the return value // must be an updated reference unless kAlwaysUpdateField is true. template ALWAYS_INLINE static MirrorType* Barrier( mirror::Object* obj, MemberOffset offset, mirror::HeapReference* ref_addr) REQUIRES_SHARED(Locks::mutator_lock_); // It's up to the implementation whether the given root gets updated // whereas the return value must be an updated reference. template ALWAYS_INLINE static MirrorType* BarrierForRoot(MirrorType** root, GcRootSource* gc_root_source = nullptr) REQUIRES_SHARED(Locks::mutator_lock_); // It's up to the implementation whether the given root gets updated // whereas the return value must be an updated reference. template ALWAYS_INLINE static MirrorType* BarrierForRoot(mirror::CompressedReference* root, GcRootSource* gc_root_source = nullptr) REQUIRES_SHARED(Locks::mutator_lock_); // Return the mirror Object if it is marked, or null if not. template ALWAYS_INLINE static MirrorType* IsMarked(MirrorType* ref) REQUIRES_SHARED(Locks::mutator_lock_); static bool IsDuringStartup(); // Without the holder object. static void AssertToSpaceInvariant(mirror::Object* ref) REQUIRES_SHARED(Locks::mutator_lock_) { AssertToSpaceInvariant(nullptr, MemberOffset(0), ref); } // With the holder object. static void AssertToSpaceInvariant(mirror::Object* obj, MemberOffset offset, mirror::Object* ref) REQUIRES_SHARED(Locks::mutator_lock_); // With GcRootSource. static void AssertToSpaceInvariant(GcRootSource* gc_root_source, mirror::Object* ref) REQUIRES_SHARED(Locks::mutator_lock_); // Without the holder object, and only with the read barrier configuration (no-op otherwise). static void MaybeAssertToSpaceInvariant(mirror::Object* ref) REQUIRES_SHARED(Locks::mutator_lock_) { if (gUseReadBarrier) { AssertToSpaceInvariant(ref); } } // ALWAYS_INLINE on this caused a performance regression b/26744236. static mirror::Object* Mark(mirror::Object* obj) REQUIRES_SHARED(Locks::mutator_lock_); static constexpr uint32_t NonGrayState() { return kNonGrayState; } static constexpr uint32_t GrayState() { return kGrayState; } // fake_address_dependency will be zero which should be bitwise-or'ed with the address of the // subsequent load to prevent the reordering of the read barrier bit load and the subsequent // object reference load (from one of `obj`'s fields). // *fake_address_dependency will be set to 0. ALWAYS_INLINE static bool IsGray(mirror::Object* obj, uintptr_t* fake_address_dependency) REQUIRES_SHARED(Locks::mutator_lock_); // This uses a load-acquire to load the read barrier bit internally to prevent the reordering of // the read barrier bit load and the subsequent load. ALWAYS_INLINE static bool IsGray(mirror::Object* obj) REQUIRES_SHARED(Locks::mutator_lock_); static bool IsValidReadBarrierState(uint32_t rb_state) { return rb_state == kNonGrayState || rb_state == kGrayState; } private: static constexpr uint32_t kNonGrayState = 0x0; // White (not marked) or black (marked through). static constexpr uint32_t kGrayState = 0x1; // Marked, but not marked through. On mark stack. static constexpr uint32_t kRBStateMask = 0x1; // The low bits for non-gray|gray. }; } // namespace art #endif // ART_RUNTIME_READ_BARRIER_H_