Index: clang/lib/CodeGen/BackendUtil.cpp =================================================================== --- clang/lib/CodeGen/BackendUtil.cpp +++ clang/lib/CodeGen/BackendUtil.cpp @@ -784,6 +784,18 @@ SI.registerCallbacks(PIC, &FAM); PassBuilder PB(TM.get(), PTO, PGOOpt, &PIC); + // Enable verify-debuginfo-preserve-each for new PM. + DebugifyEachInstrumentation Debugify; + DebugInfoPerPass DebugInfoBeforePass; + if (CodeGenOpts.EnableDIPreservationVerify) { + Debugify.setDebugifyMode(DebugifyMode::OriginalDebugInfo); + Debugify.setDebugInfoBeforePass(DebugInfoBeforePass); + + if (!CodeGenOpts.DIBugsReportFilePath.empty()) + Debugify.setOrigDIVerifyBugsReportFilePath( + CodeGenOpts.DIBugsReportFilePath); + Debugify.registerCallbacks(PIC); + } // Attempt to load pass plugins and register their callbacks with PB. for (auto &PluginFN : CodeGenOpts.PassPlugins) { auto PassPlugin = PassPlugin::Load(PluginFN); Index: clang/test/Driver/verify-debug-info-preservation.c =================================================================== --- clang/test/Driver/verify-debug-info-preservation.c +++ clang/test/Driver/verify-debug-info-preservation.c @@ -4,16 +4,26 @@ // RUN: %clang -g -Xclang -fverify-debuginfo-preserve -### %s 2>&1 \ // RUN: | FileCheck --check-prefix=VERIFYDIPRESERVE %s +// RUN: %clang -g -Xclang -fverify-debuginfo-preserve -### %s -flegacy-pass-manager 2>&1 \ +// RUN: | FileCheck --check-prefix=VERIFYDIPRESERVE %s + // VERIFYDIPRESERVE: "-fverify-debuginfo-preserve" // RUN: %clang -g -Xclang -fverify-debuginfo-preserve \ // RUN: -Xclang -fverify-debuginfo-preserve-export=%t.json -### %s 2>&1 \ // RUN: | FileCheck --check-prefix=VERIFYDIPRESERVE-JSON-EXPORT %s +// RUN: %clang -g -Xclang -fverify-debuginfo-preserve -flegacy-pass-manager \ +// RUN: -Xclang -fverify-debuginfo-preserve-export=%t.json -### %s 2>&1 \ +// RUN: | FileCheck --check-prefix=VERIFYDIPRESERVE-JSON-EXPORT %s + // VERIFYDIPRESERVE-JSON-EXPORT: "-fverify-debuginfo-preserve" // VERIFYDIPRESERVE-JSON-EXPORT: "-fverify-debuginfo-preserve-export={{.*}}" // RUN: %clang -g -Xclang -fverify-debuginfo-preserve-export=%t.json %s -S -o /dev/null 2>&1 \ // RUN: | FileCheck --check-prefix=WARN %s +// RUN: %clang -g -Xclang -fverify-debuginfo-preserve-export=%t.json -flegacy-pass-manager %s -S -o /dev/null 2>&1 \ +// RUN: | FileCheck --check-prefix=WARN %s + // WARN: warning: ignoring -fverify-debuginfo-preserve-export Index: llvm/include/llvm/Transforms/Utils/Debugify.h =================================================================== --- llvm/include/llvm/Transforms/Utils/Debugify.h +++ llvm/include/llvm/Transforms/Utils/Debugify.h @@ -101,7 +101,18 @@ llvm::StringRef NameOfWrappedPass = "", DebugInfoPerPass *DebugInfoBeforePass = nullptr); -struct NewPMDebugifyPass : public llvm::PassInfoMixin { +class NewPMDebugifyPass : public llvm::PassInfoMixin { + llvm::StringRef NameOfWrappedPass; + DebugInfoPerPass *DebugInfoBeforePass = nullptr; + enum DebugifyMode Mode = DebugifyMode::NoDebugify; +public: + NewPMDebugifyPass( + enum DebugifyMode Mode = DebugifyMode::SyntheticDebugInfo, + llvm::StringRef NameOfWrappedPass = "", + DebugInfoPerPass *DebugInfoBeforePass = nullptr) + : NameOfWrappedPass(NameOfWrappedPass), + DebugInfoBeforePass(DebugInfoBeforePass), Mode(Mode) {} + llvm::PreservedAnalyses run(llvm::Module &M, llvm::ModuleAnalysisManager &AM); }; @@ -148,18 +159,65 @@ DebugInfoPerPass *DebugInfoBeforePass = nullptr, llvm::StringRef OrigDIVerifyBugsReportFilePath = ""); -struct NewPMCheckDebugifyPass +class NewPMCheckDebugifyPass : public llvm::PassInfoMixin { + llvm::StringRef NameOfWrappedPass; + llvm::StringRef OrigDIVerifyBugsReportFilePath; + DebugifyStatsMap *StatsMap; + DebugInfoPerPass *DebugInfoBeforePass; + enum DebugifyMode Mode; + bool Strip; +public: + NewPMCheckDebugifyPass( + bool Strip = false, llvm::StringRef NameOfWrappedPass = "", + DebugifyStatsMap *StatsMap = nullptr, + enum DebugifyMode Mode = DebugifyMode::SyntheticDebugInfo, + DebugInfoPerPass *DebugInfoBeforePass = nullptr, + llvm::StringRef OrigDIVerifyBugsReportFilePath = "") + : NameOfWrappedPass(NameOfWrappedPass), + OrigDIVerifyBugsReportFilePath(OrigDIVerifyBugsReportFilePath), + StatsMap(StatsMap), DebugInfoBeforePass(DebugInfoBeforePass), Mode(Mode), + Strip(Strip) {} + llvm::PreservedAnalyses run(llvm::Module &M, llvm::ModuleAnalysisManager &AM); }; namespace llvm { void exportDebugifyStats(StringRef Path, const DebugifyStatsMap &Map); -struct DebugifyEachInstrumentation { - DebugifyStatsMap StatsMap; +class DebugifyEachInstrumentation { + llvm::StringRef OrigDIVerifyBugsReportFilePath = ""; + DebugInfoPerPass *DebugInfoBeforePass = nullptr; + enum DebugifyMode Mode = DebugifyMode::NoDebugify; + DebugifyStatsMap *DIStatsMap = nullptr; + +public: void registerCallbacks(PassInstrumentationCallbacks &PIC); + // Used within DebugifyMode::SyntheticDebugInfo mode. + void setDIStatsMap(DebugifyStatsMap &StatMap) { DIStatsMap = &StatMap; } + const DebugifyStatsMap &getDebugifyStatsMap() const { return *DIStatsMap; } + // Used within DebugifyMode::OriginalDebugInfo mode. + void setDebugInfoBeforePass(DebugInfoPerPass &PerPassMap) { + DebugInfoBeforePass = &PerPassMap; + } + DebugInfoPerPass &getDebugInfoPerPass() { return *DebugInfoBeforePass; } + + void setOrigDIVerifyBugsReportFilePath(StringRef BugsReportFilePath) { + OrigDIVerifyBugsReportFilePath = BugsReportFilePath; + } + StringRef getOrigDIVerifyBugsReportFilePath() const { + return OrigDIVerifyBugsReportFilePath; + } + + void setDebugifyMode(enum DebugifyMode M) { Mode = M; } + + bool isSyntheticDebugInfo() const { + return Mode == DebugifyMode::SyntheticDebugInfo; + } + bool isOriginalDebugInfoMode() const { + return Mode == DebugifyMode::OriginalDebugInfo; + } }; /// DebugifyCustomPassManager wraps each pass with the debugify passes if Index: llvm/lib/Transforms/Utils/Debugify.cpp =================================================================== --- llvm/lib/Transforms/Utils/Debugify.cpp +++ llvm/lib/Transforms/Utils/Debugify.cpp @@ -949,8 +949,13 @@ } PreservedAnalyses NewPMDebugifyPass::run(Module &M, ModuleAnalysisManager &) { - applyDebugifyMetadata(M, M.functions(), - "ModuleDebugify: ", /*ApplyToMF*/ nullptr); + if (Mode == DebugifyMode::SyntheticDebugInfo) + applyDebugifyMetadata(M, M.functions(), + "ModuleDebugify: ", /*ApplyToMF*/ nullptr); + else + collectDebugInfoMetadata(M, M.functions(), *DebugInfoBeforePass, + "ModuleDebugify (original debuginfo)", + NameOfWrappedPass); return PreservedAnalyses::all(); } @@ -980,8 +985,14 @@ PreservedAnalyses NewPMCheckDebugifyPass::run(Module &M, ModuleAnalysisManager &) { - checkDebugifyMetadata(M, M.functions(), "", "CheckModuleDebugify", false, - nullptr); + if (Mode == DebugifyMode::SyntheticDebugInfo) + checkDebugifyMetadata(M, M.functions(), NameOfWrappedPass, + "CheckModuleDebugify", Strip, StatsMap); + else + checkDebugInfoMetadata( + M, M.functions(), *DebugInfoBeforePass, + "CheckModuleDebugify (original debuginfo)", NameOfWrappedPass, + OrigDIVerifyBugsReportFilePath); return PreservedAnalyses::all(); } @@ -994,13 +1005,15 @@ void DebugifyEachInstrumentation::registerCallbacks( PassInstrumentationCallbacks &PIC) { - PIC.registerBeforeNonSkippedPassCallback([](StringRef P, Any IR) { + PIC.registerBeforeNonSkippedPassCallback([this](StringRef P, Any IR) { if (isIgnoredPass(P)) return; if (any_isa(IR)) - applyDebugify(*const_cast(any_cast(IR))); + applyDebugify(*const_cast(any_cast(IR)), + Mode, DebugInfoBeforePass, P); else if (any_isa(IR)) - applyDebugify(*const_cast(any_cast(IR))); + applyDebugify(*const_cast(any_cast(IR)), + Mode, DebugInfoBeforePass, P); }); PIC.registerAfterPassCallback([this](StringRef P, Any IR, const PreservedAnalyses &PassPA) { @@ -1010,12 +1023,24 @@ auto &F = *const_cast(any_cast(IR)); Module &M = *F.getParent(); auto It = F.getIterator(); - checkDebugifyMetadata(M, make_range(It, std::next(It)), P, - "CheckFunctionDebugify", /*Strip=*/true, &StatsMap); + if (Mode == DebugifyMode::SyntheticDebugInfo) + checkDebugifyMetadata(M, make_range(It, std::next(It)), P, + "CheckFunctionDebugify", /*Strip=*/true, DIStatsMap); + else + checkDebugInfoMetadata( + M, make_range(It, std::next(It)), *DebugInfoBeforePass, + "CheckModuleDebugify (original debuginfo)", + P, OrigDIVerifyBugsReportFilePath); } else if (any_isa(IR)) { auto &M = *const_cast(any_cast(IR)); - checkDebugifyMetadata(M, M.functions(), P, "CheckModuleDebugify", - /*Strip=*/true, &StatsMap); + if (Mode == DebugifyMode::SyntheticDebugInfo) + checkDebugifyMetadata(M, M.functions(), P, "CheckModuleDebugify", + /*Strip=*/true, DIStatsMap); + else + checkDebugInfoMetadata( + M, M.functions(), *DebugInfoBeforePass, + "CheckModuleDebugify (original debuginfo)", + P, OrigDIVerifyBugsReportFilePath); } }); } Index: llvm/test/DebugInfo/verify-di-preserve.ll =================================================================== --- /dev/null +++ llvm/test/DebugInfo/verify-di-preserve.ll @@ -0,0 +1,62 @@ +; RUN: opt %s -verify-debuginfo-preserve -instcombine -disable-output 2>&1 | FileCheck --check-prefix=VERIFY %s + +; VERIFY: CheckModuleDebugify (original debuginfo): + +; RUN: opt %s -verify-debuginfo-preserve -instcombine -enable-new-pm=false -disable-output 2>&1 | FileCheck --check-prefix=VERIFY-LEGACY-PM %s + +; VERIFY-LEGACY-PM: CheckModuleDebugify (original debuginfo): + + +; RUN: opt %s -verify-each-debuginfo-preserve -O2 -disable-output 2>&1 | FileCheck --check-prefix=VERIFY-EACH %s + +; VERIFY-EACH: DeadArgumentEliminationPass +; VERIFY-EACH: GlobalDCEPass + +; RUN: opt %s -verify-each-debuginfo-preserve -O2 -enable-new-pm=false -disable-output 2>&1 | FileCheck --check-prefix=VERIFY-EACH-LEGACY-PM %s + +; VERIFY-EACH-LEGACY-PM: Dead Argument Elimination +; VERIFY-EACH-LEGACY-PM: Dead Global Elimination + +; ModuleID = 'm.c' +source_filename = "m.c" +target datalayout = "e-m:e-p270:32:32-p271:32:32-p272:64:64-i64:64-f80:128-n8:16:32:64-S128" +target triple = "x86_64-unknown-linux-gnu" + +; Function Attrs: noinline nounwind optnone uwtable +define dso_local i32 @foo(i32 %i) !dbg !8 { +entry: + %i.addr = alloca i32, align 4 + store i32 %i, i32* %i.addr, align 4 + call void @llvm.dbg.declare(metadata i32* %i.addr, metadata !13, metadata !DIExpression()), !dbg !14 + %0 = load i32, i32* %i.addr, align 4, !dbg !15 + %call = call i32 @goo(i32 %0), !dbg !16 + ret i32 %call, !dbg !17 +} + +; Function Attrs: nofree nosync nounwind readnone speculatable willreturn +declare void @llvm.dbg.declare(metadata, metadata, metadata) + +declare dso_local i32 @goo(i32) + +!llvm.dbg.cu = !{!0} +!llvm.module.flags = !{!2, !3, !4, !5, !6} +!llvm.ident = !{!7} + +!0 = distinct !DICompileUnit(language: DW_LANG_C99, file: !1, producer: "clang version 14.0.0", isOptimized: false, runtimeVersion: 0, emissionKind: FullDebug, splitDebugInlining: false, nameTableKind: None) +!1 = !DIFile(filename: "m.c", directory: "/dir") +!2 = !{i32 7, !"Dwarf Version", i32 4} +!3 = !{i32 2, !"Debug Info Version", i32 3} +!4 = !{i32 1, !"wchar_size", i32 4} +!5 = !{i32 7, !"uwtable", i32 1} +!6 = !{i32 7, !"frame-pointer", i32 2} +!7 = !{!"clang version 14.0.0"} +!8 = distinct !DISubprogram(name: "foo", scope: !1, file: !1, line: 2, type: !9, scopeLine: 2, flags: DIFlagPrototyped, spFlags: DISPFlagDefinition, unit: !0, retainedNodes: !12) +!9 = !DISubroutineType(types: !10) +!10 = !{!11, !11} +!11 = !DIBasicType(name: "int", size: 32, encoding: DW_ATE_signed) +!12 = !{} +!13 = !DILocalVariable(name: "i", arg: 1, scope: !8, file: !1, line: 2, type: !11) +!14 = !DILocation(line: 2, column: 13, scope: !8) +!15 = !DILocation(line: 3, column: 14, scope: !8) +!16 = !DILocation(line: 3, column: 10, scope: !8) +!17 = !DILocation(line: 3, column: 3, scope: !8) Index: llvm/tools/opt/NewPMDriver.h =================================================================== --- llvm/tools/opt/NewPMDriver.h +++ llvm/tools/opt/NewPMDriver.h @@ -34,6 +34,9 @@ extern cl::opt DebugifyEach; extern cl::opt DebugifyExport; +extern cl::opt VerifyEachDebugInfoPreserve; +extern cl::opt VerifyDIPreserveExport; + namespace opt_tool { enum OutputKind { OK_NoOutput, @@ -75,7 +78,7 @@ bool ShouldPreserveAssemblyUseListOrder, bool ShouldPreserveBitcodeUseListOrder, bool EmitSummaryIndex, bool EmitModuleHash, - bool EnableDebugify); + bool EnableDebugify, bool VerifyDIPreserve); } // namespace llvm #endif Index: llvm/tools/opt/NewPMDriver.cpp =================================================================== --- llvm/tools/opt/NewPMDriver.cpp +++ llvm/tools/opt/NewPMDriver.cpp @@ -49,6 +49,19 @@ DebugifyExport("debugify-export", cl::desc("Export per-pass debugify statistics to this file"), cl::value_desc("filename")); + +cl::opt VerifyEachDebugInfoPreserve( + "verify-each-debuginfo-preserve", + cl::desc("Start each pass with collecting and end it with checking of " + "debug info preservation.")); + +cl::opt + VerifyDIPreserveExport("verify-di-preserve-export", + cl::desc("Export debug info preservation failures into " + "specified (JSON) file (should be abs path as we use" + " append mode to insert new JSON objects)"), + cl::value_desc("filename"), cl::init("")); + } // namespace llvm enum class DebugLogging { None, Normal, Verbose, Quiet }; @@ -280,7 +293,7 @@ bool ShouldPreserveAssemblyUseListOrder, bool ShouldPreserveBitcodeUseListOrder, bool EmitSummaryIndex, bool EmitModuleHash, - bool EnableDebugify) { + bool EnableDebugify, bool VerifyDIPreserve) { bool VerifyEachPass = VK == VK_VerifyEachPass; Optional P; @@ -337,8 +350,19 @@ PrintPassOpts); SI.registerCallbacks(PIC, &FAM); DebugifyEachInstrumentation Debugify; - if (DebugifyEach) + DebugifyStatsMap DIStatsMap; + DebugInfoPerPass DebugInfoBeforePass; + if (DebugifyEach) { + Debugify.setDIStatsMap(DIStatsMap); + Debugify.setDebugifyMode(DebugifyMode::SyntheticDebugInfo); + Debugify.registerCallbacks(PIC); + } else if (VerifyEachDebugInfoPreserve) { + Debugify.setDebugInfoBeforePass(DebugInfoBeforePass); + Debugify.setDebugifyMode(DebugifyMode::OriginalDebugInfo); + Debugify.setOrigDIVerifyBugsReportFilePath( + VerifyDIPreserveExport); Debugify.registerCallbacks(PIC); + } PipelineTuningOptions PTO; // LoopUnrolling defaults on to true and DisableLoopUnrolling is initialized @@ -419,6 +443,9 @@ MPM.addPass(VerifierPass()); if (EnableDebugify) MPM.addPass(NewPMDebugifyPass()); + if (VerifyDIPreserve) + MPM.addPass(NewPMDebugifyPass(DebugifyMode::OriginalDebugInfo, "", + &DebugInfoBeforePass)); // Add passes according to the -passes options. if (!PassPipeline.empty()) { @@ -458,7 +485,11 @@ if (VK > VK_NoVerifier) MPM.addPass(VerifierPass()); if (EnableDebugify) - MPM.addPass(NewPMCheckDebugifyPass()); + MPM.addPass(NewPMCheckDebugifyPass(false, "", &DIStatsMap)); + if (VerifyDIPreserve) + MPM.addPass(NewPMCheckDebugifyPass( + false, "", nullptr, DebugifyMode::OriginalDebugInfo, &DebugInfoBeforePass, + VerifyDIPreserveExport)); // Add any relevant output pass at the end of the pipeline. switch (OK) { @@ -506,7 +537,7 @@ OptRemarkFile->keep(); if (DebugifyEach && !DebugifyExport.empty()) - exportDebugifyStats(DebugifyExport, Debugify.StatsMap); + exportDebugifyStats(DebugifyExport, Debugify.getDebugifyStatsMap()); return true; } Index: llvm/tools/opt/opt.cpp =================================================================== --- llvm/tools/opt/opt.cpp +++ llvm/tools/opt/opt.cpp @@ -208,18 +208,6 @@ cl::desc("Start the pipeline with collecting and end it with checking of " "debug info preservation.")); -static cl::opt VerifyEachDebugInfoPreserve( - "verify-each-debuginfo-preserve", - cl::desc("Start each pass with collecting and end it with checking of " - "debug info preservation.")); - -static cl::opt - VerifyDIPreserveExport("verify-di-preserve-export", - cl::desc("Export debug info preservation failures into " - "specified (JSON) file (should be abs path as we use" - " append mode to insert new JSON objects)"), - cl::value_desc("filename"), cl::init("")); - static cl::opt PrintBreakpoints("print-breakpoints-for-testing", cl::desc("Print select breakpoints location for testing")); @@ -831,7 +819,8 @@ ThinLinkOut.get(), RemarksFile.get(), Pipeline, Passes, PluginList, OK, VK, PreserveAssemblyUseListOrder, PreserveBitcodeUseListOrder, EmitSummaryIndex, - EmitModuleHash, EnableDebugify) + EmitModuleHash, EnableDebugify, + VerifyDebugInfoPreserve) ? 0 : 1; }