blob: 4d5f6213c74813e5da35c08139beb5d801ffc6db [file] [log] [blame]
Ian Rogers2dd0e2c2013-01-24 12:42:14 -08001/*
2 * Copyright (C) 2011 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
Brian Carlstromfc0e3212013-07-17 14:40:12 -070017#ifndef ART_RUNTIME_MIRROR_OBJECT_INL_H_
18#define ART_RUNTIME_MIRROR_OBJECT_INL_H_
Ian Rogers2dd0e2c2013-01-24 12:42:14 -080019
20#include "object.h"
21
Brian Carlstromea46f952013-07-30 01:26:50 -070022#include "art_field.h"
23#include "art_method.h"
Ian Rogers2dd0e2c2013-01-24 12:42:14 -080024#include "atomic.h"
Ian Rogers4f6ad8a2013-03-18 15:27:28 -070025#include "array-inl.h"
Ian Rogers2dd0e2c2013-01-24 12:42:14 -080026#include "class.h"
Ian Rogersd9c4fc92013-10-01 19:45:43 -070027#include "lock_word-inl.h"
Ian Rogers05f30572013-02-20 12:13:11 -080028#include "monitor.h"
Mathieu Chartier52e4b432014-06-10 11:22:31 -070029#include "object_array-inl.h"
Hiroshi Yamauchi800ac2d2014-04-02 17:32:54 -070030#include "read_barrier-inl.h"
Ian Rogers2dd0e2c2013-01-24 12:42:14 -080031#include "runtime.h"
Mathieu Chartier8fa2dad2014-03-13 12:22:56 -070032#include "reference.h"
Ian Rogers05f30572013-02-20 12:13:11 -080033#include "throwable.h"
Ian Rogers2dd0e2c2013-01-24 12:42:14 -080034
35namespace art {
36namespace mirror {
37
Mingyao Yang98d1cc82014-05-15 17:02:16 -070038inline uint32_t Object::ClassSize() {
39 uint32_t vtable_entries = kVTableLength;
Fred Shih37f05ef2014-07-16 18:38:08 -070040 return Class::ComputeClassSize(true, vtable_entries, 0, 0, 0, 0, 0);
Mingyao Yang98d1cc82014-05-15 17:02:16 -070041}
42
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -070043template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogersef7d42f2014-01-06 12:55:46 -080044inline Class* Object::GetClass() {
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -070045 return GetFieldObject<Class, kVerifyFlags, kReadBarrierOption>(
Ian Rogersb0fa5dc2014-04-28 16:47:08 -070046 OFFSET_OF_OBJECT_MEMBER(Object, klass_));
Ian Rogers2dd0e2c2013-01-24 12:42:14 -080047}
48
Mathieu Chartier4e305412014-02-19 10:54:44 -080049template<VerifyObjectFlags kVerifyFlags>
Ian Rogers2dd0e2c2013-01-24 12:42:14 -080050inline void Object::SetClass(Class* new_klass) {
Ian Rogersef7d42f2014-01-06 12:55:46 -080051 // new_klass may be NULL prior to class linker initialization.
52 // We don't mark the card as this occurs as part of object allocation. Not all objects have
53 // backing cards, such as large objects.
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +010054 // We use non transactional version since we can't undo this write. We also disable checking as
55 // we may run in transaction mode here.
Mathieu Chartier4e305412014-02-19 10:54:44 -080056 SetFieldObjectWithoutWriteBarrier<false, false,
57 static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis)>(
Ian Rogersb0fa5dc2014-04-28 16:47:08 -070058 OFFSET_OF_OBJECT_MEMBER(Object, klass_), new_klass);
Ian Rogers2dd0e2c2013-01-24 12:42:14 -080059}
60
Mathieu Chartierbbd695c2014-04-16 09:48:48 -070061inline LockWord Object::GetLockWord(bool as_volatile) {
Ian Rogersb0fa5dc2014-04-28 16:47:08 -070062 if (as_volatile) {
63 return LockWord(GetField32Volatile(OFFSET_OF_OBJECT_MEMBER(Object, monitor_)));
64 }
65 return LockWord(GetField32(OFFSET_OF_OBJECT_MEMBER(Object, monitor_)));
Ian Rogersd9c4fc92013-10-01 19:45:43 -070066}
67
Mathieu Chartierbbd695c2014-04-16 09:48:48 -070068inline void Object::SetLockWord(LockWord new_val, bool as_volatile) {
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +010069 // Force use of non-transactional mode and do not check.
Ian Rogersb0fa5dc2014-04-28 16:47:08 -070070 if (as_volatile) {
71 SetField32Volatile<false, false>(OFFSET_OF_OBJECT_MEMBER(Object, monitor_), new_val.GetValue());
72 } else {
73 SetField32<false, false>(OFFSET_OF_OBJECT_MEMBER(Object, monitor_), new_val.GetValue());
74 }
Ian Rogersd9c4fc92013-10-01 19:45:43 -070075}
76
Ian Rogers228602f2014-07-10 02:07:54 -070077inline bool Object::CasLockWordWeakSequentiallyConsistent(LockWord old_val, LockWord new_val) {
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +010078 // Force use of non-transactional mode and do not check.
Ian Rogers228602f2014-07-10 02:07:54 -070079 return CasFieldWeakSequentiallyConsistent32<false, false>(
80 OFFSET_OF_OBJECT_MEMBER(Object, monitor_), old_val.GetValue(), new_val.GetValue());
Ian Rogersd9c4fc92013-10-01 19:45:43 -070081}
82
Hans Boehmd8434432014-07-11 09:56:07 -070083inline bool Object::CasLockWordWeakRelaxed(LockWord old_val, LockWord new_val) {
84 // Force use of non-transactional mode and do not check.
85 return CasFieldWeakRelaxed32<false, false>(
86 OFFSET_OF_OBJECT_MEMBER(Object, monitor_), old_val.GetValue(), new_val.GetValue());
87}
88
Ian Rogersd9c4fc92013-10-01 19:45:43 -070089inline uint32_t Object::GetLockOwnerThreadId() {
90 return Monitor::GetLockOwnerThreadId(this);
Ian Rogers05f30572013-02-20 12:13:11 -080091}
92
Mathieu Chartiere7e8a5f2014-02-14 16:59:41 -080093inline mirror::Object* Object::MonitorEnter(Thread* self) {
94 return Monitor::MonitorEnter(self, this);
Ian Rogers05f30572013-02-20 12:13:11 -080095}
96
97inline bool Object::MonitorExit(Thread* self) {
98 return Monitor::MonitorExit(self, this);
99}
100
101inline void Object::Notify(Thread* self) {
102 Monitor::Notify(self, this);
103}
104
105inline void Object::NotifyAll(Thread* self) {
106 Monitor::NotifyAll(self, this);
107}
108
109inline void Object::Wait(Thread* self) {
110 Monitor::Wait(self, this, 0, 0, true, kWaiting);
111}
112
113inline void Object::Wait(Thread* self, int64_t ms, int32_t ns) {
114 Monitor::Wait(self, this, ms, ns, true, kTimedWaiting);
115}
116
Hiroshi Yamauchi624468c2014-03-31 15:14:47 -0700117inline Object* Object::GetReadBarrierPointer() {
118#ifdef USE_BAKER_OR_BROOKS_READ_BARRIER
119 DCHECK(kUseBakerOrBrooksReadBarrier);
Hiroshi Yamauchi6a154a42014-05-02 14:26:13 -0700120 return GetFieldObject<Object, kVerifyNone, kWithoutReadBarrier>(
121 OFFSET_OF_OBJECT_MEMBER(Object, x_rb_ptr_));
Hiroshi Yamauchi9d04a202014-01-31 13:35:49 -0800122#else
123 LOG(FATAL) << "Unreachable";
124 return nullptr;
125#endif
126}
127
Hiroshi Yamauchi9103c862014-04-22 13:51:07 -0700128inline void Object::SetReadBarrierPointer(Object* rb_ptr) {
Hiroshi Yamauchi624468c2014-03-31 15:14:47 -0700129#ifdef USE_BAKER_OR_BROOKS_READ_BARRIER
130 DCHECK(kUseBakerOrBrooksReadBarrier);
Hiroshi Yamauchi9d04a202014-01-31 13:35:49 -0800131 // We don't mark the card as this occurs as part of object allocation. Not all objects have
132 // backing cards, such as large objects.
133 SetFieldObjectWithoutWriteBarrier<false, false, kVerifyNone>(
Hiroshi Yamauchi6a154a42014-05-02 14:26:13 -0700134 OFFSET_OF_OBJECT_MEMBER(Object, x_rb_ptr_), rb_ptr);
Hiroshi Yamauchi9d04a202014-01-31 13:35:49 -0800135#else
136 LOG(FATAL) << "Unreachable";
137#endif
138}
139
Hiroshi Yamauchi9103c862014-04-22 13:51:07 -0700140inline bool Object::AtomicSetReadBarrierPointer(Object* expected_rb_ptr, Object* rb_ptr) {
141#ifdef USE_BAKER_OR_BROOKS_READ_BARRIER
142 DCHECK(kUseBakerOrBrooksReadBarrier);
143 MemberOffset offset = OFFSET_OF_OBJECT_MEMBER(Object, x_rb_ptr_);
144 byte* raw_addr = reinterpret_cast<byte*>(this) + offset.SizeValue();
Ian Rogers228602f2014-07-10 02:07:54 -0700145 Atomic<uint32_t>* atomic_rb_ptr = reinterpret_cast<Atomic<uint32_t>*>(raw_addr);
Hiroshi Yamauchi9103c862014-04-22 13:51:07 -0700146 HeapReference<Object> expected_ref(HeapReference<Object>::FromMirrorPtr(expected_rb_ptr));
147 HeapReference<Object> new_ref(HeapReference<Object>::FromMirrorPtr(rb_ptr));
Hiroshi Yamauchi9103c862014-04-22 13:51:07 -0700148 do {
Ian Rogers228602f2014-07-10 02:07:54 -0700149 if (UNLIKELY(atomic_rb_ptr->LoadRelaxed() != expected_ref.reference_)) {
Hiroshi Yamauchi9103c862014-04-22 13:51:07 -0700150 // Lost the race.
151 return false;
152 }
Ian Rogers228602f2014-07-10 02:07:54 -0700153 } while (!atomic_rb_ptr->CompareExchangeWeakSequentiallyConsistent(expected_ref.reference_,
154 new_ref.reference_));
155 DCHECK_EQ(new_ref.reference_, atomic_rb_ptr->LoadRelaxed());
Hiroshi Yamauchi9103c862014-04-22 13:51:07 -0700156 return true;
157#else
158 LOG(FATAL) << "Unreachable";
159 return false;
160#endif
161}
162
Hiroshi Yamauchi624468c2014-03-31 15:14:47 -0700163inline void Object::AssertReadBarrierPointer() const {
Hiroshi Yamauchi800ac2d2014-04-02 17:32:54 -0700164 if (kUseBakerReadBarrier) {
165 Object* obj = const_cast<Object*>(this);
166 DCHECK(obj->GetReadBarrierPointer() == nullptr)
167 << "Bad Baker pointer: obj=" << reinterpret_cast<void*>(obj)
168 << " ptr=" << reinterpret_cast<void*>(obj->GetReadBarrierPointer());
169 } else if (kUseBrooksReadBarrier) {
170 Object* obj = const_cast<Object*>(this);
171 DCHECK_EQ(obj, obj->GetReadBarrierPointer())
172 << "Bad Brooks pointer: obj=" << reinterpret_cast<void*>(obj)
173 << " ptr=" << reinterpret_cast<void*>(obj->GetReadBarrierPointer());
174 } else {
175 LOG(FATAL) << "Unreachable";
176 }
Hiroshi Yamauchi9d04a202014-01-31 13:35:49 -0800177}
178
Mathieu Chartier4e305412014-02-19 10:54:44 -0800179template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800180inline bool Object::VerifierInstanceOf(Class* klass) {
Jeff Haoa3faaf42013-09-03 19:07:00 -0700181 DCHECK(klass != NULL);
Mathieu Chartier4e305412014-02-19 10:54:44 -0800182 DCHECK(GetClass<kVerifyFlags>() != NULL);
Jeff Haoa3faaf42013-09-03 19:07:00 -0700183 return klass->IsInterface() || InstanceOf(klass);
184}
185
Mathieu Chartier4e305412014-02-19 10:54:44 -0800186template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800187inline bool Object::InstanceOf(Class* klass) {
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800188 DCHECK(klass != NULL);
Mathieu Chartier4e305412014-02-19 10:54:44 -0800189 DCHECK(GetClass<kVerifyNone>() != NULL);
190 return klass->IsAssignableFrom(GetClass<kVerifyFlags>());
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800191}
192
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700193template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800194inline bool Object::IsClass() {
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700195 Class* java_lang_Class = GetClass<kVerifyFlags, kReadBarrierOption>()->
196 template GetClass<kVerifyFlags, kReadBarrierOption>();
197 return GetClass<static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis),
198 kReadBarrierOption>() == java_lang_Class;
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800199}
200
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700201template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800202inline Class* Object::AsClass() {
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700203 DCHECK((IsClass<kVerifyFlags, kReadBarrierOption>()));
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800204 return down_cast<Class*>(this);
205}
206
Mathieu Chartier4e305412014-02-19 10:54:44 -0800207template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800208inline bool Object::IsObjectArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800209 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
210 return IsArrayInstance<kVerifyFlags>() &&
211 !GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitive();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800212}
213
Mathieu Chartier4e305412014-02-19 10:54:44 -0800214template<class T, VerifyObjectFlags kVerifyFlags>
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800215inline ObjectArray<T>* Object::AsObjectArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800216 DCHECK(IsObjectArray<kVerifyFlags>());
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800217 return down_cast<ObjectArray<T>*>(this);
218}
219
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700220template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800221inline bool Object::IsArrayInstance() {
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700222 return GetClass<kVerifyFlags, kReadBarrierOption>()->
223 template IsArrayClass<kVerifyFlags, kReadBarrierOption>();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800224}
225
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700226template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800227inline bool Object::IsArtField() {
Hiroshi Yamauchi4f1ebc22014-06-25 14:30:41 -0700228 return GetClass<kVerifyFlags, kReadBarrierOption>()->
229 template IsArtFieldClass<kReadBarrierOption>();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800230}
231
Mathieu Chartier4e305412014-02-19 10:54:44 -0800232template<VerifyObjectFlags kVerifyFlags>
233inline ArtField* Object::AsArtField() {
234 DCHECK(IsArtField<kVerifyFlags>());
Brian Carlstromea46f952013-07-30 01:26:50 -0700235 return down_cast<ArtField*>(this);
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800236}
237
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700238template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800239inline bool Object::IsArtMethod() {
Hiroshi Yamauchi4f1ebc22014-06-25 14:30:41 -0700240 return GetClass<kVerifyFlags, kReadBarrierOption>()->
241 template IsArtMethodClass<kReadBarrierOption>();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800242}
243
Mathieu Chartier4e305412014-02-19 10:54:44 -0800244template<VerifyObjectFlags kVerifyFlags>
Brian Carlstromea46f952013-07-30 01:26:50 -0700245inline ArtMethod* Object::AsArtMethod() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800246 DCHECK(IsArtMethod<kVerifyFlags>());
Brian Carlstromea46f952013-07-30 01:26:50 -0700247 return down_cast<ArtMethod*>(this);
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800248}
249
Mathieu Chartier4e305412014-02-19 10:54:44 -0800250template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800251inline bool Object::IsReferenceInstance() {
Fred Shih4ee7a662014-07-11 09:59:27 -0700252 return GetClass<kVerifyFlags>()->IsTypeOfReferenceClass();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800253}
254
Mathieu Chartier4e305412014-02-19 10:54:44 -0800255template<VerifyObjectFlags kVerifyFlags>
Mathieu Chartier8fa2dad2014-03-13 12:22:56 -0700256inline Reference* Object::AsReference() {
257 DCHECK(IsReferenceInstance<kVerifyFlags>());
258 return down_cast<Reference*>(this);
259}
260
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700261template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogers05f30572013-02-20 12:13:11 -0800262inline Array* Object::AsArray() {
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700263 DCHECK((IsArrayInstance<kVerifyFlags, kReadBarrierOption>()));
Ian Rogers05f30572013-02-20 12:13:11 -0800264 return down_cast<Array*>(this);
265}
266
Mathieu Chartier4e305412014-02-19 10:54:44 -0800267template<VerifyObjectFlags kVerifyFlags>
Ian Rogers05f30572013-02-20 12:13:11 -0800268inline BooleanArray* Object::AsBooleanArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800269 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
270 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
271 DCHECK(GetClass<kNewFlags>()->GetComponentType()->IsPrimitiveBoolean());
Ian Rogers05f30572013-02-20 12:13:11 -0800272 return down_cast<BooleanArray*>(this);
273}
274
Mathieu Chartier4e305412014-02-19 10:54:44 -0800275template<VerifyObjectFlags kVerifyFlags>
Ian Rogers05f30572013-02-20 12:13:11 -0800276inline ByteArray* Object::AsByteArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800277 static const VerifyObjectFlags kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
278 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
279 DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveByte());
Ian Rogers05f30572013-02-20 12:13:11 -0800280 return down_cast<ByteArray*>(this);
281}
282
Mathieu Chartier4e305412014-02-19 10:54:44 -0800283template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800284inline ByteArray* Object::AsByteSizedArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800285 constexpr VerifyObjectFlags kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
286 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
287 DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveByte() ||
288 GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveBoolean());
Ian Rogersef7d42f2014-01-06 12:55:46 -0800289 return down_cast<ByteArray*>(this);
290}
291
Mathieu Chartier4e305412014-02-19 10:54:44 -0800292template<VerifyObjectFlags kVerifyFlags>
Ian Rogers05f30572013-02-20 12:13:11 -0800293inline CharArray* Object::AsCharArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800294 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
295 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
296 DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveChar());
Ian Rogers05f30572013-02-20 12:13:11 -0800297 return down_cast<CharArray*>(this);
298}
299
Mathieu Chartier4e305412014-02-19 10:54:44 -0800300template<VerifyObjectFlags kVerifyFlags>
Ian Rogers05f30572013-02-20 12:13:11 -0800301inline ShortArray* Object::AsShortArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800302 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
303 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
304 DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveShort());
Ian Rogers05f30572013-02-20 12:13:11 -0800305 return down_cast<ShortArray*>(this);
306}
307
Mathieu Chartier4e305412014-02-19 10:54:44 -0800308template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800309inline ShortArray* Object::AsShortSizedArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800310 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
311 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
312 DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveShort() ||
313 GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveChar());
Ian Rogersef7d42f2014-01-06 12:55:46 -0800314 return down_cast<ShortArray*>(this);
315}
316
Mathieu Chartier4e305412014-02-19 10:54:44 -0800317template<VerifyObjectFlags kVerifyFlags>
Ian Rogers05f30572013-02-20 12:13:11 -0800318inline IntArray* Object::AsIntArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800319 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
320 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
321 DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveInt() ||
322 GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveFloat());
Ian Rogers05f30572013-02-20 12:13:11 -0800323 return down_cast<IntArray*>(this);
324}
325
Mathieu Chartier4e305412014-02-19 10:54:44 -0800326template<VerifyObjectFlags kVerifyFlags>
Ian Rogers05f30572013-02-20 12:13:11 -0800327inline LongArray* Object::AsLongArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800328 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
329 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
330 DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveLong() ||
331 GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveDouble());
Ian Rogers05f30572013-02-20 12:13:11 -0800332 return down_cast<LongArray*>(this);
333}
334
Mathieu Chartier4e305412014-02-19 10:54:44 -0800335template<VerifyObjectFlags kVerifyFlags>
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100336inline FloatArray* Object::AsFloatArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800337 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
338 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
339 DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveFloat());
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100340 return down_cast<FloatArray*>(this);
341}
342
Mathieu Chartier4e305412014-02-19 10:54:44 -0800343template<VerifyObjectFlags kVerifyFlags>
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100344inline DoubleArray* Object::AsDoubleArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800345 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
346 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
347 DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveDouble());
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100348 return down_cast<DoubleArray*>(this);
349}
350
Mathieu Chartier4e305412014-02-19 10:54:44 -0800351template<VerifyObjectFlags kVerifyFlags>
Ian Rogers05f30572013-02-20 12:13:11 -0800352inline String* Object::AsString() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800353 DCHECK(GetClass<kVerifyFlags>()->IsStringClass());
Ian Rogers05f30572013-02-20 12:13:11 -0800354 return down_cast<String*>(this);
355}
356
Mathieu Chartier4e305412014-02-19 10:54:44 -0800357template<VerifyObjectFlags kVerifyFlags>
Ian Rogers05f30572013-02-20 12:13:11 -0800358inline Throwable* Object::AsThrowable() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800359 DCHECK(GetClass<kVerifyFlags>()->IsThrowableClass());
Ian Rogers05f30572013-02-20 12:13:11 -0800360 return down_cast<Throwable*>(this);
361}
362
Mathieu Chartier4e305412014-02-19 10:54:44 -0800363template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800364inline bool Object::IsWeakReferenceInstance() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800365 return GetClass<kVerifyFlags>()->IsWeakReferenceClass();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800366}
367
Mathieu Chartier4e305412014-02-19 10:54:44 -0800368template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800369inline bool Object::IsSoftReferenceInstance() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800370 return GetClass<kVerifyFlags>()->IsSoftReferenceClass();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800371}
372
Mathieu Chartier4e305412014-02-19 10:54:44 -0800373template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800374inline bool Object::IsFinalizerReferenceInstance() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800375 return GetClass<kVerifyFlags>()->IsFinalizerReferenceClass();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800376}
377
Mathieu Chartier4e305412014-02-19 10:54:44 -0800378template<VerifyObjectFlags kVerifyFlags>
Mathieu Chartier8fa2dad2014-03-13 12:22:56 -0700379inline FinalizerReference* Object::AsFinalizerReference() {
380 DCHECK(IsFinalizerReferenceInstance<kVerifyFlags>());
381 return down_cast<FinalizerReference*>(this);
382}
383
384template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800385inline bool Object::IsPhantomReferenceInstance() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800386 return GetClass<kVerifyFlags>()->IsPhantomReferenceClass();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800387}
388
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700389template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800390inline size_t Object::SizeOf() {
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800391 size_t result;
Mathieu Chartier4e305412014-02-19 10:54:44 -0800392 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700393 if (IsArrayInstance<kVerifyFlags, kReadBarrierOption>()) {
394 result = AsArray<kNewFlags, kReadBarrierOption>()->
395 template SizeOf<kNewFlags, kReadBarrierOption>();
396 } else if (IsClass<kNewFlags, kReadBarrierOption>()) {
397 result = AsClass<kNewFlags, kReadBarrierOption>()->
398 template SizeOf<kNewFlags, kReadBarrierOption>();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800399 } else {
Hiroshi Yamauchi25023c72014-05-09 11:45:53 -0700400 result = GetClass<kNewFlags, kReadBarrierOption>()->
401 template GetObjectSize<kNewFlags, kReadBarrierOption>();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800402 }
Hiroshi Yamauchi9103c862014-04-22 13:51:07 -0700403 DCHECK_GE(result, sizeof(Object))
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700404 << " class=" << PrettyTypeOf(GetClass<kNewFlags, kReadBarrierOption>());
405 DCHECK(!(IsArtField<kNewFlags, kReadBarrierOption>()) || result == sizeof(ArtField));
406 DCHECK(!(IsArtMethod<kNewFlags, kReadBarrierOption>()) || result == sizeof(ArtMethod));
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800407 return result;
408}
409
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700410template<VerifyObjectFlags kVerifyFlags, bool kIsVolatile>
Fred Shih37f05ef2014-07-16 18:38:08 -0700411inline uint8_t Object::GetFieldBoolean(MemberOffset field_offset) {
412 if (kVerifyFlags & kVerifyThis) {
413 VerifyObject(this);
414 }
415 return GetField<uint8_t, kIsVolatile>(field_offset);
416}
417
418template<VerifyObjectFlags kVerifyFlags, bool kIsVolatile>
419inline int8_t Object::GetFieldByte(MemberOffset field_offset) {
420 if (kVerifyFlags & kVerifyThis) {
421 VerifyObject(this);
422 }
423 return GetField<int8_t, kIsVolatile>(field_offset);
424}
425
426template<VerifyObjectFlags kVerifyFlags>
427inline uint8_t Object::GetFieldBooleanVolatile(MemberOffset field_offset) {
428 return GetFieldBoolean<kVerifyFlags, true>(field_offset);
429}
430
431template<VerifyObjectFlags kVerifyFlags>
432inline int8_t Object::GetFieldByteVolatile(MemberOffset field_offset) {
433 return GetFieldByte<kVerifyFlags, true>(field_offset);
434}
435
436template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags,
437 bool kIsVolatile>
438inline void Object::SetFieldBoolean(MemberOffset field_offset, uint8_t new_value)
439 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
440 if (kCheckTransaction) {
441 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
442 }
443 if (kTransactionActive) {
444 Runtime::Current()->RecordWriteFieldBoolean(this, field_offset,
445 GetFieldBoolean<kVerifyFlags, kIsVolatile>(field_offset),
446 kIsVolatile);
447 }
448 if (kVerifyFlags & kVerifyThis) {
449 VerifyObject(this);
450 }
451 SetField<uint8_t, kIsVolatile>(field_offset, new_value);
452}
453
454template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags,
455 bool kIsVolatile>
456inline void Object::SetFieldByte(MemberOffset field_offset, int8_t new_value)
457 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
458 if (kCheckTransaction) {
459 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
460 }
461 if (kTransactionActive) {
462 Runtime::Current()->RecordWriteFieldByte(this, field_offset,
463 GetFieldByte<kVerifyFlags, kIsVolatile>(field_offset),
464 kIsVolatile);
465 }
466 if (kVerifyFlags & kVerifyThis) {
467 VerifyObject(this);
468 }
469 SetField<int8_t, kIsVolatile>(field_offset, new_value);
470}
471
472template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
473inline void Object::SetFieldBooleanVolatile(MemberOffset field_offset, uint8_t new_value) {
474 return SetFieldBoolean<kTransactionActive, kCheckTransaction, kVerifyFlags, true>(
475 field_offset, new_value);
476}
477
478template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
479inline void Object::SetFieldByteVolatile(MemberOffset field_offset, int8_t new_value) {
480 return SetFieldByte<kTransactionActive, kCheckTransaction, kVerifyFlags, true>(
481 field_offset, new_value);
482}
483
484template<VerifyObjectFlags kVerifyFlags, bool kIsVolatile>
485inline uint16_t Object::GetFieldChar(MemberOffset field_offset) {
486 if (kVerifyFlags & kVerifyThis) {
487 VerifyObject(this);
488 }
489 return GetField<uint16_t, kIsVolatile>(field_offset);
490}
491
492template<VerifyObjectFlags kVerifyFlags, bool kIsVolatile>
493inline int16_t Object::GetFieldShort(MemberOffset field_offset) {
494 if (kVerifyFlags & kVerifyThis) {
495 VerifyObject(this);
496 }
497 return GetField<int16_t, kIsVolatile>(field_offset);
498}
499
500template<VerifyObjectFlags kVerifyFlags>
501inline uint16_t Object::GetFieldCharVolatile(MemberOffset field_offset) {
502 return GetFieldChar<kVerifyFlags, true>(field_offset);
503}
504
505template<VerifyObjectFlags kVerifyFlags>
506inline int16_t Object::GetFieldShortVolatile(MemberOffset field_offset) {
507 return GetFieldShort<kVerifyFlags, true>(field_offset);
508}
509
510template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags,
511 bool kIsVolatile>
512inline void Object::SetFieldChar(MemberOffset field_offset, uint16_t new_value) {
513 if (kCheckTransaction) {
514 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
515 }
516 if (kTransactionActive) {
517 Runtime::Current()->RecordWriteFieldChar(this, field_offset,
518 GetFieldChar<kVerifyFlags, kIsVolatile>(field_offset),
519 kIsVolatile);
520 }
521 if (kVerifyFlags & kVerifyThis) {
522 VerifyObject(this);
523 }
524 SetField<uint16_t, kIsVolatile>(field_offset, new_value);
525}
526
527template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags,
528 bool kIsVolatile>
529inline void Object::SetFieldShort(MemberOffset field_offset, int16_t new_value) {
530 if (kCheckTransaction) {
531 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
532 }
533 if (kTransactionActive) {
534 Runtime::Current()->RecordWriteFieldChar(this, field_offset,
535 GetFieldShort<kVerifyFlags, kIsVolatile>(field_offset),
536 kIsVolatile);
537 }
538 if (kVerifyFlags & kVerifyThis) {
539 VerifyObject(this);
540 }
541 SetField<int16_t, kIsVolatile>(field_offset, new_value);
542}
543
544template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
545inline void Object::SetFieldCharVolatile(MemberOffset field_offset, uint16_t new_value) {
546 return SetFieldChar<kTransactionActive, kCheckTransaction, kVerifyFlags, true>(
547 field_offset, new_value);
548}
549
550template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
551inline void Object::SetFieldShortVolatile(MemberOffset field_offset, int16_t new_value) {
552 return SetFieldShort<kTransactionActive, kCheckTransaction, kVerifyFlags, true>(
553 field_offset, new_value);
554}
555
556template<VerifyObjectFlags kVerifyFlags, bool kIsVolatile>
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700557inline int32_t Object::GetField32(MemberOffset field_offset) {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800558 if (kVerifyFlags & kVerifyThis) {
559 VerifyObject(this);
560 }
Fred Shih37f05ef2014-07-16 18:38:08 -0700561 return GetField<int32_t, kIsVolatile>(field_offset);
Ian Rogersb122a4b2013-11-19 18:00:50 -0800562}
563
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700564template<VerifyObjectFlags kVerifyFlags>
565inline int32_t Object::GetField32Volatile(MemberOffset field_offset) {
566 return GetField32<kVerifyFlags, true>(field_offset);
567}
568
569template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags,
570 bool kIsVolatile>
571inline void Object::SetField32(MemberOffset field_offset, int32_t new_value) {
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100572 if (kCheckTransaction) {
573 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
574 }
575 if (kTransactionActive) {
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700576 Runtime::Current()->RecordWriteField32(this, field_offset,
577 GetField32<kVerifyFlags, kIsVolatile>(field_offset),
578 kIsVolatile);
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100579 }
Mathieu Chartier4e305412014-02-19 10:54:44 -0800580 if (kVerifyFlags & kVerifyThis) {
Ian Rogersb122a4b2013-11-19 18:00:50 -0800581 VerifyObject(this);
582 }
Fred Shih37f05ef2014-07-16 18:38:08 -0700583 SetField<int32_t, kIsVolatile>(field_offset, new_value);
Ian Rogersb122a4b2013-11-19 18:00:50 -0800584}
585
Mathieu Chartier4e305412014-02-19 10:54:44 -0800586template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700587inline void Object::SetField32Volatile(MemberOffset field_offset, int32_t new_value) {
588 SetField32<kTransactionActive, kCheckTransaction, kVerifyFlags, true>(field_offset, new_value);
589}
590
Hans Boehmd8434432014-07-11 09:56:07 -0700591// TODO: Pass memory_order_ and strong/weak as arguments to avoid code duplication?
592
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700593template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
Ian Rogers228602f2014-07-10 02:07:54 -0700594inline bool Object::CasFieldWeakSequentiallyConsistent32(MemberOffset field_offset,
595 int32_t old_value, int32_t new_value) {
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100596 if (kCheckTransaction) {
597 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
598 }
599 if (kTransactionActive) {
600 Runtime::Current()->RecordWriteField32(this, field_offset, old_value, true);
601 }
Mathieu Chartier4e305412014-02-19 10:54:44 -0800602 if (kVerifyFlags & kVerifyThis) {
603 VerifyObject(this);
604 }
Ian Rogersd9c4fc92013-10-01 19:45:43 -0700605 byte* raw_addr = reinterpret_cast<byte*>(this) + field_offset.Int32Value();
Ian Rogers228602f2014-07-10 02:07:54 -0700606 AtomicInteger* atomic_addr = reinterpret_cast<AtomicInteger*>(raw_addr);
Hans Boehm30359612014-05-21 17:46:23 -0700607
Ian Rogers228602f2014-07-10 02:07:54 -0700608 return atomic_addr->CompareExchangeWeakSequentiallyConsistent(old_value, new_value);
Ian Rogersd9c4fc92013-10-01 19:45:43 -0700609}
610
Hans Boehmd8434432014-07-11 09:56:07 -0700611template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
612inline bool Object::CasFieldWeakRelaxed32(MemberOffset field_offset,
613 int32_t old_value, int32_t new_value) {
614 if (kCheckTransaction) {
615 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
616 }
617 if (kTransactionActive) {
618 Runtime::Current()->RecordWriteField32(this, field_offset, old_value, true);
619 }
620 if (kVerifyFlags & kVerifyThis) {
621 VerifyObject(this);
622 }
623 byte* raw_addr = reinterpret_cast<byte*>(this) + field_offset.Int32Value();
624 AtomicInteger* atomic_addr = reinterpret_cast<AtomicInteger*>(raw_addr);
625
626 return atomic_addr->CompareExchangeWeakRelaxed(old_value, new_value);
627}
628
629template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
630inline bool Object::CasFieldStrongSequentiallyConsistent32(MemberOffset field_offset,
631 int32_t old_value, int32_t new_value) {
632 if (kCheckTransaction) {
633 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
634 }
635 if (kTransactionActive) {
636 Runtime::Current()->RecordWriteField32(this, field_offset, old_value, true);
637 }
638 if (kVerifyFlags & kVerifyThis) {
639 VerifyObject(this);
640 }
641 byte* raw_addr = reinterpret_cast<byte*>(this) + field_offset.Int32Value();
642 AtomicInteger* atomic_addr = reinterpret_cast<AtomicInteger*>(raw_addr);
643
644 return atomic_addr->CompareExchangeStrongSequentiallyConsistent(old_value, new_value);
645}
646
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700647template<VerifyObjectFlags kVerifyFlags, bool kIsVolatile>
648inline int64_t Object::GetField64(MemberOffset field_offset) {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800649 if (kVerifyFlags & kVerifyThis) {
650 VerifyObject(this);
651 }
Fred Shih37f05ef2014-07-16 18:38:08 -0700652 return GetField<int64_t, kIsVolatile>(field_offset);
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800653}
654
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700655template<VerifyObjectFlags kVerifyFlags>
656inline int64_t Object::GetField64Volatile(MemberOffset field_offset) {
657 return GetField64<kVerifyFlags, true>(field_offset);
658}
659
660template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags,
661 bool kIsVolatile>
662inline void Object::SetField64(MemberOffset field_offset, int64_t new_value) {
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100663 if (kCheckTransaction) {
664 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
665 }
666 if (kTransactionActive) {
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700667 Runtime::Current()->RecordWriteField64(this, field_offset,
668 GetField64<kVerifyFlags, kIsVolatile>(field_offset),
669 kIsVolatile);
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100670 }
Mathieu Chartier4e305412014-02-19 10:54:44 -0800671 if (kVerifyFlags & kVerifyThis) {
Ian Rogersef7d42f2014-01-06 12:55:46 -0800672 VerifyObject(this);
673 }
Fred Shih37f05ef2014-07-16 18:38:08 -0700674 SetField<int64_t, kIsVolatile>(field_offset, new_value);
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800675}
676
Mathieu Chartier4e305412014-02-19 10:54:44 -0800677template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700678inline void Object::SetField64Volatile(MemberOffset field_offset, int64_t new_value) {
679 return SetField64<kTransactionActive, kCheckTransaction, kVerifyFlags, true>(field_offset,
680 new_value);
681}
682
Fred Shih37f05ef2014-07-16 18:38:08 -0700683template<typename kSize, bool kIsVolatile>
684inline void Object::SetField(MemberOffset field_offset, kSize new_value) {
685 byte* raw_addr = reinterpret_cast<byte*>(this) + field_offset.Int32Value();
686 kSize* addr = reinterpret_cast<kSize*>(raw_addr);
687 if (kIsVolatile) {
688 reinterpret_cast<Atomic<kSize>*>(addr)->StoreSequentiallyConsistent(new_value);
689 } else {
690 reinterpret_cast<Atomic<kSize>*>(addr)->StoreJavaData(new_value);
691 }
692}
693
694template<typename kSize, bool kIsVolatile>
695inline kSize Object::GetField(MemberOffset field_offset) {
696 const byte* raw_addr = reinterpret_cast<const byte*>(this) + field_offset.Int32Value();
697 const kSize* addr = reinterpret_cast<const kSize*>(raw_addr);
698 if (kIsVolatile) {
699 return reinterpret_cast<const Atomic<kSize>*>(addr)->LoadSequentiallyConsistent();
700 } else {
701 return reinterpret_cast<const Atomic<kSize>*>(addr)->LoadJavaData();
702 }
703}
704
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700705template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
Ian Rogers228602f2014-07-10 02:07:54 -0700706inline bool Object::CasFieldWeakSequentiallyConsistent64(MemberOffset field_offset,
707 int64_t old_value, int64_t new_value) {
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100708 if (kCheckTransaction) {
709 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
710 }
711 if (kTransactionActive) {
712 Runtime::Current()->RecordWriteField64(this, field_offset, old_value, true);
713 }
Mathieu Chartier4e305412014-02-19 10:54:44 -0800714 if (kVerifyFlags & kVerifyThis) {
715 VerifyObject(this);
716 }
Ian Rogersef7d42f2014-01-06 12:55:46 -0800717 byte* raw_addr = reinterpret_cast<byte*>(this) + field_offset.Int32Value();
Ian Rogers228602f2014-07-10 02:07:54 -0700718 Atomic<int64_t>* atomic_addr = reinterpret_cast<Atomic<int64_t>*>(raw_addr);
719 return atomic_addr->CompareExchangeWeakSequentiallyConsistent(old_value, new_value);
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800720}
721
Hans Boehmd8434432014-07-11 09:56:07 -0700722template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
723inline bool Object::CasFieldStrongSequentiallyConsistent64(MemberOffset field_offset,
724 int64_t old_value, int64_t new_value) {
725 if (kCheckTransaction) {
726 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
727 }
728 if (kTransactionActive) {
729 Runtime::Current()->RecordWriteField64(this, field_offset, old_value, true);
730 }
731 if (kVerifyFlags & kVerifyThis) {
732 VerifyObject(this);
733 }
734 byte* raw_addr = reinterpret_cast<byte*>(this) + field_offset.Int32Value();
735 Atomic<int64_t>* atomic_addr = reinterpret_cast<Atomic<int64_t>*>(raw_addr);
736 return atomic_addr->CompareExchangeStrongSequentiallyConsistent(old_value, new_value);
737}
738
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700739template<class T, VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption,
740 bool kIsVolatile>
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700741inline T* Object::GetFieldObject(MemberOffset field_offset) {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800742 if (kVerifyFlags & kVerifyThis) {
743 VerifyObject(this);
744 }
Ian Rogersef7d42f2014-01-06 12:55:46 -0800745 byte* raw_addr = reinterpret_cast<byte*>(this) + field_offset.Int32Value();
746 HeapReference<T>* objref_addr = reinterpret_cast<HeapReference<T>*>(raw_addr);
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700747 T* result = ReadBarrier::Barrier<T, kReadBarrierOption>(this, field_offset, objref_addr);
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700748 if (kIsVolatile) {
Hans Boehm30359612014-05-21 17:46:23 -0700749 // TODO: Refactor to use a SequentiallyConsistent load instead.
750 QuasiAtomic::ThreadFenceAcquire(); // Ensure visibility of operations preceding store.
Ian Rogersef7d42f2014-01-06 12:55:46 -0800751 }
Mathieu Chartier4e305412014-02-19 10:54:44 -0800752 if (kVerifyFlags & kVerifyReads) {
753 VerifyObject(result);
754 }
Ian Rogersef7d42f2014-01-06 12:55:46 -0800755 return result;
756}
757
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700758template<class T, VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700759inline T* Object::GetFieldObjectVolatile(MemberOffset field_offset) {
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700760 return GetFieldObject<T, kVerifyFlags, kReadBarrierOption, true>(field_offset);
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700761}
762
763template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags,
764 bool kIsVolatile>
765inline void Object::SetFieldObjectWithoutWriteBarrier(MemberOffset field_offset,
766 Object* new_value) {
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100767 if (kCheckTransaction) {
768 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
769 }
770 if (kTransactionActive) {
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700771 mirror::Object* obj;
772 if (kIsVolatile) {
773 obj = GetFieldObjectVolatile<Object>(field_offset);
774 } else {
775 obj = GetFieldObject<Object>(field_offset);
776 }
777 Runtime::Current()->RecordWriteFieldReference(this, field_offset, obj, true);
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100778 }
Mathieu Chartier4e305412014-02-19 10:54:44 -0800779 if (kVerifyFlags & kVerifyThis) {
Ian Rogersef7d42f2014-01-06 12:55:46 -0800780 VerifyObject(this);
781 }
Mathieu Chartier4e305412014-02-19 10:54:44 -0800782 if (kVerifyFlags & kVerifyWrites) {
783 VerifyObject(new_value);
784 }
Ian Rogersef7d42f2014-01-06 12:55:46 -0800785 byte* raw_addr = reinterpret_cast<byte*>(this) + field_offset.Int32Value();
786 HeapReference<Object>* objref_addr = reinterpret_cast<HeapReference<Object>*>(raw_addr);
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700787 if (kIsVolatile) {
Hans Boehm30359612014-05-21 17:46:23 -0700788 // TODO: Refactor to use a SequentiallyConsistent store instead.
789 QuasiAtomic::ThreadFenceRelease(); // Ensure that prior accesses are visible before store.
Ian Rogersef7d42f2014-01-06 12:55:46 -0800790 objref_addr->Assign(new_value);
Hans Boehm30359612014-05-21 17:46:23 -0700791 QuasiAtomic::ThreadFenceSequentiallyConsistent();
792 // Ensure this store occurs before any volatile loads.
Ian Rogersef7d42f2014-01-06 12:55:46 -0800793 } else {
794 objref_addr->Assign(new_value);
795 }
796}
797
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700798template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags,
799 bool kIsVolatile>
800inline void Object::SetFieldObject(MemberOffset field_offset, Object* new_value) {
801 SetFieldObjectWithoutWriteBarrier<kTransactionActive, kCheckTransaction, kVerifyFlags,
802 kIsVolatile>(field_offset, new_value);
Ian Rogersef7d42f2014-01-06 12:55:46 -0800803 if (new_value != nullptr) {
Ian Rogersef7d42f2014-01-06 12:55:46 -0800804 Runtime::Current()->GetHeap()->WriteBarrierField(this, field_offset, new_value);
Mathieu Chartier61c5ebc2014-06-05 17:42:53 -0700805 // TODO: Check field assignment could theoretically cause thread suspension, TODO: fix this.
806 CheckFieldAssignment(field_offset, new_value);
Ian Rogersef7d42f2014-01-06 12:55:46 -0800807 }
808}
809
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700810template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
811inline void Object::SetFieldObjectVolatile(MemberOffset field_offset, Object* new_value) {
812 SetFieldObject<kTransactionActive, kCheckTransaction, kVerifyFlags, true>(field_offset,
813 new_value);
814}
815
Mathieu Chartier4e305412014-02-19 10:54:44 -0800816template <VerifyObjectFlags kVerifyFlags>
817inline HeapReference<Object>* Object::GetFieldObjectReferenceAddr(MemberOffset field_offset) {
818 if (kVerifyFlags & kVerifyThis) {
819 VerifyObject(this);
820 }
821 return reinterpret_cast<HeapReference<Object>*>(reinterpret_cast<byte*>(this) +
822 field_offset.Int32Value());
823}
824
825template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
Ian Rogers228602f2014-07-10 02:07:54 -0700826inline bool Object::CasFieldWeakSequentiallyConsistentObject(MemberOffset field_offset,
827 Object* old_value, Object* new_value) {
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100828 if (kCheckTransaction) {
829 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
830 }
Mathieu Chartier4e305412014-02-19 10:54:44 -0800831 if (kVerifyFlags & kVerifyThis) {
832 VerifyObject(this);
833 }
834 if (kVerifyFlags & kVerifyWrites) {
835 VerifyObject(new_value);
836 }
837 if (kVerifyFlags & kVerifyReads) {
838 VerifyObject(old_value);
839 }
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100840 if (kTransactionActive) {
841 Runtime::Current()->RecordWriteFieldReference(this, field_offset, old_value, true);
842 }
Ian Rogersef7d42f2014-01-06 12:55:46 -0800843 HeapReference<Object> old_ref(HeapReference<Object>::FromMirrorPtr(old_value));
844 HeapReference<Object> new_ref(HeapReference<Object>::FromMirrorPtr(new_value));
Ian Rogers228602f2014-07-10 02:07:54 -0700845 byte* raw_addr = reinterpret_cast<byte*>(this) + field_offset.Int32Value();
846 Atomic<uint32_t>* atomic_addr = reinterpret_cast<Atomic<uint32_t>*>(raw_addr);
847
848 bool success = atomic_addr->CompareExchangeWeakSequentiallyConsistent(old_ref.reference_,
849 new_ref.reference_);
850
Ian Rogersef7d42f2014-01-06 12:55:46 -0800851 if (success) {
852 Runtime::Current()->GetHeap()->WriteBarrierField(this, field_offset, new_value);
853 }
854 return success;
855}
856
Hans Boehmd8434432014-07-11 09:56:07 -0700857template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
858inline bool Object::CasFieldStrongSequentiallyConsistentObject(MemberOffset field_offset,
859 Object* old_value, Object* new_value) {
860 if (kCheckTransaction) {
861 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
862 }
863 if (kVerifyFlags & kVerifyThis) {
864 VerifyObject(this);
865 }
866 if (kVerifyFlags & kVerifyWrites) {
867 VerifyObject(new_value);
868 }
869 if (kVerifyFlags & kVerifyReads) {
870 VerifyObject(old_value);
871 }
872 if (kTransactionActive) {
873 Runtime::Current()->RecordWriteFieldReference(this, field_offset, old_value, true);
874 }
875 HeapReference<Object> old_ref(HeapReference<Object>::FromMirrorPtr(old_value));
876 HeapReference<Object> new_ref(HeapReference<Object>::FromMirrorPtr(new_value));
877 byte* raw_addr = reinterpret_cast<byte*>(this) + field_offset.Int32Value();
878 Atomic<uint32_t>* atomic_addr = reinterpret_cast<Atomic<uint32_t>*>(raw_addr);
879
880 bool success = atomic_addr->CompareExchangeStrongSequentiallyConsistent(old_ref.reference_,
881 new_ref.reference_);
882
883 if (success) {
884 Runtime::Current()->GetHeap()->WriteBarrierField(this, field_offset, new_value);
885 }
886 return success;
887}
888
Mathieu Chartier407f7022014-02-18 14:37:05 -0800889template<bool kVisitClass, bool kIsStatic, typename Visitor>
890inline void Object::VisitFieldsReferences(uint32_t ref_offsets, const Visitor& visitor) {
Ian Rogerscdc1aaf2014-10-09 13:21:38 -0700891 if (!kIsStatic && (ref_offsets != mirror::Class::kClassWalkSuper)) {
892 // Instance fields and not the slow-path.
893 if (kVisitClass) {
894 visitor(this, ClassOffset(), kIsStatic);
Mathieu Chartier407f7022014-02-18 14:37:05 -0800895 }
Ian Rogerscdc1aaf2014-10-09 13:21:38 -0700896 uint32_t field_offset = mirror::kObjectHeaderSize;
Mathieu Chartier407f7022014-02-18 14:37:05 -0800897 while (ref_offsets != 0) {
Ian Rogerscdc1aaf2014-10-09 13:21:38 -0700898 if ((ref_offsets & 1) != 0) {
899 visitor(this, MemberOffset(field_offset), kIsStatic);
900 }
901 ref_offsets >>= 1;
902 field_offset += sizeof(mirror::HeapReference<mirror::Object>);
Mathieu Chartier407f7022014-02-18 14:37:05 -0800903 }
904 } else {
Mingyao Yangfaff0f02014-09-10 12:03:22 -0700905 // There is no reference offset bitmap. In the non-static case, walk up the class
Mathieu Chartier407f7022014-02-18 14:37:05 -0800906 // inheritance hierarchy and find reference offsets the hard way. In the static case, just
907 // consider this class.
908 for (mirror::Class* klass = kIsStatic ? AsClass() : GetClass(); klass != nullptr;
909 klass = kIsStatic ? nullptr : klass->GetSuperClass()) {
910 size_t num_reference_fields =
911 kIsStatic ? klass->NumReferenceStaticFields() : klass->NumReferenceInstanceFields();
912 for (size_t i = 0; i < num_reference_fields; ++i) {
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700913 mirror::ArtField* field = kIsStatic ? klass->GetStaticField(i) : klass->GetInstanceField(i);
Mathieu Chartier407f7022014-02-18 14:37:05 -0800914 MemberOffset field_offset = field->GetOffset();
915 // TODO: Do a simpler check?
Mathieu Chartier52e4b432014-06-10 11:22:31 -0700916 if (kVisitClass || field_offset.Uint32Value() != ClassOffset().Uint32Value()) {
917 visitor(this, field_offset, kIsStatic);
Mathieu Chartier407f7022014-02-18 14:37:05 -0800918 }
Mathieu Chartier407f7022014-02-18 14:37:05 -0800919 }
920 }
921 }
922}
923
924template<bool kVisitClass, typename Visitor>
925inline void Object::VisitInstanceFieldsReferences(mirror::Class* klass, const Visitor& visitor) {
926 VisitFieldsReferences<kVisitClass, false>(
927 klass->GetReferenceInstanceOffsets<kVerifyNone>(), visitor);
928}
929
930template<bool kVisitClass, typename Visitor>
931inline void Object::VisitStaticFieldsReferences(mirror::Class* klass, const Visitor& visitor) {
Mingyao Yang98d1cc82014-05-15 17:02:16 -0700932 DCHECK(!klass->IsTemp());
Mingyao Yangfaff0f02014-09-10 12:03:22 -0700933 klass->VisitFieldsReferences<kVisitClass, true>(0, visitor);
Mathieu Chartier407f7022014-02-18 14:37:05 -0800934}
935
936template <const bool kVisitClass, VerifyObjectFlags kVerifyFlags, typename Visitor,
937 typename JavaLangRefVisitor>
938inline void Object::VisitReferences(const Visitor& visitor,
939 const JavaLangRefVisitor& ref_visitor) {
940 mirror::Class* klass = GetClass<kVerifyFlags>();
Mathieu Chartier52e4b432014-06-10 11:22:31 -0700941 if (klass == Class::GetJavaLangClass()) {
942 AsClass<kVerifyNone>()->VisitReferences<kVisitClass>(klass, visitor);
943 } else if (klass->IsArrayClass()) {
944 if (klass->IsObjectArrayClass<kVerifyNone>()) {
945 AsObjectArray<mirror::Object, kVerifyNone>()->VisitReferences<kVisitClass>(visitor);
946 } else if (kVisitClass) {
947 visitor(this, ClassOffset(), false);
Mathieu Chartier407f7022014-02-18 14:37:05 -0800948 }
949 } else {
Mathieu Chartier52e4b432014-06-10 11:22:31 -0700950 DCHECK(!klass->IsVariableSize());
Mathieu Chartier580a8df2014-03-26 15:15:57 -0700951 VisitInstanceFieldsReferences<kVisitClass>(klass, visitor);
Fred Shih4ee7a662014-07-11 09:59:27 -0700952 if (UNLIKELY(klass->IsTypeOfReferenceClass<kVerifyNone>())) {
Mathieu Chartier407f7022014-02-18 14:37:05 -0800953 ref_visitor(klass, AsReference());
954 }
955 }
956}
957
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800958} // namespace mirror
959} // namespace art
960
Brian Carlstromfc0e3212013-07-17 14:40:12 -0700961#endif // ART_RUNTIME_MIRROR_OBJECT_INL_H_