1*795d594fSAndroid Build Coastguard Worker /* 2*795d594fSAndroid Build Coastguard Worker * Copyright (C) 2014 The Android Open Source Project 3*795d594fSAndroid Build Coastguard Worker * 4*795d594fSAndroid Build Coastguard Worker * Licensed under the Apache License, Version 2.0 (the "License"); 5*795d594fSAndroid Build Coastguard Worker * you may not use this file except in compliance with the License. 6*795d594fSAndroid Build Coastguard Worker * You may obtain a copy of the License at 7*795d594fSAndroid Build Coastguard Worker * 8*795d594fSAndroid Build Coastguard Worker * http://www.apache.org/licenses/LICENSE-2.0 9*795d594fSAndroid Build Coastguard Worker * 10*795d594fSAndroid Build Coastguard Worker * Unless required by applicable law or agreed to in writing, software 11*795d594fSAndroid Build Coastguard Worker * distributed under the License is distributed on an "AS IS" BASIS, 12*795d594fSAndroid Build Coastguard Worker * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13*795d594fSAndroid Build Coastguard Worker * See the License for the specific language governing permissions and 14*795d594fSAndroid Build Coastguard Worker * limitations under the License. 15*795d594fSAndroid Build Coastguard Worker */ 16*795d594fSAndroid Build Coastguard Worker 17*795d594fSAndroid Build Coastguard Worker #ifndef ART_RUNTIME_MIRROR_REFERENCE_H_ 18*795d594fSAndroid Build Coastguard Worker #define ART_RUNTIME_MIRROR_REFERENCE_H_ 19*795d594fSAndroid Build Coastguard Worker 20*795d594fSAndroid Build Coastguard Worker #include "base/locks.h" 21*795d594fSAndroid Build Coastguard Worker #include "base/macros.h" 22*795d594fSAndroid Build Coastguard Worker #include "base/pointer_size.h" 23*795d594fSAndroid Build Coastguard Worker #include "obj_ptr.h" 24*795d594fSAndroid Build Coastguard Worker #include "object.h" 25*795d594fSAndroid Build Coastguard Worker #include "read_barrier_option.h" 26*795d594fSAndroid Build Coastguard Worker 27*795d594fSAndroid Build Coastguard Worker namespace art HIDDEN { 28*795d594fSAndroid Build Coastguard Worker 29*795d594fSAndroid Build Coastguard Worker namespace gc { 30*795d594fSAndroid Build Coastguard Worker 31*795d594fSAndroid Build Coastguard Worker class ReferenceProcessor; 32*795d594fSAndroid Build Coastguard Worker class ReferenceQueue; 33*795d594fSAndroid Build Coastguard Worker 34*795d594fSAndroid Build Coastguard Worker } // namespace gc 35*795d594fSAndroid Build Coastguard Worker 36*795d594fSAndroid Build Coastguard Worker struct ReferenceOffsets; 37*795d594fSAndroid Build Coastguard Worker struct FinalizerReferenceOffsets; 38*795d594fSAndroid Build Coastguard Worker 39*795d594fSAndroid Build Coastguard Worker namespace mirror { 40*795d594fSAndroid Build Coastguard Worker 41*795d594fSAndroid Build Coastguard Worker // C++ mirror of java.lang.ref.Reference 42*795d594fSAndroid Build Coastguard Worker class MANAGED Reference : public Object { 43*795d594fSAndroid Build Coastguard Worker public: 44*795d594fSAndroid Build Coastguard Worker MIRROR_CLASS("Ljava/lang/ref/Reference;"); 45*795d594fSAndroid Build Coastguard Worker 46*795d594fSAndroid Build Coastguard Worker // Size of java.lang.ref.Reference.class. 47*795d594fSAndroid Build Coastguard Worker static uint32_t ClassSize(PointerSize pointer_size); 48*795d594fSAndroid Build Coastguard Worker 49*795d594fSAndroid Build Coastguard Worker // Size of an instance of java.lang.ref.Reference. InstanceSize()50*795d594fSAndroid Build Coastguard Worker static constexpr uint32_t InstanceSize() { 51*795d594fSAndroid Build Coastguard Worker return sizeof(Reference); 52*795d594fSAndroid Build Coastguard Worker } 53*795d594fSAndroid Build Coastguard Worker PendingNextOffset()54*795d594fSAndroid Build Coastguard Worker static MemberOffset PendingNextOffset() { 55*795d594fSAndroid Build Coastguard Worker return OFFSET_OF_OBJECT_MEMBER(Reference, pending_next_); 56*795d594fSAndroid Build Coastguard Worker } QueueOffset()57*795d594fSAndroid Build Coastguard Worker static MemberOffset QueueOffset() { 58*795d594fSAndroid Build Coastguard Worker return OFFSET_OF_OBJECT_MEMBER(Reference, queue_); 59*795d594fSAndroid Build Coastguard Worker } QueueNextOffset()60*795d594fSAndroid Build Coastguard Worker static MemberOffset QueueNextOffset() { 61*795d594fSAndroid Build Coastguard Worker return OFFSET_OF_OBJECT_MEMBER(Reference, queue_next_); 62*795d594fSAndroid Build Coastguard Worker } ReferentOffset()63*795d594fSAndroid Build Coastguard Worker static MemberOffset ReferentOffset() { 64*795d594fSAndroid Build Coastguard Worker return OFFSET_OF_OBJECT_MEMBER(Reference, referent_); 65*795d594fSAndroid Build Coastguard Worker } 66*795d594fSAndroid Build Coastguard Worker template<ReadBarrierOption kReadBarrierOption = kWithReadBarrier> GetReferent()67*795d594fSAndroid Build Coastguard Worker Object* GetReferent() REQUIRES_SHARED(Locks::mutator_lock_) { 68*795d594fSAndroid Build Coastguard Worker return GetFieldObjectVolatile<Object, kDefaultVerifyFlags, kReadBarrierOption>( 69*795d594fSAndroid Build Coastguard Worker ReferentOffset()); 70*795d594fSAndroid Build Coastguard Worker } 71*795d594fSAndroid Build Coastguard Worker template<bool kTransactionActive> 72*795d594fSAndroid Build Coastguard Worker void SetReferent(ObjPtr<Object> referent) REQUIRES_SHARED(Locks::mutator_lock_); 73*795d594fSAndroid Build Coastguard Worker template<bool kTransactionActive> ClearReferent()74*795d594fSAndroid Build Coastguard Worker void ClearReferent() REQUIRES_SHARED(Locks::mutator_lock_) { 75*795d594fSAndroid Build Coastguard Worker SetFieldObjectVolatile<kTransactionActive>(ReferentOffset(), nullptr); 76*795d594fSAndroid Build Coastguard Worker } 77*795d594fSAndroid Build Coastguard Worker 78*795d594fSAndroid Build Coastguard Worker template <ReadBarrierOption kReadBarrierOption = kWithReadBarrier> GetPendingNext()79*795d594fSAndroid Build Coastguard Worker Reference* GetPendingNext() REQUIRES_SHARED(Locks::mutator_lock_) { 80*795d594fSAndroid Build Coastguard Worker return GetFieldObject<Reference, kDefaultVerifyFlags, kReadBarrierOption>(PendingNextOffset()); 81*795d594fSAndroid Build Coastguard Worker } 82*795d594fSAndroid Build Coastguard Worker 83*795d594fSAndroid Build Coastguard Worker void SetPendingNext(ObjPtr<Reference> pending_next) REQUIRES_SHARED(Locks::mutator_lock_); 84*795d594fSAndroid Build Coastguard Worker 85*795d594fSAndroid Build Coastguard Worker // Returns true if the reference's pendingNext is null, indicating it is 86*795d594fSAndroid Build Coastguard Worker // okay to process this reference. 87*795d594fSAndroid Build Coastguard Worker // 88*795d594fSAndroid Build Coastguard Worker // If pendingNext is not null, then one of the following cases holds: 89*795d594fSAndroid Build Coastguard Worker // 1. The reference has already been enqueued to a java ReferenceQueue. In 90*795d594fSAndroid Build Coastguard Worker // this case the referent should not be considered for reference processing 91*795d594fSAndroid Build Coastguard Worker // ever again. 92*795d594fSAndroid Build Coastguard Worker // 2. The reference is currently part of a list of references that may 93*795d594fSAndroid Build Coastguard Worker // shortly be enqueued on a java ReferenceQueue. In this case the reference 94*795d594fSAndroid Build Coastguard Worker // should not be processed again until and unless the reference has been 95*795d594fSAndroid Build Coastguard Worker // removed from the list after having determined the reference is not ready 96*795d594fSAndroid Build Coastguard Worker // to be enqueued on a java ReferenceQueue. IsUnprocessed()97*795d594fSAndroid Build Coastguard Worker bool IsUnprocessed() REQUIRES_SHARED(Locks::mutator_lock_) { 98*795d594fSAndroid Build Coastguard Worker return GetPendingNext<kWithoutReadBarrier>() == nullptr; 99*795d594fSAndroid Build Coastguard Worker } 100*795d594fSAndroid Build Coastguard Worker 101*795d594fSAndroid Build Coastguard Worker private: 102*795d594fSAndroid Build Coastguard Worker // Note: This avoids a read barrier, it should only be used by the GC. GetReferentReferenceAddr()103*795d594fSAndroid Build Coastguard Worker HeapReference<Object>* GetReferentReferenceAddr() REQUIRES_SHARED(Locks::mutator_lock_) { 104*795d594fSAndroid Build Coastguard Worker return GetFieldObjectReferenceAddr<kDefaultVerifyFlags>(ReferentOffset()); 105*795d594fSAndroid Build Coastguard Worker } 106*795d594fSAndroid Build Coastguard Worker 107*795d594fSAndroid Build Coastguard Worker // Field order required by test "ValidateFieldOrderOfJavaCppUnionClasses". 108*795d594fSAndroid Build Coastguard Worker HeapReference<Reference> pending_next_; 109*795d594fSAndroid Build Coastguard Worker HeapReference<Object> queue_; 110*795d594fSAndroid Build Coastguard Worker HeapReference<Reference> queue_next_; 111*795d594fSAndroid Build Coastguard Worker HeapReference<Object> referent_; // Note this is Java volatile: 112*795d594fSAndroid Build Coastguard Worker 113*795d594fSAndroid Build Coastguard Worker friend struct art::ReferenceOffsets; // for verifying offset information 114*795d594fSAndroid Build Coastguard Worker friend class gc::ReferenceProcessor; 115*795d594fSAndroid Build Coastguard Worker friend class gc::ReferenceQueue; 116*795d594fSAndroid Build Coastguard Worker DISALLOW_IMPLICIT_CONSTRUCTORS(Reference); 117*795d594fSAndroid Build Coastguard Worker }; 118*795d594fSAndroid Build Coastguard Worker 119*795d594fSAndroid Build Coastguard Worker // C++ mirror of java.lang.ref.FinalizerReference 120*795d594fSAndroid Build Coastguard Worker class MANAGED FinalizerReference : public Reference { 121*795d594fSAndroid Build Coastguard Worker public: 122*795d594fSAndroid Build Coastguard Worker MIRROR_CLASS("Ljava/lang/ref/FinalizerReference;"); 123*795d594fSAndroid Build Coastguard Worker ZombieOffset()124*795d594fSAndroid Build Coastguard Worker static MemberOffset ZombieOffset() { 125*795d594fSAndroid Build Coastguard Worker return OFFSET_OF_OBJECT_MEMBER(FinalizerReference, zombie_); 126*795d594fSAndroid Build Coastguard Worker } 127*795d594fSAndroid Build Coastguard Worker 128*795d594fSAndroid Build Coastguard Worker template<bool kTransactionActive> 129*795d594fSAndroid Build Coastguard Worker void SetZombie(ObjPtr<Object> zombie) REQUIRES_SHARED(Locks::mutator_lock_); 130*795d594fSAndroid Build Coastguard Worker GetZombie()131*795d594fSAndroid Build Coastguard Worker Object* GetZombie() REQUIRES_SHARED(Locks::mutator_lock_) { 132*795d594fSAndroid Build Coastguard Worker return GetFieldObjectVolatile<Object>(ZombieOffset()); 133*795d594fSAndroid Build Coastguard Worker } 134*795d594fSAndroid Build Coastguard Worker 135*795d594fSAndroid Build Coastguard Worker private: 136*795d594fSAndroid Build Coastguard Worker HeapReference<FinalizerReference> next_; 137*795d594fSAndroid Build Coastguard Worker HeapReference<FinalizerReference> prev_; 138*795d594fSAndroid Build Coastguard Worker HeapReference<Object> zombie_; 139*795d594fSAndroid Build Coastguard Worker 140*795d594fSAndroid Build Coastguard Worker friend struct art::FinalizerReferenceOffsets; // for verifying offset information 141*795d594fSAndroid Build Coastguard Worker DISALLOW_IMPLICIT_CONSTRUCTORS(FinalizerReference); 142*795d594fSAndroid Build Coastguard Worker }; 143*795d594fSAndroid Build Coastguard Worker 144*795d594fSAndroid Build Coastguard Worker } // namespace mirror 145*795d594fSAndroid Build Coastguard Worker } // namespace art 146*795d594fSAndroid Build Coastguard Worker 147*795d594fSAndroid Build Coastguard Worker #endif // ART_RUNTIME_MIRROR_REFERENCE_H_ 148