forked from OSchip/llvm-project
[Function Specialisation] Fix use after free
This is a fix for a use-after-free found by the address sanitizer when compiling GCC: https://github.com/llvm/llvm-project/issues/52821 The Function Specialization pass may remove instructions, cached inside the PredicateBase class, which are later being dereferenced from the SCCPInstVisitor class. To prevent the dangling references I am lazily deleting the dead instructions after the Solver has run. Differential Revision: https://reviews.llvm.org/D118591
This commit is contained in:
parent
116c1bea65
commit
438a81a284
|
@ -276,6 +276,7 @@ class FunctionSpecializer {
|
|||
std::function<TargetLibraryInfo &(Function &)> GetTLI;
|
||||
|
||||
SmallPtrSet<Function *, 2> SpecializedFuncs;
|
||||
SmallVector<Instruction *> ReplacedWithConstant;
|
||||
|
||||
public:
|
||||
FunctionSpecializer(SCCPSolver &Solver,
|
||||
|
@ -320,6 +321,15 @@ public:
|
|||
return Changed;
|
||||
}
|
||||
|
||||
void removeDeadInstructions() {
|
||||
for (auto *I : ReplacedWithConstant) {
|
||||
LLVM_DEBUG(dbgs() << "FnSpecialization: Removing dead instruction "
|
||||
<< *I << "\n");
|
||||
I->eraseFromParent();
|
||||
}
|
||||
ReplacedWithConstant.clear();
|
||||
}
|
||||
|
||||
bool tryToReplaceWithConstant(Value *V) {
|
||||
if (!V->getType()->isSingleValueType() || isa<CallBase>(V) ||
|
||||
V->user_empty())
|
||||
|
@ -330,6 +340,10 @@ public:
|
|||
return false;
|
||||
auto *Const =
|
||||
isConstant(IV) ? Solver.getConstant(IV) : UndefValue::get(V->getType());
|
||||
|
||||
LLVM_DEBUG(dbgs() << "FnSpecialization: Replacing " << *V
|
||||
<< "\nFnSpecialization: with " << *Const << "\n");
|
||||
|
||||
V->replaceAllUsesWith(Const);
|
||||
|
||||
for (auto *U : Const->users())
|
||||
|
@ -340,7 +354,7 @@ public:
|
|||
// Remove the instruction from Block and Solver.
|
||||
if (auto *I = dyn_cast<Instruction>(V)) {
|
||||
if (I->isSafeToRemove()) {
|
||||
I->eraseFromParent();
|
||||
ReplacedWithConstant.push_back(I);
|
||||
Solver.removeLatticeValueFor(I);
|
||||
}
|
||||
}
|
||||
|
@ -886,7 +900,8 @@ bool llvm::runFunctionSpecialization(
|
|||
Changed = true;
|
||||
}
|
||||
|
||||
// Clean up the IR by removing ssa_copy intrinsics.
|
||||
// Clean up the IR by removing dead instructions and ssa_copy intrinsics.
|
||||
FS.removeDeadInstructions();
|
||||
removeSSACopy(M);
|
||||
return Changed;
|
||||
}
|
||||
|
|
|
@ -0,0 +1,58 @@
|
|||
; NOTE: Assertions have been autogenerated by utils/update_test_checks.py
|
||||
; RUN: opt -function-specialization -S < %s | FileCheck %s
|
||||
|
||||
%mystruct = type { i32, [2 x i64] }
|
||||
|
||||
define internal %mystruct* @myfunc(%mystruct* %arg) {
|
||||
; CHECK-LABEL: @myfunc(
|
||||
; CHECK-NEXT: entry:
|
||||
; CHECK-NEXT: br label [[FOR_COND:%.*]]
|
||||
; CHECK: for.cond:
|
||||
; CHECK-NEXT: br i1 true, label [[FOR_COND2:%.*]], label [[FOR_BODY:%.*]]
|
||||
; CHECK: for.body:
|
||||
; CHECK-NEXT: call void @callee(%mystruct* nonnull null)
|
||||
; CHECK-NEXT: br label [[FOR_COND]]
|
||||
; CHECK: for.cond2:
|
||||
; CHECK-NEXT: br i1 false, label [[FOR_END:%.*]], label [[FOR_BODY2:%.*]]
|
||||
; CHECK: for.body2:
|
||||
; CHECK-NEXT: [[ARRAYIDX:%.*]] = getelementptr inbounds [[MYSTRUCT:%.*]], %mystruct* null, i64 0, i32 1, i64 3
|
||||
; CHECK-NEXT: br label [[FOR_COND2]]
|
||||
; CHECK: for.end:
|
||||
; CHECK-NEXT: ret %mystruct* [[ARG:%.*]]
|
||||
;
|
||||
entry:
|
||||
br label %for.cond
|
||||
|
||||
for.cond: ; preds = %for.body, %entry
|
||||
%phi = phi %mystruct* [ undef, %for.body ], [ null, %entry ]
|
||||
%cond = icmp eq %mystruct* %phi, null
|
||||
br i1 %cond, label %for.cond2, label %for.body
|
||||
|
||||
for.body: ; preds = %for.cond
|
||||
call void @callee(%mystruct* nonnull %phi)
|
||||
br label %for.cond
|
||||
|
||||
for.cond2: ; preds = %for.body2, %for.cond
|
||||
%phi2 = phi %mystruct* [ undef, %for.body2 ], [ null, %for.cond ]
|
||||
br i1 undef, label %for.end, label %for.body2
|
||||
|
||||
for.body2: ; preds = %for.cond2
|
||||
%arrayidx = getelementptr inbounds %mystruct, %mystruct* %phi2, i64 0, i32 1, i64 3
|
||||
br label %for.cond2
|
||||
|
||||
for.end: ; preds = %for.cond2
|
||||
ret %mystruct* %arg
|
||||
}
|
||||
|
||||
define %mystruct* @caller() {
|
||||
; CHECK-LABEL: @caller(
|
||||
; CHECK-NEXT: entry:
|
||||
; CHECK-NEXT: [[CALL:%.*]] = call %mystruct* @myfunc(%mystruct* undef)
|
||||
; CHECK-NEXT: ret %mystruct* [[CALL]]
|
||||
;
|
||||
entry:
|
||||
%call = call %mystruct* @myfunc(%mystruct* undef)
|
||||
ret %mystruct* %call
|
||||
}
|
||||
|
||||
declare void @callee(%mystruct*)
|
Loading…
Reference in New Issue