【Android 逆向】ART 脱壳 ( InMemoryDexClassLoader 脱壳 | dex_file.cc 中创建 DexFile 实例对象的相关函数分析 )
文章目錄
- 前言
- 一、dalvik_system_DexFile.cc#CreateDexFile 函數分析
- 二、dex_file.cc#DexFile::Open 函數分析
- 三、dex_file.cc#DexFile::OpenCommon 函數分析
- 四、dex_file.cc#DexFile 構造函數分析
- 總結
前言
在上一篇博客 【Android 逆向】ART 脫殼 ( InMemoryDexClassLoader 脫殼 | DexFile.java 對應的 dalvik_system_DexFile.cc 本地函數分析 ) 中 , 分析了 DexFile.java 中的 createCookieWithDirectBuffer 和 createCookieWithArray 函數對應的 native 函數 ,
定義在 /art/runtime/native/dalvik_system_DexFile.cc 中的 dalvik_system_DexFile.cc 的 DexFile_createCookieWithDirectBuffer 函數 ,
這兩個函數都調用了 CreateSingleDexFileCookie 函數 , 在該函數中創建了 dex_file 對象 , 傳入了 CreateDexFile(env, std::move(data)) 參數 ;
一、dalvik_system_DexFile.cc#CreateDexFile 函數分析
在開始處調用 DexFile::Open 函數 , 返回 std::unique_ptr<const DexFile>
傳入的參數 std::string location 是 dex 文件在內存映射的起始和結束地址 ;
dalvik_system_DexFile.cc#CreateDexFile 函數源碼 :
static const DexFile* CreateDexFile(JNIEnv* env, std::unique_ptr<MemMap> dex_mem_map) {std::string location = StringPrintf("Anonymous-DexFile@%p-%p",dex_mem_map->Begin(),dex_mem_map->End());std::string error_message;// ★ 核心跳轉std::unique_ptr<const DexFile> dex_file(DexFile::Open(location,0,std::move(dex_mem_map),/* verify */ true,/* verify_location */ true,&error_message));if (dex_file == nullptr) {ScopedObjectAccess soa(env);ThrowWrappedIOException("%s", error_message.c_str());return nullptr;}if (!dex_file->DisableWrite()) {ScopedObjectAccess soa(env);ThrowWrappedIOException("Failed to make dex file read-only");return nullptr;}return dex_file.release(); }源碼路徑 : /art/runtime/native/dalvik_system_DexFile.cc#CreateDexFile
二、dex_file.cc#DexFile::Open 函數分析
傳入的 const std::string& location 參數是 dex 文件在內存中的映射起止地址 ;
在該函數中 , 又調用了 OpenCommon 函數 ;
std::unique_ptr<const DexFile> DexFile::Open(const std::string& location,uint32_t location_checksum,std::unique_ptr<MemMap> map,bool verify,bool verify_checksum,std::string* error_msg) {ScopedTrace trace(std::string("Open dex file from mapped-memory ") + location);CHECK(map.get() != nullptr);if (map->Size() < sizeof(DexFile::Header)) {*error_msg = StringPrintf("DexFile: failed to open dex file '%s' that is too short to have a header",location.c_str());return nullptr;}// ★ 核心跳轉std::unique_ptr<DexFile> dex_file = OpenCommon(map->Begin(),map->Size(),location,location_checksum,kNoOatDexFile,verify,verify_checksum,error_msg);if (dex_file != nullptr) {dex_file->mem_map_.reset(map.release());}return dex_file; }
源碼路徑 : /art/runtime/dex_file.cc
三、dex_file.cc#DexFile::OpenCommon 函數分析
在 OpenCommon 函數中 , 又新建了 DexFile 對象 , 此處調用了 DexFile 的構造函數 ;
std::unique_ptr<DexFile> DexFile::OpenCommon(const uint8_t* base,size_t size,const std::string& location,uint32_t location_checksum,const OatDexFile* oat_dex_file,bool verify,bool verify_checksum,std::string* error_msg,VerifyResult* verify_result) {if (verify_result != nullptr) {*verify_result = VerifyResult::kVerifyNotAttempted;}// ★ 核心跳轉 新建 DexFile 對象std::unique_ptr<DexFile> dex_file(new DexFile(base,size,location,location_checksum,oat_dex_file));if (dex_file == nullptr) {*error_msg = StringPrintf("Failed to open dex file '%s' from memory: %s", location.c_str(),error_msg->c_str());return nullptr;}if (!dex_file->Init(error_msg)) {dex_file.reset();return nullptr;}if (verify && !DexFileVerifier::Verify(dex_file.get(),dex_file->Begin(),dex_file->Size(),location.c_str(),verify_checksum,error_msg)) {if (verify_result != nullptr) {*verify_result = VerifyResult::kVerifyFailed;}return nullptr;}if (verify_result != nullptr) {*verify_result = VerifyResult::kVerifySucceeded;}return dex_file; }源碼路徑 : /art/runtime/dex_file.cc
四、dex_file.cc#DexFile 構造函數分析
在 dex_file.cc 中的 DexFile 構造函數中 , 也存在 dex 文件在內存中的首地址 , 該地址也可以作為脫殼點 ;
DexFile::DexFile(const uint8_t* base,size_t size,const std::string& location,uint32_t location_checksum,const OatDexFile* oat_dex_file): begin_(base),size_(size),location_(location),location_checksum_(location_checksum),header_(reinterpret_cast<const Header*>(base)),string_ids_(reinterpret_cast<const StringId*>(base + header_->string_ids_off_)),type_ids_(reinterpret_cast<const TypeId*>(base + header_->type_ids_off_)),field_ids_(reinterpret_cast<const FieldId*>(base + header_->field_ids_off_)),method_ids_(reinterpret_cast<const MethodId*>(base + header_->method_ids_off_)),proto_ids_(reinterpret_cast<const ProtoId*>(base + header_->proto_ids_off_)),class_defs_(reinterpret_cast<const ClassDef*>(base + header_->class_defs_off_)),method_handles_(nullptr),num_method_handles_(0),call_site_ids_(nullptr),num_call_site_ids_(0),oat_dex_file_(oat_dex_file) {CHECK(begin_ != nullptr) << GetLocation();CHECK_GT(size_, 0U) << GetLocation();// Check base (=header) alignment.// Must be 4-byte aligned to avoid undefined behavior when accessing// any of the sections via a pointer.CHECK_ALIGNED(begin_, alignof(Header));InitializeSectionsFromMapList(); }源碼路徑 : /art/runtime/dex_file.cc#DexFile
總結
在 InMemoryDexClassLoader 類加載器中 , 加載 dex 文件時 , 沒有對 dex 文件進行優化 ,
DexClassLoader 加載 dex 的同時 , 會對 dex 文件進行優化 ;
上面分析的所有帶 dex 文件起始地址和大小的函數 , 都可以作為脫殼點 ;
總結
以上是生活随笔為你收集整理的【Android 逆向】ART 脱壳 ( InMemoryDexClassLoader 脱壳 | dex_file.cc 中创建 DexFile 实例对象的相关函数分析 )的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 【Android 逆向】ART 脱壳 (
- 下一篇: 【Android 逆向】ART 脱壳 (