Files
swift-mirror/lib/SIL/Utils/PrunedLiveness.cpp
Michael Gottesman 04144d8214 [pruned-liveness] Change PrunedLiveBlocks so that it can propagate multiple bits of liveness rather than just one.
This is useful for making a form of PrunedLiveness that is sensitive to address
fields. I wired up the current PrunedLiveness to just use PrunedLiveBlocks with
num bits set to 1.
2022-08-22 12:11:18 -07:00

362 lines
13 KiB
C++

//===--- PrunedLiveness.cpp - Compute liveness from selected uses ---------===//
//
// This source file is part of the Swift.org open source project
//
// Copyright (c) 2014 - 2020 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
//
//===----------------------------------------------------------------------===//
#include "swift/SIL/PrunedLiveness.h"
#include "swift/SIL/BasicBlockDatastructures.h"
#include "swift/SIL/BasicBlockUtils.h"
#include "swift/SIL/OwnershipUtils.h"
using namespace swift;
/// Mark blocks live during a reverse CFG traversal from one specific block
/// containing a user.
void PrunedLiveBlocks::computeUseBlockLiveness(SILBasicBlock *userBB,
unsigned bitNo) {
// If, we are visiting this block, then it is not already LiveOut. Mark it
// LiveWithin to indicate a liveness boundary within the block.
markBlockLive(userBB, LiveWithin, bitNo);
SmallVector<SILBasicBlock *, 8> predBBWorklist({userBB});
while (!predBBWorklist.empty()) {
SILBasicBlock *bb = predBBWorklist.pop_back_val();
// The popped `bb` is live; now mark all its predecessors LiveOut.
//
// Traversal terminates at any previously visited block, including the
// blocks initialized as definition blocks.
for (auto *predBB : bb->getPredecessorBlocks()) {
switch (getBlockLiveness(predBB, bitNo)) {
case Dead:
predBBWorklist.push_back(predBB);
LLVM_FALLTHROUGH;
case LiveWithin:
markBlockLive(predBB, LiveOut, bitNo);
break;
case LiveOut:
break;
}
}
}
}
/// Update the current def's liveness based on one specific use instruction.
///
/// Return the updated liveness of the \p use block (LiveOut or LiveWithin).
///
/// Terminators are not live out of the block.
PrunedLiveBlocks::IsLive PrunedLiveBlocks::updateForUse(SILInstruction *user,
unsigned bitNo) {
SWIFT_ASSERT_ONLY(seenUse = true);
auto *bb = user->getParent();
switch (getBlockLiveness(bb, bitNo)) {
case LiveOut:
return LiveOut;
case LiveWithin:
return LiveWithin;
case Dead: {
// This use block has not yet been marked live. Mark it and its predecessor
// blocks live.
computeUseBlockLiveness(bb, bitNo);
return getBlockLiveness(bb, bitNo);
}
}
llvm_unreachable("covered switch");
}
//===----------------------------------------------------------------------===//
// MARK: PrunedLiveness
//===----------------------------------------------------------------------===//
void PrunedLiveness::updateForUse(SILInstruction *user, bool lifetimeEnding) {
auto useBlockLive = liveBlocks.updateForUse(user, 0);
// Record all uses of blocks on the liveness boundary. For blocks marked
// LiveWithin, the boundary is considered to be the last use in the block.
//
// FIXME: Why is nonLifetimeEndingUsesInLiveOut inside PrunedLiveness, and
// what does it mean? Blocks may transition to LiveOut later. Or they may
// already be LiveOut from a previous use. After computing liveness, clients
// should check uses that are in PrunedLivenessBoundary.
if (!lifetimeEnding && useBlockLive == PrunedLiveBlocks::LiveOut) {
if (nonLifetimeEndingUsesInLiveOut)
nonLifetimeEndingUsesInLiveOut->insert(user);
return;
}
// Note that a user may use the current value from multiple operands. If any
// of the uses are non-lifetime-ending, then we must consider the user
// itself non-lifetime-ending; it cannot be a final destroy point because
// the value of the non-lifetime-ending operand must be kept alive until the
// end of the user. Consider a call that takes the same value using
// different conventions:
//
// apply %f(%val, %val) : $(@guaranteed, @owned) -> ()
//
// This call is not considered the end of %val's lifetime. The @owned
// argument must be copied.
auto iterAndSuccess = users.insert({user, lifetimeEnding});
if (!iterAndSuccess.second)
iterAndSuccess.first->second &= lifetimeEnding;
}
bool PrunedLiveness::updateForBorrowingOperand(Operand *op) {
assert(op->getOperandOwnership() == OperandOwnership::Borrow);
// A nested borrow scope is considered a use-point at each scope ending
// instruction.
//
// TODO: Handle reborrowed copies by considering the extended borrow
// scope. Temporarily bail-out on reborrows because we can't handle uses
// that aren't dominated by currentDef.
if (!BorrowingOperand(op).visitScopeEndingUses([this](Operand *end) {
if (end->getOperandOwnership() == OperandOwnership::Reborrow) {
return false;
}
updateForUse(end->getUser(), /*lifetimeEnding*/ false);
return true;
})) {
return false;
}
return true;
}
void PrunedLiveness::extendAcrossLiveness(PrunedLiveness &otherLivesness) {
// update this liveness for all the interesting users in otherLiveness.
for (std::pair<SILInstruction *, bool> userAndEnd : otherLivesness.users) {
updateForUse(userAndEnd.first, userAndEnd.second);
}
}
bool PrunedLiveness::isWithinBoundaryHelper(SILInstruction *inst,
SILValue def) const {
SILBasicBlock *block = inst->getParent();
/// Returns true if \p inst is before \p def in this block.
auto foundInBlockBeforeDef = [](SILInstruction *inst, SILBasicBlock *block,
SILValue def) {
if (!def || def->getParentBlock() != block) {
return false;
}
auto *defInst = def->getDefiningInstruction();
if (!defInst) {
return false;
}
// Check if instruction is before the definition
for (SILInstruction &it :
make_range(block->begin(), defInst->getIterator())) {
if (&it == inst) {
return true;
}
}
return false;
};
switch (getBlockLiveness(block)) {
case PrunedLiveBlocks::Dead:
return false;
case PrunedLiveBlocks::LiveOut:
return !foundInBlockBeforeDef(inst, block, def);
case PrunedLiveBlocks::LiveWithin:
if (foundInBlockBeforeDef(inst, block, def)) {
return false;
}
// The boundary is within this block. This instruction is before the
// boundary iff any interesting uses occur after it.
for (SILInstruction &it :
make_range(std::next(inst->getIterator()), block->end())) {
switch (isInterestingUser(&it)) {
case PrunedLiveness::NonUser:
break;
case PrunedLiveness::NonLifetimeEndingUse:
case PrunedLiveness::LifetimeEndingUse:
return true;
}
}
return false;
}
}
bool PrunedLiveness::isWithinBoundary(SILInstruction *inst) const {
return isWithinBoundaryHelper(inst, /*def*/ SILValue());
}
bool PrunedLiveness::isWithinBoundaryOfDef(SILInstruction *inst,
SILValue def) const {
return isWithinBoundaryHelper(inst, def);
}
bool PrunedLiveness::areUsesWithinBoundaryHelper(
ArrayRef<Operand *> uses, SILValue def,
DeadEndBlocks *deadEndBlocks) const {
auto checkDeadEnd = [deadEndBlocks](SILInstruction *inst) {
return deadEndBlocks && deadEndBlocks->isDeadEnd(inst->getParent());
};
for (auto *use : uses) {
auto *user = use->getUser();
if (!isWithinBoundaryHelper(user, def) && !checkDeadEnd(user))
return false;
}
return true;
}
bool PrunedLiveness::areUsesWithinBoundary(ArrayRef<Operand *> uses,
DeadEndBlocks *deadEndBlocks) const {
return areUsesWithinBoundaryHelper(uses, SILValue(), deadEndBlocks);
}
bool PrunedLiveness::areUsesWithinBoundaryOfDef(
ArrayRef<Operand *> uses, SILValue def,
DeadEndBlocks *deadEndBlocks) const {
return areUsesWithinBoundaryHelper(uses, def, deadEndBlocks);
}
bool PrunedLiveness::areUsesOutsideBoundaryHelper(
ArrayRef<Operand *> uses, SILValue def,
DeadEndBlocks *deadEndBlocks) const {
auto checkDeadEnd = [deadEndBlocks](SILInstruction *inst) {
return deadEndBlocks && deadEndBlocks->isDeadEnd(inst->getParent());
};
for (auto *use : uses) {
auto *user = use->getUser();
if (isWithinBoundaryHelper(user, def) || checkDeadEnd(user))
return false;
}
return true;
}
bool PrunedLiveness::areUsesOutsideBoundary(
ArrayRef<Operand *> uses, DeadEndBlocks *deadEndBlocks) const {
return areUsesOutsideBoundaryHelper(uses, SILValue(), deadEndBlocks);
}
bool PrunedLiveness::areUsesOutsideBoundaryOfDef(
ArrayRef<Operand *> uses, SILValue def,
DeadEndBlocks *deadEndBlocks) const {
return areUsesOutsideBoundaryHelper(uses, def, deadEndBlocks);
}
// An SSA def meets all the criteria for pruned liveness--def dominates all
// uses with no holes in the liverange. The lifetime-ending uses are also
// recorded--destroy_value or end_borrow. However destroy_values may not
// jointly-post dominate if dead-end blocks are present.
void PrunedLiveness::computeSSALiveness(SILValue def) {
initializeDefBlock(def->getParentBlock());
for (Operand *use : def->getUses()) {
updateForUse(use->getUser(), use->isLifetimeEnding());
}
}
void PrunedLivenessBoundary::visitInsertionPoints(
llvm::function_ref<void(SILBasicBlock::iterator insertPt)> visitor,
DeadEndBlocks *deBlocks) {
for (SILInstruction *user : lastUsers) {
if (!isa<TermInst>(user)) {
visitor(std::next(user->getIterator()));
continue;
}
auto *predBB = user->getParent();
for (SILBasicBlock *succ : predBB->getSuccessors()) {
if (deBlocks && deBlocks->isDeadEnd(succ))
continue;
assert(succ->getSinglePredecessorBlock() == predBB);
visitor(succ->begin());
}
}
for (SILBasicBlock *edge : boundaryEdges) {
if (deBlocks && deBlocks->isDeadEnd(edge))
continue;
visitor(edge->begin());
}
}
// Use \p liveness to find the last use in \p bb and add it to \p
// boundary.lastUsers.
static void findLastUserInBlock(SILBasicBlock *bb,
PrunedLivenessBoundary &boundary,
const PrunedLiveness &liveness) {
for (auto instIter = bb->rbegin(), endIter = bb->rend(); instIter != endIter;
++instIter) {
auto *inst = &*instIter;
if (liveness.isInterestingUser(inst) == PrunedLiveness::NonUser)
continue;
boundary.lastUsers.push_back(inst);
return;
}
llvm_unreachable("No user in LiveWithin block");
}
void PrunedLivenessBoundary::compute(const PrunedLiveness &liveness) {
for (SILBasicBlock *bb : liveness.getDiscoveredBlocks()) {
// Process each block that has not been visited and is not LiveOut.
switch (liveness.getBlockLiveness(bb)) {
case PrunedLiveBlocks::LiveOut:
for (SILBasicBlock *succBB : bb->getSuccessors()) {
if (liveness.getBlockLiveness(succBB) == PrunedLiveBlocks::Dead) {
boundaryEdges.push_back(succBB);
}
}
break;
case PrunedLiveBlocks::LiveWithin: {
// The liveness boundary is inside this block. Insert a final destroy
// inside the block if it doesn't already have one.
findLastUserInBlock(bb, *this, liveness);
break;
}
case PrunedLiveBlocks::Dead:
llvm_unreachable("All discovered blocks must be live");
}
}
}
void PrunedLivenessBoundary::compute(const PrunedLiveness &liveness,
ArrayRef<SILBasicBlock *> postDomBlocks) {
if (postDomBlocks.empty())
return; // all paths must be dead-ends or infinite loops
BasicBlockWorklist blockWorklist(postDomBlocks[0]->getParent());
// Visit each post-dominating block as the starting point for a
// backward CFG traversal.
for (auto *bb : postDomBlocks) {
blockWorklist.push(bb);
}
while (auto *bb = blockWorklist.pop()) {
// Process each block that has not been visited and is not LiveOut.
switch (liveness.getBlockLiveness(bb)) {
case PrunedLiveBlocks::LiveOut:
// A lifetimeEndBlock may be determined to be LiveOut after analyzing
// the extended liveness. It is irrelevant for finding the boundary.
break;
case PrunedLiveBlocks::LiveWithin: {
// The liveness boundary is inside this block. Insert a final destroy
// inside the block if it doesn't already have one.
findLastUserInBlock(bb, *this, liveness);
break;
}
case PrunedLiveBlocks::Dead:
// Continue searching upward to find the pruned liveness boundary.
for (auto *predBB : bb->getPredecessorBlocks()) {
if (liveness.getBlockLiveness(predBB) == PrunedLiveBlocks::LiveOut) {
boundaryEdges.push_back(bb);
} else {
blockWorklist.pushIfNotVisited(predBB);
}
}
break;
}
}
}