| 1 | //===-- RealpathPrefixes.cpp ----------------------------------------------===// |
| 2 | // |
| 3 | // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. |
| 4 | // See https://llvm.org/LICENSE.txt for license information. |
| 5 | // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception |
| 6 | // |
| 7 | //===----------------------------------------------------------------------===// |
| 8 | |
| 9 | #include "lldb/Utility/RealpathPrefixes.h" |
| 10 | |
| 11 | #include "lldb/Utility/FileSpec.h" |
| 12 | #include "lldb/Utility/FileSpecList.h" |
| 13 | #include "lldb/Utility/LLDBLog.h" |
| 14 | #include "lldb/Utility/Log.h" |
| 15 | #include "lldb/lldb-private-types.h" |
| 16 | |
| 17 | using namespace lldb_private; |
| 18 | |
| 19 | RealpathPrefixes::RealpathPrefixes( |
| 20 | const FileSpecList &file_spec_list, |
| 21 | llvm::IntrusiveRefCntPtr<llvm::vfs::FileSystem> fs) |
| 22 | : m_fs(fs) { |
| 23 | m_prefixes.reserve(n: file_spec_list.GetSize()); |
| 24 | for (const FileSpec &file_spec : file_spec_list) { |
| 25 | m_prefixes.emplace_back(args: file_spec.GetPath()); |
| 26 | } |
| 27 | } |
| 28 | |
| 29 | std::optional<FileSpec> |
| 30 | RealpathPrefixes::ResolveSymlinks(const FileSpec &file_spec) { |
| 31 | if (m_prefixes.empty()) |
| 32 | return std::nullopt; |
| 33 | |
| 34 | // Test if `b` is a *path* prefix of `a` (not just *string* prefix). |
| 35 | // E.g. "/foo/bar" is a path prefix of "/foo/bar/baz" but not "/foo/barbaz". |
| 36 | auto is_path_prefix = [](llvm::StringRef a, llvm::StringRef b, |
| 37 | bool case_sensitive, |
| 38 | llvm::sys::path::Style style) -> bool { |
| 39 | if (case_sensitive ? a.consume_front(Prefix: b) : a.consume_front_insensitive(Prefix: b)) |
| 40 | // If `b` isn't "/", then it won't end with "/" because it comes from |
| 41 | // `FileSpec`. After `a` consumes `b`, `a` should either be empty (i.e. |
| 42 | // `a` == `b`) or end with "/" (the remainder of `a` is a subdirectory). |
| 43 | return b == "/" || a.empty() || |
| 44 | llvm::sys::path::is_separator(value: a[0], style); |
| 45 | return false; |
| 46 | }; |
| 47 | std::string file_spec_path = file_spec.GetPath(); |
| 48 | for (const std::string &prefix : m_prefixes) { |
| 49 | if (is_path_prefix(file_spec_path, prefix, file_spec.IsCaseSensitive(), |
| 50 | file_spec.GetPathStyle())) { |
| 51 | // Stats and logging. |
| 52 | IncreaseSourceRealpathAttemptCount(); |
| 53 | Log *log = GetLog(mask: LLDBLog::Source); |
| 54 | LLDB_LOGF(log, "Realpath'ing support file %s" , file_spec_path.c_str()); |
| 55 | |
| 56 | // One prefix matched. Try to realpath. |
| 57 | PathSmallString buff; |
| 58 | std::error_code ec = m_fs->getRealPath(Path: file_spec_path, Output&: buff); |
| 59 | if (ec) |
| 60 | return std::nullopt; |
| 61 | FileSpec realpath(buff, file_spec.GetPathStyle()); |
| 62 | |
| 63 | // Only return realpath if it is different from the original file_spec. |
| 64 | if (realpath != file_spec) |
| 65 | return realpath; |
| 66 | return std::nullopt; |
| 67 | } |
| 68 | } |
| 69 | // No prefix matched |
| 70 | return std::nullopt; |
| 71 | } |
| 72 | |