|  | //===--- PPCallbacksTracker.h - Preprocessor tracking -----------*- C++ -*-===// | 
|  | // | 
|  | //                     The LLVM Compiler Infrastructure | 
|  | // | 
|  | // This file is distributed under the University of Illinois Open Source | 
|  | // License. See LICENSE.TXT for details. | 
|  | // | 
|  | //===----------------------------------------------------------------------===// | 
|  | /// | 
|  | /// \file | 
|  | /// \brief Classes and definitions for preprocessor tracking. | 
|  | /// | 
|  | /// The core definition is the PPCallbacksTracker class, derived from Clang's | 
|  | /// PPCallbacks class from the Lex library, which overrides all the callbacks | 
|  | /// and collects information about each callback call, saving it in a | 
|  | /// data structure built up of CallbackCall and Argument objects, which | 
|  | /// record the preprocessor callback name and arguments in high-level string | 
|  | /// form for later inspection. | 
|  | /// | 
|  | //===----------------------------------------------------------------------===// | 
|  |  | 
|  | #ifndef PPTRACE_PPCALLBACKSTRACKER_H | 
|  | #define PPTRACE_PPCALLBACKSTRACKER_H | 
|  |  | 
|  | #include "clang/Lex/PPCallbacks.h" | 
|  | #include "clang/Lex/Preprocessor.h" | 
|  | #include "clang/Basic/SourceManager.h" | 
|  | #include "llvm/ADT/ArrayRef.h" | 
|  | #include "llvm/ADT/SmallSet.h" | 
|  | #include "llvm/ADT/StringRef.h" | 
|  | #include <string> | 
|  | #include <vector> | 
|  |  | 
|  | /// \brief This class represents one callback function argument by name | 
|  | ///   and value. | 
|  | class Argument { | 
|  | public: | 
|  | Argument(llvm::StringRef Name, llvm::StringRef Value) | 
|  | : Name(Name), Value(Value) {} | 
|  | Argument() = default; | 
|  |  | 
|  | std::string Name; | 
|  | std::string Value; | 
|  | }; | 
|  |  | 
|  | /// \brief This class represents one callback call by name and an array | 
|  | ///   of arguments. | 
|  | class CallbackCall { | 
|  | public: | 
|  | CallbackCall(llvm::StringRef Name) : Name(Name) {} | 
|  | CallbackCall() = default; | 
|  |  | 
|  | std::string Name; | 
|  | std::vector<Argument> Arguments; | 
|  | }; | 
|  |  | 
|  | /// \brief This class overrides the PPCallbacks class for tracking preprocessor | 
|  | ///   activity by means of its callback functions. | 
|  | /// | 
|  | /// This object is given a vector for storing the trace information, built up | 
|  | /// of CallbackCall and subordinate Argument objects for representing the | 
|  | /// callback calls and their arguments.  It's a reference so the vector can | 
|  | /// exist beyond the lifetime of this object, because it's deleted by the | 
|  | /// preprocessor automatically in its destructor. | 
|  | /// | 
|  | /// This class supports a mechanism for inhibiting trace output for | 
|  | /// specific callbacks by name, for the purpose of eliminating output for | 
|  | /// callbacks of no interest that might clutter the output. | 
|  | /// | 
|  | /// Following the constructor and destructor function declarations, the | 
|  | /// overidden callback functions are defined.  The remaining functions are | 
|  | /// helpers for recording the trace data, to reduce the coupling between it | 
|  | /// and the recorded data structure. | 
|  | class PPCallbacksTracker : public clang::PPCallbacks { | 
|  | public: | 
|  | /// \brief Note that all of the arguments are references, and owned | 
|  | /// by the caller. | 
|  | /// \param Ignore - Set of names of callbacks to ignore. | 
|  | /// \param CallbackCalls - Trace buffer. | 
|  | /// \param PP - The preprocessor.  Needed for getting some argument strings. | 
|  | PPCallbacksTracker(llvm::SmallSet<std::string, 4> &Ignore, | 
|  | std::vector<CallbackCall> &CallbackCalls, | 
|  | clang::Preprocessor &PP); | 
|  |  | 
|  | ~PPCallbacksTracker() override; | 
|  |  | 
|  | // Overidden callback functions. | 
|  |  | 
|  | void FileChanged(clang::SourceLocation Loc, | 
|  | clang::PPCallbacks::FileChangeReason Reason, | 
|  | clang::SrcMgr::CharacteristicKind FileType, | 
|  | clang::FileID PrevFID = clang::FileID()) override; | 
|  | void FileSkipped(const clang::FileEntry &SkippedFile, | 
|  | const clang::Token &FilenameTok, | 
|  | clang::SrcMgr::CharacteristicKind FileType) override; | 
|  | bool FileNotFound(llvm::StringRef FileName, | 
|  | llvm::SmallVectorImpl<char> &RecoveryPath) override; | 
|  | void InclusionDirective(clang::SourceLocation HashLoc, | 
|  | const clang::Token &IncludeTok, | 
|  | llvm::StringRef FileName, bool IsAngled, | 
|  | clang::CharSourceRange FilenameRange, | 
|  | const clang::FileEntry *File, | 
|  | llvm::StringRef SearchPath, | 
|  | llvm::StringRef RelativePath, | 
|  | const clang::Module *Imported, | 
|  | clang::SrcMgr::CharacteristicKind FileType) override; | 
|  | void moduleImport(clang::SourceLocation ImportLoc, clang::ModuleIdPath Path, | 
|  | const clang::Module *Imported) override; | 
|  | void EndOfMainFile() override; | 
|  | void Ident(clang::SourceLocation Loc, llvm::StringRef str) override; | 
|  | void PragmaDirective(clang::SourceLocation Loc, | 
|  | clang::PragmaIntroducerKind Introducer) override; | 
|  | void PragmaComment(clang::SourceLocation Loc, | 
|  | const clang::IdentifierInfo *Kind, | 
|  | llvm::StringRef Str) override; | 
|  | void PragmaDetectMismatch(clang::SourceLocation Loc, llvm::StringRef Name, | 
|  | llvm::StringRef Value) override; | 
|  | void PragmaDebug(clang::SourceLocation Loc, | 
|  | llvm::StringRef DebugType) override; | 
|  | void PragmaMessage(clang::SourceLocation Loc, llvm::StringRef Namespace, | 
|  | clang::PPCallbacks::PragmaMessageKind Kind, | 
|  | llvm::StringRef Str) override; | 
|  | void PragmaDiagnosticPush(clang::SourceLocation Loc, | 
|  | llvm::StringRef Namespace) override; | 
|  | void PragmaDiagnosticPop(clang::SourceLocation Loc, | 
|  | llvm::StringRef Namespace) override; | 
|  | void PragmaDiagnostic(clang::SourceLocation Loc, llvm::StringRef Namespace, | 
|  | clang::diag::Severity mapping, | 
|  | llvm::StringRef Str) override; | 
|  | void PragmaOpenCLExtension(clang::SourceLocation NameLoc, | 
|  | const clang::IdentifierInfo *Name, | 
|  | clang::SourceLocation StateLoc, | 
|  | unsigned State) override; | 
|  | void PragmaWarning(clang::SourceLocation Loc, llvm::StringRef WarningSpec, | 
|  | llvm::ArrayRef<int> Ids) override; | 
|  | void PragmaWarningPush(clang::SourceLocation Loc, int Level) override; | 
|  | void PragmaWarningPop(clang::SourceLocation Loc) override; | 
|  | void MacroExpands(const clang::Token &MacroNameTok, | 
|  | const clang::MacroDefinition &MD, clang::SourceRange Range, | 
|  | const clang::MacroArgs *Args) override; | 
|  | void MacroDefined(const clang::Token &MacroNameTok, | 
|  | const clang::MacroDirective *MD) override; | 
|  | void MacroUndefined(const clang::Token &MacroNameTok, | 
|  | const clang::MacroDefinition &MD, | 
|  | const clang::MacroDirective *Undef) override; | 
|  | void Defined(const clang::Token &MacroNameTok, | 
|  | const clang::MacroDefinition &MD, | 
|  | clang::SourceRange Range) override; | 
|  | void SourceRangeSkipped(clang::SourceRange Range, | 
|  | clang::SourceLocation EndifLoc) override; | 
|  | void If(clang::SourceLocation Loc, clang::SourceRange ConditionRange, | 
|  | ConditionValueKind ConditionValue) override; | 
|  | void Elif(clang::SourceLocation Loc, clang::SourceRange ConditionRange, | 
|  | ConditionValueKind ConditionValue, clang::SourceLocation IfLoc) override; | 
|  | void Ifdef(clang::SourceLocation Loc, const clang::Token &MacroNameTok, | 
|  | const clang::MacroDefinition &MD) override; | 
|  | void Ifndef(clang::SourceLocation Loc, const clang::Token &MacroNameTok, | 
|  | const clang::MacroDefinition &MD) override; | 
|  | void Else(clang::SourceLocation Loc, | 
|  | clang::SourceLocation IfLoc) override; | 
|  | void Endif(clang::SourceLocation Loc, | 
|  | clang::SourceLocation IfLoc) override; | 
|  |  | 
|  | // Helper functions. | 
|  |  | 
|  | /// \brief Start a new callback. | 
|  | void beginCallback(const char *Name); | 
|  |  | 
|  | /// \brief Append a string to the top trace item. | 
|  | void append(const char *Str); | 
|  |  | 
|  | /// \brief Append a bool argument to the top trace item. | 
|  | void appendArgument(const char *Name, bool Value); | 
|  |  | 
|  | /// \brief Append an int argument to the top trace item. | 
|  | void appendArgument(const char *Name, int Value); | 
|  |  | 
|  | /// \brief Append a string argument to the top trace item. | 
|  | void appendArgument(const char *Name, const char *Value); | 
|  |  | 
|  | /// \brief Append a string reference object argument to the top trace item. | 
|  | void appendArgument(const char *Name, llvm::StringRef Value); | 
|  |  | 
|  | /// \brief Append a string object argument to the top trace item. | 
|  | void appendArgument(const char *Name, const std::string &Value); | 
|  |  | 
|  | /// \brief Append a token argument to the top trace item. | 
|  | void appendArgument(const char *Name, const clang::Token &Value); | 
|  |  | 
|  | /// \brief Append an enum argument to the top trace item. | 
|  | void appendArgument(const char *Name, int Value, const char *const Strings[]); | 
|  |  | 
|  | /// \brief Append a FileID argument to the top trace item. | 
|  | void appendArgument(const char *Name, clang::FileID Value); | 
|  |  | 
|  | /// \brief Append a FileEntry argument to the top trace item. | 
|  | void appendArgument(const char *Name, const clang::FileEntry *Value); | 
|  |  | 
|  | /// \brief Append a SourceLocation argument to the top trace item. | 
|  | void appendArgument(const char *Name, clang::SourceLocation Value); | 
|  |  | 
|  | /// \brief Append a SourceRange argument to the top trace item. | 
|  | void appendArgument(const char *Name, clang::SourceRange Value); | 
|  |  | 
|  | /// \brief Append a CharSourceRange argument to the top trace item. | 
|  | void appendArgument(const char *Name, clang::CharSourceRange Value); | 
|  |  | 
|  | /// \brief Append a ModuleIdPath argument to the top trace item. | 
|  | void appendArgument(const char *Name, clang::ModuleIdPath Value); | 
|  |  | 
|  | /// \brief Append an IdentifierInfo argument to the top trace item. | 
|  | void appendArgument(const char *Name, const clang::IdentifierInfo *Value); | 
|  |  | 
|  | /// \brief Append a MacroDirective argument to the top trace item. | 
|  | void appendArgument(const char *Name, const clang::MacroDirective *Value); | 
|  |  | 
|  | /// \brief Append a MacroDefinition argument to the top trace item. | 
|  | void appendArgument(const char *Name, const clang::MacroDefinition &Value); | 
|  |  | 
|  | /// \brief Append a MacroArgs argument to the top trace item. | 
|  | void appendArgument(const char *Name, const clang::MacroArgs *Value); | 
|  |  | 
|  | /// \brief Append a Module argument to the top trace item. | 
|  | void appendArgument(const char *Name, const clang::Module *Value); | 
|  |  | 
|  | /// \brief Append a double-quoted argument to the top trace item. | 
|  | void appendQuotedArgument(const char *Name, const std::string &Value); | 
|  |  | 
|  | /// \brief Append a double-quoted file path argument to the top trace item. | 
|  | void appendFilePathArgument(const char *Name, llvm::StringRef Value); | 
|  |  | 
|  | /// \brief Get the raw source string of the range. | 
|  | llvm::StringRef getSourceString(clang::CharSourceRange Range); | 
|  |  | 
|  | /// \brief Callback trace information. | 
|  | /// We use a reference so the trace will be preserved for the caller | 
|  | /// after this object is destructed. | 
|  | std::vector<CallbackCall> &CallbackCalls; | 
|  |  | 
|  | /// \brief Names of callbacks to ignore. | 
|  | llvm::SmallSet<std::string, 4> &Ignore; | 
|  |  | 
|  | /// \brief Inhibit trace while this is set. | 
|  | bool DisableTrace; | 
|  |  | 
|  | clang::Preprocessor &PP; | 
|  | }; | 
|  |  | 
|  | #endif // PPTRACE_PPCALLBACKSTRACKER_H |