blob: 07d47d31e75881612f96dab6cb33da0b3876950e [file] [log] [blame]
Mathieu Chartier8fa2dad2014-03-13 12:22:56 -07001/*
2 * Copyright (C) 2014 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17#ifndef ART_RUNTIME_MIRROR_REFERENCE_H_
18#define ART_RUNTIME_MIRROR_REFERENCE_H_
19
Fred Shih4ee7a662014-07-11 09:59:27 -070020#include "class.h"
Mathieu Chartier8fa2dad2014-03-13 12:22:56 -070021#include "object.h"
Fred Shih4ee7a662014-07-11 09:59:27 -070022#include "object_callbacks.h"
23#include "read_barrier.h"
24#include "thread.h"
Mathieu Chartier8fa2dad2014-03-13 12:22:56 -070025
26namespace art {
27
Mathieu Chartier308351a2014-06-15 12:39:02 -070028namespace gc {
29
30class ReferenceProcessor;
31class ReferenceQueue;
32
33} // namespace gc
34
Mathieu Chartier8fa2dad2014-03-13 12:22:56 -070035struct ReferenceOffsets;
36struct FinalizerReferenceOffsets;
37
38namespace mirror {
39
40// C++ mirror of java.lang.ref.Reference
41class MANAGED Reference : public Object {
42 public:
Fred Shih4ee7a662014-07-11 09:59:27 -070043 // Size of java.lang.ref.Reference.class.
44 static uint32_t ClassSize();
45
46 // Size of an instance of java.lang.ref.Reference.
47 static constexpr uint32_t InstanceSize() {
48 return sizeof(Reference);
49 }
50
Mathieu Chartier8fa2dad2014-03-13 12:22:56 -070051 static MemberOffset PendingNextOffset() {
52 return OFFSET_OF_OBJECT_MEMBER(Reference, pending_next_);
53 }
54 static MemberOffset QueueOffset() {
55 return OFFSET_OF_OBJECT_MEMBER(Reference, queue_);
56 }
57 static MemberOffset QueueNextOffset() {
58 return OFFSET_OF_OBJECT_MEMBER(Reference, queue_next_);
59 }
60 static MemberOffset ReferentOffset() {
61 return OFFSET_OF_OBJECT_MEMBER(Reference, referent_);
62 }
Hiroshi Yamauchibfff21a2014-05-09 12:21:15 -070063 template<ReadBarrierOption kReadBarrierOption = kWithReadBarrier>
Mathieu Chartier8fa2dad2014-03-13 12:22:56 -070064 Object* GetReferent() SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
Hiroshi Yamauchibfff21a2014-05-09 12:21:15 -070065 return GetFieldObjectVolatile<Object, kDefaultVerifyFlags, kReadBarrierOption>(
66 ReferentOffset());
Mathieu Chartier8fa2dad2014-03-13 12:22:56 -070067 }
68 template<bool kTransactionActive>
69 void SetReferent(Object* referent) SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
Ian Rogersb0fa5dc2014-04-28 16:47:08 -070070 SetFieldObjectVolatile<kTransactionActive>(ReferentOffset(), referent);
Mathieu Chartier8fa2dad2014-03-13 12:22:56 -070071 }
72 template<bool kTransactionActive>
73 void ClearReferent() SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
Ian Rogersb0fa5dc2014-04-28 16:47:08 -070074 SetFieldObjectVolatile<kTransactionActive>(ReferentOffset(), nullptr);
Mathieu Chartier8fa2dad2014-03-13 12:22:56 -070075 }
Mathieu Chartier8fa2dad2014-03-13 12:22:56 -070076 // Volatile read/write is not necessary since the java pending next is only accessed from
77 // the java threads for cleared references. Once these cleared references have a null referent,
78 // we never end up reading their pending next from the GC again.
79 Reference* GetPendingNext() SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
Ian Rogersb0fa5dc2014-04-28 16:47:08 -070080 return GetFieldObject<Reference>(PendingNextOffset());
Mathieu Chartier8fa2dad2014-03-13 12:22:56 -070081 }
82 template<bool kTransactionActive>
83 void SetPendingNext(Reference* pending_next) SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
Ian Rogersb0fa5dc2014-04-28 16:47:08 -070084 SetFieldObject<kTransactionActive>(PendingNextOffset(), pending_next);
Mathieu Chartier8fa2dad2014-03-13 12:22:56 -070085 }
86
87 bool IsEnqueued() SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
88 // Since the references are stored as cyclic lists it means that once enqueued, the pending
89 // next is always non-null.
90 return GetPendingNext() != nullptr;
91 }
92
93 bool IsEnqueuable() SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
94
Fred Shih4ee7a662014-07-11 09:59:27 -070095 template<ReadBarrierOption kReadBarrierOption = kWithReadBarrier>
96 static Class* GetJavaLangRefReference() SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
97 DCHECK(java_lang_ref_Reference_ != nullptr);
98 return ReadBarrier::BarrierForRoot<mirror::Class, kReadBarrierOption>(
99 &java_lang_ref_Reference_);
100 }
101 static void SetClass(Class* klass);
102 static void ResetClass(void);
103 static void VisitRoots(RootCallback* callback, void* arg);
104
Mathieu Chartier8fa2dad2014-03-13 12:22:56 -0700105 private:
Mathieu Chartier308351a2014-06-15 12:39:02 -0700106 // Note: This avoids a read barrier, it should only be used by the GC.
107 HeapReference<Object>* GetReferentReferenceAddr() SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
108 return GetFieldObjectReferenceAddr<kDefaultVerifyFlags>(ReferentOffset());
109 }
110
Mathieu Chartier8fa2dad2014-03-13 12:22:56 -0700111 // Field order required by test "ValidateFieldOrderOfJavaCppUnionClasses".
112 HeapReference<Reference> pending_next_; // Note this is Java volatile:
113 HeapReference<Object> queue_; // Note this is Java volatile:
114 HeapReference<Reference> queue_next_; // Note this is Java volatile:
115 HeapReference<Object> referent_; // Note this is Java volatile:
116
Fred Shih4ee7a662014-07-11 09:59:27 -0700117 static Class* java_lang_ref_Reference_;
118
Mathieu Chartier8fa2dad2014-03-13 12:22:56 -0700119 friend struct art::ReferenceOffsets; // for verifying offset information
Mathieu Chartier308351a2014-06-15 12:39:02 -0700120 friend class gc::ReferenceProcessor;
121 friend class gc::ReferenceQueue;
Mathieu Chartier8fa2dad2014-03-13 12:22:56 -0700122 DISALLOW_IMPLICIT_CONSTRUCTORS(Reference);
123};
124
125// C++ mirror of java.lang.ref.FinalizerReference
126class MANAGED FinalizerReference : public Reference {
127 public:
128 static MemberOffset ZombieOffset() {
129 return OFFSET_OF_OBJECT_MEMBER(FinalizerReference, zombie_);
130 }
131
132 template<bool kTransactionActive>
133 void SetZombie(Object* zombie) SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700134 return SetFieldObjectVolatile<kTransactionActive>(ZombieOffset(), zombie);
Mathieu Chartier8fa2dad2014-03-13 12:22:56 -0700135 }
136 Object* GetZombie() SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700137 return GetFieldObjectVolatile<Object>(ZombieOffset());
Mathieu Chartier8fa2dad2014-03-13 12:22:56 -0700138 }
139
140 private:
141 HeapReference<FinalizerReference> next_;
142 HeapReference<FinalizerReference> prev_;
143 HeapReference<Object> zombie_;
144
145 friend struct art::FinalizerReferenceOffsets; // for verifying offset information
146 DISALLOW_IMPLICIT_CONSTRUCTORS(FinalizerReference);
147};
148
149} // namespace mirror
150} // namespace art
151
152#endif // ART_RUNTIME_MIRROR_REFERENCE_H_