/* Distributed under the OSI-approved BSD 3-Clause License. See accompanying file Copyright.txt or https://cmake.org/licensing for details. */ #pragma once #include "cmConfigure.h" // IWYU pragma: keep #include <iosfwd> #include <memory> #include <string> #include <utility> #include <vector> #include <cm/optional> #include "cmConstStack.h" #include "cmList.h" #include "cmSystemTools.h" /** \class cmListFileCache * \brief A class to cache list file contents. * * cmListFileCache is a class used to cache the contents of parsed * cmake list files. */ class cmMessenger; struct cmListFileArgument { enum Delimiter { Unquoted, Quoted, Bracket }; cmListFileArgument() = default; cmListFileArgument(std::string v, Delimiter d, long line) : Value(std::move(v)) , Delim(d) , Line(line) { } bool operator==(const cmListFileArgument& r) const { return (this->Value == r.Value) && (this->Delim == r.Delim); } bool operator!=(const cmListFileArgument& r) const { return !(*this == r); } std::string Value; Delimiter Delim = Unquoted; long Line = 0; }; class cmListFileFunction { public: cmListFileFunction(std::string name, long line, long lineEnd, std::vector<cmListFileArgument> args) : Impl{ std::make_shared<Implementation>(std::move(name), line, lineEnd, std::move(args)) } { } std::string const& OriginalName() const noexcept { return this->Impl->OriginalName; } std::string const& LowerCaseName() const noexcept { return this->Impl->LowerCaseName; } long Line() const noexcept { return this->Impl->Line; } long LineEnd() const noexcept { return this->Impl->LineEnd; } std::vector<cmListFileArgument> const& Arguments() const noexcept { return this->Impl->Arguments; } private: struct Implementation { Implementation(std::string name, long line, long lineEnd, std::vector<cmListFileArgument> args) : OriginalName{ std::move(name) } , LowerCaseName{ cmSystemTools::LowerCase(this->OriginalName) } , Line{ line } , LineEnd{ lineEnd } , Arguments{ std::move(args) } { } std::string OriginalName; std::string LowerCaseName; long Line = 0; long LineEnd = 0; std::vector<cmListFileArgument> Arguments; }; std::shared_ptr<Implementation const> Impl; }; class cmListFileContext { public: std::string Name; std::string FilePath; long Line = 0; static long const DeferPlaceholderLine = -1; cm::optional<std::string> DeferId; cmListFileContext() = default; // This move constructor is marked `noexcept` yet `clang-tidy` 14 reports it // as being able to throw an exception. Suppress the warning as there doesn't // seem to be any way for this to happen given the member types. // NOLINTNEXTLINE(bugprone-exception-escape) cmListFileContext(cmListFileContext&& /*other*/) noexcept = default; cmListFileContext(const cmListFileContext& /*other*/) = default; cmListFileContext& operator=(const cmListFileContext& /*other*/) = default; #if __cplusplus >= 201703L || (defined(_MSVC_LANG) && _MSVC_LANG >= 201703L) cmListFileContext& operator=(cmListFileContext&& /*other*/) noexcept = default; #else // The move assignment operators for several STL classes did not become // noexcept until C++17, which causes some tools to warn about this move // assignment operator throwing an exception when it shouldn't. cmListFileContext& operator=(cmListFileContext&& /*other*/) noexcept = delete; #endif cmListFileContext(std::string name, std::string filePath, long line) : Name(std::move(name)) , FilePath(std::move(filePath)) , Line(line) { } static cmListFileContext FromListFilePath(std::string const& filePath) { // We are entering a file-level scope but have not yet reached // any specific line or command invocation within it. This context // is useful to print when it is at the top but otherwise can be // skipped during call stack printing. cmListFileContext lfc; lfc.FilePath = filePath; return lfc; } static cmListFileContext FromListFileFunction( cmListFileFunction const& lff, std::string const& fileName, cm::optional<std::string> deferId = {}) { cmListFileContext lfc; lfc.FilePath = fileName; lfc.Line = lff.Line(); lfc.Name = lff.OriginalName(); lfc.DeferId = std::move(deferId); return lfc; } }; std::ostream& operator<<(std::ostream&, cmListFileContext const&); bool operator<(const cmListFileContext& lhs, const cmListFileContext& rhs); bool operator==(cmListFileContext const& lhs, cmListFileContext const& rhs); bool operator!=(cmListFileContext const& lhs, cmListFileContext const& rhs); // Represent a backtrace (call stack) with efficient value semantics. class cmListFileBacktrace : public cmConstStack<cmListFileContext, cmListFileBacktrace> { using cmConstStack::cmConstStack; friend class cmConstStack<cmListFileContext, cmListFileBacktrace>; }; #ifndef cmListFileCache_cxx extern template class cmConstStack<cmListFileContext, cmListFileBacktrace>; #endif // Wrap type T as a value with a backtrace. For purposes of // ordering and equality comparison, only the original value is // used. The backtrace is considered incidental. template <typename T> class BT { public: BT(T v = T(), cmListFileBacktrace bt = cmListFileBacktrace()) : Value(std::move(v)) , Backtrace(std::move(bt)) { } T Value; cmListFileBacktrace Backtrace; friend bool operator==(BT<T> const& l, BT<T> const& r) { return l.Value == r.Value; } friend bool operator<(BT<T> const& l, BT<T> const& r) { return l.Value < r.Value; } friend bool operator==(BT<T> const& l, T const& r) { return l.Value == r; } friend bool operator==(T const& l, BT<T> const& r) { return l == r.Value; } }; std::ostream& operator<<(std::ostream& os, BT<std::string> const& s); // Wrap type T as a value with potentially multiple backtraces. For purposes // of ordering and equality comparison, only the original value is used. The // backtrace is considered incidental. template <typename T> class BTs { public: BTs(T v = T(), cmListFileBacktrace bt = cmListFileBacktrace()) : Value(std::move(v)) { this->Backtraces.emplace_back(std::move(bt)); } T Value; std::vector<cmListFileBacktrace> Backtraces; friend bool operator==(BTs<T> const& l, BTs<T> const& r) { return l.Value == r.Value; } friend bool operator<(BTs<T> const& l, BTs<T> const& r) { return l.Value < r.Value; } friend bool operator==(BTs<T> const& l, T const& r) { return l.Value == r; } friend bool operator==(T const& l, BTs<T> const& r) { return l == r.Value; } }; std::vector<BT<std::string>> cmExpandListWithBacktrace( std::string const& list, cmListFileBacktrace const& bt = cmListFileBacktrace(), cmList::EmptyElements emptyArgs = cmList::EmptyElements::No); struct cmListFile { bool ParseFile(const char* path, cmMessenger* messenger, cmListFileBacktrace const& lfbt); bool ParseString(const char* str, const char* virtual_filename, cmMessenger* messenger, cmListFileBacktrace const& lfbt); std::vector<cmListFileFunction> Functions; };