blob: d0b9e1648ef8222dc662368254ba205e99ee2cb2 [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 Rogers0cfe1fb2011-08-26 03:29:44 -070019#include <string>
20
Elliott Hughes90a33692011-08-30 13:27:07 -070021#include "UniquePtr.h"
22#include "common_test.h"
Brian Carlstromc4fa2c02011-08-21 03:00:12 -070023#include "dex_cache.h"
Brian Carlstrom578bbdc2011-07-21 14:07:47 -070024#include "dex_file.h"
25#include "heap.h"
Ian Rogersce9eca62011-10-07 17:11:03 -070026#include "runtime_support.h"
Ian Rogers1f539342012-10-03 21:09:42 -070027#include "sirt_ref.h"
Carl Shapiro0e5d75d2011-07-06 18:28:37 -070028
29namespace art {
30
Brian Carlstromf734cf52011-08-17 16:28:14 -070031class ClassLinkerTest : public CommonTest {
Brian Carlstrom578bbdc2011-07-21 14:07:47 -070032 protected:
Ian Rogers00f7d0e2012-07-19 15:28:27 -070033 void AssertNonExistentClass(const std::string& descriptor)
Ian Rogersb726dcb2012-09-05 08:57:23 -070034 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
Elliott Hughesc3b77c72011-12-15 20:56:48 -080035 EXPECT_TRUE(class_linker_->FindSystemClass(descriptor.c_str()) == NULL);
Elliott Hughes885c3bd2011-08-22 16:59:20 -070036 Thread* self = Thread::Current();
37 EXPECT_TRUE(self->IsExceptionPending());
38 Object* exception = self->GetException();
39 self->ClearException();
40 Class* exception_class = class_linker_->FindSystemClass("Ljava/lang/NoClassDefFoundError;");
41 EXPECT_TRUE(exception->InstanceOf(exception_class));
Brian Carlstrom578bbdc2011-07-21 14:07:47 -070042 }
43
Ian Rogers00f7d0e2012-07-19 15:28:27 -070044 void AssertPrimitiveClass(const std::string& descriptor)
Ian Rogersb726dcb2012-09-05 08:57:23 -070045 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
Elliott Hughesc3b77c72011-12-15 20:56:48 -080046 AssertPrimitiveClass(descriptor, class_linker_->FindSystemClass(descriptor.c_str()));
Brian Carlstromd2fbb2b2011-08-23 11:57:08 -070047 }
48
Ian Rogers00f7d0e2012-07-19 15:28:27 -070049 void AssertPrimitiveClass(const std::string& descriptor, const Class* primitive)
Ian Rogersb726dcb2012-09-05 08:57:23 -070050 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
Ian Rogers6d4d9fc2011-11-30 16:24:48 -080051 ClassHelper primitive_ch(primitive);
Brian Carlstrom578bbdc2011-07-21 14:07:47 -070052 ASSERT_TRUE(primitive != NULL);
53 ASSERT_TRUE(primitive->GetClass() != NULL);
54 ASSERT_EQ(primitive->GetClass(), primitive->GetClass()->GetClass());
55 EXPECT_TRUE(primitive->GetClass()->GetSuperClass() != NULL);
Elliott Hughes91250e02011-12-13 22:30:35 -080056 ASSERT_STREQ(descriptor.c_str(), primitive_ch.GetDescriptor());
Brian Carlstrom578bbdc2011-07-21 14:07:47 -070057 EXPECT_TRUE(primitive->GetSuperClass() == NULL);
58 EXPECT_FALSE(primitive->HasSuperClass());
Brian Carlstrom74eb46a2011-08-02 20:10:14 -070059 EXPECT_TRUE(primitive->GetClassLoader() == NULL);
Brian Carlstrom25c33252011-09-18 15:58:35 -070060 EXPECT_EQ(Class::kStatusInitialized, primitive->GetStatus());
Brian Carlstrom578bbdc2011-07-21 14:07:47 -070061 EXPECT_FALSE(primitive->IsErroneous());
Brian Carlstrom25c33252011-09-18 15:58:35 -070062 EXPECT_TRUE(primitive->IsLoaded());
Elliott Hughes5fe594f2011-09-08 12:33:17 -070063 EXPECT_TRUE(primitive->IsResolved());
Brian Carlstrom25c33252011-09-18 15:58:35 -070064 EXPECT_TRUE(primitive->IsVerified());
65 EXPECT_TRUE(primitive->IsInitialized());
Brian Carlstromb63ec392011-08-27 17:38:27 -070066 EXPECT_FALSE(primitive->IsArrayInstance());
67 EXPECT_FALSE(primitive->IsArrayClass());
Brian Carlstrom5b8e4c82011-09-18 01:38:59 -070068 EXPECT_TRUE(primitive->GetComponentType() == NULL);
Brian Carlstrom578bbdc2011-07-21 14:07:47 -070069 EXPECT_FALSE(primitive->IsInterface());
70 EXPECT_TRUE(primitive->IsPublic());
71 EXPECT_TRUE(primitive->IsFinal());
72 EXPECT_TRUE(primitive->IsPrimitive());
Brian Carlstrom75cb3b42011-07-28 02:13:36 -070073 EXPECT_FALSE(primitive->IsSynthetic());
Brian Carlstrom578bbdc2011-07-21 14:07:47 -070074 EXPECT_EQ(0U, primitive->NumDirectMethods());
75 EXPECT_EQ(0U, primitive->NumVirtualMethods());
76 EXPECT_EQ(0U, primitive->NumInstanceFields());
77 EXPECT_EQ(0U, primitive->NumStaticFields());
Ian Rogersd24e2642012-06-06 21:21:43 -070078 EXPECT_EQ(0U, primitive_ch.NumDirectInterfaces());
Brian Carlstrom86927212011-09-15 11:31:11 -070079 EXPECT_TRUE(primitive->GetVTable() == NULL);
80 EXPECT_EQ(0, primitive->GetIfTableCount());
81 EXPECT_TRUE(primitive->GetIfTable() == NULL);
Brian Carlstrom578bbdc2011-07-21 14:07:47 -070082 }
83
Brian Carlstromaded5f72011-10-07 17:15:04 -070084 void AssertArrayClass(const std::string& array_descriptor,
85 const std::string& component_type,
Ian Rogers00f7d0e2012-07-19 15:28:27 -070086 ClassLoader* class_loader)
Ian Rogersb726dcb2012-09-05 08:57:23 -070087 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
Elliott Hughesc3b77c72011-12-15 20:56:48 -080088 Class* array = class_linker_->FindClass(array_descriptor.c_str(), class_loader);
Ian Rogers6d4d9fc2011-11-30 16:24:48 -080089 ClassHelper array_component_ch(array->GetComponentType());
Elliott Hughes91250e02011-12-13 22:30:35 -080090 EXPECT_STREQ(component_type.c_str(), array_component_ch.GetDescriptor());
Brian Carlstromd2fbb2b2011-08-23 11:57:08 -070091 EXPECT_EQ(class_loader, array->GetClassLoader());
92 AssertArrayClass(array_descriptor, array);
93 }
94
Ian Rogers00f7d0e2012-07-19 15:28:27 -070095 void AssertArrayClass(const std::string& array_descriptor, Class* array)
Ian Rogersb726dcb2012-09-05 08:57:23 -070096 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
Ian Rogers6d4d9fc2011-11-30 16:24:48 -080097 ClassHelper kh(array);
Brian Carlstrom578bbdc2011-07-21 14:07:47 -070098 ASSERT_TRUE(array != NULL);
99 ASSERT_TRUE(array->GetClass() != NULL);
100 ASSERT_EQ(array->GetClass(), array->GetClass()->GetClass());
101 EXPECT_TRUE(array->GetClass()->GetSuperClass() != NULL);
Elliott Hughes91250e02011-12-13 22:30:35 -0800102 ASSERT_STREQ(array_descriptor.c_str(), kh.GetDescriptor());
Brian Carlstrom578bbdc2011-07-21 14:07:47 -0700103 EXPECT_TRUE(array->GetSuperClass() != NULL);
Carl Shapiro7a909592011-07-24 19:21:59 -0700104 EXPECT_EQ(class_linker_->FindSystemClass("Ljava/lang/Object;"), array->GetSuperClass());
Brian Carlstrom578bbdc2011-07-21 14:07:47 -0700105 EXPECT_TRUE(array->HasSuperClass());
106 ASSERT_TRUE(array->GetComponentType() != NULL);
Ian Rogers6d4d9fc2011-11-30 16:24:48 -0800107 kh.ChangeClass(array->GetComponentType());
108 ASSERT_TRUE(kh.GetDescriptor() != NULL);
Brian Carlstrom25c33252011-09-18 15:58:35 -0700109 EXPECT_EQ(Class::kStatusInitialized, array->GetStatus());
Brian Carlstrom578bbdc2011-07-21 14:07:47 -0700110 EXPECT_FALSE(array->IsErroneous());
Brian Carlstrom25c33252011-09-18 15:58:35 -0700111 EXPECT_TRUE(array->IsLoaded());
Elliott Hughes5fe594f2011-09-08 12:33:17 -0700112 EXPECT_TRUE(array->IsResolved());
Brian Carlstrom25c33252011-09-18 15:58:35 -0700113 EXPECT_TRUE(array->IsVerified());
114 EXPECT_TRUE(array->IsInitialized());
Brian Carlstromb63ec392011-08-27 17:38:27 -0700115 EXPECT_FALSE(array->IsArrayInstance());
116 EXPECT_TRUE(array->IsArrayClass());
Brian Carlstrom578bbdc2011-07-21 14:07:47 -0700117 EXPECT_FALSE(array->IsInterface());
118 EXPECT_EQ(array->GetComponentType()->IsPublic(), array->IsPublic());
119 EXPECT_TRUE(array->IsFinal());
120 EXPECT_FALSE(array->IsPrimitive());
Brian Carlstrom75cb3b42011-07-28 02:13:36 -0700121 EXPECT_FALSE(array->IsSynthetic());
Brian Carlstrom578bbdc2011-07-21 14:07:47 -0700122 EXPECT_EQ(0U, array->NumDirectMethods());
123 EXPECT_EQ(0U, array->NumVirtualMethods());
124 EXPECT_EQ(0U, array->NumInstanceFields());
125 EXPECT_EQ(0U, array->NumStaticFields());
Ian Rogers6d4d9fc2011-11-30 16:24:48 -0800126 kh.ChangeClass(array);
Ian Rogersd24e2642012-06-06 21:21:43 -0700127 EXPECT_EQ(2U, kh.NumDirectInterfaces());
Brian Carlstrom86927212011-09-15 11:31:11 -0700128 EXPECT_TRUE(array->GetVTable() != NULL);
129 EXPECT_EQ(2, array->GetIfTableCount());
130 ObjectArray<InterfaceEntry>* iftable = array->GetIfTable();
131 ASSERT_TRUE(iftable != NULL);
Ian Rogersd24e2642012-06-06 21:21:43 -0700132 kh.ChangeClass(kh.GetDirectInterface(0));
Elliott Hughes91250e02011-12-13 22:30:35 -0800133 EXPECT_STREQ(kh.GetDescriptor(), "Ljava/lang/Cloneable;");
Ian Rogers6d4d9fc2011-11-30 16:24:48 -0800134 kh.ChangeClass(array);
Ian Rogersd24e2642012-06-06 21:21:43 -0700135 kh.ChangeClass(kh.GetDirectInterface(1));
Elliott Hughes91250e02011-12-13 22:30:35 -0800136 EXPECT_STREQ(kh.GetDescriptor(), "Ljava/io/Serializable;");
Brian Carlstrom578bbdc2011-07-21 14:07:47 -0700137 }
Brian Carlstrom4a96b602011-07-26 16:40:23 -0700138
Mathieu Chartier66f19252012-09-18 08:57:04 -0700139 void AssertMethod(AbstractMethod* method) SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
Ian Rogers6d4d9fc2011-11-30 16:24:48 -0800140 MethodHelper mh(method);
Brian Carlstromc4fa2c02011-08-21 03:00:12 -0700141 EXPECT_TRUE(method != NULL);
Brian Carlstroma5a97a22011-09-15 14:08:49 -0700142 EXPECT_TRUE(method->GetClass() != NULL);
Ian Rogers6d4d9fc2011-11-30 16:24:48 -0800143 EXPECT_TRUE(mh.GetName() != NULL);
144 EXPECT_TRUE(mh.GetSignature() != NULL);
Brian Carlstromc4fa2c02011-08-21 03:00:12 -0700145
Ian Rogers0cfe1fb2011-08-26 03:29:44 -0700146 EXPECT_TRUE(method->GetDexCacheStrings() != NULL);
Ian Rogers19846512012-02-24 11:42:47 -0800147 EXPECT_TRUE(method->GetDexCacheResolvedMethods() != NULL);
Ian Rogers0cfe1fb2011-08-26 03:29:44 -0700148 EXPECT_TRUE(method->GetDexCacheResolvedTypes() != NULL);
Ian Rogers0cfe1fb2011-08-26 03:29:44 -0700149 EXPECT_TRUE(method->GetDexCacheInitializedStaticStorage() != NULL);
150 EXPECT_EQ(method->GetDeclaringClass()->GetDexCache()->GetStrings(),
151 method->GetDexCacheStrings());
Ian Rogers19846512012-02-24 11:42:47 -0800152 EXPECT_EQ(method->GetDeclaringClass()->GetDexCache()->GetResolvedMethods(),
153 method->GetDexCacheResolvedMethods());
Ian Rogers0cfe1fb2011-08-26 03:29:44 -0700154 EXPECT_EQ(method->GetDeclaringClass()->GetDexCache()->GetResolvedTypes(),
155 method->GetDexCacheResolvedTypes());
Ian Rogers0cfe1fb2011-08-26 03:29:44 -0700156 EXPECT_EQ(method->GetDeclaringClass()->GetDexCache()->GetInitializedStaticStorage(),
157 method->GetDexCacheInitializedStaticStorage());
Brian Carlstromc4fa2c02011-08-21 03:00:12 -0700158 }
159
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700160 void AssertField(Class* klass, Field* field)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700161 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
Ian Rogers6d4d9fc2011-11-30 16:24:48 -0800162 FieldHelper fh(field);
Brian Carlstromc4fa2c02011-08-21 03:00:12 -0700163 EXPECT_TRUE(field != NULL);
Brian Carlstroma5a97a22011-09-15 14:08:49 -0700164 EXPECT_TRUE(field->GetClass() != NULL);
Brian Carlstromc4fa2c02011-08-21 03:00:12 -0700165 EXPECT_EQ(klass, field->GetDeclaringClass());
Ian Rogers6d4d9fc2011-11-30 16:24:48 -0800166 EXPECT_TRUE(fh.GetName() != NULL);
167 EXPECT_TRUE(fh.GetType() != NULL);
Brian Carlstromc4fa2c02011-08-21 03:00:12 -0700168 }
169
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700170 void AssertClass(const std::string& descriptor, Class* klass)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700171 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
Ian Rogers6d4d9fc2011-11-30 16:24:48 -0800172 ClassHelper kh(klass);
Elliott Hughes91250e02011-12-13 22:30:35 -0800173 EXPECT_STREQ(descriptor.c_str(), kh.GetDescriptor());
Ian Rogers6d4d9fc2011-11-30 16:24:48 -0800174 if (descriptor == "Ljava/lang/Object;") {
Brian Carlstrom74eb46a2011-08-02 20:10:14 -0700175 EXPECT_FALSE(klass->HasSuperClass());
Brian Carlstromae3ac012011-07-27 01:30:28 -0700176 } else {
Brian Carlstrom74eb46a2011-08-02 20:10:14 -0700177 EXPECT_TRUE(klass->HasSuperClass());
178 EXPECT_TRUE(klass->GetSuperClass() != NULL);
Brian Carlstromae3ac012011-07-27 01:30:28 -0700179 }
Brian Carlstroma5a97a22011-09-15 14:08:49 -0700180 EXPECT_TRUE(klass->GetClass() != NULL);
181 EXPECT_EQ(klass->GetClass(), klass->GetClass()->GetClass());
Brian Carlstromae3ac012011-07-27 01:30:28 -0700182 EXPECT_TRUE(klass->GetDexCache() != NULL);
Brian Carlstrom25c33252011-09-18 15:58:35 -0700183 EXPECT_TRUE(klass->IsLoaded());
Elliott Hughes5fe594f2011-09-08 12:33:17 -0700184 EXPECT_TRUE(klass->IsResolved());
Brian Carlstromae3ac012011-07-27 01:30:28 -0700185 EXPECT_FALSE(klass->IsErroneous());
Ian Rogers0cfe1fb2011-08-26 03:29:44 -0700186 EXPECT_FALSE(klass->IsArrayClass());
Brian Carlstrom5b8e4c82011-09-18 01:38:59 -0700187 EXPECT_TRUE(klass->GetComponentType() == NULL);
Brian Carlstromae3ac012011-07-27 01:30:28 -0700188 EXPECT_TRUE(klass->IsInSamePackage(klass));
Ian Rogers6d4d9fc2011-11-30 16:24:48 -0800189 EXPECT_TRUE(Class::IsInSamePackage(kh.GetDescriptor(), kh.GetDescriptor()));
Brian Carlstromae3ac012011-07-27 01:30:28 -0700190 if (klass->IsInterface()) {
Brian Carlstrom74eb46a2011-08-02 20:10:14 -0700191 EXPECT_TRUE(klass->IsAbstract());
192 if (klass->NumDirectMethods() == 1) {
Ian Rogers6d4d9fc2011-11-30 16:24:48 -0800193 MethodHelper mh(klass->GetDirectMethod(0));
194 EXPECT_TRUE(mh.IsClassInitializer());
Brian Carlstrom9cc262e2011-08-28 12:45:30 -0700195 EXPECT_TRUE(klass->GetDirectMethod(0)->IsDirect());
Brian Carlstrom74eb46a2011-08-02 20:10:14 -0700196 } else {
197 EXPECT_EQ(0U, klass->NumDirectMethods());
198 }
Brian Carlstromae3ac012011-07-27 01:30:28 -0700199 } else {
Brian Carlstrom74eb46a2011-08-02 20:10:14 -0700200 if (!klass->IsSynthetic()) {
201 EXPECT_NE(0U, klass->NumDirectMethods());
202 }
Brian Carlstromae3ac012011-07-27 01:30:28 -0700203 }
Brian Carlstrom86927212011-09-15 11:31:11 -0700204 EXPECT_EQ(klass->IsInterface(), klass->GetVTable() == NULL);
205 for (int i = 0; i < klass->GetIfTableCount(); i++) {
206 const InterfaceEntry* interface_entry = klass->GetIfTable()->Get(i);
207 ASSERT_TRUE(interface_entry != NULL);
208 Class* interface = interface_entry->GetInterface();
209 ASSERT_TRUE(interface != NULL);
210 EXPECT_TRUE(interface_entry->GetInterface() != NULL);
211 if (klass->IsInterface()) {
212 EXPECT_EQ(0U, interface_entry->GetMethodArrayCount());
213 } else {
214 CHECK_EQ(interface->NumVirtualMethods(), interface_entry->GetMethodArrayCount());
215 EXPECT_EQ(interface->NumVirtualMethods(), interface_entry->GetMethodArrayCount());
216 }
217 }
Brian Carlstromae3ac012011-07-27 01:30:28 -0700218 if (klass->IsAbstract()) {
Brian Carlstrom74eb46a2011-08-02 20:10:14 -0700219 EXPECT_FALSE(klass->IsFinal());
Brian Carlstromae3ac012011-07-27 01:30:28 -0700220 } else {
Brian Carlstrom74eb46a2011-08-02 20:10:14 -0700221 EXPECT_FALSE(klass->IsAnnotation());
Brian Carlstromae3ac012011-07-27 01:30:28 -0700222 }
223 if (klass->IsFinal()) {
Brian Carlstrom74eb46a2011-08-02 20:10:14 -0700224 EXPECT_FALSE(klass->IsAbstract());
225 EXPECT_FALSE(klass->IsAnnotation());
Brian Carlstromae3ac012011-07-27 01:30:28 -0700226 }
227 if (klass->IsAnnotation()) {
Brian Carlstrom74eb46a2011-08-02 20:10:14 -0700228 EXPECT_FALSE(klass->IsFinal());
229 EXPECT_TRUE(klass->IsAbstract());
Brian Carlstromae3ac012011-07-27 01:30:28 -0700230 }
231
232 EXPECT_FALSE(klass->IsPrimitive());
233 EXPECT_TRUE(klass->CanAccess(klass));
234
235 for (size_t i = 0; i < klass->NumDirectMethods(); i++) {
Mathieu Chartier66f19252012-09-18 08:57:04 -0700236 AbstractMethod* method = klass->GetDirectMethod(i);
Elliott Hughes1bac54f2012-03-16 12:48:31 -0700237 AssertMethod(method);
Brian Carlstrom9cc262e2011-08-28 12:45:30 -0700238 EXPECT_TRUE(method->IsDirect());
Jesse Wilson7833bd22011-08-09 18:31:44 -0400239 EXPECT_EQ(klass, method->GetDeclaringClass());
Brian Carlstromae3ac012011-07-27 01:30:28 -0700240 }
241
242 for (size_t i = 0; i < klass->NumVirtualMethods(); i++) {
Mathieu Chartier66f19252012-09-18 08:57:04 -0700243 AbstractMethod* method = klass->GetVirtualMethod(i);
Elliott Hughes1bac54f2012-03-16 12:48:31 -0700244 AssertMethod(method);
Brian Carlstrom9cc262e2011-08-28 12:45:30 -0700245 EXPECT_FALSE(method->IsDirect());
Brian Carlstromc4fa2c02011-08-21 03:00:12 -0700246 EXPECT_TRUE(method->GetDeclaringClass()->IsAssignableFrom(klass));
Brian Carlstromae3ac012011-07-27 01:30:28 -0700247 }
248
249 for (size_t i = 0; i < klass->NumInstanceFields(); i++) {
Jesse Wilson35baaab2011-08-10 16:18:03 -0400250 Field* field = klass->GetInstanceField(i);
Brian Carlstromd2fbb2b2011-08-23 11:57:08 -0700251 AssertField(klass, field);
Jesse Wilsonfd687c52011-08-04 19:27:35 -0700252 EXPECT_FALSE(field->IsStatic());
Brian Carlstromae3ac012011-07-27 01:30:28 -0700253 }
254
255 for (size_t i = 0; i < klass->NumStaticFields(); i++) {
Jesse Wilson35baaab2011-08-10 16:18:03 -0400256 Field* field = klass->GetStaticField(i);
Brian Carlstromd2fbb2b2011-08-23 11:57:08 -0700257 AssertField(klass, field);
Jesse Wilsonfd687c52011-08-04 19:27:35 -0700258 EXPECT_TRUE(field->IsStatic());
Elliott Hughes362f9bc2011-10-17 18:56:41 -0700259 }
Brian Carlstromae3ac012011-07-27 01:30:28 -0700260
261 // Confirm that all instances fields are packed together at the start
262 EXPECT_GE(klass->NumInstanceFields(), klass->NumReferenceInstanceFields());
Ian Rogers6d4d9fc2011-11-30 16:24:48 -0800263 FieldHelper fh;
Brian Carlstromae3ac012011-07-27 01:30:28 -0700264 for (size_t i = 0; i < klass->NumReferenceInstanceFields(); i++) {
Jesse Wilson35baaab2011-08-10 16:18:03 -0400265 Field* field = klass->GetInstanceField(i);
Ian Rogers6d4d9fc2011-11-30 16:24:48 -0800266 fh.ChangeField(field);
267 ASSERT_TRUE(!fh.IsPrimitiveType());
268 Class* field_type = fh.GetType();
Brian Carlstrom74eb46a2011-08-02 20:10:14 -0700269 ASSERT_TRUE(field_type != NULL);
Ian Rogers0cfe1fb2011-08-26 03:29:44 -0700270 ASSERT_TRUE(!field_type->IsPrimitive());
Brian Carlstromae3ac012011-07-27 01:30:28 -0700271 }
272 for (size_t i = klass->NumReferenceInstanceFields(); i < klass->NumInstanceFields(); i++) {
Jesse Wilson35baaab2011-08-10 16:18:03 -0400273 Field* field = klass->GetInstanceField(i);
Ian Rogers6d4d9fc2011-11-30 16:24:48 -0800274 fh.ChangeField(field);
275 Class* field_type = fh.GetType();
Brian Carlstrom74eb46a2011-08-02 20:10:14 -0700276 ASSERT_TRUE(field_type != NULL);
Ian Rogers6d4d9fc2011-11-30 16:24:48 -0800277 if (!fh.IsPrimitiveType() || !field_type->IsPrimitive()) {
Brian Carlstromfbfdce62011-10-05 17:33:32 -0700278 // While Reference.referent is not primitive, the ClassLinker
279 // treats it as such so that the garbage collector won't scan it.
280 EXPECT_EQ(PrettyField(field), "java.lang.Object java.lang.ref.Reference.referent");
281 }
Brian Carlstromae3ac012011-07-27 01:30:28 -0700282 }
283
284 size_t total_num_reference_instance_fields = 0;
285 Class* k = klass;
286 while (k != NULL) {
Brian Carlstrom74eb46a2011-08-02 20:10:14 -0700287 total_num_reference_instance_fields += k->NumReferenceInstanceFields();
288 k = k->GetSuperClass();
Brian Carlstromae3ac012011-07-27 01:30:28 -0700289 }
Brian Carlstrom4873d462011-08-21 15:23:39 -0700290 EXPECT_EQ(klass->GetReferenceInstanceOffsets() == 0,
Brian Carlstromae3ac012011-07-27 01:30:28 -0700291 total_num_reference_instance_fields == 0);
Brian Carlstrom4a96b602011-07-26 16:40:23 -0700292 }
293
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700294 void AssertDexFileClass(ClassLoader* class_loader, const std::string& descriptor)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700295 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
Brian Carlstromd2fbb2b2011-08-23 11:57:08 -0700296 ASSERT_TRUE(descriptor != NULL);
Elliott Hughesc3b77c72011-12-15 20:56:48 -0800297 Class* klass = class_linker_->FindSystemClass(descriptor.c_str());
Brian Carlstromd2fbb2b2011-08-23 11:57:08 -0700298 ASSERT_TRUE(klass != NULL);
Elliott Hughes91250e02011-12-13 22:30:35 -0800299 EXPECT_STREQ(descriptor.c_str(), ClassHelper(klass).GetDescriptor());
Brian Carlstromd2fbb2b2011-08-23 11:57:08 -0700300 EXPECT_EQ(class_loader, klass->GetClassLoader());
301 if (klass->IsPrimitive()) {
302 AssertPrimitiveClass(descriptor, klass);
Brian Carlstromb63ec392011-08-27 17:38:27 -0700303 } else if (klass->IsArrayClass()) {
Brian Carlstromd2fbb2b2011-08-23 11:57:08 -0700304 AssertArrayClass(descriptor, klass);
305 } else {
306 AssertClass(descriptor, klass);
307 }
Brian Carlstrom75cb3b42011-07-28 02:13:36 -0700308 }
309
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700310 void AssertDexFile(const DexFile* dex, ClassLoader* class_loader)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700311 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
Brian Carlstromae3ac012011-07-27 01:30:28 -0700312 ASSERT_TRUE(dex != NULL);
Brian Carlstrom9ea1cb12011-08-24 23:18:18 -0700313
Brian Carlstromd2fbb2b2011-08-23 11:57:08 -0700314 // Verify all the classes defined in this file
Brian Carlstrom4a96b602011-07-26 16:40:23 -0700315 for (size_t i = 0; i < dex->NumClassDefs(); i++) {
Brian Carlstromd2fbb2b2011-08-23 11:57:08 -0700316 const DexFile::ClassDef& class_def = dex->GetClassDef(i);
Brian Carlstrom4a96b602011-07-26 16:40:23 -0700317 const char* descriptor = dex->GetClassDescriptor(class_def);
Brian Carlstrom74eb46a2011-08-02 20:10:14 -0700318 AssertDexFileClass(class_loader, descriptor);
Brian Carlstrom4a96b602011-07-26 16:40:23 -0700319 }
Brian Carlstrom9ea1cb12011-08-24 23:18:18 -0700320 // Verify all the types referenced by this file
Brian Carlstromd2fbb2b2011-08-23 11:57:08 -0700321 for (size_t i = 0; i < dex->NumTypeIds(); i++) {
322 const DexFile::TypeId& type_id = dex->GetTypeId(i);
323 const char* descriptor = dex->GetTypeDescriptor(type_id);
324 AssertDexFileClass(class_loader, descriptor);
325 }
Brian Carlstrom75cb3b42011-07-28 02:13:36 -0700326 class_linker_->VisitRoots(TestRootVisitor, NULL);
Ian Rogers19846512012-02-24 11:42:47 -0800327 // Verify the dex cache has resolution methods in all resolved method slots
328 DexCache* dex_cache = class_linker_->FindDexCache(*dex);
Mathieu Chartier66f19252012-09-18 08:57:04 -0700329 ObjectArray<AbstractMethod>* resolved_methods = dex_cache->GetResolvedMethods();
Ian Rogers19846512012-02-24 11:42:47 -0800330 for (size_t i = 0; i < static_cast<size_t>(resolved_methods->GetLength()); i++) {
331 EXPECT_TRUE(resolved_methods->Get(i) != NULL);
332 }
Brian Carlstrom4a96b602011-07-26 16:40:23 -0700333 }
Brian Carlstromd2fbb2b2011-08-23 11:57:08 -0700334
Elliott Hughes1bac54f2012-03-16 12:48:31 -0700335 static void TestRootVisitor(const Object* root, void*) {
Brian Carlstromd2fbb2b2011-08-23 11:57:08 -0700336 EXPECT_TRUE(root != NULL);
337 }
Brian Carlstrom578bbdc2011-07-21 14:07:47 -0700338};
Brian Carlstroma331b3c2011-07-18 17:47:56 -0700339
Brian Carlstrom693267a2011-09-06 09:25:34 -0700340struct CheckOffset {
341 size_t cpp_offset;
342 const char* java_name;
343 CheckOffset(size_t c, const char* j) : cpp_offset(c), java_name(j) {}
344};
345
Elliott Hughes80609252011-09-23 17:24:51 -0700346template <typename T>
Brian Carlstrom693267a2011-09-06 09:25:34 -0700347struct CheckOffsets {
Brian Carlstrom3320cf42011-10-04 14:58:28 -0700348 CheckOffsets(bool is_static, const char* class_descriptor)
349 : is_static(is_static), class_descriptor(class_descriptor) {}
350 bool is_static;
Brian Carlstromdbc05252011-09-09 01:59:59 -0700351 std::string class_descriptor;
352 std::vector<CheckOffset> offsets;
Brian Carlstrom693267a2011-09-06 09:25:34 -0700353
Ian Rogersb726dcb2012-09-05 08:57:23 -0700354 bool Check() SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
Elliott Hughesc3b77c72011-12-15 20:56:48 -0800355 Class* klass = Runtime::Current()->GetClassLinker()->FindSystemClass(class_descriptor.c_str());
Brian Carlstromdbc05252011-09-09 01:59:59 -0700356 CHECK(klass != NULL) << class_descriptor;
Brian Carlstrom693267a2011-09-06 09:25:34 -0700357
358 bool error = false;
359
Brian Carlstrom3320cf42011-10-04 14:58:28 -0700360 if (!klass->IsClassClass() && !is_static) {
361 size_t expected_size = is_static ? klass->GetClassSize(): klass->GetObjectSize();
Elliott Hughes80609252011-09-23 17:24:51 -0700362 if (sizeof(T) != expected_size) {
Brian Carlstrom693267a2011-09-06 09:25:34 -0700363 LG << "Class size mismatch:"
Brian Carlstromdbc05252011-09-09 01:59:59 -0700364 << " class=" << class_descriptor
Elliott Hughes80609252011-09-23 17:24:51 -0700365 << " Java=" << expected_size
366 << " C++=" << sizeof(T);
Brian Carlstrom4b620ff2011-09-11 01:11:01 -0700367 error = true;
368 }
369 }
370
Brian Carlstrom3320cf42011-10-04 14:58:28 -0700371 size_t num_fields = is_static ? klass->NumStaticFields() : klass->NumInstanceFields();
Brian Carlstrom4b620ff2011-09-11 01:11:01 -0700372 if (offsets.size() != num_fields) {
373 LG << "Field count mismatch:"
374 << " class=" << class_descriptor
375 << " Java=" << num_fields
376 << " C++=" << offsets.size();
Brian Carlstrom693267a2011-09-06 09:25:34 -0700377 error = true;
378 }
379
Ian Rogers6d4d9fc2011-11-30 16:24:48 -0800380 FieldHelper fh;
Brian Carlstromdbc05252011-09-09 01:59:59 -0700381 for (size_t i = 0; i < offsets.size(); i++) {
Brian Carlstrom3320cf42011-10-04 14:58:28 -0700382 Field* field = is_static ? klass->GetStaticField(i) : klass->GetInstanceField(i);
Ian Rogers6d4d9fc2011-11-30 16:24:48 -0800383 fh.ChangeField(field);
384 StringPiece field_name(fh.GetName());
385 if (field_name != offsets[i].java_name) {
Brian Carlstrom693267a2011-09-06 09:25:34 -0700386 error = true;
387 }
388 }
389 if (error) {
Brian Carlstromdbc05252011-09-09 01:59:59 -0700390 for (size_t i = 0; i < offsets.size(); i++) {
391 CheckOffset& offset = offsets[i];
Brian Carlstrom3320cf42011-10-04 14:58:28 -0700392 Field* field = is_static ? klass->GetStaticField(i) : klass->GetInstanceField(i);
Ian Rogers6d4d9fc2011-11-30 16:24:48 -0800393 fh.ChangeField(field);
394 StringPiece field_name(fh.GetName());
395 if (field_name != offsets[i].java_name) {
Brian Carlstrom693267a2011-09-06 09:25:34 -0700396 LG << "JAVA FIELD ORDER MISMATCH NEXT LINE:";
397 }
398 LG << "Java field order:"
Brian Carlstromdbc05252011-09-09 01:59:59 -0700399 << " i=" << i << " class=" << class_descriptor
Ian Rogers6d4d9fc2011-11-30 16:24:48 -0800400 << " Java=" << field_name
Brian Carlstrom693267a2011-09-06 09:25:34 -0700401 << " CheckOffsets=" << offset.java_name;
402 }
403 }
404
Brian Carlstromdbc05252011-09-09 01:59:59 -0700405 for (size_t i = 0; i < offsets.size(); i++) {
406 CheckOffset& offset = offsets[i];
Brian Carlstrom3320cf42011-10-04 14:58:28 -0700407 Field* field = is_static ? klass->GetStaticField(i) : klass->GetInstanceField(i);
Brian Carlstrom693267a2011-09-06 09:25:34 -0700408 if (field->GetOffset().Uint32Value() != offset.cpp_offset) {
409 error = true;
410 }
411 }
412 if (error) {
Brian Carlstromdbc05252011-09-09 01:59:59 -0700413 for (size_t i = 0; i < offsets.size(); i++) {
414 CheckOffset& offset = offsets[i];
Brian Carlstrom3320cf42011-10-04 14:58:28 -0700415 Field* field = is_static ? klass->GetStaticField(i) : klass->GetInstanceField(i);
Brian Carlstrom693267a2011-09-06 09:25:34 -0700416 if (field->GetOffset().Uint32Value() != offset.cpp_offset) {
417 LG << "OFFSET MISMATCH NEXT LINE:";
418 }
Brian Carlstromdbc05252011-09-09 01:59:59 -0700419 LG << "Offset: class=" << class_descriptor << " field=" << offset.java_name
Brian Carlstrom693267a2011-09-06 09:25:34 -0700420 << " Java=" << field->GetOffset().Uint32Value() << " C++=" << offset.cpp_offset;
421 }
422 }
423
424 return !error;
425 };
Brian Carlstrom3320cf42011-10-04 14:58:28 -0700426
427 private:
428 DISALLOW_IMPLICIT_CONSTRUCTORS(CheckOffsets);
Brian Carlstrom693267a2011-09-06 09:25:34 -0700429};
430
Brian Carlstrom5b8e4c82011-09-18 01:38:59 -0700431// Note that ClassLinkerTest.ValidateFieldOrderOfJavaCppUnionClasses
432// is first since if it is failing, others are unlikely to succeed.
433
Elliott Hughes80609252011-09-23 17:24:51 -0700434struct ObjectOffsets : public CheckOffsets<Object> {
Brian Carlstrom3320cf42011-10-04 14:58:28 -0700435 ObjectOffsets() : CheckOffsets<Object>(false, "Ljava/lang/Object;") {
Brian Carlstromdbc05252011-09-09 01:59:59 -0700436 // alphabetical references
437 offsets.push_back(CheckOffset(OFFSETOF_MEMBER(Object, klass_), "shadow$_klass_"));
438
439 // alphabetical 32-bit
440 offsets.push_back(CheckOffset(OFFSETOF_MEMBER(Object, monitor_), "shadow$_monitor_"));
Brian Carlstrom693267a2011-09-06 09:25:34 -0700441 };
442};
443
Elliott Hughes80609252011-09-23 17:24:51 -0700444struct FieldOffsets : public CheckOffsets<Field> {
Brian Carlstrom3320cf42011-10-04 14:58:28 -0700445 FieldOffsets() : CheckOffsets<Field>(false, "Ljava/lang/reflect/Field;") {
Brian Carlstromdbc05252011-09-09 01:59:59 -0700446 // alphabetical references
Ian Rogers6d4d9fc2011-11-30 16:24:48 -0800447 offsets.push_back(CheckOffset(OFFSETOF_MEMBER(Field, declaring_class_), "declaringClass"));
Brian Carlstromdbc05252011-09-09 01:59:59 -0700448
449 // alphabetical 32-bit
Ian Rogers6d4d9fc2011-11-30 16:24:48 -0800450 offsets.push_back(CheckOffset(OFFSETOF_MEMBER(Field, access_flags_), "accessFlags"));
451 offsets.push_back(CheckOffset(OFFSETOF_MEMBER(Field, field_dex_idx_), "fieldDexIndex"));
452 offsets.push_back(CheckOffset(OFFSETOF_MEMBER(Field, offset_), "offset"));
Brian Carlstrom693267a2011-09-06 09:25:34 -0700453 };
454};
455
Mathieu Chartier66f19252012-09-18 08:57:04 -0700456struct AbstractMethodOffsets : public CheckOffsets<AbstractMethod> {
457 AbstractMethodOffsets() : CheckOffsets<AbstractMethod>(false, "Ljava/lang/reflect/AbstractMethod;") {
Brian Carlstromdbc05252011-09-09 01:59:59 -0700458 // alphabetical references
Mathieu Chartier66f19252012-09-18 08:57:04 -0700459 offsets.push_back(CheckOffset(OFFSETOF_MEMBER(AbstractMethod, declaring_class_), "declaringClass"));
460 offsets.push_back(CheckOffset(OFFSETOF_MEMBER(AbstractMethod, dex_cache_initialized_static_storage_), "dexCacheInitializedStaticStorage"));
461 offsets.push_back(CheckOffset(OFFSETOF_MEMBER(AbstractMethod, dex_cache_resolved_methods_), "dexCacheResolvedMethods"));
462 offsets.push_back(CheckOffset(OFFSETOF_MEMBER(AbstractMethod, dex_cache_resolved_types_), "dexCacheResolvedTypes"));
463 offsets.push_back(CheckOffset(OFFSETOF_MEMBER(AbstractMethod, dex_cache_strings_), "dexCacheStrings"));
Brian Carlstromdbc05252011-09-09 01:59:59 -0700464
465 // alphabetical 32-bit
Mathieu Chartier66f19252012-09-18 08:57:04 -0700466 offsets.push_back(CheckOffset(OFFSETOF_MEMBER(AbstractMethod, access_flags_), "accessFlags"));
467 offsets.push_back(CheckOffset(OFFSETOF_MEMBER(AbstractMethod, code_), "code"));
468 offsets.push_back(CheckOffset(OFFSETOF_MEMBER(AbstractMethod, code_item_offset_), "codeItemOffset"));
469 offsets.push_back(CheckOffset(OFFSETOF_MEMBER(AbstractMethod, core_spill_mask_), "coreSpillMask"));
470 offsets.push_back(CheckOffset(OFFSETOF_MEMBER(AbstractMethod, fp_spill_mask_), "fpSpillMask"));
471 offsets.push_back(CheckOffset(OFFSETOF_MEMBER(AbstractMethod, frame_size_in_bytes_), "frameSizeInBytes"));
472 offsets.push_back(CheckOffset(OFFSETOF_MEMBER(AbstractMethod, native_gc_map_), "gcMap"));
473 offsets.push_back(CheckOffset(OFFSETOF_MEMBER(AbstractMethod, invoke_stub_), "invokeStub"));
474 offsets.push_back(CheckOffset(OFFSETOF_MEMBER(AbstractMethod, mapping_table_), "mappingTable"));
475 offsets.push_back(CheckOffset(OFFSETOF_MEMBER(AbstractMethod, method_dex_index_), "methodDexIndex"));
476 offsets.push_back(CheckOffset(OFFSETOF_MEMBER(AbstractMethod, method_index_), "methodIndex"));
477 offsets.push_back(CheckOffset(OFFSETOF_MEMBER(AbstractMethod, native_method_), "nativeMethod"));
478 offsets.push_back(CheckOffset(OFFSETOF_MEMBER(AbstractMethod, vmap_table_), "vmapTable"));
Brian Carlstrom693267a2011-09-06 09:25:34 -0700479 };
480};
481
Mathieu Chartier66f19252012-09-18 08:57:04 -0700482struct ConstructorOffsets : public CheckOffsets<Constructor> {
483 // java.lang.reflect.Constructor is a subclass of java.lang.reflect.AbstractMethod
484 ConstructorOffsets() : CheckOffsets<Constructor>(false, "Ljava/lang/reflect/Constructor;") {
485 }
486};
487
488struct MethodOffsets : public CheckOffsets<Method> {
489 // java.lang.reflect.Method is a subclass of java.lang.reflect.AbstractMethod
490 MethodOffsets() : CheckOffsets<Method>(false, "Ljava/lang/reflect/Method;") {
Elliott Hughes80609252011-09-23 17:24:51 -0700491 }
492};
493
494struct ClassOffsets : public CheckOffsets<Class> {
Brian Carlstrom3320cf42011-10-04 14:58:28 -0700495 ClassOffsets() : CheckOffsets<Class>(false, "Ljava/lang/Class;") {
Brian Carlstromdbc05252011-09-09 01:59:59 -0700496 // alphabetical references
Ian Rogersd418eda2012-01-30 12:14:28 -0800497 offsets.push_back(CheckOffset(OFFSETOF_MEMBER(Class, class_loader_), "classLoader"));
498 offsets.push_back(CheckOffset(OFFSETOF_MEMBER(Class, component_type_), "componentType"));
499 offsets.push_back(CheckOffset(OFFSETOF_MEMBER(Class, dex_cache_), "dexCache"));
500 offsets.push_back(CheckOffset(OFFSETOF_MEMBER(Class, direct_methods_), "directMethods"));
501 offsets.push_back(CheckOffset(OFFSETOF_MEMBER(Class, ifields_), "iFields"));
502 offsets.push_back(CheckOffset(OFFSETOF_MEMBER(Class, iftable_), "ifTable"));
Brian Carlstromdbc05252011-09-09 01:59:59 -0700503 offsets.push_back(CheckOffset(OFFSETOF_MEMBER(Class, name_), "name"));
Ian Rogersd418eda2012-01-30 12:14:28 -0800504 offsets.push_back(CheckOffset(OFFSETOF_MEMBER(Class, sfields_), "sFields"));
505 offsets.push_back(CheckOffset(OFFSETOF_MEMBER(Class, super_class_), "superClass"));
506 offsets.push_back(CheckOffset(OFFSETOF_MEMBER(Class, verify_error_class_), "verifyErrorClass"));
507 offsets.push_back(CheckOffset(OFFSETOF_MEMBER(Class, virtual_methods_), "virtualMethods"));
508 offsets.push_back(CheckOffset(OFFSETOF_MEMBER(Class, vtable_), "vtable"));
Brian Carlstromdbc05252011-09-09 01:59:59 -0700509
510 // alphabetical 32-bit
Ian Rogersd418eda2012-01-30 12:14:28 -0800511 offsets.push_back(CheckOffset(OFFSETOF_MEMBER(Class, access_flags_), "accessFlags"));
512 offsets.push_back(CheckOffset(OFFSETOF_MEMBER(Class, class_size_), "classSize"));
513 offsets.push_back(CheckOffset(OFFSETOF_MEMBER(Class, clinit_thread_id_), "clinitThreadId"));
Ian Rogers6d4d9fc2011-11-30 16:24:48 -0800514 offsets.push_back(CheckOffset(OFFSETOF_MEMBER(Class, dex_type_idx_), "dexTypeIndex"));
Ian Rogersd418eda2012-01-30 12:14:28 -0800515 offsets.push_back(CheckOffset(OFFSETOF_MEMBER(Class, num_reference_instance_fields_), "numReferenceInstanceFields"));
516 offsets.push_back(CheckOffset(OFFSETOF_MEMBER(Class, num_reference_static_fields_), "numReferenceStaticFields"));
517 offsets.push_back(CheckOffset(OFFSETOF_MEMBER(Class, object_size_), "objectSize"));
518 offsets.push_back(CheckOffset(OFFSETOF_MEMBER(Class, primitive_type_), "primitiveType"));
519 offsets.push_back(CheckOffset(OFFSETOF_MEMBER(Class, reference_instance_offsets_), "referenceInstanceOffsets"));
520 offsets.push_back(CheckOffset(OFFSETOF_MEMBER(Class, reference_static_offsets_), "referenceStaticOffsets"));
521 offsets.push_back(CheckOffset(OFFSETOF_MEMBER(Class, status_), "status"));
Brian Carlstrom693267a2011-09-06 09:25:34 -0700522 };
523};
524
Elliott Hughes80609252011-09-23 17:24:51 -0700525struct StringOffsets : public CheckOffsets<String> {
Brian Carlstrom3320cf42011-10-04 14:58:28 -0700526 StringOffsets() : CheckOffsets<String>(false, "Ljava/lang/String;") {
Brian Carlstromdbc05252011-09-09 01:59:59 -0700527 // alphabetical references
528 offsets.push_back(CheckOffset(OFFSETOF_MEMBER(String, array_), "value"));
529
530 // alphabetical 32-bit
531 offsets.push_back(CheckOffset(OFFSETOF_MEMBER(String, count_), "count"));
532 offsets.push_back(CheckOffset(OFFSETOF_MEMBER(String, hash_code_), "hashCode"));
533 offsets.push_back(CheckOffset(OFFSETOF_MEMBER(String, offset_), "offset"));
Brian Carlstrom693267a2011-09-06 09:25:34 -0700534 };
535};
536
Elliott Hughes80609252011-09-23 17:24:51 -0700537struct ThrowableOffsets : public CheckOffsets<Throwable> {
Brian Carlstrom3320cf42011-10-04 14:58:28 -0700538 ThrowableOffsets() : CheckOffsets<Throwable>(false, "Ljava/lang/Throwable;") {
Brian Carlstromdbc05252011-09-09 01:59:59 -0700539 // alphabetical references
540 offsets.push_back(CheckOffset(OFFSETOF_MEMBER(Throwable, cause_), "cause"));
541 offsets.push_back(CheckOffset(OFFSETOF_MEMBER(Throwable, detail_message_), "detailMessage"));
542 offsets.push_back(CheckOffset(OFFSETOF_MEMBER(Throwable, stack_state_), "stackState"));
543 offsets.push_back(CheckOffset(OFFSETOF_MEMBER(Throwable, stack_trace_), "stackTrace"));
544 offsets.push_back(CheckOffset(OFFSETOF_MEMBER(Throwable, suppressed_exceptions_), "suppressedExceptions"));
Brian Carlstrom693267a2011-09-06 09:25:34 -0700545 };
546};
547
Elliott Hughes80609252011-09-23 17:24:51 -0700548struct StackTraceElementOffsets : public CheckOffsets<StackTraceElement> {
Brian Carlstrom3320cf42011-10-04 14:58:28 -0700549 StackTraceElementOffsets() : CheckOffsets<StackTraceElement>(false, "Ljava/lang/StackTraceElement;") {
Brian Carlstromdbc05252011-09-09 01:59:59 -0700550 // alphabetical references
551 offsets.push_back(CheckOffset(OFFSETOF_MEMBER(StackTraceElement, declaring_class_), "declaringClass"));
552 offsets.push_back(CheckOffset(OFFSETOF_MEMBER(StackTraceElement, file_name_), "fileName"));
553 offsets.push_back(CheckOffset(OFFSETOF_MEMBER(StackTraceElement, method_name_), "methodName"));
554 offsets.push_back(CheckOffset(OFFSETOF_MEMBER(StackTraceElement, line_number_), "lineNumber"));
Brian Carlstrom693267a2011-09-06 09:25:34 -0700555 };
556};
557
Elliott Hughes80609252011-09-23 17:24:51 -0700558struct ClassLoaderOffsets : public CheckOffsets<ClassLoader> {
Brian Carlstrom3320cf42011-10-04 14:58:28 -0700559 ClassLoaderOffsets() : CheckOffsets<ClassLoader>(false, "Ljava/lang/ClassLoader;") {
Brian Carlstromdbc05252011-09-09 01:59:59 -0700560 // alphabetical references
Brian Carlstrom87293d02012-04-01 19:53:04 -0700561 offsets.push_back(CheckOffset(OFFSETOF_MEMBER(ClassLoader, packages_), "packages"));
562 offsets.push_back(CheckOffset(OFFSETOF_MEMBER(ClassLoader, parent_), "parent"));
563 offsets.push_back(CheckOffset(OFFSETOF_MEMBER(ClassLoader, proxyCache_), "proxyCache"));
Brian Carlstrom693267a2011-09-06 09:25:34 -0700564 };
565};
566
Jesse Wilson95caa792011-10-12 18:14:17 -0400567struct ProxyOffsets : public CheckOffsets<Proxy> {
568 ProxyOffsets() : CheckOffsets<Proxy>(false, "Ljava/lang/reflect/Proxy;") {
Jesse Wilson95caa792011-10-12 18:14:17 -0400569 // alphabetical references
570 offsets.push_back(CheckOffset(OFFSETOF_MEMBER(Proxy, h_), "h"));
571 };
572};
573
Elliott Hughes80609252011-09-23 17:24:51 -0700574struct ClassClassOffsets : public CheckOffsets<ClassClass> {
Brian Carlstrom3320cf42011-10-04 14:58:28 -0700575 ClassClassOffsets() : CheckOffsets<ClassClass>(true, "Ljava/lang/Class;") {
Brian Carlstrom5b8e4c82011-09-18 01:38:59 -0700576 // padding 32-bit
577 CHECK_EQ(OFFSETOF_MEMBER(ClassClass, padding_) + 4,
578 OFFSETOF_MEMBER(ClassClass, serialVersionUID_));
579
Brian Carlstromdbc05252011-09-09 01:59:59 -0700580 // alphabetical 64-bit
581 offsets.push_back(CheckOffset(OFFSETOF_MEMBER(ClassClass, serialVersionUID_), "serialVersionUID"));
582 };
583};
584
Elliott Hughes80609252011-09-23 17:24:51 -0700585struct StringClassOffsets : public CheckOffsets<StringClass> {
Brian Carlstrom3320cf42011-10-04 14:58:28 -0700586 StringClassOffsets() : CheckOffsets<StringClass>(true, "Ljava/lang/String;") {
Brian Carlstromdbc05252011-09-09 01:59:59 -0700587 // alphabetical references
588 offsets.push_back(CheckOffset(OFFSETOF_MEMBER(StringClass, ASCII_), "ASCII"));
589 offsets.push_back(CheckOffset(OFFSETOF_MEMBER(StringClass, CASE_INSENSITIVE_ORDER_), "CASE_INSENSITIVE_ORDER"));
590
Brian Carlstrom5b8e4c82011-09-18 01:38:59 -0700591 // padding 32-bit
592 offsets.push_back(CheckOffset(OFFSETOF_MEMBER(StringClass, REPLACEMENT_CHAR_), "REPLACEMENT_CHAR"));
593
Brian Carlstromdbc05252011-09-09 01:59:59 -0700594 // alphabetical 64-bit
595 offsets.push_back(CheckOffset(OFFSETOF_MEMBER(StringClass, serialVersionUID_), "serialVersionUID"));
596 };
597};
598
Elliott Hughes80609252011-09-23 17:24:51 -0700599struct FieldClassOffsets : public CheckOffsets<FieldClass> {
Brian Carlstrom3320cf42011-10-04 14:58:28 -0700600 FieldClassOffsets() : CheckOffsets<FieldClass>(true, "Ljava/lang/reflect/Field;") {
Brian Carlstromdbc05252011-09-09 01:59:59 -0700601 // alphabetical references
602 offsets.push_back(CheckOffset(OFFSETOF_MEMBER(FieldClass, ORDER_BY_NAME_AND_DECLARING_CLASS_), "ORDER_BY_NAME_AND_DECLARING_CLASS"));
Brian Carlstromdbc05252011-09-09 01:59:59 -0700603 };
604};
605
Elliott Hughes80609252011-09-23 17:24:51 -0700606struct MethodClassOffsets : public CheckOffsets<MethodClass> {
Brian Carlstrom3320cf42011-10-04 14:58:28 -0700607 MethodClassOffsets() : CheckOffsets<MethodClass>(true, "Ljava/lang/reflect/Method;") {
Brian Carlstromdbc05252011-09-09 01:59:59 -0700608 // alphabetical references
Brian Carlstromdbc05252011-09-09 01:59:59 -0700609 offsets.push_back(CheckOffset(OFFSETOF_MEMBER(MethodClass, ORDER_BY_SIGNATURE_), "ORDER_BY_SIGNATURE"));
Brian Carlstrom693267a2011-09-06 09:25:34 -0700610 };
611};
612
Mathieu Chartier66f19252012-09-18 08:57:04 -0700613struct DexCacheOffsets : public CheckOffsets<DexCache> {
614 DexCacheOffsets() : CheckOffsets<DexCache>(false, "Ljava/lang/DexCache;") {
615 // alphabetical references
616 offsets.push_back(CheckOffset(OFFSETOF_MEMBER(DexCache, initialized_static_storage_), "initializedStaticStorage"));
617 offsets.push_back(CheckOffset(OFFSETOF_MEMBER(DexCache, location_), "location"));
618 offsets.push_back(CheckOffset(OFFSETOF_MEMBER(DexCache, resolved_fields_), "resolvedFields"));
619 offsets.push_back(CheckOffset(OFFSETOF_MEMBER(DexCache, resolved_methods_), "resolvedMethods"));
620 offsets.push_back(CheckOffset(OFFSETOF_MEMBER(DexCache, resolved_types_), "resolvedTypes"));
621 offsets.push_back(CheckOffset(OFFSETOF_MEMBER(DexCache, strings_), "strings"));
622 offsets.push_back(CheckOffset(OFFSETOF_MEMBER(DexCache, dex_file_), "dexFile"));
623 };
624};
625
Jesse Wilson46cdd4b2011-07-28 17:40:48 -0400626// C++ fields must exactly match the fields in the Java classes. If this fails,
Jesse Wilsonac5b9e22011-07-27 15:11:13 -0400627// reorder the fields in the C++ class. Managed class fields are ordered by
Brian Carlstrom693267a2011-09-06 09:25:34 -0700628// ClassLinker::LinkFields.
Jesse Wilson46cdd4b2011-07-28 17:40:48 -0400629TEST_F(ClassLinkerTest, ValidateFieldOrderOfJavaCppUnionClasses) {
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700630 ScopedObjectAccess soa(Thread::Current());
Brian Carlstrom693267a2011-09-06 09:25:34 -0700631 EXPECT_TRUE(ObjectOffsets().Check());
Elliott Hughes80609252011-09-23 17:24:51 -0700632 EXPECT_TRUE(ConstructorOffsets().Check());
Brian Carlstrom693267a2011-09-06 09:25:34 -0700633 EXPECT_TRUE(MethodOffsets().Check());
Mathieu Chartier66f19252012-09-18 08:57:04 -0700634 EXPECT_TRUE(FieldOffsets().Check());
635 EXPECT_TRUE(AbstractMethodOffsets().Check());
Brian Carlstrom693267a2011-09-06 09:25:34 -0700636 EXPECT_TRUE(ClassOffsets().Check());
637 EXPECT_TRUE(StringOffsets().Check());
638 EXPECT_TRUE(ThrowableOffsets().Check());
639 EXPECT_TRUE(StackTraceElementOffsets().Check());
640 EXPECT_TRUE(ClassLoaderOffsets().Check());
Jesse Wilson95caa792011-10-12 18:14:17 -0400641 EXPECT_TRUE(ProxyOffsets().Check());
Mathieu Chartier66f19252012-09-18 08:57:04 -0700642 EXPECT_TRUE(DexCacheOffsets().Check());
Brian Carlstromdbc05252011-09-09 01:59:59 -0700643
644 EXPECT_TRUE(ClassClassOffsets().Check());
645 EXPECT_TRUE(StringClassOffsets().Check());
646 EXPECT_TRUE(FieldClassOffsets().Check());
647 EXPECT_TRUE(MethodClassOffsets().Check());
Brian Carlstrom74eb46a2011-08-02 20:10:14 -0700648}
649
Brian Carlstrom5b8e4c82011-09-18 01:38:59 -0700650TEST_F(ClassLinkerTest, FindClassNonexistent) {
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700651 ScopedObjectAccess soa(Thread::Current());
Brian Carlstrom5b8e4c82011-09-18 01:38:59 -0700652 AssertNonExistentClass("NoSuchClass;");
653 AssertNonExistentClass("LNoSuchClass;");
654}
655
656TEST_F(ClassLinkerTest, FindClassNested) {
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700657 ScopedObjectAccess soa(Thread::Current());
Ian Rogers1f539342012-10-03 21:09:42 -0700658 SirtRef<ClassLoader> class_loader(soa.Self(), soa.Decode<ClassLoader*>(LoadDex("Nested")));
Brian Carlstrom5b8e4c82011-09-18 01:38:59 -0700659
Brian Carlstrom40381fb2011-10-19 14:13:40 -0700660 Class* outer = class_linker_->FindClass("LNested;", class_loader.get());
Brian Carlstrom5b8e4c82011-09-18 01:38:59 -0700661 ASSERT_TRUE(outer != NULL);
662 EXPECT_EQ(0U, outer->NumVirtualMethods());
663 EXPECT_EQ(1U, outer->NumDirectMethods());
664
Brian Carlstrom40381fb2011-10-19 14:13:40 -0700665 Class* inner = class_linker_->FindClass("LNested$Inner;", class_loader.get());
Brian Carlstrom5b8e4c82011-09-18 01:38:59 -0700666 ASSERT_TRUE(inner != NULL);
667 EXPECT_EQ(0U, inner->NumVirtualMethods());
668 EXPECT_EQ(1U, inner->NumDirectMethods());
669}
670
671TEST_F(ClassLinkerTest, FindClass_Primitives) {
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700672 ScopedObjectAccess soa(Thread::Current());
Brian Carlstromaded5f72011-10-07 17:15:04 -0700673 const std::string expected("BCDFIJSZV");
Elliott Hughesdb7d5e92011-12-16 18:47:37 -0800674 for (int ch = 1; ch < 256; ++ch) {
675 std::string descriptor;
676 descriptor.push_back(ch);
Brian Carlstromaded5f72011-10-07 17:15:04 -0700677 if (expected.find(ch) == std::string::npos) {
Brian Carlstrom5b8e4c82011-09-18 01:38:59 -0700678 AssertNonExistentClass(descriptor);
679 } else {
680 AssertPrimitiveClass(descriptor);
681 }
682 }
683}
684
685TEST_F(ClassLinkerTest, FindClass) {
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700686 ScopedObjectAccess soa(Thread::Current());
Brian Carlstrom5b8e4c82011-09-18 01:38:59 -0700687 Class* JavaLangObject = class_linker_->FindSystemClass("Ljava/lang/Object;");
Ian Rogers6d4d9fc2011-11-30 16:24:48 -0800688 ClassHelper kh(JavaLangObject);
Brian Carlstrom5b8e4c82011-09-18 01:38:59 -0700689 ASSERT_TRUE(JavaLangObject != NULL);
690 ASSERT_TRUE(JavaLangObject->GetClass() != NULL);
691 ASSERT_EQ(JavaLangObject->GetClass(), JavaLangObject->GetClass()->GetClass());
692 EXPECT_EQ(JavaLangObject, JavaLangObject->GetClass()->GetSuperClass());
Elliott Hughes91250e02011-12-13 22:30:35 -0800693 ASSERT_STREQ(kh.GetDescriptor(), "Ljava/lang/Object;");
Brian Carlstrom5b8e4c82011-09-18 01:38:59 -0700694 EXPECT_TRUE(JavaLangObject->GetSuperClass() == NULL);
695 EXPECT_FALSE(JavaLangObject->HasSuperClass());
696 EXPECT_TRUE(JavaLangObject->GetClassLoader() == NULL);
Brian Carlstrom25c33252011-09-18 15:58:35 -0700697 EXPECT_EQ(Class::kStatusResolved, JavaLangObject->GetStatus());
Brian Carlstrom5b8e4c82011-09-18 01:38:59 -0700698 EXPECT_FALSE(JavaLangObject->IsErroneous());
Brian Carlstrom25c33252011-09-18 15:58:35 -0700699 EXPECT_TRUE(JavaLangObject->IsLoaded());
Brian Carlstrom5b8e4c82011-09-18 01:38:59 -0700700 EXPECT_TRUE(JavaLangObject->IsResolved());
Brian Carlstrom25c33252011-09-18 15:58:35 -0700701 EXPECT_FALSE(JavaLangObject->IsVerified());
702 EXPECT_FALSE(JavaLangObject->IsInitialized());
Brian Carlstrom5b8e4c82011-09-18 01:38:59 -0700703 EXPECT_FALSE(JavaLangObject->IsArrayInstance());
704 EXPECT_FALSE(JavaLangObject->IsArrayClass());
705 EXPECT_TRUE(JavaLangObject->GetComponentType() == NULL);
706 EXPECT_FALSE(JavaLangObject->IsInterface());
707 EXPECT_TRUE(JavaLangObject->IsPublic());
708 EXPECT_FALSE(JavaLangObject->IsFinal());
709 EXPECT_FALSE(JavaLangObject->IsPrimitive());
710 EXPECT_FALSE(JavaLangObject->IsSynthetic());
711 EXPECT_EQ(2U, JavaLangObject->NumDirectMethods());
712 EXPECT_EQ(11U, JavaLangObject->NumVirtualMethods());
713 EXPECT_EQ(2U, JavaLangObject->NumInstanceFields());
Ian Rogers6d4d9fc2011-11-30 16:24:48 -0800714 FieldHelper fh(JavaLangObject->GetInstanceField(0));
715 EXPECT_STREQ(fh.GetName(), "shadow$_klass_");
716 fh.ChangeField(JavaLangObject->GetInstanceField(1));
717 EXPECT_STREQ(fh.GetName(), "shadow$_monitor_");
Brian Carlstrom5b8e4c82011-09-18 01:38:59 -0700718
719 EXPECT_EQ(0U, JavaLangObject->NumStaticFields());
Ian Rogersd24e2642012-06-06 21:21:43 -0700720 EXPECT_EQ(0U, kh.NumDirectInterfaces());
Brian Carlstrom5b8e4c82011-09-18 01:38:59 -0700721
Ian Rogers1f539342012-10-03 21:09:42 -0700722 SirtRef<ClassLoader> class_loader(soa.Self(), soa.Decode<ClassLoader*>(LoadDex("MyClass")));
Brian Carlstrom5b8e4c82011-09-18 01:38:59 -0700723 AssertNonExistentClass("LMyClass;");
Brian Carlstrom40381fb2011-10-19 14:13:40 -0700724 Class* MyClass = class_linker_->FindClass("LMyClass;", class_loader.get());
Ian Rogers6d4d9fc2011-11-30 16:24:48 -0800725 kh.ChangeClass(MyClass);
Brian Carlstrom5b8e4c82011-09-18 01:38:59 -0700726 ASSERT_TRUE(MyClass != NULL);
727 ASSERT_TRUE(MyClass->GetClass() != NULL);
728 ASSERT_EQ(MyClass->GetClass(), MyClass->GetClass()->GetClass());
729 EXPECT_EQ(JavaLangObject, MyClass->GetClass()->GetSuperClass());
Elliott Hughes91250e02011-12-13 22:30:35 -0800730 ASSERT_STREQ(kh.GetDescriptor(), "LMyClass;");
Brian Carlstrom5b8e4c82011-09-18 01:38:59 -0700731 EXPECT_TRUE(MyClass->GetSuperClass() == JavaLangObject);
732 EXPECT_TRUE(MyClass->HasSuperClass());
Brian Carlstrom40381fb2011-10-19 14:13:40 -0700733 EXPECT_EQ(class_loader.get(), MyClass->GetClassLoader());
Brian Carlstrom25c33252011-09-18 15:58:35 -0700734 EXPECT_EQ(Class::kStatusResolved, MyClass->GetStatus());
Brian Carlstrom5b8e4c82011-09-18 01:38:59 -0700735 EXPECT_FALSE(MyClass->IsErroneous());
Brian Carlstrom25c33252011-09-18 15:58:35 -0700736 EXPECT_TRUE(MyClass->IsLoaded());
Brian Carlstrom5b8e4c82011-09-18 01:38:59 -0700737 EXPECT_TRUE(MyClass->IsResolved());
Brian Carlstrom25c33252011-09-18 15:58:35 -0700738 EXPECT_FALSE(MyClass->IsVerified());
739 EXPECT_FALSE(MyClass->IsInitialized());
Brian Carlstrom5b8e4c82011-09-18 01:38:59 -0700740 EXPECT_FALSE(MyClass->IsArrayInstance());
741 EXPECT_FALSE(MyClass->IsArrayClass());
742 EXPECT_TRUE(MyClass->GetComponentType() == NULL);
743 EXPECT_FALSE(MyClass->IsInterface());
744 EXPECT_FALSE(MyClass->IsPublic());
745 EXPECT_FALSE(MyClass->IsFinal());
746 EXPECT_FALSE(MyClass->IsPrimitive());
747 EXPECT_FALSE(MyClass->IsSynthetic());
748 EXPECT_EQ(1U, MyClass->NumDirectMethods());
749 EXPECT_EQ(0U, MyClass->NumVirtualMethods());
750 EXPECT_EQ(0U, MyClass->NumInstanceFields());
751 EXPECT_EQ(0U, MyClass->NumStaticFields());
Ian Rogersd24e2642012-06-06 21:21:43 -0700752 EXPECT_EQ(0U, kh.NumDirectInterfaces());
Brian Carlstrom5b8e4c82011-09-18 01:38:59 -0700753
754 EXPECT_EQ(JavaLangObject->GetClass()->GetClass(), MyClass->GetClass()->GetClass());
755
756 // created by class_linker
757 AssertArrayClass("[C", "C", NULL);
758 AssertArrayClass("[Ljava/lang/Object;", "Ljava/lang/Object;", NULL);
759 // synthesized on the fly
760 AssertArrayClass("[[C", "[C", NULL);
Brian Carlstrom40381fb2011-10-19 14:13:40 -0700761 AssertArrayClass("[[[LMyClass;", "[[LMyClass;", class_loader.get());
Brian Carlstrom5b8e4c82011-09-18 01:38:59 -0700762 // or not available at all
763 AssertNonExistentClass("[[[[LNonExistentClass;");
764}
765
766TEST_F(ClassLinkerTest, LibCore) {
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700767 ScopedObjectAccess soa(Thread::Current());
Brian Carlstroma004aa92012-02-08 18:05:09 -0800768 AssertDexFile(java_lang_dex_file_, NULL);
Brian Carlstrom5b8e4c82011-09-18 01:38:59 -0700769}
770
Ian Rogersa15e67d2012-02-28 13:51:55 -0800771// The first reference array element must be a multiple of 4 bytes from the
Jesse Wilsondf4189c2011-08-09 17:10:28 -0400772// start of the object
773TEST_F(ClassLinkerTest, ValidateObjectArrayElementsOffset) {
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700774 ScopedObjectAccess soa(Thread::Current());
Jesse Wilsondf4189c2011-08-09 17:10:28 -0400775 Class* array_class = class_linker_->FindSystemClass("[Ljava/lang/String;");
776 ObjectArray<String>* array = ObjectArray<String>::Alloc(array_class, 0);
777 uint32_t array_offset = reinterpret_cast<uint32_t>(array);
Ian Rogers0cfe1fb2011-08-26 03:29:44 -0700778 uint32_t data_offset =
Ian Rogersa15e67d2012-02-28 13:51:55 -0800779 array_offset + ObjectArray<String>::DataOffset(sizeof(String*)).Uint32Value();
780 if (sizeof(String*) == sizeof(int32_t)) {
781 EXPECT_TRUE(IsAligned<4>(data_offset)); // Check 4 byte alignment.
782 } else {
783 EXPECT_TRUE(IsAligned<8>(data_offset)); // Check 8 byte alignment.
784 }
Jesse Wilsondf4189c2011-08-09 17:10:28 -0400785}
786
787TEST_F(ClassLinkerTest, ValidatePrimitiveArrayElementsOffset) {
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700788 ScopedObjectAccess soa(Thread::Current());
Ian Rogers1f539342012-10-03 21:09:42 -0700789 SirtRef<LongArray> long_array(soa.Self(), LongArray::Alloc(0));
Ian Rogersa15e67d2012-02-28 13:51:55 -0800790 EXPECT_EQ(class_linker_->FindSystemClass("[J"), long_array->GetClass());
791 uintptr_t data_offset = reinterpret_cast<uintptr_t>(long_array->GetData());
792 EXPECT_TRUE(IsAligned<8>(data_offset)); // Longs require 8 byte alignment
793
Ian Rogers1f539342012-10-03 21:09:42 -0700794 SirtRef<DoubleArray> double_array(soa.Self(), DoubleArray::Alloc(0));
Ian Rogersa15e67d2012-02-28 13:51:55 -0800795 EXPECT_EQ(class_linker_->FindSystemClass("[D"), double_array->GetClass());
796 data_offset = reinterpret_cast<uintptr_t>(double_array->GetData());
797 EXPECT_TRUE(IsAligned<8>(data_offset)); // Doubles require 8 byte alignment
798
Ian Rogers1f539342012-10-03 21:09:42 -0700799 SirtRef<IntArray> int_array(soa.Self(), IntArray::Alloc(0));
Ian Rogersa15e67d2012-02-28 13:51:55 -0800800 EXPECT_EQ(class_linker_->FindSystemClass("[I"), int_array->GetClass());
801 data_offset = reinterpret_cast<uintptr_t>(int_array->GetData());
802 EXPECT_TRUE(IsAligned<4>(data_offset)); // Ints require 4 byte alignment
803
Ian Rogers1f539342012-10-03 21:09:42 -0700804 SirtRef<CharArray> char_array(soa.Self(), CharArray::Alloc(0));
Ian Rogersa15e67d2012-02-28 13:51:55 -0800805 EXPECT_EQ(class_linker_->FindSystemClass("[C"), char_array->GetClass());
806 data_offset = reinterpret_cast<uintptr_t>(char_array->GetData());
807 EXPECT_TRUE(IsAligned<2>(data_offset)); // Chars require 2 byte alignment
808
Ian Rogers1f539342012-10-03 21:09:42 -0700809 SirtRef<ShortArray> short_array(soa.Self(), ShortArray::Alloc(0));
Ian Rogersa15e67d2012-02-28 13:51:55 -0800810 EXPECT_EQ(class_linker_->FindSystemClass("[S"), short_array->GetClass());
811 data_offset = reinterpret_cast<uintptr_t>(short_array->GetData());
812 EXPECT_TRUE(IsAligned<2>(data_offset)); // Shorts require 2 byte alignment
813
814 // Take it as given that bytes and booleans have byte alignment
Jesse Wilsondf4189c2011-08-09 17:10:28 -0400815}
816
Elliott Hughes33203b52011-09-20 19:42:01 -0700817TEST_F(ClassLinkerTest, ValidateBoxedTypes) {
818 // Validate that the "value" field is always the 0th field in each of java.lang's box classes.
819 // This lets UnboxPrimitive avoid searching for the field by name at runtime.
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700820 ScopedObjectAccess soa(Thread::Current());
Elliott Hughes33203b52011-09-20 19:42:01 -0700821 Class* c;
822 c = class_linker_->FindClass("Ljava/lang/Boolean;", NULL);
Ian Rogers6d4d9fc2011-11-30 16:24:48 -0800823 FieldHelper fh(c->GetIFields()->Get(0));
824 EXPECT_STREQ("value", fh.GetName());
Elliott Hughes33203b52011-09-20 19:42:01 -0700825 c = class_linker_->FindClass("Ljava/lang/Byte;", NULL);
Ian Rogers6d4d9fc2011-11-30 16:24:48 -0800826 fh.ChangeField(c->GetIFields()->Get(0));
827 EXPECT_STREQ("value", fh.GetName());
Elliott Hughes33203b52011-09-20 19:42:01 -0700828 c = class_linker_->FindClass("Ljava/lang/Character;", NULL);
Ian Rogers6d4d9fc2011-11-30 16:24:48 -0800829 fh.ChangeField(c->GetIFields()->Get(0));
830 EXPECT_STREQ("value", fh.GetName());
Elliott Hughes33203b52011-09-20 19:42:01 -0700831 c = class_linker_->FindClass("Ljava/lang/Double;", NULL);
Ian Rogers6d4d9fc2011-11-30 16:24:48 -0800832 fh.ChangeField(c->GetIFields()->Get(0));
833 EXPECT_STREQ("value", fh.GetName());
Elliott Hughes33203b52011-09-20 19:42:01 -0700834 c = class_linker_->FindClass("Ljava/lang/Float;", NULL);
Ian Rogers6d4d9fc2011-11-30 16:24:48 -0800835 fh.ChangeField(c->GetIFields()->Get(0));
836 EXPECT_STREQ("value", fh.GetName());
Elliott Hughes33203b52011-09-20 19:42:01 -0700837 c = class_linker_->FindClass("Ljava/lang/Integer;", NULL);
Ian Rogers6d4d9fc2011-11-30 16:24:48 -0800838 fh.ChangeField(c->GetIFields()->Get(0));
839 EXPECT_STREQ("value", fh.GetName());
Elliott Hughes33203b52011-09-20 19:42:01 -0700840 c = class_linker_->FindClass("Ljava/lang/Long;", NULL);
Ian Rogers6d4d9fc2011-11-30 16:24:48 -0800841 fh.ChangeField(c->GetIFields()->Get(0));
842 EXPECT_STREQ("value", fh.GetName());
Elliott Hughes33203b52011-09-20 19:42:01 -0700843 c = class_linker_->FindClass("Ljava/lang/Short;", NULL);
Ian Rogers6d4d9fc2011-11-30 16:24:48 -0800844 fh.ChangeField(c->GetIFields()->Get(0));
845 EXPECT_STREQ("value", fh.GetName());
Elliott Hughes33203b52011-09-20 19:42:01 -0700846}
847
Brian Carlstrom74eb46a2011-08-02 20:10:14 -0700848TEST_F(ClassLinkerTest, TwoClassLoadersOneClass) {
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700849 ScopedObjectAccess soa(Thread::Current());
Ian Rogers1f539342012-10-03 21:09:42 -0700850 SirtRef<ClassLoader> class_loader_1(soa.Self(), soa.Decode<ClassLoader*>(LoadDex("MyClass")));
851 SirtRef<ClassLoader> class_loader_2(soa.Self(), soa.Decode<ClassLoader*>(LoadDex("MyClass")));
Brian Carlstrom40381fb2011-10-19 14:13:40 -0700852 Class* MyClass_1 = class_linker_->FindClass("LMyClass;", class_loader_1.get());
853 Class* MyClass_2 = class_linker_->FindClass("LMyClass;", class_loader_2.get());
Brian Carlstrom74eb46a2011-08-02 20:10:14 -0700854 EXPECT_TRUE(MyClass_1 != NULL);
855 EXPECT_TRUE(MyClass_2 != NULL);
856 EXPECT_NE(MyClass_1, MyClass_2);
Jesse Wilsonac5b9e22011-07-27 15:11:13 -0400857}
858
Jesse Wilson7833bd22011-08-09 18:31:44 -0400859TEST_F(ClassLinkerTest, StaticFields) {
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700860 ScopedObjectAccess soa(Thread::Current());
Ian Rogers1f539342012-10-03 21:09:42 -0700861 SirtRef<ClassLoader> class_loader(soa.Self(), soa.Decode<ClassLoader*>(LoadDex("Statics")));
Brian Carlstrom40381fb2011-10-19 14:13:40 -0700862 Class* statics = class_linker_->FindClass("LStatics;", class_loader.get());
Ian Rogers0045a292012-03-31 21:08:41 -0700863 class_linker_->EnsureInitialized(statics, true, true);
Jesse Wilson7833bd22011-08-09 18:31:44 -0400864
jeffhaoabcfde32011-09-29 15:05:18 -0700865 // Static final primitives that are initialized by a compile-time constant
866 // expression resolve to a copy of a constant value from the constant pool.
867 // So <clinit> should be null.
Mathieu Chartier66f19252012-09-18 08:57:04 -0700868 AbstractMethod* clinit = statics->FindDirectMethod("<clinit>", "()V");
jeffhaoabcfde32011-09-29 15:05:18 -0700869 EXPECT_TRUE(clinit == NULL);
870
871 EXPECT_EQ(9U, statics->NumStaticFields());
Jesse Wilson7833bd22011-08-09 18:31:44 -0400872
Brian Carlstrom6b4ef022011-10-23 14:59:04 -0700873 Field* s0 = statics->FindStaticField("s0", "Z");
Ian Rogers6d4d9fc2011-11-30 16:24:48 -0800874 FieldHelper fh(s0);
Elliott Hughes91250e02011-12-13 22:30:35 -0800875 EXPECT_STREQ(ClassHelper(s0->GetClass()).GetDescriptor(), "Ljava/lang/reflect/Field;");
Ian Rogers6d4d9fc2011-11-30 16:24:48 -0800876 EXPECT_TRUE(fh.GetTypeAsPrimitiveType() == Primitive::kPrimBoolean);
jeffhaoabcfde32011-09-29 15:05:18 -0700877 EXPECT_EQ(true, s0->GetBoolean(NULL));
Brian Carlstrom4873d462011-08-21 15:23:39 -0700878 s0->SetBoolean(NULL, false);
Jesse Wilson7833bd22011-08-09 18:31:44 -0400879
Brian Carlstrom6b4ef022011-10-23 14:59:04 -0700880 Field* s1 = statics->FindStaticField("s1", "B");
Ian Rogers6d4d9fc2011-11-30 16:24:48 -0800881 fh.ChangeField(s1);
882 EXPECT_TRUE(fh.GetTypeAsPrimitiveType() == Primitive::kPrimByte);
jeffhaoabcfde32011-09-29 15:05:18 -0700883 EXPECT_EQ(5, s1->GetByte(NULL));
Brian Carlstrom4873d462011-08-21 15:23:39 -0700884 s1->SetByte(NULL, 6);
Jesse Wilson7833bd22011-08-09 18:31:44 -0400885
Brian Carlstrom6b4ef022011-10-23 14:59:04 -0700886 Field* s2 = statics->FindStaticField("s2", "C");
Ian Rogers6d4d9fc2011-11-30 16:24:48 -0800887 fh.ChangeField(s2);
888 EXPECT_TRUE(fh.GetTypeAsPrimitiveType() == Primitive::kPrimChar);
jeffhaoabcfde32011-09-29 15:05:18 -0700889 EXPECT_EQ('a', s2->GetChar(NULL));
Brian Carlstrom4873d462011-08-21 15:23:39 -0700890 s2->SetChar(NULL, 'b');
Jesse Wilson7833bd22011-08-09 18:31:44 -0400891
Brian Carlstrom6b4ef022011-10-23 14:59:04 -0700892 Field* s3 = statics->FindStaticField("s3", "S");
Ian Rogers6d4d9fc2011-11-30 16:24:48 -0800893 fh.ChangeField(s3);
894 EXPECT_TRUE(fh.GetTypeAsPrimitiveType() == Primitive::kPrimShort);
Ian Rogers466bb252011-10-14 03:29:56 -0700895 EXPECT_EQ(-536, s3->GetShort(NULL));
896 s3->SetShort(NULL, -535);
Jesse Wilson7833bd22011-08-09 18:31:44 -0400897
Brian Carlstrom6b4ef022011-10-23 14:59:04 -0700898 Field* s4 = statics->FindStaticField("s4", "I");
Ian Rogers6d4d9fc2011-11-30 16:24:48 -0800899 fh.ChangeField(s4);
900 EXPECT_TRUE(fh.GetTypeAsPrimitiveType() == Primitive::kPrimInt);
jeffhaoabcfde32011-09-29 15:05:18 -0700901 EXPECT_EQ(2000000000, s4->GetInt(NULL));
Brian Carlstrom4873d462011-08-21 15:23:39 -0700902 s4->SetInt(NULL, 2000000001);
Jesse Wilson7833bd22011-08-09 18:31:44 -0400903
Brian Carlstrom6b4ef022011-10-23 14:59:04 -0700904 Field* s5 = statics->FindStaticField("s5", "J");
Ian Rogers6d4d9fc2011-11-30 16:24:48 -0800905 fh.ChangeField(s5);
906 EXPECT_TRUE(fh.GetTypeAsPrimitiveType() == Primitive::kPrimLong);
jeffhaoabcfde32011-09-29 15:05:18 -0700907 EXPECT_EQ(0x1234567890abcdefLL, s5->GetLong(NULL));
Brian Carlstrom4873d462011-08-21 15:23:39 -0700908 s5->SetLong(NULL, 0x34567890abcdef12LL);
Jesse Wilson7833bd22011-08-09 18:31:44 -0400909
Brian Carlstrom6b4ef022011-10-23 14:59:04 -0700910 Field* s6 = statics->FindStaticField("s6", "F");
Ian Rogers6d4d9fc2011-11-30 16:24:48 -0800911 fh.ChangeField(s6);
912 EXPECT_TRUE(fh.GetTypeAsPrimitiveType() == Primitive::kPrimFloat);
jeffhaoabcfde32011-09-29 15:05:18 -0700913 EXPECT_EQ(0.5, s6->GetFloat(NULL));
Brian Carlstrom4873d462011-08-21 15:23:39 -0700914 s6->SetFloat(NULL, 0.75);
Jesse Wilson7833bd22011-08-09 18:31:44 -0400915
Brian Carlstrom6b4ef022011-10-23 14:59:04 -0700916 Field* s7 = statics->FindStaticField("s7", "D");
Ian Rogers6d4d9fc2011-11-30 16:24:48 -0800917 fh.ChangeField(s7);
918 EXPECT_TRUE(fh.GetTypeAsPrimitiveType() == Primitive::kPrimDouble);
jeffhaoabcfde32011-09-29 15:05:18 -0700919 EXPECT_EQ(16777217, s7->GetDouble(NULL));
Brian Carlstrom4873d462011-08-21 15:23:39 -0700920 s7->SetDouble(NULL, 16777219);
Jesse Wilson7833bd22011-08-09 18:31:44 -0400921
Brian Carlstrom6b4ef022011-10-23 14:59:04 -0700922 Field* s8 = statics->FindStaticField("s8", "Ljava/lang/String;");
Ian Rogers6d4d9fc2011-11-30 16:24:48 -0800923 fh.ChangeField(s8);
924 EXPECT_TRUE(fh.GetTypeAsPrimitiveType() == Primitive::kPrimNot);
jeffhaoabcfde32011-09-29 15:05:18 -0700925 EXPECT_TRUE(s8->GetObject(NULL)->AsString()->Equals("android"));
Brian Carlstrom4873d462011-08-21 15:23:39 -0700926 s8->SetObject(NULL, String::AllocFromModifiedUtf8("robot"));
Jesse Wilson7833bd22011-08-09 18:31:44 -0400927
Brian Carlstrom2e3d1b22012-01-09 18:01:56 -0800928 // TODO: Remove EXPECT_FALSE when GCC can handle EXPECT_EQ
929 // http://code.google.com/p/googletest/issues/detail?id=322
930 EXPECT_FALSE( s0->GetBoolean(NULL));
Brian Carlstrom4873d462011-08-21 15:23:39 -0700931 EXPECT_EQ(6, s1->GetByte(NULL));
932 EXPECT_EQ('b', s2->GetChar(NULL));
Ian Rogers466bb252011-10-14 03:29:56 -0700933 EXPECT_EQ(-535, s3->GetShort(NULL));
Brian Carlstrom4873d462011-08-21 15:23:39 -0700934 EXPECT_EQ(2000000001, s4->GetInt(NULL));
935 EXPECT_EQ(0x34567890abcdef12LL, s5->GetLong(NULL));
936 EXPECT_EQ(0.75, s6->GetFloat(NULL));
937 EXPECT_EQ(16777219, s7->GetDouble(NULL));
938 EXPECT_TRUE(s8->GetObject(NULL)->AsString()->Equals("robot"));
Jesse Wilson7833bd22011-08-09 18:31:44 -0400939}
940
Brian Carlstrom30b94452011-08-25 21:35:26 -0700941TEST_F(ClassLinkerTest, Interfaces) {
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700942 ScopedObjectAccess soa(Thread::Current());
Ian Rogers1f539342012-10-03 21:09:42 -0700943 SirtRef<ClassLoader> class_loader(soa.Self(), soa.Decode<ClassLoader*>(LoadDex("Interfaces")));
Brian Carlstrom40381fb2011-10-19 14:13:40 -0700944 Class* I = class_linker_->FindClass("LInterfaces$I;", class_loader.get());
945 Class* J = class_linker_->FindClass("LInterfaces$J;", class_loader.get());
946 Class* K = class_linker_->FindClass("LInterfaces$K;", class_loader.get());
947 Class* A = class_linker_->FindClass("LInterfaces$A;", class_loader.get());
948 Class* B = class_linker_->FindClass("LInterfaces$B;", class_loader.get());
Brian Carlstrom30b94452011-08-25 21:35:26 -0700949 EXPECT_TRUE(I->IsAssignableFrom(A));
950 EXPECT_TRUE(J->IsAssignableFrom(A));
jeffhao5dbddee2011-09-07 16:38:26 -0700951 EXPECT_TRUE(J->IsAssignableFrom(K));
952 EXPECT_TRUE(K->IsAssignableFrom(B));
953 EXPECT_TRUE(J->IsAssignableFrom(B));
Brian Carlstrom30b94452011-08-25 21:35:26 -0700954
Mathieu Chartier66f19252012-09-18 08:57:04 -0700955 AbstractMethod* Ii = I->FindVirtualMethod("i", "()V");
956 AbstractMethod* Jj1 = J->FindVirtualMethod("j1", "()V");
957 AbstractMethod* Jj2 = J->FindVirtualMethod("j2", "()V");
958 AbstractMethod* Kj1 = K->FindInterfaceMethod("j1", "()V");
959 AbstractMethod* Kj2 = K->FindInterfaceMethod("j2", "()V");
960 AbstractMethod* Kk = K->FindInterfaceMethod("k", "()V");
961 AbstractMethod* Ai = A->FindVirtualMethod("i", "()V");
962 AbstractMethod* Aj1 = A->FindVirtualMethod("j1", "()V");
963 AbstractMethod* Aj2 = A->FindVirtualMethod("j2", "()V");
Brian Carlstrom30b94452011-08-25 21:35:26 -0700964 ASSERT_TRUE(Ii != NULL);
965 ASSERT_TRUE(Jj1 != NULL);
966 ASSERT_TRUE(Jj2 != NULL);
jeffhao5dbddee2011-09-07 16:38:26 -0700967 ASSERT_TRUE(Kj1 != NULL);
968 ASSERT_TRUE(Kj2 != NULL);
969 ASSERT_TRUE(Kk != NULL);
Brian Carlstrom30b94452011-08-25 21:35:26 -0700970 ASSERT_TRUE(Ai != NULL);
971 ASSERT_TRUE(Aj1 != NULL);
972 ASSERT_TRUE(Aj2 != NULL);
Brian Carlstrom30b94452011-08-25 21:35:26 -0700973 EXPECT_NE(Ii, Ai);
974 EXPECT_NE(Jj1, Aj1);
975 EXPECT_NE(Jj2, Aj2);
jeffhao5dbddee2011-09-07 16:38:26 -0700976 EXPECT_EQ(Kj1, Jj1);
977 EXPECT_EQ(Kj2, Jj2);
Ian Rogersa32a6fd2012-02-06 20:18:44 -0800978 EXPECT_EQ(Ai, A->FindVirtualMethodForInterface(Ii));
979 EXPECT_EQ(Aj1, A->FindVirtualMethodForInterface(Jj1));
980 EXPECT_EQ(Aj2, A->FindVirtualMethodForInterface(Jj2));
Brian Carlstrom30b94452011-08-25 21:35:26 -0700981 EXPECT_EQ(Ai, A->FindVirtualMethodForVirtualOrInterface(Ii));
982 EXPECT_EQ(Aj1, A->FindVirtualMethodForVirtualOrInterface(Jj1));
983 EXPECT_EQ(Aj2, A->FindVirtualMethodForVirtualOrInterface(Jj2));
Ian Rogersd24e2642012-06-06 21:21:43 -0700984
985 Field* Afoo = A->FindStaticField("foo", "Ljava/lang/String;");
986 Field* Bfoo = B->FindStaticField("foo", "Ljava/lang/String;");
987 Field* Jfoo = J->FindStaticField("foo", "Ljava/lang/String;");
988 Field* Kfoo = K->FindStaticField("foo", "Ljava/lang/String;");
989 ASSERT_TRUE(Afoo != NULL);
990 EXPECT_EQ(Afoo, Bfoo);
991 EXPECT_EQ(Afoo, Jfoo);
992 EXPECT_EQ(Afoo, Kfoo);
Brian Carlstrom30b94452011-08-25 21:35:26 -0700993}
994
Elliott Hughesf3778f62012-01-26 14:14:35 -0800995TEST_F(ClassLinkerTest, ResolveVerifyAndClinit) {
jeffhaoabcfde32011-09-29 15:05:18 -0700996 // pretend we are trying to get the static storage for the StaticsFromCode class.
Brian Carlstrom193a44d2011-09-04 12:01:42 -0700997
jeffhaoabcfde32011-09-29 15:05:18 -0700998 // case 1, get the uninitialized storage from StaticsFromCode.<clinit>
999 // case 2, get the initialized storage from StaticsFromCode.getS0
Brian Carlstrom193a44d2011-09-04 12:01:42 -07001000
Ian Rogers00f7d0e2012-07-19 15:28:27 -07001001 ScopedObjectAccess soa(Thread::Current());
1002 jobject jclass_loader = LoadDex("StaticsFromCode");
Ian Rogers1f539342012-10-03 21:09:42 -07001003 SirtRef<ClassLoader> class_loader(soa.Self(), soa.Decode<ClassLoader*>(jclass_loader));
Ian Rogers00f7d0e2012-07-19 15:28:27 -07001004 const DexFile* dex_file = Runtime::Current()->GetCompileTimeClassPath(jclass_loader)[0];
Brian Carlstrom848a4b32011-09-04 11:29:27 -07001005 CHECK(dex_file != NULL);
1006
Brian Carlstrom40381fb2011-10-19 14:13:40 -07001007 Class* klass = class_linker_->FindClass("LStaticsFromCode;", class_loader.get());
Mathieu Chartier66f19252012-09-18 08:57:04 -07001008 AbstractMethod* clinit = klass->FindDirectMethod("<clinit>", "()V");
1009 AbstractMethod* getS0 = klass->FindDirectMethod("getS0", "()Ljava/lang/Object;");
Ian Rogers9b1a4f42011-11-14 18:35:10 -08001010 const DexFile::StringId* string_id = dex_file->FindStringId("LStaticsFromCode;");
1011 ASSERT_TRUE(string_id != NULL);
1012 const DexFile::TypeId* type_id = dex_file->FindTypeId(dex_file->GetIndexForStringId(*string_id));
1013 ASSERT_TRUE(type_id != NULL);
1014 uint32_t type_idx = dex_file->GetIndexForTypeId(*type_id);
Brian Carlstrom848a4b32011-09-04 11:29:27 -07001015 EXPECT_TRUE(clinit->GetDexCacheInitializedStaticStorage()->Get(type_idx) == NULL);
Elliott Hughesf3778f62012-01-26 14:14:35 -08001016 StaticStorageBase* uninit = ResolveVerifyAndClinit(type_idx, clinit, Thread::Current(), true, false);
Brian Carlstrom193a44d2011-09-04 12:01:42 -07001017 EXPECT_TRUE(uninit != NULL);
1018 EXPECT_TRUE(clinit->GetDexCacheInitializedStaticStorage()->Get(type_idx) == NULL);
Elliott Hughesf3778f62012-01-26 14:14:35 -08001019 StaticStorageBase* init = ResolveVerifyAndClinit(type_idx, getS0, Thread::Current(), true, false);
Brian Carlstrom193a44d2011-09-04 12:01:42 -07001020 EXPECT_TRUE(init != NULL);
1021 EXPECT_EQ(init, clinit->GetDexCacheInitializedStaticStorage()->Get(type_idx));
Brian Carlstrom848a4b32011-09-04 11:29:27 -07001022}
1023
Elliott Hughes20cde902011-10-04 17:37:27 -07001024TEST_F(ClassLinkerTest, FinalizableBit) {
Ian Rogers00f7d0e2012-07-19 15:28:27 -07001025 ScopedObjectAccess soa(Thread::Current());
Elliott Hughes20cde902011-10-04 17:37:27 -07001026 Class* c;
1027
1028 // Object has a finalize method, but we know it's empty.
1029 c = class_linker_->FindSystemClass("Ljava/lang/Object;");
1030 EXPECT_FALSE(c->IsFinalizable());
1031
1032 // Enum has a finalize method to prevent its subclasses from implementing one.
1033 c = class_linker_->FindSystemClass("Ljava/lang/Enum;");
1034 EXPECT_FALSE(c->IsFinalizable());
1035
1036 // RoundingMode is an enum.
1037 c = class_linker_->FindSystemClass("Ljava/math/RoundingMode;");
1038 EXPECT_FALSE(c->IsFinalizable());
1039
1040 // RandomAccessFile extends Object and overrides finalize.
1041 c = class_linker_->FindSystemClass("Ljava/io/RandomAccessFile;");
1042 EXPECT_TRUE(c->IsFinalizable());
1043
1044 // FileInputStream is finalizable and extends InputStream which isn't.
1045 c = class_linker_->FindSystemClass("Ljava/io/InputStream;");
1046 EXPECT_FALSE(c->IsFinalizable());
1047 c = class_linker_->FindSystemClass("Ljava/io/FileInputStream;");
1048 EXPECT_TRUE(c->IsFinalizable());
1049
1050 // ScheduledThreadPoolExecutor doesn't have a finalize method but
1051 // extends ThreadPoolExecutor which does.
1052 c = class_linker_->FindSystemClass("Ljava/util/concurrent/ThreadPoolExecutor;");
1053 EXPECT_TRUE(c->IsFinalizable());
1054 c = class_linker_->FindSystemClass("Ljava/util/concurrent/ScheduledThreadPoolExecutor;");
1055 EXPECT_TRUE(c->IsFinalizable());
1056}
1057
Ian Rogers6d4d9fc2011-11-30 16:24:48 -08001058TEST_F(ClassLinkerTest, ClassRootDescriptors) {
Ian Rogers00f7d0e2012-07-19 15:28:27 -07001059 ScopedObjectAccess soa(Thread::Current());
Ian Rogers6d4d9fc2011-11-30 16:24:48 -08001060 ClassHelper kh;
1061 for (int i = 0; i < ClassLinker::kClassRootsMax; i++) {
1062 Class* klass = class_linker_->GetClassRoot(ClassLinker::ClassRoot(i));
1063 kh.ChangeClass(klass);
1064 EXPECT_TRUE(kh.GetDescriptor() != NULL);
Elliott Hughes91250e02011-12-13 22:30:35 -08001065 EXPECT_STREQ(kh.GetDescriptor(),
Ian Rogers6d4d9fc2011-11-30 16:24:48 -08001066 class_linker_->GetClassRootDescriptor(ClassLinker::ClassRoot(i))) << " i = " << i;
1067 }
1068}
1069
Jesse Wilsondf4189c2011-08-09 17:10:28 -04001070} // namespace art