object-inl.h revision 12b58b23de974232e991c650405f929f8b0dcc9f
1/*
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
17#ifndef ART_RUNTIME_MIRROR_OBJECT_INL_H_
18#define ART_RUNTIME_MIRROR_OBJECT_INL_H_
19
20#include "object.h"
21
22#include "art_field.h"
23#include "art_method.h"
24#include "atomic.h"
25#include "array-inl.h"
26#include "class.h"
27#include "class_flags.h"
28#include "class_linker.h"
29#include "class_loader-inl.h"
30#include "dex_cache-inl.h"
31#include "lock_word-inl.h"
32#include "monitor.h"
33#include "object_array-inl.h"
34#include "object_reference-inl.h"
35#include "obj_ptr-inl.h"
36#include "read_barrier-inl.h"
37#include "reference.h"
38#include "runtime.h"
39#include "string-inl.h"
40#include "throwable.h"
41
42namespace art {
43namespace mirror {
44
45inline uint32_t Object::ClassSize(PointerSize pointer_size) {
46  uint32_t vtable_entries = kVTableLength;
47  return Class::ComputeClassSize(true, vtable_entries, 0, 0, 0, 0, 0, pointer_size);
48}
49
50template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
51inline Class* Object::GetClass() {
52  return GetFieldObject<Class, kVerifyFlags, kReadBarrierOption>(
53      OFFSET_OF_OBJECT_MEMBER(Object, klass_));
54}
55
56template<VerifyObjectFlags kVerifyFlags>
57inline void Object::SetClass(ObjPtr<Class> new_klass) {
58  // new_klass may be null prior to class linker initialization.
59  // We don't mark the card as this occurs as part of object allocation. Not all objects have
60  // backing cards, such as large objects.
61  // We use non transactional version since we can't undo this write. We also disable checking as
62  // we may run in transaction mode here.
63  SetFieldObjectWithoutWriteBarrier<false, false,
64      static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis)>(
65      OFFSET_OF_OBJECT_MEMBER(Object, klass_), new_klass);
66}
67
68template<VerifyObjectFlags kVerifyFlags>
69inline LockWord Object::GetLockWord(bool as_volatile) {
70  if (as_volatile) {
71    return LockWord(GetField32Volatile<kVerifyFlags>(OFFSET_OF_OBJECT_MEMBER(Object, monitor_)));
72  }
73  return LockWord(GetField32<kVerifyFlags>(OFFSET_OF_OBJECT_MEMBER(Object, monitor_)));
74}
75
76template<VerifyObjectFlags kVerifyFlags>
77inline void Object::SetLockWord(LockWord new_val, bool as_volatile) {
78  // Force use of non-transactional mode and do not check.
79  if (as_volatile) {
80    SetField32Volatile<false, false, kVerifyFlags>(
81        OFFSET_OF_OBJECT_MEMBER(Object, monitor_), new_val.GetValue());
82  } else {
83    SetField32<false, false, kVerifyFlags>(
84        OFFSET_OF_OBJECT_MEMBER(Object, monitor_), new_val.GetValue());
85  }
86}
87
88inline bool Object::CasLockWordWeakSequentiallyConsistent(LockWord old_val, LockWord new_val) {
89  // Force use of non-transactional mode and do not check.
90  return CasFieldWeakSequentiallyConsistent32<false, false>(
91      OFFSET_OF_OBJECT_MEMBER(Object, monitor_), old_val.GetValue(), new_val.GetValue());
92}
93
94inline bool Object::CasLockWordWeakRelaxed(LockWord old_val, LockWord new_val) {
95  // Force use of non-transactional mode and do not check.
96  return CasFieldWeakRelaxed32<false, false>(
97      OFFSET_OF_OBJECT_MEMBER(Object, monitor_), old_val.GetValue(), new_val.GetValue());
98}
99
100inline bool Object::CasLockWordWeakRelease(LockWord old_val, LockWord new_val) {
101  // Force use of non-transactional mode and do not check.
102  return CasFieldWeakRelease32<false, false>(
103      OFFSET_OF_OBJECT_MEMBER(Object, monitor_), old_val.GetValue(), new_val.GetValue());
104}
105
106inline uint32_t Object::GetLockOwnerThreadId() {
107  return Monitor::GetLockOwnerThreadId(this);
108}
109
110inline mirror::Object* Object::MonitorEnter(Thread* self) {
111  return Monitor::MonitorEnter(self, this, /*trylock*/false);
112}
113
114inline mirror::Object* Object::MonitorTryEnter(Thread* self) {
115  return Monitor::MonitorEnter(self, this, /*trylock*/true);
116}
117
118inline bool Object::MonitorExit(Thread* self) {
119  return Monitor::MonitorExit(self, this);
120}
121
122inline void Object::Notify(Thread* self) {
123  Monitor::Notify(self, this);
124}
125
126inline void Object::NotifyAll(Thread* self) {
127  Monitor::NotifyAll(self, this);
128}
129
130inline void Object::Wait(Thread* self) {
131  Monitor::Wait(self, this, 0, 0, true, kWaiting);
132}
133
134inline void Object::Wait(Thread* self, int64_t ms, int32_t ns) {
135  Monitor::Wait(self, this, ms, ns, true, kTimedWaiting);
136}
137
138inline uint32_t Object::GetReadBarrierState(uintptr_t* fake_address_dependency) {
139#ifdef USE_BAKER_READ_BARRIER
140  CHECK(kUseBakerReadBarrier);
141#if defined(__arm__)
142  uintptr_t obj = reinterpret_cast<uintptr_t>(this);
143  uintptr_t result;
144  DCHECK_EQ(OFFSETOF_MEMBER(Object, monitor_), 4U);
145  // Use inline assembly to prevent the compiler from optimizing away the false dependency.
146  __asm__ __volatile__(
147      "ldr %[result], [%[obj], #4]\n\t"
148      // This instruction is enough to "fool the compiler and the CPU" by having `fad` always be
149      // null, without them being able to assume that fact.
150      "eor %[fad], %[result], %[result]\n\t"
151      : [result] "+r" (result), [fad] "=r" (*fake_address_dependency)
152      : [obj] "r" (obj));
153  DCHECK_EQ(*fake_address_dependency, 0U);
154  LockWord lw(static_cast<uint32_t>(result));
155  uint32_t rb_state = lw.ReadBarrierState();
156  return rb_state;
157#elif defined(__aarch64__)
158  uintptr_t obj = reinterpret_cast<uintptr_t>(this);
159  uintptr_t result;
160  DCHECK_EQ(OFFSETOF_MEMBER(Object, monitor_), 4U);
161  // Use inline assembly to prevent the compiler from optimizing away the false dependency.
162  __asm__ __volatile__(
163      "ldr %w[result], [%[obj], #4]\n\t"
164      // This instruction is enough to "fool the compiler and the CPU" by having `fad` always be
165      // null, without them being able to assume that fact.
166      "eor %[fad], %[result], %[result]\n\t"
167      : [result] "+r" (result), [fad] "=r" (*fake_address_dependency)
168      : [obj] "r" (obj));
169  DCHECK_EQ(*fake_address_dependency, 0U);
170  LockWord lw(static_cast<uint32_t>(result));
171  uint32_t rb_state = lw.ReadBarrierState();
172  return rb_state;
173#elif defined(__i386__) || defined(__x86_64__)
174  LockWord lw = GetLockWord(false);
175  // i386/x86_64 don't need fake address dependency. Use a compiler fence to avoid compiler
176  // reordering.
177  *fake_address_dependency = 0;
178  std::atomic_signal_fence(std::memory_order_acquire);
179  uint32_t rb_state = lw.ReadBarrierState();
180  return rb_state;
181#else
182  // mips/mips64
183  LOG(FATAL) << "Unreachable";
184  UNREACHABLE();
185  UNUSED(fake_address_dependency);
186#endif
187#else  // !USE_BAKER_READ_BARRIER
188  LOG(FATAL) << "Unreachable";
189  UNREACHABLE();
190  UNUSED(fake_address_dependency);
191#endif
192}
193
194inline uint32_t Object::GetReadBarrierState() {
195#ifdef USE_BAKER_READ_BARRIER
196  DCHECK(kUseBakerReadBarrier);
197  LockWord lw(GetField<uint32_t, /*kIsVolatile*/false>(OFFSET_OF_OBJECT_MEMBER(Object, monitor_)));
198  uint32_t rb_state = lw.ReadBarrierState();
199  DCHECK(ReadBarrier::IsValidReadBarrierState(rb_state)) << rb_state;
200  return rb_state;
201#else
202  LOG(FATAL) << "Unreachable";
203  UNREACHABLE();
204#endif
205}
206
207inline uint32_t Object::GetReadBarrierStateAcquire() {
208#ifdef USE_BAKER_READ_BARRIER
209  DCHECK(kUseBakerReadBarrier);
210  LockWord lw(GetFieldAcquire<uint32_t>(OFFSET_OF_OBJECT_MEMBER(Object, monitor_)));
211  uint32_t rb_state = lw.ReadBarrierState();
212  DCHECK(ReadBarrier::IsValidReadBarrierState(rb_state)) << rb_state;
213  return rb_state;
214#else
215  LOG(FATAL) << "Unreachable";
216  UNREACHABLE();
217#endif
218}
219
220inline uint32_t Object::GetMarkBit() {
221#ifdef USE_READ_BARRIER
222  return GetLockWord(false).MarkBitState();
223#else
224  LOG(FATAL) << "Unreachable";
225  UNREACHABLE();
226#endif
227}
228
229inline void Object::SetReadBarrierState(uint32_t rb_state) {
230#ifdef USE_BAKER_READ_BARRIER
231  DCHECK(kUseBakerReadBarrier);
232  DCHECK(ReadBarrier::IsValidReadBarrierState(rb_state)) << rb_state;
233  LockWord lw = GetLockWord(false);
234  lw.SetReadBarrierState(rb_state);
235  SetLockWord(lw, false);
236#else
237  LOG(FATAL) << "Unreachable";
238  UNREACHABLE();
239  UNUSED(rb_state);
240#endif
241}
242
243template<bool kCasRelease>
244inline bool Object::AtomicSetReadBarrierState(uint32_t expected_rb_state, uint32_t rb_state) {
245#ifdef USE_BAKER_READ_BARRIER
246  DCHECK(kUseBakerReadBarrier);
247  DCHECK(ReadBarrier::IsValidReadBarrierState(expected_rb_state)) << expected_rb_state;
248  DCHECK(ReadBarrier::IsValidReadBarrierState(rb_state)) << rb_state;
249  LockWord expected_lw;
250  LockWord new_lw;
251  do {
252    LockWord lw = GetLockWord(false);
253    if (UNLIKELY(lw.ReadBarrierState() != expected_rb_state)) {
254      // Lost the race.
255      return false;
256    }
257    expected_lw = lw;
258    expected_lw.SetReadBarrierState(expected_rb_state);
259    new_lw = lw;
260    new_lw.SetReadBarrierState(rb_state);
261    // ConcurrentCopying::ProcessMarkStackRef uses this with kCasRelease == true.
262    // If kCasRelease == true, use a CAS release so that when GC updates all the fields of
263    // an object and then changes the object from gray to black, the field updates (stores) will be
264    // visible (won't be reordered after this CAS.)
265  } while (!(kCasRelease ?
266             CasLockWordWeakRelease(expected_lw, new_lw) :
267             CasLockWordWeakRelaxed(expected_lw, new_lw)));
268  return true;
269#else
270  UNUSED(expected_rb_state, rb_state);
271  LOG(FATAL) << "Unreachable";
272  UNREACHABLE();
273#endif
274}
275
276inline bool Object::AtomicSetMarkBit(uint32_t expected_mark_bit, uint32_t mark_bit) {
277  LockWord expected_lw;
278  LockWord new_lw;
279  do {
280    LockWord lw = GetLockWord(false);
281    if (UNLIKELY(lw.MarkBitState() != expected_mark_bit)) {
282      // Lost the race.
283      return false;
284    }
285    expected_lw = lw;
286    new_lw = lw;
287    new_lw.SetMarkBitState(mark_bit);
288    // Since this is only set from the mutator, we can use the non release Cas.
289  } while (!CasLockWordWeakRelaxed(expected_lw, new_lw));
290  return true;
291}
292
293
294inline void Object::AssertReadBarrierState() const {
295  CHECK(kUseBakerReadBarrier);
296  Object* obj = const_cast<Object*>(this);
297  DCHECK(obj->GetReadBarrierState() == ReadBarrier::WhiteState())
298      << "Bad Baker pointer: obj=" << reinterpret_cast<void*>(obj)
299      << " rb_state" << reinterpret_cast<void*>(obj->GetReadBarrierState());
300}
301
302template<VerifyObjectFlags kVerifyFlags>
303inline bool Object::VerifierInstanceOf(ObjPtr<Class> klass) {
304  DCHECK(klass != nullptr);
305  DCHECK(GetClass<kVerifyFlags>() != nullptr);
306  return klass->IsInterface() || InstanceOf(klass);
307}
308
309template<VerifyObjectFlags kVerifyFlags>
310inline bool Object::InstanceOf(ObjPtr<Class> klass) {
311  DCHECK(klass != nullptr);
312  DCHECK(GetClass<kVerifyNone>() != nullptr);
313  return klass->IsAssignableFrom(GetClass<kVerifyFlags>());
314}
315
316template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
317inline bool Object::IsClass() {
318  Class* java_lang_Class = GetClass<kVerifyFlags, kReadBarrierOption>()->
319      template GetClass<kVerifyFlags, kReadBarrierOption>();
320  return GetClass<static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis),
321      kReadBarrierOption>() == java_lang_Class;
322}
323
324template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
325inline Class* Object::AsClass() {
326  DCHECK((IsClass<kVerifyFlags, kReadBarrierOption>()));
327  return down_cast<Class*>(this);
328}
329
330template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
331inline bool Object::IsObjectArray() {
332  constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
333  return IsArrayInstance<kVerifyFlags, kReadBarrierOption>() &&
334      !GetClass<kNewFlags, kReadBarrierOption>()->
335          template GetComponentType<kNewFlags, kReadBarrierOption>()->IsPrimitive();
336}
337
338template<class T, VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
339inline ObjectArray<T>* Object::AsObjectArray() {
340  DCHECK((IsObjectArray<kVerifyFlags, kReadBarrierOption>()));
341  return down_cast<ObjectArray<T>*>(this);
342}
343
344template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
345inline bool Object::IsArrayInstance() {
346  return GetClass<kVerifyFlags, kReadBarrierOption>()->
347      template IsArrayClass<kVerifyFlags, kReadBarrierOption>();
348}
349
350template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
351inline bool Object::IsReferenceInstance() {
352  return GetClass<kVerifyFlags, kReadBarrierOption>()->IsTypeOfReferenceClass();
353}
354
355template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
356inline Reference* Object::AsReference() {
357  DCHECK((IsReferenceInstance<kVerifyFlags, kReadBarrierOption>()));
358  return down_cast<Reference*>(this);
359}
360
361template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
362inline Array* Object::AsArray() {
363  DCHECK((IsArrayInstance<kVerifyFlags, kReadBarrierOption>()));
364  return down_cast<Array*>(this);
365}
366
367template<VerifyObjectFlags kVerifyFlags>
368inline BooleanArray* Object::AsBooleanArray() {
369  constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
370  DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
371  DCHECK(GetClass<kNewFlags>()->GetComponentType()->IsPrimitiveBoolean());
372  return down_cast<BooleanArray*>(this);
373}
374
375template<VerifyObjectFlags kVerifyFlags>
376inline ByteArray* Object::AsByteArray() {
377  constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
378  DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
379  DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveByte());
380  return down_cast<ByteArray*>(this);
381}
382
383template<VerifyObjectFlags kVerifyFlags>
384inline ByteArray* Object::AsByteSizedArray() {
385  constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
386  DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
387  DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveByte() ||
388         GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveBoolean());
389  return down_cast<ByteArray*>(this);
390}
391
392template<VerifyObjectFlags kVerifyFlags>
393inline CharArray* Object::AsCharArray() {
394  constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
395  DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
396  DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveChar());
397  return down_cast<CharArray*>(this);
398}
399
400template<VerifyObjectFlags kVerifyFlags>
401inline ShortArray* Object::AsShortArray() {
402  constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
403  DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
404  DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveShort());
405  return down_cast<ShortArray*>(this);
406}
407
408template<VerifyObjectFlags kVerifyFlags>
409inline ShortArray* Object::AsShortSizedArray() {
410  constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
411  DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
412  DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveShort() ||
413         GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveChar());
414  return down_cast<ShortArray*>(this);
415}
416
417template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
418inline bool Object::IsIntArray() {
419  constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
420  ObjPtr<Class> klass = GetClass<kVerifyFlags, kReadBarrierOption>();
421  ObjPtr<Class> component_type = klass->GetComponentType<kVerifyFlags, kReadBarrierOption>();
422  return component_type != nullptr && component_type->template IsPrimitiveInt<kNewFlags>();
423}
424
425template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
426inline IntArray* Object::AsIntArray() {
427  DCHECK((IsIntArray<kVerifyFlags, kReadBarrierOption>()));
428  return down_cast<IntArray*>(this);
429}
430
431template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
432inline bool Object::IsLongArray() {
433  constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
434  ObjPtr<Class> klass = GetClass<kVerifyFlags, kReadBarrierOption>();
435  ObjPtr<Class> component_type = klass->GetComponentType<kVerifyFlags, kReadBarrierOption>();
436  return component_type != nullptr && component_type->template IsPrimitiveLong<kNewFlags>();
437}
438
439template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
440inline LongArray* Object::AsLongArray() {
441  DCHECK((IsLongArray<kVerifyFlags, kReadBarrierOption>()));
442  return down_cast<LongArray*>(this);
443}
444
445template<VerifyObjectFlags kVerifyFlags>
446inline bool Object::IsFloatArray() {
447  constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
448  auto* component_type = GetClass<kVerifyFlags>()->GetComponentType();
449  return component_type != nullptr && component_type->template IsPrimitiveFloat<kNewFlags>();
450}
451
452template<VerifyObjectFlags kVerifyFlags>
453inline FloatArray* Object::AsFloatArray() {
454  DCHECK(IsFloatArray<kVerifyFlags>());
455  constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
456  DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
457  DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveFloat());
458  return down_cast<FloatArray*>(this);
459}
460
461template<VerifyObjectFlags kVerifyFlags>
462inline bool Object::IsDoubleArray() {
463  constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
464  auto* component_type = GetClass<kVerifyFlags>()->GetComponentType();
465  return component_type != nullptr && component_type->template IsPrimitiveDouble<kNewFlags>();
466}
467
468template<VerifyObjectFlags kVerifyFlags>
469inline DoubleArray* Object::AsDoubleArray() {
470  DCHECK(IsDoubleArray<kVerifyFlags>());
471  constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
472  DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
473  DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveDouble());
474  return down_cast<DoubleArray*>(this);
475}
476
477template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
478inline bool Object::IsString() {
479  return GetClass<kVerifyFlags, kReadBarrierOption>()->IsStringClass();
480}
481
482template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
483inline String* Object::AsString() {
484  DCHECK((IsString<kVerifyFlags, kReadBarrierOption>()));
485  return down_cast<String*>(this);
486}
487
488template<VerifyObjectFlags kVerifyFlags>
489inline Throwable* Object::AsThrowable() {
490  DCHECK(GetClass<kVerifyFlags>()->IsThrowableClass());
491  return down_cast<Throwable*>(this);
492}
493
494template<VerifyObjectFlags kVerifyFlags>
495inline bool Object::IsWeakReferenceInstance() {
496  return GetClass<kVerifyFlags>()->IsWeakReferenceClass();
497}
498
499template<VerifyObjectFlags kVerifyFlags>
500inline bool Object::IsSoftReferenceInstance() {
501  return GetClass<kVerifyFlags>()->IsSoftReferenceClass();
502}
503
504template<VerifyObjectFlags kVerifyFlags>
505inline bool Object::IsFinalizerReferenceInstance() {
506  return GetClass<kVerifyFlags>()->IsFinalizerReferenceClass();
507}
508
509template<VerifyObjectFlags kVerifyFlags>
510inline FinalizerReference* Object::AsFinalizerReference() {
511  DCHECK(IsFinalizerReferenceInstance<kVerifyFlags>());
512  return down_cast<FinalizerReference*>(this);
513}
514
515template<VerifyObjectFlags kVerifyFlags>
516inline bool Object::IsPhantomReferenceInstance() {
517  return GetClass<kVerifyFlags>()->IsPhantomReferenceClass();
518}
519
520template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
521inline size_t Object::SizeOf() {
522  size_t result;
523  constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
524  if (IsArrayInstance<kVerifyFlags, kReadBarrierOption>()) {
525    result = AsArray<kNewFlags, kReadBarrierOption>()->
526        template SizeOf<kNewFlags, kReadBarrierOption>();
527  } else if (IsClass<kNewFlags, kReadBarrierOption>()) {
528    result = AsClass<kNewFlags, kReadBarrierOption>()->
529        template SizeOf<kNewFlags, kReadBarrierOption>();
530  } else if (GetClass<kNewFlags, kReadBarrierOption>()->IsStringClass()) {
531    result = AsString<kNewFlags, kReadBarrierOption>()->
532        template SizeOf<kNewFlags>();
533  } else {
534    result = GetClass<kNewFlags, kReadBarrierOption>()->
535        template GetObjectSize<kNewFlags, kReadBarrierOption>();
536  }
537  DCHECK_GE(result, sizeof(Object))
538      << " class=" << Class::PrettyClass(GetClass<kNewFlags, kReadBarrierOption>());
539  return result;
540}
541
542template<VerifyObjectFlags kVerifyFlags, bool kIsVolatile>
543inline uint8_t Object::GetFieldBoolean(MemberOffset field_offset) {
544  if (kVerifyFlags & kVerifyThis) {
545    VerifyObject(this);
546  }
547  return GetField<uint8_t, kIsVolatile>(field_offset);
548}
549
550template<VerifyObjectFlags kVerifyFlags, bool kIsVolatile>
551inline int8_t Object::GetFieldByte(MemberOffset field_offset) {
552  if (kVerifyFlags & kVerifyThis) {
553    VerifyObject(this);
554  }
555  return GetField<int8_t, kIsVolatile>(field_offset);
556}
557
558template<VerifyObjectFlags kVerifyFlags>
559inline uint8_t Object::GetFieldBooleanVolatile(MemberOffset field_offset) {
560  return GetFieldBoolean<kVerifyFlags, true>(field_offset);
561}
562
563template<VerifyObjectFlags kVerifyFlags>
564inline int8_t Object::GetFieldByteVolatile(MemberOffset field_offset) {
565  return GetFieldByte<kVerifyFlags, true>(field_offset);
566}
567
568template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags,
569    bool kIsVolatile>
570inline void Object::SetFieldBoolean(MemberOffset field_offset, uint8_t new_value)
571    REQUIRES_SHARED(Locks::mutator_lock_) {
572  if (kCheckTransaction) {
573    DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
574  }
575  if (kTransactionActive) {
576    Runtime::Current()->RecordWriteFieldBoolean(this, field_offset,
577                                           GetFieldBoolean<kVerifyFlags, kIsVolatile>(field_offset),
578                                           kIsVolatile);
579  }
580  if (kVerifyFlags & kVerifyThis) {
581    VerifyObject(this);
582  }
583  SetField<uint8_t, kIsVolatile>(field_offset, new_value);
584}
585
586template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags,
587    bool kIsVolatile>
588inline void Object::SetFieldByte(MemberOffset field_offset, int8_t new_value)
589    REQUIRES_SHARED(Locks::mutator_lock_) {
590  if (kCheckTransaction) {
591    DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
592  }
593  if (kTransactionActive) {
594    Runtime::Current()->RecordWriteFieldByte(this, field_offset,
595                                           GetFieldByte<kVerifyFlags, kIsVolatile>(field_offset),
596                                           kIsVolatile);
597  }
598  if (kVerifyFlags & kVerifyThis) {
599    VerifyObject(this);
600  }
601  SetField<int8_t, kIsVolatile>(field_offset, new_value);
602}
603
604template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
605inline void Object::SetFieldBooleanVolatile(MemberOffset field_offset, uint8_t new_value) {
606  return SetFieldBoolean<kTransactionActive, kCheckTransaction, kVerifyFlags, true>(
607      field_offset, new_value);
608}
609
610template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
611inline void Object::SetFieldByteVolatile(MemberOffset field_offset, int8_t new_value) {
612  return SetFieldByte<kTransactionActive, kCheckTransaction, kVerifyFlags, true>(
613      field_offset, new_value);
614}
615
616template<VerifyObjectFlags kVerifyFlags, bool kIsVolatile>
617inline uint16_t Object::GetFieldChar(MemberOffset field_offset) {
618  if (kVerifyFlags & kVerifyThis) {
619    VerifyObject(this);
620  }
621  return GetField<uint16_t, kIsVolatile>(field_offset);
622}
623
624template<VerifyObjectFlags kVerifyFlags, bool kIsVolatile>
625inline int16_t Object::GetFieldShort(MemberOffset field_offset) {
626  if (kVerifyFlags & kVerifyThis) {
627    VerifyObject(this);
628  }
629  return GetField<int16_t, kIsVolatile>(field_offset);
630}
631
632template<VerifyObjectFlags kVerifyFlags>
633inline uint16_t Object::GetFieldCharVolatile(MemberOffset field_offset) {
634  return GetFieldChar<kVerifyFlags, true>(field_offset);
635}
636
637template<VerifyObjectFlags kVerifyFlags>
638inline int16_t Object::GetFieldShortVolatile(MemberOffset field_offset) {
639  return GetFieldShort<kVerifyFlags, true>(field_offset);
640}
641
642template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags,
643    bool kIsVolatile>
644inline void Object::SetFieldChar(MemberOffset field_offset, uint16_t new_value) {
645  if (kCheckTransaction) {
646    DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
647  }
648  if (kTransactionActive) {
649    Runtime::Current()->RecordWriteFieldChar(this, field_offset,
650                                           GetFieldChar<kVerifyFlags, kIsVolatile>(field_offset),
651                                           kIsVolatile);
652  }
653  if (kVerifyFlags & kVerifyThis) {
654    VerifyObject(this);
655  }
656  SetField<uint16_t, kIsVolatile>(field_offset, new_value);
657}
658
659template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags,
660    bool kIsVolatile>
661inline void Object::SetFieldShort(MemberOffset field_offset, int16_t new_value) {
662  if (kCheckTransaction) {
663    DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
664  }
665  if (kTransactionActive) {
666    Runtime::Current()->RecordWriteFieldChar(this, field_offset,
667                                           GetFieldShort<kVerifyFlags, kIsVolatile>(field_offset),
668                                           kIsVolatile);
669  }
670  if (kVerifyFlags & kVerifyThis) {
671    VerifyObject(this);
672  }
673  SetField<int16_t, kIsVolatile>(field_offset, new_value);
674}
675
676template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
677inline void Object::SetFieldCharVolatile(MemberOffset field_offset, uint16_t new_value) {
678  return SetFieldChar<kTransactionActive, kCheckTransaction, kVerifyFlags, true>(
679      field_offset, new_value);
680}
681
682template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
683inline void Object::SetFieldShortVolatile(MemberOffset field_offset, int16_t new_value) {
684  return SetFieldShort<kTransactionActive, kCheckTransaction, kVerifyFlags, true>(
685      field_offset, new_value);
686}
687
688template<VerifyObjectFlags kVerifyFlags, bool kIsVolatile>
689inline int32_t Object::GetField32(MemberOffset field_offset) {
690  if (kVerifyFlags & kVerifyThis) {
691    VerifyObject(this);
692  }
693  return GetField<int32_t, kIsVolatile>(field_offset);
694}
695
696template<VerifyObjectFlags kVerifyFlags>
697inline int32_t Object::GetField32Volatile(MemberOffset field_offset) {
698  return GetField32<kVerifyFlags, true>(field_offset);
699}
700
701template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags,
702    bool kIsVolatile>
703inline void Object::SetField32(MemberOffset field_offset, int32_t new_value) {
704  if (kCheckTransaction) {
705    DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
706  }
707  if (kTransactionActive) {
708    Runtime::Current()->RecordWriteField32(this, field_offset,
709                                           GetField32<kVerifyFlags, kIsVolatile>(field_offset),
710                                           kIsVolatile);
711  }
712  if (kVerifyFlags & kVerifyThis) {
713    VerifyObject(this);
714  }
715  SetField<int32_t, kIsVolatile>(field_offset, new_value);
716}
717
718template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
719inline void Object::SetField32Volatile(MemberOffset field_offset, int32_t new_value) {
720  SetField32<kTransactionActive, kCheckTransaction, kVerifyFlags, true>(field_offset, new_value);
721}
722
723// TODO: Pass memory_order_ and strong/weak as arguments to avoid code duplication?
724
725template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
726inline bool Object::CasFieldWeakSequentiallyConsistent32(MemberOffset field_offset,
727                                                         int32_t old_value, int32_t new_value) {
728  if (kCheckTransaction) {
729    DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
730  }
731  if (kTransactionActive) {
732    Runtime::Current()->RecordWriteField32(this, field_offset, old_value, true);
733  }
734  if (kVerifyFlags & kVerifyThis) {
735    VerifyObject(this);
736  }
737  uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
738  AtomicInteger* atomic_addr = reinterpret_cast<AtomicInteger*>(raw_addr);
739
740  return atomic_addr->CompareExchangeWeakSequentiallyConsistent(old_value, new_value);
741}
742
743template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
744inline bool Object::CasFieldWeakRelaxed32(MemberOffset field_offset,
745                                          int32_t old_value, int32_t new_value) {
746  if (kCheckTransaction) {
747    DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
748  }
749  if (kTransactionActive) {
750    Runtime::Current()->RecordWriteField32(this, field_offset, old_value, true);
751  }
752  if (kVerifyFlags & kVerifyThis) {
753    VerifyObject(this);
754  }
755  uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
756  AtomicInteger* atomic_addr = reinterpret_cast<AtomicInteger*>(raw_addr);
757
758  return atomic_addr->CompareExchangeWeakRelaxed(old_value, new_value);
759}
760
761template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
762inline bool Object::CasFieldWeakRelease32(MemberOffset field_offset,
763                                          int32_t old_value, int32_t new_value) {
764  if (kCheckTransaction) {
765    DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
766  }
767  if (kTransactionActive) {
768    Runtime::Current()->RecordWriteField32(this, field_offset, old_value, true);
769  }
770  if (kVerifyFlags & kVerifyThis) {
771    VerifyObject(this);
772  }
773  uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
774  AtomicInteger* atomic_addr = reinterpret_cast<AtomicInteger*>(raw_addr);
775
776  return atomic_addr->CompareExchangeWeakRelease(old_value, new_value);
777}
778
779template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
780inline bool Object::CasFieldStrongSequentiallyConsistent32(MemberOffset field_offset,
781                                                           int32_t old_value, int32_t new_value) {
782  if (kCheckTransaction) {
783    DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
784  }
785  if (kTransactionActive) {
786    Runtime::Current()->RecordWriteField32(this, field_offset, old_value, true);
787  }
788  if (kVerifyFlags & kVerifyThis) {
789    VerifyObject(this);
790  }
791  uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
792  AtomicInteger* atomic_addr = reinterpret_cast<AtomicInteger*>(raw_addr);
793
794  return atomic_addr->CompareExchangeStrongSequentiallyConsistent(old_value, new_value);
795}
796
797template<VerifyObjectFlags kVerifyFlags, bool kIsVolatile>
798inline int64_t Object::GetField64(MemberOffset field_offset) {
799  if (kVerifyFlags & kVerifyThis) {
800    VerifyObject(this);
801  }
802  return GetField<int64_t, kIsVolatile>(field_offset);
803}
804
805template<VerifyObjectFlags kVerifyFlags>
806inline int64_t Object::GetField64Volatile(MemberOffset field_offset) {
807  return GetField64<kVerifyFlags, true>(field_offset);
808}
809
810template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags,
811    bool kIsVolatile>
812inline void Object::SetField64(MemberOffset field_offset, int64_t new_value) {
813  if (kCheckTransaction) {
814    DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
815  }
816  if (kTransactionActive) {
817    Runtime::Current()->RecordWriteField64(this, field_offset,
818                                           GetField64<kVerifyFlags, kIsVolatile>(field_offset),
819                                           kIsVolatile);
820  }
821  if (kVerifyFlags & kVerifyThis) {
822    VerifyObject(this);
823  }
824  SetField<int64_t, kIsVolatile>(field_offset, new_value);
825}
826
827template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
828inline void Object::SetField64Volatile(MemberOffset field_offset, int64_t new_value) {
829  return SetField64<kTransactionActive, kCheckTransaction, kVerifyFlags, true>(field_offset,
830                                                                               new_value);
831}
832
833template<typename kSize, bool kIsVolatile>
834inline void Object::SetField(MemberOffset field_offset, kSize new_value) {
835  uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
836  kSize* addr = reinterpret_cast<kSize*>(raw_addr);
837  if (kIsVolatile) {
838    reinterpret_cast<Atomic<kSize>*>(addr)->StoreSequentiallyConsistent(new_value);
839  } else {
840    reinterpret_cast<Atomic<kSize>*>(addr)->StoreJavaData(new_value);
841  }
842}
843
844template<typename kSize, bool kIsVolatile>
845inline kSize Object::GetField(MemberOffset field_offset) {
846  const uint8_t* raw_addr = reinterpret_cast<const uint8_t*>(this) + field_offset.Int32Value();
847  const kSize* addr = reinterpret_cast<const kSize*>(raw_addr);
848  if (kIsVolatile) {
849    return reinterpret_cast<const Atomic<kSize>*>(addr)->LoadSequentiallyConsistent();
850  } else {
851    return reinterpret_cast<const Atomic<kSize>*>(addr)->LoadJavaData();
852  }
853}
854
855template<typename kSize>
856inline kSize Object::GetFieldAcquire(MemberOffset field_offset) {
857  const uint8_t* raw_addr = reinterpret_cast<const uint8_t*>(this) + field_offset.Int32Value();
858  const kSize* addr = reinterpret_cast<const kSize*>(raw_addr);
859  return reinterpret_cast<const Atomic<kSize>*>(addr)->LoadAcquire();
860}
861
862template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
863inline bool Object::CasFieldWeakSequentiallyConsistent64(MemberOffset field_offset,
864                                                         int64_t old_value, int64_t new_value) {
865  if (kCheckTransaction) {
866    DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
867  }
868  if (kTransactionActive) {
869    Runtime::Current()->RecordWriteField64(this, field_offset, old_value, true);
870  }
871  if (kVerifyFlags & kVerifyThis) {
872    VerifyObject(this);
873  }
874  uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
875  Atomic<int64_t>* atomic_addr = reinterpret_cast<Atomic<int64_t>*>(raw_addr);
876  return atomic_addr->CompareExchangeWeakSequentiallyConsistent(old_value, new_value);
877}
878
879template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
880inline bool Object::CasFieldStrongSequentiallyConsistent64(MemberOffset field_offset,
881                                                           int64_t old_value, int64_t new_value) {
882  if (kCheckTransaction) {
883    DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
884  }
885  if (kTransactionActive) {
886    Runtime::Current()->RecordWriteField64(this, field_offset, old_value, true);
887  }
888  if (kVerifyFlags & kVerifyThis) {
889    VerifyObject(this);
890  }
891  uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
892  Atomic<int64_t>* atomic_addr = reinterpret_cast<Atomic<int64_t>*>(raw_addr);
893  return atomic_addr->CompareExchangeStrongSequentiallyConsistent(old_value, new_value);
894}
895
896template<class T, VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption,
897         bool kIsVolatile>
898inline T* Object::GetFieldObject(MemberOffset field_offset) {
899  if (kVerifyFlags & kVerifyThis) {
900    VerifyObject(this);
901  }
902  uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
903  HeapReference<T>* objref_addr = reinterpret_cast<HeapReference<T>*>(raw_addr);
904  T* result = ReadBarrier::Barrier<T, kReadBarrierOption>(this, field_offset, objref_addr);
905  if (kIsVolatile) {
906    // TODO: Refactor to use a SequentiallyConsistent load instead.
907    QuasiAtomic::ThreadFenceAcquire();  // Ensure visibility of operations preceding store.
908  }
909  if (kVerifyFlags & kVerifyReads) {
910    VerifyObject(result);
911  }
912  return result;
913}
914
915template<class T, VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
916inline T* Object::GetFieldObjectVolatile(MemberOffset field_offset) {
917  return GetFieldObject<T, kVerifyFlags, kReadBarrierOption, true>(field_offset);
918}
919
920template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags,
921    bool kIsVolatile>
922inline void Object::SetFieldObjectWithoutWriteBarrier(MemberOffset field_offset,
923                                                      ObjPtr<Object> new_value) {
924  if (kCheckTransaction) {
925    DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
926  }
927  if (kTransactionActive) {
928    ObjPtr<Object> obj;
929    if (kIsVolatile) {
930      obj = GetFieldObjectVolatile<Object>(field_offset);
931    } else {
932      obj = GetFieldObject<Object>(field_offset);
933    }
934    Runtime::Current()->RecordWriteFieldReference(this, field_offset, obj.Ptr(), true);
935  }
936  if (kVerifyFlags & kVerifyThis) {
937    VerifyObject(this);
938  }
939  if (kVerifyFlags & kVerifyWrites) {
940    VerifyObject(new_value);
941  }
942  uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
943  HeapReference<Object>* objref_addr = reinterpret_cast<HeapReference<Object>*>(raw_addr);
944  if (kIsVolatile) {
945    // TODO: Refactor to use a SequentiallyConsistent store instead.
946    QuasiAtomic::ThreadFenceRelease();  // Ensure that prior accesses are visible before store.
947    objref_addr->Assign(new_value.Ptr());
948    QuasiAtomic::ThreadFenceSequentiallyConsistent();
949                                // Ensure this store occurs before any volatile loads.
950  } else {
951    objref_addr->Assign(new_value.Ptr());
952  }
953}
954
955template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags,
956    bool kIsVolatile>
957inline void Object::SetFieldObject(MemberOffset field_offset, ObjPtr<Object> new_value) {
958  SetFieldObjectWithoutWriteBarrier<kTransactionActive, kCheckTransaction, kVerifyFlags,
959      kIsVolatile>(field_offset, new_value);
960  if (new_value != nullptr) {
961    Runtime::Current()->GetHeap()->WriteBarrierField(this, field_offset, new_value);
962    // TODO: Check field assignment could theoretically cause thread suspension, TODO: fix this.
963    CheckFieldAssignment(field_offset, new_value);
964  }
965}
966
967template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
968inline void Object::SetFieldObjectVolatile(MemberOffset field_offset, ObjPtr<Object> new_value) {
969  SetFieldObject<kTransactionActive, kCheckTransaction, kVerifyFlags, true>(field_offset,
970                                                                            new_value);
971}
972
973template <VerifyObjectFlags kVerifyFlags>
974inline HeapReference<Object>* Object::GetFieldObjectReferenceAddr(MemberOffset field_offset) {
975  if (kVerifyFlags & kVerifyThis) {
976    VerifyObject(this);
977  }
978  return reinterpret_cast<HeapReference<Object>*>(reinterpret_cast<uint8_t*>(this) +
979      field_offset.Int32Value());
980}
981
982template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
983inline bool Object::CasFieldWeakSequentiallyConsistentObject(MemberOffset field_offset,
984                                                             ObjPtr<Object> old_value,
985                                                             ObjPtr<Object> new_value) {
986  bool success = CasFieldWeakSequentiallyConsistentObjectWithoutWriteBarrier<
987      kTransactionActive, kCheckTransaction, kVerifyFlags>(field_offset, old_value, new_value);
988  if (success) {
989    Runtime::Current()->GetHeap()->WriteBarrierField(this, field_offset, new_value);
990  }
991  return success;
992}
993
994template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
995inline bool Object::CasFieldWeakSequentiallyConsistentObjectWithoutWriteBarrier(
996    MemberOffset field_offset,
997    ObjPtr<Object> old_value,
998    ObjPtr<Object> new_value) {
999  if (kCheckTransaction) {
1000    DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
1001  }
1002  if (kVerifyFlags & kVerifyThis) {
1003    VerifyObject(this);
1004  }
1005  if (kVerifyFlags & kVerifyWrites) {
1006    VerifyObject(new_value);
1007  }
1008  if (kVerifyFlags & kVerifyReads) {
1009    VerifyObject(old_value);
1010  }
1011  if (kTransactionActive) {
1012    Runtime::Current()->RecordWriteFieldReference(this, field_offset, old_value, true);
1013  }
1014  HeapReference<Object> old_ref(HeapReference<Object>::FromObjPtr(old_value));
1015  HeapReference<Object> new_ref(HeapReference<Object>::FromObjPtr(new_value));
1016  uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
1017  Atomic<uint32_t>* atomic_addr = reinterpret_cast<Atomic<uint32_t>*>(raw_addr);
1018
1019  bool success = atomic_addr->CompareExchangeWeakSequentiallyConsistent(old_ref.reference_,
1020                                                                        new_ref.reference_);
1021  return success;
1022}
1023
1024template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
1025inline bool Object::CasFieldStrongSequentiallyConsistentObject(MemberOffset field_offset,
1026                                                               ObjPtr<Object> old_value,
1027                                                               ObjPtr<Object> new_value) {
1028  bool success = CasFieldStrongSequentiallyConsistentObjectWithoutWriteBarrier<
1029      kTransactionActive, kCheckTransaction, kVerifyFlags>(field_offset, old_value, new_value);
1030  if (success) {
1031    Runtime::Current()->GetHeap()->WriteBarrierField(this, field_offset, new_value);
1032  }
1033  return success;
1034}
1035
1036template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
1037inline bool Object::CasFieldStrongSequentiallyConsistentObjectWithoutWriteBarrier(
1038    MemberOffset field_offset,
1039    ObjPtr<Object> old_value,
1040    ObjPtr<Object> new_value) {
1041  if (kCheckTransaction) {
1042    DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
1043  }
1044  if (kVerifyFlags & kVerifyThis) {
1045    VerifyObject(this);
1046  }
1047  if (kVerifyFlags & kVerifyWrites) {
1048    VerifyObject(new_value);
1049  }
1050  if (kVerifyFlags & kVerifyReads) {
1051    VerifyObject(old_value);
1052  }
1053  if (kTransactionActive) {
1054    Runtime::Current()->RecordWriteFieldReference(this, field_offset, old_value, true);
1055  }
1056  HeapReference<Object> old_ref(HeapReference<Object>::FromObjPtr(old_value));
1057  HeapReference<Object> new_ref(HeapReference<Object>::FromObjPtr(new_value));
1058  uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
1059  Atomic<uint32_t>* atomic_addr = reinterpret_cast<Atomic<uint32_t>*>(raw_addr);
1060
1061  bool success = atomic_addr->CompareExchangeStrongSequentiallyConsistent(old_ref.reference_,
1062                                                                          new_ref.reference_);
1063  return success;
1064}
1065
1066template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
1067inline bool Object::CasFieldWeakRelaxedObjectWithoutWriteBarrier(
1068    MemberOffset field_offset,
1069    ObjPtr<Object> old_value,
1070    ObjPtr<Object> new_value) {
1071  if (kCheckTransaction) {
1072    DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
1073  }
1074  if (kVerifyFlags & kVerifyThis) {
1075    VerifyObject(this);
1076  }
1077  if (kVerifyFlags & kVerifyWrites) {
1078    VerifyObject(new_value);
1079  }
1080  if (kVerifyFlags & kVerifyReads) {
1081    VerifyObject(old_value);
1082  }
1083  if (kTransactionActive) {
1084    Runtime::Current()->RecordWriteFieldReference(this, field_offset, old_value, true);
1085  }
1086  HeapReference<Object> old_ref(HeapReference<Object>::FromObjPtr(old_value));
1087  HeapReference<Object> new_ref(HeapReference<Object>::FromObjPtr(new_value));
1088  uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
1089  Atomic<uint32_t>* atomic_addr = reinterpret_cast<Atomic<uint32_t>*>(raw_addr);
1090
1091  bool success = atomic_addr->CompareExchangeWeakRelaxed(old_ref.reference_,
1092                                                         new_ref.reference_);
1093  return success;
1094}
1095
1096template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
1097inline bool Object::CasFieldStrongRelaxedObjectWithoutWriteBarrier(
1098    MemberOffset field_offset,
1099    ObjPtr<Object> old_value,
1100    ObjPtr<Object> new_value) {
1101  if (kCheckTransaction) {
1102    DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
1103  }
1104  if (kVerifyFlags & kVerifyThis) {
1105    VerifyObject(this);
1106  }
1107  if (kVerifyFlags & kVerifyWrites) {
1108    VerifyObject(new_value);
1109  }
1110  if (kVerifyFlags & kVerifyReads) {
1111    VerifyObject(old_value);
1112  }
1113  if (kTransactionActive) {
1114    Runtime::Current()->RecordWriteFieldReference(this, field_offset, old_value, true);
1115  }
1116  HeapReference<Object> old_ref(HeapReference<Object>::FromObjPtr(old_value));
1117  HeapReference<Object> new_ref(HeapReference<Object>::FromObjPtr(new_value));
1118  uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
1119  Atomic<uint32_t>* atomic_addr = reinterpret_cast<Atomic<uint32_t>*>(raw_addr);
1120
1121  bool success = atomic_addr->CompareExchangeStrongRelaxed(old_ref.reference_,
1122                                                           new_ref.reference_);
1123  return success;
1124}
1125
1126template<bool kIsStatic,
1127         VerifyObjectFlags kVerifyFlags,
1128         ReadBarrierOption kReadBarrierOption,
1129         typename Visitor>
1130inline void Object::VisitFieldsReferences(uint32_t ref_offsets, const Visitor& visitor) {
1131  if (!kIsStatic && (ref_offsets != mirror::Class::kClassWalkSuper)) {
1132    // Instance fields and not the slow-path.
1133    uint32_t field_offset = mirror::kObjectHeaderSize;
1134    while (ref_offsets != 0) {
1135      if ((ref_offsets & 1) != 0) {
1136        visitor(this, MemberOffset(field_offset), kIsStatic);
1137      }
1138      ref_offsets >>= 1;
1139      field_offset += sizeof(mirror::HeapReference<mirror::Object>);
1140    }
1141  } else {
1142    // There is no reference offset bitmap. In the non-static case, walk up the class
1143    // inheritance hierarchy and find reference offsets the hard way. In the static case, just
1144    // consider this class.
1145    for (ObjPtr<Class> klass = kIsStatic
1146            ? AsClass<kVerifyFlags, kReadBarrierOption>()
1147            : GetClass<kVerifyFlags, kReadBarrierOption>();
1148        klass != nullptr;
1149        klass = kIsStatic ? nullptr : klass->GetSuperClass<kVerifyFlags, kReadBarrierOption>()) {
1150      const size_t num_reference_fields =
1151          kIsStatic ? klass->NumReferenceStaticFields() : klass->NumReferenceInstanceFields();
1152      if (num_reference_fields == 0u) {
1153        continue;
1154      }
1155      // Presumably GC can happen when we are cross compiling, it should not cause performance
1156      // problems to do pointer size logic.
1157      MemberOffset field_offset = kIsStatic
1158          ? klass->GetFirstReferenceStaticFieldOffset<kVerifyFlags, kReadBarrierOption>(
1159              Runtime::Current()->GetClassLinker()->GetImagePointerSize())
1160          : klass->GetFirstReferenceInstanceFieldOffset<kVerifyFlags, kReadBarrierOption>();
1161      for (size_t i = 0u; i < num_reference_fields; ++i) {
1162        // TODO: Do a simpler check?
1163        if (field_offset.Uint32Value() != ClassOffset().Uint32Value()) {
1164          visitor(this, field_offset, kIsStatic);
1165        }
1166        field_offset = MemberOffset(field_offset.Uint32Value() +
1167                                    sizeof(mirror::HeapReference<mirror::Object>));
1168      }
1169    }
1170  }
1171}
1172
1173template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption, typename Visitor>
1174inline void Object::VisitInstanceFieldsReferences(ObjPtr<Class> klass, const Visitor& visitor) {
1175  VisitFieldsReferences<false, kVerifyFlags, kReadBarrierOption>(
1176      klass->GetReferenceInstanceOffsets<kVerifyFlags>(), visitor);
1177}
1178
1179template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption, typename Visitor>
1180inline void Object::VisitStaticFieldsReferences(ObjPtr<Class> klass, const Visitor& visitor) {
1181  DCHECK(!klass->IsTemp());
1182  klass->VisitFieldsReferences<true, kVerifyFlags, kReadBarrierOption>(0, visitor);
1183}
1184
1185template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
1186inline bool Object::IsClassLoader() {
1187  return GetClass<kVerifyFlags, kReadBarrierOption>()->IsClassLoaderClass();
1188}
1189
1190template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
1191inline mirror::ClassLoader* Object::AsClassLoader() {
1192  DCHECK((IsClassLoader<kVerifyFlags, kReadBarrierOption>()));
1193  return down_cast<mirror::ClassLoader*>(this);
1194}
1195
1196template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
1197inline bool Object::IsDexCache() {
1198  return GetClass<kVerifyFlags, kReadBarrierOption>()->IsDexCacheClass();
1199}
1200
1201template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
1202inline mirror::DexCache* Object::AsDexCache() {
1203  DCHECK((IsDexCache<kVerifyFlags, kReadBarrierOption>()));
1204  return down_cast<mirror::DexCache*>(this);
1205}
1206
1207template <bool kVisitNativeRoots,
1208          VerifyObjectFlags kVerifyFlags,
1209          ReadBarrierOption kReadBarrierOption,
1210          typename Visitor,
1211          typename JavaLangRefVisitor>
1212inline void Object::VisitReferences(const Visitor& visitor,
1213                                    const JavaLangRefVisitor& ref_visitor) {
1214  ObjPtr<Class> klass = GetClass<kVerifyFlags, kReadBarrierOption>();
1215  visitor(this, ClassOffset(), false);
1216  const uint32_t class_flags = klass->GetClassFlags<kVerifyNone>();
1217  if (LIKELY(class_flags == kClassFlagNormal)) {
1218    DCHECK((!klass->IsVariableSize<kVerifyFlags, kReadBarrierOption>()));
1219    VisitInstanceFieldsReferences<kVerifyFlags, kReadBarrierOption>(klass, visitor);
1220    DCHECK((!klass->IsClassClass<kVerifyFlags, kReadBarrierOption>()));
1221    DCHECK(!klass->IsStringClass());
1222    DCHECK(!klass->IsClassLoaderClass());
1223    DCHECK((!klass->IsArrayClass<kVerifyFlags, kReadBarrierOption>()));
1224  } else {
1225    if ((class_flags & kClassFlagNoReferenceFields) == 0) {
1226      DCHECK(!klass->IsStringClass());
1227      if (class_flags == kClassFlagClass) {
1228        DCHECK((klass->IsClassClass<kVerifyFlags, kReadBarrierOption>()));
1229        ObjPtr<Class> as_klass = AsClass<kVerifyNone, kReadBarrierOption>();
1230        as_klass->VisitReferences<kVisitNativeRoots, kVerifyFlags, kReadBarrierOption>(klass,
1231                                                                                       visitor);
1232      } else if (class_flags == kClassFlagObjectArray) {
1233        DCHECK((klass->IsObjectArrayClass<kVerifyFlags, kReadBarrierOption>()));
1234        AsObjectArray<mirror::Object, kVerifyNone, kReadBarrierOption>()->VisitReferences(visitor);
1235      } else if ((class_flags & kClassFlagReference) != 0) {
1236        VisitInstanceFieldsReferences<kVerifyFlags, kReadBarrierOption>(klass, visitor);
1237        ref_visitor(klass, AsReference<kVerifyFlags, kReadBarrierOption>());
1238      } else if (class_flags == kClassFlagDexCache) {
1239        mirror::DexCache* const dex_cache = AsDexCache<kVerifyFlags, kReadBarrierOption>();
1240        dex_cache->VisitReferences<kVisitNativeRoots,
1241                                   kVerifyFlags,
1242                                   kReadBarrierOption>(klass, visitor);
1243      } else {
1244        mirror::ClassLoader* const class_loader = AsClassLoader<kVerifyFlags, kReadBarrierOption>();
1245        class_loader->VisitReferences<kVisitNativeRoots,
1246                                      kVerifyFlags,
1247                                      kReadBarrierOption>(klass, visitor);
1248      }
1249    } else if (kIsDebugBuild) {
1250      CHECK((!klass->IsClassClass<kVerifyFlags, kReadBarrierOption>()));
1251      CHECK((!klass->IsObjectArrayClass<kVerifyFlags, kReadBarrierOption>()));
1252      // String still has instance fields for reflection purposes but these don't exist in
1253      // actual string instances.
1254      if (!klass->IsStringClass()) {
1255        size_t total_reference_instance_fields = 0;
1256        ObjPtr<Class> super_class = klass;
1257        do {
1258          total_reference_instance_fields += super_class->NumReferenceInstanceFields();
1259          super_class = super_class->GetSuperClass<kVerifyFlags, kReadBarrierOption>();
1260        } while (super_class != nullptr);
1261        // The only reference field should be the object's class. This field is handled at the
1262        // beginning of the function.
1263        CHECK_EQ(total_reference_instance_fields, 1u);
1264      }
1265    }
1266  }
1267}
1268}  // namespace mirror
1269}  // namespace art
1270
1271#endif  // ART_RUNTIME_MIRROR_OBJECT_INL_H_
1272