Skip to content

Commit 11fc817

Browse files
committedAug 13, 2015
[DeadStoreElimination] remove a redundant store even if the load is in a different block.
DeadStoreElimination does eliminate a store if it stores a value which was loaded from the same memory location. So far this worked only if the store is in the same block as the load. Now we can also handle stores which are in a different block than the load. Example: define i32 @test(i1, i32*) { entry: %l2 = load i32, i32* %1, align 4 br i1 %0, label %bb1, label %bb2 bb1: br label %bb3 bb2: ; This store is redundant store i32 %l2, i32* %1, align 4 br label %bb3 bb3: ret i32 0 } Differential Revision: http://reviews.llvm.org/D11854 llvm-svn: 244901
1 parent ef1ac01 commit 11fc817

File tree

2 files changed

+218
-10
lines changed

2 files changed

+218
-10
lines changed
 

‎llvm/lib/Transforms/Scalar/DeadStoreElimination.cpp

+71-10
Original file line numberDiff line numberDiff line change
@@ -40,6 +40,7 @@ using namespace llvm;
4040

4141
#define DEBUG_TYPE "dse"
4242

43+
STATISTIC(NumRedundantStores, "Number of redundant stores deleted");
4344
STATISTIC(NumFastStores, "Number of stores deleted");
4445
STATISTIC(NumFastOther , "Number of other instrs removed");
4546

@@ -76,6 +77,7 @@ namespace {
7677
}
7778

