/* * Copyright (C) 2020 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. */ #define LOG_TAG "installd" #include "run_dex2oat.h" #include #include #include #include #include #include #include #include #include #include #include #include "unique_file.h" using android::base::Basename; using android::base::StringPrintf; namespace android { namespace installd { namespace { // Should minidebug info be included in compiled artifacts? Even if this value is // "true," usage might still be conditional to other constraints, e.g., system // property overrides. static constexpr bool kEnableMinidebugInfo = true; static constexpr const char* kMinidebugInfoSystemProperty = "dalvik.vm.dex2oat-minidebuginfo"; static constexpr bool kMinidebugInfoSystemPropertyDefault = false; static constexpr const char* kMinidebugDex2oatFlag = "--generate-mini-debug-info"; static constexpr const char* kDisableCompactDexFlag = "--compact-dex-level=none"; std::vector SplitBySpaces(const std::string& str) { if (str.empty()) { return {}; } return android::base::Split(str, " "); } } // namespace RunDex2Oat::RunDex2Oat(const char* dex2oat_bin, ExecVHelper* execv_helper) : dex2oat_bin_(dex2oat_bin), execv_helper_(execv_helper) {} void RunDex2Oat::Initialize(const UniqueFile& output_oat, const UniqueFile& output_vdex, const UniqueFile& output_image, const UniqueFile& input_dex, const UniqueFile& input_vdex, const UniqueFile& dex_metadata, const UniqueFile& profile, const char* class_loader_context, const std::string& class_loader_context_fds, int swap_fd, const char* instruction_set, const char* compiler_filter, bool debuggable, bool post_bootcomplete, bool for_restore, int target_sdk_version, bool enable_hidden_api_checks, bool generate_compact_dex, bool use_jitzygote, bool background_job_compile, const char* compilation_reason) { PrepareBootImageFlags(use_jitzygote); PrepareInputFileFlags(output_oat, output_vdex, output_image, input_dex, input_vdex, dex_metadata, profile, swap_fd, class_loader_context, class_loader_context_fds); PrepareCompilerConfigFlags(input_vdex, output_vdex, instruction_set, compiler_filter, debuggable, target_sdk_version, enable_hidden_api_checks, generate_compact_dex, compilation_reason); PrepareCompilerRuntimeAndPerfConfigFlags(post_bootcomplete, for_restore, background_job_compile); const std::string dex2oat_flags = GetProperty("dalvik.vm.dex2oat-flags", ""); std::vector dex2oat_flags_args = SplitBySpaces(dex2oat_flags); ALOGV("dalvik.vm.dex2oat-flags=%s\n", dex2oat_flags.c_str()); // Do not add args after dex2oat_flags, they should override others for debugging. for (auto it = dex2oat_flags_args.begin(); it != dex2oat_flags_args.end(); ++it) { AddArg(*it); } execv_helper_->PrepareArgs(dex2oat_bin_); } RunDex2Oat::~RunDex2Oat() {} void RunDex2Oat::PrepareBootImageFlags(bool use_jitzygote) { if (use_jitzygote) { // Don't pass a boot image because JIT Zygote should decide which image to use. Typically, // it does not use any boot image on disk. AddArg("--force-jit-zygote"); } else { AddArg(MapPropertyToArg("dalvik.vm.boot-image", "--boot-image=%s")); } } void RunDex2Oat::PrepareInputFileFlags(const UniqueFile& output_oat, const UniqueFile& output_vdex, const UniqueFile& output_image, const UniqueFile& input_dex, const UniqueFile& input_vdex, const UniqueFile& dex_metadata, const UniqueFile& profile, int swap_fd, const char* class_loader_context, const std::string& class_loader_context_fds) { std::string input_basename = Basename(input_dex.path()); LOG(VERBOSE) << "Running " << dex2oat_bin_ << " in=" << input_basename << " out=" << output_oat.path(); AddArg(StringPrintf("--zip-fd=%d", input_dex.fd())); AddArg(StringPrintf("--zip-location=%s", input_basename.c_str())); AddArg(StringPrintf("--oat-fd=%d", output_oat.fd())); AddArg(StringPrintf("--oat-location=%s", output_oat.path().c_str())); AddArg(StringPrintf("--input-vdex-fd=%d", input_vdex.fd())); AddArg(StringPrintf("--output-vdex-fd=%d", output_vdex.fd())); if (output_image.fd() >= 0) { AddArg(StringPrintf("--app-image-fd=%d", output_image.fd())); AddArg(MapPropertyToArg("dalvik.vm.appimageformat", "--image-format=%s")); } if (dex_metadata.fd() > -1) { AddArg("--dm-fd=" + std::to_string(dex_metadata.fd())); } if (profile.fd() != -1) { AddArg(StringPrintf("--profile-file-fd=%d", profile.fd())); } if (swap_fd >= 0) { AddArg(StringPrintf("--swap-fd=%d", swap_fd)); } // Get the directory of the apk to pass as a base classpath directory. { std::string apk_dir(input_dex.path()); size_t dir_index = apk_dir.rfind('/'); if (dir_index != std::string::npos) { apk_dir = apk_dir.substr(0, dir_index); AddArg(StringPrintf("--classpath-dir=%s", apk_dir.c_str())); } } if (class_loader_context != nullptr) { AddArg(StringPrintf("--class-loader-context=%s", class_loader_context)); if (!class_loader_context_fds.empty()) { AddArg(StringPrintf("--class-loader-context-fds=%s", class_loader_context_fds.c_str())); } } } void RunDex2Oat::PrepareCompilerConfigFlags(const UniqueFile& input_vdex, const UniqueFile& output_vdex, const char* instruction_set, const char* compiler_filter, bool debuggable, int target_sdk_version, bool enable_hidden_api_checks, bool generate_compact_dex, const char* compilation_reason) { // Disable cdex if update input vdex is true since this combination of options is not // supported. const bool disable_cdex = !generate_compact_dex || (input_vdex.fd() == output_vdex.fd()); if (disable_cdex) { AddArg(kDisableCompactDexFlag); } // ISA related { AddArg(StringPrintf("--instruction-set=%s", instruction_set)); const std::string dex2oat_isa_features_key = StringPrintf("dalvik.vm.isa.%s.features", instruction_set); std::string instruction_set_features_arg = MapPropertyToArg(dex2oat_isa_features_key, "--instruction-set-features=%s"); AddArg(instruction_set_features_arg); const std::string dex2oat_isa_variant_key = StringPrintf("dalvik.vm.isa.%s.variant", instruction_set); std::string instruction_set_variant_arg = MapPropertyToArg(dex2oat_isa_variant_key, "--instruction-set-variant=%s"); AddArg(instruction_set_variant_arg); } // Compute compiler filter. if (compiler_filter != nullptr) { AddArg(StringPrintf("--compiler-filter=%s", compiler_filter)); } else { AddArg(MapPropertyToArg("dalvik.vm.dex2oat-filter", "--compiler-filter=%s")); } if (compilation_reason != nullptr) { AddArg(std::string("--compilation-reason=") + compilation_reason); } AddArg(MapPropertyToArg("dalvik.vm.dex2oat-max-image-block-size", "--max-image-block-size=%s")); AddArg(MapPropertyToArg("dalvik.vm.dex2oat-very-large", "--very-large-app-threshold=%s")); std::string resolve_startup_string_arg = MapPropertyToArg( "persist.device_config.runtime.dex2oat_resolve_startup_strings", "--resolve-startup-const-strings=%s"); if (resolve_startup_string_arg.empty()) { // If empty, fall back to system property. resolve_startup_string_arg = MapPropertyToArg("dalvik.vm.dex2oat-resolve-startup-strings", "--resolve-startup-const-strings=%s"); } AddArg(resolve_startup_string_arg); // Debug related { // Check whether all apps should be compiled debuggable. if (!debuggable) { debuggable = GetProperty("dalvik.vm.always_debuggable", "") == "1"; } if (debuggable) { AddArg("--debuggable"); } const bool generate_debug_info = GetBoolProperty("debug.generate-debug-info", false); if (generate_debug_info) { AddArg("--generate-debug-info"); } { bool generate_minidebug_info = kEnableMinidebugInfo && GetBoolProperty(kMinidebugInfoSystemProperty, kMinidebugInfoSystemPropertyDefault); if (generate_minidebug_info) { AddArg(kMinidebugDex2oatFlag); } } } // On-device signing related. odsign sets the system property odsign.verification.success if // AOT artifacts have the expected signatures. const bool trust_art_apex_data_files = GetBoolProperty("odsign.verification.success", false); if (!trust_art_apex_data_files) { AddRuntimeArg("-Xdeny-art-apex-data-files"); } if (target_sdk_version != 0) { AddRuntimeArg(StringPrintf("-Xtarget-sdk-version:%d", target_sdk_version)); } if (enable_hidden_api_checks) { AddRuntimeArg("-Xhidden-api-policy:enabled"); } } void RunDex2Oat::PrepareCompilerRuntimeAndPerfConfigFlags(bool post_bootcomplete, bool for_restore, bool background_job_compile) { // CPU set { std::string cpu_set_format = "--cpu-set=%s"; std::string dex2oat_cpu_set_arg = post_bootcomplete ? (for_restore ? MapPropertyToArgWithBackup( "dalvik.vm.restore-dex2oat-cpu-set", "dalvik.vm.dex2oat-cpu-set", cpu_set_format) : (background_job_compile ? MapPropertyToArgWithBackup( "dalvik.vm.background-dex2oat-cpu-set", "dalvik.vm.dex2oat-cpu-set", cpu_set_format) : MapPropertyToArg("dalvik.vm.dex2oat-cpu-set", cpu_set_format))) : MapPropertyToArg("dalvik.vm.boot-dex2oat-cpu-set", cpu_set_format); AddArg(dex2oat_cpu_set_arg); } // Number of threads { std::string threads_format = "-j%s"; std::string dex2oat_threads_arg = post_bootcomplete ? (for_restore ? MapPropertyToArgWithBackup( "dalvik.vm.restore-dex2oat-threads", "dalvik.vm.dex2oat-threads", threads_format) : (background_job_compile ? MapPropertyToArgWithBackup( "dalvik.vm.background-dex2oat-threads", "dalvik.vm.dex2oat-threads", threads_format) : MapPropertyToArg("dalvik.vm.dex2oat-threads", threads_format))) : MapPropertyToArg("dalvik.vm.boot-dex2oat-threads", threads_format); AddArg(dex2oat_threads_arg); } AddRuntimeArg(MapPropertyToArg("dalvik.vm.dex2oat-Xms", "-Xms%s")); AddRuntimeArg(MapPropertyToArg("dalvik.vm.dex2oat-Xmx", "-Xmx%s")); // Enable compiling dex files in isolation on low ram devices. // It takes longer but reduces the memory footprint. if (GetBoolProperty("ro.config.low_ram", false)) { AddArg("--compile-individually"); } } void RunDex2Oat::Exec(int exit_code) { execv_helper_->Exec(exit_code); } void RunDex2Oat::AddArg(const std::string& arg) { execv_helper_->AddArg(arg); } void RunDex2Oat::AddRuntimeArg(const std::string& arg) { execv_helper_->AddRuntimeArg(arg); } std::string RunDex2Oat::GetProperty(const std::string& key, const std::string& default_value) { return android::base::GetProperty(key, default_value); } bool RunDex2Oat::GetBoolProperty(const std::string& key, bool default_value) { return android::base::GetBoolProperty(key, default_value); } std::string RunDex2Oat::MapPropertyToArg(const std::string& property, const std::string& format, const std::string& default_value) { std::string prop = GetProperty(property, default_value); if (!prop.empty()) { return StringPrintf(format.c_str(), prop.c_str()); } return ""; } std::string RunDex2Oat::MapPropertyToArgWithBackup( const std::string& property, const std::string& backupProperty, const std::string& format, const std::string& default_value) { std::string value = GetProperty(property, default_value); if (!value.empty()) { return StringPrintf(format.c_str(), value.c_str()); } return MapPropertyToArg(backupProperty, format, default_value); } } // namespace installd } // namespace android