blob: 6ec207ededfb929f72810ddaea968dc602f6e161 [file]
/*
* Copyright (C) 2022 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 "testing.h"
#include <string>
#include <vector>
#include "android-base/file.h"
#include "android-base/stringprintf.h"
#include "android-base/strings.h"
#include "base/file_utils.h"
#include "base/globals.h"
#include "base/os.h"
namespace art {
namespace testing {
std::string GetAndroidBuildTop() {
CHECK(IsHost());
std::string android_build_top;
// Look at how we were invoked to find the expected directory.
std::string argv;
if (android::base::ReadFileToString("/proc/self/cmdline", &argv)) {
// /proc/self/cmdline is the programs 'argv' with elements delimited by '\0'.
std::filesystem::path path(argv.substr(0, argv.find('\0')));
path = std::filesystem::absolute(path);
// Walk up until we find the one of the well-known directories.
for (; path.parent_path() != path; path = path.parent_path()) {
// We are running tests from out/host/linux-x86 on developer machine.
if (path.filename() == std::filesystem::path("linux-x86")) {
android_build_top = path.parent_path().parent_path().parent_path();
break;
}
// We are running tests from testcases (extracted from zip) on tradefed.
// The first path is for remote runs and the second path for local runs.
if (path.filename() == std::filesystem::path("testcases") ||
path.filename().string().starts_with("host_testcases")) {
android_build_top = path.append("art_common");
break;
}
}
}
CHECK(!android_build_top.empty());
// Check that the expected directory matches the environment variable.
const char* android_build_top_from_env = getenv("ANDROID_BUILD_TOP");
android_build_top = std::filesystem::path(android_build_top).string();
CHECK(!android_build_top.empty());
if (android_build_top_from_env != nullptr) {
if (std::filesystem::weakly_canonical(android_build_top).string() !=
std::filesystem::weakly_canonical(android_build_top_from_env).string()) {
android_build_top = android_build_top_from_env;
}
} else {
setenv("ANDROID_BUILD_TOP", android_build_top.c_str(), /*overwrite=*/0);
}
if (android_build_top.back() != '/') {
android_build_top += '/';
}
return android_build_top;
}
std::string GetAndroidHostOut() {
CHECK(IsHost());
// Check that the expected directory matches the environment variable.
// ANDROID_HOST_OUT is set by envsetup or unset and is the full path to host binaries/libs
const char* android_host_out_from_env = getenv("ANDROID_HOST_OUT");
// OUT_DIR is a user-settable ENV_VAR that controls where soong puts build artifacts. It can
// either be relative to ANDROID_BUILD_TOP or a concrete path.
const char* android_out_dir = getenv("OUT_DIR");
// Take account of OUT_DIR setting.
if (android_out_dir == nullptr) {
android_out_dir = "out";
}
std::string android_host_out;
if (android_out_dir[0] == '/') {
android_host_out = (std::filesystem::path(android_out_dir) / "host" / "linux-x86").string();
} else {
android_host_out =
(std::filesystem::path(GetAndroidBuildTop()) / android_out_dir / "host" / "linux-x86")
.string();
}
std::filesystem::path expected(android_host_out);
if (android_host_out_from_env != nullptr) {
std::filesystem::path from_env(std::filesystem::weakly_canonical(android_host_out_from_env));
if (std::filesystem::weakly_canonical(expected).string() != from_env.string()) {
LOG(WARNING) << "Execution path (" << expected << ") not below ANDROID_HOST_OUT (" << from_env
<< ")! Using env-var.";
expected = from_env;
}
} else {
setenv("ANDROID_HOST_OUT", android_host_out.c_str(), /*overwrite=*/0);
}
return expected.string();
}
std::string GetHostBootClasspathInstallRoot() {
CHECK(IsHost());
std::string build_install_root = GetAndroidHostOut() + "/testcases/art_common/out/host/linux-x86";
// Look for the `apex` subdirectory as a discriminator to check the location.
if (OS::DirectoryExists((build_install_root + "/apex").c_str())) {
// This is the path where "m art-host-tests" installs support files for host
// tests, so use it when the tests are run in a build tree (which is the
// case when testing locally).
return build_install_root;
}
if (OS::DirectoryExists((GetAndroidRoot() + "/apex").c_str())) {
// This is the location for host tests in CI when the files are unzipped
// from art-host-tests.zip.
return GetAndroidRoot();
}
LOG(ERROR) << "Neither location has a boot classpath (forgot \"m art-host-tests\"?): "
<< build_install_root << " or " << GetAndroidRoot();
return "<no boot classpath found>";
}
static std::string GetDexFileName(const std::string& jar_prefix, const std::string& prefix) {
const char* apexPath =
(jar_prefix == "conscrypt") ?
kAndroidConscryptApexDefaultPath :
(jar_prefix == "core-icu4j" ? kAndroidI18nApexDefaultPath : kAndroidArtApexDefaultPath);
return android::base::StringPrintf(
"%s%s/javalib/%s.jar", prefix.c_str(), apexPath, jar_prefix.c_str());
}
static std::vector<std::string> GetPrefixedDexFileNames(const std::string& prefix,
const std::vector<std::string>& modules) {
std::vector<std::string> result;
result.reserve(modules.size());
for (const std::string& module : modules) {
result.push_back(GetDexFileName(module, prefix));
}
return result;
}
std::vector<std::string> GetLibCoreModuleNames(bool core_only) {
// Note: This must start with the CORE_IMG_JARS in Android.common_path.mk because that's what we
// use for compiling the boot.art image. It may contain additional modules from TEST_CORE_JARS.
// CORE_IMG_JARS modules.
std::vector<std::string> modules{
"core-oj",
"core-libart",
"okhttp",
"bouncycastle",
"apache-xml",
};
// Additional modules.
if (!core_only) {
modules.push_back("core-icu4j");
modules.push_back("conscrypt");
}
return modules;
}
std::vector<std::string> GetLibCoreDexFileNames(const std::vector<std::string>& modules) {
return GetPrefixedDexFileNames(kIsTargetBuild ? "" : GetHostBootClasspathInstallRoot(), modules);
}
std::vector<std::string> GetLibCoreDexFileNames(const std::string& prefix, bool core_only) {
std::vector<std::string> modules = GetLibCoreModuleNames(core_only);
return GetPrefixedDexFileNames(prefix, modules);
}
std::vector<std::string> GetLibCoreDexLocations(const std::vector<std::string>& modules) {
std::string prefix = "";
if (IsHost()) {
std::string android_root = GetAndroidRoot();
std::string build_top = GetAndroidBuildTop();
CHECK(android_root.starts_with(build_top))
<< " android_root=" << android_root << " build_top=" << build_top;
prefix = android_root.substr(build_top.size());
}
return GetPrefixedDexFileNames(prefix, modules);
}
std::vector<std::string> GetLibCoreDexLocations(bool core_only) {
std::vector<std::string> modules = GetLibCoreModuleNames(core_only);
return GetLibCoreDexLocations(modules);
}
std::string GetClassPathOption(const char* option, const std::vector<std::string>& class_path) {
return option + android::base::Join(class_path, ':');
}
} // namespace testing
} // namespace art