Files
swift-mirror/SwiftCompilerSources/Sources/Optimizer/FunctionPasses/EscapeInfoDumper.swift
Erik Eckstein 87ae9e3a73 add passes to dump the results of EscapeInfo
And add test files which uses the passes for verification of EscapeInfo
2022-04-22 09:50:07 +02:00

135 lines
4.2 KiB
Swift

//===--- EscapeInfoDumper.swift - Dumps escape information ----------------===//
//
// 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
/// Dumps the results of escape analysis.
///
/// Dumps the EscapeInfo query results for all `alloc_stack` instructions in a function.
///
/// This pass is used for testing EscapeInfo.
let escapeInfoDumper = FunctionPass(name: "dump-escape-info", {
(function: Function, context: PassContext) in
print("Escape information for \(function.name):")
var escapeInfo = EscapeInfo(calleeAnalysis: context.calleeAnalysis)
for block in function.blocks {
for inst in block.instructions {
if let allocRef = inst as? AllocRefInst {
var results = Set<String>()
let escapes = escapeInfo.isEscaping(object: allocRef,
visitUse: { op, path, _ in
if op.instruction is ReturnInst {
results.insert("return[\(path)]")
return .ignore
}
return .continueWalking
},
visitDef: { def, path, followStores in
guard let arg = def as? FunctionArgument else {
return .continueWalkingUp
}
results.insert("arg\(arg.index)[\(path)]")
return .continueWalkingDown
})
let res: String
if escapes {
res = "global"
} else if results.isEmpty {
res = " - "
} else {
res = Array(results).sorted().joined(separator: ",")
}
print("\(res): \(allocRef)")
}
}
}
print("End function \(function.name)\n")
})
/// Dumps the results of address-related escape analysis.
///
/// Dumps the EscapeInfo query results for addresses escaping to function calls.
/// The `fix_lifetime` instruction is used as marker for addresses and values to query.
///
/// This pass is used for testing EscapeInfo.
let addressEscapeInfoDumper = FunctionPass(name: "dump-addr-escape-info", {
(function: Function, context: PassContext) in
print("Address escape information for \(function.name):")
var valuesToCheck = [Value]()
var applies = [Instruction]()
for block in function.blocks {
for inst in block.instructions {
switch inst {
case let fli as FixLifetimeInst:
valuesToCheck.append(fli.operand)
case is FullApplySite:
applies.append(inst)
default:
break
}
}
}
var escapeInfo = EscapeInfo(calleeAnalysis: context.calleeAnalysis)
// test `isEscaping(addressesOf:)`
for value in valuesToCheck {
print("value:\(value)")
for apply in applies {
let path = AliasAnalysis.getPtrOrAddressPath(for: value)
let escaping = escapeInfo.isEscaping(addressesOf: value, path: path,
visitUse: { op, _, _ in
let user = op.instruction
if user == apply {
return .markEscaping
}
if user is ReturnInst {
// Anything which is returned cannot escape to an instruction inside the function.
return .ignore
}
return .continueWalking
})
print(" \(escaping ? "==>" : "- ") \(apply)")
}
}
// test `canReferenceSameField` for each pair of `fix_lifetime`.
if !valuesToCheck.isEmpty {
for lhsIdx in 0..<(valuesToCheck.count - 1) {
for rhsIdx in (lhsIdx + 1) ..< valuesToCheck.count {
print("pair \(lhsIdx) - \(rhsIdx)")
let lhs = valuesToCheck[lhsIdx]
let rhs = valuesToCheck[rhsIdx]
print(lhs)
print(rhs)
if escapeInfo.canReferenceSameField(
lhs, path: AliasAnalysis.getPtrOrAddressPath(for: lhs),
rhs, path: AliasAnalysis.getPtrOrAddressPath(for: rhs)) {
print("may alias")
} else {
print("no alias")
}
}
}
}
print("End function \(function.name)\n")
})