Files
swift-mirror/lib/SILGen/ManagedValue.cpp
Michael Gottesman adcfda6f6d [silgenpattern] When binding values, use a ManagedValue instead of an RValue to avoid implicitly exploding tuples.
In SILGenPattern, we need to be able to unforward cleanups when we explode
tuples. Thus we can't use RValue in SILGenPattern since it may implicitly
explode tuples (which without modifying RValue itself we can not
unforward). This patch removes a specific RValue usage that we can replace with
the use of a ManagedValue instead.

rdar://49903264
2019-05-16 12:41:08 -07:00

245 lines
8.3 KiB
C++

//===--- ManagedValue.cpp - Value with cleanup ----------------------------===//
//
// This source file is part of the Swift.org open source project
//
// Copyright (c) 2014 - 2017 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
//
//===----------------------------------------------------------------------===//
//
// A storage structure for holding a destructured rvalue with an optional
// cleanup(s).
// Ownership of the rvalue can be "forwarded" to disable the associated
// cleanup(s).
//
//===----------------------------------------------------------------------===//
#include "ManagedValue.h"
#include "SILGenFunction.h"
using namespace swift;
using namespace Lowering;
/// Emit a copy of this value with independent ownership.
ManagedValue ManagedValue::copy(SILGenFunction &SGF, SILLocation loc) const {
auto &lowering = SGF.getTypeLowering(getType());
if (lowering.isTrivial())
return *this;
if (getType().isObject()) {
return SGF.B.createCopyValue(loc, *this, lowering);
}
SILValue buf = SGF.emitTemporaryAllocation(loc, getType());
SGF.B.createCopyAddr(loc, getValue(), buf, IsNotTake, IsInitialization);
return SGF.emitManagedRValueWithCleanup(buf, lowering);
}
/// Emit a copy of this value with independent ownership.
ManagedValue ManagedValue::formalAccessCopy(SILGenFunction &SGF,
SILLocation loc) {
assert(SGF.isInFormalEvaluationScope() &&
"Can only perform a formal access copy in a formal evaluation scope");
auto &lowering = SGF.getTypeLowering(getType());
if (lowering.isTrivial())
return *this;
if (getType().isObject()) {
return SGF.B.createFormalAccessCopyValue(loc, *this);
}
SILValue buf = SGF.emitTemporaryAllocation(loc, getType());
return SGF.B.createFormalAccessCopyAddr(loc, *this, buf, IsNotTake,
IsInitialization);
}
/// Store a copy of this value with independent ownership into the given
/// uninitialized address.
void ManagedValue::copyInto(SILGenFunction &SGF, SILLocation loc,
SILValue dest) {
auto &lowering = SGF.getTypeLowering(getType());
if (lowering.isAddressOnly() && SGF.silConv.useLoweredAddresses()) {
SGF.B.createCopyAddr(loc, getValue(), dest, IsNotTake, IsInitialization);
return;
}
SILValue copy = lowering.emitCopyValue(SGF.B, loc, getValue());
lowering.emitStoreOfCopy(SGF.B, loc, copy, dest, IsInitialization);
}
void ManagedValue::copyInto(SILGenFunction &SGF, SILLocation loc,
Initialization *dest) {
dest->copyOrInitValueInto(SGF, loc, *this, /*isInit*/ false);
dest->finishInitialization(SGF);
}
/// This is the same operation as 'copy', but works on +0 values that don't
/// have cleanups. It returns a +1 value with one.
ManagedValue ManagedValue::copyUnmanaged(SILGenFunction &SGF, SILLocation loc) {
if (getType().isObject()) {
return SGF.B.createCopyValue(loc, *this);
}
SILValue result = SGF.emitTemporaryAllocation(loc, getType());
SGF.B.createCopyAddr(loc, getValue(), result, IsNotTake, IsInitialization);
return SGF.emitManagedRValueWithCleanup(result);
}
/// This is the same operation as 'copy', but works on +0 values that don't
/// have cleanups. It returns a +1 value with one.
ManagedValue ManagedValue::formalAccessCopyUnmanaged(SILGenFunction &SGF,
SILLocation loc) {
assert(SGF.isInFormalEvaluationScope());
if (getType().isObject()) {
return SGF.B.createFormalAccessCopyValue(loc, *this);
}
SILValue result = SGF.emitTemporaryAllocation(loc, getType());
return SGF.B.createFormalAccessCopyAddr(loc, *this, result, IsNotTake,
IsInitialization);
}
/// Disable the cleanup for this value.
void ManagedValue::forwardCleanup(SILGenFunction &SGF) const {
assert(hasCleanup() && "value doesn't have cleanup!");
SGF.Cleanups.forwardCleanup(getCleanup());
}
/// Forward this value, deactivating the cleanup and returning the
/// underlying value.
SILValue ManagedValue::forward(SILGenFunction &SGF) const {
if (hasCleanup())
forwardCleanup(SGF);
return getValue();
}
void ManagedValue::forwardInto(SILGenFunction &SGF, SILLocation loc,
SILValue address) {
assert(isPlusOne(SGF));
auto &addrTL = SGF.getTypeLowering(address->getType());
SGF.emitSemanticStore(loc, forward(SGF), address, addrTL, IsInitialization);
}
void ManagedValue::assignInto(SILGenFunction &SGF, SILLocation loc,
SILValue address) {
assert(isPlusOne(SGF));
auto &addrTL = SGF.getTypeLowering(address->getType());
SGF.emitSemanticStore(loc, forward(SGF), address, addrTL,
IsNotInitialization);
}
void ManagedValue::forwardInto(SILGenFunction &SGF, SILLocation loc,
Initialization *dest) {
assert(isPlusOne(SGF));
dest->copyOrInitValueInto(SGF, loc, *this, /*isInit*/ true);
dest->finishInitialization(SGF);
}
ManagedValue ManagedValue::borrow(SILGenFunction &SGF, SILLocation loc) const {
assert(getValue() && "cannot borrow an invalid or in-context value");
if (isLValue())
return *this;
if (getType().isAddress())
return ManagedValue::forUnmanaged(getValue());
return SGF.emitManagedBeginBorrow(loc, getValue());
}
ManagedValue ManagedValue::formalAccessBorrow(SILGenFunction &SGF,
SILLocation loc) const {
assert(SGF.isInFormalEvaluationScope());
assert(getValue() && "cannot borrow an invalid or in-context value");
if (isLValue())
return *this;
if (getType().isAddress())
return ManagedValue::forUnmanaged(getValue());
return SGF.emitFormalEvaluationManagedBeginBorrow(loc, getValue());
}
ManagedValue ManagedValue::materialize(SILGenFunction &SGF,
SILLocation loc) const {
auto temporary = SGF.emitTemporaryAllocation(loc, getType());
bool hadCleanup = hasCleanup();
// The temporary memory is +0 if the value was.
if (hadCleanup) {
SGF.B.emitStoreValueOperation(loc, forward(SGF), temporary,
StoreOwnershipQualifier::Init);
// SEMANTIC SIL TODO: This should really be called a temporary LValue.
return ManagedValue::forOwnedAddressRValue(temporary,
SGF.enterDestroyCleanup(temporary));
} else {
auto object = SGF.emitManagedBeginBorrow(loc, getValue());
SGF.emitManagedStoreBorrow(loc, object.getValue(), temporary);
return ManagedValue::forBorrowedAddressRValue(temporary);
}
}
void ManagedValue::print(raw_ostream &os) const {
if (SILValue v = getValue()) {
v->print(os);
}
}
void ManagedValue::dump() const {
dump(llvm::errs());
}
void ManagedValue::dump(raw_ostream &os, unsigned indent) const {
os.indent(indent);
if (isInContext()) {
os << "InContext\n";
return;
}
if (isLValue()) os << "[lvalue] ";
if (hasCleanup()) os << "[cleanup] ";
if (SILValue v = getValue()) {
v->print(os);
} else {
os << "<null>\n";
}
}
ManagedValue ManagedValue::ensurePlusOne(SILGenFunction &SGF,
SILLocation loc) const {
// Undef can pair with any type of ownership, so it is effectively a +1 value.
if (isa<SILUndef>(getValue()))
return *this;
if (!isPlusOne(SGF)) {
return copy(SGF, loc);
}
return *this;
}
bool ManagedValue::isPlusOne(SILGenFunction &SGF) const {
// If this value is SILUndef, return true. SILUndef can always be passed to +1
// APIs.
if (isa<SILUndef>(getValue()))
return true;
// Ignore trivial values since for our purposes they are always at +1 since
// they can always be passed to +1 APIs.
if (getType().isTrivial(SGF.F))
return true;
// If we have an object and the object has any ownership, the same
// property applies.
if (getType().isObject() && getOwnershipKind() == ValueOwnershipKind::Any)
return true;
return hasCleanup();
}
bool ManagedValue::isPlusZero() const {
// SILUndef can always be passed to +0 APIs.
if (isa<SILUndef>(getValue()))
return true;
// Otherwise, just check if we have a cleanup.
return !hasCleanup();
}