diff options
Diffstat (limited to 'libarttools/tools/tools.cc')
| -rw-r--r-- | libarttools/tools/tools.cc | 121 |
1 files changed, 119 insertions, 2 deletions
diff --git a/libarttools/tools/tools.cc b/libarttools/tools/tools.cc index a3a91e81e2..3d5301ad7f 100644 --- a/libarttools/tools/tools.cc +++ b/libarttools/tools/tools.cc @@ -16,12 +16,129 @@ #include "tools.h" +#include <errno.h> +#include <fnmatch.h> + +#include <algorithm> +#include <filesystem> +#include <functional> +#include <string> +#include <string_view> +#include <system_error> +#include <vector> + +#include "android-base/logging.h" +#include "fmt/format.h" + namespace art { namespace tools { -std::string getMsg() { - return "hello world!"; +namespace { + +using ::std::placeholders::_1; + +using ::fmt::literals::operator""_format; // NOLINT + +// Returns true if `path_prefix` matches `pattern` or can be a prefix of a path that matches +// `pattern` (i.e., `path_prefix` represents a directory that may contain a file whose path matches +// `pattern`). +bool PartialMatch(const std::filesystem::path& pattern, const std::filesystem::path& path_prefix) { + for (std::filesystem::path::const_iterator pattern_it = pattern.begin(), + path_prefix_it = path_prefix.begin(); + ; // NOLINT + pattern_it++, path_prefix_it++) { + if (path_prefix_it == path_prefix.end()) { + return true; + } + if (pattern_it == pattern.end()) { + return false; + } + if (*pattern_it == "**") { + return true; + } + if (fnmatch(pattern_it->c_str(), path_prefix_it->c_str(), /*flags=*/0) != 0) { + return false; + } + } } +bool FullMatchRecursive(const std::filesystem::path& pattern, + std::filesystem::path::const_iterator pattern_it, + const std::filesystem::path& path, + std::filesystem::path::const_iterator path_it, + bool double_asterisk_visited = false) { + if (pattern_it == pattern.end() && path_it == path.end()) { + return true; + } + if (pattern_it == pattern.end()) { + return false; + } + if (*pattern_it == "**") { + DCHECK(!double_asterisk_visited); + std::filesystem::path::const_iterator next_pattern_it = pattern_it; + return FullMatchRecursive( + pattern, ++next_pattern_it, path, path_it, /*double_asterisk_visited=*/true) || + (path_it != path.end() && FullMatchRecursive(pattern, pattern_it, path, ++path_it)); + } + if (path_it == path.end()) { + return false; + } + if (fnmatch(pattern_it->c_str(), path_it->c_str(), /*flags=*/0) != 0) { + return false; + } + return FullMatchRecursive(pattern, ++pattern_it, path, ++path_it); } + +// Returns true if `path` fully matches `pattern`. +bool FullMatch(const std::filesystem::path& pattern, const std::filesystem::path& path) { + return FullMatchRecursive(pattern, pattern.begin(), path, path.begin()); } + +void MatchGlobRecursive(const std::vector<std::filesystem::path>& patterns, + const std::filesystem::path& root_dir, + /*out*/ std::vector<std::string>* results) { + std::error_code ec; + for (auto it = std::filesystem::recursive_directory_iterator( + root_dir, std::filesystem::directory_options::skip_permission_denied, ec); + !ec && it != std::filesystem::end(it); + it.increment(ec)) { + const std::filesystem::directory_entry& entry = *it; + if (std::none_of(patterns.begin(), patterns.end(), std::bind(PartialMatch, _1, entry.path()))) { + // Avoid unnecessary I/O and SELinux denials. + it.disable_recursion_pending(); + continue; + } + std::error_code ec2; + if (entry.is_regular_file(ec2) && + std::any_of(patterns.begin(), patterns.end(), std::bind(FullMatch, _1, entry.path()))) { + results->push_back(entry.path()); + } + if (ec2) { + // It's expected that we don't have permission to stat some dirs/files, and we don't care + // about them. + if (ec2.value() != EACCES) { + LOG(ERROR) << "Unable to lstat '{}': {}"_format(entry.path().string(), ec2.message()); + } + continue; + } + } + if (ec) { + LOG(ERROR) << "Unable to walk through '{}': {}"_format(root_dir.string(), ec.message()); + } +} + +} // namespace + +std::vector<std::string> Glob(const std::vector<std::string>& patterns, std::string_view root_dir) { + std::vector<std::filesystem::path> parsed_patterns; + parsed_patterns.reserve(patterns.size()); + for (std::string_view pattern : patterns) { + parsed_patterns.emplace_back(pattern); + } + std::vector<std::string> results; + MatchGlobRecursive(parsed_patterns, root_dir, &results); + return results; +} + +} // namespace tools +} // namespace art |