/*
** Copyright 2016, 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 "otapreopt_parameters.h"
#include <android-base/logging.h>
#include "dexopt.h"
#include "installd_constants.h"
#include "otapreopt_utils.h"
#ifndef LOG_TAG
#define LOG_TAG "otapreopt"
#endif
namespace android {
namespace installd {
static bool ParseBool(const char* in) {
if (strcmp(in, "true") == 0) {
return true;
}
return false;
}
static const char* ParseNull(const char* arg) {
return (strcmp(arg, "!") == 0) ? nullptr : arg;
}
static bool ParseUInt(const char* in, uint32_t* out) {
char* end;
long long int result = strtoll(in, &end, 0);
if (in == end || *end != '\0') {
return false;
}
if (result < std::numeric_limits<uint32_t>::min() ||
std::numeric_limits<uint32_t>::max() < result) {
return false;
}
*out = static_cast<uint32_t>(result);
return true;
}
bool OTAPreoptParameters::ReadArguments(int argc, const char** argv) {
// Expected command line:
// target-slot [version] dexopt {DEXOPT_PARAMETERS}
const char* target_slot_arg = argv[1];
if (target_slot_arg == nullptr) {
LOG(ERROR) << "Missing parameters";
return false;
}
// Sanitize value. Only allow (a-zA-Z0-9_)+.
target_slot = target_slot_arg;
if (!ValidateTargetSlotSuffix(target_slot)) {
LOG(ERROR) << "Target slot suffix not legal: " << target_slot;
return false;
}
// Check for version or "dexopt" next.
if (argv[2] == nullptr) {
LOG(ERROR) << "Missing parameters";
return false;
}
if (std::string("dexopt").compare(argv[2]) == 0) {
// This is version 1 (N) or pre-versioning version 2.
constexpr int kV2ArgCount = 1 // "otapreopt"
+ 1 // slot
+ 1 // "dexopt"
+ 1 // apk_path
+ 1 // uid
+ 1 // pkg
+ 1 // isa
+ 1 // dexopt_needed
+ 1 // oat_dir
+ 1 // dexopt_flags
+ 1 // filter
+ 1 // volume
+ 1 // libs
+ 1; // seinfo
if (argc == kV2ArgCount) {
return ReadArgumentsPostV1(2, argv, false);
} else {
return ReadArgumentsV1(argv);
}
}
uint32_t version;
if (!ParseUInt(argv[2], &version)) {
LOG(ERROR) << "Could not parse version: " << argv[2];
return false;
}
return ReadArgumentsPostV1(version, argv, true);
}
static int ReplaceMask(int input, int old_mask, int new_mask) {
return (input & old_mask) != 0 ? new_mask : 0;
}
void OTAPreoptParameters::SetDefaultsForPostV1Arguments() {
// Set se_info to null. It is only relevant for secondary dex files, which we won't
// receive from a v1 A side.
se_info = nullptr;
// Set downgrade to false. It is only relevant when downgrading compiler
// filter, which is not the case during ota.
downgrade = false;
// Set target_sdk_version to 0, ie the platform SDK version. This is
// conservative and may force some classes to verify at runtime.
target_sdk_version = 0;
// Set the profile name to the primary apk profile.
profile_name = "primary.prof";
// By default we don't have a dex metadata file.
dex_metadata_path = nullptr;
// The compilation reason is ab-ota (match the system property pm.dexopt.ab-ota)
compilation_reason = "ab-ota";
// Flag is enabled by default for A/B otas.
dexopt_flags = DEXOPT_GENERATE_COMPACT_DEX;
}
bool OTAPreoptParameters::ReadArgumentsV1(const char** argv) {
// Check for "dexopt".
if (argv[2] == nullptr) {
LOG(ERROR) << "Missing parameters";
return false;
}
if (std::string("dexopt").compare(argv[2]) != 0) {
LOG(ERROR) << "Expected \"dexopt\" but found: " << argv[2];
return false;
}
SetDefaultsForPostV1Arguments();
size_t param_index = 0;
for (;; ++param_index) {
const char* param = argv[3 + param_index];
if (param == nullptr) {
break;
}
switch (param_index) {
case 0:
apk_path = param;
break;
case 1:
uid = atoi(param);
break;
case 2:
pkgName = param;
break;
case 3:
instruction_set = param;
break;
case 4: {
// Version 1 had:
// DEXOPT_DEX2OAT_NEEDED = 1
// DEXOPT_PATCHOAT_NEEDED = 2
// DEXOPT_SELF_PATCHOAT_NEEDED = 3
// We will simply use DEX2OAT_FROM_SCRATCH.
dexopt_needed = DEX2OAT_FROM_SCRATCH;
break;
}
case 5:
oat_dir = param;
break;
case 6: {
// Version 1 had:
constexpr int OLD_DEXOPT_PUBLIC = 1 << 1;
// Note: DEXOPT_SAFEMODE has been removed.
// constexpr int OLD_DEXOPT_SAFEMODE = 1 << 2;
constexpr int OLD_DEXOPT_DEBUGGABLE = 1 << 3;
constexpr int OLD_DEXOPT_BOOTCOMPLETE = 1 << 4;
constexpr int OLD_DEXOPT_PROFILE_GUIDED = 1 << 5;
constexpr int OLD_DEXOPT_OTA = 1 << 6;
static_assert(DEXOPT_GENERATE_COMPACT_DEX > OLD_DEXOPT_OTA, "must not overlap");
int input = atoi(param);
dexopt_flags |=
ReplaceMask(input, OLD_DEXOPT_PUBLIC, DEXOPT_PUBLIC) |
ReplaceMask(input, OLD_DEXOPT_DEBUGGABLE, DEXOPT_DEBUGGABLE) |
ReplaceMask(input, OLD_DEXOPT_BOOTCOMPLETE, DEXOPT_BOOTCOMPLETE) |
ReplaceMask(input, OLD_DEXOPT_PROFILE_GUIDED, DEXOPT_PROFILE_GUIDED) |
ReplaceMask(input, OLD_DEXOPT_OTA, 0);
break;
}
case 7:
compiler_filter = param;
break;
case 8:
volume_uuid = ParseNull(param);
break;
case 9:
shared_libraries = ParseNull(param);
break;
default:
LOG(ERROR) << "Too many arguments, got " << param;
return false;
}
}
if (param_index != 10) {
LOG(ERROR) << "Not enough parameters";
return false;
}
return true;
}
bool OTAPreoptParameters::ReadArgumentsPostV1(uint32_t version, const char** argv, bool versioned) {
size_t num_args_expected = 0;
switch (version) {
case 2: num_args_expected = 11; break;
case 3: num_args_expected = 12; break;
case 4: num_args_expected = 13; break;
case 5: num_args_expected = 14; break;
case 6: num_args_expected = 15; break;
case 7:
// Version 8 adds a new dexopt flag: DEXOPT_GENERATE_COMPACT_DEX
case 8: num_args_expected = 16; break;
// Version 9 adds a new dexopt flag: DEXOPT_GENERATE_APP_IMAGE
case 9: num_args_expected = 16; break;
default:
LOG(ERROR) << "Don't know how to read arguments for version " << version;
return false;
}
size_t dexopt_index = versioned ? 3 : 2;
// Check for "dexopt".
if (argv[dexopt_index] == nullptr) {
LOG(ERROR) << "Missing parameters";
return false;
}
if (std::string("dexopt").compare(argv[dexopt_index]) != 0) {
LOG(ERROR) << "Expected \"dexopt\" but found: " << argv[dexopt_index];
return false;
}
// Validate the number of arguments.
size_t num_args_actual = 0;
while (argv[dexopt_index + 1 + num_args_actual] != nullptr) {
num_args_actual++;
}
if (num_args_actual != num_args_expected) {
LOG(ERROR) << "Invalid number of arguments. expected="
<< num_args_expected << " actual=" << num_args_actual;
return false;
}
// The number of arguments is OK.
// Configure the default values for the parameters that were added after V1.
// The default values will be overwritten in case they are passed as arguments.
SetDefaultsForPostV1Arguments();
for (size_t param_index = 0; param_index < num_args_actual; ++param_index) {
const char* param = argv[dexopt_index + 1 + param_index];
switch (param_index) {
case 0:
apk_path = param;
break;
case 1:
uid = atoi(param);
break;
case 2:
pkgName = param;
break;
case 3:
instruction_set = param;
break;
case 4:
dexopt_needed = atoi(param);
break;
case 5:
oat_dir = param;
break;
case 6:
dexopt_flags = atoi(param);
// Add CompactDex generation flag for versions less than 8 since it wasn't passed
// from the package manager. Only conditionally set the flag here so that it can
// be fully controlled by the package manager.
dexopt_flags |= (version < 8) ? DEXOPT_GENERATE_COMPACT_DEX : 0u;
break;
case 7:
compiler_filter = param;
break;
case 8:
volume_uuid = ParseNull(param);
break;
case 9:
shared_libraries = ParseNull(param);
break;
case 10:
se_info = ParseNull(param);
break;
case 11:
downgrade = ParseBool(param);
break;
case 12:
target_sdk_version = atoi(param);
break;
case 13:
profile_name = ParseNull(param);
break;
case 14:
dex_metadata_path = ParseNull(param);
break;
case 15:
compilation_reason = ParseNull(param);
break;
default:
LOG(FATAL) << "Should not get here. Did you call ReadArguments "
<< "with the right expectation? index=" << param_index
<< " num_args=" << num_args_actual;
return false;
}
}
return true;
}
} // namespace installd
} // namespace android