|  | //===--- FS.cpp - File system related utils ----------------------*- C++-*-===// | 
|  | // | 
|  | // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. | 
|  | // See https://llvm.org/LICENSE.txt for license information. | 
|  | // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception | 
|  | // | 
|  | //===----------------------------------------------------------------------===// | 
|  |  | 
|  | #include "FS.h" | 
|  | #include "clang/Basic/LLVM.h" | 
|  | #include "llvm/ADT/None.h" | 
|  | #include "llvm/Support/Path.h" | 
|  | #include "llvm/Support/VirtualFileSystem.h" | 
|  |  | 
|  | namespace clang { | 
|  | namespace clangd { | 
|  |  | 
|  | PreambleFileStatusCache::PreambleFileStatusCache(llvm::StringRef MainFilePath){ | 
|  | assert(llvm::sys::path::is_absolute(MainFilePath)); | 
|  | llvm::SmallString<256> MainFileCanonical(MainFilePath); | 
|  | llvm::sys::path::remove_dots(MainFileCanonical, /*remove_dot_dot=*/true); | 
|  | this->MainFilePath = std::string(MainFileCanonical.str()); | 
|  | } | 
|  |  | 
|  | void PreambleFileStatusCache::update(const llvm::vfs::FileSystem &FS, | 
|  | llvm::vfs::Status S) { | 
|  | // Canonicalize path for later lookup, which is usually by absolute path. | 
|  | llvm::SmallString<32> PathStore(S.getName()); | 
|  | if (FS.makeAbsolute(PathStore)) | 
|  | return; | 
|  | llvm::sys::path::remove_dots(PathStore, /*remove_dot_dot=*/true); | 
|  | // Do not cache status for the main file. | 
|  | if (PathStore == MainFilePath) | 
|  | return; | 
|  | // Stores the latest status in cache as it can change in a preamble build. | 
|  | StatCache.insert({PathStore, std::move(S)}); | 
|  | } | 
|  |  | 
|  | llvm::Optional<llvm::vfs::Status> | 
|  | PreambleFileStatusCache::lookup(llvm::StringRef File) const { | 
|  | // Canonicalize to match the cached form. | 
|  | // Lookup tends to be first by absolute path, so no need to make absolute. | 
|  | llvm::SmallString<256> PathLookup(File); | 
|  | llvm::sys::path::remove_dots(PathLookup, /*remove_dot_dot=*/true); | 
|  |  | 
|  | auto I = StatCache.find(PathLookup); | 
|  | if (I != StatCache.end()) | 
|  | // Returned Status name should always match the requested File. | 
|  | return llvm::vfs::Status::copyWithNewName(I->getValue(), File); | 
|  | return None; | 
|  | } | 
|  |  | 
|  | llvm::IntrusiveRefCntPtr<llvm::vfs::FileSystem> | 
|  | PreambleFileStatusCache::getProducingFS( | 
|  | llvm::IntrusiveRefCntPtr<llvm::vfs::FileSystem> FS) { | 
|  | // This invalidates old status in cache if files are re-`open()`ed or | 
|  | // re-`stat()`ed in case file status has changed during preamble build. | 
|  | class CollectFS : public llvm::vfs::ProxyFileSystem { | 
|  | public: | 
|  | CollectFS(llvm::IntrusiveRefCntPtr<llvm::vfs::FileSystem> FS, | 
|  | PreambleFileStatusCache &StatCache) | 
|  | : ProxyFileSystem(std::move(FS)), StatCache(StatCache) {} | 
|  |  | 
|  | llvm::ErrorOr<std::unique_ptr<llvm::vfs::File>> | 
|  | openFileForRead(const llvm::Twine &Path) override { | 
|  | auto File = getUnderlyingFS().openFileForRead(Path); | 
|  | if (!File || !*File) | 
|  | return File; | 
|  | // Eagerly stat opened file, as the followup `status` call on the file | 
|  | // doesn't necessarily go through this FS. This puts some extra work on | 
|  | // preamble build, but it should be worth it as preamble can be reused | 
|  | // many times (e.g. code completion) and the repeated status call is | 
|  | // likely to be cached in the underlying file system anyway. | 
|  | if (auto S = File->get()->status()) | 
|  | StatCache.update(getUnderlyingFS(), std::move(*S)); | 
|  | return File; | 
|  | } | 
|  |  | 
|  | llvm::ErrorOr<llvm::vfs::Status> status(const llvm::Twine &Path) override { | 
|  | auto S = getUnderlyingFS().status(Path); | 
|  | if (S) | 
|  | StatCache.update(getUnderlyingFS(), *S); | 
|  | return S; | 
|  | } | 
|  |  | 
|  | private: | 
|  | PreambleFileStatusCache &StatCache; | 
|  | }; | 
|  | return llvm::IntrusiveRefCntPtr<CollectFS>( | 
|  | new CollectFS(std::move(FS), *this)); | 
|  | } | 
|  |  | 
|  | llvm::IntrusiveRefCntPtr<llvm::vfs::FileSystem> | 
|  | PreambleFileStatusCache::getConsumingFS( | 
|  | llvm::IntrusiveRefCntPtr<llvm::vfs::FileSystem> FS) const { | 
|  | class CacheVFS : public llvm::vfs::ProxyFileSystem { | 
|  | public: | 
|  | CacheVFS(llvm::IntrusiveRefCntPtr<llvm::vfs::FileSystem> FS, | 
|  | const PreambleFileStatusCache &StatCache) | 
|  | : ProxyFileSystem(std::move(FS)), StatCache(StatCache) {} | 
|  |  | 
|  | llvm::ErrorOr<llvm::vfs::Status> status(const llvm::Twine &Path) override { | 
|  | if (auto S = StatCache.lookup(Path.str())) | 
|  | return *S; | 
|  | return getUnderlyingFS().status(Path); | 
|  | } | 
|  |  | 
|  | private: | 
|  | const PreambleFileStatusCache &StatCache; | 
|  | }; | 
|  | return llvm::IntrusiveRefCntPtr<CacheVFS>(new CacheVFS(std::move(FS), *this)); | 
|  | } | 
|  |  | 
|  | Path removeDots(PathRef File) { | 
|  | llvm::SmallString<128> CanonPath(File); | 
|  | llvm::sys::path::remove_dots(CanonPath, /*remove_dot_dot=*/true); | 
|  | return CanonPath.str().str(); | 
|  | } | 
|  |  | 
|  | } // namespace clangd | 
|  | } // namespace clang |