You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
387 lines
15 KiB
387 lines
15 KiB
/*
|
|
* 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 <memory>
|
|
#include <string>
|
|
#include <vector>
|
|
|
|
#include <android-base/file.h>
|
|
#include <android-base/logging.h>
|
|
#include <android-base/properties.h>
|
|
#include <android-base/scopeguard.h>
|
|
#include <android-base/stringprintf.h>
|
|
#include <android-base/strings.h>
|
|
#include <log/log.h>
|
|
#include <server_configurable_flags/get_flags.h>
|
|
|
|
#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";
|
|
|
|
// Location of the JIT Zygote image.
|
|
static const char* kJitZygoteImage =
|
|
"boot.art:/nonx/boot-framework.art!/system/etc/boot-image.prof";
|
|
|
|
std::vector<std::string> 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_image,
|
|
const char* compilation_reason) {
|
|
PrepareBootImageFlags(use_jitzygote_image);
|
|
|
|
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);
|
|
|
|
const std::string dex2oat_flags = GetProperty("dalvik.vm.dex2oat-flags", "");
|
|
std::vector<std::string> 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_image) {
|
|
std::string boot_image;
|
|
if (use_jitzygote_image) {
|
|
boot_image = StringPrintf("--boot-image=%s", kJitZygoteImage);
|
|
} else {
|
|
boot_image = MapPropertyToArg("dalvik.vm.boot-image", "--boot-image=%s");
|
|
}
|
|
AddArg(boot_image);
|
|
}
|
|
|
|
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.
|
|
{
|
|
std::string dex2oat_compiler_filter_arg;
|
|
{
|
|
// If we are booting without the real /data, don't spend time compiling.
|
|
std::string vold_decrypt = GetProperty("vold.decrypt", "");
|
|
bool skip_compilation = vold_decrypt == "trigger_restart_min_framework" ||
|
|
vold_decrypt == "1";
|
|
|
|
bool have_dex2oat_relocation_skip_flag = false;
|
|
if (skip_compilation) {
|
|
dex2oat_compiler_filter_arg = "--compiler-filter=extract";
|
|
have_dex2oat_relocation_skip_flag = true;
|
|
} else if (compiler_filter != nullptr) {
|
|
dex2oat_compiler_filter_arg = StringPrintf("--compiler-filter=%s",
|
|
compiler_filter);
|
|
}
|
|
if (have_dex2oat_relocation_skip_flag) {
|
|
AddRuntimeArg("-Xnorelocate");
|
|
}
|
|
}
|
|
|
|
if (dex2oat_compiler_filter_arg.empty()) {
|
|
dex2oat_compiler_filter_arg = MapPropertyToArg("dalvik.vm.dex2oat-filter",
|
|
"--compiler-filter=%s");
|
|
}
|
|
AddArg(dex2oat_compiler_filter_arg);
|
|
|
|
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) {
|
|
// 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)
|
|
: 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)
|
|
: 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
|