| Ian Rogers | ef7d42f | 2014-01-06 12:55:46 -0800 | [diff] [blame] | 1 | /* |
| 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_OBJECT_REFERENCE_H_ |
| 18 | #define ART_RUNTIME_MIRROR_OBJECT_REFERENCE_H_ |
| 19 | |
| David Sehr | c431b9d | 2018-03-02 12:01:51 -0800 | [diff] [blame] | 20 | #include "base/atomic.h" |
| Andreas Gampe | 7fbc4a5 | 2018-11-28 08:26:47 -0800 | [diff] [blame] | 21 | #include "base/locks.h" // For Locks::mutator_lock_. |
| Andreas Gampe | 09659c2 | 2017-09-18 18:23:32 -0700 | [diff] [blame] | 22 | #include "heap_poisoning.h" |
| Mathieu Chartier | a058fdf | 2016-10-06 15:13:58 -0700 | [diff] [blame] | 23 | #include "obj_ptr.h" |
| Andreas Gampe | 5a0430d | 2019-01-04 14:33:57 -0800 | [diff] [blame] | 24 | #include "runtime_globals.h" |
| Ian Rogers | ef7d42f | 2014-01-06 12:55:46 -0800 | [diff] [blame] | 25 | |
| 26 | namespace art { |
| 27 | namespace mirror { |
| 28 | |
| 29 | class Object; |
| 30 | |
| 31 | // Classes shared with the managed side of the world need to be packed so that they don't have |
| 32 | // extra platform specific padding. |
| 33 | #define MANAGED PACKED(4) |
| 34 | |
| Hans Boehm | cc55e1d | 2017-07-27 15:28:07 -0700 | [diff] [blame] | 35 | template<bool kPoisonReferences, class MirrorType> |
| 36 | class PtrCompression { |
| 37 | public: |
| 38 | // Compress reference to its bit representation. |
| 39 | static uint32_t Compress(MirrorType* mirror_ptr) { |
| 40 | uintptr_t as_bits = reinterpret_cast<uintptr_t>(mirror_ptr); |
| 41 | return static_cast<uint32_t>(kPoisonReferences ? -as_bits : as_bits); |
| 42 | } |
| 43 | |
| 44 | // Uncompress an encoded reference from its bit representation. |
| 45 | static MirrorType* Decompress(uint32_t ref) { |
| 46 | uintptr_t as_bits = kPoisonReferences ? -ref : ref; |
| 47 | return reinterpret_cast<MirrorType*>(as_bits); |
| 48 | } |
| 49 | |
| 50 | // Convert an ObjPtr to a compressed reference. |
| Vladimir Marko | f57f2ed | 2019-03-25 15:55:42 +0000 | [diff] [blame] | 51 | static uint32_t Compress(ObjPtr<MirrorType> ptr) REQUIRES_SHARED(Locks::mutator_lock_); |
| Hans Boehm | cc55e1d | 2017-07-27 15:28:07 -0700 | [diff] [blame] | 52 | }; |
| 53 | |
| Ian Rogers | ef7d42f | 2014-01-06 12:55:46 -0800 | [diff] [blame] | 54 | // Value type representing a reference to a mirror::Object of type MirrorType. |
| 55 | template<bool kPoisonReferences, class MirrorType> |
| 56 | class MANAGED ObjectReference { |
| Hans Boehm | cc55e1d | 2017-07-27 15:28:07 -0700 | [diff] [blame] | 57 | private: |
| 58 | using Compression = PtrCompression<kPoisonReferences, MirrorType>; |
| 59 | |
| Ian Rogers | ef7d42f | 2014-01-06 12:55:46 -0800 | [diff] [blame] | 60 | public: |
| Chris Wailes | 0c61be4 | 2018-09-26 17:27:34 -0700 | [diff] [blame] | 61 | /* |
| 62 | * Returns a pointer to the mirror of the managed object this reference is for. |
| 63 | * |
| 64 | * This does NOT return the current object (which isn't derived from, and |
| 65 | * therefor cannot be a mirror::Object) as a mirror pointer. Instead, this |
| 66 | * returns a pointer to the mirror of the managed object this refers to. |
| 67 | * |
| 68 | * TODO (chriswailes): Rename to GetPtr(). |
| 69 | */ |
| Hans Boehm | cc55e1d | 2017-07-27 15:28:07 -0700 | [diff] [blame] | 70 | MirrorType* AsMirrorPtr() const { |
| 71 | return Compression::Decompress(reference_); |
| Ian Rogers | ef7d42f | 2014-01-06 12:55:46 -0800 | [diff] [blame] | 72 | } |
| 73 | |
| Hans Boehm | cc55e1d | 2017-07-27 15:28:07 -0700 | [diff] [blame] | 74 | void Assign(MirrorType* other) { |
| 75 | reference_ = Compression::Compress(other); |
| Ian Rogers | ef7d42f | 2014-01-06 12:55:46 -0800 | [diff] [blame] | 76 | } |
| 77 | |
| Hans Boehm | cc55e1d | 2017-07-27 15:28:07 -0700 | [diff] [blame] | 78 | void Assign(ObjPtr<MirrorType> ptr) REQUIRES_SHARED(Locks::mutator_lock_); |
| Mathieu Chartier | 1a5337f | 2016-10-13 13:48:23 -0700 | [diff] [blame] | 79 | |
| Ian Rogers | ef7d42f | 2014-01-06 12:55:46 -0800 | [diff] [blame] | 80 | void Clear() { |
| 81 | reference_ = 0; |
| Mathieu Chartier | bb87e0f | 2015-04-03 11:21:55 -0700 | [diff] [blame] | 82 | DCHECK(IsNull()); |
| 83 | } |
| 84 | |
| 85 | bool IsNull() const { |
| 86 | return reference_ == 0; |
| Ian Rogers | ef7d42f | 2014-01-06 12:55:46 -0800 | [diff] [blame] | 87 | } |
| 88 | |
| 89 | uint32_t AsVRegValue() const { |
| 90 | return reference_; |
| 91 | } |
| 92 | |
| Hans Boehm | cc55e1d | 2017-07-27 15:28:07 -0700 | [diff] [blame] | 93 | static ObjectReference<kPoisonReferences, MirrorType> FromMirrorPtr(MirrorType* mirror_ptr) |
| 94 | REQUIRES_SHARED(Locks::mutator_lock_) { |
| 95 | return ObjectReference<kPoisonReferences, MirrorType>(mirror_ptr); |
| 96 | } |
| 97 | |
| Ian Rogers | ef7d42f | 2014-01-06 12:55:46 -0800 | [diff] [blame] | 98 | protected: |
| Hans Boehm | cc55e1d | 2017-07-27 15:28:07 -0700 | [diff] [blame] | 99 | explicit ObjectReference(MirrorType* mirror_ptr) REQUIRES_SHARED(Locks::mutator_lock_) |
| 100 | : reference_(Compression::Compress(mirror_ptr)) { |
| Ian Rogers | ef7d42f | 2014-01-06 12:55:46 -0800 | [diff] [blame] | 101 | } |
| 102 | |
| Ian Rogers | ef7d42f | 2014-01-06 12:55:46 -0800 | [diff] [blame] | 103 | // The encoded reference to a mirror::Object. |
| 104 | uint32_t reference_; |
| 105 | }; |
| 106 | |
| 107 | // References between objects within the managed heap. |
| Hans Boehm | cc55e1d | 2017-07-27 15:28:07 -0700 | [diff] [blame] | 108 | // Similar API to ObjectReference, but not a value type. Supports atomic access. |
| Ian Rogers | ef7d42f | 2014-01-06 12:55:46 -0800 | [diff] [blame] | 109 | template<class MirrorType> |
| Hans Boehm | cc55e1d | 2017-07-27 15:28:07 -0700 | [diff] [blame] | 110 | class MANAGED HeapReference { |
| 111 | private: |
| 112 | using Compression = PtrCompression<kPoisonHeapReferences, MirrorType>; |
| 113 | |
| Ian Rogers | ef7d42f | 2014-01-06 12:55:46 -0800 | [diff] [blame] | 114 | public: |
| Mathieu Chartier | ad8ebb3 | 2017-08-09 20:13:18 -0700 | [diff] [blame] | 115 | HeapReference() REQUIRES_SHARED(Locks::mutator_lock_) : HeapReference(nullptr) {} |
| 116 | |
| Hans Boehm | cc55e1d | 2017-07-27 15:28:07 -0700 | [diff] [blame] | 117 | template <bool kIsVolatile = false> |
| 118 | MirrorType* AsMirrorPtr() const REQUIRES_SHARED(Locks::mutator_lock_) { |
| 119 | return Compression::Decompress( |
| Orion Hodson | 88591fe | 2018-03-06 13:35:43 +0000 | [diff] [blame] | 120 | kIsVolatile ? reference_.load(std::memory_order_seq_cst) : reference_.LoadJavaData()); |
| Hans Boehm | cc55e1d | 2017-07-27 15:28:07 -0700 | [diff] [blame] | 121 | } |
| 122 | |
| 123 | template <bool kIsVolatile = false> |
| 124 | void Assign(MirrorType* other) REQUIRES_SHARED(Locks::mutator_lock_) { |
| 125 | if (kIsVolatile) { |
| Orion Hodson | 88591fe | 2018-03-06 13:35:43 +0000 | [diff] [blame] | 126 | reference_.store(Compression::Compress(other), std::memory_order_seq_cst); |
| Hans Boehm | cc55e1d | 2017-07-27 15:28:07 -0700 | [diff] [blame] | 127 | } else { |
| 128 | reference_.StoreJavaData(Compression::Compress(other)); |
| 129 | } |
| 130 | } |
| 131 | |
| 132 | template <bool kIsVolatile = false> |
| 133 | void Assign(ObjPtr<MirrorType> ptr) REQUIRES_SHARED(Locks::mutator_lock_); |
| 134 | |
| 135 | void Clear() { |
| 136 | reference_.StoreJavaData(0); |
| 137 | DCHECK(IsNull()); |
| 138 | } |
| 139 | |
| 140 | bool IsNull() const { |
| 141 | return reference_.LoadJavaData() == 0; |
| 142 | } |
| 143 | |
| Ian Rogers | ef7d42f | 2014-01-06 12:55:46 -0800 | [diff] [blame] | 144 | static HeapReference<MirrorType> FromMirrorPtr(MirrorType* mirror_ptr) |
| Andreas Gampe | bdf7f1c | 2016-08-30 16:38:47 -0700 | [diff] [blame] | 145 | REQUIRES_SHARED(Locks::mutator_lock_) { |
| Ian Rogers | ef7d42f | 2014-01-06 12:55:46 -0800 | [diff] [blame] | 146 | return HeapReference<MirrorType>(mirror_ptr); |
| 147 | } |
| Mathieu Chartier | a058fdf | 2016-10-06 15:13:58 -0700 | [diff] [blame] | 148 | |
| Hiroshi Yamauchi | 65f5f24 | 2016-12-19 11:44:47 -0800 | [diff] [blame] | 149 | bool CasWeakRelaxed(MirrorType* old_ptr, MirrorType* new_ptr) |
| 150 | REQUIRES_SHARED(Locks::mutator_lock_); |
| 151 | |
| Ian Rogers | ef7d42f | 2014-01-06 12:55:46 -0800 | [diff] [blame] | 152 | private: |
| Chih-Hung Hsieh | a593118 | 2016-09-01 15:08:13 -0700 | [diff] [blame] | 153 | explicit HeapReference(MirrorType* mirror_ptr) REQUIRES_SHARED(Locks::mutator_lock_) |
| Mathieu Chartier | ad8ebb3 | 2017-08-09 20:13:18 -0700 | [diff] [blame] | 154 | : reference_(Compression::Compress(mirror_ptr)) {} |
| Hans Boehm | cc55e1d | 2017-07-27 15:28:07 -0700 | [diff] [blame] | 155 | |
| 156 | // The encoded reference to a mirror::Object. Atomically updateable. |
| 157 | Atomic<uint32_t> reference_; |
| Ian Rogers | ef7d42f | 2014-01-06 12:55:46 -0800 | [diff] [blame] | 158 | }; |
| 159 | |
| Mathieu Chartier | a058fdf | 2016-10-06 15:13:58 -0700 | [diff] [blame] | 160 | static_assert(sizeof(mirror::HeapReference<mirror::Object>) == kHeapReferenceSize, |
| 161 | "heap reference size does not match"); |
| 162 | |
| Mathieu Chartier | d3ed9a3 | 2015-04-10 14:23:35 -0700 | [diff] [blame] | 163 | // Standard compressed reference used in the runtime. Used for StackReference and GC roots. |
| Mathieu Chartier | bb87e0f | 2015-04-03 11:21:55 -0700 | [diff] [blame] | 164 | template<class MirrorType> |
| 165 | class MANAGED CompressedReference : public mirror::ObjectReference<false, MirrorType> { |
| 166 | public: |
| Andreas Gampe | bdf7f1c | 2016-08-30 16:38:47 -0700 | [diff] [blame] | 167 | CompressedReference<MirrorType>() REQUIRES_SHARED(Locks::mutator_lock_) |
| Mathieu Chartier | bb87e0f | 2015-04-03 11:21:55 -0700 | [diff] [blame] | 168 | : mirror::ObjectReference<false, MirrorType>(nullptr) {} |
| 169 | |
| 170 | static CompressedReference<MirrorType> FromMirrorPtr(MirrorType* p) |
| Andreas Gampe | bdf7f1c | 2016-08-30 16:38:47 -0700 | [diff] [blame] | 171 | REQUIRES_SHARED(Locks::mutator_lock_) { |
| Mathieu Chartier | bb87e0f | 2015-04-03 11:21:55 -0700 | [diff] [blame] | 172 | return CompressedReference<MirrorType>(p); |
| 173 | } |
| 174 | |
| 175 | private: |
| Chih-Hung Hsieh | a593118 | 2016-09-01 15:08:13 -0700 | [diff] [blame] | 176 | explicit CompressedReference(MirrorType* p) REQUIRES_SHARED(Locks::mutator_lock_) |
| Mathieu Chartier | bb87e0f | 2015-04-03 11:21:55 -0700 | [diff] [blame] | 177 | : mirror::ObjectReference<false, MirrorType>(p) {} |
| 178 | }; |
| 179 | |
| Ian Rogers | ef7d42f | 2014-01-06 12:55:46 -0800 | [diff] [blame] | 180 | } // namespace mirror |
| 181 | } // namespace art |
| 182 | |
| 183 | #endif // ART_RUNTIME_MIRROR_OBJECT_REFERENCE_H_ |