diff options
Diffstat (limited to 'llvm/lib/Transforms/Utils/Debugify.cpp')
| -rw-r--r-- | llvm/lib/Transforms/Utils/Debugify.cpp | 103 | 
1 files changed, 82 insertions, 21 deletions
| diff --git a/llvm/lib/Transforms/Utils/Debugify.cpp b/llvm/lib/Transforms/Utils/Debugify.cpp index 8f98d81a3d79..3e4d53c10dc9 100644 --- a/llvm/lib/Transforms/Utils/Debugify.cpp +++ b/llvm/lib/Transforms/Utils/Debugify.cpp @@ -20,6 +20,7 @@  #include "llvm/IR/Instructions.h"  #include "llvm/IR/IntrinsicInst.h"  #include "llvm/IR/Module.h" +#include "llvm/IR/PassInstrumentation.h"  #include "llvm/Pass.h"  #include "llvm/Support/CommandLine.h" @@ -198,6 +199,18 @@ bool llvm::applyDebugifyMetadata(    return true;  } +static bool applyDebugify(Function &F) { +  Module &M = *F.getParent(); +  auto FuncIt = F.getIterator(); +  return applyDebugifyMetadata(M, make_range(FuncIt, std::next(FuncIt)), +                               "FunctionDebugify: ", /*ApplyToMF=*/nullptr); +} + +static bool applyDebugify(Module &M) { +  return applyDebugifyMetadata(M, M.functions(), +                               "ModuleDebugify: ", /*ApplyToMF=*/nullptr); +} +  bool llvm::stripDebugifyMetadata(Module &M) {    bool Changed = false; @@ -226,9 +239,7 @@ bool llvm::stripDebugifyMetadata(Module &M) {    NamedMDNode *NMD = M.getModuleFlagsMetadata();    if (!NMD)      return Changed; -  SmallVector<MDNode *, 4> Flags; -  for (MDNode *Flag : NMD->operands()) -    Flags.push_back(Flag); +  SmallVector<MDNode *, 4> Flags(NMD->operands());    NMD->clearOperands();    for (MDNode *Flag : Flags) {      MDString *Key = dyn_cast_or_null<MDString>(Flag->getOperand(1)); @@ -383,10 +394,7 @@ bool checkDebugifyMetadata(Module &M,  /// ModulePass for attaching synthetic debug info to everything, used with the  /// legacy module pass manager.  struct DebugifyModulePass : public ModulePass { -  bool runOnModule(Module &M) override { -    return applyDebugifyMetadata(M, M.functions(), -                                 "ModuleDebugify: ", /*ApplyToMF*/ nullptr); -  } +  bool runOnModule(Module &M) override { return applyDebugify(M); }    DebugifyModulePass() : ModulePass(ID) {} @@ -400,12 +408,7 @@ struct DebugifyModulePass : public ModulePass {  /// FunctionPass for attaching synthetic debug info to instructions within a  /// single function, used with the legacy module pass manager.  struct DebugifyFunctionPass : public FunctionPass { -  bool runOnFunction(Function &F) override { -    Module &M = *F.getParent(); -    auto FuncIt = F.getIterator(); -    return applyDebugifyMetadata(M, make_range(FuncIt, std::next(FuncIt)), -                                 "FunctionDebugify: ", /*ApplyToMF*/ nullptr); -  } +  bool runOnFunction(Function &F) override { return applyDebugify(F); }    DebugifyFunctionPass() : FunctionPass(ID) {} @@ -472,9 +475,32 @@ private:  } // end anonymous namespace -ModulePass *createDebugifyModulePass() { return new DebugifyModulePass(); } +void llvm::exportDebugifyStats(StringRef Path, const DebugifyStatsMap &Map) { +  std::error_code EC; +  raw_fd_ostream OS{Path, EC}; +  if (EC) { +    errs() << "Could not open file: " << EC.message() << ", " << Path << '\n'; +    return; +  } + +  OS << "Pass Name" << ',' << "# of missing debug values" << ',' +     << "# of missing locations" << ',' << "Missing/Expected value ratio" << ',' +     << "Missing/Expected location ratio" << '\n'; +  for (const auto &Entry : Map) { +    StringRef Pass = Entry.first; +    DebugifyStatistics Stats = Entry.second; + +    OS << Pass << ',' << Stats.NumDbgValuesMissing << ',' +       << Stats.NumDbgLocsMissing << ',' << Stats.getMissingValueRatio() << ',' +       << Stats.getEmptyLocationRatio() << '\n'; +  } +} + +ModulePass *llvm::createDebugifyModulePass() { +  return new DebugifyModulePass(); +} -FunctionPass *createDebugifyFunctionPass() { +FunctionPass *llvm::createDebugifyFunctionPass() {    return new DebugifyFunctionPass();  } @@ -484,15 +510,15 @@ PreservedAnalyses NewPMDebugifyPass::run(Module &M, ModuleAnalysisManager &) {    return PreservedAnalyses::all();  } -ModulePass *createCheckDebugifyModulePass(bool Strip, -                                          StringRef NameOfWrappedPass, -                                          DebugifyStatsMap *StatsMap) { +ModulePass *llvm::createCheckDebugifyModulePass(bool Strip, +                                                StringRef NameOfWrappedPass, +                                                DebugifyStatsMap *StatsMap) {    return new CheckDebugifyModulePass(Strip, NameOfWrappedPass, StatsMap);  } -FunctionPass *createCheckDebugifyFunctionPass(bool Strip, -                                              StringRef NameOfWrappedPass, -                                              DebugifyStatsMap *StatsMap) { +FunctionPass * +llvm::createCheckDebugifyFunctionPass(bool Strip, StringRef NameOfWrappedPass, +                                      DebugifyStatsMap *StatsMap) {    return new CheckDebugifyFunctionPass(Strip, NameOfWrappedPass, StatsMap);  } @@ -503,6 +529,41 @@ PreservedAnalyses NewPMCheckDebugifyPass::run(Module &M,    return PreservedAnalyses::all();  } +static bool isIgnoredPass(StringRef PassID) { +  return isSpecialPass(PassID, {"PassManager", "PassAdaptor", +                                "AnalysisManagerProxy", "PrintFunctionPass", +                                "PrintModulePass", "BitcodeWriterPass", +                                "ThinLTOBitcodeWriterPass", "VerifierPass"}); +} + +void DebugifyEachInstrumentation::registerCallbacks( +    PassInstrumentationCallbacks &PIC) { +  PIC.registerBeforeNonSkippedPassCallback([](StringRef P, Any IR) { +    if (isIgnoredPass(P)) +      return; +    if (any_isa<const Function *>(IR)) +      applyDebugify(*const_cast<Function *>(any_cast<const Function *>(IR))); +    else if (any_isa<const Module *>(IR)) +      applyDebugify(*const_cast<Module *>(any_cast<const Module *>(IR))); +  }); +  PIC.registerAfterPassCallback([this](StringRef P, Any IR, +                                       const PreservedAnalyses &PassPA) { +    if (isIgnoredPass(P)) +      return; +    if (any_isa<const Function *>(IR)) { +      auto &F = *const_cast<Function *>(any_cast<const Function *>(IR)); +      Module &M = *F.getParent(); +      auto It = F.getIterator(); +      checkDebugifyMetadata(M, make_range(It, std::next(It)), P, +                            "CheckFunctionDebugify", /*Strip=*/true, &StatsMap); +    } else if (any_isa<const Module *>(IR)) { +      auto &M = *const_cast<Module *>(any_cast<const Module *>(IR)); +      checkDebugifyMetadata(M, M.functions(), P, "CheckModuleDebugify", +                            /*Strip=*/true, &StatsMap); +    } +  }); +} +  char DebugifyModulePass::ID = 0;  static RegisterPass<DebugifyModulePass> DM("debugify",                                             "Attach debug info to everything"); | 
