compiler_driver-inl.h revision 1a5337fff2cc6cb9d563c8b32aca75f485d23373
1/* 2 * Copyright (C) 2012 The Android Open Source Project 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License. 15 */ 16 17#ifndef ART_COMPILER_DRIVER_COMPILER_DRIVER_INL_H_ 18#define ART_COMPILER_DRIVER_COMPILER_DRIVER_INL_H_ 19 20#include "compiler_driver.h" 21 22#include "art_field-inl.h" 23#include "art_method-inl.h" 24#include "base/enums.h" 25#include "class_linker-inl.h" 26#include "dex_compilation_unit.h" 27#include "mirror/class_loader.h" 28#include "mirror/dex_cache-inl.h" 29#include "scoped_thread_state_change-inl.h" 30#include "handle_scope-inl.h" 31 32namespace art { 33 34inline mirror::ClassLoader* CompilerDriver::GetClassLoader(const ScopedObjectAccess& soa, 35 const DexCompilationUnit* mUnit) { 36 return soa.Decode<mirror::ClassLoader>(mUnit->GetClassLoader()).Ptr(); 37} 38 39inline mirror::Class* CompilerDriver::ResolveClass( 40 const ScopedObjectAccess& soa, Handle<mirror::DexCache> dex_cache, 41 Handle<mirror::ClassLoader> class_loader, uint16_t cls_index, 42 const DexCompilationUnit* mUnit) { 43 DCHECK_EQ(dex_cache->GetDexFile(), mUnit->GetDexFile()); 44 DCHECK_EQ(class_loader.Get(), GetClassLoader(soa, mUnit)); 45 mirror::Class* cls = mUnit->GetClassLinker()->ResolveType( 46 *mUnit->GetDexFile(), cls_index, dex_cache, class_loader); 47 DCHECK_EQ(cls == nullptr, soa.Self()->IsExceptionPending()); 48 if (UNLIKELY(cls == nullptr)) { 49 // Clean up any exception left by type resolution. 50 soa.Self()->ClearException(); 51 } 52 return cls; 53} 54 55inline mirror::Class* CompilerDriver::ResolveCompilingMethodsClass( 56 const ScopedObjectAccess& soa, Handle<mirror::DexCache> dex_cache, 57 Handle<mirror::ClassLoader> class_loader, const DexCompilationUnit* mUnit) { 58 DCHECK_EQ(dex_cache->GetDexFile(), mUnit->GetDexFile()); 59 DCHECK_EQ(class_loader.Get(), GetClassLoader(soa, mUnit)); 60 const DexFile::MethodId& referrer_method_id = 61 mUnit->GetDexFile()->GetMethodId(mUnit->GetDexMethodIndex()); 62 return ResolveClass(soa, dex_cache, class_loader, referrer_method_id.class_idx_, mUnit); 63} 64 65inline ArtField* CompilerDriver::ResolveFieldWithDexFile( 66 const ScopedObjectAccess& soa, Handle<mirror::DexCache> dex_cache, 67 Handle<mirror::ClassLoader> class_loader, const DexFile* dex_file, 68 uint32_t field_idx, bool is_static) { 69 DCHECK_EQ(dex_cache->GetDexFile(), dex_file); 70 ArtField* resolved_field = Runtime::Current()->GetClassLinker()->ResolveField( 71 *dex_file, field_idx, dex_cache, class_loader, is_static); 72 DCHECK_EQ(resolved_field == nullptr, soa.Self()->IsExceptionPending()); 73 if (UNLIKELY(resolved_field == nullptr)) { 74 // Clean up any exception left by type resolution. 75 soa.Self()->ClearException(); 76 return nullptr; 77 } 78 if (UNLIKELY(resolved_field->IsStatic() != is_static)) { 79 // ClassLinker can return a field of the wrong kind directly from the DexCache. 80 // Silently return null on such incompatible class change. 81 return nullptr; 82 } 83 return resolved_field; 84} 85 86inline ArtField* CompilerDriver::ResolveField( 87 const ScopedObjectAccess& soa, Handle<mirror::DexCache> dex_cache, 88 Handle<mirror::ClassLoader> class_loader, const DexCompilationUnit* mUnit, 89 uint32_t field_idx, bool is_static) { 90 DCHECK_EQ(class_loader.Get(), GetClassLoader(soa, mUnit)); 91 return ResolveFieldWithDexFile(soa, dex_cache, class_loader, mUnit->GetDexFile(), field_idx, 92 is_static); 93} 94 95inline std::pair<bool, bool> CompilerDriver::IsFastInstanceField( 96 mirror::DexCache* dex_cache, mirror::Class* referrer_class, 97 ArtField* resolved_field, uint16_t field_idx) { 98 DCHECK(!resolved_field->IsStatic()); 99 ObjPtr<mirror::Class> fields_class = resolved_field->GetDeclaringClass(); 100 bool fast_get = referrer_class != nullptr && 101 referrer_class->CanAccessResolvedField(fields_class, 102 resolved_field, 103 dex_cache, 104 field_idx); 105 bool fast_put = fast_get && (!resolved_field->IsFinal() || fields_class == referrer_class); 106 return std::make_pair(fast_get, fast_put); 107} 108 109template <typename ArtMember> 110inline bool CompilerDriver::CanAccessResolvedMember(mirror::Class* referrer_class ATTRIBUTE_UNUSED, 111 mirror::Class* access_to ATTRIBUTE_UNUSED, 112 ArtMember* member ATTRIBUTE_UNUSED, 113 mirror::DexCache* dex_cache ATTRIBUTE_UNUSED, 114 uint32_t field_idx ATTRIBUTE_UNUSED) { 115 // Not defined for ArtMember values other than ArtField or ArtMethod. 116 UNREACHABLE(); 117} 118 119template <> 120inline bool CompilerDriver::CanAccessResolvedMember<ArtField>(mirror::Class* referrer_class, 121 mirror::Class* access_to, 122 ArtField* field, 123 mirror::DexCache* dex_cache, 124 uint32_t field_idx) { 125 return referrer_class->CanAccessResolvedField(access_to, field, dex_cache, field_idx); 126} 127 128template <> 129inline bool CompilerDriver::CanAccessResolvedMember<ArtMethod>( 130 mirror::Class* referrer_class, 131 mirror::Class* access_to, 132 ArtMethod* method, 133 mirror::DexCache* dex_cache, 134 uint32_t field_idx) { 135 return referrer_class->CanAccessResolvedMethod(access_to, method, dex_cache, field_idx); 136} 137 138template <typename ArtMember> 139inline std::pair<bool, bool> CompilerDriver::IsClassOfStaticMemberAvailableToReferrer( 140 mirror::DexCache* dex_cache, 141 mirror::Class* referrer_class, 142 ArtMember* resolved_member, 143 uint16_t member_idx, 144 uint32_t* storage_index) { 145 DCHECK(resolved_member->IsStatic()); 146 if (LIKELY(referrer_class != nullptr)) { 147 ObjPtr<mirror::Class> members_class = resolved_member->GetDeclaringClass(); 148 if (members_class == referrer_class) { 149 *storage_index = members_class->GetDexTypeIndex(); 150 return std::make_pair(true, true); 151 } 152 if (CanAccessResolvedMember<ArtMember>( 153 referrer_class, members_class.Ptr(), resolved_member, dex_cache, member_idx)) { 154 // We have the resolved member, we must make it into a index for the referrer 155 // in its static storage (which may fail if it doesn't have a slot for it) 156 // TODO: for images we can elide the static storage base null check 157 // if we know there's a non-null entry in the image 158 const DexFile* dex_file = dex_cache->GetDexFile(); 159 uint32_t storage_idx = DexFile::kDexNoIndex; 160 if (LIKELY(members_class->GetDexCache() == dex_cache)) { 161 // common case where the dex cache of both the referrer and the member are the same, 162 // no need to search the dex file 163 storage_idx = members_class->GetDexTypeIndex(); 164 } else { 165 // Search dex file for localized ssb index, may fail if member's class is a parent 166 // of the class mentioned in the dex file and there is no dex cache entry. 167 storage_idx = resolved_member->GetDeclaringClass()->FindTypeIndexInOtherDexFile(*dex_file); 168 } 169 if (storage_idx != DexFile::kDexNoIndex) { 170 *storage_index = storage_idx; 171 return std::make_pair(true, !resolved_member->IsFinal()); 172 } 173 } 174 } 175 // Conservative defaults. 176 *storage_index = DexFile::kDexNoIndex; 177 return std::make_pair(false, false); 178} 179 180inline std::pair<bool, bool> CompilerDriver::IsFastStaticField( 181 mirror::DexCache* dex_cache, mirror::Class* referrer_class, 182 ArtField* resolved_field, uint16_t field_idx, uint32_t* storage_index) { 183 return IsClassOfStaticMemberAvailableToReferrer( 184 dex_cache, referrer_class, resolved_field, field_idx, storage_index); 185} 186 187inline bool CompilerDriver::IsClassOfStaticMethodAvailableToReferrer( 188 mirror::DexCache* dex_cache, mirror::Class* referrer_class, 189 ArtMethod* resolved_method, uint16_t method_idx, uint32_t* storage_index) { 190 std::pair<bool, bool> result = IsClassOfStaticMemberAvailableToReferrer( 191 dex_cache, referrer_class, resolved_method, method_idx, storage_index); 192 // Only the first member of `result` is meaningful, as there is no 193 // "write access" to a method. 194 return result.first; 195} 196 197inline ArtMethod* CompilerDriver::ResolveMethod( 198 ScopedObjectAccess& soa, Handle<mirror::DexCache> dex_cache, 199 Handle<mirror::ClassLoader> class_loader, const DexCompilationUnit* mUnit, 200 uint32_t method_idx, InvokeType invoke_type, bool check_incompatible_class_change) { 201 DCHECK_EQ(class_loader.Get(), GetClassLoader(soa, mUnit)); 202 ArtMethod* resolved_method = 203 check_incompatible_class_change 204 ? mUnit->GetClassLinker()->ResolveMethod<ClassLinker::kForceICCECheck>( 205 *dex_cache->GetDexFile(), method_idx, dex_cache, class_loader, nullptr, invoke_type) 206 : mUnit->GetClassLinker()->ResolveMethod<ClassLinker::kNoICCECheckForCache>( 207 *dex_cache->GetDexFile(), method_idx, dex_cache, class_loader, nullptr, invoke_type); 208 if (UNLIKELY(resolved_method == nullptr)) { 209 DCHECK(soa.Self()->IsExceptionPending()); 210 // Clean up any exception left by type resolution. 211 soa.Self()->ClearException(); 212 } 213 return resolved_method; 214} 215 216} // namespace art 217 218#endif // ART_COMPILER_DRIVER_COMPILER_DRIVER_INL_H_ 219