Index: lib/Transforms/Scalar/ADCE.cpp
===================================================================
--- lib/Transforms/Scalar/ADCE.cpp
+++ lib/Transforms/Scalar/ADCE.cpp
@@ -26,6 +26,7 @@
 #include "llvm/IR/BasicBlock.h"
 #include "llvm/IR/CFG.h"
 #include "llvm/IR/DebugInfoMetadata.h"
+#include "llvm/IR/IRBuilder.h"
 #include "llvm/IR/InstIterator.h"
 #include "llvm/IR/Instructions.h"
 #include "llvm/IR/IntrinsicInst.h"
@@ -37,11 +38,17 @@
 #define DEBUG_TYPE "adce"
 
 STATISTIC(NumRemoved, "Number of instructions removed");
+STATISTIC(NumBranchesRemoved, "Number of branch instructions removed");
 
 // This is a tempoary option until we change the interface
 // to this pass based on optimization level.
 static cl::opt<bool> RemoveControlFlowFlag("adce-remove-control-flow",
-                                           cl::init(false), cl::Hidden);
+                                           cl::init(true), cl::Hidden);
+
+// This option enables removing of may-be-infinite loops which have no other
+// effect.
+static cl::opt<bool> RemoveLoops("adce-remove-loops", cl::init(false),
+                                 cl::Hidden);
 
 namespace {
 /// Information about Instructions
@@ -97,8 +104,9 @@
   /// Set of blocks with not known to have live terminators.
   SmallPtrSet<BasicBlock *, 16> BlocksWithDeadTerminators;
 
-  /// The set of blocks which we have determined are live in the
-  /// most recent iteration of propagating liveness.
+  /// The set of blocks which we have determined whose control
+  /// dependence sources must be live and which have not had
+  /// those dependences analyized.
   SmallPtrSet<BasicBlock *, 16> NewLiveBlocks;
 
   /// Set up auxiliary data structures for Instructions and BasicBlocks and
@@ -113,7 +121,10 @@
   void markLiveInstructions();
   /// Mark an instruction as live.
   void markLive(Instruction *I);
-  
+  /// Mark a block as live.
+  void markLive(BlockInfoType &BB);
+  void markLive(BasicBlock *BB) { markLive(BlockInfo[BB]); }
+
   /// Mark terminators of control predecessors of a PHI node live.
   void markPhiLive(PHINode *PN);
 
@@ -130,6 +141,18 @@
   /// was removed.
   bool removeDeadInstructions();
 
+  /// Identify connected sections of the control flow grap which have
+  /// dead terminators and rewrite the control flow graph to remove them.
+  void updateDeadRegions();
+  void updateDeadRegions(BasicBlock *BB,
+                         SmallPtrSet<BasicBlock *, 16> *VisitedBlocks);
+
+  /// Make the terminator of this block an unconditional branch to \p Target.
+  void makeUnconditional(BasicBlock *BB, BasicBlock *Target);
+  /// The collection of unconditional branch instructions created
+  /// by makeUnconditional.
+  SmallVector<Instruction *, 16> NewBranches;
+
 public:
   AggressiveDeadCodeElimination(Function &F, PostDominatorTree &PDT)
       : F(F), PDT(PDT) {}
@@ -186,23 +209,45 @@
   if (!RemoveControlFlowFlag)
     return;
 
-  // This is temporary: will update with post order traveral to
-  // find loop bottoms
-  SmallPtrSet<BasicBlock *, 16> Seen;
-  for (auto &BB : F) {
-    Seen.insert(&BB);
-    TerminatorInst *Term = BB.getTerminator();
-    if (isLive(Term))
-      continue;
+  if (!RemoveLoops) {
 
-    for (auto Succ : successors(&BB))
-      if (Seen.count(Succ)) {
-        // back edge....
-        markLive(Term);
+    // Iterate over blocks in lexical order and
+    // treat all edges to a block already seen as loop back edges
+    // and mark the branch live it if there is a back edge.
+    SmallPtrSet<BasicBlock *, 16> Seen;
+    for (auto &BB : F) {
+      Seen.insert(&BB);
+      TerminatorInst *Term = BB.getTerminator();
+      if (isLive(Term))
+        continue;
+
+      for (auto Succ : successors(&BB))
+        if (Seen.count(Succ)) {
+          // back edge....
+          markLive(Term);
+          break;
+        }
+    }
+  }
+
+  // Mark blocks live if there is no path from the block to the
+  // return of the function or a successor for which this is true.
+  // This protects IDFCalculator which cannot handle such blocks.
+  for (auto &BBInfoPair : BlockInfo) {
+    auto &BBInfo = BBInfoPair.second;
+    if (BBInfo.terminatorIsLive())
+      continue;
+    auto *BB = BBInfo.BB;
+    if (!PDT.getNode(BB)) {
+      markLive(BBInfo.Terminator);
+      continue;
+    }
+    for (auto Succ : successors(BB))
+      if (!PDT.getNode(Succ)) {
+        markLive(BBInfo.Terminator);
         break;
       }
   }
-  // End temporary handling of loops.
 
   // Mark blocks live if there is no path from the block to the
   // return of the function or a successor for which this is true.
@@ -278,32 +323,19 @@
       Instruction *LiveInst = Worklist.pop_back_val();
       DEBUG(dbgs() << "work live: "; LiveInst->dump(););
 
-      // Collect the live debug info scopes attached to this instruction.
-      if (const DILocation *DL = LiveInst->getDebugLoc())
-        collectLiveScopes(*DL);
-
       for (Use &OI : LiveInst->operands())
         if (Instruction *Inst = dyn_cast<Instruction>(OI))
           markLive(Inst);
-      
+
       if (auto *PN = dyn_cast<PHINode>(LiveInst))
         markPhiLive(PN);
     }
+
+    // After data flow liveness has been identified, examine which branch
+    // decisions are required to determine live instructions are executed.
     markLiveBranchesFromControlDependences();
 
-    if (Worklist.empty()) {
-      // Temporary until we can actually delete branches.
-      SmallVector<TerminatorInst *, 16> DeadTerminators;
-      for (auto *BB : BlocksWithDeadTerminators)
-        DeadTerminators.push_back(BB->getTerminator());
-      for (auto *I : DeadTerminators)
-        markLive(I);
-      assert(BlocksWithDeadTerminators.empty());
-      // End temporary.
-    }
   } while (!Worklist.empty());
-
-  assert(BlocksWithDeadTerminators.empty());
 }
 
 void AggressiveDeadCodeElimination::markLive(Instruction *I) {
@@ -316,13 +348,26 @@
   Info.Live = true;
   Worklist.push_back(I);
 
+  // Collect the live debug info scopes attached to this instruction.
+  if (const DILocation *DL = I->getDebugLoc())
+    collectLiveScopes(*DL);
+
   // Mark the containing block live
   auto &BBInfo = *Info.Block;
-  if (BBInfo.Terminator == I)
+  if (BBInfo.Terminator == I) {
     BlocksWithDeadTerminators.erase(BBInfo.BB);
+    // For live terminators, mark destination blocks
+    // live to preserve this control flow edges.
+    if (!BBInfo.UnconditionalBranch)
+      for (auto *BB : successors(I->getParent()))
+        markLive(BB);
+  }
+  markLive(BBInfo);
+}
+
+void AggressiveDeadCodeElimination::markLive(BlockInfoType &BBInfo) {
   if (BBInfo.Live)
     return;
-
   DEBUG(dbgs() << "mark block live: " << BBInfo.BB->getName() << '\n');
   BBInfo.Live = true;
   if (!BBInfo.CFLive) {
@@ -332,7 +377,7 @@
 
   // Mark unconditional branches at the end of live
   // blocks as live since there is no work to do for them later
-  if (BBInfo.UnconditionalBranch && I != BBInfo.Terminator)
+  if (BBInfo.UnconditionalBranch)
     markLive(BBInfo.Terminator);
 }
 
@@ -421,8 +466,38 @@
 //===----------------------------------------------------------------------===//
 bool AggressiveDeadCodeElimination::removeDeadInstructions() {
 
+  // Updates control and dataflow around dead blocks
+  updateDeadRegions();
+
+  // Mark newly inserted branches live; as a side effect loop invalidates the
+  // pointers stored in BlockInfo.
+  for (Instruction *I : NewBranches)
+    InstInfo[I].Live = true;
+
+  DEBUG({
+    for (Instruction &I : instructions(F)) {
+      // Check if the instruction is alive.
+      if (isLive(&I))
+        continue;
+
+      if (auto *DII = dyn_cast<DbgInfoIntrinsic>(&I)) {
+        // Check if the scope of this variable location is alive.
+        if (AliveScopes.count(DII->getDebugLoc()->getScope()))
+          continue;
+
+        // If intrinsic is pointing at a live SSA value, there may be an
+        // earlier optimization bug: if we know the location of the variable,
+        // why isn't the scope of the location alive?
+        if (Value *V = DII->getVariableLocation())
+          if (Instruction *II = dyn_cast<Instruction>(V))
+            if (isLive(II))
+              dbgs() << "Dropping debug info for " << *DII << "\n";
+      }
+    }
+  });
+
   // The inverse of the live set is the dead set.  These are those instructions
-  // which have no side effects and do not influence the control flow or return
+  // that have no side effects and do not influence the control flow or return
   // value of the function, and may therefore be deleted safely.
   // NOTE: We reuse the Worklist vector here for memory efficiency.
   for (Instruction &I : instructions(F)) {
@@ -430,23 +505,12 @@
     if (isLive(&I))
       continue;
 
-    assert(!I.isTerminator() && "NYI: Removing Control Flow");
-
     if (auto *DII = dyn_cast<DbgInfoIntrinsic>(&I)) {
       // Check if the scope of this variable location is alive.
       if (AliveScopes.count(DII->getDebugLoc()->getScope()))
         continue;
 
       // Fallthrough and drop the intrinsic.
-      DEBUG({
-        // If intrinsic is pointing at a live SSA value, there may be an
-        // earlier optimization bug: if we know the location of the variable,
-        // why isn't the scope of the location alive?
-        if (Value *V = DII->getVariableLocation())
-          if (Instruction *II = dyn_cast<Instruction>(V))
-            if (isLive(II))
-              dbgs() << "Dropping debug info for " << *DII << "\n";
-      });
     }
 
     // Prepare to delete.
@@ -459,9 +523,163 @@
     I->eraseFromParent();
   }
 
+  for (auto BB : BlocksWithDeadTerminators) {
+    auto &DB = BlockInfo[BB];
+    if (!DB.Live) {
+      NumBranchesRemoved += 1;
+      DB.BB->eraseFromParent();
+    }
+  }
+
   return !Worklist.empty();
 }
 
+// A dead region is the set of dead blocks with a common live post-dominator.
+void AggressiveDeadCodeElimination::updateDeadRegions() {
+
+  DEBUG({
+    dbgs() << "final dead terminator blocks: " << '\n';
+    for (auto *BB : BlocksWithDeadTerminators)
+      dbgs() << '\t' << BB->getName()
+             << (BlockInfo[BB].Live ? " LIVE\n" : "\n");
+  });
+
+  // Set of blocks processed so far.
+  SmallPtrSet<BasicBlock *, 16> VisitedBlocks;
+
+  // Find an unprocsesed block which identifies a new dead region.
+  for (BasicBlock *BB : BlocksWithDeadTerminators) {
+    if (!VisitedBlocks.insert(BB).second)
+      continue;
+    updateDeadRegions(BB, &VisitedBlocks);
+  }
+}
+
+void AggressiveDeadCodeElimination::updateDeadRegions(
+    BasicBlock *BB, SmallPtrSet<BasicBlock *, 16> *VisitedBlocks) {
+
+  // Gather all basic blocks in a connected region
+  // of dead blocks.
+  SmallPtrSet<BasicBlock *, 16> DeadRegionBlocks;
+
+  // Live blocks which are control flow predecessors of some block
+  // in this region.
+  SmallPtrSet<BasicBlock *, 16> LivePredecessors;
+
+  // The unique live post-dominator for this set of blocks.
+  BasicBlock *LivePDOM = nullptr;
+
+  // Explore the connected component
+  SmallVector<BasicBlock *, 16> DeadWorklist;
+  auto addDeadBlock = [VisitedBlocks, &DeadWorklist](BasicBlock *BB) {
+    if (VisitedBlocks->insert(BB).second)
+      DeadWorklist.push_back(BB);
+  };
+
+  // Find connected component of dead blocks.
+  DeadWorklist.push_back(BB);
+  while (!DeadWorklist.empty()) {
+    BasicBlock *BB = DeadWorklist.pop_back_val();
+    assert(!isLive(BB->getTerminator()));
+    DeadRegionBlocks.insert(BB);
+
+    // If this block is not live, then scan to see if there
+    // are predecessors blocks which also have dead terminators and
+    // should be part of this region.
+    if (!BlockInfo[BB].Live) {
+      for (auto PredBB : predecessors(BB)) {
+        if (BlocksWithDeadTerminators.count(PredBB))
+          addDeadBlock(PredBB);
+        else
+          LivePredecessors.insert(PredBB);
+      }
+    } else
+      LivePredecessors.insert(BB);
+
+    // Scan for dead successors. A live successor will be the
+    // unique post-dominator for all of the blocks in this region.
+    for (auto SuccBB : successors(BB)) {
+      if (!BlockInfo[SuccBB].Live)
+        addDeadBlock(SuccBB);
+      else if (LivePDOM)
+        assert(LivePDOM == SuccBB);
+      else
+        LivePDOM = SuccBB;
+    }
+  }
+
+  // Update each live phi based on the new incoming edges.
+  for (auto it = LivePDOM->begin(); PHINode *PN = dyn_cast<PHINode>(it); ++it) {
+    if (!isLive(PN))
+      continue;
+
+    // Unique value which reaches LivePDOM from any predecessor in
+    // DeadRegionBlocks.
+    Value *DeadPathValue = nullptr;
+
+    // Remove reaching definitions from dead predecessors, record reaching value
+    // from these dead blocks.
+    for (int64_t i = PN->getNumIncomingValues() - 1; i >= 0; --i) {
+      auto *PredBB = PN->getIncomingBlock(i);
+      if (DeadRegionBlocks.count(PredBB) == 0)
+        continue;
+
+      if (!DeadPathValue)
+        DeadPathValue = PN->getIncomingValue(i);
+      else
+        assert(DeadPathValue == PN->getIncomingValue(i));
+      PN->removeIncomingValue(i, /*DeletePHIIfEmpty*/ false);
+    }
+    assert(DeadPathValue && "Failed to find value for dead predecessor");
+    // Add edges from live predecessors.
+    for (auto PredBB : LivePredecessors)
+      PN->addIncoming(DeadPathValue, PredBB);
+  }
+
+  // Update control flow edges to bypass dead blocks.
+  for (auto PredBB : LivePredecessors) {
+    auto PredTerm = PredBB->getTerminator();
+    if (!isLive(PredBB->getTerminator())) {
+      // This is a live block but the terminator is not live so all
+      // control reaches the LivePdom, so we make the terminator
+      // an unconditional branch to that destination (B29 in the example
+      // above).
+      makeUnconditional(PredBB, LivePDOM);
+      continue;
+    }
+
+    auto NumSucc = PredTerm->getNumSuccessors();
+    for (unsigned Idx = 0; Idx != NumSucc; ++Idx) {
+      auto Succ = PredTerm->getSuccessor(Idx);
+      if (DeadRegionBlocks.count(Succ))
+        PredTerm->setSuccessor(Idx, LivePDOM);
+    }
+  }
+}
+
+void AggressiveDeadCodeElimination::makeUnconditional(BasicBlock *BB,
+                                                      BasicBlock *Target) {
+  TerminatorInst *PredTerm = BB->getTerminator();
+  // Collect the live debug info scopes attached to this instruction.
+  if (const DILocation *DL = PredTerm->getDebugLoc())
+    collectLiveScopes(*DL);
+
+  // Just mark live an existing unconditional branch
+  if (isUnconditionalBranch(PredTerm)) {
+    PredTerm->setSuccessor(0, Target);
+    InstInfo[PredTerm].Live = true;
+    return;
+  }
+  DEBUG(dbgs() << "making unconditional " << BB->getName() << '\n');
+  NumBranchesRemoved += 1;
+  IRBuilder<> Builder(PredTerm);
+  auto NewTerm = Builder.CreateBr(Target);
+  // Don't update InstInfo to protect pointers into it from BlockInfoVec
+  NewBranches.push_back(NewTerm);
+  if (const DILocation *DL = PredTerm->getDebugLoc())
+    NewTerm->setDebugLoc(DL);
+}
+
 //===----------------------------------------------------------------------===//
 //
 // Pass Manager integration code
@@ -494,7 +712,8 @@
 
   void getAnalysisUsage(AnalysisUsage &AU) const override {
     AU.addRequired<PostDominatorTreeWrapperPass>();
-    AU.setPreservesCFG(); // TODO -- will remove when we start removing branches
+    if (!RemoveControlFlowFlag)
+      AU.setPreservesCFG();
     AU.addPreserved<GlobalsAAWrapperPass>();
   }
 };
