diff options
Diffstat (limited to 'artd/path_utils.cc')
| -rw-r--r-- | artd/path_utils.cc | 280 |
1 files changed, 280 insertions, 0 deletions
diff --git a/artd/path_utils.cc b/artd/path_utils.cc new file mode 100644 index 0000000000..6ef0eafb53 --- /dev/null +++ b/artd/path_utils.cc @@ -0,0 +1,280 @@ +/* + * 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 "path_utils.h" + +#include <filesystem> +#include <string> +#include <vector> + +#include "aidl/com/android/server/art/BnArtd.h" +#include "android-base/errors.h" +#include "android-base/result.h" +#include "android-base/strings.h" +#include "arch/instruction_set.h" +#include "base/file_utils.h" +#include "file_utils.h" +#include "fmt/format.h" +#include "oat_file_assistant.h" +#include "tools/tools.h" + +namespace art { +namespace artd { + +namespace { + +using ::aidl::com::android::server::art::ArtifactsPath; +using ::aidl::com::android::server::art::DexMetadataPath; +using ::aidl::com::android::server::art::ProfilePath; +using ::aidl::com::android::server::art::VdexPath; +using ::android::base::Error; +using ::android::base::Result; + +using ::fmt::literals::operator""_format; // NOLINT + +using PrebuiltProfilePath = ProfilePath::PrebuiltProfilePath; +using PrimaryCurProfilePath = ProfilePath::PrimaryCurProfilePath; +using PrimaryRefProfilePath = ProfilePath::PrimaryRefProfilePath; +using SecondaryCurProfilePath = ProfilePath::SecondaryCurProfilePath; +using SecondaryRefProfilePath = ProfilePath::SecondaryRefProfilePath; +using TmpProfilePath = ProfilePath::TmpProfilePath; +using WritableProfilePath = ProfilePath::WritableProfilePath; + +Result<void> ValidateAbsoluteNormalPath(const std::string& path_str) { + if (path_str.empty()) { + return Errorf("Path is empty"); + } + if (path_str.find('\0') != std::string::npos) { + return Errorf("Path '{}' has invalid character '\\0'", path_str); + } + std::filesystem::path path(path_str); + if (!path.is_absolute()) { + return Errorf("Path '{}' is not an absolute path", path_str); + } + if (path.lexically_normal() != path_str) { + return Errorf("Path '{}' is not in normal form", path_str); + } + return {}; +} + +Result<void> ValidatePathElementSubstring(const std::string& path_element_substring, + const std::string& name) { + if (path_element_substring.empty()) { + return Errorf("{} is empty", name); + } + if (path_element_substring.find('/') != std::string::npos) { + return Errorf("{} '{}' has invalid character '/'", name, path_element_substring); + } + if (path_element_substring.find('\0') != std::string::npos) { + return Errorf("{} '{}' has invalid character '\\0'", name, path_element_substring); + } + return {}; +} + +Result<void> ValidatePathElement(const std::string& path_element, const std::string& name) { + OR_RETURN(ValidatePathElementSubstring(path_element, name)); + if (path_element == "." || path_element == "..") { + return Errorf("Invalid {} '{}'", name, path_element); + } + return {}; +} + +Result<std::string> GetAndroidDataOrError() { + std::string error_msg; + std::string result = GetAndroidDataSafe(&error_msg); + if (!error_msg.empty()) { + return Error() << error_msg; + } + return result; +} + +Result<std::string> GetAndroidExpandOrError() { + std::string error_msg; + std::string result = GetAndroidExpandSafe(&error_msg); + if (!error_msg.empty()) { + return Error() << error_msg; + } + return result; +} + +Result<std::string> GetArtRootOrError() { + std::string error_msg; + std::string result = GetArtRootSafe(&error_msg); + if (!error_msg.empty()) { + return Error() << error_msg; + } + return result; +} + +} // namespace + +Result<std::vector<std::string>> ListManagedFiles() { + std::string android_data = OR_RETURN(GetAndroidDataOrError()); + std::string android_expand = OR_RETURN(GetAndroidExpandOrError()); + + // See `art::tools::Glob` for the syntax. + std::vector<std::string> patterns = { + // Profiles for primary dex files. + android_data + "/misc/profiles/**", + // Artifacts for primary dex files. + android_data + "/dalvik-cache/**", + }; + + for (const std::string& data_root : {android_data, android_expand + "/*"}) { + // Artifacts for primary dex files. + patterns.push_back(data_root + "/app/*/*/oat/**"); + // Profiles and artifacts for secondary dex files. Those files are in app data directories, so + // we use more granular patterns to avoid accidentally deleting apps' files. + for (const char* user_dir : {"/user", "/user_de"}) { + std::string secondary_oat_dir = data_root + user_dir + "/*/*/**/oat"; + for (const char* maybe_tmp_suffix : {"", ".*.tmp"}) { + patterns.push_back(secondary_oat_dir + "/*.prof" + maybe_tmp_suffix); + patterns.push_back(secondary_oat_dir + "/*/*.odex" + maybe_tmp_suffix); + patterns.push_back(secondary_oat_dir + "/*/*.vdex" + maybe_tmp_suffix); + patterns.push_back(secondary_oat_dir + "/*/*.art" + maybe_tmp_suffix); + } + } + } + + return tools::Glob(patterns); +} + +Result<void> ValidateDexPath(const std::string& dex_path) { + OR_RETURN(ValidateAbsoluteNormalPath(dex_path)); + return {}; +} + +Result<std::string> BuildArtBinPath(const std::string& binary_name) { + return "{}/bin/{}"_format(OR_RETURN(GetArtRootOrError()), binary_name); +} + +Result<std::string> BuildOatPath(const ArtifactsPath& artifacts_path) { + OR_RETURN(ValidateDexPath(artifacts_path.dexPath)); + + InstructionSet isa = GetInstructionSetFromString(artifacts_path.isa.c_str()); + if (isa == InstructionSet::kNone) { + return Errorf("Instruction set '{}' is invalid", artifacts_path.isa); + } + + std::string error_msg; + std::string path; + if (artifacts_path.isInDalvikCache) { + // Apps' OAT files are never in ART APEX data. + if (!OatFileAssistant::DexLocationToOatFilename( + artifacts_path.dexPath, isa, /*deny_art_apex_data_files=*/true, &path, &error_msg)) { + return Error() << error_msg; + } + return path; + } else { + if (!OatFileAssistant::DexLocationToOdexFilename( + artifacts_path.dexPath, isa, &path, &error_msg)) { + return Error() << error_msg; + } + return path; + } +} + +Result<std::string> BuildPrimaryRefProfilePath( + const PrimaryRefProfilePath& primary_ref_profile_path) { + OR_RETURN(ValidatePathElement(primary_ref_profile_path.packageName, "packageName")); + OR_RETURN(ValidatePathElementSubstring(primary_ref_profile_path.profileName, "profileName")); + return "{}/misc/profiles/ref/{}/{}.prof"_format(OR_RETURN(GetAndroidDataOrError()), + primary_ref_profile_path.packageName, + primary_ref_profile_path.profileName); +} + +Result<std::string> BuildPrebuiltProfilePath(const PrebuiltProfilePath& prebuilt_profile_path) { + OR_RETURN(ValidateDexPath(prebuilt_profile_path.dexPath)); + return prebuilt_profile_path.dexPath + ".prof"; +} + +Result<std::string> BuildPrimaryCurProfilePath( + const PrimaryCurProfilePath& primary_cur_profile_path) { + OR_RETURN(ValidatePathElement(primary_cur_profile_path.packageName, "packageName")); + OR_RETURN(ValidatePathElementSubstring(primary_cur_profile_path.profileName, "profileName")); + return "{}/misc/profiles/cur/{}/{}/{}.prof"_format(OR_RETURN(GetAndroidDataOrError()), + primary_cur_profile_path.userId, + primary_cur_profile_path.packageName, + primary_cur_profile_path.profileName); +} + +Result<std::string> BuildSecondaryRefProfilePath( + const SecondaryRefProfilePath& secondary_ref_profile_path) { + OR_RETURN(ValidateDexPath(secondary_ref_profile_path.dexPath)); + std::filesystem::path dex_path(secondary_ref_profile_path.dexPath); + return "{}/oat/{}.prof"_format(dex_path.parent_path().string(), dex_path.filename().string()); +} + +Result<std::string> BuildSecondaryCurProfilePath( + const SecondaryCurProfilePath& secondary_cur_profile_path) { + OR_RETURN(ValidateDexPath(secondary_cur_profile_path.dexPath)); + std::filesystem::path dex_path(secondary_cur_profile_path.dexPath); + return "{}/oat/{}.cur.prof"_format(dex_path.parent_path().string(), dex_path.filename().string()); +} + +Result<std::string> BuildFinalProfilePath(const TmpProfilePath& tmp_profile_path) { + const WritableProfilePath& final_path = tmp_profile_path.finalPath; + switch (final_path.getTag()) { + case WritableProfilePath::forPrimary: + return BuildPrimaryRefProfilePath(final_path.get<WritableProfilePath::forPrimary>()); + case WritableProfilePath::forSecondary: + return BuildSecondaryRefProfilePath(final_path.get<WritableProfilePath::forSecondary>()); + // No default. All cases should be explicitly handled, or the compilation will fail. + } + // This should never happen. Just in case we get a non-enumerator value. + LOG(FATAL) << "Unexpected writable profile path type {}"_format(final_path.getTag()); +} + +Result<std::string> BuildTmpProfilePath(const TmpProfilePath& tmp_profile_path) { + OR_RETURN(ValidatePathElementSubstring(tmp_profile_path.id, "id")); + return NewFile::BuildTempPath(OR_RETURN(BuildFinalProfilePath(tmp_profile_path)), + tmp_profile_path.id); +} + +Result<std::string> BuildDexMetadataPath(const DexMetadataPath& dex_metadata_path) { + OR_RETURN(ValidateDexPath(dex_metadata_path.dexPath)); + return ReplaceFileExtension(dex_metadata_path.dexPath, "dm"); +} + +Result<std::string> BuildProfileOrDmPath(const ProfilePath& profile_path) { + switch (profile_path.getTag()) { + case ProfilePath::primaryRefProfilePath: + return BuildPrimaryRefProfilePath(profile_path.get<ProfilePath::primaryRefProfilePath>()); + case ProfilePath::prebuiltProfilePath: + return BuildPrebuiltProfilePath(profile_path.get<ProfilePath::prebuiltProfilePath>()); + case ProfilePath::primaryCurProfilePath: + return BuildPrimaryCurProfilePath(profile_path.get<ProfilePath::primaryCurProfilePath>()); + case ProfilePath::secondaryRefProfilePath: + return BuildSecondaryRefProfilePath(profile_path.get<ProfilePath::secondaryRefProfilePath>()); + case ProfilePath::secondaryCurProfilePath: + return BuildSecondaryCurProfilePath(profile_path.get<ProfilePath::secondaryCurProfilePath>()); + case ProfilePath::tmpProfilePath: + return BuildTmpProfilePath(profile_path.get<ProfilePath::tmpProfilePath>()); + case ProfilePath::dexMetadataPath: + return BuildDexMetadataPath(profile_path.get<ProfilePath::dexMetadataPath>()); + // No default. All cases should be explicitly handled, or the compilation will fail. + } + // This should never happen. Just in case we get a non-enumerator value. + LOG(FATAL) << "Unexpected profile path type {}"_format(profile_path.getTag()); +} + +Result<std::string> BuildVdexPath(const VdexPath& vdex_path) { + DCHECK(vdex_path.getTag() == VdexPath::artifactsPath); + return OatPathToVdexPath(OR_RETURN(BuildOatPath(vdex_path.get<VdexPath::artifactsPath>()))); +} + +} // namespace artd +} // namespace art |