//===--- ProfileList.h - ProfileList filter ---------------------*- 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 // //===----------------------------------------------------------------------===// // // User-provided filters include/exclude profile instrumentation in certain // functions or files. // //===----------------------------------------------------------------------===// #include "clang/Basic/ProfileList.h" #include "clang/Basic/FileManager.h" #include "clang/Basic/SourceManager.h" #include "llvm/Support/SpecialCaseList.h" #include "llvm/Support/raw_ostream.h" #include using namespace clang; namespace clang { class ProfileSpecialCaseList : public llvm::SpecialCaseList { public: static std::unique_ptr create(const std::vector &Paths, llvm::vfs::FileSystem &VFS, std::string &Error); static std::unique_ptr createOrDie(const std::vector &Paths, llvm::vfs::FileSystem &VFS); bool isEmpty() const { return Sections.empty(); } bool hasPrefix(StringRef Prefix) const { for (auto &SectionIter : Sections) if (SectionIter.Entries.count(Prefix) > 0) return true; return false; } }; std::unique_ptr ProfileSpecialCaseList::create(const std::vector &Paths, llvm::vfs::FileSystem &VFS, std::string &Error) { auto PSCL = std::make_unique(); if (PSCL->createInternal(Paths, VFS, Error)) return PSCL; return nullptr; } std::unique_ptr ProfileSpecialCaseList::createOrDie(const std::vector &Paths, llvm::vfs::FileSystem &VFS) { std::string Error; if (auto PSCL = create(Paths, VFS, Error)) return PSCL; llvm::report_fatal_error(llvm::Twine(Error)); } } ProfileList::ProfileList(ArrayRef Paths, SourceManager &SM) : SCL(ProfileSpecialCaseList::createOrDie( Paths, SM.getFileManager().getVirtualFileSystem())), Empty(SCL->isEmpty()), SM(SM) {} ProfileList::~ProfileList() = default; static StringRef getSectionName(CodeGenOptions::ProfileInstrKind Kind) { switch (Kind) { case CodeGenOptions::ProfileNone: return ""; case CodeGenOptions::ProfileClangInstr: return "clang"; case CodeGenOptions::ProfileIRInstr: return "llvm"; case CodeGenOptions::ProfileCSIRInstr: return "csllvm"; } llvm_unreachable("Unhandled CodeGenOptions::ProfileInstrKind enum"); } ProfileList::ExclusionType ProfileList::getDefault(CodeGenOptions::ProfileInstrKind Kind) const { StringRef Section = getSectionName(Kind); // Check for "default:" if (SCL->inSection(Section, "default", "allow")) return Allow; if (SCL->inSection(Section, "default", "skip")) return Skip; if (SCL->inSection(Section, "default", "forbid")) return Forbid; // If any cases use "fun" or "src", set the default to FORBID. if (SCL->hasPrefix("fun") || SCL->hasPrefix("src")) return Forbid; return Allow; } std::optional ProfileList::inSection(StringRef Section, StringRef Prefix, StringRef Query) const { if (SCL->inSection(Section, Prefix, Query, "allow")) return Allow; if (SCL->inSection(Section, Prefix, Query, "skip")) return Skip; if (SCL->inSection(Section, Prefix, Query, "forbid")) return Forbid; if (SCL->inSection(Section, Prefix, Query)) return Allow; return std::nullopt; } std::optional ProfileList::isFunctionExcluded(StringRef FunctionName, CodeGenOptions::ProfileInstrKind Kind) const { StringRef Section = getSectionName(Kind); // Check for "function:=" if (auto V = inSection(Section, "function", FunctionName)) return V; if (SCL->inSection(Section, "!fun", FunctionName)) return Forbid; if (SCL->inSection(Section, "fun", FunctionName)) return Allow; return std::nullopt; } std::optional ProfileList::isLocationExcluded(SourceLocation Loc, CodeGenOptions::ProfileInstrKind Kind) const { return isFileExcluded(SM.getFilename(SM.getFileLoc(Loc)), Kind); } std::optional ProfileList::isFileExcluded(StringRef FileName, CodeGenOptions::ProfileInstrKind Kind) const { StringRef Section = getSectionName(Kind); // Check for "source:=" if (auto V = inSection(Section, "source", FileName)) return V; if (SCL->inSection(Section, "!src", FileName)) return Forbid; if (SCL->inSection(Section, "src", FileName)) return Allow; return std::nullopt; }