2016-11-19 161 views
1

在ir中有一個分支我想完全刪除(condtion + branch + true_basic_block + false_basic_block)。它看起來像這樣:從llvm刪除完整分支ir

%4 = icmp sge i32 %2, %3 
br i1 %4, label %5, label %7 

; <label>:5          ; preds = %0 
%6 = load i32* %x, align 4 
store i32 %6, i32* %z, align 4 
br label %9 

; <label>:7          ; preds = %0 
%8 = load i32* %y, align 4 
store i32 %8, i32* %z, align 4 
br label %9 

; <label>:9          ; preds = %7, %5 
%10 = call dereferenceable(140) %"class.std::basic_ostream"*@_ZStlsISt11char_traitsIcEERSt13basic_ostreamIcT_ES5_PKc(%"class.std::basic_ostream"* dereferenceable(140) @_ZSt4cout, i8* getelementptr inbounds ([5 x i8]* @.str, i32 0, i32 0)) 
%11 = load i32* %z, align 4 
%12 = call dereferenceable(140) %"class.std::basic_ostream"* @_ZNSolsEi(%"class.std::basic_ostream"* %10, i32 %11) 
%13 = call dereferenceable(140) %"class.std::basic_ostream"* @_ZNSolsEPFRSoS_E(%"class.std::basic_ostream"* %12, %"class.std::basic_ostream"* (%"class.std::basic_ostream"*)* @_ZSt4endlIcSt11char_traitsIcEERSt13basic_ostreamIT_T0_ES6_) 
ret i32 0 

現在把它刪除,是否有removeBranch功能,或者我需要刪除的指令一個接一個。我一直在嘗試後一種方式,但我已經看到從「基本塊在主沒有終止符」的每個錯誤「使用仍然當def被破壞」,以及更多..我使用了erasefromparent,replaceinstwithvalue,replaceinstwithinst,removefromparent等

任何人都可以足以指出我在正確的方向嗎?

這是我function_pass:

bool runOnFunction(Function &F) override { 
    for (auto& B : F) 
    for (auto& I : B) 
      if(auto* brn = dyn_cast<BranchInst>(&I))  
       if(brn->isConditional()){ 
        Instruction* cond = dyn_cast<Instruction>(brn->getCondition()); 
        if(cond->getOpcode() == Instruction::ICmp){ 
         branch_vector.push_back(brn); 
         //removeConditionalBranch(dyn_cast<BranchInst>(brn)); 
        } 
       } 

/*For now just delete the branches in the vector.*/ 
for(auto b : branch_vector) 
     removeConditionalBranch(dyn_cast<BranchInst>(b)); 

return true; 
} 

這是輸出:output

回答

1

我不知道任何RemoveBranch實用功能的,但這樣的事情應該工作。這個想法是刪除分支指令,然後刪除任何因此而死的東西,然後將初始塊與連接塊合併。

// for DeleteDeadBlock, MergeBlockIntoPredecessor 
#include "llvm/Transforms/Utils/BasicBlockUtils.h" 
// for RecursivelyDeleteTriviallyDeadInstructions 
#include "llvm/Transforms/Utils/Local.h" 

void removeConditionalBranch(BranchInst *Branch) { 
    assert(Branch && 
     Branch->isConditional() && 
     Branch->getNumSuccessors() == 2); 

    BasicBlock *Parent = Branch->getParent(); 
    BasicBlock *ThenBlock = Branch->getSuccessor(0); 
    BasicBlock *ElseBlock = Branch->getSuccessor(1); 

    BasicBlock *ThenSuccessor = ThenBlock->getUniqueSuccessor(); 
    BasicBlock *ElseSuccessor = ElseBlock->getUniqueSuccessor(); 
    assert(ThenSuccessor && ElseSuccessor && ThenSuccessor == ElseSuccessor); 

    Branch->eraseFromParent(); 
    RecursivelyDeleteTriviallyDeadInstructions(Branch->getCondition()); 
    DeleteDeadBlock(ThenBlock); 
    DeleteDeadBlock(ElseBlock); 

    IRBuilder<> Builder(Parent); 
    Builder.CreateBr(ThenSuccessor); 
    bool Merged = MergeBlockIntoPredecessor(ThenSuccessor); 
    assert(Merged); 
} 

此代碼只處理您已經證明了簡單的情況下,與當時和else塊都無條件地轉移到共同的跳躍加入塊(它將失敗,對任何事情更復雜的斷言錯誤)。更復雜的控制流將會有點棘手,但您仍應該能夠使用此代碼作爲起點。

+0

我應該在遍歷指令時還是在迭代循環之外調用removeConditionalBranch(在迭代過程中將其存儲在向量中)?我在兩種方法中都遇到了分段錯誤... –

+0

@GauravSharma一般來說,在迭代它們時,不應該添加/刪除指令。我只是測試了這個,我沒有收到段錯誤,所以你必須提供更多的信息。 –

+0

@ Ismail Badawi:我已經添加了pass和它的輸出...請參閱.... –