7879
bool runOnBasicBlock(BasicBlock &BB);
80+
bool MemoryIsNotModifiedBetween(LoadInst *LI, StoreInst *SI);
7981
bool HandleFree(CallInst *F);
8082
bool handleEndBlock(BasicBlock &BB);
8183
void RemoveAccessedObjects(const MemoryLocation &LoadedLoc,
@@ -495,19 +497,14 @@ bool DSE::runOnBasicBlock(BasicBlock &BB) {
495497
if (!hasMemoryWrite(Inst, *TLI))
496498
continue;
497499

498-
MemDepResult InstDep = MD->getDependency(Inst);
499-
500-
// Ignore any store where we can't find a local dependence.
501-
// FIXME: cross-block DSE would be fun. :)
502-
if (!InstDep.isDef() && !InstDep.isClobber())
503-
continue;
504-
505500
// If we're storing the same value back to a pointer that we just
506501
// loaded from, then the store can be removed.
507502
if (StoreInst *SI = dyn_cast<StoreInst>(Inst)) {
508-
if (LoadInst *DepLoad = dyn_cast<LoadInst>(InstDep.getInst())) {
503+
if (LoadInst *DepLoad = dyn_cast<LoadInst>(SI->getValueOperand())) {
509504
if (SI->getPointerOperand() == DepLoad->getPointerOperand() &&
510-
SI->getOperand(0) == DepLoad && isRemovable(SI)) {
505+
isRemovable(SI) &&
506+
MemoryIsNotModifiedBetween(DepLoad, SI)) {
507+
511508
DEBUG(dbgs() << "DSE: Remove Store Of Load from same pointer:\n "
512509
<< "LOAD: " << *DepLoad << "\n STORE: " << *SI << '\n');
513510

@@ -521,13 +518,20 @@ bool DSE::runOnBasicBlock(BasicBlock &BB) {
521518
BBI = BB.begin();
522519
else if (BBI != BB.begin()) // Revisit this instruction if possible.
523520
--BBI;
524-
++NumFastStores;
521+
++NumRedundantStores;
525522
MadeChange = true;
526523
continue;
527524
}
528525
}
529526
}
530527

528+
MemDepResult InstDep = MD->getDependency(Inst);
529+
530+
// Ignore any store where we can't find a local dependence.
531+
// FIXME: cross-block DSE would be fun. :)
532+
if (!InstDep.isDef() && !InstDep.isClobber())
533+
continue;
534+
531535
// Figure out what location is being stored to.
532536
MemoryLocation Loc = getLocForWrite(Inst, *AA);
533537

@@ -627,6 +631,63 @@ bool DSE::runOnBasicBlock(BasicBlock &BB) {
627631
return MadeChange;
628632
}
629633

634+
/// Returns true if the memory which is accessed by the store instruction is not
635+
/// modified between the load and the store instruction.
636+
/// Precondition: The store instruction must be dominated by the load
637+
/// instruction.
638+
bool DSE::MemoryIsNotModifiedBetween(LoadInst *LI, StoreInst *SI) {
639+
SmallVector<BasicBlock *, 16> WorkList;
640+
SmallPtrSet<BasicBlock *, 8> Visited;
641+
BasicBlock::iterator LoadBBI(LI);
642+
++LoadBBI;
643+
BasicBlock::iterator StoreBBI(SI);
644+
BasicBlock *LoadBB = LI->getParent();
645+
BasicBlock *StoreBB = SI->getParent();
646+
MemoryLocation StoreLoc = MemoryLocation::get(SI);
647+
648+
// Start checking the store-block.
649+
WorkList.push_back(StoreBB);
650+
bool isFirstBlock = true;
651+
652+
// Check all blocks going backward until we reach the load-block.
653+
while (!WorkList.empty()) {
654+
BasicBlock *B = WorkList.pop_back_val();
655+
656+
// Ignore instructions before LI if this is the LoadBB.
657+
BasicBlock::iterator BI = (B == LoadBB ? LoadBBI : B->begin());
658+
659+
BasicBlock::iterator EI;
660+
if (isFirstBlock) {
661+
// Ignore instructions after SI if this is the first visit of StoreBB.
662+
assert(B == StoreBB && "first block is not the store block");
663+
EI = StoreBBI;
664+
isFirstBlock = false;
665+
} else {
666+
// It's not StoreBB or (in case of a loop) the second visit of StoreBB.
667+
// In this case we also have to look at instructions after SI.
668+
EI = B->end();
669+
}
670+
for (; BI != EI; ++BI) {
671+
Instruction *I = BI;
672+
if (I->mayWriteToMemory() && I != SI) {
673+
auto Res = AA->getModRefInfo(I, StoreLoc);
674+
if (Res != MRI_NoModRef)
675+
return false;
676+
}
677+
}
678+
if (B != LoadBB) {
679+
assert(B != &LoadBB->getParent()->getEntryBlock() &&
680+
"Should not hit the entry block because SI must be dominated by LI");
681+
for (auto PredI = pred_begin(B), PE = pred_end(B); PredI != PE; ++PredI) {
682+
if (!Visited.insert(*PredI).second)
683+
continue;
684+
WorkList.push_back(*PredI);
685+
}
686+
}
687+
}
688+
return true;
689+
}
690+
630691
/// Find all blocks that will unconditionally lead to the block BB and append
631692
/// them to F.
632693
static void FindUnconditionalPreds(SmallVectorImpl<BasicBlock *> &Blocks,

‎llvm/test/Transforms/DeadStoreElimination/simple.ll

+147
Original file line numberDiff line numberDiff line change
@@ -350,3 +350,150 @@ define i8* @test25(i8* %p) nounwind {
350350
store i8 %tmp, i8* %p.4, align 1
351351
ret i8* %q
352352
}
353+
354+
; Remove redundant store if loaded value is in another block.
355+
; CHECK-LABEL: @test26(
356+
; CHECK-NOT: store
357+
; CHECK: ret
358+
define i32 @test26(i1 %c, i32* %p) {
359+
entry:
360+
%v = load i32, i32* %p, align 4
361+
br i1 %c, label %bb1, label %bb2
362+
bb1:
363+
br label %bb3
364+
bb2:
365+
store i32 %v, i32* %p, align 4
366+
br label %bb3
367+
bb3:
368+
ret i32 0
369+
}
370+
371+
; Remove redundant store if loaded value is in another block.
372+
; CHECK-LABEL: @test27(
373+
; CHECK-NOT: store
374+
; CHECK: ret
375+
define i32 @test27(i1 %c, i32* %p) {
376+
entry:
377+
%v = load i32, i32* %p, align 4
378+
br i1 %c, label %bb1, label %bb2
379+
bb1:
380+
br label %bb3
381+
bb2:
382+
br label %bb3
383+
bb3:
384+
store i32 %v, i32* %p, align 4
385+
ret i32 0
386+
}
387+
388+
; Don't remove redundant store because of may-aliased store.
389+
; CHECK-LABEL: @test28(
390+
; CHECK: bb3:
391+
; CHECK-NEXT: store i32 %v
392+
define i32 @test28(i1 %c, i32* %p, i32* %p2, i32 %i) {
393+
entry:
394+
%v = load i32, i32* %p, align 4
395+
396+
; Might overwrite value at %p
397+
store i32 %i, i32* %p2, align 4
398+
br i1 %c, label %bb1, label %bb2
399+
bb1:
400+
br label %bb3
401+
bb2:
402+
br label %bb3
403+
bb3:
404+
store i32 %v, i32* %p, align 4
405+
ret i32 0
406+
}
407+
408+
; Don't remove redundant store because of may-aliased store.
409+
; CHECK-LABEL: @test29(
410+
; CHECK: bb3:
411+
; CHECK-NEXT: store i32 %v
412+
define i32 @test29(i1 %c, i32* %p, i32* %p2, i32 %i) {
413+
entry:
414+
%v = load i32, i32* %p, align 4
415+
br i1 %c, label %bb1, label %bb2
416+
bb1:
417+
br label %bb3
418+
bb2:
419+
; Might overwrite value at %p
420+
store i32 %i, i32* %p2, align 4
421+
br label %bb3
422+
bb3:
423+
store i32 %v, i32* %p, align 4
424+
ret i32 0
425+
}
426+
427+
declare void @unknown_func()
428+
429+
; Don't remove redundant store because of unknown call.
430+
; CHECK-LABEL: @test30(
431+
; CHECK: bb3:
432+
; CHECK-NEXT: store i32 %v
433+
define i32 @test30(i1 %c, i32* %p, i32 %i) {
434+
entry:
435+
%v = load i32, i32* %p, align 4
436+
br i1 %c, label %bb1, label %bb2
437+
bb1:
438+
br label %bb3
439+
bb2:
440+
; Might overwrite value at %p
441+
call void @unknown_func()
442+
br label %bb3
443+
bb3:
444+
store i32 %v, i32* %p, align 4
445+
ret i32 0
446+
}
447+
448+
; Remove redundant store if loaded value is in another block inside a loop.
449+
; CHECK-LABEL: @test31(
450+
; CHECK-NOT: store
451+
; CHECK: ret
452+
define i32 @test31(i1 %c, i32* %p, i32 %i) {
453+
entry:
454+
%v = load i32, i32* %p, align 4
455+
br label %bb1
456+
bb1:
457+
store i32 %v, i32* %p, align 4
458+
br i1 undef, label %bb1, label %bb2
459+
bb2:
460+
ret i32 0
461+
}
462+
463+
; Don't remove redundant store in a loop with a may-alias store.
464+
; CHECK-LABEL: @test32(
465+
; CHECK: bb1:
466+
; CHECK-NEXT: store i32 %v
467+
; CHECK-NEXT: call void @unknown_func
468+
define i32 @test32(i1 %c, i32* %p, i32 %i) {
469+
entry:
470+
%v = load i32, i32* %p, align 4
471+
br label %bb1
472+
bb1:
473+
store i32 %v, i32* %p, align 4
474+
; Might read and overwrite value at %p
475+
call void @unknown_func()
476+
br i1 undef, label %bb1, label %bb2
477+
bb2:
478+
ret i32 0
479+
}
480+
481+
; Remove redundant store, which is in the lame loop as the load.
482+
; CHECK-LABEL: @test33(
483+
; CHECK-NOT: store
484+
; CHECK: ret
485+
define i32 @test33(i1 %c, i32* %p, i32 %i) {
486+
entry:
487+
br label %bb1
488+
bb1:
489+
%v = load i32, i32* %p, align 4
490+
br label %bb2
491+
bb2:
492+
store i32 %v, i32* %p, align 4
493+
; Might read and overwrite value at %p, but doesn't matter.
494+
call void @unknown_func()
495+
br i1 undef, label %bb1, label %bb3
496+
bb3:
497+
ret i32 0
498+
}
499+

0 commit comments

Comments
 (0)
Please sign in to comment.