blob: 089ef5731028759ef502a1244cf52bf7ff5d1cea [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
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -070038template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogersef7d42f2014-01-06 12:55:46 -080039inline Class* Object::GetClass() {
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -070040 return GetFieldObject<Class, kVerifyFlags, kReadBarrierOption>(
Ian Rogersb0fa5dc2014-04-28 16:47:08 -070041 OFFSET_OF_OBJECT_MEMBER(Object, klass_));
Ian Rogers2dd0e2c2013-01-24 12:42:14 -080042}
43
Mathieu Chartier4e305412014-02-19 10:54:44 -080044template<VerifyObjectFlags kVerifyFlags>
Ian Rogers2dd0e2c2013-01-24 12:42:14 -080045inline void Object::SetClass(Class* new_klass) {
Ian Rogersef7d42f2014-01-06 12:55:46 -080046 // new_klass may be NULL prior to class linker initialization.
47 // We don't mark the card as this occurs as part of object allocation. Not all objects have
48 // backing cards, such as large objects.
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +010049 // We use non transactional version since we can't undo this write. We also disable checking as
50 // we may run in transaction mode here.
Mathieu Chartier4e305412014-02-19 10:54:44 -080051 SetFieldObjectWithoutWriteBarrier<false, false,
52 static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis)>(
Ian Rogersb0fa5dc2014-04-28 16:47:08 -070053 OFFSET_OF_OBJECT_MEMBER(Object, klass_), new_klass);
Ian Rogers2dd0e2c2013-01-24 12:42:14 -080054}
55
Mathieu Chartierbbd695c2014-04-16 09:48:48 -070056inline LockWord Object::GetLockWord(bool as_volatile) {
Ian Rogersb0fa5dc2014-04-28 16:47:08 -070057 if (as_volatile) {
58 return LockWord(GetField32Volatile(OFFSET_OF_OBJECT_MEMBER(Object, monitor_)));
59 }
60 return LockWord(GetField32(OFFSET_OF_OBJECT_MEMBER(Object, monitor_)));
Ian Rogersd9c4fc92013-10-01 19:45:43 -070061}
62
Mathieu Chartierbbd695c2014-04-16 09:48:48 -070063inline void Object::SetLockWord(LockWord new_val, bool as_volatile) {
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +010064 // Force use of non-transactional mode and do not check.
Ian Rogersb0fa5dc2014-04-28 16:47:08 -070065 if (as_volatile) {
66 SetField32Volatile<false, false>(OFFSET_OF_OBJECT_MEMBER(Object, monitor_), new_val.GetValue());
67 } else {
68 SetField32<false, false>(OFFSET_OF_OBJECT_MEMBER(Object, monitor_), new_val.GetValue());
69 }
Ian Rogersd9c4fc92013-10-01 19:45:43 -070070}
71
Ian Rogers228602f2014-07-10 02:07:54 -070072inline bool Object::CasLockWordWeakSequentiallyConsistent(LockWord old_val, LockWord new_val) {
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +010073 // Force use of non-transactional mode and do not check.
Ian Rogers228602f2014-07-10 02:07:54 -070074 return CasFieldWeakSequentiallyConsistent32<false, false>(
75 OFFSET_OF_OBJECT_MEMBER(Object, monitor_), old_val.GetValue(), new_val.GetValue());
Ian Rogersd9c4fc92013-10-01 19:45:43 -070076}
77
78inline uint32_t Object::GetLockOwnerThreadId() {
79 return Monitor::GetLockOwnerThreadId(this);
Ian Rogers05f30572013-02-20 12:13:11 -080080}
81
Mathieu Chartiere7e8a5f2014-02-14 16:59:41 -080082inline mirror::Object* Object::MonitorEnter(Thread* self) {
83 return Monitor::MonitorEnter(self, this);
Ian Rogers05f30572013-02-20 12:13:11 -080084}
85
86inline bool Object::MonitorExit(Thread* self) {
87 return Monitor::MonitorExit(self, this);
88}
89
90inline void Object::Notify(Thread* self) {
91 Monitor::Notify(self, this);
92}
93
94inline void Object::NotifyAll(Thread* self) {
95 Monitor::NotifyAll(self, this);
96}
97
98inline void Object::Wait(Thread* self) {
99 Monitor::Wait(self, this, 0, 0, true, kWaiting);
100}
101
102inline void Object::Wait(Thread* self, int64_t ms, int32_t ns) {
103 Monitor::Wait(self, this, ms, ns, true, kTimedWaiting);
104}
105
Hiroshi Yamauchi624468c2014-03-31 15:14:47 -0700106inline Object* Object::GetReadBarrierPointer() {
107#ifdef USE_BAKER_OR_BROOKS_READ_BARRIER
108 DCHECK(kUseBakerOrBrooksReadBarrier);
Hiroshi Yamauchi6a154a42014-05-02 14:26:13 -0700109 return GetFieldObject<Object, kVerifyNone, kWithoutReadBarrier>(
110 OFFSET_OF_OBJECT_MEMBER(Object, x_rb_ptr_));
Hiroshi Yamauchi9d04a202014-01-31 13:35:49 -0800111#else
112 LOG(FATAL) << "Unreachable";
113 return nullptr;
114#endif
115}
116
Hiroshi Yamauchi9103c862014-04-22 13:51:07 -0700117inline void Object::SetReadBarrierPointer(Object* rb_ptr) {
Hiroshi Yamauchi624468c2014-03-31 15:14:47 -0700118#ifdef USE_BAKER_OR_BROOKS_READ_BARRIER
119 DCHECK(kUseBakerOrBrooksReadBarrier);
Hiroshi Yamauchi9d04a202014-01-31 13:35:49 -0800120 // We don't mark the card as this occurs as part of object allocation. Not all objects have
121 // backing cards, such as large objects.
122 SetFieldObjectWithoutWriteBarrier<false, false, kVerifyNone>(
Hiroshi Yamauchi6a154a42014-05-02 14:26:13 -0700123 OFFSET_OF_OBJECT_MEMBER(Object, x_rb_ptr_), rb_ptr);
Hiroshi Yamauchi9d04a202014-01-31 13:35:49 -0800124#else
125 LOG(FATAL) << "Unreachable";
126#endif
127}
128
Hiroshi Yamauchi9103c862014-04-22 13:51:07 -0700129inline bool Object::AtomicSetReadBarrierPointer(Object* expected_rb_ptr, Object* rb_ptr) {
130#ifdef USE_BAKER_OR_BROOKS_READ_BARRIER
131 DCHECK(kUseBakerOrBrooksReadBarrier);
132 MemberOffset offset = OFFSET_OF_OBJECT_MEMBER(Object, x_rb_ptr_);
133 byte* raw_addr = reinterpret_cast<byte*>(this) + offset.SizeValue();
Ian Rogers228602f2014-07-10 02:07:54 -0700134 Atomic<uint32_t>* atomic_rb_ptr = reinterpret_cast<Atomic<uint32_t>*>(raw_addr);
Hiroshi Yamauchi9103c862014-04-22 13:51:07 -0700135 HeapReference<Object> expected_ref(HeapReference<Object>::FromMirrorPtr(expected_rb_ptr));
136 HeapReference<Object> new_ref(HeapReference<Object>::FromMirrorPtr(rb_ptr));
Hiroshi Yamauchi9103c862014-04-22 13:51:07 -0700137 do {
Ian Rogers228602f2014-07-10 02:07:54 -0700138 if (UNLIKELY(atomic_rb_ptr->LoadRelaxed() != expected_ref.reference_)) {
Hiroshi Yamauchi9103c862014-04-22 13:51:07 -0700139 // Lost the race.
140 return false;
141 }
Ian Rogers228602f2014-07-10 02:07:54 -0700142 } while (!atomic_rb_ptr->CompareExchangeWeakSequentiallyConsistent(expected_ref.reference_,
143 new_ref.reference_));
144 DCHECK_EQ(new_ref.reference_, atomic_rb_ptr->LoadRelaxed());
Hiroshi Yamauchi9103c862014-04-22 13:51:07 -0700145 return true;
146#else
147 LOG(FATAL) << "Unreachable";
148 return false;
149#endif
150}
151
Hiroshi Yamauchi624468c2014-03-31 15:14:47 -0700152inline void Object::AssertReadBarrierPointer() const {
Hiroshi Yamauchi800ac2d2014-04-02 17:32:54 -0700153 if (kUseBakerReadBarrier) {
154 Object* obj = const_cast<Object*>(this);
155 DCHECK(obj->GetReadBarrierPointer() == nullptr)
156 << "Bad Baker pointer: obj=" << reinterpret_cast<void*>(obj)
157 << " ptr=" << reinterpret_cast<void*>(obj->GetReadBarrierPointer());
158 } else if (kUseBrooksReadBarrier) {
159 Object* obj = const_cast<Object*>(this);
160 DCHECK_EQ(obj, obj->GetReadBarrierPointer())
161 << "Bad Brooks pointer: obj=" << reinterpret_cast<void*>(obj)
162 << " ptr=" << reinterpret_cast<void*>(obj->GetReadBarrierPointer());
163 } else {
164 LOG(FATAL) << "Unreachable";
165 }
Hiroshi Yamauchi9d04a202014-01-31 13:35:49 -0800166}
167
Mathieu Chartier4e305412014-02-19 10:54:44 -0800168template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800169inline bool Object::VerifierInstanceOf(Class* klass) {
Jeff Haoa3faaf42013-09-03 19:07:00 -0700170 DCHECK(klass != NULL);
Mathieu Chartier4e305412014-02-19 10:54:44 -0800171 DCHECK(GetClass<kVerifyFlags>() != NULL);
Jeff Haoa3faaf42013-09-03 19:07:00 -0700172 return klass->IsInterface() || InstanceOf(klass);
173}
174
Mathieu Chartier4e305412014-02-19 10:54:44 -0800175template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800176inline bool Object::InstanceOf(Class* klass) {
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800177 DCHECK(klass != NULL);
Mathieu Chartier4e305412014-02-19 10:54:44 -0800178 DCHECK(GetClass<kVerifyNone>() != NULL);
179 return klass->IsAssignableFrom(GetClass<kVerifyFlags>());
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800180}
181
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700182template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800183inline bool Object::IsClass() {
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700184 Class* java_lang_Class = GetClass<kVerifyFlags, kReadBarrierOption>()->
185 template GetClass<kVerifyFlags, kReadBarrierOption>();
186 return GetClass<static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis),
187 kReadBarrierOption>() == java_lang_Class;
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800188}
189
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700190template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800191inline Class* Object::AsClass() {
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700192 DCHECK((IsClass<kVerifyFlags, kReadBarrierOption>()));
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800193 return down_cast<Class*>(this);
194}
195
Mathieu Chartier4e305412014-02-19 10:54:44 -0800196template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800197inline bool Object::IsObjectArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800198 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
199 return IsArrayInstance<kVerifyFlags>() &&
200 !GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitive();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800201}
202
Mathieu Chartier4e305412014-02-19 10:54:44 -0800203template<class T, VerifyObjectFlags kVerifyFlags>
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800204inline ObjectArray<T>* Object::AsObjectArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800205 DCHECK(IsObjectArray<kVerifyFlags>());
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800206 return down_cast<ObjectArray<T>*>(this);
207}
208
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700209template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800210inline bool Object::IsArrayInstance() {
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700211 return GetClass<kVerifyFlags, kReadBarrierOption>()->
212 template IsArrayClass<kVerifyFlags, kReadBarrierOption>();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800213}
214
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700215template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800216inline bool Object::IsArtField() {
Hiroshi Yamauchi4f1ebc22014-06-25 14:30:41 -0700217 return GetClass<kVerifyFlags, kReadBarrierOption>()->
218 template IsArtFieldClass<kReadBarrierOption>();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800219}
220
Mathieu Chartier4e305412014-02-19 10:54:44 -0800221template<VerifyObjectFlags kVerifyFlags>
222inline ArtField* Object::AsArtField() {
223 DCHECK(IsArtField<kVerifyFlags>());
Brian Carlstromea46f952013-07-30 01:26:50 -0700224 return down_cast<ArtField*>(this);
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800225}
226
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700227template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800228inline bool Object::IsArtMethod() {
Hiroshi Yamauchi4f1ebc22014-06-25 14:30:41 -0700229 return GetClass<kVerifyFlags, kReadBarrierOption>()->
230 template IsArtMethodClass<kReadBarrierOption>();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800231}
232
Mathieu Chartier4e305412014-02-19 10:54:44 -0800233template<VerifyObjectFlags kVerifyFlags>
Brian Carlstromea46f952013-07-30 01:26:50 -0700234inline ArtMethod* Object::AsArtMethod() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800235 DCHECK(IsArtMethod<kVerifyFlags>());
Brian Carlstromea46f952013-07-30 01:26:50 -0700236 return down_cast<ArtMethod*>(this);
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800237}
238
Mathieu Chartier4e305412014-02-19 10:54:44 -0800239template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800240inline bool Object::IsReferenceInstance() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800241 return GetClass<kVerifyFlags>()->IsReferenceClass();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800242}
243
Mathieu Chartier4e305412014-02-19 10:54:44 -0800244template<VerifyObjectFlags kVerifyFlags>
Mathieu Chartier8fa2dad2014-03-13 12:22:56 -0700245inline Reference* Object::AsReference() {
246 DCHECK(IsReferenceInstance<kVerifyFlags>());
247 return down_cast<Reference*>(this);
248}
249
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700250template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogers05f30572013-02-20 12:13:11 -0800251inline Array* Object::AsArray() {
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700252 DCHECK((IsArrayInstance<kVerifyFlags, kReadBarrierOption>()));
Ian Rogers05f30572013-02-20 12:13:11 -0800253 return down_cast<Array*>(this);
254}
255
Mathieu Chartier4e305412014-02-19 10:54:44 -0800256template<VerifyObjectFlags kVerifyFlags>
Ian Rogers05f30572013-02-20 12:13:11 -0800257inline BooleanArray* Object::AsBooleanArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800258 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
259 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
260 DCHECK(GetClass<kNewFlags>()->GetComponentType()->IsPrimitiveBoolean());
Ian Rogers05f30572013-02-20 12:13:11 -0800261 return down_cast<BooleanArray*>(this);
262}
263
Mathieu Chartier4e305412014-02-19 10:54:44 -0800264template<VerifyObjectFlags kVerifyFlags>
Ian Rogers05f30572013-02-20 12:13:11 -0800265inline ByteArray* Object::AsByteArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800266 static const VerifyObjectFlags kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
267 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
268 DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveByte());
Ian Rogers05f30572013-02-20 12:13:11 -0800269 return down_cast<ByteArray*>(this);
270}
271
Mathieu Chartier4e305412014-02-19 10:54:44 -0800272template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800273inline ByteArray* Object::AsByteSizedArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800274 constexpr VerifyObjectFlags kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
275 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
276 DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveByte() ||
277 GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveBoolean());
Ian Rogersef7d42f2014-01-06 12:55:46 -0800278 return down_cast<ByteArray*>(this);
279}
280
Mathieu Chartier4e305412014-02-19 10:54:44 -0800281template<VerifyObjectFlags kVerifyFlags>
Ian Rogers05f30572013-02-20 12:13:11 -0800282inline CharArray* Object::AsCharArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800283 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
284 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
285 DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveChar());
Ian Rogers05f30572013-02-20 12:13:11 -0800286 return down_cast<CharArray*>(this);
287}
288
Mathieu Chartier4e305412014-02-19 10:54:44 -0800289template<VerifyObjectFlags kVerifyFlags>
Ian Rogers05f30572013-02-20 12:13:11 -0800290inline ShortArray* Object::AsShortArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800291 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
292 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
293 DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveShort());
Ian Rogers05f30572013-02-20 12:13:11 -0800294 return down_cast<ShortArray*>(this);
295}
296
Mathieu Chartier4e305412014-02-19 10:54:44 -0800297template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800298inline ShortArray* Object::AsShortSizedArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800299 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
300 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
301 DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveShort() ||
302 GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveChar());
Ian Rogersef7d42f2014-01-06 12:55:46 -0800303 return down_cast<ShortArray*>(this);
304}
305
Mathieu Chartier4e305412014-02-19 10:54:44 -0800306template<VerifyObjectFlags kVerifyFlags>
Ian Rogers05f30572013-02-20 12:13:11 -0800307inline IntArray* Object::AsIntArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800308 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
309 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
310 DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveInt() ||
311 GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveFloat());
Ian Rogers05f30572013-02-20 12:13:11 -0800312 return down_cast<IntArray*>(this);
313}
314
Mathieu Chartier4e305412014-02-19 10:54:44 -0800315template<VerifyObjectFlags kVerifyFlags>
Ian Rogers05f30572013-02-20 12:13:11 -0800316inline LongArray* Object::AsLongArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800317 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
318 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
319 DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveLong() ||
320 GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveDouble());
Ian Rogers05f30572013-02-20 12:13:11 -0800321 return down_cast<LongArray*>(this);
322}
323
Mathieu Chartier4e305412014-02-19 10:54:44 -0800324template<VerifyObjectFlags kVerifyFlags>
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100325inline FloatArray* Object::AsFloatArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800326 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
327 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
328 DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveFloat());
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100329 return down_cast<FloatArray*>(this);
330}
331
Mathieu Chartier4e305412014-02-19 10:54:44 -0800332template<VerifyObjectFlags kVerifyFlags>
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100333inline DoubleArray* Object::AsDoubleArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800334 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
335 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
336 DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveDouble());
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100337 return down_cast<DoubleArray*>(this);
338}
339
Mathieu Chartier4e305412014-02-19 10:54:44 -0800340template<VerifyObjectFlags kVerifyFlags>
Ian Rogers05f30572013-02-20 12:13:11 -0800341inline String* Object::AsString() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800342 DCHECK(GetClass<kVerifyFlags>()->IsStringClass());
Ian Rogers05f30572013-02-20 12:13:11 -0800343 return down_cast<String*>(this);
344}
345
Mathieu Chartier4e305412014-02-19 10:54:44 -0800346template<VerifyObjectFlags kVerifyFlags>
Ian Rogers05f30572013-02-20 12:13:11 -0800347inline Throwable* Object::AsThrowable() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800348 DCHECK(GetClass<kVerifyFlags>()->IsThrowableClass());
Ian Rogers05f30572013-02-20 12:13:11 -0800349 return down_cast<Throwable*>(this);
350}
351
Mathieu Chartier4e305412014-02-19 10:54:44 -0800352template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800353inline bool Object::IsWeakReferenceInstance() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800354 return GetClass<kVerifyFlags>()->IsWeakReferenceClass();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800355}
356
Mathieu Chartier4e305412014-02-19 10:54:44 -0800357template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800358inline bool Object::IsSoftReferenceInstance() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800359 return GetClass<kVerifyFlags>()->IsSoftReferenceClass();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800360}
361
Mathieu Chartier4e305412014-02-19 10:54:44 -0800362template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800363inline bool Object::IsFinalizerReferenceInstance() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800364 return GetClass<kVerifyFlags>()->IsFinalizerReferenceClass();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800365}
366
Mathieu Chartier4e305412014-02-19 10:54:44 -0800367template<VerifyObjectFlags kVerifyFlags>
Mathieu Chartier8fa2dad2014-03-13 12:22:56 -0700368inline FinalizerReference* Object::AsFinalizerReference() {
369 DCHECK(IsFinalizerReferenceInstance<kVerifyFlags>());
370 return down_cast<FinalizerReference*>(this);
371}
372
373template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800374inline bool Object::IsPhantomReferenceInstance() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800375 return GetClass<kVerifyFlags>()->IsPhantomReferenceClass();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800376}
377
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700378template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800379inline size_t Object::SizeOf() {
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800380 size_t result;
Mathieu Chartier4e305412014-02-19 10:54:44 -0800381 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700382 if (IsArrayInstance<kVerifyFlags, kReadBarrierOption>()) {
383 result = AsArray<kNewFlags, kReadBarrierOption>()->
384 template SizeOf<kNewFlags, kReadBarrierOption>();
385 } else if (IsClass<kNewFlags, kReadBarrierOption>()) {
386 result = AsClass<kNewFlags, kReadBarrierOption>()->
387 template SizeOf<kNewFlags, kReadBarrierOption>();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800388 } else {
Hiroshi Yamauchi25023c72014-05-09 11:45:53 -0700389 result = GetClass<kNewFlags, kReadBarrierOption>()->
390 template GetObjectSize<kNewFlags, kReadBarrierOption>();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800391 }
Hiroshi Yamauchi9103c862014-04-22 13:51:07 -0700392 DCHECK_GE(result, sizeof(Object))
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700393 << " class=" << PrettyTypeOf(GetClass<kNewFlags, kReadBarrierOption>());
394 DCHECK(!(IsArtField<kNewFlags, kReadBarrierOption>()) || result == sizeof(ArtField));
395 DCHECK(!(IsArtMethod<kNewFlags, kReadBarrierOption>()) || result == sizeof(ArtMethod));
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800396 return result;
397}
398
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700399template<VerifyObjectFlags kVerifyFlags, bool kIsVolatile>
400inline int32_t Object::GetField32(MemberOffset field_offset) {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800401 if (kVerifyFlags & kVerifyThis) {
402 VerifyObject(this);
403 }
Ian Rogersb122a4b2013-11-19 18:00:50 -0800404 const byte* raw_addr = reinterpret_cast<const byte*>(this) + field_offset.Int32Value();
405 const int32_t* word_addr = reinterpret_cast<const int32_t*>(raw_addr);
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700406 if (UNLIKELY(kIsVolatile)) {
Hans Boehm30359612014-05-21 17:46:23 -0700407 return reinterpret_cast<const Atomic<int32_t>*>(word_addr)->LoadSequentiallyConsistent();
Ian Rogersb122a4b2013-11-19 18:00:50 -0800408 } else {
Hans Boehm30359612014-05-21 17:46:23 -0700409 return reinterpret_cast<const Atomic<int32_t>*>(word_addr)->LoadJavaData();
Ian Rogersb122a4b2013-11-19 18:00:50 -0800410 }
411}
412
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700413template<VerifyObjectFlags kVerifyFlags>
414inline int32_t Object::GetField32Volatile(MemberOffset field_offset) {
415 return GetField32<kVerifyFlags, true>(field_offset);
416}
417
418template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags,
419 bool kIsVolatile>
420inline void Object::SetField32(MemberOffset field_offset, int32_t new_value) {
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100421 if (kCheckTransaction) {
422 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
423 }
424 if (kTransactionActive) {
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700425 Runtime::Current()->RecordWriteField32(this, field_offset,
426 GetField32<kVerifyFlags, kIsVolatile>(field_offset),
427 kIsVolatile);
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100428 }
Mathieu Chartier4e305412014-02-19 10:54:44 -0800429 if (kVerifyFlags & kVerifyThis) {
Ian Rogersb122a4b2013-11-19 18:00:50 -0800430 VerifyObject(this);
431 }
432 byte* raw_addr = reinterpret_cast<byte*>(this) + field_offset.Int32Value();
Ian Rogers29501cf2014-02-07 21:00:25 -0800433 int32_t* word_addr = reinterpret_cast<int32_t*>(raw_addr);
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700434 if (kIsVolatile) {
Hans Boehm30359612014-05-21 17:46:23 -0700435 reinterpret_cast<Atomic<int32_t>*>(word_addr)->StoreSequentiallyConsistent(new_value);
Ian Rogersb122a4b2013-11-19 18:00:50 -0800436 } else {
Hans Boehm30359612014-05-21 17:46:23 -0700437 reinterpret_cast<Atomic<int32_t>*>(word_addr)->StoreJavaData(new_value);
Ian Rogersb122a4b2013-11-19 18:00:50 -0800438 }
439}
440
Mathieu Chartier4e305412014-02-19 10:54:44 -0800441template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700442inline void Object::SetField32Volatile(MemberOffset field_offset, int32_t new_value) {
443 SetField32<kTransactionActive, kCheckTransaction, kVerifyFlags, true>(field_offset, new_value);
444}
445
446template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
Ian Rogers228602f2014-07-10 02:07:54 -0700447inline bool Object::CasFieldWeakSequentiallyConsistent32(MemberOffset field_offset,
448 int32_t old_value, int32_t new_value) {
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100449 if (kCheckTransaction) {
450 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
451 }
452 if (kTransactionActive) {
453 Runtime::Current()->RecordWriteField32(this, field_offset, old_value, true);
454 }
Mathieu Chartier4e305412014-02-19 10:54:44 -0800455 if (kVerifyFlags & kVerifyThis) {
456 VerifyObject(this);
457 }
Ian Rogersd9c4fc92013-10-01 19:45:43 -0700458 byte* raw_addr = reinterpret_cast<byte*>(this) + field_offset.Int32Value();
Ian Rogers228602f2014-07-10 02:07:54 -0700459 AtomicInteger* atomic_addr = reinterpret_cast<AtomicInteger*>(raw_addr);
Hans Boehm30359612014-05-21 17:46:23 -0700460
Ian Rogers228602f2014-07-10 02:07:54 -0700461 return atomic_addr->CompareExchangeWeakSequentiallyConsistent(old_value, new_value);
Ian Rogersd9c4fc92013-10-01 19:45:43 -0700462}
463
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700464template<VerifyObjectFlags kVerifyFlags, bool kIsVolatile>
465inline int64_t Object::GetField64(MemberOffset field_offset) {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800466 if (kVerifyFlags & kVerifyThis) {
467 VerifyObject(this);
468 }
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800469 const byte* raw_addr = reinterpret_cast<const byte*>(this) + field_offset.Int32Value();
470 const int64_t* addr = reinterpret_cast<const int64_t*>(raw_addr);
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700471 if (kIsVolatile) {
Hans Boehm30359612014-05-21 17:46:23 -0700472 return reinterpret_cast<const Atomic<int64_t>*>(addr)->LoadSequentiallyConsistent();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800473 } else {
Hans Boehm30359612014-05-21 17:46:23 -0700474 return reinterpret_cast<const Atomic<int64_t>*>(addr)->LoadJavaData();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800475 }
476}
477
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700478template<VerifyObjectFlags kVerifyFlags>
479inline int64_t Object::GetField64Volatile(MemberOffset field_offset) {
480 return GetField64<kVerifyFlags, true>(field_offset);
481}
482
483template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags,
484 bool kIsVolatile>
485inline void Object::SetField64(MemberOffset field_offset, int64_t new_value) {
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100486 if (kCheckTransaction) {
487 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
488 }
489 if (kTransactionActive) {
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700490 Runtime::Current()->RecordWriteField64(this, field_offset,
491 GetField64<kVerifyFlags, kIsVolatile>(field_offset),
492 kIsVolatile);
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100493 }
Mathieu Chartier4e305412014-02-19 10:54:44 -0800494 if (kVerifyFlags & kVerifyThis) {
Ian Rogersef7d42f2014-01-06 12:55:46 -0800495 VerifyObject(this);
496 }
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800497 byte* raw_addr = reinterpret_cast<byte*>(this) + field_offset.Int32Value();
498 int64_t* addr = reinterpret_cast<int64_t*>(raw_addr);
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700499 if (kIsVolatile) {
Hans Boehm30359612014-05-21 17:46:23 -0700500 reinterpret_cast<Atomic<int64_t>*>(addr)->StoreSequentiallyConsistent(new_value);
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800501 } else {
Hans Boehm30359612014-05-21 17:46:23 -0700502 reinterpret_cast<Atomic<int64_t>*>(addr)->StoreJavaData(new_value);
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800503 }
504}
505
Mathieu Chartier4e305412014-02-19 10:54:44 -0800506template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700507inline void Object::SetField64Volatile(MemberOffset field_offset, int64_t new_value) {
508 return SetField64<kTransactionActive, kCheckTransaction, kVerifyFlags, true>(field_offset,
509 new_value);
510}
511
512template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
Ian Rogers228602f2014-07-10 02:07:54 -0700513inline bool Object::CasFieldWeakSequentiallyConsistent64(MemberOffset field_offset,
514 int64_t old_value, int64_t new_value) {
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100515 if (kCheckTransaction) {
516 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
517 }
518 if (kTransactionActive) {
519 Runtime::Current()->RecordWriteField64(this, field_offset, old_value, true);
520 }
Mathieu Chartier4e305412014-02-19 10:54:44 -0800521 if (kVerifyFlags & kVerifyThis) {
522 VerifyObject(this);
523 }
Ian Rogersef7d42f2014-01-06 12:55:46 -0800524 byte* raw_addr = reinterpret_cast<byte*>(this) + field_offset.Int32Value();
Ian Rogers228602f2014-07-10 02:07:54 -0700525 Atomic<int64_t>* atomic_addr = reinterpret_cast<Atomic<int64_t>*>(raw_addr);
526 return atomic_addr->CompareExchangeWeakSequentiallyConsistent(old_value, new_value);
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800527}
528
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700529template<class T, VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption,
530 bool kIsVolatile>
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700531inline T* Object::GetFieldObject(MemberOffset field_offset) {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800532 if (kVerifyFlags & kVerifyThis) {
533 VerifyObject(this);
534 }
Ian Rogersef7d42f2014-01-06 12:55:46 -0800535 byte* raw_addr = reinterpret_cast<byte*>(this) + field_offset.Int32Value();
536 HeapReference<T>* objref_addr = reinterpret_cast<HeapReference<T>*>(raw_addr);
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700537 T* result = ReadBarrier::Barrier<T, kReadBarrierOption>(this, field_offset, objref_addr);
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700538 if (kIsVolatile) {
Hans Boehm30359612014-05-21 17:46:23 -0700539 // TODO: Refactor to use a SequentiallyConsistent load instead.
540 QuasiAtomic::ThreadFenceAcquire(); // Ensure visibility of operations preceding store.
Ian Rogersef7d42f2014-01-06 12:55:46 -0800541 }
Mathieu Chartier4e305412014-02-19 10:54:44 -0800542 if (kVerifyFlags & kVerifyReads) {
543 VerifyObject(result);
544 }
Ian Rogersef7d42f2014-01-06 12:55:46 -0800545 return result;
546}
547
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700548template<class T, VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700549inline T* Object::GetFieldObjectVolatile(MemberOffset field_offset) {
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700550 return GetFieldObject<T, kVerifyFlags, kReadBarrierOption, true>(field_offset);
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700551}
552
553template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags,
554 bool kIsVolatile>
555inline void Object::SetFieldObjectWithoutWriteBarrier(MemberOffset field_offset,
556 Object* new_value) {
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100557 if (kCheckTransaction) {
558 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
559 }
560 if (kTransactionActive) {
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700561 mirror::Object* obj;
562 if (kIsVolatile) {
563 obj = GetFieldObjectVolatile<Object>(field_offset);
564 } else {
565 obj = GetFieldObject<Object>(field_offset);
566 }
567 Runtime::Current()->RecordWriteFieldReference(this, field_offset, obj, true);
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100568 }
Mathieu Chartier4e305412014-02-19 10:54:44 -0800569 if (kVerifyFlags & kVerifyThis) {
Ian Rogersef7d42f2014-01-06 12:55:46 -0800570 VerifyObject(this);
571 }
Mathieu Chartier4e305412014-02-19 10:54:44 -0800572 if (kVerifyFlags & kVerifyWrites) {
573 VerifyObject(new_value);
574 }
Ian Rogersef7d42f2014-01-06 12:55:46 -0800575 byte* raw_addr = reinterpret_cast<byte*>(this) + field_offset.Int32Value();
576 HeapReference<Object>* objref_addr = reinterpret_cast<HeapReference<Object>*>(raw_addr);
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700577 if (kIsVolatile) {
Hans Boehm30359612014-05-21 17:46:23 -0700578 // TODO: Refactor to use a SequentiallyConsistent store instead.
579 QuasiAtomic::ThreadFenceRelease(); // Ensure that prior accesses are visible before store.
Ian Rogersef7d42f2014-01-06 12:55:46 -0800580 objref_addr->Assign(new_value);
Hans Boehm30359612014-05-21 17:46:23 -0700581 QuasiAtomic::ThreadFenceSequentiallyConsistent();
582 // Ensure this store occurs before any volatile loads.
Ian Rogersef7d42f2014-01-06 12:55:46 -0800583 } else {
584 objref_addr->Assign(new_value);
585 }
586}
587
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700588template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags,
589 bool kIsVolatile>
590inline void Object::SetFieldObject(MemberOffset field_offset, Object* new_value) {
591 SetFieldObjectWithoutWriteBarrier<kTransactionActive, kCheckTransaction, kVerifyFlags,
592 kIsVolatile>(field_offset, new_value);
Ian Rogersef7d42f2014-01-06 12:55:46 -0800593 if (new_value != nullptr) {
Ian Rogersef7d42f2014-01-06 12:55:46 -0800594 Runtime::Current()->GetHeap()->WriteBarrierField(this, field_offset, new_value);
Mathieu Chartier61c5ebc2014-06-05 17:42:53 -0700595 // TODO: Check field assignment could theoretically cause thread suspension, TODO: fix this.
596 CheckFieldAssignment(field_offset, new_value);
Ian Rogersef7d42f2014-01-06 12:55:46 -0800597 }
598}
599
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700600template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
601inline void Object::SetFieldObjectVolatile(MemberOffset field_offset, Object* new_value) {
602 SetFieldObject<kTransactionActive, kCheckTransaction, kVerifyFlags, true>(field_offset,
603 new_value);
604}
605
Mathieu Chartier4e305412014-02-19 10:54:44 -0800606template <VerifyObjectFlags kVerifyFlags>
607inline HeapReference<Object>* Object::GetFieldObjectReferenceAddr(MemberOffset field_offset) {
608 if (kVerifyFlags & kVerifyThis) {
609 VerifyObject(this);
610 }
611 return reinterpret_cast<HeapReference<Object>*>(reinterpret_cast<byte*>(this) +
612 field_offset.Int32Value());
613}
614
615template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
Ian Rogers228602f2014-07-10 02:07:54 -0700616inline bool Object::CasFieldWeakSequentiallyConsistentObject(MemberOffset field_offset,
617 Object* old_value, Object* new_value) {
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100618 if (kCheckTransaction) {
619 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
620 }
Mathieu Chartier4e305412014-02-19 10:54:44 -0800621 if (kVerifyFlags & kVerifyThis) {
622 VerifyObject(this);
623 }
624 if (kVerifyFlags & kVerifyWrites) {
625 VerifyObject(new_value);
626 }
627 if (kVerifyFlags & kVerifyReads) {
628 VerifyObject(old_value);
629 }
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100630 if (kTransactionActive) {
631 Runtime::Current()->RecordWriteFieldReference(this, field_offset, old_value, true);
632 }
Ian Rogersef7d42f2014-01-06 12:55:46 -0800633 HeapReference<Object> old_ref(HeapReference<Object>::FromMirrorPtr(old_value));
634 HeapReference<Object> new_ref(HeapReference<Object>::FromMirrorPtr(new_value));
Ian Rogers228602f2014-07-10 02:07:54 -0700635 byte* raw_addr = reinterpret_cast<byte*>(this) + field_offset.Int32Value();
636 Atomic<uint32_t>* atomic_addr = reinterpret_cast<Atomic<uint32_t>*>(raw_addr);
637
638 bool success = atomic_addr->CompareExchangeWeakSequentiallyConsistent(old_ref.reference_,
639 new_ref.reference_);
640
Ian Rogersef7d42f2014-01-06 12:55:46 -0800641 if (success) {
642 Runtime::Current()->GetHeap()->WriteBarrierField(this, field_offset, new_value);
643 }
644 return success;
645}
646
Mathieu Chartier407f7022014-02-18 14:37:05 -0800647template<bool kVisitClass, bool kIsStatic, typename Visitor>
648inline void Object::VisitFieldsReferences(uint32_t ref_offsets, const Visitor& visitor) {
649 if (LIKELY(ref_offsets != CLASS_WALK_SUPER)) {
650 if (!kVisitClass) {
651 // Mask out the class from the reference offsets.
652 ref_offsets ^= kWordHighBitMask;
653 }
654 DCHECK_EQ(ClassOffset().Uint32Value(), 0U);
655 // Found a reference offset bitmap. Visit the specified offsets.
656 while (ref_offsets != 0) {
657 size_t right_shift = CLZ(ref_offsets);
658 MemberOffset field_offset = CLASS_OFFSET_FROM_CLZ(right_shift);
659 visitor(this, field_offset, kIsStatic);
660 ref_offsets &= ~(CLASS_HIGH_BIT >> right_shift);
661 }
662 } else {
663 // There is no reference offset bitmap. In the non-static case, walk up the class
664 // inheritance hierarchy and find reference offsets the hard way. In the static case, just
665 // consider this class.
666 for (mirror::Class* klass = kIsStatic ? AsClass() : GetClass(); klass != nullptr;
667 klass = kIsStatic ? nullptr : klass->GetSuperClass()) {
668 size_t num_reference_fields =
669 kIsStatic ? klass->NumReferenceStaticFields() : klass->NumReferenceInstanceFields();
670 for (size_t i = 0; i < num_reference_fields; ++i) {
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700671 mirror::ArtField* field = kIsStatic ? klass->GetStaticField(i) : klass->GetInstanceField(i);
Mathieu Chartier407f7022014-02-18 14:37:05 -0800672 MemberOffset field_offset = field->GetOffset();
673 // TODO: Do a simpler check?
Mathieu Chartier52e4b432014-06-10 11:22:31 -0700674 if (kVisitClass || field_offset.Uint32Value() != ClassOffset().Uint32Value()) {
675 visitor(this, field_offset, kIsStatic);
Mathieu Chartier407f7022014-02-18 14:37:05 -0800676 }
Mathieu Chartier407f7022014-02-18 14:37:05 -0800677 }
678 }
679 }
680}
681
682template<bool kVisitClass, typename Visitor>
683inline void Object::VisitInstanceFieldsReferences(mirror::Class* klass, const Visitor& visitor) {
684 VisitFieldsReferences<kVisitClass, false>(
685 klass->GetReferenceInstanceOffsets<kVerifyNone>(), visitor);
686}
687
688template<bool kVisitClass, typename Visitor>
689inline void Object::VisitStaticFieldsReferences(mirror::Class* klass, const Visitor& visitor) {
690 klass->VisitFieldsReferences<kVisitClass, true>(
691 klass->GetReferenceStaticOffsets<kVerifyNone>(), visitor);
692}
693
694template <const bool kVisitClass, VerifyObjectFlags kVerifyFlags, typename Visitor,
695 typename JavaLangRefVisitor>
696inline void Object::VisitReferences(const Visitor& visitor,
697 const JavaLangRefVisitor& ref_visitor) {
698 mirror::Class* klass = GetClass<kVerifyFlags>();
Mathieu Chartier52e4b432014-06-10 11:22:31 -0700699 if (klass == Class::GetJavaLangClass()) {
700 AsClass<kVerifyNone>()->VisitReferences<kVisitClass>(klass, visitor);
701 } else if (klass->IsArrayClass()) {
702 if (klass->IsObjectArrayClass<kVerifyNone>()) {
703 AsObjectArray<mirror::Object, kVerifyNone>()->VisitReferences<kVisitClass>(visitor);
704 } else if (kVisitClass) {
705 visitor(this, ClassOffset(), false);
Mathieu Chartier407f7022014-02-18 14:37:05 -0800706 }
707 } else {
Mathieu Chartier52e4b432014-06-10 11:22:31 -0700708 DCHECK(!klass->IsVariableSize());
Mathieu Chartier580a8df2014-03-26 15:15:57 -0700709 VisitInstanceFieldsReferences<kVisitClass>(klass, visitor);
710 if (UNLIKELY(klass->IsReferenceClass<kVerifyNone>())) {
Mathieu Chartier407f7022014-02-18 14:37:05 -0800711 ref_visitor(klass, AsReference());
712 }
713 }
714}
715
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800716} // namespace mirror
717} // namespace art
718
Brian Carlstromfc0e3212013-07-17 14:40:12 -0700719#endif // ART_RUNTIME_MIRROR_OBJECT_INL_H_