// Copyright 2019 The Chromium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #include "base/android/bundle_utils.h" #include #include #include "base/android/jni_android.h" #include "base/android/jni_string.h" #include "base/base_jni_headers/BundleUtils_jni.h" #include "base/files/file_path.h" #include "base/logging.h" // These symbols are added by the lld linker when creating a partitioned shared // library. The symbols live in the base library, and are used to properly load // the other partitions (feature libraries) when needed. struct PartitionIndexEntry { int32_t name_relptr; int32_t addr_relptr; uint32_t size; }; static_assert(sizeof(PartitionIndexEntry) == 12U, "Unexpected PartitionIndexEntry size"); // Marked as weak_import because these symbols are lld-specific. The method that // uses them will only be invoked in builds that have lld-generated partitions. extern PartitionIndexEntry __part_index_begin[] __attribute__((weak_import)); extern PartitionIndexEntry __part_index_end[] __attribute__((weak_import)); extern "C" { // Marked as weak_import because this symbol is either supplied by the system // (on Android N+), or by Crazy Linker (Android M and prior). extern void* android_dlopen_ext(const char* __filename, int __flags, const android_dlextinfo* __info) __attribute__((weak_import)); } // extern "C" namespace base { namespace android { namespace { const void* ReadRelPtr(const int32_t* relptr) { return reinterpret_cast(relptr) + *relptr; } } // namespace // static std::string BundleUtils::ResolveLibraryPath(const std::string& library_name) { JNIEnv* env = AttachCurrentThread(); ScopedJavaLocalRef java_path = Java_BundleUtils_getNativeLibraryPath( env, base::android::ConvertUTF8ToJavaString(env, library_name)); // TODO(https://crbug.com/1019853): Remove this tolerance. if (!java_path) { return std::string(); } return base::android::ConvertJavaStringToUTF8(env, java_path); } // static bool BundleUtils::IsBundle() { return Java_BundleUtils_isBundleForNative( base::android::AttachCurrentThread()); } // static void* BundleUtils::DlOpenModuleLibraryPartition(const std::string& library_name, const std::string& partition) { // TODO(https://crbug.com/1019853): Remove this tolerance. std::string library_path = ResolveLibraryPath(library_name); if (library_path.empty()) { return nullptr; } // Linear search is required here because the partition descriptors are not // ordered. If a large number of partitions come into existence, lld could be // modified to sort the partitions. DCHECK(__part_index_begin != nullptr); DCHECK(__part_index_end != nullptr); for (const PartitionIndexEntry* part = __part_index_begin; part != __part_index_end; ++part) { std::string name( reinterpret_cast(ReadRelPtr(&part->name_relptr))); if (name == partition) { android_dlextinfo info = {}; info.flags = ANDROID_DLEXT_RESERVED_ADDRESS; info.reserved_addr = const_cast(ReadRelPtr(&part->addr_relptr)); info.reserved_size = part->size; DCHECK(android_dlopen_ext != nullptr); return android_dlopen_ext(library_path.c_str(), RTLD_LOCAL, &info); } } NOTREACHED(); return nullptr; } } // namespace android } // namespace base