| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325 | /* Distributed under the OSI-approved BSD 3-Clause License.  See accompanying   file Copyright.txt or https://cmake.org/licensing for details.  *//* This code was originally taken from part of the Clang-Tidy LLVM project and * modified for use with CMake under the following original license: *///===--- HeaderGuard.cpp - clang-tidy//-------------------------------------===////// 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 "UsePragmaOnceCheck.h"#include <algorithm>#include <cassert>#include <clang/Frontend/CompilerInstance.h>#include <clang/Lex/PPCallbacks.h>#include <clang/Lex/Preprocessor.h>#include <clang/Tooling/Tooling.h>#include <llvm/Support/Path.h>namespace clang {namespace tidy {namespace cmake {/// canonicalize a path by removing ./ and ../ components.static std::string cleanPath(StringRef Path){  SmallString<256> Result = Path;  llvm::sys::path::remove_dots(Result, true);  return std::string(Result.str());}namespace {// This class is a workaround for the fact that PPCallbacks doesn't give us the// location of the hash for an #ifndef, #define, or #endif, so we have to find// it ourselves. We can't lex backwards, and attempting to turn on the// preprocessor's backtracking functionality wreaks havoc, so we have to// instantiate a second lexer and lex all the way from the beginning of the// file. Cache the results of this lexing so that we don't have to do it more// times than needed.//// TODO: Upstream a change to LLVM to give us the location of the hash in// PPCallbacks so we don't have to do this workaround.class DirectiveCache{public:  DirectiveCache(Preprocessor* PP, FileID FID)    : PP(PP)    , FID(FID)  {    SourceManager& SM = this->PP->getSourceManager();    const FileEntry* Entry = SM.getFileEntryForID(FID);    assert(Entry && "Invalid FileID given");    Lexer MyLexer(FID, SM.getMemoryBufferForFileOrFake(Entry), SM,                  this->PP->getLangOpts());    Token Tok;    while (!MyLexer.LexFromRawLexer(Tok)) {      if (Tok.getKind() == tok::hash) {        assert(SM.getFileID(Tok.getLocation()) == this->FID &&               "Token FileID does not match passed FileID");        if (!this->HashLocs.empty()) {          assert(SM.getFileOffset(this->HashLocs.back()) <                   SM.getFileOffset(Tok.getLocation()) &&                 "Tokens in file are not in order");        }        this->HashLocs.push_back(Tok.getLocation());      }    }  }  SourceRange createRangeForIfndef(SourceLocation IfndefMacroTokLoc)  {    // The #ifndef of an include guard is likely near the beginning of the    // file, so search from the front.    return SourceRange(this->findPreviousHashFromFront(IfndefMacroTokLoc),                       IfndefMacroTokLoc);  }  SourceRange createRangeForDefine(SourceLocation DefineMacroTokLoc)  {    // The #define of an include guard is likely near the beginning of the    // file, so search from the front.    return SourceRange(this->findPreviousHashFromFront(DefineMacroTokLoc),                       DefineMacroTokLoc);  }  SourceRange createRangeForEndif(SourceLocation EndifLoc)  {    // The #endif of an include guard is likely near the end of the file, so    // search from the back.    return SourceRange(this->findPreviousHashFromBack(EndifLoc), EndifLoc);  }private:  Preprocessor* PP;  FileID FID;  SmallVector<SourceLocation> HashLocs;  SourceLocation findPreviousHashFromFront(SourceLocation Loc)  {    SourceManager& SM = this->PP->getSourceManager();    Loc = SM.getExpansionLoc(Loc);    assert(SM.getFileID(Loc) == this->FID &&           "Loc FileID does not match our FileID");    auto It = std::find_if(      this->HashLocs.begin(), this->HashLocs.end(),      [&SM, &Loc](const SourceLocation& OtherLoc) -> bool {        return SM.getFileOffset(OtherLoc) >= SM.getFileOffset(Loc);      });    assert(It != this->HashLocs.begin() &&           "No hash associated with passed Loc");    return *--It;  }  SourceLocation findPreviousHashFromBack(SourceLocation Loc)  {    SourceManager& SM = this->PP->getSourceManager();    Loc = SM.getExpansionLoc(Loc);    assert(SM.getFileID(Loc) == this->FID &&           "Loc FileID does not match our FileID");    auto It =      std::find_if(this->HashLocs.rbegin(), this->HashLocs.rend(),                   [&SM, &Loc](const SourceLocation& OtherLoc) -> bool {                     return SM.getFileOffset(OtherLoc) < SM.getFileOffset(Loc);                   });    assert(It != this->HashLocs.rend() &&           "No hash associated with passed Loc");    return *It;  }};class UsePragmaOncePPCallbacks : public PPCallbacks{public:  UsePragmaOncePPCallbacks(Preprocessor* PP, UsePragmaOnceCheck* Check)    : PP(PP)    , Check(Check)  {  }  void FileChanged(SourceLocation Loc, FileChangeReason Reason,                   SrcMgr::CharacteristicKind FileType,                   FileID PrevFID) override  {    // Record all files we enter. We'll need them to diagnose headers without    // guards.    SourceManager& SM = this->PP->getSourceManager();    if (Reason == EnterFile && FileType == SrcMgr::C_User) {      if (const FileEntry* FE = SM.getFileEntryForID(SM.getFileID(Loc))) {        std::string FileName = cleanPath(FE->getName());        this->Files[FileName] = FE;      }    }  }  void Ifndef(SourceLocation Loc, const Token& MacroNameTok,              const MacroDefinition& MD) override  {    if (MD) {      return;    }    // Record #ifndefs that succeeded. We also need the Location of the Name.    this->Ifndefs[MacroNameTok.getIdentifierInfo()] =      std::make_pair(Loc, MacroNameTok.getLocation());  }  void MacroDefined(const Token& MacroNameTok,                    const MacroDirective* MD) override  {    // Record all defined macros. We store the whole token to get info on the    // name later.    this->Macros.emplace_back(MacroNameTok, MD->getMacroInfo());  }  void Endif(SourceLocation Loc, SourceLocation IfLoc) override  {    // Record all #endif and the corresponding #ifs (including #ifndefs).    this->EndIfs[IfLoc] = Loc;  }  void EndOfMainFile() override  {    // Now that we have all this information from the preprocessor, use it!    SourceManager& SM = this->PP->getSourceManager();    for (const auto& MacroEntry : this->Macros) {      const MacroInfo* MI = MacroEntry.second;      // We use clang's header guard detection. This has the advantage of also      // emitting a warning for cases where a pseudo header guard is found but      // preceded by something blocking the header guard optimization.      if (!MI->isUsedForHeaderGuard()) {        continue;      }      const FileEntry* FE =        SM.getFileEntryForID(SM.getFileID(MI->getDefinitionLoc()));      std::string FileName = cleanPath(FE->getName());      this->Files.erase(FileName);      // Look up Locations for this guard.      SourceLocation Ifndef =        this->Ifndefs[MacroEntry.first.getIdentifierInfo()].second;      SourceLocation Define = MacroEntry.first.getLocation();      SourceLocation EndIf =        this          ->EndIfs[this->Ifndefs[MacroEntry.first.getIdentifierInfo()].first];      StringRef CurHeaderGuard =        MacroEntry.first.getIdentifierInfo()->getName();      std::vector<FixItHint> FixIts;      HeaderSearch& HeaderInfo = this->PP->getHeaderSearchInfo();      HeaderFileInfo& Info = HeaderInfo.getFileInfo(FE);      DirectiveCache Cache(this->PP, SM.getFileID(MI->getDefinitionLoc()));      SourceRange IfndefSrcRange = Cache.createRangeForIfndef(Ifndef);      SourceRange DefineSrcRange = Cache.createRangeForDefine(Define);      SourceRange EndifSrcRange = Cache.createRangeForEndif(EndIf);      if (Info.isPragmaOnce) {        FixIts.push_back(FixItHint::CreateRemoval(IfndefSrcRange));      } else {        FixIts.push_back(          FixItHint::CreateReplacement(IfndefSrcRange, "#pragma once"));      }      FixIts.push_back(FixItHint::CreateRemoval(DefineSrcRange));      FixIts.push_back(FixItHint::CreateRemoval(EndifSrcRange));      this->Check->diag(IfndefSrcRange.getBegin(), "use #pragma once")        << FixIts;    }    // Emit warnings for headers that are missing guards.    checkGuardlessHeaders();    clearAllState();  }  /// Looks for files that were visited but didn't have a header guard.  /// Emits a warning with fixits suggesting adding one.  void checkGuardlessHeaders()  {    // Look for header files that didn't have a header guard. Emit a warning    // and fix-its to add the guard.    // TODO: Insert the guard after top comments.    for (const auto& FE : this->Files) {      StringRef FileName = FE.getKey();      if (!Check->shouldSuggestToAddPragmaOnce(FileName)) {        continue;      }      SourceManager& SM = this->PP->getSourceManager();      FileID FID = SM.translateFile(FE.getValue());      SourceLocation StartLoc = SM.getLocForStartOfFile(FID);      if (StartLoc.isInvalid()) {        continue;      }      HeaderSearch& HeaderInfo = this->PP->getHeaderSearchInfo();      HeaderFileInfo& Info = HeaderInfo.getFileInfo(FE.second);      if (Info.isPragmaOnce) {        continue;      }      this->Check->diag(StartLoc, "use #pragma once")        << FixItHint::CreateInsertion(StartLoc, "#pragma once\n");    }  }private:  void clearAllState()  {    this->Macros.clear();    this->Files.clear();    this->Ifndefs.clear();    this->EndIfs.clear();  }  std::vector<std::pair<Token, const MacroInfo*>> Macros;  llvm::StringMap<const FileEntry*> Files;  std::map<const IdentifierInfo*, std::pair<SourceLocation, SourceLocation>>    Ifndefs;  std::map<SourceLocation, SourceLocation> EndIfs;  Preprocessor* PP;  UsePragmaOnceCheck* Check;};} // namespacevoid UsePragmaOnceCheck::storeOptions(ClangTidyOptions::OptionMap& Opts){  this->Options.store(Opts, "HeaderFileExtensions",                      RawStringHeaderFileExtensions);}void UsePragmaOnceCheck::registerPPCallbacks(const SourceManager& SM,                                             Preprocessor* PP,                                             Preprocessor* ModuleExpanderPP){  PP->addPPCallbacks(std::make_unique<UsePragmaOncePPCallbacks>(PP, this));}bool UsePragmaOnceCheck::shouldSuggestToAddPragmaOnce(StringRef FileName){  return utils::isFileExtension(FileName, this->HeaderFileExtensions);}} // namespace cmake} // namespace tidy} // namespace clang
 |