Files
swift-mirror/SwiftCompilerSources/Sources/Optimizer/Analysis/DominatorTree.swift

101 lines
3.0 KiB
Swift

//===--- DominatorTree.swift - the dominator tree -------------------------===//
//
// This source file is part of the Swift.org open source project
//
// Copyright (c) 2014 - 2022 Apple Inc. and the Swift project authors
// Licensed under Apache License v2.0 with Runtime Library Exception
//
// See https://swift.org/LICENSE.txt for license information
// See https://swift.org/CONTRIBUTORS.txt for the list of Swift project authors
//
//===----------------------------------------------------------------------===//
import SIL
import OptimizerBridging
struct DominatorTree {
let bridged: BridgedDomTree
func getChildren(of block: BasicBlock) -> DomChildren {
return DomChildren(bridgedDomTree: bridged, bb: block)
}
/// Returns the immediate dominator of `block`, i.e. the parent of `block` in the tree.
func getImmediateDominator(of block: BasicBlock) -> BasicBlock? {
bridged.getImmediateDominator(block.bridged).block
}
/// Returns the sub-dominator-tree of `startBlock` in dominance order.
///
/// Blocks - including their sub-tree - are only included if `filter` returns true.
func getDominanceOrder(startingAt startBlock: BasicBlock, filter: (BasicBlock) -> Bool) -> [BasicBlock] {
guard filter(startBlock) else {
return []
}
var order = [BasicBlock]()
order.append(startBlock)
var idx = 0
while idx < order.count {
let block = order[idx]
idx += 1
order.append(contentsOf: getChildren(of: block).lazy.filter(filter))
}
return order
}
}
struct DomChildren: BridgedRandomAccessCollection {
private let bridgedDomTree: BridgedDomTree
private let block: BasicBlock
let count: Int
var startIndex: Int { return 0 }
var endIndex: Int { return count }
init(bridgedDomTree: BridgedDomTree, bb: BasicBlock) {
self.bridgedDomTree = bridgedDomTree
self.block = bb
self.count = bridgedDomTree.getNumberOfChildren(bb.bridged)
}
subscript(_ index: Int) -> BasicBlock {
assert(index >= startIndex && index < endIndex)
return bridgedDomTree.getChildAt(block.bridged, index).block
}
}
extension BasicBlock {
func dominates(_ other: BasicBlock, _ domTree: DominatorTree) -> Bool {
domTree.bridged.dominates(self.bridged, other.bridged)
}
func strictlyDominates(_ other: BasicBlock, _ domTree: DominatorTree) -> Bool {
dominates(other, domTree) && self != other
}
}
//===--------------------------------------------------------------------===//
// Tests
//===--------------------------------------------------------------------===//
let domtreeTest = FunctionTest("domtree") {
function, arguments, context in
let domtree = context.dominatorTree
let startBlock = arguments.takeBlock()
let order = domtree.getDominanceOrder(startingAt: startBlock, filter: { block in
if let sli = block.instructions.first! as? StringLiteralInst,
sli.value == "skip"
{
return false
}
return true
})
print("order: \(order.map { $0.index })")
}