/* * 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 #include "base/bit_utils.h" #include "obj_ptr.h" #include "thread-current-inl.h" namespace art { template inline uintptr_t ObjPtr::GetCurrentTrimedCookie() { Thread* self = Thread::Current(); if (UNLIKELY(self == nullptr)) { return kCookieMask; } return self->GetPoisonObjectCookie() & kCookieMask; } template inline bool ObjPtr::IsValid() const { if (!kObjPtrPoisoning || IsNull()) { return true; } return GetCookie() == GetCurrentTrimedCookie(); } template inline void ObjPtr::AssertValid() const { if (kObjPtrPoisoning) { CHECK(IsValid()) << "Stale object pointer " << PtrUnchecked() << " , expected cookie " << GetCurrentTrimedCookie() << " but got " << GetCookie(); } } template inline uintptr_t ObjPtr::Encode(MirrorType* ptr) { uintptr_t ref = reinterpret_cast(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 template > */> inline ObjPtr::ObjPtr(Type* ptr) : reference_(Encode(static_cast(ptr))) { } template template > */> inline ObjPtr::ObjPtr(const ObjPtr& other) : reference_(other.reference_) { if (kObjPtrPoisoningValidateOnCopy) { AssertValid(); } } template template > */> inline ObjPtr& ObjPtr::operator=(const ObjPtr& other) { reference_ = other.reference_; if (kObjPtrPoisoningValidateOnCopy) { AssertValid(); } return *this; } template OBJPTR_INLINE ObjPtr& ObjPtr::operator=(MirrorType* ptr) { Assign(ptr); return *this; } template inline void ObjPtr::Assign(MirrorType* ptr) { reference_ = Encode(ptr); } template inline MirrorType* ObjPtr::operator->() const { return Ptr(); } template inline MirrorType* ObjPtr::Ptr() const { AssertValid(); return PtrUnchecked(); } template template inline ObjPtr ObjPtr::DownCast(ObjPtr ptr) { static_assert(std::is_base_of_v, "Target type must be a subtype of source type"); return static_cast(ptr.Ptr()); } template template inline ObjPtr ObjPtr::DownCast(SourceType* ptr) { static_assert(std::is_base_of_v, "Target type must be a subtype of source type"); return static_cast(ptr); } template size_t HashObjPtr::operator()(const ObjPtr& ptr) const { return std::hash()(ptr.Ptr()); } template inline std::enable_if_t || std::is_base_of_v, bool> operator==(ObjPtr lhs, ObjPtr rhs) { return lhs.Ptr() == rhs.Ptr(); } template inline std::enable_if_t || std::is_base_of_v, bool> operator==(const MirrorType1* lhs, ObjPtr rhs) { return lhs == rhs.Ptr(); } template inline std::enable_if_t || std::is_base_of_v, bool> operator==(ObjPtr lhs, const MirrorType2* rhs) { return lhs.Ptr() == rhs; } template inline std::enable_if_t || std::is_base_of_v, bool> operator!=(ObjPtr lhs, ObjPtr rhs) { return !(lhs == rhs); } template inline std::enable_if_t || std::is_base_of_v, bool> operator!=(const MirrorType1* lhs, ObjPtr rhs) { return !(lhs == rhs); } template inline std::enable_if_t || std::is_base_of_v, bool> operator!=(ObjPtr lhs, const MirrorType2* rhs) { return !(lhs == rhs); } template inline std::ostream& operator<<(std::ostream& os, ObjPtr 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_