blob: a0aaa9e8c7eebd0d1242a9f9d65376a50ed763a5 [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 Shapiro894d0fa2011-06-30 14:48:49 -070016
Elliott Hughes90a33692011-08-30 13:27:07 -070017#include "object.h"
18
19#include <stdint.h>
20#include <stdio.h>
Ian Rogers700a4022014-05-19 16:49:03 -070021#include <memory>
Elliott Hughes90a33692011-08-30 13:27:07 -070022
Ian Rogers2dd0e2c2013-01-24 12:42:14 -080023#include "array-inl.h"
Brian Carlstromea46f952013-07-30 01:26:50 -070024#include "art_field-inl.h"
Ian Rogers1ff3c982014-08-12 02:30:58 -070025#include "art_method-inl.h"
Ian Rogers1f539342012-10-03 21:09:42 -070026#include "asm_support.h"
Ian Rogers2dd0e2c2013-01-24 12:42:14 -080027#include "class-inl.h"
Brian Carlstrom578bbdc2011-07-21 14:07:47 -070028#include "class_linker.h"
Ian Rogers2dd0e2c2013-01-24 12:42:14 -080029#include "class_linker-inl.h"
Brian Carlstroma1ce1fe2014-02-24 23:23:58 -080030#include "common_runtime_test.h"
Brian Carlstrom578bbdc2011-07-21 14:07:47 -070031#include "dex_file.h"
Mingyao Yang98d1cc82014-05-15 17:02:16 -070032#include "entrypoints/entrypoint_utils-inl.h"
Ian Rogers1d54e732013-05-02 21:10:01 -070033#include "gc/accounting/card_table-inl.h"
34#include "gc/heap.h"
Ian Rogers1ff3c982014-08-12 02:30:58 -070035#include "handle_scope-inl.h"
Ian Rogers2dd0e2c2013-01-24 12:42:14 -080036#include "iftable-inl.h"
Ian Rogers1ff3c982014-08-12 02:30:58 -070037#include "method_helper-inl.h"
Ian Rogers2dd0e2c2013-01-24 12:42:14 -080038#include "object-inl.h"
39#include "object_array-inl.h"
Ian Rogerse63db272014-07-15 15:36:11 -070040#include "scoped_thread_state_change.h"
Ian Rogersb0fa5dc2014-04-28 16:47:08 -070041#include "string-inl.h"
Carl Shapiro894d0fa2011-06-30 14:48:49 -070042
43namespace art {
Ian Rogers2dd0e2c2013-01-24 12:42:14 -080044namespace mirror {
Carl Shapiro894d0fa2011-06-30 14:48:49 -070045
Brian Carlstroma1ce1fe2014-02-24 23:23:58 -080046class ObjectTest : public CommonRuntimeTest {
Brian Carlstrom0b138b22011-07-27 15:19:17 -070047 protected:
Brian Carlstrom0a02d122013-09-06 15:42:40 -070048 void AssertString(int32_t expected_utf16_length,
Brian Carlstrom0b138b22011-07-27 15:19:17 -070049 const char* utf8_in,
50 const char* utf16_expected_le,
Ian Rogers00f7d0e2012-07-19 15:28:27 -070051 int32_t expected_hash)
Ian Rogersb726dcb2012-09-05 08:57:23 -070052 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
Ian Rogers700a4022014-05-19 16:49:03 -070053 std::unique_ptr<uint16_t[]> utf16_expected(new uint16_t[expected_utf16_length]);
Brian Carlstrom0a02d122013-09-06 15:42:40 -070054 for (int32_t i = 0; i < expected_utf16_length; i++) {
Brian Carlstrom0b138b22011-07-27 15:19:17 -070055 uint16_t ch = (((utf16_expected_le[i*2 + 0] & 0xff) << 8) |
56 ((utf16_expected_le[i*2 + 1] & 0xff) << 0));
57 utf16_expected[i] = ch;
58 }
59
Ian Rogers50b35e22012-10-04 10:09:15 -070060 Thread* self = Thread::Current();
Mathieu Chartiereb8167a2014-05-07 15:43:14 -070061 StackHandleScope<1> hs(self);
62 Handle<String> string(
63 hs.NewHandle(String::AllocFromModifiedUtf8(self, expected_utf16_length, utf8_in)));
Brian Carlstrom0a02d122013-09-06 15:42:40 -070064 ASSERT_EQ(expected_utf16_length, string->GetLength());
Carl Shapiro8860c0e2011-08-04 17:36:16 -070065 ASSERT_TRUE(string->GetCharArray() != NULL);
Jesse Wilsonfd687c52011-08-04 19:27:35 -070066 ASSERT_TRUE(string->GetCharArray()->GetData() != NULL);
Brian Carlstrom0a02d122013-09-06 15:42:40 -070067 // strlen is necessary because the 1-character string "\x00\x00" is interpreted as ""
68 ASSERT_TRUE(string->Equals(utf8_in) || (expected_utf16_length == 1 && strlen(utf8_in) == 0));
69 ASSERT_TRUE(string->Equals(StringPiece(utf8_in)) || (expected_utf16_length == 1 && strlen(utf8_in) == 0));
70 for (int32_t i = 0; i < expected_utf16_length; i++) {
Jesse Wilsonfd687c52011-08-04 19:27:35 -070071 EXPECT_EQ(utf16_expected[i], string->CharAt(i));
Brian Carlstrom0b138b22011-07-27 15:19:17 -070072 }
Ian Rogers0cfe1fb2011-08-26 03:29:44 -070073 EXPECT_EQ(expected_hash, string->GetHashCode());
Brian Carlstrom0b138b22011-07-27 15:19:17 -070074 }
75};
Brian Carlstroma331b3c2011-07-18 17:47:56 -070076
Ian Rogers68d8b422014-07-17 11:09:10 -070077// Keep constants in sync.
78TEST_F(ObjectTest, Constants) {
Ian Rogerscdc1aaf2014-10-09 13:21:38 -070079 EXPECT_EQ(kObjectReferenceSize, sizeof(HeapReference<Object>));
80 EXPECT_EQ(kObjectHeaderSize, sizeof(Object));
Ian Rogers68d8b422014-07-17 11:09:10 -070081}
82
Brian Carlstroma331b3c2011-07-18 17:47:56 -070083TEST_F(ObjectTest, IsInSamePackage) {
Carl Shapiro894d0fa2011-06-30 14:48:49 -070084 // Matches
Ian Rogers6d4d9fc2011-11-30 16:24:48 -080085 EXPECT_TRUE(Class::IsInSamePackage("Ljava/lang/Object;", "Ljava/lang/Class;"));
86 EXPECT_TRUE(Class::IsInSamePackage("LFoo;", "LBar;"));
Carl Shapiro894d0fa2011-06-30 14:48:49 -070087
88 // Mismatches
Ian Rogers6d4d9fc2011-11-30 16:24:48 -080089 EXPECT_FALSE(Class::IsInSamePackage("Ljava/lang/Object;", "Ljava/io/File;"));
90 EXPECT_FALSE(Class::IsInSamePackage("Ljava/lang/Object;", "Ljava/lang/reflect/Method;"));
Carl Shapiro894d0fa2011-06-30 14:48:49 -070091}
92
Elliott Hughes081be7f2011-09-18 16:50:26 -070093TEST_F(ObjectTest, Clone) {
Ian Rogers00f7d0e2012-07-19 15:28:27 -070094 ScopedObjectAccess soa(Thread::Current());
Mathieu Chartierf8322842014-05-16 10:59:25 -070095 StackHandleScope<2> hs(soa.Self());
Mathieu Chartiereb8167a2014-05-07 15:43:14 -070096 Handle<ObjectArray<Object>> a1(
97 hs.NewHandle(class_linker_->AllocObjectArray<Object>(soa.Self(), 256)));
Elliott Hughes081be7f2011-09-18 16:50:26 -070098 size_t s1 = a1->SizeOf();
Ian Rogers50b35e22012-10-04 10:09:15 -070099 Object* clone = a1->Clone(soa.Self());
Elliott Hughes081be7f2011-09-18 16:50:26 -0700100 EXPECT_EQ(s1, clone->SizeOf());
101 EXPECT_TRUE(clone->GetClass() == a1->GetClass());
102}
103
Brian Carlstrom578bbdc2011-07-21 14:07:47 -0700104TEST_F(ObjectTest, AllocObjectArray) {
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700105 ScopedObjectAccess soa(Thread::Current());
Mathieu Chartierf8322842014-05-16 10:59:25 -0700106 StackHandleScope<2> hs(soa.Self());
Ian Rogers700a4022014-05-19 16:49:03 -0700107 Handle<ObjectArray<Object>> oa(
Mathieu Chartiereb8167a2014-05-07 15:43:14 -0700108 hs.NewHandle(class_linker_->AllocObjectArray<Object>(soa.Self(), 2)));
Elliott Hughesd8ddfd52011-08-15 14:32:53 -0700109 EXPECT_EQ(2, oa->GetLength());
Brian Carlstromf7ed11a2011-08-09 17:55:51 -0700110 EXPECT_TRUE(oa->Get(0) == NULL);
111 EXPECT_TRUE(oa->Get(1) == NULL);
Mathieu Chartiereb8167a2014-05-07 15:43:14 -0700112 oa->Set<false>(0, oa.Get());
113 EXPECT_TRUE(oa->Get(0) == oa.Get());
Brian Carlstromf7ed11a2011-08-09 17:55:51 -0700114 EXPECT_TRUE(oa->Get(1) == NULL);
Mathieu Chartiereb8167a2014-05-07 15:43:14 -0700115 oa->Set<false>(1, oa.Get());
116 EXPECT_TRUE(oa->Get(0) == oa.Get());
117 EXPECT_TRUE(oa->Get(1) == oa.Get());
Brian Carlstromf7ed11a2011-08-09 17:55:51 -0700118
Ian Rogers98379392014-02-24 16:53:16 -0800119 Class* aioobe = class_linker_->FindSystemClass(soa.Self(),
120 "Ljava/lang/ArrayIndexOutOfBoundsException;");
Elliott Hughes710a0cb2011-08-16 14:32:37 -0700121
122 EXPECT_TRUE(oa->Get(-1) == NULL);
Ian Rogers1f539342012-10-03 21:09:42 -0700123 EXPECT_TRUE(soa.Self()->IsExceptionPending());
Ian Rogers62d6c772013-02-27 08:32:07 -0800124 EXPECT_EQ(aioobe, soa.Self()->GetException(NULL)->GetClass());
Ian Rogers1f539342012-10-03 21:09:42 -0700125 soa.Self()->ClearException();
Elliott Hughes710a0cb2011-08-16 14:32:37 -0700126
127 EXPECT_TRUE(oa->Get(2) == NULL);
Ian Rogers1f539342012-10-03 21:09:42 -0700128 EXPECT_TRUE(soa.Self()->IsExceptionPending());
Ian Rogers62d6c772013-02-27 08:32:07 -0800129 EXPECT_EQ(aioobe, soa.Self()->GetException(NULL)->GetClass());
Ian Rogers1f539342012-10-03 21:09:42 -0700130 soa.Self()->ClearException();
Elliott Hughes710a0cb2011-08-16 14:32:37 -0700131
Brian Carlstromf7ed11a2011-08-09 17:55:51 -0700132 ASSERT_TRUE(oa->GetClass() != NULL);
Mathieu Chartierf8322842014-05-16 10:59:25 -0700133 Handle<mirror::Class> klass(hs.NewHandle(oa->GetClass()));
134 ASSERT_EQ(2U, klass->NumDirectInterfaces());
Ian Rogers98379392014-02-24 16:53:16 -0800135 EXPECT_EQ(class_linker_->FindSystemClass(soa.Self(), "Ljava/lang/Cloneable;"),
Mathieu Chartierf8322842014-05-16 10:59:25 -0700136 mirror::Class::GetDirectInterface(soa.Self(), klass, 0));
Ian Rogers98379392014-02-24 16:53:16 -0800137 EXPECT_EQ(class_linker_->FindSystemClass(soa.Self(), "Ljava/io/Serializable;"),
Mathieu Chartierf8322842014-05-16 10:59:25 -0700138 mirror::Class::GetDirectInterface(soa.Self(), klass, 1));
Carl Shapiro0e5d75d2011-07-06 18:28:37 -0700139}
140
Elliott Hughes68f4fa02011-08-21 10:46:59 -0700141TEST_F(ObjectTest, AllocArray) {
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700142 ScopedObjectAccess soa(Thread::Current());
Ian Rogers98379392014-02-24 16:53:16 -0800143 Class* c = class_linker_->FindSystemClass(soa.Self(), "[I");
Mathieu Chartiereb8167a2014-05-07 15:43:14 -0700144 StackHandleScope<1> hs(soa.Self());
Andreas Gampe5a4b8a22014-09-11 08:30:08 -0700145 MutableHandle<Array> a(
Hiroshi Yamauchif0edfc32014-09-25 11:46:46 -0700146 hs.NewHandle(Array::Alloc<true>(soa.Self(), c, 1, c->GetComponentSizeShift(),
Mathieu Chartiereb8167a2014-05-07 15:43:14 -0700147 Runtime::Current()->GetHeap()->GetCurrentAllocator())));
Ian Rogers6fac4472014-02-25 17:01:10 -0800148 EXPECT_TRUE(c == a->GetClass());
149 EXPECT_EQ(1, a->GetLength());
Elliott Hughes68f4fa02011-08-21 10:46:59 -0700150
Ian Rogers98379392014-02-24 16:53:16 -0800151 c = class_linker_->FindSystemClass(soa.Self(), "[Ljava/lang/Object;");
Hiroshi Yamauchif0edfc32014-09-25 11:46:46 -0700152 a.Assign(Array::Alloc<true>(soa.Self(), c, 1, c->GetComponentSizeShift(),
Mathieu Chartiereb8167a2014-05-07 15:43:14 -0700153 Runtime::Current()->GetHeap()->GetCurrentAllocator()));
Ian Rogers6fac4472014-02-25 17:01:10 -0800154 EXPECT_TRUE(c == a->GetClass());
155 EXPECT_EQ(1, a->GetLength());
Elliott Hughes68f4fa02011-08-21 10:46:59 -0700156
Ian Rogers98379392014-02-24 16:53:16 -0800157 c = class_linker_->FindSystemClass(soa.Self(), "[[Ljava/lang/Object;");
Hiroshi Yamauchif0edfc32014-09-25 11:46:46 -0700158 a.Assign(Array::Alloc<true>(soa.Self(), c, 1, c->GetComponentSizeShift(),
Mathieu Chartiereb8167a2014-05-07 15:43:14 -0700159 Runtime::Current()->GetHeap()->GetCurrentAllocator()));
Ian Rogers6fac4472014-02-25 17:01:10 -0800160 EXPECT_TRUE(c == a->GetClass());
161 EXPECT_EQ(1, a->GetLength());
162}
163
164TEST_F(ObjectTest, AllocArray_FillUsable) {
165 ScopedObjectAccess soa(Thread::Current());
166 Class* c = class_linker_->FindSystemClass(soa.Self(), "[B");
Mathieu Chartiereb8167a2014-05-07 15:43:14 -0700167 StackHandleScope<1> hs(soa.Self());
Andreas Gampe5a4b8a22014-09-11 08:30:08 -0700168 MutableHandle<Array> a(
Hiroshi Yamauchif0edfc32014-09-25 11:46:46 -0700169 hs.NewHandle(Array::Alloc<true, true>(soa.Self(), c, 1, c->GetComponentSizeShift(),
170 Runtime::Current()->GetHeap()->GetCurrentAllocator())));
Ian Rogers6fac4472014-02-25 17:01:10 -0800171 EXPECT_TRUE(c == a->GetClass());
172 EXPECT_LE(1, a->GetLength());
173
174 c = class_linker_->FindSystemClass(soa.Self(), "[I");
Hiroshi Yamauchif0edfc32014-09-25 11:46:46 -0700175 a.Assign(Array::Alloc<true, true>(soa.Self(), c, 2, c->GetComponentSizeShift(),
176 Runtime::Current()->GetHeap()->GetCurrentAllocator()));
Ian Rogers6fac4472014-02-25 17:01:10 -0800177 EXPECT_TRUE(c == a->GetClass());
178 EXPECT_LE(2, a->GetLength());
179
180 c = class_linker_->FindSystemClass(soa.Self(), "[Ljava/lang/Object;");
Hiroshi Yamauchif0edfc32014-09-25 11:46:46 -0700181 a.Assign(Array::Alloc<true, true>(soa.Self(), c, 2, c->GetComponentSizeShift(),
182 Runtime::Current()->GetHeap()->GetCurrentAllocator()));
Ian Rogers6fac4472014-02-25 17:01:10 -0800183 EXPECT_TRUE(c == a->GetClass());
184 EXPECT_LE(2, a->GetLength());
185
186 c = class_linker_->FindSystemClass(soa.Self(), "[[Ljava/lang/Object;");
Hiroshi Yamauchif0edfc32014-09-25 11:46:46 -0700187 a.Assign(Array::Alloc<true, true>(soa.Self(), c, 2, c->GetComponentSizeShift(),
188 Runtime::Current()->GetHeap()->GetCurrentAllocator()));
Ian Rogers6fac4472014-02-25 17:01:10 -0800189 EXPECT_TRUE(c == a->GetClass());
190 EXPECT_LE(2, a->GetLength());
Elliott Hughes68f4fa02011-08-21 10:46:59 -0700191}
192
Elliott Hughes710a0cb2011-08-16 14:32:37 -0700193template<typename ArrayT>
194void TestPrimitiveArray(ClassLinker* cl) {
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700195 ScopedObjectAccess soa(Thread::Current());
Elliott Hughes710a0cb2011-08-16 14:32:37 -0700196 typedef typename ArrayT::ElementType T;
197
Ian Rogers50b35e22012-10-04 10:09:15 -0700198 ArrayT* a = ArrayT::Alloc(soa.Self(), 2);
Elliott Hughes710a0cb2011-08-16 14:32:37 -0700199 EXPECT_EQ(2, a->GetLength());
200 EXPECT_EQ(0, a->Get(0));
201 EXPECT_EQ(0, a->Get(1));
202 a->Set(0, T(123));
203 EXPECT_EQ(T(123), a->Get(0));
204 EXPECT_EQ(0, a->Get(1));
205 a->Set(1, T(321));
206 EXPECT_EQ(T(123), a->Get(0));
207 EXPECT_EQ(T(321), a->Get(1));
208
Ian Rogers98379392014-02-24 16:53:16 -0800209 Class* aioobe = cl->FindSystemClass(soa.Self(), "Ljava/lang/ArrayIndexOutOfBoundsException;");
Elliott Hughes710a0cb2011-08-16 14:32:37 -0700210
211 EXPECT_EQ(0, a->Get(-1));
Ian Rogers1f539342012-10-03 21:09:42 -0700212 EXPECT_TRUE(soa.Self()->IsExceptionPending());
Ian Rogers62d6c772013-02-27 08:32:07 -0800213 EXPECT_EQ(aioobe, soa.Self()->GetException(NULL)->GetClass());
Ian Rogers1f539342012-10-03 21:09:42 -0700214 soa.Self()->ClearException();
Elliott Hughes710a0cb2011-08-16 14:32:37 -0700215
216 EXPECT_EQ(0, a->Get(2));
Ian Rogers1f539342012-10-03 21:09:42 -0700217 EXPECT_TRUE(soa.Self()->IsExceptionPending());
Ian Rogers62d6c772013-02-27 08:32:07 -0800218 EXPECT_EQ(aioobe, soa.Self()->GetException(NULL)->GetClass());
Ian Rogers1f539342012-10-03 21:09:42 -0700219 soa.Self()->ClearException();
Elliott Hughes710a0cb2011-08-16 14:32:37 -0700220}
221
222TEST_F(ObjectTest, PrimitiveArray_Boolean_Alloc) {
223 TestPrimitiveArray<BooleanArray>(class_linker_);
224}
225TEST_F(ObjectTest, PrimitiveArray_Byte_Alloc) {
226 TestPrimitiveArray<ByteArray>(class_linker_);
227}
228TEST_F(ObjectTest, PrimitiveArray_Char_Alloc) {
229 TestPrimitiveArray<CharArray>(class_linker_);
230}
Elliott Hughes710a0cb2011-08-16 14:32:37 -0700231TEST_F(ObjectTest, PrimitiveArray_Int_Alloc) {
232 TestPrimitiveArray<IntArray>(class_linker_);
233}
234TEST_F(ObjectTest, PrimitiveArray_Long_Alloc) {
235 TestPrimitiveArray<LongArray>(class_linker_);
236}
237TEST_F(ObjectTest, PrimitiveArray_Short_Alloc) {
238 TestPrimitiveArray<ShortArray>(class_linker_);
239}
240
Ian Rogers647b1a82014-10-10 11:02:11 -0700241TEST_F(ObjectTest, PrimitiveArray_Double_Alloc) {
242 typedef DoubleArray ArrayT;
243 ScopedObjectAccess soa(Thread::Current());
244 typedef typename ArrayT::ElementType T;
245
246 ArrayT* a = ArrayT::Alloc(soa.Self(), 2);
247 EXPECT_EQ(2, a->GetLength());
248 EXPECT_DOUBLE_EQ(0, a->Get(0));
249 EXPECT_DOUBLE_EQ(0, a->Get(1));
250 a->Set(0, T(123));
251 EXPECT_DOUBLE_EQ(T(123), a->Get(0));
252 EXPECT_DOUBLE_EQ(0, a->Get(1));
253 a->Set(1, T(321));
254 EXPECT_DOUBLE_EQ(T(123), a->Get(0));
255 EXPECT_DOUBLE_EQ(T(321), a->Get(1));
256
257 Class* aioobe = class_linker_->FindSystemClass(soa.Self(),
258 "Ljava/lang/ArrayIndexOutOfBoundsException;");
259
260 EXPECT_DOUBLE_EQ(0, a->Get(-1));
261 EXPECT_TRUE(soa.Self()->IsExceptionPending());
262 EXPECT_EQ(aioobe, soa.Self()->GetException(NULL)->GetClass());
263 soa.Self()->ClearException();
264
265 EXPECT_DOUBLE_EQ(0, a->Get(2));
266 EXPECT_TRUE(soa.Self()->IsExceptionPending());
267 EXPECT_EQ(aioobe, soa.Self()->GetException(NULL)->GetClass());
268 soa.Self()->ClearException();
269}
270
271TEST_F(ObjectTest, PrimitiveArray_Float_Alloc) {
272 typedef FloatArray ArrayT;
273 ScopedObjectAccess soa(Thread::Current());
274 typedef typename ArrayT::ElementType T;
275
276 ArrayT* a = ArrayT::Alloc(soa.Self(), 2);
277 EXPECT_FLOAT_EQ(2, a->GetLength());
278 EXPECT_FLOAT_EQ(0, a->Get(0));
279 EXPECT_FLOAT_EQ(0, a->Get(1));
280 a->Set(0, T(123));
281 EXPECT_FLOAT_EQ(T(123), a->Get(0));
282 EXPECT_FLOAT_EQ(0, a->Get(1));
283 a->Set(1, T(321));
284 EXPECT_FLOAT_EQ(T(123), a->Get(0));
285 EXPECT_FLOAT_EQ(T(321), a->Get(1));
286
287 Class* aioobe = class_linker_->FindSystemClass(soa.Self(),
288 "Ljava/lang/ArrayIndexOutOfBoundsException;");
289
290 EXPECT_FLOAT_EQ(0, a->Get(-1));
291 EXPECT_TRUE(soa.Self()->IsExceptionPending());
292 EXPECT_EQ(aioobe, soa.Self()->GetException(NULL)->GetClass());
293 soa.Self()->ClearException();
294
295 EXPECT_FLOAT_EQ(0, a->Get(2));
296 EXPECT_TRUE(soa.Self()->IsExceptionPending());
297 EXPECT_EQ(aioobe, soa.Self()->GetException(NULL)->GetClass());
298 soa.Self()->ClearException();
299}
300
301
Elliott Hughesb408de72011-10-04 14:35:05 -0700302TEST_F(ObjectTest, CheckAndAllocArrayFromCode) {
Ian Rogersb886da82011-09-23 16:27:54 -0700303 // pretend we are trying to call 'new char[3]' from String.toCharArray
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700304 ScopedObjectAccess soa(Thread::Current());
Ian Rogers98379392014-02-24 16:53:16 -0800305 Class* java_util_Arrays = class_linker_->FindSystemClass(soa.Self(), "Ljava/util/Arrays;");
Brian Carlstromea46f952013-07-30 01:26:50 -0700306 ArtMethod* sort = java_util_Arrays->FindDirectMethod("sort", "([I)V");
Ian Rogers9b1a4f42011-11-14 18:35:10 -0800307 const DexFile::StringId* string_id = java_lang_dex_file_->FindStringId("[I");
308 ASSERT_TRUE(string_id != NULL);
309 const DexFile::TypeId* type_id = java_lang_dex_file_->FindTypeId(
310 java_lang_dex_file_->GetIndexForStringId(*string_id));
311 ASSERT_TRUE(type_id != NULL);
312 uint32_t type_idx = java_lang_dex_file_->GetIndexForTypeId(*type_id);
Hiroshi Yamauchie4e23c02013-12-06 13:38:43 -0800313 Object* array = CheckAndAllocArrayFromCodeInstrumented(type_idx, sort, 3, Thread::Current(), false,
314 Runtime::Current()->GetHeap()->GetCurrentAllocator());
Brian Carlstromb63ec392011-08-27 17:38:27 -0700315 EXPECT_TRUE(array->IsArrayInstance());
316 EXPECT_EQ(3, array->AsArray()->GetLength());
317 EXPECT_TRUE(array->GetClass()->IsArrayClass());
318 EXPECT_TRUE(array->GetClass()->GetComponentType()->IsPrimitive());
319}
320
Ian Rogers64b6d142012-10-29 16:34:15 -0700321TEST_F(ObjectTest, CreateMultiArray) {
322 ScopedObjectAccess soa(Thread::Current());
323
Mathieu Chartiereb8167a2014-05-07 15:43:14 -0700324 StackHandleScope<2> hs(soa.Self());
325 Handle<Class> c(hs.NewHandle(class_linker_->FindSystemClass(soa.Self(), "I")));
Andreas Gampe5a4b8a22014-09-11 08:30:08 -0700326 MutableHandle<IntArray> dims(hs.NewHandle(IntArray::Alloc(soa.Self(), 1)));
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100327 dims->Set<false>(0, 1);
Mathieu Chartier5bb99032014-02-08 16:20:58 -0800328 Array* multi = Array::CreateMultiArray(soa.Self(), c, dims);
Ian Rogers98379392014-02-24 16:53:16 -0800329 EXPECT_TRUE(multi->GetClass() == class_linker_->FindSystemClass(soa.Self(), "[I"));
Ian Rogers64b6d142012-10-29 16:34:15 -0700330 EXPECT_EQ(1, multi->GetLength());
331
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100332 dims->Set<false>(0, -1);
Mathieu Chartier5bb99032014-02-08 16:20:58 -0800333 multi = Array::CreateMultiArray(soa.Self(), c, dims);
Ian Rogers64b6d142012-10-29 16:34:15 -0700334 EXPECT_TRUE(soa.Self()->IsExceptionPending());
Ian Rogers62d6c772013-02-27 08:32:07 -0800335 EXPECT_EQ(PrettyDescriptor(soa.Self()->GetException(NULL)->GetClass()),
Ian Rogers64b6d142012-10-29 16:34:15 -0700336 "java.lang.NegativeArraySizeException");
337 soa.Self()->ClearException();
338
Mathieu Chartiereb8167a2014-05-07 15:43:14 -0700339 dims.Assign(IntArray::Alloc(soa.Self(), 2));
Ian Rogers64b6d142012-10-29 16:34:15 -0700340 for (int i = 1; i < 20; ++i) {
341 for (int j = 0; j < 20; ++j) {
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100342 dims->Set<false>(0, i);
343 dims->Set<false>(1, j);
Mathieu Chartier5bb99032014-02-08 16:20:58 -0800344 multi = Array::CreateMultiArray(soa.Self(), c, dims);
Ian Rogers98379392014-02-24 16:53:16 -0800345 EXPECT_TRUE(multi->GetClass() == class_linker_->FindSystemClass(soa.Self(), "[[I"));
Ian Rogers64b6d142012-10-29 16:34:15 -0700346 EXPECT_EQ(i, multi->GetLength());
347 for (int k = 0; k < i; ++k) {
348 Array* outer = multi->AsObjectArray<Array>()->Get(k);
Ian Rogers98379392014-02-24 16:53:16 -0800349 EXPECT_TRUE(outer->GetClass() == class_linker_->FindSystemClass(soa.Self(), "[I"));
Ian Rogers64b6d142012-10-29 16:34:15 -0700350 EXPECT_EQ(j, outer->GetLength());
351 }
352 }
353 }
354}
355
Brian Carlstromb9edb842011-08-28 16:31:06 -0700356TEST_F(ObjectTest, StaticFieldFromCode) {
jeffhaoabcfde32011-09-29 15:05:18 -0700357 // pretend we are trying to access 'Static.s0' from StaticsFromCode.<clinit>
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700358 ScopedObjectAccess soa(Thread::Current());
359 jobject class_loader = LoadDex("StaticsFromCode");
360 const DexFile* dex_file = Runtime::Current()->GetCompileTimeClassPath(class_loader)[0];
Brian Carlstrom848a4b32011-09-04 11:29:27 -0700361 CHECK(dex_file != NULL);
362
Mathieu Chartiereb8167a2014-05-07 15:43:14 -0700363 StackHandleScope<2> hs(soa.Self());
364 Handle<mirror::ClassLoader> loader(hs.NewHandle(soa.Decode<ClassLoader*>(class_loader)));
Ian Rogers98379392014-02-24 16:53:16 -0800365 Class* klass = class_linker_->FindClass(soa.Self(), "LStaticsFromCode;", loader);
Ian Rogers241b5de2013-10-09 17:58:57 -0700366 ArtMethod* clinit = klass->FindClassInitializer();
Ian Rogers9b1a4f42011-11-14 18:35:10 -0800367 const DexFile::StringId* klass_string_id = dex_file->FindStringId("LStaticsFromCode;");
368 ASSERT_TRUE(klass_string_id != NULL);
369 const DexFile::TypeId* klass_type_id = dex_file->FindTypeId(
370 dex_file->GetIndexForStringId(*klass_string_id));
371 ASSERT_TRUE(klass_type_id != NULL);
372
373 const DexFile::StringId* type_string_id = dex_file->FindStringId("Ljava/lang/Object;");
374 ASSERT_TRUE(type_string_id != NULL);
375 const DexFile::TypeId* type_type_id = dex_file->FindTypeId(
376 dex_file->GetIndexForStringId(*type_string_id));
377 ASSERT_TRUE(type_type_id != NULL);
378
379 const DexFile::StringId* name_str_id = dex_file->FindStringId("s0");
380 ASSERT_TRUE(name_str_id != NULL);
381
382 const DexFile::FieldId* field_id = dex_file->FindFieldId(
383 *klass_type_id, *name_str_id, *type_type_id);
384 ASSERT_TRUE(field_id != NULL);
385 uint32_t field_idx = dex_file->GetIndexForFieldId(*field_id);
386
Sebastien Hertzd4beb6b2013-10-02 17:07:20 +0200387 ArtField* field = FindFieldFromCode<StaticObjectRead, true>(field_idx, clinit, Thread::Current(),
Ian Rogersef7d42f2014-01-06 12:55:46 -0800388 sizeof(HeapReference<Object>));
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700389 Object* s0 = field->GetObj(klass);
Ian Rogers64b6d142012-10-29 16:34:15 -0700390 EXPECT_TRUE(s0 != NULL);
Brian Carlstromb9edb842011-08-28 16:31:06 -0700391
Mathieu Chartiereb8167a2014-05-07 15:43:14 -0700392 Handle<CharArray> char_array(hs.NewHandle(CharArray::Alloc(soa.Self(), 0)));
393 field->SetObj<false>(field->GetDeclaringClass(), char_array.Get());
394 EXPECT_EQ(char_array.Get(), field->GetObj(klass));
Brian Carlstromb9edb842011-08-28 16:31:06 -0700395
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100396 field->SetObj<false>(field->GetDeclaringClass(), NULL);
Ian Rogers2fa6b2e2012-10-17 00:10:17 -0700397 EXPECT_EQ(NULL, field->GetObj(klass));
Brian Carlstrom9f30b382011-08-28 22:41:38 -0700398
Brian Carlstromea46f952013-07-30 01:26:50 -0700399 // TODO: more exhaustive tests of all 6 cases of ArtField::*FromCode
Brian Carlstromb9edb842011-08-28 16:31:06 -0700400}
401
Brian Carlstrom0b138b22011-07-27 15:19:17 -0700402TEST_F(ObjectTest, String) {
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700403 ScopedObjectAccess soa(Thread::Current());
Brian Carlstrom0b138b22011-07-27 15:19:17 -0700404 // Test the empty string.
405 AssertString(0, "", "", 0);
406
407 // Test one-byte characters.
408 AssertString(1, " ", "\x00\x20", 0x20);
409 AssertString(1, "", "\x00\x00", 0);
410 AssertString(1, "\x7f", "\x00\x7f", 0x7f);
411 AssertString(2, "hi", "\x00\x68\x00\x69", (31 * 0x68) + 0x69);
412
413 // Test two-byte characters.
414 AssertString(1, "\xc2\x80", "\x00\x80", 0x80);
415 AssertString(1, "\xd9\xa6", "\x06\x66", 0x0666);
416 AssertString(1, "\xdf\xbf", "\x07\xff", 0x07ff);
417 AssertString(3, "h\xd9\xa6i", "\x00\x68\x06\x66\x00\x69", (31 * ((31 * 0x68) + 0x0666)) + 0x69);
418
419 // Test three-byte characters.
420 AssertString(1, "\xe0\xa0\x80", "\x08\x00", 0x0800);
421 AssertString(1, "\xe1\x88\xb4", "\x12\x34", 0x1234);
422 AssertString(1, "\xef\xbf\xbf", "\xff\xff", 0xffff);
423 AssertString(3, "h\xe1\x88\xb4i", "\x00\x68\x12\x34\x00\x69", (31 * ((31 * 0x68) + 0x1234)) + 0x69);
424}
Jesse Wilsoncbe9fc02011-07-29 18:59:50 -0400425
Jesse Wilsonf7e85a52011-08-01 18:45:58 -0700426TEST_F(ObjectTest, StringEqualsUtf8) {
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700427 ScopedObjectAccess soa(Thread::Current());
Mathieu Chartiereb8167a2014-05-07 15:43:14 -0700428 StackHandleScope<2> hs(soa.Self());
429 Handle<String> string(hs.NewHandle(String::AllocFromModifiedUtf8(soa.Self(), "android")));
Carl Shapiro8860c0e2011-08-04 17:36:16 -0700430 EXPECT_TRUE(string->Equals("android"));
431 EXPECT_FALSE(string->Equals("Android"));
432 EXPECT_FALSE(string->Equals("ANDROID"));
433 EXPECT_FALSE(string->Equals(""));
434 EXPECT_FALSE(string->Equals("and"));
435 EXPECT_FALSE(string->Equals("androids"));
Jesse Wilsonf7e85a52011-08-01 18:45:58 -0700436
Mathieu Chartiereb8167a2014-05-07 15:43:14 -0700437 Handle<String> empty(hs.NewHandle(String::AllocFromModifiedUtf8(soa.Self(), "")));
Carl Shapiro8860c0e2011-08-04 17:36:16 -0700438 EXPECT_TRUE(empty->Equals(""));
439 EXPECT_FALSE(empty->Equals("a"));
Jesse Wilsoncbe9fc02011-07-29 18:59:50 -0400440}
441
442TEST_F(ObjectTest, StringEquals) {
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700443 ScopedObjectAccess soa(Thread::Current());
Mathieu Chartiereb8167a2014-05-07 15:43:14 -0700444 StackHandleScope<3> hs(soa.Self());
445 Handle<String> string(hs.NewHandle(String::AllocFromModifiedUtf8(soa.Self(), "android")));
446 Handle<String> string_2(hs.NewHandle(String::AllocFromModifiedUtf8(soa.Self(), "android")));
447 EXPECT_TRUE(string->Equals(string_2.Get()));
Carl Shapiro8860c0e2011-08-04 17:36:16 -0700448 EXPECT_FALSE(string->Equals("Android"));
449 EXPECT_FALSE(string->Equals("ANDROID"));
450 EXPECT_FALSE(string->Equals(""));
451 EXPECT_FALSE(string->Equals("and"));
452 EXPECT_FALSE(string->Equals("androids"));
Jesse Wilsoncbe9fc02011-07-29 18:59:50 -0400453
Mathieu Chartiereb8167a2014-05-07 15:43:14 -0700454 Handle<String> empty(hs.NewHandle(String::AllocFromModifiedUtf8(soa.Self(), "")));
Carl Shapiro8860c0e2011-08-04 17:36:16 -0700455 EXPECT_TRUE(empty->Equals(""));
456 EXPECT_FALSE(empty->Equals("a"));
457}
458
Ian Rogers64b6d142012-10-29 16:34:15 -0700459TEST_F(ObjectTest, StringCompareTo) {
460 ScopedObjectAccess soa(Thread::Current());
Mathieu Chartiereb8167a2014-05-07 15:43:14 -0700461 StackHandleScope<5> hs(soa.Self());
462 Handle<String> string(hs.NewHandle(String::AllocFromModifiedUtf8(soa.Self(), "android")));
463 Handle<String> string_2(hs.NewHandle(String::AllocFromModifiedUtf8(soa.Self(), "android")));
464 Handle<String> string_3(hs.NewHandle(String::AllocFromModifiedUtf8(soa.Self(), "Android")));
465 Handle<String> string_4(hs.NewHandle(String::AllocFromModifiedUtf8(soa.Self(), "and")));
466 Handle<String> string_5(hs.NewHandle(String::AllocFromModifiedUtf8(soa.Self(), "")));
467 EXPECT_EQ(0, string->CompareTo(string_2.Get()));
468 EXPECT_LT(0, string->CompareTo(string_3.Get()));
469 EXPECT_GT(0, string_3->CompareTo(string.Get()));
470 EXPECT_LT(0, string->CompareTo(string_4.Get()));
471 EXPECT_GT(0, string_4->CompareTo(string.Get()));
472 EXPECT_LT(0, string->CompareTo(string_5.Get()));
473 EXPECT_GT(0, string_5->CompareTo(string.Get()));
Ian Rogers64b6d142012-10-29 16:34:15 -0700474}
475
jeffhao0ce13152012-03-27 19:45:50 -0700476TEST_F(ObjectTest, StringLength) {
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700477 ScopedObjectAccess soa(Thread::Current());
Mathieu Chartiereb8167a2014-05-07 15:43:14 -0700478 StackHandleScope<1> hs(soa.Self());
479 Handle<String> string(hs.NewHandle(String::AllocFromModifiedUtf8(soa.Self(), "android")));
jeffhao0ce13152012-03-27 19:45:50 -0700480 EXPECT_EQ(string->GetLength(), 7);
481 EXPECT_EQ(string->GetUtfLength(), 7);
482
Sebastien Hertzee1d79a2014-02-21 15:46:30 +0100483 string->SetOffset(2);
484 string->SetCount(5);
jeffhao0ce13152012-03-27 19:45:50 -0700485 EXPECT_TRUE(string->Equals("droid"));
486 EXPECT_EQ(string->GetLength(), 5);
487 EXPECT_EQ(string->GetUtfLength(), 5);
488}
489
Carl Shapiro8860c0e2011-08-04 17:36:16 -0700490TEST_F(ObjectTest, DescriptorCompare) {
Mathieu Chartier590fee92013-09-13 13:46:47 -0700491 // Two classloaders conflicts in compile_time_class_paths_.
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700492 ScopedObjectAccess soa(Thread::Current());
Carl Shapiro8860c0e2011-08-04 17:36:16 -0700493 ClassLinker* linker = class_linker_;
494
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700495 jobject jclass_loader_1 = LoadDex("ProtoCompare");
496 jobject jclass_loader_2 = LoadDex("ProtoCompare2");
Mathieu Chartierbfd9a432014-05-21 17:43:44 -0700497 StackHandleScope<4> hs(soa.Self());
Mathieu Chartiereb8167a2014-05-07 15:43:14 -0700498 Handle<ClassLoader> class_loader_1(hs.NewHandle(soa.Decode<ClassLoader*>(jclass_loader_1)));
499 Handle<ClassLoader> class_loader_2(hs.NewHandle(soa.Decode<ClassLoader*>(jclass_loader_2)));
Carl Shapiro8860c0e2011-08-04 17:36:16 -0700500
Ian Rogers98379392014-02-24 16:53:16 -0800501 Class* klass1 = linker->FindClass(soa.Self(), "LProtoCompare;", class_loader_1);
Carl Shapiro8860c0e2011-08-04 17:36:16 -0700502 ASSERT_TRUE(klass1 != NULL);
Ian Rogers98379392014-02-24 16:53:16 -0800503 Class* klass2 = linker->FindClass(soa.Self(), "LProtoCompare2;", class_loader_2);
Carl Shapiro8860c0e2011-08-04 17:36:16 -0700504 ASSERT_TRUE(klass2 != NULL);
505
Brian Carlstromea46f952013-07-30 01:26:50 -0700506 ArtMethod* m1_1 = klass1->GetVirtualMethod(0);
Mathieu Chartierbfd9a432014-05-21 17:43:44 -0700507 EXPECT_STREQ(m1_1->GetName(), "m1");
Brian Carlstromea46f952013-07-30 01:26:50 -0700508 ArtMethod* m2_1 = klass1->GetVirtualMethod(1);
Mathieu Chartierbfd9a432014-05-21 17:43:44 -0700509 EXPECT_STREQ(m2_1->GetName(), "m2");
Brian Carlstromea46f952013-07-30 01:26:50 -0700510 ArtMethod* m3_1 = klass1->GetVirtualMethod(2);
Mathieu Chartierbfd9a432014-05-21 17:43:44 -0700511 EXPECT_STREQ(m3_1->GetName(), "m3");
Brian Carlstromea46f952013-07-30 01:26:50 -0700512 ArtMethod* m4_1 = klass1->GetVirtualMethod(3);
Mathieu Chartierbfd9a432014-05-21 17:43:44 -0700513 EXPECT_STREQ(m4_1->GetName(), "m4");
Carl Shapiro8860c0e2011-08-04 17:36:16 -0700514
Brian Carlstromea46f952013-07-30 01:26:50 -0700515 ArtMethod* m1_2 = klass2->GetVirtualMethod(0);
Mathieu Chartierbfd9a432014-05-21 17:43:44 -0700516 EXPECT_STREQ(m1_2->GetName(), "m1");
Brian Carlstromea46f952013-07-30 01:26:50 -0700517 ArtMethod* m2_2 = klass2->GetVirtualMethod(1);
Mathieu Chartierbfd9a432014-05-21 17:43:44 -0700518 EXPECT_STREQ(m2_2->GetName(), "m2");
Brian Carlstromea46f952013-07-30 01:26:50 -0700519 ArtMethod* m3_2 = klass2->GetVirtualMethod(2);
Mathieu Chartierbfd9a432014-05-21 17:43:44 -0700520 EXPECT_STREQ(m3_2->GetName(), "m3");
Brian Carlstromea46f952013-07-30 01:26:50 -0700521 ArtMethod* m4_2 = klass2->GetVirtualMethod(3);
Mathieu Chartierbfd9a432014-05-21 17:43:44 -0700522 EXPECT_STREQ(m4_2->GetName(), "m4");
Carl Shapiro8860c0e2011-08-04 17:36:16 -0700523
Andreas Gampe5a4b8a22014-09-11 08:30:08 -0700524 MutableMethodHelper mh(hs.NewHandle(m1_1));
525 MutableMethodHelper mh2(hs.NewHandle(m1_2));
Ian Rogers6d4d9fc2011-11-30 16:24:48 -0800526 EXPECT_TRUE(mh.HasSameNameAndSignature(&mh2));
527 EXPECT_TRUE(mh2.HasSameNameAndSignature(&mh));
Carl Shapiro8860c0e2011-08-04 17:36:16 -0700528
Ian Rogers6d4d9fc2011-11-30 16:24:48 -0800529 mh.ChangeMethod(m2_1);
530 mh2.ChangeMethod(m2_2);
531 EXPECT_TRUE(mh.HasSameNameAndSignature(&mh2));
532 EXPECT_TRUE(mh2.HasSameNameAndSignature(&mh));
Carl Shapiro8860c0e2011-08-04 17:36:16 -0700533
Ian Rogers6d4d9fc2011-11-30 16:24:48 -0800534 mh.ChangeMethod(m3_1);
535 mh2.ChangeMethod(m3_2);
536 EXPECT_TRUE(mh.HasSameNameAndSignature(&mh2));
537 EXPECT_TRUE(mh2.HasSameNameAndSignature(&mh));
Carl Shapiro8860c0e2011-08-04 17:36:16 -0700538
Ian Rogers6d4d9fc2011-11-30 16:24:48 -0800539 mh.ChangeMethod(m4_1);
540 mh2.ChangeMethod(m4_2);
541 EXPECT_TRUE(mh.HasSameNameAndSignature(&mh2));
542 EXPECT_TRUE(mh2.HasSameNameAndSignature(&mh));
Jesse Wilsoncbe9fc02011-07-29 18:59:50 -0400543}
544
Jesse Wilsonfd687c52011-08-04 19:27:35 -0700545TEST_F(ObjectTest, StringHashCode) {
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700546 ScopedObjectAccess soa(Thread::Current());
Mathieu Chartiereb8167a2014-05-07 15:43:14 -0700547 StackHandleScope<3> hs(soa.Self());
548 Handle<String> empty(hs.NewHandle(String::AllocFromModifiedUtf8(soa.Self(), "")));
549 Handle<String> A(hs.NewHandle(String::AllocFromModifiedUtf8(soa.Self(), "A")));
550 Handle<String> ABC(hs.NewHandle(String::AllocFromModifiedUtf8(soa.Self(), "ABC")));
Brian Carlstrom40381fb2011-10-19 14:13:40 -0700551
552 EXPECT_EQ(0, empty->GetHashCode());
553 EXPECT_EQ(65, A->GetHashCode());
554 EXPECT_EQ(64578, ABC->GetHashCode());
Jesse Wilsonfd687c52011-08-04 19:27:35 -0700555}
556
Brian Carlstromf7ed11a2011-08-09 17:55:51 -0700557TEST_F(ObjectTest, InstanceOf) {
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700558 ScopedObjectAccess soa(Thread::Current());
559 jobject jclass_loader = LoadDex("XandY");
Mathieu Chartiereb8167a2014-05-07 15:43:14 -0700560 StackHandleScope<3> hs(soa.Self());
561 Handle<ClassLoader> class_loader(hs.NewHandle(soa.Decode<ClassLoader*>(jclass_loader)));
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700562
Ian Rogers98379392014-02-24 16:53:16 -0800563 Class* X = class_linker_->FindClass(soa.Self(), "LX;", class_loader);
564 Class* Y = class_linker_->FindClass(soa.Self(), "LY;", class_loader);
Brian Carlstromf7ed11a2011-08-09 17:55:51 -0700565 ASSERT_TRUE(X != NULL);
566 ASSERT_TRUE(Y != NULL);
567
Mathieu Chartiereb8167a2014-05-07 15:43:14 -0700568 Handle<Object> x(hs.NewHandle(X->AllocObject(soa.Self())));
569 Handle<Object> y(hs.NewHandle(Y->AllocObject(soa.Self())));
570 ASSERT_TRUE(x.Get() != NULL);
571 ASSERT_TRUE(y.Get() != NULL);
Brian Carlstromf7ed11a2011-08-09 17:55:51 -0700572
Brian Carlstromf7ed11a2011-08-09 17:55:51 -0700573 EXPECT_TRUE(x->InstanceOf(X));
574 EXPECT_FALSE(x->InstanceOf(Y));
575 EXPECT_TRUE(y->InstanceOf(X));
576 EXPECT_TRUE(y->InstanceOf(Y));
Brian Carlstrom5b8e4c82011-09-18 01:38:59 -0700577
Ian Rogers98379392014-02-24 16:53:16 -0800578 Class* java_lang_Class = class_linker_->FindSystemClass(soa.Self(), "Ljava/lang/Class;");
579 Class* Object_array_class = class_linker_->FindSystemClass(soa.Self(), "[Ljava/lang/Object;");
Elliott Hughes92f14b22011-10-06 12:29:54 -0700580
581 EXPECT_FALSE(java_lang_Class->InstanceOf(Object_array_class));
582 EXPECT_TRUE(Object_array_class->InstanceOf(java_lang_Class));
583
584 // All array classes implement Cloneable and Serializable.
Ian Rogers50b35e22012-10-04 10:09:15 -0700585 Object* array = ObjectArray<Object>::Alloc(soa.Self(), Object_array_class, 1);
Ian Rogers98379392014-02-24 16:53:16 -0800586 Class* java_lang_Cloneable = class_linker_->FindSystemClass(soa.Self(), "Ljava/lang/Cloneable;");
587 Class* java_io_Serializable = class_linker_->FindSystemClass(soa.Self(), "Ljava/io/Serializable;");
Elliott Hughes92f14b22011-10-06 12:29:54 -0700588 EXPECT_TRUE(array->InstanceOf(java_lang_Cloneable));
589 EXPECT_TRUE(array->InstanceOf(java_io_Serializable));
Brian Carlstromf7ed11a2011-08-09 17:55:51 -0700590}
591
592TEST_F(ObjectTest, IsAssignableFrom) {
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700593 ScopedObjectAccess soa(Thread::Current());
594 jobject jclass_loader = LoadDex("XandY");
Mathieu Chartiereb8167a2014-05-07 15:43:14 -0700595 StackHandleScope<1> hs(soa.Self());
596 Handle<ClassLoader> class_loader(hs.NewHandle(soa.Decode<ClassLoader*>(jclass_loader)));
Ian Rogers98379392014-02-24 16:53:16 -0800597 Class* X = class_linker_->FindClass(soa.Self(), "LX;", class_loader);
598 Class* Y = class_linker_->FindClass(soa.Self(), "LY;", class_loader);
Brian Carlstromf7ed11a2011-08-09 17:55:51 -0700599
600 EXPECT_TRUE(X->IsAssignableFrom(X));
601 EXPECT_TRUE(X->IsAssignableFrom(Y));
602 EXPECT_FALSE(Y->IsAssignableFrom(X));
603 EXPECT_TRUE(Y->IsAssignableFrom(Y));
Ian Rogersd81871c2011-10-03 13:57:23 -0700604
605 // class final String implements CharSequence, ..
Ian Rogers98379392014-02-24 16:53:16 -0800606 Class* string = class_linker_->FindSystemClass(soa.Self(), "Ljava/lang/String;");
607 Class* charseq = class_linker_->FindSystemClass(soa.Self(), "Ljava/lang/CharSequence;");
Ian Rogersd81871c2011-10-03 13:57:23 -0700608 // Can String be assigned to CharSequence without a cast?
609 EXPECT_TRUE(charseq->IsAssignableFrom(string));
610 // Can CharSequence be assigned to String without a cast?
611 EXPECT_FALSE(string->IsAssignableFrom(charseq));
612
613 // Primitive types are only assignable to themselves
614 const char* prims = "ZBCSIJFD";
615 Class* prim_types[strlen(prims)];
616 for (size_t i = 0; i < strlen(prims); i++) {
617 prim_types[i] = class_linker_->FindPrimitiveClass(prims[i]);
618 }
619 for (size_t i = 0; i < strlen(prims); i++) {
620 for (size_t j = 0; i < strlen(prims); i++) {
621 if (i == j) {
622 EXPECT_TRUE(prim_types[i]->IsAssignableFrom(prim_types[j]));
623 } else {
624 EXPECT_FALSE(prim_types[i]->IsAssignableFrom(prim_types[j]));
625 }
626 }
627 }
Brian Carlstromf7ed11a2011-08-09 17:55:51 -0700628}
629
630TEST_F(ObjectTest, IsAssignableFromArray) {
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700631 ScopedObjectAccess soa(Thread::Current());
632 jobject jclass_loader = LoadDex("XandY");
Mathieu Chartiereb8167a2014-05-07 15:43:14 -0700633 StackHandleScope<1> hs(soa.Self());
634 Handle<ClassLoader> class_loader(hs.NewHandle(soa.Decode<ClassLoader*>(jclass_loader)));
Ian Rogers98379392014-02-24 16:53:16 -0800635 Class* X = class_linker_->FindClass(soa.Self(), "LX;", class_loader);
636 Class* Y = class_linker_->FindClass(soa.Self(), "LY;", class_loader);
Brian Carlstromf7ed11a2011-08-09 17:55:51 -0700637 ASSERT_TRUE(X != NULL);
638 ASSERT_TRUE(Y != NULL);
639
Ian Rogers98379392014-02-24 16:53:16 -0800640 Class* YA = class_linker_->FindClass(soa.Self(), "[LY;", class_loader);
641 Class* YAA = class_linker_->FindClass(soa.Self(), "[[LY;", class_loader);
Brian Carlstromf7ed11a2011-08-09 17:55:51 -0700642 ASSERT_TRUE(YA != NULL);
643 ASSERT_TRUE(YAA != NULL);
644
Ian Rogers98379392014-02-24 16:53:16 -0800645 Class* XAA = class_linker_->FindClass(soa.Self(), "[[LX;", class_loader);
Brian Carlstromf7ed11a2011-08-09 17:55:51 -0700646 ASSERT_TRUE(XAA != NULL);
647
Ian Rogers98379392014-02-24 16:53:16 -0800648 Class* O = class_linker_->FindSystemClass(soa.Self(), "Ljava/lang/Object;");
649 Class* OA = class_linker_->FindSystemClass(soa.Self(), "[Ljava/lang/Object;");
650 Class* OAA = class_linker_->FindSystemClass(soa.Self(), "[[Ljava/lang/Object;");
651 Class* OAAA = class_linker_->FindSystemClass(soa.Self(), "[[[Ljava/lang/Object;");
Brian Carlstromf7ed11a2011-08-09 17:55:51 -0700652 ASSERT_TRUE(O != NULL);
653 ASSERT_TRUE(OA != NULL);
654 ASSERT_TRUE(OAA != NULL);
655 ASSERT_TRUE(OAAA != NULL);
656
Ian Rogers98379392014-02-24 16:53:16 -0800657 Class* S = class_linker_->FindSystemClass(soa.Self(), "Ljava/io/Serializable;");
658 Class* SA = class_linker_->FindSystemClass(soa.Self(), "[Ljava/io/Serializable;");
659 Class* SAA = class_linker_->FindSystemClass(soa.Self(), "[[Ljava/io/Serializable;");
Brian Carlstromf7ed11a2011-08-09 17:55:51 -0700660 ASSERT_TRUE(S != NULL);
661 ASSERT_TRUE(SA != NULL);
662 ASSERT_TRUE(SAA != NULL);
663
Ian Rogers98379392014-02-24 16:53:16 -0800664 Class* IA = class_linker_->FindSystemClass(soa.Self(), "[I");
Brian Carlstromf7ed11a2011-08-09 17:55:51 -0700665 ASSERT_TRUE(IA != NULL);
666
667 EXPECT_TRUE(YAA->IsAssignableFrom(YAA)); // identity
668 EXPECT_TRUE(XAA->IsAssignableFrom(YAA)); // element superclass
669 EXPECT_FALSE(YAA->IsAssignableFrom(XAA));
670 EXPECT_FALSE(Y->IsAssignableFrom(YAA));
671 EXPECT_FALSE(YA->IsAssignableFrom(YAA));
672 EXPECT_TRUE(O->IsAssignableFrom(YAA)); // everything is an Object
673 EXPECT_TRUE(OA->IsAssignableFrom(YAA));
674 EXPECT_TRUE(OAA->IsAssignableFrom(YAA));
675 EXPECT_TRUE(S->IsAssignableFrom(YAA)); // all arrays are Serializable
676 EXPECT_TRUE(SA->IsAssignableFrom(YAA));
677 EXPECT_FALSE(SAA->IsAssignableFrom(YAA)); // unless Y was Serializable
678
679 EXPECT_FALSE(IA->IsAssignableFrom(OA));
680 EXPECT_FALSE(OA->IsAssignableFrom(IA));
681 EXPECT_TRUE(O->IsAssignableFrom(IA));
682}
683
Elliott Hughescdf53122011-08-19 15:46:09 -0700684TEST_F(ObjectTest, FindInstanceField) {
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700685 ScopedObjectAccess soa(Thread::Current());
Mathieu Chartiereb8167a2014-05-07 15:43:14 -0700686 StackHandleScope<1> hs(soa.Self());
687 Handle<String> s(hs.NewHandle(String::AllocFromModifiedUtf8(soa.Self(), "ABC")));
688 ASSERT_TRUE(s.Get() != NULL);
Elliott Hughescdf53122011-08-19 15:46:09 -0700689 Class* c = s->GetClass();
690 ASSERT_TRUE(c != NULL);
691
692 // Wrong type.
Brian Carlstrom6b4ef022011-10-23 14:59:04 -0700693 EXPECT_TRUE(c->FindDeclaredInstanceField("count", "J") == NULL);
694 EXPECT_TRUE(c->FindInstanceField("count", "J") == NULL);
Elliott Hughescdf53122011-08-19 15:46:09 -0700695
696 // Wrong name.
Brian Carlstrom6b4ef022011-10-23 14:59:04 -0700697 EXPECT_TRUE(c->FindDeclaredInstanceField("Count", "I") == NULL);
698 EXPECT_TRUE(c->FindInstanceField("Count", "I") == NULL);
Elliott Hughescdf53122011-08-19 15:46:09 -0700699
700 // Right name and type.
Brian Carlstromea46f952013-07-30 01:26:50 -0700701 ArtField* f1 = c->FindDeclaredInstanceField("count", "I");
702 ArtField* f2 = c->FindInstanceField("count", "I");
Elliott Hughescdf53122011-08-19 15:46:09 -0700703 EXPECT_TRUE(f1 != NULL);
704 EXPECT_TRUE(f2 != NULL);
705 EXPECT_EQ(f1, f2);
706
707 // TODO: check that s.count == 3.
708
709 // Ensure that we handle superclass fields correctly...
Ian Rogers98379392014-02-24 16:53:16 -0800710 c = class_linker_->FindSystemClass(soa.Self(), "Ljava/lang/StringBuilder;");
Elliott Hughescdf53122011-08-19 15:46:09 -0700711 ASSERT_TRUE(c != NULL);
712 // No StringBuilder.count...
Brian Carlstrom6b4ef022011-10-23 14:59:04 -0700713 EXPECT_TRUE(c->FindDeclaredInstanceField("count", "I") == NULL);
Elliott Hughescdf53122011-08-19 15:46:09 -0700714 // ...but there is an AbstractStringBuilder.count.
Brian Carlstrom6b4ef022011-10-23 14:59:04 -0700715 EXPECT_TRUE(c->FindInstanceField("count", "I") != NULL);
Elliott Hughescdf53122011-08-19 15:46:09 -0700716}
717
718TEST_F(ObjectTest, FindStaticField) {
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700719 ScopedObjectAccess soa(Thread::Current());
Mathieu Chartierf8322842014-05-16 10:59:25 -0700720 StackHandleScope<4> hs(soa.Self());
Mathieu Chartiereb8167a2014-05-07 15:43:14 -0700721 Handle<String> s(hs.NewHandle(String::AllocFromModifiedUtf8(soa.Self(), "ABC")));
722 ASSERT_TRUE(s.Get() != NULL);
Mathieu Chartierf8322842014-05-16 10:59:25 -0700723 Handle<Class> c(hs.NewHandle(s->GetClass()));
724 ASSERT_TRUE(c.Get() != NULL);
Elliott Hughescdf53122011-08-19 15:46:09 -0700725
726 // Wrong type.
Brian Carlstrom6b4ef022011-10-23 14:59:04 -0700727 EXPECT_TRUE(c->FindDeclaredStaticField("CASE_INSENSITIVE_ORDER", "I") == NULL);
Mathieu Chartierf8322842014-05-16 10:59:25 -0700728 EXPECT_TRUE(mirror::Class::FindStaticField(soa.Self(), c, "CASE_INSENSITIVE_ORDER", "I") == NULL);
Elliott Hughescdf53122011-08-19 15:46:09 -0700729
730 // Wrong name.
Brian Carlstrom6b4ef022011-10-23 14:59:04 -0700731 EXPECT_TRUE(c->FindDeclaredStaticField("cASE_INSENSITIVE_ORDER", "Ljava/util/Comparator;") == NULL);
Mathieu Chartierf8322842014-05-16 10:59:25 -0700732 EXPECT_TRUE(
733 mirror::Class::FindStaticField(soa.Self(), c, "cASE_INSENSITIVE_ORDER",
734 "Ljava/util/Comparator;") == NULL);
Elliott Hughescdf53122011-08-19 15:46:09 -0700735
736 // Right name and type.
Mathieu Chartierf8322842014-05-16 10:59:25 -0700737 Handle<ArtField> f1(hs.NewHandle(
738 c->FindDeclaredStaticField("CASE_INSENSITIVE_ORDER", "Ljava/util/Comparator;")));
739 Handle<ArtField> f2(hs.NewHandle(
740 mirror::Class::FindStaticField(soa.Self(), c, "CASE_INSENSITIVE_ORDER",
741 "Ljava/util/Comparator;")));
742 EXPECT_TRUE(f1.Get() != NULL);
743 EXPECT_TRUE(f2.Get() != NULL);
744 EXPECT_EQ(f1.Get(), f2.Get());
Elliott Hughescdf53122011-08-19 15:46:09 -0700745
746 // TODO: test static fields via superclasses.
747 // TODO: test static fields via interfaces.
748 // TODO: test that interfaces trump superclasses.
749}
750
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800751} // namespace mirror
Carl Shapiro894d0fa2011-06-30 14:48:49 -0700752} // namespace art