/* * Copyright (C) 2019 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 "linkerconfig/variableloader.h" #include #include #include #include #include #include "linkerconfig/environment.h" #include "linkerconfig/librarylistloader.h" #include "linkerconfig/log.h" #include "linkerconfig/stringutil.h" #include "linkerconfig/variables.h" using android::base::Result; using android::linkerconfig::modules::GetProductVndkVersion; using android::linkerconfig::modules::GetVendorVndkVersion; using android::linkerconfig::modules::TrimPrefix; using android::linkerconfig::modules::Variables; namespace { using namespace android::linkerconfig::generator; void LoadVndkVersionVariable() { Variables::AddValue("VENDOR_VNDK_VERSION", GetVendorVndkVersion()); Variables::AddValue("PRODUCT_VNDK_VERSION", GetProductVndkVersion()); } Result GetRealPath(std::string target_path) { char resolved_path[PATH_MAX]; if (realpath(target_path.c_str(), resolved_path) != nullptr) { return resolved_path; } return ErrnoErrorf("Failed to get realpath from {}", target_path); } void LoadVariableFromPartitionPath(const std::string& root, std::string variable_name, std::string partition) { auto real_path = GetRealPath(root + partition); if (real_path.ok()) { Variables::AddValue(variable_name, TrimPrefix(*real_path, root)); } else { LOG(WARNING) << real_path.error(); Variables::AddValue(variable_name, partition); } } void LoadPartitionPathVariables(const std::string& root) { // TODO(b/141714913): generalize path handling LoadVariableFromPartitionPath(root, "PRODUCT", "/product"); LoadVariableFromPartitionPath(root, "SYSTEM_EXT", "/system_ext"); } void LoadVndkLibraryListVariables(const std::string& root, const std::string& vndk_version, const std::string& partition) { if (vndk_version == "") { return; } const std::string vndk_path = root + "/apex/com.android.vndk.v" + vndk_version; // Skip loading if VNDK APEX is not available if (::access(vndk_path.c_str(), F_OK) != 0) { PLOG(ERROR) << "Unable to access VNDK APEX at path: " << vndk_path; return; } const std::string vndksp_libraries_path = vndk_path + "/etc/vndksp.libraries." + vndk_version + ".txt"; const std::string vndkcore_libraries_path = vndk_path + "/etc/vndkcore.libraries." + vndk_version + ".txt"; const std::string vndkprivate_libraries_path = vndk_path + "/etc/vndkprivate.libraries." + vndk_version + ".txt"; const std::string sanitizer_library_path = root + "/system/etc/sanitizer.libraries.txt"; Variables::AddValue("VNDK_SAMEPROCESS_LIBRARIES_" + partition, GetPublicLibrariesString(vndksp_libraries_path, vndkprivate_libraries_path)); Variables::AddValue("VNDK_CORE_LIBRARIES_" + partition, GetPublicLibrariesString(vndkcore_libraries_path, vndkprivate_libraries_path)); Variables::AddValue("SANITIZER_DEFAULT_" + partition, GetPublicLibrariesString(sanitizer_library_path, vndkcore_libraries_path)); if (partition == "VENDOR") { auto vndkcorevariant_library_path = root + "/system/etc/vndkcorevariant.libraries.txt"; Variables::AddValue("VNDK_USING_CORE_VARIANT_LIBRARIES", GetPublicLibrariesString(vndkcorevariant_library_path, vndkprivate_libraries_path)); } } void LoadLlndkLibraryListVariables(const std::string& root, const std::string& vndk_version, const std::string& partition) { if (vndk_version == "") { // llndk.libraries.txt in the system image does not contain version // information in the file name because system has fixed target version // which is always higher or equal than vendor / product images, and llndk // libraries (and its list) are backward compatible. const std::string llndk_libraries_in_system_path = root + "/system/etc/llndk.libraries.txt"; Variables::AddValue("LLNDK_LIBRARIES_" + partition, GetLibrariesString(llndk_libraries_in_system_path)); } else { const std::string vndk_path = root + "/apex/com.android.vndk.v" + vndk_version; // Skip loading if VNDK APEX is not available if (::access(vndk_path.c_str(), F_OK) != 0) { PLOG(ERROR) << "Unable to access VNDK APEX at path: " << vndk_path; return; } const std::string llndk_libraries_path = vndk_path + "/etc/llndk.libraries." + vndk_version + ".txt"; const std::string vndkprivate_libraries_path = vndk_path + "/etc/vndkprivate.libraries." + vndk_version + ".txt"; Variables::AddValue("LLNDK_LIBRARIES_" + partition, GetPublicLibrariesString(llndk_libraries_path, vndkprivate_libraries_path)); Variables::AddValue("PRIVATE_LLNDK_LIBRARIES_" + partition, GetPrivateLibrariesString(llndk_libraries_path, vndkprivate_libraries_path)); } } void LoadLibraryListVariables(const std::string& root) { LoadVndkLibraryListVariables(root, GetVendorVndkVersion(), "VENDOR"); LoadVndkLibraryListVariables(root, GetProductVndkVersion(), "PRODUCT"); LoadLlndkLibraryListVariables(root, GetVendorVndkVersion(), "VENDOR"); LoadLlndkLibraryListVariables(root, GetProductVndkVersion(), "PRODUCT"); auto llndk_library_path = root + "/system/etc/llndk.libraries.txt"; auto sanitizer_library_path = root + "/system/etc/sanitizer.libraries.txt"; Variables::AddValue("SANITIZER_RUNTIME_LIBRARIES", GetLibrariesString(sanitizer_library_path)); Variables::AddValue( "SANITIZER_LIBRARIES_LLNDK", GetPrivateLibrariesString(sanitizer_library_path, llndk_library_path)); } } // namespace namespace android { namespace linkerconfig { namespace generator { void LoadVariables(const std::string& root) { LoadVndkVersionVariable(); LoadPartitionPathVariables(root); LoadLibraryListVariables(root); } } // namespace generator } // namespace linkerconfig } // namespace android