#include "slang-ir-simplify-cfg.h" #include "slang-ir-insts.h" #include "slang-ir.h" namespace Slang { bool processFunc(IRGlobalValueWithCode* func) { auto firstBlock = func->getFirstBlock(); if (!firstBlock) return false; SharedIRBuilder sharedBuilder(func->getModule()); IRBuilder builder(&sharedBuilder); bool changed = false; List workList; HashSet processedBlock; workList.add(func->getFirstBlock()); while (workList.getCount()) { auto block = workList.getFirst(); workList.fastRemoveAt(0); while (block) { if (auto loop = as(block->getTerminator())) { // If continue block is unreachable, remove it. auto continueBlock = loop->getContinueBlock(); if (continueBlock && !continueBlock->hasMoreThanOneUse()) { loop->continueBlock.set(loop->getTargetBlock()); continueBlock->removeAndDeallocate(); } // If there isn't any actual back jumps into loop target, remove the header // and turn it into a normal branch. auto targetBlock = loop->getTargetBlock(); if (targetBlock->getPredecessors().getCount() == 1 && *targetBlock->getPredecessors().begin() == block) { builder.setInsertBefore(loop); List args; for (UInt i = 0; i < loop->getArgCount(); i++) { args.add(loop->getArg(i)); } builder.emitBranch(targetBlock, args.getCount(), args.getBuffer()); loop->removeAndDeallocate(); } } // If `block` does not end with an unconditional branch, bail. if (block->getTerminator()->getOp() != kIROp_unconditionalBranch) break; auto branch = as(block->getTerminator()); auto successor = branch->getTargetBlock(); // Only perform the merge if `block` is the only predecessor of `successor`. // We also need to make sure not to merge a block that serves as the // merge point in CFG. Such blocks will have more than one use. if (successor->hasMoreThanOneUse()) break; if (block->hasMoreThanOneUse()) break; changed = true; Index paramIndex = 0; auto inst = successor->getFirstDecorationOrChild(); while (inst) { auto next = inst->getNextInst(); if (inst->getOp() == kIROp_Param) { inst->replaceUsesWith(branch->getArg(paramIndex)); paramIndex++; } else { inst->removeFromParent(); inst->insertAtEnd(block); } inst = next; } branch->removeAndDeallocate(); assert(!successor->hasUses()); successor->removeAndDeallocate(); } for (auto successor : block->getSuccessors()) { if (processedBlock.Add(successor)) { workList.add(successor); } } } return changed; } bool simplifyCFG(IRModule* module) { bool changed = false; for (auto inst : module->getGlobalInsts()) { if (auto func = as(inst)) { changed |= processFunc(func); } } return changed; } bool simplifyCFG(IRGlobalValueWithCode* func) { return processFunc(func); } } // namespace Slang