diff options
author | Mike Aizatsky <aizatsky@chromium.org> | 2015-11-09 18:56:31 +0000 |
---|---|---|
committer | Mike Aizatsky <aizatsky@chromium.org> | 2015-11-09 18:56:31 +0000 |
commit | 662b4fd325d410181d91031f0155e60c520eebe6 (patch) | |
tree | 283c6efc13f982c838ebef8a9941ed79ec6f7c73 /llvm | |
parent | cee6a6a63b56c9317dd73edefa93ea35037fa2f6 (diff) | |
download | bcm5719-llvm-662b4fd325d410181d91031f0155e60c520eebe6.tar.gz bcm5719-llvm-662b4fd325d410181d91031f0155e60c520eebe6.zip |
Moving FileManager::removeDotPaths to llvm::sys::path::remove_dots
Differential Revision: http://reviews.llvm.org/D14393
llvm-svn: 252499
Diffstat (limited to 'llvm')
-rw-r--r-- | llvm/include/llvm/Support/Path.h | 7 | ||||
-rw-r--r-- | llvm/lib/Support/Path.cpp | 35 | ||||
-rw-r--r-- | llvm/unittests/Support/Path.cpp | 20 |
3 files changed, 62 insertions, 0 deletions
diff --git a/llvm/include/llvm/Support/Path.h b/llvm/include/llvm/Support/Path.h index 1e56be4a5fc..955cc991d9b 100644 --- a/llvm/include/llvm/Support/Path.h +++ b/llvm/include/llvm/Support/Path.h @@ -423,6 +423,13 @@ bool is_relative(const Twine &path); /// @result The cleaned-up \a path. StringRef remove_leading_dotslash(StringRef path); +/// @brief In-place remove any './' and optionally '../' components from a path. +/// +/// @param path processed path +/// @param remove_dot_dot specify if '../' should be removed +/// @result True if path was changed +bool remove_dots(SmallVectorImpl<char> &path, bool remove_dot_dot = false); + } // end namespace path } // end namespace sys } // end namespace llvm diff --git a/llvm/lib/Support/Path.cpp b/llvm/lib/Support/Path.cpp index f45774bca7b..4952f59fc24 100644 --- a/llvm/lib/Support/Path.cpp +++ b/llvm/lib/Support/Path.cpp @@ -671,6 +671,41 @@ StringRef remove_leading_dotslash(StringRef Path) { return Path; } +static SmallString<256> remove_dots(StringRef path, bool remove_dot_dot) { + SmallVector<StringRef, 16> components; + + // Skip the root path, then look for traversal in the components. + StringRef rel = path::relative_path(path); + for (StringRef C : llvm::make_range(path::begin(rel), path::end(rel))) { + if (C == ".") + continue; + if (remove_dot_dot) { + if (C == "..") { + if (!components.empty()) + components.pop_back(); + continue; + } + } + components.push_back(C); + } + + SmallString<256> buffer = path::root_path(path); + for (StringRef C : components) + path::append(buffer, C); + return buffer; +} + +bool remove_dots(SmallVectorImpl<char> &path, bool remove_dot_dot) { + StringRef p(path.data(), path.size()); + + SmallString<256> result = remove_dots(p, remove_dot_dot); + if (result == path) + return false; + + path.swap(result); + return true; +} + } // end namespace path namespace fs { diff --git a/llvm/unittests/Support/Path.cpp b/llvm/unittests/Support/Path.cpp index a7a6a4add7c..07ad3fc6b63 100644 --- a/llvm/unittests/Support/Path.cpp +++ b/llvm/unittests/Support/Path.cpp @@ -844,4 +844,24 @@ TEST(Support, RemoveLeadingDotSlash) { Path2 = path::remove_leading_dotslash(Path2); EXPECT_EQ(Path2, ""); } + +static std::string remove_dots(StringRef path, + bool remove_dot_dot) { + SmallString<256> buffer(path); + path::remove_dots(buffer, remove_dot_dot); + return buffer.str(); +} + +TEST(Support, RemoveDots) { + EXPECT_EQ("foolz/wat", remove_dots("././/foolz/wat", false)); + EXPECT_EQ("", remove_dots("./////", false)); + + EXPECT_EQ("a/../b/c", remove_dots("./a/../b/c", false)); + EXPECT_EQ("b/c", remove_dots("./a/../b/c", true)); + EXPECT_EQ("c", remove_dots("././c", true)); + + SmallString<64> Path1("././c"); + EXPECT_TRUE(path::remove_dots(Path1, true)); + EXPECT_EQ("c", Path1); +} } // anonymous namespace |