forked from OSchip/llvm-project
[VFS] More consistent support for Windows
Removed some #ifdefs specific to Windows handling of VFS paths. This eliminates most of the differences between the Windows and non-Windows code paths. Making this work required some changes to account for the fact that VFS file paths can be Posix style or Windows style, so you cannot just assume that they use the host's native path style. In one case, this means implementing our own version of make_absolute, since the filesystem code in Support doesn't have styles in the sense that the path code does. Differential Review: https://reviews.llvm.org/D71092
This commit is contained in:
parent
5aa6e246a1
commit
da45bd2321
|
@ -10,7 +10,7 @@
|
||||||
// Preprocessor (__FILE__ macro and # directives):
|
// Preprocessor (__FILE__ macro and # directives):
|
||||||
|
|
||||||
// RUN: %clang_cc1 -I %t -ivfsoverlay %t.external.yaml -E %s | FileCheck -check-prefix=CHECK-PP-EXTERNAL %s
|
// RUN: %clang_cc1 -I %t -ivfsoverlay %t.external.yaml -E %s | FileCheck -check-prefix=CHECK-PP-EXTERNAL %s
|
||||||
// CHECK-PP-EXTERNAL: # {{[0-9]*}} "[[NAME:.*Inputs.external-names.h]]"
|
// CHECK-PP-EXTERNAL: # {{[0-9]*}} "[[NAME:.*Inputs(/|\\\\)external-names.h]]"
|
||||||
// CHECK-PP-EXTERNAL-NEXT: void foo(char **c) {
|
// CHECK-PP-EXTERNAL-NEXT: void foo(char **c) {
|
||||||
// CHECK-PP-EXTERNAL-NEXT: *c = "[[NAME]]";
|
// CHECK-PP-EXTERNAL-NEXT: *c = "[[NAME]]";
|
||||||
|
|
||||||
|
|
|
@ -705,16 +705,6 @@ private:
|
||||||
bool IsFallthrough = true;
|
bool IsFallthrough = true;
|
||||||
/// @}
|
/// @}
|
||||||
|
|
||||||
/// Virtual file paths and external files could be canonicalized without "..",
|
|
||||||
/// "." and "./" in their paths. FIXME: some unittests currently fail on
|
|
||||||
/// win32 when using remove_dots and remove_leading_dotslash on paths.
|
|
||||||
bool UseCanonicalizedPaths =
|
|
||||||
#ifdef _WIN32
|
|
||||||
false;
|
|
||||||
#else
|
|
||||||
true;
|
|
||||||
#endif
|
|
||||||
|
|
||||||
RedirectingFileSystem(IntrusiveRefCntPtr<FileSystem> ExternalFS);
|
RedirectingFileSystem(IntrusiveRefCntPtr<FileSystem> ExternalFS);
|
||||||
|
|
||||||
/// Looks up the path <tt>[Start, End)</tt> in \p From, possibly
|
/// Looks up the path <tt>[Start, End)</tt> in \p From, possibly
|
||||||
|
|
|
@ -990,6 +990,28 @@ std::error_code InMemoryFileSystem::isLocal(const Twine &Path, bool &Result) {
|
||||||
// RedirectingFileSystem implementation
|
// RedirectingFileSystem implementation
|
||||||
//===-----------------------------------------------------------------------===/
|
//===-----------------------------------------------------------------------===/
|
||||||
|
|
||||||
|
namespace {
|
||||||
|
|
||||||
|
/// Removes leading "./" as well as path components like ".." and ".".
|
||||||
|
static llvm::SmallString<256> canonicalize(llvm::StringRef Path) {
|
||||||
|
// First detect the path style in use by checking the first separator.
|
||||||
|
llvm::sys::path::Style style = llvm::sys::path::Style::native;
|
||||||
|
const size_t n = Path.find_first_of("/\\");
|
||||||
|
if (n != static_cast<size_t>(-1))
|
||||||
|
style = (Path[n] == '/') ? llvm::sys::path::Style::posix
|
||||||
|
: llvm::sys::path::Style::windows;
|
||||||
|
|
||||||
|
// Now remove the dots. Explicitly specifying the path style prevents the
|
||||||
|
// direction of the slashes from changing.
|
||||||
|
llvm::SmallString<256> result =
|
||||||
|
llvm::sys::path::remove_leading_dotslash(Path, style);
|
||||||
|
llvm::sys::path::remove_dots(result, /*remove_dot_dot=*/true, style);
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
} // anonymous namespace
|
||||||
|
|
||||||
|
|
||||||
RedirectingFileSystem::RedirectingFileSystem(IntrusiveRefCntPtr<FileSystem> FS)
|
RedirectingFileSystem::RedirectingFileSystem(IntrusiveRefCntPtr<FileSystem> FS)
|
||||||
: ExternalFS(std::move(FS)) {
|
: ExternalFS(std::move(FS)) {
|
||||||
if (ExternalFS)
|
if (ExternalFS)
|
||||||
|
@ -1083,7 +1105,23 @@ std::error_code RedirectingFileSystem::makeAbsolute(SmallVectorImpl<char> &Path)
|
||||||
if (!WorkingDir)
|
if (!WorkingDir)
|
||||||
return WorkingDir.getError();
|
return WorkingDir.getError();
|
||||||
|
|
||||||
llvm::sys::fs::make_absolute(WorkingDir.get(), Path);
|
// We can't use sys::fs::make_absolute because that assumes the path style
|
||||||
|
// is native and there is no way to override that. Since we know WorkingDir
|
||||||
|
// is absolute, we can use it to determine which style we actually have and
|
||||||
|
// append Path ourselves.
|
||||||
|
sys::path::Style style = sys::path::Style::windows;
|
||||||
|
if (sys::path::is_absolute(WorkingDir.get(), sys::path::Style::posix)) {
|
||||||
|
style = sys::path::Style::posix;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::string Result = WorkingDir.get();
|
||||||
|
StringRef Dir(Result);
|
||||||
|
if (!Dir.endswith(sys::path::get_separator(style))) {
|
||||||
|
Result += sys::path::get_separator(style);
|
||||||
|
}
|
||||||
|
Result.append(Path.data(), Path.size());
|
||||||
|
Path.assign(Result.begin(), Result.end());
|
||||||
|
|
||||||
return {};
|
return {};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1318,8 +1356,8 @@ class llvm::vfs::RedirectingFileSystemParser {
|
||||||
bool HasContents = false; // external or otherwise
|
bool HasContents = false; // external or otherwise
|
||||||
std::vector<std::unique_ptr<RedirectingFileSystem::Entry>>
|
std::vector<std::unique_ptr<RedirectingFileSystem::Entry>>
|
||||||
EntryArrayContents;
|
EntryArrayContents;
|
||||||
std::string ExternalContentsPath;
|
SmallString<256> ExternalContentsPath;
|
||||||
std::string Name;
|
SmallString<256> Name;
|
||||||
yaml::Node *NameValueNode = nullptr;
|
yaml::Node *NameValueNode = nullptr;
|
||||||
auto UseExternalName =
|
auto UseExternalName =
|
||||||
RedirectingFileSystem::RedirectingFileEntry::NK_NotSet;
|
RedirectingFileSystem::RedirectingFileEntry::NK_NotSet;
|
||||||
|
@ -1342,16 +1380,9 @@ class llvm::vfs::RedirectingFileSystemParser {
|
||||||
return nullptr;
|
return nullptr;
|
||||||
|
|
||||||
NameValueNode = I.getValue();
|
NameValueNode = I.getValue();
|
||||||
if (FS->UseCanonicalizedPaths) {
|
// Guarantee that old YAML files containing paths with ".." and "."
|
||||||
SmallString<256> Path(Value);
|
// are properly canonicalized before read into the VFS.
|
||||||
// Guarantee that old YAML files containing paths with ".." and "."
|
Name = canonicalize(Value).str();
|
||||||
// are properly canonicalized before read into the VFS.
|
|
||||||
Path = sys::path::remove_leading_dotslash(Path);
|
|
||||||
sys::path::remove_dots(Path, /*remove_dot_dot=*/true);
|
|
||||||
Name = std::string(Path.str());
|
|
||||||
} else {
|
|
||||||
Name = std::string(Value);
|
|
||||||
}
|
|
||||||
} else if (Key == "type") {
|
} else if (Key == "type") {
|
||||||
if (!parseScalarString(I.getValue(), Value, Buffer))
|
if (!parseScalarString(I.getValue(), Value, Buffer))
|
||||||
return nullptr;
|
return nullptr;
|
||||||
|
@ -1404,13 +1435,10 @@ class llvm::vfs::RedirectingFileSystemParser {
|
||||||
FullPath = Value;
|
FullPath = Value;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (FS->UseCanonicalizedPaths) {
|
// Guarantee that old YAML files containing paths with ".." and "."
|
||||||
// Guarantee that old YAML files containing paths with ".." and "."
|
// are properly canonicalized before read into the VFS.
|
||||||
// are properly canonicalized before read into the VFS.
|
FullPath = canonicalize(FullPath);
|
||||||
FullPath = sys::path::remove_leading_dotslash(FullPath);
|
ExternalContentsPath = FullPath.str();
|
||||||
sys::path::remove_dots(FullPath, /*remove_dot_dot=*/true);
|
|
||||||
}
|
|
||||||
ExternalContentsPath = std::string(FullPath.str());
|
|
||||||
} else if (Key == "use-external-name") {
|
} else if (Key == "use-external-name") {
|
||||||
bool Val;
|
bool Val;
|
||||||
if (!parseScalarBool(I.getValue(), Val))
|
if (!parseScalarBool(I.getValue(), Val))
|
||||||
|
@ -1654,14 +1682,10 @@ RedirectingFileSystem::lookupPath(const Twine &Path_) const {
|
||||||
if (std::error_code EC = makeAbsolute(Path))
|
if (std::error_code EC = makeAbsolute(Path))
|
||||||
return EC;
|
return EC;
|
||||||
|
|
||||||
// Canonicalize path by removing ".", "..", "./", etc components. This is
|
// Canonicalize path by removing ".", "..", "./", components. This is
|
||||||
// a VFS request, do bot bother about symlinks in the path components
|
// a VFS request, do not bother about symlinks in the path components
|
||||||
// but canonicalize in order to perform the correct entry search.
|
// but canonicalize in order to perform the correct entry search.
|
||||||
if (UseCanonicalizedPaths) {
|
Path = canonicalize(Path);
|
||||||
Path = sys::path::remove_leading_dotslash(Path);
|
|
||||||
sys::path::remove_dots(Path, /*remove_dot_dot=*/true);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (Path.empty())
|
if (Path.empty())
|
||||||
return make_error_code(llvm::errc::invalid_argument);
|
return make_error_code(llvm::errc::invalid_argument);
|
||||||
|
|
||||||
|
@ -1680,16 +1704,9 @@ ErrorOr<RedirectingFileSystem::Entry *>
|
||||||
RedirectingFileSystem::lookupPath(sys::path::const_iterator Start,
|
RedirectingFileSystem::lookupPath(sys::path::const_iterator Start,
|
||||||
sys::path::const_iterator End,
|
sys::path::const_iterator End,
|
||||||
RedirectingFileSystem::Entry *From) const {
|
RedirectingFileSystem::Entry *From) const {
|
||||||
#ifndef _WIN32
|
|
||||||
assert(!isTraversalComponent(*Start) &&
|
assert(!isTraversalComponent(*Start) &&
|
||||||
!isTraversalComponent(From->getName()) &&
|
!isTraversalComponent(From->getName()) &&
|
||||||
"Paths should not contain traversal components");
|
"Paths should not contain traversal components");
|
||||||
#else
|
|
||||||
// FIXME: this is here to support windows, remove it once canonicalized
|
|
||||||
// paths become globally default.
|
|
||||||
if (Start->equals("."))
|
|
||||||
++Start;
|
|
||||||
#endif
|
|
||||||
|
|
||||||
StringRef FromName = From->getName();
|
StringRef FromName = From->getName();
|
||||||
|
|
||||||
|
|
|
@ -2148,11 +2148,9 @@ TEST_F(VFSFromYAMLTest, WorkingDirectoryFallthrough) {
|
||||||
ASSERT_FALSE(Status.getError());
|
ASSERT_FALSE(Status.getError());
|
||||||
EXPECT_TRUE(Status->exists());
|
EXPECT_TRUE(Status->exists());
|
||||||
|
|
||||||
#if !defined(_WIN32)
|
|
||||||
Status = FS->status("../bar/baz/a");
|
Status = FS->status("../bar/baz/a");
|
||||||
ASSERT_FALSE(Status.getError());
|
ASSERT_FALSE(Status.getError());
|
||||||
EXPECT_TRUE(Status->exists());
|
EXPECT_TRUE(Status->exists());
|
||||||
#endif
|
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_F(VFSFromYAMLTest, WorkingDirectoryFallthroughInvalid) {
|
TEST_F(VFSFromYAMLTest, WorkingDirectoryFallthroughInvalid) {
|
||||||
|
|
Loading…
Reference in New Issue