/* * Copyright (C) 2011 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #include "dex_cache.h" #include #include "art_method-inl.h" #include "class_linker.h" #include "common_runtime_test.h" #include "handle_scope-inl.h" #include "linear_alloc.h" #include "mirror/class_loader-inl.h" #include "mirror/dex_cache-inl.h" #include "scoped_thread_state_change-inl.h" namespace art { namespace mirror { class DexCacheTest : public CommonRuntimeTest {}; class DexCacheMethodHandlesTest : public DexCacheTest { protected: void SetUpRuntimeOptions(RuntimeOptions* options) override { CommonRuntimeTest::SetUpRuntimeOptions(options); } }; TEST_F(DexCacheTest, Open) { ScopedObjectAccess soa(Thread::Current()); StackHandleScope<1> hs(soa.Self()); ASSERT_TRUE(java_lang_dex_file_ != nullptr); Handle dex_cache( hs.NewHandle(class_linker_->AllocAndInitializeDexCache( soa.Self(), *java_lang_dex_file_, Runtime::Current()->GetLinearAlloc()))); ASSERT_TRUE(dex_cache != nullptr); EXPECT_TRUE(dex_cache->StaticStringSize() == dex_cache->NumStrings() || java_lang_dex_file_->NumStringIds() == dex_cache->NumStrings()); EXPECT_TRUE(dex_cache->StaticTypeSize() == dex_cache->NumResolvedTypes() || java_lang_dex_file_->NumTypeIds() == dex_cache->NumResolvedTypes()); EXPECT_TRUE(dex_cache->StaticMethodSize() == dex_cache->NumResolvedMethods() || java_lang_dex_file_->NumMethodIds() == dex_cache->NumResolvedMethods()); EXPECT_TRUE(dex_cache->StaticArtFieldSize() == dex_cache->NumResolvedFields() || java_lang_dex_file_->NumFieldIds() == dex_cache->NumResolvedFields()); EXPECT_TRUE(dex_cache->StaticMethodTypeSize() == dex_cache->NumResolvedMethodTypes() || java_lang_dex_file_->NumProtoIds() == dex_cache->NumResolvedMethodTypes()); } TEST_F(DexCacheMethodHandlesTest, Open) { ScopedObjectAccess soa(Thread::Current()); StackHandleScope<1> hs(soa.Self()); ASSERT_TRUE(java_lang_dex_file_ != nullptr); Handle dex_cache( hs.NewHandle(class_linker_->AllocAndInitializeDexCache( soa.Self(), *java_lang_dex_file_, Runtime::Current()->GetLinearAlloc()))); EXPECT_TRUE(dex_cache->StaticMethodTypeSize() == dex_cache->NumResolvedMethodTypes() || java_lang_dex_file_->NumProtoIds() == dex_cache->NumResolvedMethodTypes()); } TEST_F(DexCacheTest, LinearAlloc) { ScopedObjectAccess soa(Thread::Current()); jobject jclass_loader(LoadDex("Main")); ASSERT_TRUE(jclass_loader != nullptr); StackHandleScope<1> hs(soa.Self()); Handle class_loader(hs.NewHandle( soa.Decode(jclass_loader))); ObjPtr klass = class_linker_->FindClass(soa.Self(), "LMain;", class_loader); ASSERT_TRUE(klass != nullptr); LinearAlloc* const linear_alloc = klass->GetClassLoader()->GetAllocator(); EXPECT_NE(linear_alloc, runtime_->GetLinearAlloc()); EXPECT_TRUE(linear_alloc->Contains(klass->GetDexCache()->GetResolvedMethods())); } TEST_F(DexCacheTest, TestResolvedFieldAccess) { ScopedObjectAccess soa(Thread::Current()); jobject jclass_loader(LoadDex("Packages")); ASSERT_TRUE(jclass_loader != nullptr); StackHandleScope<3> hs(soa.Self()); Handle class_loader(hs.NewHandle( soa.Decode(jclass_loader))); Handle klass1 = hs.NewHandle(class_linker_->FindClass(soa.Self(), "Lpackage1/Package1;", class_loader)); ASSERT_TRUE(klass1 != nullptr); Handle klass2 = hs.NewHandle(class_linker_->FindClass(soa.Self(), "Lpackage2/Package2;", class_loader)); ASSERT_TRUE(klass2 != nullptr); EXPECT_OBJ_PTR_EQ(klass1->GetDexCache(), klass2->GetDexCache()); EXPECT_NE(klass1->NumStaticFields(), 0u); for (ArtField& field : klass2->GetSFields()) { EXPECT_FALSE( klass1->ResolvedFieldAccessTest( klass2.Get(), &field, klass1->GetDexCache(), field.GetDexFieldIndex())); } } TEST_F(DexCacheMethodHandlesTest, TestResolvedMethodTypes) { ScopedObjectAccess soa(Thread::Current()); jobject jclass_loader(LoadDex("MethodTypes")); ASSERT_TRUE(jclass_loader != nullptr); StackHandleScope<5> hs(soa.Self()); Handle class_loader(hs.NewHandle( soa.Decode(jclass_loader))); Handle method_types( hs.NewHandle(class_linker_->FindClass(soa.Self(), "LMethodTypes;", class_loader))); class_linker_->EnsureInitialized(soa.Self(), method_types, true, true); ArtMethod* method1 = method_types->FindClassMethod( "method1", "(Ljava/lang/String;)Ljava/lang/String;", kRuntimePointerSize); ASSERT_TRUE(method1 != nullptr); ASSERT_FALSE(method1->IsDirect()); ArtMethod* method2 = method_types->FindClassMethod( "method2", "(Ljava/lang/String;Ljava/lang/String;)Ljava/lang/String;", kRuntimePointerSize); ASSERT_TRUE(method2 != nullptr); ASSERT_FALSE(method2->IsDirect()); const DexFile& dex_file = *(method1->GetDexFile()); Handle dex_cache = hs.NewHandle( class_linker_->FindDexCache(Thread::Current(), dex_file)); const dex::MethodId& method1_id = dex_file.GetMethodId(method1->GetDexMethodIndex()); const dex::MethodId& method2_id = dex_file.GetMethodId(method2->GetDexMethodIndex()); Handle method1_type = hs.NewHandle( class_linker_->ResolveMethodType(soa.Self(), method1_id.proto_idx_, dex_cache, class_loader)); Handle method2_type = hs.NewHandle( class_linker_->ResolveMethodType(soa.Self(), method2_id.proto_idx_, dex_cache, class_loader)); EXPECT_EQ(method1_type.Get(), dex_cache->GetResolvedMethodType(method1_id.proto_idx_)); EXPECT_EQ(method2_type.Get(), dex_cache->GetResolvedMethodType(method2_id.proto_idx_)); // The MethodTypes dex file contains a single interface with two abstract // methods. It must therefore contain precisely two method IDs. ASSERT_EQ(2u, dex_file.NumProtoIds()); ASSERT_EQ(dex_file.NumProtoIds(), dex_cache->NumResolvedMethodTypes()); MethodTypeDexCacheType* method_types_cache = dex_cache->GetResolvedMethodTypes(); for (size_t i = 0; i < dex_file.NumProtoIds(); ++i) { const MethodTypeDexCachePair pair = method_types_cache[i].load(std::memory_order_relaxed); if (dex::ProtoIndex(pair.index) == method1_id.proto_idx_) { ASSERT_EQ(method1_type.Get(), pair.object.Read()); } else if (dex::ProtoIndex(pair.index) == method2_id.proto_idx_) { ASSERT_EQ(method2_type.Get(), pair.object.Read()); } else { ASSERT_TRUE(false); } } } } // namespace mirror } // namespace art