Index: test/Transforms/ADCE/2002-05-23-ZeroArgPHITest.ll
===================================================================
--- test/Transforms/ADCE/2002-05-23-ZeroArgPHITest.ll
+++ test/Transforms/ADCE/2002-05-23-ZeroArgPHITest.ll
@@ -4,7 +4,8 @@
 ; removed even though there were uses still around.  Now the uses are filled
 ; in with a dummy value before the PHI is deleted.
 ;
-; RUN: opt < %s -adce
+; RUN: opt < %s -S -adce | grep bb1
+; RUN: opt < %s -S -adce -adce-remove-loops | FileCheck %s
 	
         %node_t = type { double*, %node_t*, %node_t**, double**, double*, i32, i32 }
 
@@ -14,6 +15,8 @@
         store %node_t* %nodelist, %node_t** %nodelist.upgrd.1
         br label %bb1
 
+; CHECK-NOT: bb1:
+; CHECK-NOT: bb2:
 bb1:            ; preds = %bb0
         %reg107 = load %node_t*, %node_t** %nodelist.upgrd.1              ; <%node_t*> [#uses=2]
         %cond211 = icmp eq %node_t* %reg107, null               ; <i1> [#uses=1]
Index: test/Transforms/ADCE/2002-05-28-Crash-distilled.ll
===================================================================
--- test/Transforms/ADCE/2002-05-28-Crash-distilled.ll
+++ test/Transforms/ADCE/2002-05-28-Crash-distilled.ll
@@ -1,12 +1,14 @@
 ; This testcase is a distilled form of: 2002-05-28-Crash.ll
 
 ; RUN: opt < %s -adce 
+; RUN: opt < %s -adce -adce-remove-loops -S | FileCheck %s
 
 define float @test(i32 %i) {
         %F = sitofp i32 %i to float             ; <float> [#uses=1]
         %I = bitcast i32 %i to i32              ; <i32> [#uses=1]
         br label %Loop
 
+; CHECK-NOT: Loop:
 Loop:           ; preds = %Loop, %0
         %B = icmp ne i32 %I, 0          ; <i1> [#uses=1]
         br i1 %B, label %Out, label %Loop
Index: test/Transforms/ADCE/2002-05-28-Crash.ll
===================================================================
--- test/Transforms/ADCE/2002-05-28-Crash.ll
+++ test/Transforms/ADCE/2002-05-28-Crash.ll
@@ -12,6 +12,7 @@
 ;}
 ;
 ; RUN: opt < %s -adce
+; RUN: opt < %s -adce -adce-remove-loops -S | FileCheck %s
 
 define i32 @rx_bitset_empty(i32 %size, i32* %set) {
 bb1:
@@ -30,6 +31,7 @@
         %cond232 = icmp ne i32 %reg125, 0               ; <i1> [#uses=1]
         br i1 %cond232, label %bb3, label %bb2
 
+; CHECK-NOT: bb2:
 bb2:            ; preds = %bb2, %bb1
         %cann-indvar = phi i32 [ 0, %bb1 ], [ %add1-indvar, %bb2 ]              ; <i32> [#uses=2]
         %reg130-scale = mul i32 %cann-indvar, -1                ; <i32> [#uses=1]
Index: test/Transforms/ADCE/2002-07-17-AssertionFailure.ll
===================================================================
--- test/Transforms/ADCE/2002-07-17-AssertionFailure.ll
+++ test/Transforms/ADCE/2002-07-17-AssertionFailure.ll
@@ -3,11 +3,12 @@
 ; block in this function, it would work fine, but that would be the part we 
 ; have to fix now, wouldn't it....
 ;
-; RUN: opt < %s -adce
+; RUN: opt < %s -adce -S | FileCheck %s
 
 define void @foo(i8* %reg5481) {
         %cast611 = bitcast i8* %reg5481 to i8**         ; <i8**> [#uses=1]
         %reg162 = load i8*, i8** %cast611            ; <i8*> [#uses=1]
+; CHECK-NOT: ptrtoint
         ptrtoint i8* %reg162 to i32             ; <i32>:1 [#uses=0]
         ret void
 }
Index: test/Transforms/ADCE/2002-07-17-PHIAssertion.ll
===================================================================
--- test/Transforms/ADCE/2002-07-17-PHIAssertion.ll
+++ test/Transforms/ADCE/2002-07-17-PHIAssertion.ll
@@ -1,6 +1,6 @@
 ; This testcase was extracted from the gzip SPEC benchmark
 ;
-; RUN: opt < %s -adce
+; RUN: opt < %s -adce | FileCheck %s
 
 @bk = external global i32               ; <i32*> [#uses=2]
 @hufts = external global i32            ; <i32*> [#uses=1]
@@ -16,6 +16,8 @@
 bb3:            ; preds = %bb2
         br label %UnifiedExitNode
 
+; CHECK-NOT: bb4:
+; CHECK-NOT: bb5:
 bb4:            ; preds = %bb2
         %reg117 = load i32, i32* @hufts              ; <i32> [#uses=2]
         %cond241 = icmp ule i32 %reg117, %reg128                ; <i1> [#uses=1]
Index: test/Transforms/ADCE/2002-07-29-Segfault.ll
===================================================================
--- test/Transforms/ADCE/2002-07-29-Segfault.ll
+++ test/Transforms/ADCE/2002-07-29-Segfault.ll
@@ -1,4 +1,5 @@
 ; RUN: opt < %s -adce -disable-output
+; RUN: opt < %s -adce -disable-output -adce-remove-loops
 
 define void @test() {
         br label %BB3
Index: test/Transforms/ADCE/2003-01-22-PredecessorProblem.ll
===================================================================
--- test/Transforms/ADCE/2003-01-22-PredecessorProblem.ll
+++ test/Transforms/ADCE/2003-01-22-PredecessorProblem.ll
@@ -1,14 +1,17 @@
 ; Testcase reduced from 197.parser by bugpoint
 ; RUN: opt < %s -adce 
+; RUN: opt < %s -adce -adce-remove-loops -S | FileCheck %s
 
 define void @conjunction_prune() {
 ; <label>:0
         br label %bb19
 
+; CHECK-NOT: bb19:
 bb19:           ; preds = %bb23, %bb22, %0
         %reg205 = phi i8* [ null, %bb22 ], [ null, %bb23 ], [ null, %0 ]                ; <i8*> [#uses=1]
         br i1 false, label %bb21, label %bb22
 
+; CHECK-NOT: bb21:
 bb21:           ; preds = %bb19
         %cast455 = bitcast i8* %reg205 to i8**          ; <i8**> [#uses=0]
         br label %bb22
Index: test/Transforms/ADCE/2003-04-25-PHIPostDominateProblem.ll
===================================================================
--- test/Transforms/ADCE/2003-04-25-PHIPostDominateProblem.ll
+++ test/Transforms/ADCE/2003-04-25-PHIPostDominateProblem.ll
@@ -2,7 +2,8 @@
 ; entries for it's postdominator.  But I think this can only happen when the 
 ; PHI node is dead, so we just avoid patching up dead PHI nodes.
 
-; RUN: opt < %s -adce
+; RUN: opt < %s -adce                    -S | FileCheck %s
+; RUN: opt < %s -adce -adce-remove-loops -S | FileCheck %s
 
 target datalayout = "e-p:32:32"
 
@@ -14,6 +15,8 @@
         %k.1 = phi i32 [ %k.0, %endif ], [ 0, %entry ]          ; <i32> [#uses=1]
         br i1 false, label %no_exit, label %return
 
+; CHECK-NOT: then:
+; CHECK-NOT: else:
 no_exit:                ; preds = %loopentry
         br i1 false, label %then, label %else
 
Index: test/Transforms/ADCE/2003-06-11-InvalidCFG.ll
===================================================================
--- test/Transforms/ADCE/2003-06-11-InvalidCFG.ll
+++ test/Transforms/ADCE/2003-06-11-InvalidCFG.ll
@@ -1,4 +1,5 @@
 ; RUN: opt < %s -adce -disable-output
+; RUN: opt < %s -adce -adce-remove-loops -disable-output
 
 @G = external global i32*               ; <i32**> [#uses=1]
 
Index: test/Transforms/ADCE/2003-06-24-BadSuccessor.ll
===================================================================
--- test/Transforms/ADCE/2003-06-24-BadSuccessor.ll
+++ test/Transforms/ADCE/2003-06-24-BadSuccessor.ll
@@ -1,4 +1,6 @@
 ; RUN: opt < %s -adce -disable-output
+; RUN: opt < %s -adce -adce-remove-loops=true -disable-output
+
 target datalayout = "e-p:32:32"
 	%struct..CppObjTypeDesc = type { i32, i16, i16 }
 	%struct..TypeToken = type { i32, i16, i16 }
@@ -30,6 +32,7 @@
 	br i1 false, label %no_exit.1, label %loopentry.0
 
 no_exit.1:		; preds = %loopentry.1
+; CHECK: switch
 	switch i32 0, label %label.17 [
 		 i32 2, label %label.11
 		 i32 19, label %label.10
Index: test/Transforms/ADCE/2003-06-24-BasicFunctionality.ll
===================================================================
--- test/Transforms/ADCE/2003-06-24-BasicFunctionality.ll
+++ test/Transforms/ADCE/2003-06-24-BasicFunctionality.ll
@@ -1,4 +1,5 @@
-; RUN: opt < %s -adce -simplifycfg -S | not grep then:
+; RUN: opt < %s -adce -S | FileCheck %s
+; RUN: opt < %s -adce -adce-remove-loops -S | FileCheck %s
 
 define void @dead_test8(i32* %data.1, i32 %idx.1) {
 entry:
@@ -20,6 +21,10 @@
         %tmp.161 = icmp ne i32 %k.1, %tmp.14            ; <i1> [#uses=1]
         br i1 %tmp.161, label %then, label %else
 
+; CHECK-NOT: %tmp.161
+; CHECK-NOT: then:
+; CHECK-NOT: else
+
 then:           ; preds = %no_exit
         %inc.0 = add i32 %k.1, 1                ; <i32> [#uses=1]
         br label %endif
Index: test/Transforms/ADCE/2003-09-15-InfLoopCrash.ll
===================================================================
--- test/Transforms/ADCE/2003-09-15-InfLoopCrash.ll
+++ test/Transforms/ADCE/2003-09-15-InfLoopCrash.ll
@@ -1,4 +1,5 @@
 ; RUN: opt < %s -adce -disable-output
+; RUN: opt < %s -adce -adce-remove-loops -disable-output
 
 define i32 @main() {
         br label %loop
Index: test/Transforms/ADCE/2003-11-16-MissingPostDominanceInfo.ll
===================================================================
--- test/Transforms/ADCE/2003-11-16-MissingPostDominanceInfo.ll
+++ test/Transforms/ADCE/2003-11-16-MissingPostDominanceInfo.ll
@@ -1,4 +1,6 @@
 ; RUN: opt < %s -adce -simplifycfg -S | grep call
+; RUN: opt < %s -adce -adce-remove-loops -simplifycfg -S | grep call
+
 declare void @exit(i32)
 
 define i32 @main(i32 %argc) {
Index: test/Transforms/ADCE/2004-05-04-UnreachableBlock.ll
===================================================================
--- test/Transforms/ADCE/2004-05-04-UnreachableBlock.ll
+++ test/Transforms/ADCE/2004-05-04-UnreachableBlock.ll
@@ -1,4 +1,5 @@
 ; RUN: opt < %s -adce -disable-output
+; RUN: opt < %s -adce -adce-remove-loops -disable-output
 
 define void @test() {
 entry:
Index: test/Transforms/ADCE/2016-09-06.ll
===================================================================
--- /dev/null
+++ test/Transforms/ADCE/2016-09-06.ll
@@ -0,0 +1,59 @@
+; RUN: opt < %s -sroa -adce -adce-remove-loops -S | FileCheck %s
+; ModuleID = 'test1.bc'
+source_filename = "test1.c"
+target datalayout = "e-m:e-i64:64-f80:128-n8:16:32:64-S128"
+target triple = "x86_64-unknown-linux-gnu"
+
+; Function Attrs: nounwind uwtable
+define i32 @foo(i32, i32, i32) #0 {
+  %4 = alloca i32, align 4
+  %5 = alloca i32, align 4
+  %6 = alloca i32, align 4
+  %7 = alloca i32, align 4
+  %8 = alloca i32, align 4
+  store i32 %0, i32* %4, align 4
+  store i32 %1, i32* %5, align 4
+  store i32 %2, i32* %6, align 4
+  store i32 0, i32* %7, align 4
+  %9 = load i32, i32* %5, align 4
+  %I10 = icmp ne i32 %9, 0
+  br i1 %I10, label %B11, label %B21
+
+B11: 
+  store i32 0, i32* %8, align 4
+  br label %B12
+
+; CHECK-NOT: B12:
+; CHECK-NOT: B16:
+; CHECK-NOT: B17:
+; CHECK-NOT: B20:
+
+B12:
+  %I13 = load i32, i32* %8, align 4
+  %I14 = load i32, i32* %6, align 4
+  %I15 = icmp slt i32 %I13, %I14
+  br i1 %I15, label %B16, label %B20
+
+B16: 
+  br label %B17
+
+B17: 
+  %I18 = load i32, i32* %8, align 4
+  %I19 = add nsw i32 %I18, 1
+  store i32 %I19, i32* %8, align 4
+  br label %B12
+
+B20:
+  store i32 1, i32* %7, align 4
+  br label %B21
+
+B21: 
+  %I22 = load i32, i32* %7, align 4
+  ret i32 %I22
+}
+
+attributes #0 = { nounwind uwtable "correctly-rounded-divide-sqrt-fp-math"="false" "disable-tail-calls"="false" "less-precise-fpmad"="false" "no-frame-pointer-elim"="true" "no-frame-pointer-elim-non-leaf" "no-infs-fp-math"="false" "no-jump-tables"="false" "no-nans-fp-math"="false" "no-signed-zeros-fp-math"="false" "no-trapping-math"="false" "stack-protector-buffer-size"="8" "target-cpu"="x86-64" "target-features"="+fxsr,+mmx,+sse,+sse2,+x87" "unsafe-fp-math"="false" "use-soft-float"="false" }
+
+!llvm.ident = !{!0}
+
+!0 = !{!"clang version 4.0.0 (http://llvm.org/git/clang.git 5864a13abf4490e76ae2eb0896198e1305927df2)"}
Index: test/Transforms/ADCE/basictest1.ll
===================================================================
--- test/Transforms/ADCE/basictest1.ll
+++ test/Transforms/ADCE/basictest1.ll
@@ -1,4 +1,6 @@
-; RUN: opt < %s -adce -simplifycfg | llvm-dis	
+; RUN: opt < %s -adce -S | FileCheck %s
+; RUN: opt < %s -adce -adce-remove-loops -S | FileCheck %s
+
 %FILE = type { i32, i8*, i8*, i8, i8, i32, i32, i32 }
 	%spec_fd_t = type { i32, i32, i32, i8* }
 @__iob = external global [20 x %FILE]		; <[20 x %FILE]*> [#uses=1]
@@ -26,6 +28,8 @@
 	%cond266 = icmp sle i32 %reg109, 4		; <i1> [#uses=1]
 	br i1 %cond266, label %bb3, label %bb2
 
+; CHECK-NOT: bb2:
+
 bb2:		; preds = %0
 	%cast273 = getelementptr [17 x i8], [17 x i8]* @.LC12, i64 0, i64 0		; <i8*> [#uses=0]
 	br label %bb3
@@ -57,6 +61,9 @@
 	%cond271 = icmp sle i32 %reg134, 4		; <i1> [#uses=1]
 	br i1 %cond271, label %bb8, label %bb7
 
+; CHECK-NOT: bb7:
+; CHECK-NOT: bb8:
+
 bb7:		; preds = %bb6
 	%cast277 = getelementptr [4 x i8], [4 x i8]* @.LC10, i64 0, i64 0		; <i8*> [#uses=0]
 	br label %bb8
@@ -79,6 +86,8 @@
 	%cond272 = icmp sle i32 %reg163, 4		; <i1> [#uses=1]
 	br i1 %cond272, label %bb11, label %bb10
 
+; CHECK-NOT: bb10:
+
 bb10:		; preds = %bb9
 	%cast279 = getelementptr [4 x i8], [4 x i8]* @.LC11, i64 0, i64 0		; <i8*> [#uses=0]
 	br label %bb11
Index: test/Transforms/ADCE/basictest2.ll
===================================================================
--- test/Transforms/ADCE/basictest2.ll
+++ test/Transforms/ADCE/basictest2.ll
@@ -1,4 +1,6 @@
-; RUN: opt < %s -adce -simplifycfg | llvm-dis
+; RUN: opt < %s -adce -disable-output
+; RUN: opt < %s -adce -adce-remove-loops -S | FileCheck %s
+
 	%FILE = type { i32, i8*, i8*, i8, i8, i32, i32, i32 }
 	%spec_fd_t = type { i32, i32, i32, i8* }
 @__iob = external global [20 x %FILE]		; <[20 x %FILE]*> [#uses=1]
@@ -26,6 +28,7 @@
 	%cond266 = icmp sle i32 %reg109, 4		; <i1> [#uses=1]
 	br i1 %cond266, label %bb3, label %bb2
 
+; CHECK-NOT: bb2:
 bb2:		; preds = %0
 	%cast273 = getelementptr [17 x i8], [17 x i8]* @.LC12, i64 0, i64 0		; <i8*> [#uses=0]
 	br label %bb3
@@ -52,6 +55,8 @@
 	%cond270 = icmp slt i32 %reg1321, %reg1331		; <i1> [#uses=1]
 	br i1 %cond270, label %bb9, label %bb6
 
+; CHECK-NOT: bb7:
+; CHECK-NOT: bb8:
 bb6:		; preds = %bb5
 	%reg134 = load i32, i32* @dbglvl		; <i32> [#uses=1]
 	%cond271 = icmp sle i32 %reg134, 4		; <i1> [#uses=1]
@@ -79,6 +84,7 @@
 	%cond272 = icmp sle i32 %reg163, 4		; <i1> [#uses=1]
 	br i1 %cond272, label %bb11, label %bb10
 
+; CHECK-NOT: bb10:
 bb10:		; preds = %bb9
 	%cast279 = getelementptr [4 x i8], [4 x i8]* @.LC11, i64 0, i64 0		; <i8*> [#uses=0]
 	br label %bb11