/* 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 #include #include #include #include #include #include #include "cmStateSnapshot.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 cmCommandContext { struct cmCommandName { std::string Original; std::string Lower; cmCommandName() = default; cmCommandName(std::string name) : Original(std::move(name)) , Lower(cmSystemTools::LowerCase(this->Original)) { } } Name; long Line = 0; cmCommandContext() = default; cmCommandContext(std::string name, long line) : Name(std::move(name)) , Line(line) { } }; 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 cmListFileContext { public: std::string Name; std::string FilePath; long Line = 0; static long const DeferPlaceholderLine = -1; cm::optional DeferId; cmListFileContext() = default; cmListFileContext(std::string name, std::string filePath, long line) : Name(std::move(name)) , FilePath(std::move(filePath)) , Line(line) { } #if __cplusplus < 201703L && (!defined(_MSVC_LANG) || _MSVC_LANG < 201703L) cmListFileContext(const cmListFileContext& /*other*/) = default; cmListFileContext(cmListFileContext&& /*other*/) = default; cmListFileContext& operator=(const cmListFileContext& /*other*/) = default; cmListFileContext& operator=(cmListFileContext&& /*other*/) = delete; #endif static cmListFileContext FromCommandContext( cmCommandContext const& lfcc, std::string const& fileName, cm::optional deferId = {}) { cmListFileContext lfc; lfc.FilePath = fileName; lfc.Line = lfcc.Line; lfc.Name = lfcc.Name.Original; 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); class cmListFileFunction { public: cmListFileFunction(std::string name, long line, std::vector args) : Impl{ std::make_shared(std::move(name), line, std::move(args)) } { } std::string const& OriginalName() const noexcept { return this->Impl->Name.Original; } std::string const& LowerCaseName() const noexcept { return this->Impl->Name.Lower; } long Line() const noexcept { return this->Impl->Line; } std::vector const& Arguments() const noexcept { return this->Impl->Arguments; } operator cmCommandContext const&() const noexcept { return *this->Impl; } private: struct Implementation : public cmCommandContext { Implementation(std::string name, long line, std::vector args) : cmCommandContext{ std::move(name), line } , Arguments{ std::move(args) } { } std::vector Arguments; }; std::shared_ptr Impl; }; // Represent a backtrace (call stack). Provide value semantics // but use efficient reference-counting underneath to avoid copies. class cmListFileBacktrace { public: // Default-constructed backtrace may not be used until after // set via assignment from a backtrace constructed with a // valid snapshot. cmListFileBacktrace() = default; // Construct an empty backtrace whose bottom sits in the directory // indicated by the given valid snapshot. cmListFileBacktrace(cmStateSnapshot const& snapshot); cmStateSnapshot GetBottom() const; // Get a backtrace with the given file scope added to the top. // May not be called until after construction with a valid snapshot. cmListFileBacktrace Push(std::string const& file) const; // Get a backtrace with the given call context added to the top. // May not be called until after construction with a valid snapshot. cmListFileBacktrace Push(cmListFileContext const& lfc) const; // Get a backtrace with the top level removed. // May not be called until after a matching Push. cmListFileBacktrace Pop() const; // Get the context at the top of the backtrace. // This may be called only if Empty() would return false. cmListFileContext const& Top() const; // Print the top of the backtrace. void PrintTitle(std::ostream& out) const; // Print the call stack below the top of the backtrace. void PrintCallStack(std::ostream& out) const; // Get the number of 'frames' in this backtrace size_t Depth() const; // Return true if this backtrace is empty. bool Empty() const; private: struct Entry; std::shared_ptr TopEntry; cmListFileBacktrace(std::shared_ptr parent, cmListFileContext const& lfc); cmListFileBacktrace(std::shared_ptr top); }; // 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 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 const& l, BT const& r) { return l.Value == r.Value; } friend bool operator<(BT const& l, BT const& r) { return l.Value < r.Value; } friend bool operator==(BT const& l, T const& r) { return l.Value == r; } friend bool operator==(T const& l, BT const& r) { return l == r.Value; } }; std::ostream& operator<<(std::ostream& os, BT 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 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 Backtraces; friend bool operator==(BTs const& l, BTs const& r) { return l.Value == r.Value; } friend bool operator<(BTs const& l, BTs const& r) { return l.Value < r.Value; } friend bool operator==(BTs const& l, T const& r) { return l.Value == r; } friend bool operator==(T const& l, BTs const& r) { return l == r.Value; } }; std::vector> ExpandListWithBacktrace( std::string const& list, cmListFileBacktrace const& bt = cmListFileBacktrace()); 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 Functions; };