/* * 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_REFLECTIVE_HANDLE_H_ #define ART_RUNTIME_REFLECTIVE_HANDLE_H_ #include "base/macros.h" #include "base/value_object.h" #include "reflective_reference.h" namespace art HIDDEN { // This is a holder similar to Handle that is used to hold reflective references to ArtField and // ArtMethod structures. A reflective reference is one that must be updated if the underlying class // or instances are replaced due to structural redefinition or some other process. In general these // don't need to be used. It's only when it's important that a reference to a field not become // obsolete and it needs to be held over a suspend point that this should be used. template class ReflectiveHandle : public ValueObject { public: static_assert(std::is_same_v || std::is_same_v, "Expected ArtField or ArtMethod"); ReflectiveHandle() : reference_(nullptr) {} ALWAYS_INLINE ReflectiveHandle(const ReflectiveHandle& handle) = default; ALWAYS_INLINE ReflectiveHandle& operator=(const ReflectiveHandle& handle) = default; ALWAYS_INLINE explicit ReflectiveHandle(ReflectiveReference* reference) : reference_(reference) {} ALWAYS_INLINE T& operator*() const REQUIRES_SHARED(Locks::mutator_lock_) { return *Get(); } ALWAYS_INLINE T* operator->() const REQUIRES_SHARED(Locks::mutator_lock_) { return Get(); } ALWAYS_INLINE T* Get() const REQUIRES_SHARED(Locks::mutator_lock_) { return reference_->Ptr(); } ALWAYS_INLINE bool IsNull() const { // It's safe to null-check it without a read barrier. return reference_->IsNull(); } ALWAYS_INLINE bool operator!=(std::nullptr_t) const REQUIRES_SHARED(Locks::mutator_lock_) { return !IsNull(); } ALWAYS_INLINE bool operator==(std::nullptr_t) const REQUIRES_SHARED(Locks::mutator_lock_) { return IsNull(); } protected: ReflectiveReference* reference_; private: friend class BaseReflectiveHandleScope; template friend class StackReflectiveHandleScope; }; // Handles that support assignment. template class MutableReflectiveHandle : public ReflectiveHandle { public: MutableReflectiveHandle() {} ALWAYS_INLINE MutableReflectiveHandle(const MutableReflectiveHandle& handle) = default; ALWAYS_INLINE MutableReflectiveHandle& operator=(const MutableReflectiveHandle& handle) = default; ALWAYS_INLINE explicit MutableReflectiveHandle(ReflectiveReference* reference) : ReflectiveHandle(reference) {} ALWAYS_INLINE T* Assign(T* reference) REQUIRES_SHARED(Locks::mutator_lock_) { ReflectiveReference* ref = ReflectiveHandle::reference_; T* old = ref->Ptr(); ref->Assign(reference); return old; } private: friend class BaseReflectiveHandleScope; template friend class StackReflectiveHandleScope; }; template class ReflectiveHandleWrapper : public MutableReflectiveHandle { public: ReflectiveHandleWrapper(T** obj, const MutableReflectiveHandle& handle) : MutableReflectiveHandle(handle), obj_(obj) { } ReflectiveHandleWrapper(const ReflectiveHandleWrapper&) = default; ~ReflectiveHandleWrapper() { *obj_ = MutableReflectiveHandle::Get(); } private: T** const obj_; }; } // namespace art #endif // ART_RUNTIME_REFLECTIVE_HANDLE_H_