安卓类加载器补充InMemoryDexClassLoader
InMemoryDexClassLoader加载内存dex的流程 Android 8新增了InMemoryDexClassLoader,专门用于加载内存dex,依旧以15的r3为例
InMemoryDexClassLoader InMemoryDexClassLoader
是一个继承自 BaseDexClassLoader
的类,用于在运行时从内存中加载 Dex 文件,而无需将其存储到磁盘。它主要用于动态加载字节缓冲区中的类。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 public final class InMemoryDexClassLoader extends BaseDexClassLoader { public InMemoryDexClassLoader (@NonNull ByteBuffer @NonNull [] dexBuffers, @Nullable String librarySearchPath, @Nullable ClassLoader parent) { super (dexBuffers, librarySearchPath, parent); } public InMemoryDexClassLoader (@NonNull ByteBuffer @NonNull [] dexBuffers,@Nullable ClassLoader parent) { this (dexBuffers, null , parent); } public InMemoryDexClassLoader (@NonNull ByteBuffer dexBuffer, @Nullable ClassLoader parent) { this (new ByteBuffer [] { dexBuffer }, parent); } }
我们来看看父类的BaseDexClassLoader 构造方法
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 public class BaseDexClassLoader extends ClassLoader {... public BaseDexClassLoader (ByteBuffer[] dexFiles, String librarySearchPath, ClassLoader parent) { super (parent); this .sharedLibraryLoaders = null ; this .sharedLibraryLoadersAfter = null ; this .pathList = new DexPathList (this , librarySearchPath); this .pathList.initByteBufferDexPath(dexFiles); this .pathList.maybeRunBackgroundVerification(this ); } }
DexPathList() 1 2 3 4 5 6 7 8 9 10 11 12 13 public DexPathList (ClassLoader definingContext, String librarySearchPath) { if (definingContext == null ) { throw new NullPointerException ("definingContext == null" ); } this .definingContext = definingContext; this .nativeLibraryDirectories = splitPaths(librarySearchPath, false ); this .systemNativeLibraryDirectories =splitPaths(System.getProperty("java.library.path" ), true ); this .nativeLibraryPathElements = makePathElements(getAllNativeLibraryDirectories()); }
initByteBufferDexPath 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 void initByteBufferDexPath (ByteBuffer[] dexFiles) {... final List<IOException> suppressedExceptions = new ArrayList <IOException>(); try { Element[] null_elements = null ; DexFile dex = new DexFile (dexFiles, definingContext, null_elements); dexElements = new Element [] { new Element (dex) }; } catch (IOException suppressed) { System.logE("Unable to load dex files" , suppressed); suppressedExceptions.add(suppressed); dexElements = new Element [0 ]; } if (suppressedExceptions.size() > 0 ) { dexElementsSuppressedExceptions = suppressedExceptions.toArray( new IOException [suppressedExceptions.size()]); } }
DexFile 调用openInMemoryDexFile()
方法,返回的cookie是用于操作dex文件的唯一凭证,可以理解为”身份码”。
1 2 3 4 5 6 7 DexFile(ByteBuffer[] bufs, ClassLoader loader, DexPathList.Element[] elements) throws IOException { mCookie = openInMemoryDexFiles(bufs, loader, elements); mInternalCookie = mCookie; mFileName = null ; }
openInMemoryDexFile() 最后调用openInMemoryDexFilesNative
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 private static Object openInMemoryDexFiles (ByteBuffer[] bufs, ClassLoader loader, DexPathList.Element[] elements) throws IOException { byte [][] arrays = new byte [bufs.length][]; int [] starts = new int [bufs.length]; int [] ends = new int [bufs.length]; for (int i = 0 ; i < bufs.length; ++i) { arrays[i] = bufs[i].isDirect() ? null : bufs[i].array(); starts[i] = bufs[i].position(); ends[i] = bufs[i].limit(); } return openInMemoryDexFilesNative(bufs, arrays, starts, ends, loader, elements); }
openInMemoryDexFilesNative 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 static jobject DexFile_openInMemoryDexFilesNative (JNIEnv* env, jclass, jobjectArray buffers, jobjectArray arrays, jintArray jstarts, jintArray jends, jobject class_loader, jobjectArray dex_elements) {... ... std::vector<std::string> error_msgs; const OatFile* oat_file = nullptr ; std::vector<std::unique_ptr<const DexFile>> dex_files = Runtime::Current ()->GetOatFileManager ().OpenDexFilesFromOat (std::move (dex_mem_maps), class_loader, dex_elements, &oat_file, &error_msgs); return CreateCookieFromOatFileManagerResult (env, dex_files, oat_file, error_msgs); }
CreateCookieFromOatFileManagerResult 用于根据加载 .dex
文件的结果,创建一个供 Java 层使用的句柄(通常是一个 jlongArray
,表示加载的 .dex
文件的句柄)。其主要职责是处理加载结果中的异常情况,或者将成功加载的 DexFile
信息转换为一个数组返回给 Java。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 static jobject CreateCookieFromOatFileManagerResult ( JNIEnv* env, std::vector<std::unique_ptr<const DexFile>>& dex_files, const OatFile* oat_file, const std::vector<std::string>& error_msgs) { ClassLinker* linker = Runtime::Current ()->GetClassLinker (); if (dex_files.empty ()) { ... } jlongArray array = ConvertDexFilesToJavaArray (env, oat_file, dex_files); ... return array; }
OpenDexFilesFromOat 它的主要作用是解析 OAT 文件并提取出嵌入的 DEX 文件,为后续的类加载过程服务
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 std::vector<std::unique_ptr<const DexFile>> OatFileManager::OpenDexFilesFromOat( const char * dex_location, jobject class_loader, jobjectArray dex_elements, const OatFile** out_oat_file, std::vector<std::string>* error_msgs) { ScopedTrace trace (StringPrintf("%s(%s)" , __FUNCTION__, dex_location) ); CHECK(dex_location != nullptr); CHECK(error_msgs != nullptr); std::vector<std::unique_ptr<const DexFile>> dex_files; std::unique_ptr<ClassLoaderContext> context ( ClassLoaderContext::CreateContextForClassLoader(class_loader, dex_elements) ); if (class_loader == nullptr) { LOG(WARNING) << "Opening an oat file without a class loader. " << "Are you using the deprecated DexFile APIs?" ; } else if (context != nullptr) { auto oat_file_assistant = std::make_unique<OatFileAssistant>(dex_location, kRuntimeISA, context.get(), runtime->GetOatFilesExecutable(), only_use_system_oat_files_); ... std::unique_ptr<const OatFile> oat_file(oat_file_assistant->GetBestOatFile().release()); VLOG(oat) << "OatFileAssistant(" << dex_location << ").GetBestOatFile()=" << (oat_file != nullptr ? oat_file->GetLocation() : "" ) << " (executable=" << (oat_file != nullptr ? oat_file->IsExecutable() : false ) << ")" ; ... if (!added_image_space) { DCHECK(dex_files.empty()); ... if (oat_file != nullptr) { dex_files = oat_file_assistant->LoadDexFiles(*oat_file.get(), dex_location); ... } } if (dex_files.empty()) { ScopedTrace failed_to_open_dex_files ("FailedToOpenDexFilesFromOat" ) ; error_msgs->push_back("Failed to open dex files from " + odex_location); } else if (should_madvise) { } } if (oat_file != nullptr) { VLOG(class_linker) << "Registering " << oat_file->GetLocation(); *out_oat_file = RegisterOatFile(std::move(oat_file)); } } else { ... } } if (dex_files.empty()) { std::string error_msg; static constexpr bool kVerifyChecksum = true ; ArtDexFileLoader dex_file_loader (dex_location) ; if (!dex_file_loader.Open(Runtime::Current()->IsVerificationEnabled(), kVerifyChecksum, &error_msg, &dex_files)) { ScopedTrace fail_to_open_dex_from_apk ("FailedToOpenDexFilesFromApk" ) ; LOG(WARNING) << error_msg; error_msgs->push_back("Failed to open dex files from " + std::string(dex_location) + " because: " + error_msg); } } ... return dex_files; }
LoadDexFiles() 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 bool OatFileAssistant::LoadDexFiles(const OatFile& oat_file, const std::string& dex_location, std::vector<std::unique_ptr<const DexFile>>* out_dex_files) { std::string error_msg; const OatDexFile* oat_dex_file = oat_file.GetOatDexFile(dex_location.c_str(), &error_msg); if (oat_dex_file == nullptr) { LOG(WARNING) << error_msg; return false ; } std::unique_ptr<const DexFile> dex_file = oat_dex_file->OpenDexFile(&error_msg); if (dex_file.get() == nullptr) { LOG(WARNING) << "Failed to open dex file from oat dex file: " << error_msg; return false ; } out_dex_files->push_back(std::move(dex_file)); for (size_t i = 1 ;; i++) { std::string multidex_dex_location = DexFileLoader::GetMultiDexLocation(i, dex_location.c_str()); oat_dex_file = oat_file.GetOatDexFile(multidex_dex_location.c_str()); if (oat_dex_file == nullptr) { break ; } dex_file = oat_dex_file->OpenDexFile(&error_msg); if (dex_file.get() == nullptr) { LOG(WARNING) << "Failed to open dex file from oat dex file: " << error_msg; return false ; } out_dex_files->push_back(std::move(dex_file)); } return true ; }
GetOatDexFile() 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 const OatDexFile* OatFile::GetOatDexFile(const char * dex_location, std::string* error_msg) const { const OatDexFile* oat_dex_file = nullptr; std::string_view key (dex_location) ; auto primary_it = oat_dex_files_.find(key); if (primary_it != oat_dex_files_.end()) { oat_dex_file = primary_it->second; DCHECK(oat_dex_file != nullptr); } else { MutexLock mu (Thread::Current() , secondary_lookup_lock_); auto secondary_lb = secondary_oat_dex_files_.lower_bound(key); if (secondary_lb != secondary_oat_dex_files_.end() && key == secondary_lb->first) { oat_dex_file = secondary_lb->second; } else { std::string dex_canonical_location = DexFileLoader::GetDexCanonicalLocation(dex_location); if (dex_canonical_location != dex_location) { std::string_view canonical_key (dex_canonical_location) ; auto canonical_it = oat_dex_files_.find(canonical_key); if (canonical_it != oat_dex_files_.end()) { oat_dex_file = canonical_it->second; } } string_cache_.emplace_back(key.data(), key.length()); std::string_view key_copy (string_cache_.back() ); secondary_oat_dex_files_.PutBefore(secondary_lb, key_copy, oat_dex_file); } } if (oat_dex_file == nullptr) { if (error_msg != nullptr) { std::string dex_canonical_location = DexFileLoader::GetDexCanonicalLocation(dex_location); *error_msg = "Failed to find OatDexFile for DexFile " + std::string(dex_location) + " (canonical path " + dex_canonical_location + ") in OatFile " + GetLocation(); } return nullptr; } return oat_dex_file; }
OatDexFile::OpenDexFile 1 2 3 4 5 6 7 8 9 10 11 12 13 std::unique_ptr<const DexFile> OatDexFile::OpenDexFile(std::string* error_msg) const { ScopedTrace trace (__PRETTY_FUNCTION__) ; static constexpr bool kVerify = false ; static constexpr bool kVerifyChecksum = false ; ArtDexFileLoader dex_file_loader (dex_file_container_, dex_file_location_) ; return dex_file_loader.OpenOne(dex_file_pointer_ - dex_file_container_->Begin(), dex_file_location_checksum_, this , kVerify, kVerifyChecksum, error_msg); }
dex_file_loader.OpenOne 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 std::unique_ptr<const DexFile> DexFileLoader::OpenOne(size_t header_offset, uint32_t location_checksum, const OatDexFile* oat_dex_file, bool verify, bool verify_checksum, std::string* error_msg) { DEXFILE_SCOPED_TRACE(std::string("Open dex file " ) + location_); uint32_t magic; if (!InitAndReadMagic(header_offset, &magic, error_msg) || !MapRootContainer(error_msg)) { DCHECK(!error_msg->empty()); return {}; } DCHECK(root_container_ != nullptr); DCHECK_LE(header_offset, root_container_->Size()); std::unique_ptr<const DexFile> dex_file = OpenCommon(root_container_, root_container_->Begin() + header_offset, root_container_->Size() - header_offset, location_, location_checksum, oat_dex_file, verify, verify_checksum, error_msg, nullptr); return dex_file; }
DexFileLoader::OpenCommon 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 std::unique_ptr<DexFile> DexFileLoader::OpenCommon(std::shared_ptr<DexFileContainer> container, const uint8_t* base, size_t app_compat_size, const std::string& location, std::optional<uint32_t> location_checksum, const OatDexFile* oat_dex_file, bool verify, bool verify_checksum, std::string* error_msg, DexFileLoaderErrorCode* error_code ) { if (container == nullptr) { container = std::make_shared<MemoryDexFileContainer>(base, app_compat_size); } CHECK_GE(base, container->Begin()); CHECK_LE(base, container->End()); const size_t size = container->End() - base; if (error_code != nullptr) { *error_code = DexFileLoaderErrorCode::kDexFileError; } std::unique_ptr<DexFile> dex_file; auto header = reinterpret_cast<const DexFile::Header*>(base); if (size >= sizeof(StandardDexFile::Header) && StandardDexFile::IsMagicValid(base)) { uint32_t checksum = location_checksum.value_or(header->checksum_); dex_file.reset(new StandardDexFile (base, location, checksum, oat_dex_file, container)); } else if (size >= sizeof(CompactDexFile::Header) && CompactDexFile::IsMagicValid(base)) { uint32_t checksum = location_checksum.value_or(header->checksum_); dex_file.reset(new CompactDexFile (base, location, checksum, oat_dex_file, container)); } else { *error_msg = StringPrintf("Invalid or truncated dex file '%s'" , location.c_str()); } if (dex_file == nullptr) { *error_msg = StringPrintf("Failed to open dex file '%s': %s" , location.c_str(), error_msg->c_str()); return nullptr; } if (!dex_file->Init(error_msg)) { dex_file.reset(); return nullptr; } if (verify && !dex_file->IsCompactDexFile()) { DEXFILE_SCOPED_TRACE(std::string("Verify dex file " ) + location); if (!dex::Verify(dex_file.get(), location.c_str(), verify_checksum, error_msg)) { if (error_code != nullptr) { *error_code = DexFileLoaderErrorCode::kVerifyError; } return nullptr; } } if (error_code != nullptr) { *error_code = DexFileLoaderErrorCode::kNoError; } return dex_file; }
从这里可以看出,不管是DexClassLoader、PathClassLoader,还是InMemoryDexClassLoader,尽管它们的流程可能会不太一样,但最终都会调用OpenCommon()
方法创建dexfile,而且该方法的参数含有dex的起始地址(base)和大小(size),因此这里可以作为一个非常好的脱壳点。