blob: a5aa0d08fedcc5e4547782eb01c3fa148599f7c4 [file] [log] [blame]
Elliott Hughes2faa5f12012-01-30 14:42:07 -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 */
Carl Shapiro0e5d75d2011-07-06 18:28:37 -070016
Brian Carlstrom578bbdc2011-07-21 14:07:47 -070017#include "class_linker.h"
Elliott Hughes90a33692011-08-30 13:27:07 -070018
Ian Rogers700a4022014-05-19 16:49:03 -070019#include <memory>
Ian Rogers0cfe1fb2011-08-26 03:29:44 -070020#include <string>
21
Mathieu Chartierc7853442015-03-27 14:35:38 -070022#include "art_field-inl.h"
Mathieu Chartiere401d142015-04-22 13:56:20 -070023#include "art_method-inl.h"
Andreas Gampe542451c2016-07-26 09:02:02 -070024#include "base/enums.h"
Ian Rogers2dd0e2c2013-01-24 12:42:14 -080025#include "class_linker-inl.h"
Brian Carlstroma1ce1fe2014-02-24 23:23:58 -080026#include "common_runtime_test.h"
Brian Carlstrom578bbdc2011-07-21 14:07:47 -070027#include "dex_file.h"
Alex Light705ad492015-09-21 11:36:30 -070028#include "experimental_flags.h"
Mingyao Yang98d1cc82014-05-15 17:02:16 -070029#include "entrypoints/entrypoint_utils-inl.h"
Ian Rogers1d54e732013-05-02 21:10:01 -070030#include "gc/heap.h"
Mathieu Chartierdaaf3262015-03-24 13:30:28 -070031#include "mirror/accessible_object.h"
Ian Rogers2dd0e2c2013-01-24 12:42:14 -080032#include "mirror/class-inl.h"
33#include "mirror/dex_cache.h"
Neil Fuller16b21cd2016-08-12 09:37:02 +010034#include "mirror/executable.h"
Mathieu Chartierdaaf3262015-03-24 13:30:28 -070035#include "mirror/field.h"
Narayan Kamathafa48272016-08-03 12:46:58 +010036#include "mirror/method_type.h"
37#include "mirror/method_handle_impl.h"
Ian Rogers2dd0e2c2013-01-24 12:42:14 -080038#include "mirror/object-inl.h"
39#include "mirror/object_array-inl.h"
40#include "mirror/proxy.h"
Mathieu Chartier8fa2dad2014-03-13 12:22:56 -070041#include "mirror/reference.h"
Ian Rogers2dd0e2c2013-01-24 12:42:14 -080042#include "mirror/stack_trace_element.h"
Mingyao Yang98d1cc82014-05-15 17:02:16 -070043#include "mirror/string-inl.h"
Mathieu Chartiereb8167a2014-05-07 15:43:14 -070044#include "handle_scope-inl.h"
Mathieu Chartier0795f232016-09-27 18:43:30 -070045#include "scoped_thread_state_change-inl.h"
Ian Rogerse63db272014-07-15 15:36:11 -070046#include "thread-inl.h"
Carl Shapiro0e5d75d2011-07-06 18:28:37 -070047
48namespace art {
49
Brian Carlstroma1ce1fe2014-02-24 23:23:58 -080050class ClassLinkerTest : public CommonRuntimeTest {
Brian Carlstrom578bbdc2011-07-21 14:07:47 -070051 protected:
Ian Rogers00f7d0e2012-07-19 15:28:27 -070052 void AssertNonExistentClass(const std::string& descriptor)
Andreas Gampebdf7f1c2016-08-30 16:38:47 -070053 REQUIRES_SHARED(Locks::mutator_lock_) {
Elliott Hughes885c3bd2011-08-22 16:59:20 -070054 Thread* self = Thread::Current();
Ian Rogerscdc1aaf2014-10-09 13:21:38 -070055 EXPECT_TRUE(class_linker_->FindSystemClass(self, descriptor.c_str()) == nullptr);
Elliott Hughes885c3bd2011-08-22 16:59:20 -070056 EXPECT_TRUE(self->IsExceptionPending());
Nicolas Geoffray14691c52015-03-05 10:40:17 +000057 mirror::Object* exception = self->GetException();
Elliott Hughes885c3bd2011-08-22 16:59:20 -070058 self->ClearException();
Ian Rogers98379392014-02-24 16:53:16 -080059 mirror::Class* exception_class =
60 class_linker_->FindSystemClass(self, "Ljava/lang/NoClassDefFoundError;");
Elliott Hughes885c3bd2011-08-22 16:59:20 -070061 EXPECT_TRUE(exception->InstanceOf(exception_class));
Brian Carlstrom578bbdc2011-07-21 14:07:47 -070062 }
63
Ian Rogers00f7d0e2012-07-19 15:28:27 -070064 void AssertPrimitiveClass(const std::string& descriptor)
Andreas Gampebdf7f1c2016-08-30 16:38:47 -070065 REQUIRES_SHARED(Locks::mutator_lock_) {
Ian Rogers98379392014-02-24 16:53:16 -080066 Thread* self = Thread::Current();
67 AssertPrimitiveClass(descriptor, class_linker_->FindSystemClass(self, descriptor.c_str()));
Brian Carlstromd2fbb2b2011-08-23 11:57:08 -070068 }
69
Ian Rogersef7d42f2014-01-06 12:55:46 -080070 void AssertPrimitiveClass(const std::string& descriptor, mirror::Class* primitive)
Andreas Gampebdf7f1c2016-08-30 16:38:47 -070071 REQUIRES_SHARED(Locks::mutator_lock_) {
Ian Rogerscdc1aaf2014-10-09 13:21:38 -070072 ASSERT_TRUE(primitive != nullptr);
73 ASSERT_TRUE(primitive->GetClass() != nullptr);
Brian Carlstrom578bbdc2011-07-21 14:07:47 -070074 ASSERT_EQ(primitive->GetClass(), primitive->GetClass()->GetClass());
Ian Rogerscdc1aaf2014-10-09 13:21:38 -070075 EXPECT_TRUE(primitive->GetClass()->GetSuperClass() != nullptr);
Ian Rogers1ff3c982014-08-12 02:30:58 -070076 std::string temp;
77 ASSERT_STREQ(descriptor.c_str(), primitive->GetDescriptor(&temp));
Ian Rogerscdc1aaf2014-10-09 13:21:38 -070078 EXPECT_TRUE(primitive->GetSuperClass() == nullptr);
Brian Carlstrom578bbdc2011-07-21 14:07:47 -070079 EXPECT_FALSE(primitive->HasSuperClass());
Ian Rogerscdc1aaf2014-10-09 13:21:38 -070080 EXPECT_TRUE(primitive->GetClassLoader() == nullptr);
Brian Carlstromea46f952013-07-30 01:26:50 -070081 EXPECT_EQ(mirror::Class::kStatusInitialized, primitive->GetStatus());
Brian Carlstrom578bbdc2011-07-21 14:07:47 -070082 EXPECT_FALSE(primitive->IsErroneous());
Brian Carlstrom25c33252011-09-18 15:58:35 -070083 EXPECT_TRUE(primitive->IsLoaded());
Elliott Hughes5fe594f2011-09-08 12:33:17 -070084 EXPECT_TRUE(primitive->IsResolved());
Brian Carlstrom25c33252011-09-18 15:58:35 -070085 EXPECT_TRUE(primitive->IsVerified());
86 EXPECT_TRUE(primitive->IsInitialized());
Brian Carlstromb63ec392011-08-27 17:38:27 -070087 EXPECT_FALSE(primitive->IsArrayInstance());
88 EXPECT_FALSE(primitive->IsArrayClass());
Ian Rogerscdc1aaf2014-10-09 13:21:38 -070089 EXPECT_TRUE(primitive->GetComponentType() == nullptr);
Brian Carlstrom578bbdc2011-07-21 14:07:47 -070090 EXPECT_FALSE(primitive->IsInterface());
91 EXPECT_TRUE(primitive->IsPublic());
92 EXPECT_TRUE(primitive->IsFinal());
93 EXPECT_TRUE(primitive->IsPrimitive());
Brian Carlstrom75cb3b42011-07-28 02:13:36 -070094 EXPECT_FALSE(primitive->IsSynthetic());
Brian Carlstrom578bbdc2011-07-21 14:07:47 -070095 EXPECT_EQ(0U, primitive->NumDirectMethods());
96 EXPECT_EQ(0U, primitive->NumVirtualMethods());
97 EXPECT_EQ(0U, primitive->NumInstanceFields());
98 EXPECT_EQ(0U, primitive->NumStaticFields());
Mathieu Chartierf8322842014-05-16 10:59:25 -070099 EXPECT_EQ(0U, primitive->NumDirectInterfaces());
Mingyao Yang2cdbad72014-07-16 10:44:41 -0700100 EXPECT_FALSE(primitive->HasVTable());
Brian Carlstrom86927212011-09-15 11:31:11 -0700101 EXPECT_EQ(0, primitive->GetIfTableCount());
Ian Rogerscdc1aaf2014-10-09 13:21:38 -0700102 EXPECT_TRUE(primitive->GetIfTable() == nullptr);
Elliott Hughes00626c22013-06-14 15:04:14 -0700103 EXPECT_EQ(kAccPublic | kAccFinal | kAccAbstract, primitive->GetAccessFlags());
Brian Carlstrom578bbdc2011-07-21 14:07:47 -0700104 }
105
Artem Udovichenkoa62cb9b2016-06-30 09:18:25 +0000106 void AssertObjectClass(mirror::Class* JavaLangObject)
Andreas Gampebdf7f1c2016-08-30 16:38:47 -0700107 REQUIRES_SHARED(Locks::mutator_lock_) {
Artem Udovichenkoa62cb9b2016-06-30 09:18:25 +0000108 ASSERT_TRUE(JavaLangObject != nullptr);
109 ASSERT_TRUE(JavaLangObject->GetClass() != nullptr);
110 ASSERT_EQ(JavaLangObject->GetClass(),
111 JavaLangObject->GetClass()->GetClass());
112 EXPECT_EQ(JavaLangObject, JavaLangObject->GetClass()->GetSuperClass());
113 std::string temp;
114 ASSERT_STREQ(JavaLangObject->GetDescriptor(&temp), "Ljava/lang/Object;");
115 EXPECT_TRUE(JavaLangObject->GetSuperClass() == nullptr);
116 EXPECT_FALSE(JavaLangObject->HasSuperClass());
117 EXPECT_TRUE(JavaLangObject->GetClassLoader() == nullptr);
118 EXPECT_EQ(mirror::Class::kStatusInitialized, JavaLangObject->GetStatus());
119 EXPECT_FALSE(JavaLangObject->IsErroneous());
120 EXPECT_TRUE(JavaLangObject->IsLoaded());
121 EXPECT_TRUE(JavaLangObject->IsResolved());
122 EXPECT_TRUE(JavaLangObject->IsVerified());
123 EXPECT_TRUE(JavaLangObject->IsInitialized());
124 EXPECT_FALSE(JavaLangObject->IsArrayInstance());
125 EXPECT_FALSE(JavaLangObject->IsArrayClass());
126 EXPECT_TRUE(JavaLangObject->GetComponentType() == nullptr);
127 EXPECT_FALSE(JavaLangObject->IsInterface());
128 EXPECT_TRUE(JavaLangObject->IsPublic());
129 EXPECT_FALSE(JavaLangObject->IsFinal());
130 EXPECT_FALSE(JavaLangObject->IsPrimitive());
131 EXPECT_FALSE(JavaLangObject->IsSynthetic());
132 EXPECT_EQ(2U, JavaLangObject->NumDirectMethods());
133 EXPECT_EQ(11U, JavaLangObject->NumVirtualMethods());
134 if (!kUseBrooksReadBarrier) {
135 EXPECT_EQ(2U, JavaLangObject->NumInstanceFields());
136 } else {
137 EXPECT_EQ(4U, JavaLangObject->NumInstanceFields());
138 }
139 EXPECT_STREQ(JavaLangObject->GetInstanceField(0)->GetName(),
140 "shadow$_klass_");
141 EXPECT_STREQ(JavaLangObject->GetInstanceField(1)->GetName(),
142 "shadow$_monitor_");
143 if (kUseBrooksReadBarrier) {
144 EXPECT_STREQ(JavaLangObject->GetInstanceField(2)->GetName(),
145 "shadow$_x_rb_ptr_");
146 EXPECT_STREQ(JavaLangObject->GetInstanceField(3)->GetName(),
147 "shadow$_x_xpadding_");
148 }
149
150 EXPECT_EQ(0U, JavaLangObject->NumStaticFields());
151 EXPECT_EQ(0U, JavaLangObject->NumDirectInterfaces());
152
Andreas Gampe542451c2016-07-26 09:02:02 -0700153 PointerSize pointer_size = class_linker_->GetImagePointerSize();
Artem Udovichenkoa62cb9b2016-06-30 09:18:25 +0000154 ArtMethod* unimplemented = runtime_->GetImtUnimplementedMethod();
155 ImTable* imt = JavaLangObject->GetImt(pointer_size);
156 ASSERT_NE(nullptr, imt);
157 for (size_t i = 0; i < ImTable::kSize; ++i) {
158 ASSERT_EQ(unimplemented, imt->Get(i, pointer_size));
159 }
160 }
161
Brian Carlstromaded5f72011-10-07 17:15:04 -0700162 void AssertArrayClass(const std::string& array_descriptor,
163 const std::string& component_type,
Brian Carlstromea46f952013-07-30 01:26:50 -0700164 mirror::ClassLoader* class_loader)
Andreas Gampebdf7f1c2016-08-30 16:38:47 -0700165 REQUIRES_SHARED(Locks::mutator_lock_) {
Mathieu Chartierc528dba2013-11-26 12:00:11 -0800166 Thread* self = Thread::Current();
Mathieu Chartiereb8167a2014-05-07 15:43:14 -0700167 StackHandleScope<2> hs(self);
168 Handle<mirror::ClassLoader> loader(hs.NewHandle(class_loader));
169 Handle<mirror::Class> array(
170 hs.NewHandle(class_linker_->FindClass(self, array_descriptor.c_str(), loader)));
Ian Rogers1ff3c982014-08-12 02:30:58 -0700171 std::string temp;
172 EXPECT_STREQ(component_type.c_str(), array->GetComponentType()->GetDescriptor(&temp));
Brian Carlstromd2fbb2b2011-08-23 11:57:08 -0700173 EXPECT_EQ(class_loader, array->GetClassLoader());
Elliott Hughes00626c22013-06-14 15:04:14 -0700174 EXPECT_EQ(kAccFinal | kAccAbstract, (array->GetAccessFlags() & (kAccFinal | kAccAbstract)));
Brian Carlstromd2fbb2b2011-08-23 11:57:08 -0700175 AssertArrayClass(array_descriptor, array);
176 }
177
Mathieu Chartier0cd81352014-05-22 16:48:55 -0700178 void AssertArrayClass(const std::string& array_descriptor, Handle<mirror::Class> array)
Andreas Gampebdf7f1c2016-08-30 16:38:47 -0700179 REQUIRES_SHARED(Locks::mutator_lock_) {
Ian Rogerscdc1aaf2014-10-09 13:21:38 -0700180 ASSERT_TRUE(array.Get() != nullptr);
181 ASSERT_TRUE(array->GetClass() != nullptr);
Brian Carlstrom578bbdc2011-07-21 14:07:47 -0700182 ASSERT_EQ(array->GetClass(), array->GetClass()->GetClass());
Ian Rogerscdc1aaf2014-10-09 13:21:38 -0700183 EXPECT_TRUE(array->GetClass()->GetSuperClass() != nullptr);
Ian Rogers1ff3c982014-08-12 02:30:58 -0700184 std::string temp;
185 ASSERT_STREQ(array_descriptor.c_str(), array->GetDescriptor(&temp));
Ian Rogerscdc1aaf2014-10-09 13:21:38 -0700186 EXPECT_TRUE(array->GetSuperClass() != nullptr);
Ian Rogers98379392014-02-24 16:53:16 -0800187 Thread* self = Thread::Current();
188 EXPECT_EQ(class_linker_->FindSystemClass(self, "Ljava/lang/Object;"), array->GetSuperClass());
Brian Carlstrom578bbdc2011-07-21 14:07:47 -0700189 EXPECT_TRUE(array->HasSuperClass());
Ian Rogerscdc1aaf2014-10-09 13:21:38 -0700190 ASSERT_TRUE(array->GetComponentType() != nullptr);
Ian Rogers1ff3c982014-08-12 02:30:58 -0700191 ASSERT_GT(strlen(array->GetComponentType()->GetDescriptor(&temp)), 0U);
Brian Carlstromea46f952013-07-30 01:26:50 -0700192 EXPECT_EQ(mirror::Class::kStatusInitialized, array->GetStatus());
Brian Carlstrom578bbdc2011-07-21 14:07:47 -0700193 EXPECT_FALSE(array->IsErroneous());
Brian Carlstrom25c33252011-09-18 15:58:35 -0700194 EXPECT_TRUE(array->IsLoaded());
Elliott Hughes5fe594f2011-09-08 12:33:17 -0700195 EXPECT_TRUE(array->IsResolved());
Brian Carlstrom25c33252011-09-18 15:58:35 -0700196 EXPECT_TRUE(array->IsVerified());
197 EXPECT_TRUE(array->IsInitialized());
Brian Carlstromb63ec392011-08-27 17:38:27 -0700198 EXPECT_FALSE(array->IsArrayInstance());
199 EXPECT_TRUE(array->IsArrayClass());
Brian Carlstrom578bbdc2011-07-21 14:07:47 -0700200 EXPECT_FALSE(array->IsInterface());
201 EXPECT_EQ(array->GetComponentType()->IsPublic(), array->IsPublic());
202 EXPECT_TRUE(array->IsFinal());
203 EXPECT_FALSE(array->IsPrimitive());
Brian Carlstrom75cb3b42011-07-28 02:13:36 -0700204 EXPECT_FALSE(array->IsSynthetic());
Brian Carlstrom578bbdc2011-07-21 14:07:47 -0700205 EXPECT_EQ(0U, array->NumDirectMethods());
206 EXPECT_EQ(0U, array->NumVirtualMethods());
207 EXPECT_EQ(0U, array->NumInstanceFields());
208 EXPECT_EQ(0U, array->NumStaticFields());
Mathieu Chartierf8322842014-05-16 10:59:25 -0700209 EXPECT_EQ(2U, array->NumDirectInterfaces());
Artem Udovichenkoa62cb9b2016-06-30 09:18:25 +0000210 EXPECT_TRUE(array->ShouldHaveImt());
211 EXPECT_TRUE(array->ShouldHaveEmbeddedVTable());
Brian Carlstrom86927212011-09-15 11:31:11 -0700212 EXPECT_EQ(2, array->GetIfTableCount());
Ian Rogerscdc1aaf2014-10-09 13:21:38 -0700213 ASSERT_TRUE(array->GetIfTable() != nullptr);
Mathieu Chartierf8322842014-05-16 10:59:25 -0700214 mirror::Class* direct_interface0 = mirror::Class::GetDirectInterface(self, array, 0);
215 EXPECT_TRUE(direct_interface0 != nullptr);
Ian Rogers1ff3c982014-08-12 02:30:58 -0700216 EXPECT_STREQ(direct_interface0->GetDescriptor(&temp), "Ljava/lang/Cloneable;");
Mathieu Chartierf8322842014-05-16 10:59:25 -0700217 mirror::Class* direct_interface1 = mirror::Class::GetDirectInterface(self, array, 1);
Ian Rogers1ff3c982014-08-12 02:30:58 -0700218 EXPECT_STREQ(direct_interface1->GetDescriptor(&temp), "Ljava/io/Serializable;");
Mathieu Chartierb74cd292014-05-29 14:31:33 -0700219 mirror::Class* array_ptr = array->GetComponentType();
220 EXPECT_EQ(class_linker_->FindArrayClass(self, &array_ptr), array.Get());
Artem Udovichenkoa62cb9b2016-06-30 09:18:25 +0000221
Andreas Gampe542451c2016-07-26 09:02:02 -0700222 PointerSize pointer_size = class_linker_->GetImagePointerSize();
Artem Udovichenkoa62cb9b2016-06-30 09:18:25 +0000223 mirror::Class* JavaLangObject =
224 class_linker_->FindSystemClass(self, "Ljava/lang/Object;");
225 ImTable* JavaLangObject_imt = JavaLangObject->GetImt(pointer_size);
226 // IMT of a array class should be shared with the IMT of the java.lag.Object
227 ASSERT_EQ(JavaLangObject_imt, array->GetImt(pointer_size));
Brian Carlstrom578bbdc2011-07-21 14:07:47 -0700228 }
Brian Carlstrom4a96b602011-07-26 16:40:23 -0700229
Andreas Gampebdf7f1c2016-08-30 16:38:47 -0700230 void AssertMethod(ArtMethod* method) REQUIRES_SHARED(Locks::mutator_lock_) {
Andreas Gampe58a5af82014-07-31 16:23:49 -0700231 EXPECT_TRUE(method != nullptr);
Mathieu Chartiere401d142015-04-22 13:56:20 -0700232 EXPECT_TRUE(method->GetDeclaringClass() != nullptr);
Andreas Gampe58a5af82014-07-31 16:23:49 -0700233 EXPECT_TRUE(method->GetName() != nullptr);
Mathieu Chartierbfd9a432014-05-21 17:43:44 -0700234 EXPECT_TRUE(method->GetSignature() != Signature::NoSignature());
Brian Carlstromc4fa2c02011-08-21 03:00:12 -0700235
Andreas Gampe542451c2016-07-26 09:02:02 -0700236 EXPECT_TRUE(method->HasDexCacheResolvedMethods(kRuntimePointerSize));
237 EXPECT_TRUE(method->HasDexCacheResolvedTypes(kRuntimePointerSize));
Andreas Gampe58a5af82014-07-31 16:23:49 -0700238 EXPECT_TRUE(method->HasSameDexCacheResolvedMethods(
Vladimir Marko05792b92015-08-03 11:56:49 +0100239 method->GetDeclaringClass()->GetDexCache()->GetResolvedMethods(),
Andreas Gampe542451c2016-07-26 09:02:02 -0700240 kRuntimePointerSize));
Andreas Gampe58a5af82014-07-31 16:23:49 -0700241 EXPECT_TRUE(method->HasSameDexCacheResolvedTypes(
Vladimir Marko05792b92015-08-03 11:56:49 +0100242 method->GetDeclaringClass()->GetDexCache()->GetResolvedTypes(),
Andreas Gampe542451c2016-07-26 09:02:02 -0700243 kRuntimePointerSize));
Brian Carlstromc4fa2c02011-08-21 03:00:12 -0700244 }
245
Mathieu Chartierc7853442015-03-27 14:35:38 -0700246 void AssertField(mirror::Class* klass, ArtField* field)
Andreas Gampebdf7f1c2016-08-30 16:38:47 -0700247 REQUIRES_SHARED(Locks::mutator_lock_) {
Ian Rogerscdc1aaf2014-10-09 13:21:38 -0700248 EXPECT_TRUE(field != nullptr);
Brian Carlstromc4fa2c02011-08-21 03:00:12 -0700249 EXPECT_EQ(klass, field->GetDeclaringClass());
Ian Rogerscdc1aaf2014-10-09 13:21:38 -0700250 EXPECT_TRUE(field->GetName() != nullptr);
Mathieu Chartierdaaf3262015-03-24 13:30:28 -0700251 EXPECT_TRUE(field->GetType<true>() != nullptr);
Brian Carlstromc4fa2c02011-08-21 03:00:12 -0700252 }
253
Mathieu Chartier0cd81352014-05-22 16:48:55 -0700254 void AssertClass(const std::string& descriptor, Handle<mirror::Class> klass)
Andreas Gampebdf7f1c2016-08-30 16:38:47 -0700255 REQUIRES_SHARED(Locks::mutator_lock_) {
Ian Rogers1ff3c982014-08-12 02:30:58 -0700256 std::string temp;
257 EXPECT_STREQ(descriptor.c_str(), klass->GetDescriptor(&temp));
Ian Rogers6d4d9fc2011-11-30 16:24:48 -0800258 if (descriptor == "Ljava/lang/Object;") {
Brian Carlstrom74eb46a2011-08-02 20:10:14 -0700259 EXPECT_FALSE(klass->HasSuperClass());
Brian Carlstromae3ac012011-07-27 01:30:28 -0700260 } else {
Brian Carlstrom74eb46a2011-08-02 20:10:14 -0700261 EXPECT_TRUE(klass->HasSuperClass());
Ian Rogerscdc1aaf2014-10-09 13:21:38 -0700262 EXPECT_TRUE(klass->GetSuperClass() != nullptr);
Brian Carlstromae3ac012011-07-27 01:30:28 -0700263 }
Ian Rogerscdc1aaf2014-10-09 13:21:38 -0700264 EXPECT_TRUE(klass->GetClass() != nullptr);
Brian Carlstroma5a97a22011-09-15 14:08:49 -0700265 EXPECT_EQ(klass->GetClass(), klass->GetClass()->GetClass());
Ian Rogerscdc1aaf2014-10-09 13:21:38 -0700266 EXPECT_TRUE(klass->GetDexCache() != nullptr);
Brian Carlstrom25c33252011-09-18 15:58:35 -0700267 EXPECT_TRUE(klass->IsLoaded());
Elliott Hughes5fe594f2011-09-08 12:33:17 -0700268 EXPECT_TRUE(klass->IsResolved());
Brian Carlstromae3ac012011-07-27 01:30:28 -0700269 EXPECT_FALSE(klass->IsErroneous());
Ian Rogers0cfe1fb2011-08-26 03:29:44 -0700270 EXPECT_FALSE(klass->IsArrayClass());
Ian Rogerscdc1aaf2014-10-09 13:21:38 -0700271 EXPECT_TRUE(klass->GetComponentType() == nullptr);
Mathieu Chartiereb8167a2014-05-07 15:43:14 -0700272 EXPECT_TRUE(klass->IsInSamePackage(klass.Get()));
Mathieu Chartiereace4582014-11-24 18:29:54 -0800273 EXPECT_TRUE(klass->GetDexCacheStrings() != nullptr);
274 EXPECT_EQ(klass->GetDexCacheStrings(), klass->GetDexCache()->GetStrings());
Ian Rogers1ff3c982014-08-12 02:30:58 -0700275 std::string temp2;
276 EXPECT_TRUE(mirror::Class::IsInSamePackage(klass->GetDescriptor(&temp),
277 klass->GetDescriptor(&temp2)));
Brian Carlstromae3ac012011-07-27 01:30:28 -0700278 if (klass->IsInterface()) {
Brian Carlstrom74eb46a2011-08-02 20:10:14 -0700279 EXPECT_TRUE(klass->IsAbstract());
Alex Light42d511d2016-02-03 13:45:41 -0800280 // Check that all direct methods are static (either <clinit> or a regular static method).
Andreas Gampe542451c2016-07-26 09:02:02 -0700281 for (ArtMethod& m : klass->GetDirectMethods(kRuntimePointerSize)) {
Alex Light42d511d2016-02-03 13:45:41 -0800282 EXPECT_TRUE(m.IsStatic());
283 EXPECT_TRUE(m.IsDirect());
Brian Carlstrom74eb46a2011-08-02 20:10:14 -0700284 }
Brian Carlstromae3ac012011-07-27 01:30:28 -0700285 } else {
Brian Carlstrom74eb46a2011-08-02 20:10:14 -0700286 if (!klass->IsSynthetic()) {
287 EXPECT_NE(0U, klass->NumDirectMethods());
288 }
Brian Carlstromae3ac012011-07-27 01:30:28 -0700289 }
Mingyao Yang2cdbad72014-07-16 10:44:41 -0700290 EXPECT_EQ(klass->IsInterface(), !klass->HasVTable());
Ian Rogersef7d42f2014-01-06 12:55:46 -0800291 mirror::IfTable* iftable = klass->GetIfTable();
Brian Carlstrom86927212011-09-15 11:31:11 -0700292 for (int i = 0; i < klass->GetIfTableCount(); i++) {
Brian Carlstromea46f952013-07-30 01:26:50 -0700293 mirror::Class* interface = iftable->GetInterface(i);
Ian Rogerscdc1aaf2014-10-09 13:21:38 -0700294 ASSERT_TRUE(interface != nullptr);
Brian Carlstrom86927212011-09-15 11:31:11 -0700295 if (klass->IsInterface()) {
Ian Rogers9bc81912012-10-11 21:43:36 -0700296 EXPECT_EQ(0U, iftable->GetMethodArrayCount(i));
Brian Carlstrom86927212011-09-15 11:31:11 -0700297 } else {
Alex Light705ad492015-09-21 11:36:30 -0700298 EXPECT_EQ(interface->NumDeclaredVirtualMethods(), iftable->GetMethodArrayCount(i));
Brian Carlstrom86927212011-09-15 11:31:11 -0700299 }
300 }
Brian Carlstromae3ac012011-07-27 01:30:28 -0700301 if (klass->IsAbstract()) {
Brian Carlstrom74eb46a2011-08-02 20:10:14 -0700302 EXPECT_FALSE(klass->IsFinal());
Brian Carlstromae3ac012011-07-27 01:30:28 -0700303 } else {
Brian Carlstrom74eb46a2011-08-02 20:10:14 -0700304 EXPECT_FALSE(klass->IsAnnotation());
Brian Carlstromae3ac012011-07-27 01:30:28 -0700305 }
306 if (klass->IsFinal()) {
Brian Carlstrom74eb46a2011-08-02 20:10:14 -0700307 EXPECT_FALSE(klass->IsAbstract());
308 EXPECT_FALSE(klass->IsAnnotation());
Brian Carlstromae3ac012011-07-27 01:30:28 -0700309 }
310 if (klass->IsAnnotation()) {
Brian Carlstrom74eb46a2011-08-02 20:10:14 -0700311 EXPECT_FALSE(klass->IsFinal());
312 EXPECT_TRUE(klass->IsAbstract());
Brian Carlstromae3ac012011-07-27 01:30:28 -0700313 }
314
315 EXPECT_FALSE(klass->IsPrimitive());
Mathieu Chartiereb8167a2014-05-07 15:43:14 -0700316 EXPECT_TRUE(klass->CanAccess(klass.Get()));
Brian Carlstromae3ac012011-07-27 01:30:28 -0700317
Andreas Gampe542451c2016-07-26 09:02:02 -0700318 for (ArtMethod& method : klass->GetDirectMethods(kRuntimePointerSize)) {
Mathieu Chartiere401d142015-04-22 13:56:20 -0700319 AssertMethod(&method);
320 EXPECT_TRUE(method.IsDirect());
321 EXPECT_EQ(klass.Get(), method.GetDeclaringClass());
Brian Carlstromae3ac012011-07-27 01:30:28 -0700322 }
323
Andreas Gampe542451c2016-07-26 09:02:02 -0700324 for (ArtMethod& method : klass->GetDeclaredVirtualMethods(kRuntimePointerSize)) {
Mathieu Chartiere401d142015-04-22 13:56:20 -0700325 AssertMethod(&method);
326 EXPECT_FALSE(method.IsDirect());
Alex Lighte64300b2015-12-15 15:02:47 -0800327 EXPECT_EQ(klass.Get(), method.GetDeclaringClass());
328 }
329
Andreas Gampe542451c2016-07-26 09:02:02 -0700330 for (ArtMethod& method : klass->GetCopiedMethods(kRuntimePointerSize)) {
Alex Lighte64300b2015-12-15 15:02:47 -0800331 AssertMethod(&method);
332 EXPECT_FALSE(method.IsDirect());
Alex Light36121492016-02-22 13:43:29 -0800333 EXPECT_TRUE(method.IsCopied());
Alex Lighte64300b2015-12-15 15:02:47 -0800334 EXPECT_TRUE(method.GetDeclaringClass()->IsInterface())
335 << "declaring class: " << PrettyClass(method.GetDeclaringClass());
336 EXPECT_TRUE(method.GetDeclaringClass()->IsAssignableFrom(klass.Get()))
337 << "declaring class: " << PrettyClass(method.GetDeclaringClass());
Brian Carlstromae3ac012011-07-27 01:30:28 -0700338 }
339
340 for (size_t i = 0; i < klass->NumInstanceFields(); i++) {
Mathieu Chartierc7853442015-03-27 14:35:38 -0700341 ArtField* field = klass->GetInstanceField(i);
Mathieu Chartiereb8167a2014-05-07 15:43:14 -0700342 AssertField(klass.Get(), field);
Jesse Wilsonfd687c52011-08-04 19:27:35 -0700343 EXPECT_FALSE(field->IsStatic());
Brian Carlstromae3ac012011-07-27 01:30:28 -0700344 }
345
346 for (size_t i = 0; i < klass->NumStaticFields(); i++) {
Mathieu Chartierc7853442015-03-27 14:35:38 -0700347 ArtField* field = klass->GetStaticField(i);
Mathieu Chartiereb8167a2014-05-07 15:43:14 -0700348 AssertField(klass.Get(), field);
Jesse Wilsonfd687c52011-08-04 19:27:35 -0700349 EXPECT_TRUE(field->IsStatic());
Elliott Hughes362f9bc2011-10-17 18:56:41 -0700350 }
Brian Carlstromae3ac012011-07-27 01:30:28 -0700351
Vladimir Marko76649e82014-11-10 18:32:59 +0000352 // Confirm that all instances field offsets are packed together at the start.
Brian Carlstromae3ac012011-07-27 01:30:28 -0700353 EXPECT_GE(klass->NumInstanceFields(), klass->NumReferenceInstanceFields());
Vladimir Marko76649e82014-11-10 18:32:59 +0000354 MemberOffset start_ref_offset = klass->GetFirstReferenceInstanceFieldOffset();
355 MemberOffset end_ref_offset(start_ref_offset.Uint32Value() +
356 klass->NumReferenceInstanceFields() *
357 sizeof(mirror::HeapReference<mirror::Object>));
358 MemberOffset current_ref_offset = start_ref_offset;
359 for (size_t i = 0; i < klass->NumInstanceFields(); i++) {
Mathieu Chartierc7853442015-03-27 14:35:38 -0700360 ArtField* field = klass->GetInstanceField(i);
361 mirror::Class* field_type = field->GetType<true>();
Ian Rogerscdc1aaf2014-10-09 13:21:38 -0700362 ASSERT_TRUE(field_type != nullptr);
Vladimir Marko76649e82014-11-10 18:32:59 +0000363 if (!field->IsPrimitiveType()) {
364 ASSERT_TRUE(!field_type->IsPrimitive());
365 ASSERT_EQ(current_ref_offset.Uint32Value(), field->GetOffset().Uint32Value());
366 if (current_ref_offset.Uint32Value() == end_ref_offset.Uint32Value()) {
367 // While Reference.referent is not primitive, the ClassLinker
368 // treats it as such so that the garbage collector won't scan it.
Mathieu Chartierc7853442015-03-27 14:35:38 -0700369 EXPECT_EQ(PrettyField(field),
Vladimir Marko76649e82014-11-10 18:32:59 +0000370 "java.lang.Object java.lang.ref.Reference.referent");
371 } else {
372 current_ref_offset = MemberOffset(current_ref_offset.Uint32Value() +
373 sizeof(mirror::HeapReference<mirror::Object>));
374 }
375 } else {
376 if (field->GetOffset().Uint32Value() < end_ref_offset.Uint32Value()) {
377 // Shuffled before references.
378 ASSERT_LT(field->GetOffset().Uint32Value(), start_ref_offset.Uint32Value());
379 CHECK(!IsAligned<4>(field->GetOffset().Uint32Value()));
380 }
Brian Carlstromfbfdce62011-10-05 17:33:32 -0700381 }
Brian Carlstromae3ac012011-07-27 01:30:28 -0700382 }
Vladimir Marko76649e82014-11-10 18:32:59 +0000383 ASSERT_EQ(end_ref_offset.Uint32Value(), current_ref_offset.Uint32Value());
Brian Carlstromae3ac012011-07-27 01:30:28 -0700384
Ian Rogerscdc1aaf2014-10-09 13:21:38 -0700385 uint32_t total_num_reference_instance_fields = 0;
Mathieu Chartiereb8167a2014-05-07 15:43:14 -0700386 mirror::Class* k = klass.Get();
Ian Rogerscdc1aaf2014-10-09 13:21:38 -0700387 while (k != nullptr) {
Brian Carlstrom74eb46a2011-08-02 20:10:14 -0700388 total_num_reference_instance_fields += k->NumReferenceInstanceFields();
389 k = k->GetSuperClass();
Brian Carlstromae3ac012011-07-27 01:30:28 -0700390 }
Ian Rogerscdc1aaf2014-10-09 13:21:38 -0700391 EXPECT_GE(total_num_reference_instance_fields, 1U); // Should always have Object's class.
392 if (klass->GetReferenceInstanceOffsets() != mirror::Class::kClassWalkSuper) {
393 // The reference instance offsets have a bit set for each reference offset.
394 // +1 for Object's class.
395 EXPECT_EQ(static_cast<uint32_t>(POPCOUNT(klass->GetReferenceInstanceOffsets())) + 1,
396 total_num_reference_instance_fields);
397 }
Brian Carlstrom4a96b602011-07-26 16:40:23 -0700398 }
399
Brian Carlstromea46f952013-07-30 01:26:50 -0700400 void AssertDexFileClass(mirror::ClassLoader* class_loader, const std::string& descriptor)
Andreas Gampebdf7f1c2016-08-30 16:38:47 -0700401 REQUIRES_SHARED(Locks::mutator_lock_) {
Ian Rogerscdc1aaf2014-10-09 13:21:38 -0700402 ASSERT_TRUE(descriptor != nullptr);
Ian Rogers98379392014-02-24 16:53:16 -0800403 Thread* self = Thread::Current();
Mathieu Chartiereb8167a2014-05-07 15:43:14 -0700404 StackHandleScope<1> hs(self);
405 Handle<mirror::Class> klass(
406 hs.NewHandle(class_linker_->FindSystemClass(self, descriptor.c_str())));
407 ASSERT_TRUE(klass.Get() != nullptr);
Ian Rogers1ff3c982014-08-12 02:30:58 -0700408 std::string temp;
409 EXPECT_STREQ(descriptor.c_str(), klass.Get()->GetDescriptor(&temp));
Brian Carlstromd2fbb2b2011-08-23 11:57:08 -0700410 EXPECT_EQ(class_loader, klass->GetClassLoader());
411 if (klass->IsPrimitive()) {
Mathieu Chartiereb8167a2014-05-07 15:43:14 -0700412 AssertPrimitiveClass(descriptor, klass.Get());
Brian Carlstromb63ec392011-08-27 17:38:27 -0700413 } else if (klass->IsArrayClass()) {
Brian Carlstromd2fbb2b2011-08-23 11:57:08 -0700414 AssertArrayClass(descriptor, klass);
415 } else {
416 AssertClass(descriptor, klass);
417 }
Brian Carlstrom75cb3b42011-07-28 02:13:36 -0700418 }
419
Richard Uhlerfbef44d2014-12-23 09:48:51 -0800420 void AssertDexFile(const DexFile& dex, mirror::ClassLoader* class_loader)
Andreas Gampebdf7f1c2016-08-30 16:38:47 -0700421 REQUIRES_SHARED(Locks::mutator_lock_) {
Brian Carlstromd2fbb2b2011-08-23 11:57:08 -0700422 // Verify all the classes defined in this file
Richard Uhlerfbef44d2014-12-23 09:48:51 -0800423 for (size_t i = 0; i < dex.NumClassDefs(); i++) {
424 const DexFile::ClassDef& class_def = dex.GetClassDef(i);
425 const char* descriptor = dex.GetClassDescriptor(class_def);
Brian Carlstrom74eb46a2011-08-02 20:10:14 -0700426 AssertDexFileClass(class_loader, descriptor);
Brian Carlstrom4a96b602011-07-26 16:40:23 -0700427 }
Brian Carlstrom9ea1cb12011-08-24 23:18:18 -0700428 // Verify all the types referenced by this file
Richard Uhlerfbef44d2014-12-23 09:48:51 -0800429 for (size_t i = 0; i < dex.NumTypeIds(); i++) {
430 const DexFile::TypeId& type_id = dex.GetTypeId(i);
431 const char* descriptor = dex.GetTypeDescriptor(type_id);
Brian Carlstromd2fbb2b2011-08-23 11:57:08 -0700432 AssertDexFileClass(class_loader, descriptor);
433 }
Mathieu Chartierbb87e0f2015-04-03 11:21:55 -0700434 TestRootVisitor visitor;
435 class_linker_->VisitRoots(&visitor, kVisitRootFlagAllRoots);
Ian Rogers19846512012-02-24 11:42:47 -0800436 // Verify the dex cache has resolution methods in all resolved method slots
Mathieu Chartier673ed3d2015-08-28 14:56:43 -0700437 mirror::DexCache* dex_cache = class_linker_->FindDexCache(Thread::Current(), dex);
Mathieu Chartiere401d142015-04-22 13:56:20 -0700438 auto* resolved_methods = dex_cache->GetResolvedMethods();
Vladimir Marko05792b92015-08-03 11:56:49 +0100439 for (size_t i = 0, num_methods = dex_cache->NumResolvedMethods(); i != num_methods; ++i) {
440 EXPECT_TRUE(
Andreas Gampe542451c2016-07-26 09:02:02 -0700441 mirror::DexCache::GetElementPtrSize(resolved_methods, i, kRuntimePointerSize) != nullptr)
Mathieu Chartiere401d142015-04-22 13:56:20 -0700442 << dex.GetLocation() << " i=" << i;
Ian Rogers19846512012-02-24 11:42:47 -0800443 }
Brian Carlstrom4a96b602011-07-26 16:40:23 -0700444 }
Brian Carlstromd2fbb2b2011-08-23 11:57:08 -0700445
Mathieu Chartierbb87e0f2015-04-03 11:21:55 -0700446 class TestRootVisitor : public SingleRootVisitor {
447 public:
448 void VisitRoot(mirror::Object* root, const RootInfo& info ATTRIBUTE_UNUSED) OVERRIDE {
449 EXPECT_TRUE(root != nullptr);
450 }
451 };
Brian Carlstrom578bbdc2011-07-21 14:07:47 -0700452};
Brian Carlstroma331b3c2011-07-18 17:47:56 -0700453
Brian Carlstrom693267a2011-09-06 09:25:34 -0700454struct CheckOffset {
455 size_t cpp_offset;
456 const char* java_name;
457 CheckOffset(size_t c, const char* j) : cpp_offset(c), java_name(j) {}
458};
459
Elliott Hughes80609252011-09-23 17:24:51 -0700460template <typename T>
Brian Carlstrom693267a2011-09-06 09:25:34 -0700461struct CheckOffsets {
Andreas Gampe277ccbd2014-11-03 21:36:10 -0800462 CheckOffsets(bool is_static_in, const char* class_descriptor_in)
463 : is_static(is_static_in), class_descriptor(class_descriptor_in) {}
Brian Carlstrom3320cf42011-10-04 14:58:28 -0700464 bool is_static;
Brian Carlstromdbc05252011-09-09 01:59:59 -0700465 std::string class_descriptor;
466 std::vector<CheckOffset> offsets;
Brian Carlstrom693267a2011-09-06 09:25:34 -0700467
Andreas Gampebdf7f1c2016-08-30 16:38:47 -0700468 bool Check() REQUIRES_SHARED(Locks::mutator_lock_) {
Ian Rogers98379392014-02-24 16:53:16 -0800469 Thread* self = Thread::Current();
470 mirror::Class* klass =
471 Runtime::Current()->GetClassLinker()->FindSystemClass(self, class_descriptor.c_str());
Ian Rogerscdc1aaf2014-10-09 13:21:38 -0700472 CHECK(klass != nullptr) << class_descriptor;
Brian Carlstrom693267a2011-09-06 09:25:34 -0700473
474 bool error = false;
475
Mathieu Chartiere401d142015-04-22 13:56:20 -0700476 // Classes have a different size due to padding field. Strings are variable length.
477 if (!klass->IsClassClass() && !klass->IsStringClass() && !is_static) {
Mathieu Chartierdaaf3262015-03-24 13:30:28 -0700478 // Currently only required for AccessibleObject since of the padding fields. The class linker
479 // says AccessibleObject is 9 bytes but sizeof(AccessibleObject) is 12 bytes due to padding.
480 // The RoundUp is to get around this case.
481 static constexpr size_t kPackAlignment = 4;
482 size_t expected_size = RoundUp(is_static ? klass->GetClassSize(): klass->GetObjectSize(),
483 kPackAlignment);
Elliott Hughes80609252011-09-23 17:24:51 -0700484 if (sizeof(T) != expected_size) {
Brian Carlstrom265091e2013-01-30 14:08:26 -0800485 LOG(ERROR) << "Class size mismatch:"
Brian Carlstromdbc05252011-09-09 01:59:59 -0700486 << " class=" << class_descriptor
Elliott Hughes80609252011-09-23 17:24:51 -0700487 << " Java=" << expected_size
488 << " C++=" << sizeof(T);
Brian Carlstrom4b620ff2011-09-11 01:11:01 -0700489 error = true;
490 }
491 }
492
Brian Carlstrom3320cf42011-10-04 14:58:28 -0700493 size_t num_fields = is_static ? klass->NumStaticFields() : klass->NumInstanceFields();
Brian Carlstrom4b620ff2011-09-11 01:11:01 -0700494 if (offsets.size() != num_fields) {
Brian Carlstrom265091e2013-01-30 14:08:26 -0800495 LOG(ERROR) << "Field count mismatch:"
Brian Carlstrom4b620ff2011-09-11 01:11:01 -0700496 << " class=" << class_descriptor
497 << " Java=" << num_fields
498 << " C++=" << offsets.size();
Brian Carlstrom693267a2011-09-06 09:25:34 -0700499 error = true;
500 }
501
Brian Carlstromdbc05252011-09-09 01:59:59 -0700502 for (size_t i = 0; i < offsets.size(); i++) {
Mathieu Chartierc7853442015-03-27 14:35:38 -0700503 ArtField* field = is_static ? klass->GetStaticField(i) : klass->GetInstanceField(i);
Mathieu Chartier61c5ebc2014-06-05 17:42:53 -0700504 StringPiece field_name(field->GetName());
Ian Rogers6d4d9fc2011-11-30 16:24:48 -0800505 if (field_name != offsets[i].java_name) {
Brian Carlstrom693267a2011-09-06 09:25:34 -0700506 error = true;
507 }
508 }
509 if (error) {
Brian Carlstromdbc05252011-09-09 01:59:59 -0700510 for (size_t i = 0; i < offsets.size(); i++) {
511 CheckOffset& offset = offsets[i];
Mathieu Chartierc7853442015-03-27 14:35:38 -0700512 ArtField* field = is_static ? klass->GetStaticField(i) : klass->GetInstanceField(i);
Mathieu Chartier61c5ebc2014-06-05 17:42:53 -0700513 StringPiece field_name(field->GetName());
Ian Rogers6d4d9fc2011-11-30 16:24:48 -0800514 if (field_name != offsets[i].java_name) {
Brian Carlstrom265091e2013-01-30 14:08:26 -0800515 LOG(ERROR) << "JAVA FIELD ORDER MISMATCH NEXT LINE:";
Brian Carlstrom693267a2011-09-06 09:25:34 -0700516 }
Brian Carlstrom265091e2013-01-30 14:08:26 -0800517 LOG(ERROR) << "Java field order:"
Brian Carlstromdbc05252011-09-09 01:59:59 -0700518 << " i=" << i << " class=" << class_descriptor
Ian Rogers6d4d9fc2011-11-30 16:24:48 -0800519 << " Java=" << field_name
Brian Carlstrom693267a2011-09-06 09:25:34 -0700520 << " CheckOffsets=" << offset.java_name;
521 }
522 }
523
Brian Carlstromdbc05252011-09-09 01:59:59 -0700524 for (size_t i = 0; i < offsets.size(); i++) {
525 CheckOffset& offset = offsets[i];
Mathieu Chartierc7853442015-03-27 14:35:38 -0700526 ArtField* field = is_static ? klass->GetStaticField(i) : klass->GetInstanceField(i);
Brian Carlstrom693267a2011-09-06 09:25:34 -0700527 if (field->GetOffset().Uint32Value() != offset.cpp_offset) {
528 error = true;
529 }
530 }
531 if (error) {
Brian Carlstromdbc05252011-09-09 01:59:59 -0700532 for (size_t i = 0; i < offsets.size(); i++) {
533 CheckOffset& offset = offsets[i];
Mathieu Chartierc7853442015-03-27 14:35:38 -0700534 ArtField* field = is_static ? klass->GetStaticField(i) : klass->GetInstanceField(i);
Brian Carlstrom693267a2011-09-06 09:25:34 -0700535 if (field->GetOffset().Uint32Value() != offset.cpp_offset) {
Brian Carlstrom265091e2013-01-30 14:08:26 -0800536 LOG(ERROR) << "OFFSET MISMATCH NEXT LINE:";
Brian Carlstrom693267a2011-09-06 09:25:34 -0700537 }
Brian Carlstrom265091e2013-01-30 14:08:26 -0800538 LOG(ERROR) << "Offset: class=" << class_descriptor << " field=" << offset.java_name
Brian Carlstrom693267a2011-09-06 09:25:34 -0700539 << " Java=" << field->GetOffset().Uint32Value() << " C++=" << offset.cpp_offset;
540 }
541 }
542
543 return !error;
544 };
Brian Carlstrom3320cf42011-10-04 14:58:28 -0700545
Mathieu Chartierfc58af42015-04-16 18:00:39 -0700546 void addOffset(size_t offset, const char* name) {
547 offsets.push_back(CheckOffset(offset, name));
548 }
549
Brian Carlstrom3320cf42011-10-04 14:58:28 -0700550 private:
551 DISALLOW_IMPLICIT_CONSTRUCTORS(CheckOffsets);
Brian Carlstrom693267a2011-09-06 09:25:34 -0700552};
553
Brian Carlstrom5b8e4c82011-09-18 01:38:59 -0700554// Note that ClassLinkerTest.ValidateFieldOrderOfJavaCppUnionClasses
555// is first since if it is failing, others are unlikely to succeed.
556
Brian Carlstromea46f952013-07-30 01:26:50 -0700557struct ObjectOffsets : public CheckOffsets<mirror::Object> {
558 ObjectOffsets() : CheckOffsets<mirror::Object>(false, "Ljava/lang/Object;") {
Mathieu Chartierfc58af42015-04-16 18:00:39 -0700559 addOffset(OFFSETOF_MEMBER(mirror::Object, klass_), "shadow$_klass_");
560 addOffset(OFFSETOF_MEMBER(mirror::Object, monitor_), "shadow$_monitor_");
Hiroshi Yamauchi60f63f52015-04-23 16:12:40 -0700561#ifdef USE_BROOKS_READ_BARRIER
Mathieu Chartierfc58af42015-04-16 18:00:39 -0700562 addOffset(OFFSETOF_MEMBER(mirror::Object, x_rb_ptr_), "shadow$_x_rb_ptr_");
563 addOffset(OFFSETOF_MEMBER(mirror::Object, x_xpadding_), "shadow$_x_xpadding_");
Hiroshi Yamauchi9d04a202014-01-31 13:35:49 -0800564#endif
Brian Carlstrom693267a2011-09-06 09:25:34 -0700565 };
566};
567
Brian Carlstromea46f952013-07-30 01:26:50 -0700568struct ClassOffsets : public CheckOffsets<mirror::Class> {
569 ClassOffsets() : CheckOffsets<mirror::Class>(false, "Ljava/lang/Class;") {
Mathieu Chartierfc58af42015-04-16 18:00:39 -0700570 addOffset(OFFSETOF_MEMBER(mirror::Class, access_flags_), "accessFlags");
Mathieu Chartier52a7f5c2015-08-18 18:35:52 -0700571 addOffset(OFFSETOF_MEMBER(mirror::Class, class_flags_), "classFlags");
Mathieu Chartierfc58af42015-04-16 18:00:39 -0700572 addOffset(OFFSETOF_MEMBER(mirror::Class, class_loader_), "classLoader");
573 addOffset(OFFSETOF_MEMBER(mirror::Class, class_size_), "classSize");
574 addOffset(OFFSETOF_MEMBER(mirror::Class, clinit_thread_id_), "clinitThreadId");
575 addOffset(OFFSETOF_MEMBER(mirror::Class, component_type_), "componentType");
Alex Lighte64300b2015-12-15 15:02:47 -0800576 addOffset(OFFSETOF_MEMBER(mirror::Class, copied_methods_offset_), "copiedMethodsOffset");
Mathieu Chartierfc58af42015-04-16 18:00:39 -0700577 addOffset(OFFSETOF_MEMBER(mirror::Class, dex_cache_), "dexCache");
578 addOffset(OFFSETOF_MEMBER(mirror::Class, dex_cache_strings_), "dexCacheStrings");
579 addOffset(OFFSETOF_MEMBER(mirror::Class, dex_class_def_idx_), "dexClassDefIndex");
580 addOffset(OFFSETOF_MEMBER(mirror::Class, dex_type_idx_), "dexTypeIndex");
Mathieu Chartierfc58af42015-04-16 18:00:39 -0700581 addOffset(OFFSETOF_MEMBER(mirror::Class, ifields_), "iFields");
582 addOffset(OFFSETOF_MEMBER(mirror::Class, iftable_), "ifTable");
Alex Lighte64300b2015-12-15 15:02:47 -0800583 addOffset(OFFSETOF_MEMBER(mirror::Class, methods_), "methods");
Mathieu Chartierfc58af42015-04-16 18:00:39 -0700584 addOffset(OFFSETOF_MEMBER(mirror::Class, name_), "name");
Mathieu Chartierfc58af42015-04-16 18:00:39 -0700585 addOffset(OFFSETOF_MEMBER(mirror::Class, num_reference_instance_fields_),
586 "numReferenceInstanceFields");
587 addOffset(OFFSETOF_MEMBER(mirror::Class, num_reference_static_fields_),
588 "numReferenceStaticFields");
Mathieu Chartierfc58af42015-04-16 18:00:39 -0700589 addOffset(OFFSETOF_MEMBER(mirror::Class, object_size_), "objectSize");
Mathieu Chartier93bbee02016-08-31 09:38:40 -0700590 addOffset(OFFSETOF_MEMBER(mirror::Class, object_size_alloc_fast_path_),
591 "objectSizeAllocFastPath");
Mathieu Chartierfc58af42015-04-16 18:00:39 -0700592 addOffset(OFFSETOF_MEMBER(mirror::Class, primitive_type_), "primitiveType");
593 addOffset(OFFSETOF_MEMBER(mirror::Class, reference_instance_offsets_),
594 "referenceInstanceOffsets");
595 addOffset(OFFSETOF_MEMBER(mirror::Class, sfields_), "sFields");
596 addOffset(OFFSETOF_MEMBER(mirror::Class, status_), "status");
597 addOffset(OFFSETOF_MEMBER(mirror::Class, super_class_), "superClass");
Andreas Gampe99babb62015-11-02 16:20:00 -0800598 addOffset(OFFSETOF_MEMBER(mirror::Class, verify_error_), "verifyError");
Alex Lighte64300b2015-12-15 15:02:47 -0800599 addOffset(OFFSETOF_MEMBER(mirror::Class, virtual_methods_offset_), "virtualMethodsOffset");
Mathieu Chartierfc58af42015-04-16 18:00:39 -0700600 addOffset(OFFSETOF_MEMBER(mirror::Class, vtable_), "vtable");
Brian Carlstrom693267a2011-09-06 09:25:34 -0700601 };
602};
603
Brian Carlstromea46f952013-07-30 01:26:50 -0700604struct StringOffsets : public CheckOffsets<mirror::String> {
605 StringOffsets() : CheckOffsets<mirror::String>(false, "Ljava/lang/String;") {
Mathieu Chartierfc58af42015-04-16 18:00:39 -0700606 addOffset(OFFSETOF_MEMBER(mirror::String, count_), "count");
Narayan Kamathd1ef4362015-11-12 11:49:06 +0000607 addOffset(OFFSETOF_MEMBER(mirror::String, hash_code_), "hash");
Brian Carlstrom693267a2011-09-06 09:25:34 -0700608 };
609};
610
Brian Carlstromea46f952013-07-30 01:26:50 -0700611struct ThrowableOffsets : public CheckOffsets<mirror::Throwable> {
612 ThrowableOffsets() : CheckOffsets<mirror::Throwable>(false, "Ljava/lang/Throwable;") {
Narayan Kamathd1ef4362015-11-12 11:49:06 +0000613 addOffset(OFFSETOF_MEMBER(mirror::Throwable, backtrace_), "backtrace");
Mathieu Chartierfc58af42015-04-16 18:00:39 -0700614 addOffset(OFFSETOF_MEMBER(mirror::Throwable, cause_), "cause");
615 addOffset(OFFSETOF_MEMBER(mirror::Throwable, detail_message_), "detailMessage");
Mathieu Chartierfc58af42015-04-16 18:00:39 -0700616 addOffset(OFFSETOF_MEMBER(mirror::Throwable, stack_trace_), "stackTrace");
617 addOffset(OFFSETOF_MEMBER(mirror::Throwable, suppressed_exceptions_), "suppressedExceptions");
Brian Carlstrom693267a2011-09-06 09:25:34 -0700618 };
619};
620
Brian Carlstromea46f952013-07-30 01:26:50 -0700621struct StackTraceElementOffsets : public CheckOffsets<mirror::StackTraceElement> {
Mathieu Chartierfc58af42015-04-16 18:00:39 -0700622 StackTraceElementOffsets() : CheckOffsets<mirror::StackTraceElement>(
623 false, "Ljava/lang/StackTraceElement;") {
624 addOffset(OFFSETOF_MEMBER(mirror::StackTraceElement, declaring_class_), "declaringClass");
625 addOffset(OFFSETOF_MEMBER(mirror::StackTraceElement, file_name_), "fileName");
626 addOffset(OFFSETOF_MEMBER(mirror::StackTraceElement, line_number_), "lineNumber");
627 addOffset(OFFSETOF_MEMBER(mirror::StackTraceElement, method_name_), "methodName");
Brian Carlstrom693267a2011-09-06 09:25:34 -0700628 };
629};
630
Brian Carlstromea46f952013-07-30 01:26:50 -0700631struct ClassLoaderOffsets : public CheckOffsets<mirror::ClassLoader> {
632 ClassLoaderOffsets() : CheckOffsets<mirror::ClassLoader>(false, "Ljava/lang/ClassLoader;") {
Mathieu Chartier951ec2c2015-09-22 08:50:05 -0700633 addOffset(OFFSETOF_MEMBER(mirror::ClassLoader, allocator_), "allocator");
Mathieu Chartier6b069532015-08-05 15:08:12 -0700634 addOffset(OFFSETOF_MEMBER(mirror::ClassLoader, class_table_), "classTable");
Mathieu Chartierfc58af42015-04-16 18:00:39 -0700635 addOffset(OFFSETOF_MEMBER(mirror::ClassLoader, packages_), "packages");
636 addOffset(OFFSETOF_MEMBER(mirror::ClassLoader, parent_), "parent");
637 addOffset(OFFSETOF_MEMBER(mirror::ClassLoader, proxyCache_), "proxyCache");
Brian Carlstrom693267a2011-09-06 09:25:34 -0700638 };
639};
640
Brian Carlstromea46f952013-07-30 01:26:50 -0700641struct ProxyOffsets : public CheckOffsets<mirror::Proxy> {
642 ProxyOffsets() : CheckOffsets<mirror::Proxy>(false, "Ljava/lang/reflect/Proxy;") {
Mathieu Chartierfc58af42015-04-16 18:00:39 -0700643 addOffset(OFFSETOF_MEMBER(mirror::Proxy, h_), "h");
Jesse Wilson95caa792011-10-12 18:14:17 -0400644 };
645};
646
Brian Carlstromea46f952013-07-30 01:26:50 -0700647struct DexCacheOffsets : public CheckOffsets<mirror::DexCache> {
648 DexCacheOffsets() : CheckOffsets<mirror::DexCache>(false, "Ljava/lang/DexCache;") {
Mathieu Chartierfc58af42015-04-16 18:00:39 -0700649 addOffset(OFFSETOF_MEMBER(mirror::DexCache, dex_), "dex");
650 addOffset(OFFSETOF_MEMBER(mirror::DexCache, dex_file_), "dexFile");
651 addOffset(OFFSETOF_MEMBER(mirror::DexCache, location_), "location");
Vladimir Marko05792b92015-08-03 11:56:49 +0100652 addOffset(OFFSETOF_MEMBER(mirror::DexCache, num_resolved_fields_), "numResolvedFields");
653 addOffset(OFFSETOF_MEMBER(mirror::DexCache, num_resolved_methods_), "numResolvedMethods");
654 addOffset(OFFSETOF_MEMBER(mirror::DexCache, num_resolved_types_), "numResolvedTypes");
655 addOffset(OFFSETOF_MEMBER(mirror::DexCache, num_strings_), "numStrings");
Mathieu Chartierfc58af42015-04-16 18:00:39 -0700656 addOffset(OFFSETOF_MEMBER(mirror::DexCache, resolved_fields_), "resolvedFields");
657 addOffset(OFFSETOF_MEMBER(mirror::DexCache, resolved_methods_), "resolvedMethods");
658 addOffset(OFFSETOF_MEMBER(mirror::DexCache, resolved_types_), "resolvedTypes");
659 addOffset(OFFSETOF_MEMBER(mirror::DexCache, strings_), "strings");
Mathieu Chartier66f19252012-09-18 08:57:04 -0700660 };
661};
662
Mathieu Chartier8fa2dad2014-03-13 12:22:56 -0700663struct ReferenceOffsets : public CheckOffsets<mirror::Reference> {
664 ReferenceOffsets() : CheckOffsets<mirror::Reference>(false, "Ljava/lang/ref/Reference;") {
Mathieu Chartierfc58af42015-04-16 18:00:39 -0700665 addOffset(OFFSETOF_MEMBER(mirror::Reference, pending_next_), "pendingNext");
666 addOffset(OFFSETOF_MEMBER(mirror::Reference, queue_), "queue");
667 addOffset(OFFSETOF_MEMBER(mirror::Reference, queue_next_), "queueNext");
668 addOffset(OFFSETOF_MEMBER(mirror::Reference, referent_), "referent");
Mathieu Chartier8fa2dad2014-03-13 12:22:56 -0700669 };
670};
671
672struct FinalizerReferenceOffsets : public CheckOffsets<mirror::FinalizerReference> {
Mathieu Chartierfc58af42015-04-16 18:00:39 -0700673 FinalizerReferenceOffsets() : CheckOffsets<mirror::FinalizerReference>(
674 false, "Ljava/lang/ref/FinalizerReference;") {
675 addOffset(OFFSETOF_MEMBER(mirror::FinalizerReference, next_), "next");
676 addOffset(OFFSETOF_MEMBER(mirror::FinalizerReference, prev_), "prev");
677 addOffset(OFFSETOF_MEMBER(mirror::FinalizerReference, zombie_), "zombie");
Mathieu Chartier8fa2dad2014-03-13 12:22:56 -0700678 };
679};
680
Mathieu Chartierdaaf3262015-03-24 13:30:28 -0700681struct AccessibleObjectOffsets : public CheckOffsets<mirror::AccessibleObject> {
Mathieu Chartierfc58af42015-04-16 18:00:39 -0700682 AccessibleObjectOffsets() : CheckOffsets<mirror::AccessibleObject>(
683 false, "Ljava/lang/reflect/AccessibleObject;") {
Narayan Kamathd1ef4362015-11-12 11:49:06 +0000684 addOffset(mirror::AccessibleObject::FlagOffset().Uint32Value(), "override");
Mathieu Chartierdaaf3262015-03-24 13:30:28 -0700685 };
686};
687
688struct FieldOffsets : public CheckOffsets<mirror::Field> {
689 FieldOffsets() : CheckOffsets<mirror::Field>(false, "Ljava/lang/reflect/Field;") {
Mathieu Chartierfc58af42015-04-16 18:00:39 -0700690 addOffset(OFFSETOF_MEMBER(mirror::Field, access_flags_), "accessFlags");
691 addOffset(OFFSETOF_MEMBER(mirror::Field, declaring_class_), "declaringClass");
692 addOffset(OFFSETOF_MEMBER(mirror::Field, dex_field_index_), "dexFieldIndex");
693 addOffset(OFFSETOF_MEMBER(mirror::Field, offset_), "offset");
694 addOffset(OFFSETOF_MEMBER(mirror::Field, type_), "type");
695 };
696};
697
Neil Fuller16b21cd2016-08-12 09:37:02 +0100698struct ExecutableOffsets : public CheckOffsets<mirror::Executable> {
699 ExecutableOffsets() : CheckOffsets<mirror::Executable>(
700 false, "Ljava/lang/reflect/Executable;") {
Neil Fuller0e844392016-09-08 13:43:31 +0100701 addOffset(OFFSETOF_MEMBER(mirror::Executable, access_flags_), "accessFlags");
702 addOffset(OFFSETOF_MEMBER(mirror::Executable, art_method_), "artMethod");
703 addOffset(OFFSETOF_MEMBER(mirror::Executable, declaring_class_), "declaringClass");
704 addOffset(OFFSETOF_MEMBER(mirror::Executable, declaring_class_of_overridden_method_),
705 "declaringClassOfOverriddenMethod");
706 addOffset(OFFSETOF_MEMBER(mirror::Executable, dex_method_index_), "dexMethodIndex");
Neil Fuller60458a02016-09-01 15:32:44 +0100707 addOffset(OFFSETOF_MEMBER(mirror::Executable, has_real_parameter_data_),
708 "hasRealParameterData");
709 addOffset(OFFSETOF_MEMBER(mirror::Executable, parameters_), "parameters");
Neil Fuller16b21cd2016-08-12 09:37:02 +0100710 };
711};
712
Narayan Kamathafa48272016-08-03 12:46:58 +0100713struct MethodTypeOffsets : public CheckOffsets<mirror::MethodType> {
714 MethodTypeOffsets() : CheckOffsets<mirror::MethodType>(
715 false, "Ljava/lang/invoke/MethodType;") {
716 addOffset(OFFSETOF_MEMBER(mirror::MethodType, form_), "form");
717 addOffset(OFFSETOF_MEMBER(mirror::MethodType, method_descriptor_), "methodDescriptor");
718 addOffset(OFFSETOF_MEMBER(mirror::MethodType, p_types_), "ptypes");
719 addOffset(OFFSETOF_MEMBER(mirror::MethodType, r_type_), "rtype");
720 addOffset(OFFSETOF_MEMBER(mirror::MethodType, wrap_alt_), "wrapAlt");
721 }
722};
723
724struct MethodHandleImplOffsets : public CheckOffsets<mirror::MethodHandleImpl> {
725 MethodHandleImplOffsets() : CheckOffsets<mirror::MethodHandleImpl>(
726 false, "Ljava/lang/invoke/MethodHandle;") {
727 addOffset(OFFSETOF_MEMBER(mirror::MethodHandleImpl, art_field_or_method_), "artFieldOrMethod");
728 addOffset(OFFSETOF_MEMBER(mirror::MethodHandleImpl, as_type_cache_), "asTypeCache");
729 addOffset(OFFSETOF_MEMBER(mirror::MethodHandleImpl, handle_kind_), "handleKind");
730 addOffset(OFFSETOF_MEMBER(mirror::MethodHandleImpl, method_type_), "type");
731 }
732};
733
Jesse Wilson46cdd4b2011-07-28 17:40:48 -0400734// C++ fields must exactly match the fields in the Java classes. If this fails,
Jesse Wilsonac5b9e22011-07-27 15:11:13 -0400735// reorder the fields in the C++ class. Managed class fields are ordered by
Brian Carlstrom693267a2011-09-06 09:25:34 -0700736// ClassLinker::LinkFields.
Jesse Wilson46cdd4b2011-07-28 17:40:48 -0400737TEST_F(ClassLinkerTest, ValidateFieldOrderOfJavaCppUnionClasses) {
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700738 ScopedObjectAccess soa(Thread::Current());
Brian Carlstrom693267a2011-09-06 09:25:34 -0700739 EXPECT_TRUE(ObjectOffsets().Check());
Brian Carlstrom693267a2011-09-06 09:25:34 -0700740 EXPECT_TRUE(ClassOffsets().Check());
741 EXPECT_TRUE(StringOffsets().Check());
742 EXPECT_TRUE(ThrowableOffsets().Check());
743 EXPECT_TRUE(StackTraceElementOffsets().Check());
744 EXPECT_TRUE(ClassLoaderOffsets().Check());
Jesse Wilson95caa792011-10-12 18:14:17 -0400745 EXPECT_TRUE(ProxyOffsets().Check());
Mathieu Chartier66f19252012-09-18 08:57:04 -0700746 EXPECT_TRUE(DexCacheOffsets().Check());
Mathieu Chartier8fa2dad2014-03-13 12:22:56 -0700747 EXPECT_TRUE(ReferenceOffsets().Check());
748 EXPECT_TRUE(FinalizerReferenceOffsets().Check());
Mathieu Chartierdaaf3262015-03-24 13:30:28 -0700749 EXPECT_TRUE(AccessibleObjectOffsets().Check());
750 EXPECT_TRUE(FieldOffsets().Check());
Neil Fuller16b21cd2016-08-12 09:37:02 +0100751 EXPECT_TRUE(ExecutableOffsets().Check());
Narayan Kamathafa48272016-08-03 12:46:58 +0100752 EXPECT_TRUE(MethodTypeOffsets().Check());
753 EXPECT_TRUE(MethodHandleImplOffsets().Check());
Brian Carlstrom74eb46a2011-08-02 20:10:14 -0700754}
755
Brian Carlstrom5b8e4c82011-09-18 01:38:59 -0700756TEST_F(ClassLinkerTest, FindClassNonexistent) {
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700757 ScopedObjectAccess soa(Thread::Current());
Brian Carlstrom5b8e4c82011-09-18 01:38:59 -0700758 AssertNonExistentClass("NoSuchClass;");
759 AssertNonExistentClass("LNoSuchClass;");
760}
761
Andreas Gampe81c6f8d2015-03-25 17:19:53 -0700762TEST_F(ClassLinkerTest, GetDexFiles) {
763 ScopedObjectAccess soa(Thread::Current());
764
765 jobject jclass_loader = LoadDex("Nested");
766 std::vector<const DexFile*> dex_files(GetDexFiles(jclass_loader));
767 ASSERT_EQ(dex_files.size(), 1U);
768 EXPECT_TRUE(EndsWith(dex_files[0]->GetLocation(), "Nested.jar"));
769
770 jobject jclass_loader2 = LoadDex("MultiDex");
771 std::vector<const DexFile*> dex_files2(GetDexFiles(jclass_loader2));
772 ASSERT_EQ(dex_files2.size(), 2U);
773 EXPECT_TRUE(EndsWith(dex_files2[0]->GetLocation(), "MultiDex.jar"));
774}
775
Brian Carlstrom5b8e4c82011-09-18 01:38:59 -0700776TEST_F(ClassLinkerTest, FindClassNested) {
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700777 ScopedObjectAccess soa(Thread::Current());
Mathieu Chartiereb8167a2014-05-07 15:43:14 -0700778 StackHandleScope<1> hs(soa.Self());
779 Handle<mirror::ClassLoader> class_loader(
Mathieu Chartier0795f232016-09-27 18:43:30 -0700780 hs.NewHandle(soa.Decode<mirror::ClassLoader>(LoadDex("Nested"))));
Brian Carlstrom5b8e4c82011-09-18 01:38:59 -0700781
Ian Rogers98379392014-02-24 16:53:16 -0800782 mirror::Class* outer = class_linker_->FindClass(soa.Self(), "LNested;", class_loader);
Ian Rogerscdc1aaf2014-10-09 13:21:38 -0700783 ASSERT_TRUE(outer != nullptr);
Brian Carlstrom5b8e4c82011-09-18 01:38:59 -0700784 EXPECT_EQ(0U, outer->NumVirtualMethods());
785 EXPECT_EQ(1U, outer->NumDirectMethods());
786
Ian Rogers98379392014-02-24 16:53:16 -0800787 mirror::Class* inner = class_linker_->FindClass(soa.Self(), "LNested$Inner;", class_loader);
Ian Rogerscdc1aaf2014-10-09 13:21:38 -0700788 ASSERT_TRUE(inner != nullptr);
Brian Carlstrom5b8e4c82011-09-18 01:38:59 -0700789 EXPECT_EQ(0U, inner->NumVirtualMethods());
790 EXPECT_EQ(1U, inner->NumDirectMethods());
791}
792
793TEST_F(ClassLinkerTest, FindClass_Primitives) {
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700794 ScopedObjectAccess soa(Thread::Current());
Brian Carlstromaded5f72011-10-07 17:15:04 -0700795 const std::string expected("BCDFIJSZV");
Elliott Hughesdb7d5e92011-12-16 18:47:37 -0800796 for (int ch = 1; ch < 256; ++ch) {
797 std::string descriptor;
798 descriptor.push_back(ch);
Brian Carlstromaded5f72011-10-07 17:15:04 -0700799 if (expected.find(ch) == std::string::npos) {
Brian Carlstrom5b8e4c82011-09-18 01:38:59 -0700800 AssertNonExistentClass(descriptor);
801 } else {
802 AssertPrimitiveClass(descriptor);
803 }
804 }
805}
806
807TEST_F(ClassLinkerTest, FindClass) {
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700808 ScopedObjectAccess soa(Thread::Current());
Ian Rogers98379392014-02-24 16:53:16 -0800809 mirror::Class* JavaLangObject = class_linker_->FindSystemClass(soa.Self(), "Ljava/lang/Object;");
Artem Udovichenkoa62cb9b2016-06-30 09:18:25 +0000810 AssertObjectClass(JavaLangObject);
Brian Carlstrom5b8e4c82011-09-18 01:38:59 -0700811
Mathieu Chartiereb8167a2014-05-07 15:43:14 -0700812 StackHandleScope<1> hs(soa.Self());
813 Handle<mirror::ClassLoader> class_loader(
Mathieu Chartier0795f232016-09-27 18:43:30 -0700814 hs.NewHandle(soa.Decode<mirror::ClassLoader>(LoadDex("MyClass"))));
Brian Carlstrom5b8e4c82011-09-18 01:38:59 -0700815 AssertNonExistentClass("LMyClass;");
Ian Rogers98379392014-02-24 16:53:16 -0800816 mirror::Class* MyClass = class_linker_->FindClass(soa.Self(), "LMyClass;", class_loader);
Ian Rogerscdc1aaf2014-10-09 13:21:38 -0700817 ASSERT_TRUE(MyClass != nullptr);
818 ASSERT_TRUE(MyClass->GetClass() != nullptr);
Brian Carlstrom5b8e4c82011-09-18 01:38:59 -0700819 ASSERT_EQ(MyClass->GetClass(), MyClass->GetClass()->GetClass());
820 EXPECT_EQ(JavaLangObject, MyClass->GetClass()->GetSuperClass());
Artem Udovichenkoa62cb9b2016-06-30 09:18:25 +0000821 std::string temp;
Ian Rogers1ff3c982014-08-12 02:30:58 -0700822 ASSERT_STREQ(MyClass->GetDescriptor(&temp), "LMyClass;");
Brian Carlstrom5b8e4c82011-09-18 01:38:59 -0700823 EXPECT_TRUE(MyClass->GetSuperClass() == JavaLangObject);
824 EXPECT_TRUE(MyClass->HasSuperClass());
Mathieu Chartiereb8167a2014-05-07 15:43:14 -0700825 EXPECT_EQ(class_loader.Get(), MyClass->GetClassLoader());
Brian Carlstromea46f952013-07-30 01:26:50 -0700826 EXPECT_EQ(mirror::Class::kStatusResolved, MyClass->GetStatus());
Brian Carlstrom5b8e4c82011-09-18 01:38:59 -0700827 EXPECT_FALSE(MyClass->IsErroneous());
Brian Carlstrom25c33252011-09-18 15:58:35 -0700828 EXPECT_TRUE(MyClass->IsLoaded());
Brian Carlstrom5b8e4c82011-09-18 01:38:59 -0700829 EXPECT_TRUE(MyClass->IsResolved());
Brian Carlstrom25c33252011-09-18 15:58:35 -0700830 EXPECT_FALSE(MyClass->IsVerified());
831 EXPECT_FALSE(MyClass->IsInitialized());
Brian Carlstrom5b8e4c82011-09-18 01:38:59 -0700832 EXPECT_FALSE(MyClass->IsArrayInstance());
833 EXPECT_FALSE(MyClass->IsArrayClass());
Ian Rogerscdc1aaf2014-10-09 13:21:38 -0700834 EXPECT_TRUE(MyClass->GetComponentType() == nullptr);
Brian Carlstrom5b8e4c82011-09-18 01:38:59 -0700835 EXPECT_FALSE(MyClass->IsInterface());
836 EXPECT_FALSE(MyClass->IsPublic());
837 EXPECT_FALSE(MyClass->IsFinal());
838 EXPECT_FALSE(MyClass->IsPrimitive());
839 EXPECT_FALSE(MyClass->IsSynthetic());
840 EXPECT_EQ(1U, MyClass->NumDirectMethods());
841 EXPECT_EQ(0U, MyClass->NumVirtualMethods());
842 EXPECT_EQ(0U, MyClass->NumInstanceFields());
843 EXPECT_EQ(0U, MyClass->NumStaticFields());
Mathieu Chartierf8322842014-05-16 10:59:25 -0700844 EXPECT_EQ(0U, MyClass->NumDirectInterfaces());
Brian Carlstrom5b8e4c82011-09-18 01:38:59 -0700845
846 EXPECT_EQ(JavaLangObject->GetClass()->GetClass(), MyClass->GetClass()->GetClass());
847
848 // created by class_linker
Ian Rogerscdc1aaf2014-10-09 13:21:38 -0700849 AssertArrayClass("[C", "C", nullptr);
850 AssertArrayClass("[Ljava/lang/Object;", "Ljava/lang/Object;", nullptr);
Brian Carlstrom5b8e4c82011-09-18 01:38:59 -0700851 // synthesized on the fly
Ian Rogerscdc1aaf2014-10-09 13:21:38 -0700852 AssertArrayClass("[[C", "[C", nullptr);
Mathieu Chartiereb8167a2014-05-07 15:43:14 -0700853 AssertArrayClass("[[[LMyClass;", "[[LMyClass;", class_loader.Get());
Brian Carlstrom5b8e4c82011-09-18 01:38:59 -0700854 // or not available at all
855 AssertNonExistentClass("[[[[LNonExistentClass;");
856}
857
858TEST_F(ClassLinkerTest, LibCore) {
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700859 ScopedObjectAccess soa(Thread::Current());
Richard Uhlerfbef44d2014-12-23 09:48:51 -0800860 ASSERT_TRUE(java_lang_dex_file_ != nullptr);
861 AssertDexFile(*java_lang_dex_file_, nullptr);
Brian Carlstrom5b8e4c82011-09-18 01:38:59 -0700862}
863
Ian Rogersa15e67d2012-02-28 13:51:55 -0800864// The first reference array element must be a multiple of 4 bytes from the
Jesse Wilsondf4189c2011-08-09 17:10:28 -0400865// start of the object
866TEST_F(ClassLinkerTest, ValidateObjectArrayElementsOffset) {
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700867 ScopedObjectAccess soa(Thread::Current());
Ian Rogers98379392014-02-24 16:53:16 -0800868 mirror::Class* array_class = class_linker_->FindSystemClass(soa.Self(), "[Ljava/lang/String;");
Brian Carlstromea46f952013-07-30 01:26:50 -0700869 mirror::ObjectArray<mirror::String>* array =
870 mirror::ObjectArray<mirror::String>::Alloc(soa.Self(), array_class, 0);
Ian Rogersef7d42f2014-01-06 12:55:46 -0800871 uintptr_t data_offset =
872 reinterpret_cast<uintptr_t>(array->GetRawData(sizeof(mirror::HeapReference<mirror::String>),
873 0));
874 if (sizeof(mirror::HeapReference<mirror::String>) == sizeof(int32_t)) {
Ian Rogersa15e67d2012-02-28 13:51:55 -0800875 EXPECT_TRUE(IsAligned<4>(data_offset)); // Check 4 byte alignment.
876 } else {
877 EXPECT_TRUE(IsAligned<8>(data_offset)); // Check 8 byte alignment.
878 }
Jesse Wilsondf4189c2011-08-09 17:10:28 -0400879}
880
881TEST_F(ClassLinkerTest, ValidatePrimitiveArrayElementsOffset) {
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700882 ScopedObjectAccess soa(Thread::Current());
Mathieu Chartiereb8167a2014-05-07 15:43:14 -0700883 StackHandleScope<5> hs(soa.Self());
884 Handle<mirror::LongArray> long_array(hs.NewHandle(mirror::LongArray::Alloc(soa.Self(), 0)));
Ian Rogers98379392014-02-24 16:53:16 -0800885 EXPECT_EQ(class_linker_->FindSystemClass(soa.Self(), "[J"), long_array->GetClass());
Ian Rogersa15e67d2012-02-28 13:51:55 -0800886 uintptr_t data_offset = reinterpret_cast<uintptr_t>(long_array->GetData());
887 EXPECT_TRUE(IsAligned<8>(data_offset)); // Longs require 8 byte alignment
888
Mathieu Chartiereb8167a2014-05-07 15:43:14 -0700889 Handle<mirror::DoubleArray> double_array(hs.NewHandle(mirror::DoubleArray::Alloc(soa.Self(), 0)));
Ian Rogers98379392014-02-24 16:53:16 -0800890 EXPECT_EQ(class_linker_->FindSystemClass(soa.Self(), "[D"), double_array->GetClass());
Ian Rogersa15e67d2012-02-28 13:51:55 -0800891 data_offset = reinterpret_cast<uintptr_t>(double_array->GetData());
892 EXPECT_TRUE(IsAligned<8>(data_offset)); // Doubles require 8 byte alignment
893
Mathieu Chartiereb8167a2014-05-07 15:43:14 -0700894 Handle<mirror::IntArray> int_array(hs.NewHandle(mirror::IntArray::Alloc(soa.Self(), 0)));
Ian Rogers98379392014-02-24 16:53:16 -0800895 EXPECT_EQ(class_linker_->FindSystemClass(soa.Self(), "[I"), int_array->GetClass());
Ian Rogersa15e67d2012-02-28 13:51:55 -0800896 data_offset = reinterpret_cast<uintptr_t>(int_array->GetData());
897 EXPECT_TRUE(IsAligned<4>(data_offset)); // Ints require 4 byte alignment
898
Mathieu Chartiereb8167a2014-05-07 15:43:14 -0700899 Handle<mirror::CharArray> char_array(hs.NewHandle(mirror::CharArray::Alloc(soa.Self(), 0)));
Ian Rogers98379392014-02-24 16:53:16 -0800900 EXPECT_EQ(class_linker_->FindSystemClass(soa.Self(), "[C"), char_array->GetClass());
Ian Rogersa15e67d2012-02-28 13:51:55 -0800901 data_offset = reinterpret_cast<uintptr_t>(char_array->GetData());
902 EXPECT_TRUE(IsAligned<2>(data_offset)); // Chars require 2 byte alignment
903
Mathieu Chartiereb8167a2014-05-07 15:43:14 -0700904 Handle<mirror::ShortArray> short_array(hs.NewHandle(mirror::ShortArray::Alloc(soa.Self(), 0)));
Ian Rogers98379392014-02-24 16:53:16 -0800905 EXPECT_EQ(class_linker_->FindSystemClass(soa.Self(), "[S"), short_array->GetClass());
Ian Rogersa15e67d2012-02-28 13:51:55 -0800906 data_offset = reinterpret_cast<uintptr_t>(short_array->GetData());
907 EXPECT_TRUE(IsAligned<2>(data_offset)); // Shorts require 2 byte alignment
908
909 // Take it as given that bytes and booleans have byte alignment
Jesse Wilsondf4189c2011-08-09 17:10:28 -0400910}
911
Elliott Hughes33203b52011-09-20 19:42:01 -0700912TEST_F(ClassLinkerTest, ValidateBoxedTypes) {
913 // Validate that the "value" field is always the 0th field in each of java.lang's box classes.
914 // This lets UnboxPrimitive avoid searching for the field by name at runtime.
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700915 ScopedObjectAccess soa(Thread::Current());
Mathieu Chartier9865bde2015-12-21 09:58:16 -0800916 ScopedNullHandle<mirror::ClassLoader> class_loader;
Brian Carlstromea46f952013-07-30 01:26:50 -0700917 mirror::Class* c;
Ian Rogers98379392014-02-24 16:53:16 -0800918 c = class_linker_->FindClass(soa.Self(), "Ljava/lang/Boolean;", class_loader);
Mathieu Chartier54d220e2015-07-30 16:20:06 -0700919 EXPECT_STREQ("value", c->GetIFieldsPtr()->At(0).GetName());
Ian Rogers98379392014-02-24 16:53:16 -0800920 c = class_linker_->FindClass(soa.Self(), "Ljava/lang/Byte;", class_loader);
Mathieu Chartier54d220e2015-07-30 16:20:06 -0700921 EXPECT_STREQ("value", c->GetIFieldsPtr()->At(0).GetName());
Ian Rogers98379392014-02-24 16:53:16 -0800922 c = class_linker_->FindClass(soa.Self(), "Ljava/lang/Character;", class_loader);
Mathieu Chartier54d220e2015-07-30 16:20:06 -0700923 EXPECT_STREQ("value", c->GetIFieldsPtr()->At(0).GetName());
Ian Rogers98379392014-02-24 16:53:16 -0800924 c = class_linker_->FindClass(soa.Self(), "Ljava/lang/Double;", class_loader);
Mathieu Chartier54d220e2015-07-30 16:20:06 -0700925 EXPECT_STREQ("value", c->GetIFieldsPtr()->At(0).GetName());
Ian Rogers98379392014-02-24 16:53:16 -0800926 c = class_linker_->FindClass(soa.Self(), "Ljava/lang/Float;", class_loader);
Mathieu Chartier54d220e2015-07-30 16:20:06 -0700927 EXPECT_STREQ("value", c->GetIFieldsPtr()->At(0).GetName());
Ian Rogers98379392014-02-24 16:53:16 -0800928 c = class_linker_->FindClass(soa.Self(), "Ljava/lang/Integer;", class_loader);
Mathieu Chartier54d220e2015-07-30 16:20:06 -0700929 EXPECT_STREQ("value", c->GetIFieldsPtr()->At(0).GetName());
Ian Rogers98379392014-02-24 16:53:16 -0800930 c = class_linker_->FindClass(soa.Self(), "Ljava/lang/Long;", class_loader);
Mathieu Chartier54d220e2015-07-30 16:20:06 -0700931 EXPECT_STREQ("value", c->GetIFieldsPtr()->At(0).GetName());
Ian Rogers98379392014-02-24 16:53:16 -0800932 c = class_linker_->FindClass(soa.Self(), "Ljava/lang/Short;", class_loader);
Mathieu Chartier54d220e2015-07-30 16:20:06 -0700933 EXPECT_STREQ("value", c->GetIFieldsPtr()->At(0).GetName());
Elliott Hughes33203b52011-09-20 19:42:01 -0700934}
935
Brian Carlstrom74eb46a2011-08-02 20:10:14 -0700936TEST_F(ClassLinkerTest, TwoClassLoadersOneClass) {
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700937 ScopedObjectAccess soa(Thread::Current());
Mathieu Chartiereb8167a2014-05-07 15:43:14 -0700938 StackHandleScope<2> hs(soa.Self());
939 Handle<mirror::ClassLoader> class_loader_1(
Mathieu Chartier0795f232016-09-27 18:43:30 -0700940 hs.NewHandle(soa.Decode<mirror::ClassLoader>(LoadDex("MyClass"))));
Mathieu Chartiereb8167a2014-05-07 15:43:14 -0700941 Handle<mirror::ClassLoader> class_loader_2(
Mathieu Chartier0795f232016-09-27 18:43:30 -0700942 hs.NewHandle(soa.Decode<mirror::ClassLoader>(LoadDex("MyClass"))));
Ian Rogers98379392014-02-24 16:53:16 -0800943 mirror::Class* MyClass_1 = class_linker_->FindClass(soa.Self(), "LMyClass;", class_loader_1);
944 mirror::Class* MyClass_2 = class_linker_->FindClass(soa.Self(), "LMyClass;", class_loader_2);
Mathieu Chartiereb8167a2014-05-07 15:43:14 -0700945 EXPECT_TRUE(MyClass_1 != nullptr);
946 EXPECT_TRUE(MyClass_2 != nullptr);
Brian Carlstrom74eb46a2011-08-02 20:10:14 -0700947 EXPECT_NE(MyClass_1, MyClass_2);
Jesse Wilsonac5b9e22011-07-27 15:11:13 -0400948}
949
Jesse Wilson7833bd22011-08-09 18:31:44 -0400950TEST_F(ClassLinkerTest, StaticFields) {
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700951 ScopedObjectAccess soa(Thread::Current());
Mathieu Chartiereb8167a2014-05-07 15:43:14 -0700952 StackHandleScope<2> hs(soa.Self());
953 Handle<mirror::ClassLoader> class_loader(
Mathieu Chartier0795f232016-09-27 18:43:30 -0700954 hs.NewHandle(soa.Decode<mirror::ClassLoader>(LoadDex("Statics"))));
Mathieu Chartiereb8167a2014-05-07 15:43:14 -0700955 Handle<mirror::Class> statics(
956 hs.NewHandle(class_linker_->FindClass(soa.Self(), "LStatics;", class_loader)));
Ian Rogers7b078e82014-09-10 14:44:24 -0700957 class_linker_->EnsureInitialized(soa.Self(), statics, true, true);
Jesse Wilson7833bd22011-08-09 18:31:44 -0400958
jeffhaoabcfde32011-09-29 15:05:18 -0700959 // Static final primitives that are initialized by a compile-time constant
960 // expression resolve to a copy of a constant value from the constant pool.
961 // So <clinit> should be null.
Andreas Gampe542451c2016-07-26 09:02:02 -0700962 ArtMethod* clinit = statics->FindDirectMethod("<clinit>", "()V", kRuntimePointerSize);
Ian Rogerscdc1aaf2014-10-09 13:21:38 -0700963 EXPECT_TRUE(clinit == nullptr);
jeffhaoabcfde32011-09-29 15:05:18 -0700964
965 EXPECT_EQ(9U, statics->NumStaticFields());
Jesse Wilson7833bd22011-08-09 18:31:44 -0400966
Mathieu Chartierc7853442015-03-27 14:35:38 -0700967 ArtField* s0 = mirror::Class::FindStaticField(soa.Self(), statics, "s0", "Z");
Mathieu Chartier61c5ebc2014-06-05 17:42:53 -0700968 EXPECT_EQ(s0->GetTypeAsPrimitiveType(), Primitive::kPrimBoolean);
Mathieu Chartiereb8167a2014-05-07 15:43:14 -0700969 EXPECT_EQ(true, s0->GetBoolean(statics.Get()));
970 s0->SetBoolean<false>(statics.Get(), false);
Jesse Wilson7833bd22011-08-09 18:31:44 -0400971
Mathieu Chartierc7853442015-03-27 14:35:38 -0700972 ArtField* s1 = mirror::Class::FindStaticField(soa.Self(), statics, "s1", "B");
Mathieu Chartier61c5ebc2014-06-05 17:42:53 -0700973 EXPECT_EQ(s1->GetTypeAsPrimitiveType(), Primitive::kPrimByte);
Mathieu Chartiereb8167a2014-05-07 15:43:14 -0700974 EXPECT_EQ(5, s1->GetByte(statics.Get()));
975 s1->SetByte<false>(statics.Get(), 6);
Jesse Wilson7833bd22011-08-09 18:31:44 -0400976
Mathieu Chartierc7853442015-03-27 14:35:38 -0700977 ArtField* s2 = mirror::Class::FindStaticField(soa.Self(), statics, "s2", "C");
Mathieu Chartier61c5ebc2014-06-05 17:42:53 -0700978 EXPECT_EQ(s2->GetTypeAsPrimitiveType(), Primitive::kPrimChar);
Mathieu Chartiereb8167a2014-05-07 15:43:14 -0700979 EXPECT_EQ('a', s2->GetChar(statics.Get()));
980 s2->SetChar<false>(statics.Get(), 'b');
Jesse Wilson7833bd22011-08-09 18:31:44 -0400981
Mathieu Chartierc7853442015-03-27 14:35:38 -0700982 ArtField* s3 = mirror::Class::FindStaticField(soa.Self(), statics, "s3", "S");
Mathieu Chartier61c5ebc2014-06-05 17:42:53 -0700983 EXPECT_EQ(s3->GetTypeAsPrimitiveType(), Primitive::kPrimShort);
Mathieu Chartiereb8167a2014-05-07 15:43:14 -0700984 EXPECT_EQ(-536, s3->GetShort(statics.Get()));
985 s3->SetShort<false>(statics.Get(), -535);
Jesse Wilson7833bd22011-08-09 18:31:44 -0400986
Mathieu Chartierc7853442015-03-27 14:35:38 -0700987 ArtField* s4 = mirror::Class::FindStaticField(soa.Self(), statics, "s4", "I");
Mathieu Chartier61c5ebc2014-06-05 17:42:53 -0700988 EXPECT_EQ(s4->GetTypeAsPrimitiveType(), Primitive::kPrimInt);
Mathieu Chartiereb8167a2014-05-07 15:43:14 -0700989 EXPECT_EQ(2000000000, s4->GetInt(statics.Get()));
990 s4->SetInt<false>(statics.Get(), 2000000001);
Jesse Wilson7833bd22011-08-09 18:31:44 -0400991
Mathieu Chartierc7853442015-03-27 14:35:38 -0700992 ArtField* s5 = mirror::Class::FindStaticField(soa.Self(), statics, "s5", "J");
Mathieu Chartier61c5ebc2014-06-05 17:42:53 -0700993 EXPECT_EQ(s5->GetTypeAsPrimitiveType(), Primitive::kPrimLong);
Mathieu Chartiereb8167a2014-05-07 15:43:14 -0700994 EXPECT_EQ(0x1234567890abcdefLL, s5->GetLong(statics.Get()));
995 s5->SetLong<false>(statics.Get(), INT64_C(0x34567890abcdef12));
Jesse Wilson7833bd22011-08-09 18:31:44 -0400996
Mathieu Chartierc7853442015-03-27 14:35:38 -0700997 ArtField* s6 = mirror::Class::FindStaticField(soa.Self(), statics, "s6", "F");
Mathieu Chartier61c5ebc2014-06-05 17:42:53 -0700998 EXPECT_EQ(s6->GetTypeAsPrimitiveType(), Primitive::kPrimFloat);
Ian Rogers647b1a82014-10-10 11:02:11 -0700999 EXPECT_DOUBLE_EQ(0.5, s6->GetFloat(statics.Get()));
Mathieu Chartiereb8167a2014-05-07 15:43:14 -07001000 s6->SetFloat<false>(statics.Get(), 0.75);
Jesse Wilson7833bd22011-08-09 18:31:44 -04001001
Mathieu Chartierc7853442015-03-27 14:35:38 -07001002 ArtField* s7 = mirror::Class::FindStaticField(soa.Self(), statics, "s7", "D");
Mathieu Chartier61c5ebc2014-06-05 17:42:53 -07001003 EXPECT_EQ(s7->GetTypeAsPrimitiveType(), Primitive::kPrimDouble);
Ian Rogers647b1a82014-10-10 11:02:11 -07001004 EXPECT_DOUBLE_EQ(16777217.0, s7->GetDouble(statics.Get()));
Mathieu Chartiereb8167a2014-05-07 15:43:14 -07001005 s7->SetDouble<false>(statics.Get(), 16777219);
Jesse Wilson7833bd22011-08-09 18:31:44 -04001006
Mathieu Chartierc7853442015-03-27 14:35:38 -07001007 ArtField* s8 = mirror::Class::FindStaticField(soa.Self(), statics, "s8",
Mathieu Chartierf8322842014-05-16 10:59:25 -07001008 "Ljava/lang/String;");
Mathieu Chartier61c5ebc2014-06-05 17:42:53 -07001009 EXPECT_EQ(s8->GetTypeAsPrimitiveType(), Primitive::kPrimNot);
Mathieu Chartiereb8167a2014-05-07 15:43:14 -07001010 EXPECT_TRUE(s8->GetObject(statics.Get())->AsString()->Equals("android"));
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +01001011 s8->SetObject<false>(s8->GetDeclaringClass(),
1012 mirror::String::AllocFromModifiedUtf8(soa.Self(), "robot"));
Jesse Wilson7833bd22011-08-09 18:31:44 -04001013
Brian Carlstrom2e3d1b22012-01-09 18:01:56 -08001014 // TODO: Remove EXPECT_FALSE when GCC can handle EXPECT_EQ
1015 // http://code.google.com/p/googletest/issues/detail?id=322
Mathieu Chartiereb8167a2014-05-07 15:43:14 -07001016 EXPECT_FALSE(s0->GetBoolean(statics.Get()));
1017 EXPECT_EQ(6, s1->GetByte(statics.Get()));
1018 EXPECT_EQ('b', s2->GetChar(statics.Get()));
1019 EXPECT_EQ(-535, s3->GetShort(statics.Get()));
1020 EXPECT_EQ(2000000001, s4->GetInt(statics.Get()));
1021 EXPECT_EQ(INT64_C(0x34567890abcdef12), s5->GetLong(statics.Get()));
Ian Rogers647b1a82014-10-10 11:02:11 -07001022 EXPECT_FLOAT_EQ(0.75, s6->GetFloat(statics.Get()));
1023 EXPECT_DOUBLE_EQ(16777219.0, s7->GetDouble(statics.Get()));
Mathieu Chartiereb8167a2014-05-07 15:43:14 -07001024 EXPECT_TRUE(s8->GetObject(statics.Get())->AsString()->Equals("robot"));
Jesse Wilson7833bd22011-08-09 18:31:44 -04001025}
1026
Brian Carlstrom30b94452011-08-25 21:35:26 -07001027TEST_F(ClassLinkerTest, Interfaces) {
Ian Rogers00f7d0e2012-07-19 15:28:27 -07001028 ScopedObjectAccess soa(Thread::Current());
Mathieu Chartierf8322842014-05-16 10:59:25 -07001029 StackHandleScope<6> hs(soa.Self());
Mathieu Chartiereb8167a2014-05-07 15:43:14 -07001030 Handle<mirror::ClassLoader> class_loader(
Mathieu Chartier0795f232016-09-27 18:43:30 -07001031 hs.NewHandle(soa.Decode<mirror::ClassLoader>(LoadDex("Interfaces"))));
Mathieu Chartierf8322842014-05-16 10:59:25 -07001032 Handle<mirror::Class> I(
1033 hs.NewHandle(class_linker_->FindClass(soa.Self(), "LInterfaces$I;", class_loader)));
1034 Handle<mirror::Class> J(
1035 hs.NewHandle(class_linker_->FindClass(soa.Self(), "LInterfaces$J;", class_loader)));
1036 Handle<mirror::Class> K(
1037 hs.NewHandle(class_linker_->FindClass(soa.Self(), "LInterfaces$K;", class_loader)));
1038 Handle<mirror::Class> A(
1039 hs.NewHandle(class_linker_->FindClass(soa.Self(), "LInterfaces$A;", class_loader)));
1040 Handle<mirror::Class> B(
1041 hs.NewHandle(class_linker_->FindClass(soa.Self(), "LInterfaces$B;", class_loader)));
1042 EXPECT_TRUE(I->IsAssignableFrom(A.Get()));
1043 EXPECT_TRUE(J->IsAssignableFrom(A.Get()));
1044 EXPECT_TRUE(J->IsAssignableFrom(K.Get()));
1045 EXPECT_TRUE(K->IsAssignableFrom(B.Get()));
1046 EXPECT_TRUE(J->IsAssignableFrom(B.Get()));
Brian Carlstrom30b94452011-08-25 21:35:26 -07001047
Ian Rogersd91d6d62013-09-25 20:26:14 -07001048 const Signature void_sig = I->GetDexCache()->GetDexFile()->CreateSignature("()V");
Andreas Gampe542451c2016-07-26 09:02:02 -07001049 ArtMethod* Ii = I->FindVirtualMethod("i", void_sig, kRuntimePointerSize);
1050 ArtMethod* Jj1 = J->FindVirtualMethod("j1", void_sig, kRuntimePointerSize);
1051 ArtMethod* Jj2 = J->FindVirtualMethod("j2", void_sig, kRuntimePointerSize);
1052 ArtMethod* Kj1 = K->FindInterfaceMethod("j1", void_sig, kRuntimePointerSize);
1053 ArtMethod* Kj2 = K->FindInterfaceMethod("j2", void_sig, kRuntimePointerSize);
1054 ArtMethod* Kk = K->FindInterfaceMethod("k", void_sig, kRuntimePointerSize);
1055 ArtMethod* Ai = A->FindVirtualMethod("i", void_sig, kRuntimePointerSize);
1056 ArtMethod* Aj1 = A->FindVirtualMethod("j1", void_sig, kRuntimePointerSize);
1057 ArtMethod* Aj2 = A->FindVirtualMethod("j2", void_sig, kRuntimePointerSize);
Ian Rogerscdc1aaf2014-10-09 13:21:38 -07001058 ASSERT_TRUE(Ii != nullptr);
1059 ASSERT_TRUE(Jj1 != nullptr);
1060 ASSERT_TRUE(Jj2 != nullptr);
1061 ASSERT_TRUE(Kj1 != nullptr);
1062 ASSERT_TRUE(Kj2 != nullptr);
1063 ASSERT_TRUE(Kk != nullptr);
1064 ASSERT_TRUE(Ai != nullptr);
1065 ASSERT_TRUE(Aj1 != nullptr);
1066 ASSERT_TRUE(Aj2 != nullptr);
Brian Carlstrom30b94452011-08-25 21:35:26 -07001067 EXPECT_NE(Ii, Ai);
1068 EXPECT_NE(Jj1, Aj1);
1069 EXPECT_NE(Jj2, Aj2);
jeffhao5dbddee2011-09-07 16:38:26 -07001070 EXPECT_EQ(Kj1, Jj1);
1071 EXPECT_EQ(Kj2, Jj2);
Andreas Gampe542451c2016-07-26 09:02:02 -07001072 EXPECT_EQ(Ai, A->FindVirtualMethodForInterface(Ii, kRuntimePointerSize));
1073 EXPECT_EQ(Aj1, A->FindVirtualMethodForInterface(Jj1, kRuntimePointerSize));
1074 EXPECT_EQ(Aj2, A->FindVirtualMethodForInterface(Jj2, kRuntimePointerSize));
1075 EXPECT_EQ(Ai, A->FindVirtualMethodForVirtualOrInterface(Ii, kRuntimePointerSize));
1076 EXPECT_EQ(Aj1, A->FindVirtualMethodForVirtualOrInterface(Jj1, kRuntimePointerSize));
1077 EXPECT_EQ(Aj2, A->FindVirtualMethodForVirtualOrInterface(Jj2, kRuntimePointerSize));
Ian Rogersd24e2642012-06-06 21:21:43 -07001078
Mathieu Chartiere401d142015-04-22 13:56:20 -07001079 ArtField* Afoo = mirror::Class::FindStaticField(soa.Self(), A, "foo", "Ljava/lang/String;");
1080 ArtField* Bfoo = mirror::Class::FindStaticField(soa.Self(), B, "foo", "Ljava/lang/String;");
1081 ArtField* Jfoo = mirror::Class::FindStaticField(soa.Self(), J, "foo", "Ljava/lang/String;");
1082 ArtField* Kfoo = mirror::Class::FindStaticField(soa.Self(), K, "foo", "Ljava/lang/String;");
Ian Rogerscdc1aaf2014-10-09 13:21:38 -07001083 ASSERT_TRUE(Afoo != nullptr);
Ian Rogersd24e2642012-06-06 21:21:43 -07001084 EXPECT_EQ(Afoo, Bfoo);
1085 EXPECT_EQ(Afoo, Jfoo);
1086 EXPECT_EQ(Afoo, Kfoo);
Brian Carlstrom30b94452011-08-25 21:35:26 -07001087}
1088
Elliott Hughesf3778f62012-01-26 14:14:35 -08001089TEST_F(ClassLinkerTest, ResolveVerifyAndClinit) {
jeffhaoabcfde32011-09-29 15:05:18 -07001090 // pretend we are trying to get the static storage for the StaticsFromCode class.
Brian Carlstrom193a44d2011-09-04 12:01:42 -07001091
jeffhaoabcfde32011-09-29 15:05:18 -07001092 // case 1, get the uninitialized storage from StaticsFromCode.<clinit>
1093 // case 2, get the initialized storage from StaticsFromCode.getS0
Brian Carlstrom193a44d2011-09-04 12:01:42 -07001094
Ian Rogers00f7d0e2012-07-19 15:28:27 -07001095 ScopedObjectAccess soa(Thread::Current());
1096 jobject jclass_loader = LoadDex("StaticsFromCode");
Andreas Gampe81c6f8d2015-03-25 17:19:53 -07001097 const DexFile* dex_file = GetFirstDexFile(jclass_loader);
Mathieu Chartiereb8167a2014-05-07 15:43:14 -07001098 StackHandleScope<1> hs(soa.Self());
1099 Handle<mirror::ClassLoader> class_loader(
Mathieu Chartier0795f232016-09-27 18:43:30 -07001100 hs.NewHandle(soa.Decode<mirror::ClassLoader>(jclass_loader)));
Ian Rogers98379392014-02-24 16:53:16 -08001101 mirror::Class* klass = class_linker_->FindClass(soa.Self(), "LStaticsFromCode;", class_loader);
Andreas Gampe542451c2016-07-26 09:02:02 -07001102 ArtMethod* clinit = klass->FindClassInitializer(kRuntimePointerSize);
1103 ArtMethod* getS0 = klass->FindDirectMethod("getS0", "()Ljava/lang/Object;", kRuntimePointerSize);
Mathieu Chartier9507fa22015-10-29 15:08:57 -07001104 const DexFile::TypeId* type_id = dex_file->FindTypeId("LStaticsFromCode;");
Ian Rogerscdc1aaf2014-10-09 13:21:38 -07001105 ASSERT_TRUE(type_id != nullptr);
Ian Rogers9b1a4f42011-11-14 18:35:10 -08001106 uint32_t type_idx = dex_file->GetIndexForTypeId(*type_id);
Mathieu Chartiere401d142015-04-22 13:56:20 -07001107 mirror::Class* uninit = ResolveVerifyAndClinit(type_idx, clinit, soa.Self(), true, false);
Ian Rogerscdc1aaf2014-10-09 13:21:38 -07001108 EXPECT_TRUE(uninit != nullptr);
Ian Rogers5ddb4102014-01-07 08:58:46 -08001109 EXPECT_FALSE(uninit->IsInitialized());
Mathieu Chartiere401d142015-04-22 13:56:20 -07001110 mirror::Class* init = ResolveVerifyAndClinit(type_idx, getS0, soa.Self(), true, false);
Ian Rogerscdc1aaf2014-10-09 13:21:38 -07001111 EXPECT_TRUE(init != nullptr);
Ian Rogers5ddb4102014-01-07 08:58:46 -08001112 EXPECT_TRUE(init->IsInitialized());
Brian Carlstrom848a4b32011-09-04 11:29:27 -07001113}
1114
Elliott Hughes20cde902011-10-04 17:37:27 -07001115TEST_F(ClassLinkerTest, FinalizableBit) {
Ian Rogers00f7d0e2012-07-19 15:28:27 -07001116 ScopedObjectAccess soa(Thread::Current());
Brian Carlstromea46f952013-07-30 01:26:50 -07001117 mirror::Class* c;
Elliott Hughes20cde902011-10-04 17:37:27 -07001118
1119 // Object has a finalize method, but we know it's empty.
Ian Rogers98379392014-02-24 16:53:16 -08001120 c = class_linker_->FindSystemClass(soa.Self(), "Ljava/lang/Object;");
Elliott Hughes20cde902011-10-04 17:37:27 -07001121 EXPECT_FALSE(c->IsFinalizable());
1122
1123 // Enum has a finalize method to prevent its subclasses from implementing one.
Ian Rogers98379392014-02-24 16:53:16 -08001124 c = class_linker_->FindSystemClass(soa.Self(), "Ljava/lang/Enum;");
Elliott Hughes20cde902011-10-04 17:37:27 -07001125 EXPECT_FALSE(c->IsFinalizable());
1126
1127 // RoundingMode is an enum.
Ian Rogers98379392014-02-24 16:53:16 -08001128 c = class_linker_->FindSystemClass(soa.Self(), "Ljava/math/RoundingMode;");
Elliott Hughes20cde902011-10-04 17:37:27 -07001129 EXPECT_FALSE(c->IsFinalizable());
1130
1131 // RandomAccessFile extends Object and overrides finalize.
Ian Rogers98379392014-02-24 16:53:16 -08001132 c = class_linker_->FindSystemClass(soa.Self(), "Ljava/io/RandomAccessFile;");
Elliott Hughes20cde902011-10-04 17:37:27 -07001133 EXPECT_TRUE(c->IsFinalizable());
1134
1135 // FileInputStream is finalizable and extends InputStream which isn't.
Ian Rogers98379392014-02-24 16:53:16 -08001136 c = class_linker_->FindSystemClass(soa.Self(), "Ljava/io/InputStream;");
Elliott Hughes20cde902011-10-04 17:37:27 -07001137 EXPECT_FALSE(c->IsFinalizable());
Ian Rogers98379392014-02-24 16:53:16 -08001138 c = class_linker_->FindSystemClass(soa.Self(), "Ljava/io/FileInputStream;");
Elliott Hughes20cde902011-10-04 17:37:27 -07001139 EXPECT_TRUE(c->IsFinalizable());
1140
1141 // ScheduledThreadPoolExecutor doesn't have a finalize method but
1142 // extends ThreadPoolExecutor which does.
Ian Rogers98379392014-02-24 16:53:16 -08001143 c = class_linker_->FindSystemClass(soa.Self(), "Ljava/util/concurrent/ThreadPoolExecutor;");
Elliott Hughes20cde902011-10-04 17:37:27 -07001144 EXPECT_TRUE(c->IsFinalizable());
Ian Rogers98379392014-02-24 16:53:16 -08001145 c = class_linker_->FindSystemClass(soa.Self(), "Ljava/util/concurrent/ScheduledThreadPoolExecutor;");
Elliott Hughes20cde902011-10-04 17:37:27 -07001146 EXPECT_TRUE(c->IsFinalizable());
1147}
1148
Ian Rogers6d4d9fc2011-11-30 16:24:48 -08001149TEST_F(ClassLinkerTest, ClassRootDescriptors) {
Ian Rogers00f7d0e2012-07-19 15:28:27 -07001150 ScopedObjectAccess soa(Thread::Current());
Ian Rogers1ff3c982014-08-12 02:30:58 -07001151 std::string temp;
Ian Rogers6d4d9fc2011-11-30 16:24:48 -08001152 for (int i = 0; i < ClassLinker::kClassRootsMax; i++) {
Brian Carlstromea46f952013-07-30 01:26:50 -07001153 mirror::Class* klass = class_linker_->GetClassRoot(ClassLinker::ClassRoot(i));
Ian Rogers1ff3c982014-08-12 02:30:58 -07001154 EXPECT_GT(strlen(klass->GetDescriptor(&temp)), 0U);
1155 EXPECT_STREQ(klass->GetDescriptor(&temp),
Ian Rogers6d4d9fc2011-11-30 16:24:48 -08001156 class_linker_->GetClassRootDescriptor(ClassLinker::ClassRoot(i))) << " i = " << i;
1157 }
1158}
1159
Mingyao Yang98d1cc82014-05-15 17:02:16 -07001160TEST_F(ClassLinkerTest, ValidatePredefinedClassSizes) {
1161 ScopedObjectAccess soa(Thread::Current());
Mathieu Chartier9865bde2015-12-21 09:58:16 -08001162 ScopedNullHandle<mirror::ClassLoader> class_loader;
Mingyao Yang98d1cc82014-05-15 17:02:16 -07001163 mirror::Class* c;
1164
1165 c = class_linker_->FindClass(soa.Self(), "Ljava/lang/Class;", class_loader);
Mathieu Chartiere401d142015-04-22 13:56:20 -07001166 ASSERT_TRUE(c != nullptr);
Andreas Gampe542451c2016-07-26 09:02:02 -07001167 EXPECT_EQ(c->GetClassSize(), mirror::Class::ClassClassSize(kRuntimePointerSize));
Mingyao Yang98d1cc82014-05-15 17:02:16 -07001168
1169 c = class_linker_->FindClass(soa.Self(), "Ljava/lang/Object;", class_loader);
Mathieu Chartiere401d142015-04-22 13:56:20 -07001170 ASSERT_TRUE(c != nullptr);
Andreas Gampe542451c2016-07-26 09:02:02 -07001171 EXPECT_EQ(c->GetClassSize(), mirror::Object::ClassSize(kRuntimePointerSize));
Mingyao Yang98d1cc82014-05-15 17:02:16 -07001172
1173 c = class_linker_->FindClass(soa.Self(), "Ljava/lang/String;", class_loader);
Mathieu Chartiere401d142015-04-22 13:56:20 -07001174 ASSERT_TRUE(c != nullptr);
Andreas Gampe542451c2016-07-26 09:02:02 -07001175 EXPECT_EQ(c->GetClassSize(), mirror::String::ClassSize(kRuntimePointerSize));
Mingyao Yang98d1cc82014-05-15 17:02:16 -07001176
1177 c = class_linker_->FindClass(soa.Self(), "Ljava/lang/DexCache;", class_loader);
Mathieu Chartiere401d142015-04-22 13:56:20 -07001178 ASSERT_TRUE(c != nullptr);
Andreas Gampe542451c2016-07-26 09:02:02 -07001179 EXPECT_EQ(c->GetClassSize(), mirror::DexCache::ClassSize(kRuntimePointerSize));
Mingyao Yang98d1cc82014-05-15 17:02:16 -07001180}
1181
Mathieu Chartiere401d142015-04-22 13:56:20 -07001182static void CheckMethod(ArtMethod* method, bool verified)
Andreas Gampebdf7f1c2016-08-30 16:38:47 -07001183 REQUIRES_SHARED(Locks::mutator_lock_) {
Andreas Gampe48498592014-09-10 19:48:05 -07001184 if (!method->IsNative() && !method->IsAbstract()) {
Igor Murashkindf707e42016-02-02 16:56:50 -08001185 EXPECT_EQ((method->GetAccessFlags() & kAccSkipAccessChecks) != 0U, verified)
Andreas Gampe48498592014-09-10 19:48:05 -07001186 << PrettyMethod(method, true);
1187 }
1188}
1189
Igor Murashkindf707e42016-02-02 16:56:50 -08001190static void CheckVerificationAttempted(mirror::Class* c, bool preverified)
Andreas Gampebdf7f1c2016-08-30 16:38:47 -07001191 REQUIRES_SHARED(Locks::mutator_lock_) {
Igor Murashkindf707e42016-02-02 16:56:50 -08001192 EXPECT_EQ((c->GetAccessFlags() & kAccVerificationAttempted) != 0U, preverified)
Andreas Gampe48498592014-09-10 19:48:05 -07001193 << "Class " << PrettyClass(c) << " not as expected";
Andreas Gampe542451c2016-07-26 09:02:02 -07001194 for (auto& m : c->GetMethods(kRuntimePointerSize)) {
Mathieu Chartiere401d142015-04-22 13:56:20 -07001195 CheckMethod(&m, preverified);
Andreas Gampe48498592014-09-10 19:48:05 -07001196 }
1197}
1198
1199TEST_F(ClassLinkerTest, Preverified_InitializedBoot) {
1200 ScopedObjectAccess soa(Thread::Current());
1201
1202 mirror::Class* JavaLangObject = class_linker_->FindSystemClass(soa.Self(), "Ljava/lang/Object;");
Ian Rogerscdc1aaf2014-10-09 13:21:38 -07001203 ASSERT_TRUE(JavaLangObject != nullptr);
Andreas Gampe48498592014-09-10 19:48:05 -07001204 EXPECT_TRUE(JavaLangObject->IsInitialized()) << "Not testing already initialized class from the "
1205 "core";
Igor Murashkindf707e42016-02-02 16:56:50 -08001206 CheckVerificationAttempted(JavaLangObject, true);
Andreas Gampe48498592014-09-10 19:48:05 -07001207}
1208
1209TEST_F(ClassLinkerTest, Preverified_UninitializedBoot) {
1210 ScopedObjectAccess soa(Thread::Current());
1211
1212 StackHandleScope<1> hs(soa.Self());
1213
1214 Handle<mirror::Class> security_manager(hs.NewHandle(class_linker_->FindSystemClass(
1215 soa.Self(), "Ljava/lang/SecurityManager;")));
1216 EXPECT_FALSE(security_manager->IsInitialized()) << "Not testing uninitialized class from the "
1217 "core";
1218
Igor Murashkindf707e42016-02-02 16:56:50 -08001219 CheckVerificationAttempted(security_manager.Get(), false);
Andreas Gampe48498592014-09-10 19:48:05 -07001220
Ian Rogers7b078e82014-09-10 14:44:24 -07001221 class_linker_->EnsureInitialized(soa.Self(), security_manager, true, true);
Igor Murashkindf707e42016-02-02 16:56:50 -08001222 CheckVerificationAttempted(security_manager.Get(), true);
Andreas Gampe48498592014-09-10 19:48:05 -07001223}
1224
1225TEST_F(ClassLinkerTest, Preverified_App) {
1226 ScopedObjectAccess soa(Thread::Current());
1227
1228 StackHandleScope<2> hs(soa.Self());
1229 Handle<mirror::ClassLoader> class_loader(
Mathieu Chartier0795f232016-09-27 18:43:30 -07001230 hs.NewHandle(soa.Decode<mirror::ClassLoader>(LoadDex("Statics"))));
Andreas Gampe48498592014-09-10 19:48:05 -07001231 Handle<mirror::Class> statics(
1232 hs.NewHandle(class_linker_->FindClass(soa.Self(), "LStatics;", class_loader)));
1233
Igor Murashkindf707e42016-02-02 16:56:50 -08001234 CheckVerificationAttempted(statics.Get(), false);
Andreas Gampe48498592014-09-10 19:48:05 -07001235
Ian Rogers7b078e82014-09-10 14:44:24 -07001236 class_linker_->EnsureInitialized(soa.Self(), statics, true, true);
Igor Murashkindf707e42016-02-02 16:56:50 -08001237 CheckVerificationAttempted(statics.Get(), true);
Andreas Gampe48498592014-09-10 19:48:05 -07001238}
1239
Sebastien Hertz6963e442014-11-26 22:11:27 +01001240TEST_F(ClassLinkerTest, IsBootStrapClassLoaded) {
1241 ScopedObjectAccess soa(Thread::Current());
1242
1243 StackHandleScope<3> hs(soa.Self());
1244 Handle<mirror::ClassLoader> class_loader(
Mathieu Chartier0795f232016-09-27 18:43:30 -07001245 hs.NewHandle(soa.Decode<mirror::ClassLoader>(LoadDex("Statics"))));
Sebastien Hertz6963e442014-11-26 22:11:27 +01001246
1247 // java.lang.Object is a bootstrap class.
1248 Handle<mirror::Class> jlo_class(
1249 hs.NewHandle(class_linker_->FindSystemClass(soa.Self(), "Ljava/lang/Object;")));
1250 ASSERT_TRUE(jlo_class.Get() != nullptr);
1251 EXPECT_TRUE(jlo_class.Get()->IsBootStrapClassLoaded());
1252
1253 // Statics is not a bootstrap class.
1254 Handle<mirror::Class> statics(
1255 hs.NewHandle(class_linker_->FindClass(soa.Self(), "LStatics;", class_loader)));
1256 ASSERT_TRUE(statics.Get() != nullptr);
1257 EXPECT_FALSE(statics.Get()->IsBootStrapClassLoaded());
1258}
1259
Mathieu Chartier76172162016-01-26 14:54:06 -08001260// Regression test for b/26799552.
1261TEST_F(ClassLinkerTest, RegisterDexFileName) {
1262 ScopedObjectAccess soa(Thread::Current());
1263 StackHandleScope<2> hs(soa.Self());
1264 ClassLinker* class_linker = Runtime::Current()->GetClassLinker();
1265 MutableHandle<mirror::DexCache> dex_cache(hs.NewHandle<mirror::DexCache>(nullptr));
1266 {
1267 ReaderMutexLock mu(soa.Self(), *class_linker->DexLock());
1268 for (const ClassLinker::DexCacheData& data : class_linker->GetDexCachesData()) {
1269 dex_cache.Assign(down_cast<mirror::DexCache*>(soa.Self()->DecodeJObject(data.weak_root)));
1270 if (dex_cache.Get() != nullptr) {
1271 break;
1272 }
1273 }
1274 ASSERT_TRUE(dex_cache.Get() != nullptr);
1275 }
1276 // Make a copy of the dex cache and change the name.
1277 dex_cache.Assign(dex_cache->Clone(soa.Self())->AsDexCache());
1278 const uint16_t data[] = { 0x20AC, 0x20A1 };
1279 Handle<mirror::String> location(hs.NewHandle(mirror::String::AllocFromUtf16(soa.Self(),
1280 arraysize(data),
1281 data)));
1282 dex_cache->SetLocation(location.Get());
1283 const DexFile* old_dex_file = dex_cache->GetDexFile();
1284
Hiroshi Yamauchi906ae5d2016-02-22 13:30:35 -08001285 std::unique_ptr<DexFile> dex_file(new DexFile(old_dex_file->Begin(),
1286 old_dex_file->Size(),
1287 location->ToModifiedUtf8(),
1288 0u,
Hiroshi Yamauchi906ae5d2016-02-22 13:30:35 -08001289 nullptr));
Mathieu Chartier76172162016-01-26 14:54:06 -08001290 {
1291 WriterMutexLock mu(soa.Self(), *class_linker->DexLock());
1292 // Check that inserting with a UTF16 name works.
1293 class_linker->RegisterDexFileLocked(*dex_file, dex_cache);
1294 }
1295}
1296
Jesse Wilsondf4189c2011-08-09 17:10:28 -04001297} // namespace art