25
25
#include " llvm/IR/PassInstrumentation.h"
26
26
#include " llvm/Pass.h"
27
27
#include " llvm/Support/CommandLine.h"
28
+ #include " llvm/Support/JSON.h"
28
29
29
30
#define DEBUG_TYPE " debugify"
30
31
@@ -334,25 +335,36 @@ bool llvm::collectDebugInfoMetadata(Module &M,
334
335
static bool checkFunctions (const DebugFnMap &DIFunctionsBefore,
335
336
const DebugFnMap &DIFunctionsAfter,
336
337
StringRef NameOfWrappedPass,
337
- StringRef FileNameFromCU) {
338
+ StringRef FileNameFromCU, bool ShouldWriteIntoJSON,
339
+ llvm::json::Array &Bugs) {
338
340
bool Preserved = true ;
339
341
for (const auto &F : DIFunctionsAfter) {
340
342
if (F.second )
341
343
continue ;
342
344
auto SPIt = DIFunctionsBefore.find (F.first );
343
345
if (SPIt == DIFunctionsBefore.end ()) {
344
- dbg () << " ERROR: " << NameOfWrappedPass
345
- << " did not generate DISubprogram for " << F.first << " from "
346
- << FileNameFromCU << ' \n ' ;
346
+ if (ShouldWriteIntoJSON)
347
+ Bugs.push_back (llvm::json::Object ({{" metadata" , " DISubprogram" },
348
+ {" name" , F.first },
349
+ {" action" , " not-generate" }}));
350
+ else
351
+ dbg () << " ERROR: " << NameOfWrappedPass
352
+ << " did not generate DISubprogram for " << F.first << " from "
353
+ << FileNameFromCU << ' \n ' ;
347
354
Preserved = false ;
348
355
} else {
349
356
auto SP = SPIt->second ;
350
357
if (!SP)
351
358
continue ;
352
359
// If the function had the SP attached before the pass, consider it as
353
360
// a debug info bug.
354
- dbg () << " ERROR: " << NameOfWrappedPass << " dropped DISubprogram of "
355
- << F.first << " from " << FileNameFromCU << ' \n ' ;
361
+ if (ShouldWriteIntoJSON)
362
+ Bugs.push_back (llvm::json::Object ({{" metadata" , " DISubprogram" },
363
+ {" name" , F.first },
364
+ {" action" , " drop" }}));
365
+ else
366
+ dbg () << " ERROR: " << NameOfWrappedPass << " dropped DISubprogram of "
367
+ << F.first << " from " << FileNameFromCU << ' \n ' ;
356
368
Preserved = false ;
357
369
}
358
370
}
@@ -366,7 +378,9 @@ static bool checkInstructions(const DebugInstMap &DILocsBefore,
366
378
const DebugInstMap &DILocsAfter,
367
379
const WeakInstValueMap &InstToDelete,
368
380
StringRef NameOfWrappedPass,
369
- StringRef FileNameFromCU) {
381
+ StringRef FileNameFromCU,
382
+ bool ShouldWriteIntoJSON,
383
+ llvm::json::Array &Bugs) {
370
384
bool Preserved = true ;
371
385
for (const auto &L : DILocsAfter) {
372
386
if (L.second )
@@ -382,34 +396,73 @@ static bool checkInstructions(const DebugInstMap &DILocsBefore,
382
396
auto FnName = Instr->getFunction ()->getName ();
383
397
auto BB = Instr->getParent ();
384
398
auto BBName = BB->hasName () ? BB->getName () : " no-name" ;
399
+ auto InstName = Instruction::getOpcodeName (Instr->getOpcode ());
385
400
386
401
auto InstrIt = DILocsBefore.find (Instr);
387
402
if (InstrIt == DILocsBefore.end ()) {
388
- dbg () << " WARNING: " << NameOfWrappedPass
389
- << " did not generate DILocation for " << *Instr
390
- << " (BB: " << BBName << " , Fn: " << FnName
391
- << " , File: " << FileNameFromCU << " )\n " ;
403
+ if (ShouldWriteIntoJSON)
404
+ Bugs.push_back (llvm::json::Object ({{" metadata" , " DILocation" },
405
+ {" fn-name" , FnName.str ()},
406
+ {" bb-name" , BBName.str ()},
407
+ {" instr" , InstName},
408
+ {" action" , " not-generate" }}));
409
+ else
410
+ dbg () << " WARNING: " << NameOfWrappedPass
411
+ << " did not generate DILocation for " << *Instr
412
+ << " (BB: " << BBName << " , Fn: " << FnName
413
+ << " , File: " << FileNameFromCU << " )\n " ;
392
414
Preserved = false ;
393
415
} else {
394
416
if (!InstrIt->second )
395
417
continue ;
396
418
// If the instr had the !dbg attached before the pass, consider it as
397
419
// a debug info issue.
398
- dbg () << " WARNING: " << NameOfWrappedPass << " dropped DILocation of "
399
- << *Instr << " (BB: " << BBName << " , Fn: " << FnName
400
- << " , File: " << FileNameFromCU << " )\n " ;
420
+ if (ShouldWriteIntoJSON)
421
+ Bugs.push_back (llvm::json::Object ({{" metadata" , " DILocation" },
422
+ {" fn-name" , FnName.str ()},
423
+ {" bb-name" , BBName.str ()},
424
+ {" instr" , InstName},
425
+ {" action" , " drop" }}));
426
+ else
427
+ dbg () << " WARNING: " << NameOfWrappedPass << " dropped DILocation of "
428
+ << *Instr << " (BB: " << BBName << " , Fn: " << FnName
429
+ << " , File: " << FileNameFromCU << " )\n " ;
401
430
Preserved = false ;
402
431
}
403
432
}
404
433
405
434
return Preserved;
406
435
}
407
436
437
+ // Write the json data into the specifed file.
438
+ static void writeJSON (StringRef OrigDIVerifyBugsReportFilePath,
439
+ StringRef FileNameFromCU, StringRef NameOfWrappedPass,
440
+ llvm::json::Array &Bugs) {
441
+ std::error_code EC;
442
+ raw_fd_ostream OS_FILE{OrigDIVerifyBugsReportFilePath, EC,
443
+ sys::fs::OF_Append | sys::fs::OF_Text};
444
+ if (EC) {
445
+ errs () << " Could not open file: " << EC.message () << " , "
446
+ << OrigDIVerifyBugsReportFilePath << ' \n ' ;
447
+ return ;
448
+ }
449
+
450
+ OS_FILE << " {\" file\" :\" " << FileNameFromCU << " \" , " ;
451
+
452
+ StringRef PassName = NameOfWrappedPass != " " ? NameOfWrappedPass : " no-name" ;
453
+ OS_FILE << " \" pass\" :\" " << PassName << " \" , " ;
454
+
455
+ llvm::json::Value BugsToPrint{std::move (Bugs)};
456
+ OS_FILE << " \" bugs\" : " << BugsToPrint;
457
+
458
+ OS_FILE << " }\n " ;
459
+ }
460
+
408
461
bool llvm::checkDebugInfoMetadata (Module &M,
409
462
iterator_range<Module::iterator> Functions,
410
463
DebugInfoPerPassMap &DIPreservationMap,
411
- StringRef Banner,
412
- StringRef NameOfWrappedPass ) {
464
+ StringRef Banner, StringRef NameOfWrappedPass,
465
+ StringRef OrigDIVerifyBugsReportFilePath ) {
413
466
LLVM_DEBUG (dbgs () << Banner << " : (after) " << NameOfWrappedPass << ' \n ' );
414
467
415
468
if (!M.getNamedMetadata (" llvm.dbg.cu" )) {
@@ -428,7 +481,8 @@ bool llvm::checkDebugInfoMetadata(Module &M,
428
481
// TODO: Collect metadata other than DISubprograms.
429
482
// Collect the DISubprogram.
430
483
auto *SP = F.getSubprogram ();
431
- DIPreservationAfter[NameOfWrappedPass].DIFunctions .insert ({F.getName (), SP});
484
+ DIPreservationAfter[NameOfWrappedPass].DIFunctions .insert (
485
+ {F.getName (), SP});
432
486
if (SP)
433
487
LLVM_DEBUG (dbgs () << " Collecting subprogram: " << *SP << ' \n ' );
434
488
@@ -467,14 +521,22 @@ bool llvm::checkDebugInfoMetadata(Module &M,
467
521
468
522
auto InstToDelete = DIPreservationAfter[NameOfWrappedPass].InstToDelete ;
469
523
470
- bool ResultForFunc = checkFunctions (DIFunctionsBefore, DIFunctionsAfter,
471
- NameOfWrappedPass, FileNameFromCU);
472
- bool ResultForInsts =
473
- checkInstructions (DILocsBefore, DILocsAfter, InstToDelete,
474
- NameOfWrappedPass, FileNameFromCU);
524
+ bool ShouldWriteIntoJSON = !OrigDIVerifyBugsReportFilePath.empty ();
525
+ llvm::json::Array Bugs;
526
+
527
+ bool ResultForFunc =
528
+ checkFunctions (DIFunctionsBefore, DIFunctionsAfter, NameOfWrappedPass,
529
+ FileNameFromCU, ShouldWriteIntoJSON, Bugs);
530
+ bool ResultForInsts = checkInstructions (
531
+ DILocsBefore, DILocsAfter, InstToDelete, NameOfWrappedPass,
532
+ FileNameFromCU, ShouldWriteIntoJSON, Bugs);
475
533
bool Result = ResultForFunc && ResultForInsts;
476
534
477
535
StringRef ResultBanner = NameOfWrappedPass != " " ? NameOfWrappedPass : Banner;
536
+ if (ShouldWriteIntoJSON && !Bugs.empty ())
537
+ writeJSON (OrigDIVerifyBugsReportFilePath, FileNameFromCU, NameOfWrappedPass,
538
+ Bugs);
539
+
478
540
if (Result)
479
541
dbg () << ResultBanner << " : PASS\n " ;
480
542
else
@@ -680,15 +742,18 @@ struct CheckDebugifyModulePass : public ModulePass {
680
742
" CheckModuleDebugify" , Strip, StatsMap);
681
743
return checkDebugInfoMetadata (
682
744
M, M.functions (), *DIPreservationMap,
683
- " CheckModuleDebugify (original debuginfo)" , NameOfWrappedPass);
745
+ " CheckModuleDebugify (original debuginfo)" , NameOfWrappedPass,
746
+ OrigDIVerifyBugsReportFilePath);
684
747
}
685
748
686
749
CheckDebugifyModulePass (
687
750
bool Strip = false , StringRef NameOfWrappedPass = " " ,
688
751
DebugifyStatsMap *StatsMap = nullptr ,
689
752
enum DebugifyMode Mode = DebugifyMode::SyntheticDebugInfo,
690
- DebugInfoPerPassMap *DIPreservationMap = nullptr )
753
+ DebugInfoPerPassMap *DIPreservationMap = nullptr ,
754
+ StringRef OrigDIVerifyBugsReportFilePath = " " )
691
755
: ModulePass(ID), NameOfWrappedPass(NameOfWrappedPass),
756
+ OrigDIVerifyBugsReportFilePath (OrigDIVerifyBugsReportFilePath),
692
757
StatsMap(StatsMap), DIPreservationMap(DIPreservationMap), Mode(Mode),
693
758
Strip(Strip) {}
694
759
@@ -700,6 +765,7 @@ struct CheckDebugifyModulePass : public ModulePass {
700
765
701
766
private:
702
767
StringRef NameOfWrappedPass;
768
+ StringRef OrigDIVerifyBugsReportFilePath;
703
769
DebugifyStatsMap *StatsMap;
704
770
DebugInfoPerPassMap *DIPreservationMap;
705
771
enum DebugifyMode Mode;
@@ -718,15 +784,18 @@ struct CheckDebugifyFunctionPass : public FunctionPass {
718
784
Strip, StatsMap);
719
785
return checkDebugInfoMetadata (
720
786
M, make_range (FuncIt, std::next (FuncIt)), *DIPreservationMap,
721
- " CheckFunctionDebugify (original debuginfo)" , NameOfWrappedPass);
787
+ " CheckFunctionDebugify (original debuginfo)" , NameOfWrappedPass,
788
+ OrigDIVerifyBugsReportFilePath);
722
789
}
723
790
724
791
CheckDebugifyFunctionPass (
725
792
bool Strip = false , StringRef NameOfWrappedPass = " " ,
726
793
DebugifyStatsMap *StatsMap = nullptr ,
727
794
enum DebugifyMode Mode = DebugifyMode::SyntheticDebugInfo,
728
- DebugInfoPerPassMap *DIPreservationMap = nullptr )
795
+ DebugInfoPerPassMap *DIPreservationMap = nullptr ,
796
+ StringRef OrigDIVerifyBugsReportFilePath = " " )
729
797
: FunctionPass(ID), NameOfWrappedPass(NameOfWrappedPass),
798
+ OrigDIVerifyBugsReportFilePath (OrigDIVerifyBugsReportFilePath),
730
799
StatsMap(StatsMap), DIPreservationMap(DIPreservationMap), Mode(Mode),
731
800
Strip(Strip) {}
732
801
@@ -738,6 +807,7 @@ struct CheckDebugifyFunctionPass : public FunctionPass {
738
807
739
808
private:
740
809
StringRef NameOfWrappedPass;
810
+ StringRef OrigDIVerifyBugsReportFilePath;
741
811
DebugifyStatsMap *StatsMap;
742
812
DebugInfoPerPassMap *DIPreservationMap;
743
813
enum DebugifyMode Mode;
@@ -794,22 +864,26 @@ PreservedAnalyses NewPMDebugifyPass::run(Module &M, ModuleAnalysisManager &) {
794
864
795
865
ModulePass *createCheckDebugifyModulePass (
796
866
bool Strip, StringRef NameOfWrappedPass, DebugifyStatsMap *StatsMap,
797
- enum DebugifyMode Mode, DebugInfoPerPassMap *DIPreservationMap) {
867
+ enum DebugifyMode Mode, DebugInfoPerPassMap *DIPreservationMap,
868
+ StringRef OrigDIVerifyBugsReportFilePath) {
798
869
if (Mode == DebugifyMode::SyntheticDebugInfo)
799
870
return new CheckDebugifyModulePass (Strip, NameOfWrappedPass, StatsMap);
800
871
assert (Mode == DebugifyMode::OriginalDebugInfo && " Must be original mode" );
801
872
return new CheckDebugifyModulePass (false , NameOfWrappedPass, nullptr , Mode,
802
- DIPreservationMap);
873
+ DIPreservationMap,
874
+ OrigDIVerifyBugsReportFilePath);
803
875
}
804
876
805
877
FunctionPass *createCheckDebugifyFunctionPass (
806
878
bool Strip, StringRef NameOfWrappedPass, DebugifyStatsMap *StatsMap,
807
- enum DebugifyMode Mode, DebugInfoPerPassMap *DIPreservationMap) {
879
+ enum DebugifyMode Mode, DebugInfoPerPassMap *DIPreservationMap,
880
+ StringRef OrigDIVerifyBugsReportFilePath) {
808
881
if (Mode == DebugifyMode::SyntheticDebugInfo)
809
882
return new CheckDebugifyFunctionPass (Strip, NameOfWrappedPass, StatsMap);
810
883
assert (Mode == DebugifyMode::OriginalDebugInfo && " Must be original mode" );
811
884
return new CheckDebugifyFunctionPass (false , NameOfWrappedPass, nullptr , Mode,
812
- DIPreservationMap);
885
+ DIPreservationMap,
886
+ OrigDIVerifyBugsReportFilePath);
813
887
}
814
888
815
889
PreservedAnalyses NewPMCheckDebugifyPass::run (Module &M,
0 commit comments