|
43 | 43 | #include "llvm/IR/DebugInfoMetadata.h"
|
44 | 44 | #include "llvm/IR/DebugLoc.h"
|
45 | 45 | #include "llvm/IR/DerivedTypes.h"
|
| 46 | +#include "llvm/IR/EHPersonalities.h" |
46 | 47 | #include "llvm/IR/Function.h"
|
47 | 48 | #include "llvm/IR/GlobalAlias.h"
|
48 | 49 | #include "llvm/IR/GlobalValue.h"
|
@@ -643,21 +644,70 @@ static uint64_t GetCtorAndDtorPriority(Triple &TargetTriple) {
|
643 | 644 |
|
644 | 645 | namespace {
|
645 | 646 | /// Helper RAII class to post-process inserted asan runtime calls during a
|
646 |
| -/// pass on a single Function. This is a no-op implementation, for a first NFC |
647 |
| -/// commit. Coming up: detect and add "funclet" opBundle to function calls that |
648 |
| -/// need them. |
| 647 | +/// pass on a single Function. Upon end of scope, detects and applies the |
| 648 | +/// required funclet OpBundle. |
649 | 649 | class RuntimeCallInserter {
|
650 | 650 | Function *OwnerFn = nullptr;
|
| 651 | + bool TrackInsertedCalls = false; |
| 652 | + SmallVector<CallInst *> InsertedCalls; |
651 | 653 |
|
652 | 654 | public:
|
653 |
| - RuntimeCallInserter(Function &Fn) : OwnerFn(&Fn) {} |
| 655 | + RuntimeCallInserter(Function &Fn) : OwnerFn(&Fn) { |
| 656 | + if (Fn.hasPersonalityFn()) { |
| 657 | + auto Personality = classifyEHPersonality(Fn.getPersonalityFn()); |
| 658 | + if (isScopedEHPersonality(Personality)) |
| 659 | + TrackInsertedCalls = true; |
| 660 | + } |
| 661 | + } |
| 662 | + |
| 663 | + ~RuntimeCallInserter() { |
| 664 | + if (InsertedCalls.empty()) |
| 665 | + return; |
| 666 | + assert(TrackInsertedCalls && "Calls were wrongly tracked"); |
| 667 | + |
| 668 | + DenseMap<BasicBlock *, ColorVector> BlockColors = colorEHFunclets(*OwnerFn); |
| 669 | + for (CallInst *CI : InsertedCalls) { |
| 670 | + BasicBlock *BB = CI->getParent(); |
| 671 | + assert(BB && "Instruction doesn't belong to a BasicBlock"); |
| 672 | + assert(BB->getParent() == OwnerFn && |
| 673 | + "Instruction doesn't belong to the expected Function!"); |
| 674 | + |
| 675 | + ColorVector &Colors = BlockColors[BB]; |
| 676 | + // funclet opbundles are only valid in monochromatic BBs. |
| 677 | + // Note that unreachable BBs are seen as colorless by colorEHFunclets() |
| 678 | + // and will be DCE'ed later. |
| 679 | + if (Colors.empty()) |
| 680 | + continue; |
| 681 | + if (Colors.size() != 1) { |
| 682 | + OwnerFn->getContext().emitError( |
| 683 | + "Instruction's BasicBlock is not monochromatic"); |
| 684 | + continue; |
| 685 | + } |
| 686 | + |
| 687 | + BasicBlock *Color = Colors.front(); |
| 688 | + Instruction *EHPad = Color->getFirstNonPHI(); |
| 689 | + |
| 690 | + if (EHPad && EHPad->isEHPad()) { |
| 691 | + // Replace CI with a clone with an added funclet OperandBundle |
| 692 | + OperandBundleDef OB("funclet", EHPad); |
| 693 | + auto *NewCall = |
| 694 | + CallBase::addOperandBundle(CI, LLVMContext::OB_funclet, OB, CI); |
| 695 | + NewCall->copyMetadata(*CI); |
| 696 | + CI->replaceAllUsesWith(NewCall); |
| 697 | + CI->eraseFromParent(); |
| 698 | + } |
| 699 | + } |
| 700 | + } |
654 | 701 |
|
655 | 702 | CallInst *createRuntimeCall(IRBuilder<> &IRB, FunctionCallee Callee,
|
656 | 703 | ArrayRef<Value *> Args = {},
|
657 | 704 | const Twine &Name = "") {
|
658 | 705 | assert(IRB.GetInsertBlock()->getParent() == OwnerFn);
|
659 |
| - (void)OwnerFn; |
660 |
| - return IRB.CreateCall(Callee, Args, Name, nullptr); |
| 706 | + |
| 707 | + CallInst *Inst = IRB.CreateCall(Callee, Args, Name, nullptr); |
| 708 | + if (TrackInsertedCalls) |
| 709 | + InsertedCalls.push_back(Inst); |
| 710 | + return Inst; |
661 | 711 | }
|
662 | 712 | };
|
663 | 713 |
|
|
0 commit comments