[llvm] [WebAssembly] Incorporate SCCs into WebAssemblyFixIrreducibleControlFlow (PR #181755)
Derek Schuff via llvm-commits
llvm-commits at lists.llvm.org
Fri Feb 20 11:47:27 PST 2026
================
@@ -98,137 +107,142 @@ class ReachabilityGraph {
calculate();
}
- bool canReach(MachineBasicBlock *From, MachineBasicBlock *To) const {
- assert(inRegion(From) && inRegion(To));
- auto I = Reachable.find(From);
- if (I == Reachable.end())
- return false;
- return I->second.count(To);
- }
-
- // "Loopers" are blocks that are in a loop. We detect these by finding blocks
- // that can reach themselves.
- const BlockSet &getLoopers() const { return Loopers; }
-
// Get all blocks that are loop entries.
const BlockSet &getLoopEntries() const { return LoopEntries; }
+ const BlockSet &getLoopEntriesForSCC(unsigned SCCId) const {
+ return LoopEntriesBySCC[SCCId];
+ }
- // Get all blocks that enter a particular loop from outside.
- const BlockSet &getLoopEnterers(MachineBasicBlock *LoopEntry) const {
- assert(inRegion(LoopEntry));
- auto I = LoopEnterers.find(LoopEntry);
- assert(I != LoopEnterers.end());
- return I->second;
+ unsigned getSCCId(MachineBasicBlock *MBB) const {
+ return getNode(MBB)->SCCId;
}
+ friend struct GraphTraits<ReachabilityGraph *>;
+
private:
MachineBasicBlock *Entry;
const BlockSet &Blocks;
- BlockSet Loopers, LoopEntries;
- DenseMap<MachineBasicBlock *, BlockSet> LoopEnterers;
+ BlockSet LoopEntries;
+ SmallVector<BlockSet, 0> LoopEntriesBySCC;
bool inRegion(MachineBasicBlock *MBB) const { return Blocks.count(MBB); }
- // Maps a block to all the other blocks it can reach.
- DenseMap<MachineBasicBlock *, BlockSet> Reachable;
+ SmallVector<ReachabilityNode, 0> Nodes;
+ DenseMap<MachineBasicBlock *, ReachabilityNode *> MBBToNodeMap;
- void calculate() {
- // Reachability computation work list. Contains pairs of recent additions
- // (A, B) where we just added a link A => B.
- using BlockPair = std::pair<MachineBasicBlock *, MachineBasicBlock *>;
- SmallVector<BlockPair, 4> WorkList;
+ ReachabilityNode *getNode(MachineBasicBlock *MBB) const {
+ return MBBToNodeMap.at(MBB);
+ }
- // Add all relevant direct branches.
- for (auto *MBB : Blocks) {
- for (auto *Succ : MBB->successors()) {
- if (Succ != Entry && inRegion(Succ)) {
- Reachable[MBB].insert(Succ);
- WorkList.emplace_back(MBB, Succ);
- }
- }
- }
+ void calculate();
+};
+} // end anonymous namespace
- while (!WorkList.empty()) {
- MachineBasicBlock *MBB, *Succ;
- std::tie(MBB, Succ) = WorkList.pop_back_val();
- assert(inRegion(MBB) && Succ != Entry && inRegion(Succ));
- if (MBB != Entry) {
- // We recently added MBB => Succ, and that means we may have enabled
- // Pred => MBB => Succ.
- for (auto *Pred : MBB->predecessors()) {
- if (Reachable[Pred].insert(Succ).second) {
- WorkList.emplace_back(Pred, Succ);
- }
- }
- }
- }
+namespace llvm {
+template <> struct GraphTraits<ReachabilityGraph *> {
+ typedef ReachabilityNode NodeType;
+ typedef NodeType *NodeRef;
+ typedef SmallVectorImpl<NodeRef>::iterator ChildIteratorType;
----------------
dschuff wrote:
AFAIK yes. The advantage of `using` is that you can use it with templates, i.e. with typedef you can freeze the parameter like you do here, but with `using` you can also leave template parameters undefined for further instantiations later.
https://github.com/llvm/llvm-project/pull/181755
More information about the llvm-commits
mailing list