mirror of
https://github.com/apple/swift.git
synced 2025-12-14 20:36:38 +01:00
135 lines
4.2 KiB
Swift
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")
|
|
})
|