| /* |
| * Copyright (C) 2016 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_OBJ_PTR_INL_H_ |
| #define ART_RUNTIME_OBJ_PTR_INL_H_ |
| |
| #include <ostream> |
| |
| #include "base/bit_utils.h" |
| #include "obj_ptr.h" |
| #include "thread-current-inl.h" |
| |
| namespace art { |
| |
| template<class MirrorType> |
| inline uintptr_t ObjPtr<MirrorType>::GetCurrentTrimedCookie() { |
| Thread* self = Thread::Current(); |
| if (UNLIKELY(self == nullptr)) { |
| return kCookieMask; |
| } |
| return self->GetPoisonObjectCookie() & kCookieMask; |
| } |
| |
| template<class MirrorType> |
| inline bool ObjPtr<MirrorType>::IsValid() const { |
| if (!kObjPtrPoisoning || IsNull()) { |
| return true; |
| } |
| return GetCookie() == GetCurrentTrimedCookie(); |
| } |
| |
| template<class MirrorType> |
| inline void ObjPtr<MirrorType>::AssertValid() const { |
| if (kObjPtrPoisoning) { |
| CHECK(IsValid()) << "Stale object pointer " << PtrUnchecked() << " , expected cookie " |
| << GetCurrentTrimedCookie() << " but got " << GetCookie(); |
| } |
| } |
| |
| template<class MirrorType> |
| inline uintptr_t ObjPtr<MirrorType>::Encode(MirrorType* ptr) { |
| uintptr_t ref = reinterpret_cast<uintptr_t>(ptr); |
| DCHECK_ALIGNED(ref, kObjectAlignment); |
| if (kObjPtrPoisoning && ref != 0) { |
| DCHECK_LE(ref, 0xFFFFFFFFU); |
| ref >>= kObjectAlignmentShift; |
| // Put cookie in high bits. |
| ref |= GetCurrentTrimedCookie() << kCookieShift; |
| } |
| return ref; |
| } |
| |
| template<class MirrorType> |
| template <typename Type, |
| typename /* = typename std::enable_if_t<std::is_base_of_v<MirrorType, Type>> */> |
| inline ObjPtr<MirrorType>::ObjPtr(Type* ptr) |
| : reference_(Encode(static_cast<MirrorType*>(ptr))) { |
| } |
| |
| template<class MirrorType> |
| template <typename Type, |
| typename /* = typename std::enable_if_t<std::is_base_of_v<MirrorType, Type>> */> |
| inline ObjPtr<MirrorType>::ObjPtr(const ObjPtr<Type>& other) |
| : reference_(other.reference_) { |
| if (kObjPtrPoisoningValidateOnCopy) { |
| AssertValid(); |
| } |
| } |
| |
| template<class MirrorType> |
| template <typename Type, |
| typename /* = typename std::enable_if_t<std::is_base_of_v<MirrorType, Type>> */> |
| inline ObjPtr<MirrorType>& ObjPtr<MirrorType>::operator=(const ObjPtr<Type>& other) { |
| reference_ = other.reference_; |
| if (kObjPtrPoisoningValidateOnCopy) { |
| AssertValid(); |
| } |
| return *this; |
| } |
| |
| template<class MirrorType> |
| OBJPTR_INLINE ObjPtr<MirrorType>& ObjPtr<MirrorType>::operator=(MirrorType* ptr) { |
| Assign(ptr); |
| return *this; |
| } |
| |
| template<class MirrorType> |
| inline void ObjPtr<MirrorType>::Assign(MirrorType* ptr) { |
| reference_ = Encode(ptr); |
| } |
| |
| template<class MirrorType> |
| inline MirrorType* ObjPtr<MirrorType>::operator->() const { |
| return Ptr(); |
| } |
| |
| template<class MirrorType> |
| inline MirrorType* ObjPtr<MirrorType>::Ptr() const { |
| AssertValid(); |
| return PtrUnchecked(); |
| } |
| |
| template<class MirrorType> |
| template <typename SourceType> |
| inline ObjPtr<MirrorType> ObjPtr<MirrorType>::DownCast(ObjPtr<SourceType> ptr) { |
| static_assert(std::is_base_of_v<SourceType, MirrorType>, |
| "Target type must be a subtype of source type"); |
| return static_cast<MirrorType*>(ptr.Ptr()); |
| } |
| |
| template<class MirrorType> |
| template <typename SourceType> |
| inline ObjPtr<MirrorType> ObjPtr<MirrorType>::DownCast(SourceType* ptr) { |
| static_assert(std::is_base_of_v<SourceType, MirrorType>, |
| "Target type must be a subtype of source type"); |
| return static_cast<MirrorType*>(ptr); |
| } |
| |
| template<class MirrorType> |
| size_t HashObjPtr::operator()(const ObjPtr<MirrorType>& ptr) const { |
| return std::hash<MirrorType*>()(ptr.Ptr()); |
| } |
| |
| template<class MirrorType1, class MirrorType2> |
| inline std::enable_if_t<std::is_base_of_v<MirrorType1, MirrorType2> || |
| std::is_base_of_v<MirrorType2, MirrorType1>, bool> |
| operator==(ObjPtr<MirrorType1> lhs, ObjPtr<MirrorType2> rhs) { |
| return lhs.Ptr() == rhs.Ptr(); |
| } |
| |
| template<class MirrorType1, class MirrorType2> |
| inline std::enable_if_t<std::is_base_of_v<MirrorType1, MirrorType2> || |
| std::is_base_of_v<MirrorType2, MirrorType1>, bool> |
| operator==(const MirrorType1* lhs, ObjPtr<MirrorType2> rhs) { |
| return lhs == rhs.Ptr(); |
| } |
| |
| template<class MirrorType1, class MirrorType2> |
| inline std::enable_if_t<std::is_base_of_v<MirrorType1, MirrorType2> || |
| std::is_base_of_v<MirrorType2, MirrorType1>, bool> |
| operator==(ObjPtr<MirrorType1> lhs, const MirrorType2* rhs) { |
| return lhs.Ptr() == rhs; |
| } |
| |
| template<class MirrorType1, class MirrorType2> |
| inline std::enable_if_t<std::is_base_of_v<MirrorType1, MirrorType2> || |
| std::is_base_of_v<MirrorType2, MirrorType1>, bool> |
| operator!=(ObjPtr<MirrorType1> lhs, ObjPtr<MirrorType2> rhs) { |
| return !(lhs == rhs); |
| } |
| |
| template<class MirrorType1, class MirrorType2> |
| inline std::enable_if_t<std::is_base_of_v<MirrorType1, MirrorType2> || |
| std::is_base_of_v<MirrorType2, MirrorType1>, bool> |
| operator!=(const MirrorType1* lhs, ObjPtr<MirrorType2> rhs) { |
| return !(lhs == rhs); |
| } |
| |
| template<class MirrorType1, class MirrorType2> |
| inline std::enable_if_t<std::is_base_of_v<MirrorType1, MirrorType2> || |
| std::is_base_of_v<MirrorType2, MirrorType1>, bool> |
| operator!=(ObjPtr<MirrorType1> lhs, const MirrorType2* rhs) { |
| return !(lhs == rhs); |
| } |
| |
| template<class MirrorType> |
| inline std::ostream& operator<<(std::ostream& os, ObjPtr<MirrorType> ptr) { |
| // May be used for dumping bad pointers, do not use the checked version. |
| return os << ptr.PtrUnchecked(); |
| } |
| |
| } // namespace art |
| |
| #endif // ART_RUNTIME_OBJ_PTR_INL_H_ |