Skip to content

Commit a1c1b32

Browse files
committed
[MemAccessUtils] Added RelativeAccessStorageWithBase.
The new "relative" version of AccessStorageWithBase carries additional information about the walk from the specified address back to the base. For now, that includes the original address and the most transformative sort of cast that was encountered.
1 parent 8a0e5cd commit a1c1b32

File tree

3 files changed

+83
-17
lines changed

3 files changed

+83
-17
lines changed

include/swift/SIL/MemAccessUtils.h

+45-5
Original file line numberDiff line numberDiff line change
@@ -260,6 +260,16 @@ enum class NestedAccessType { StopAtAccessBegin, IgnoreAccessBegin };
260260
/// previous.
261261
enum class AccessUseType { Exact, Inner, Overlapping };
262262

263+
/// When walking from a value to its storage, casts may be encountered. The
264+
/// cases describe variety of encountered casts, categorized by the kind of
265+
/// transformation that the casts perform.
266+
///
267+
/// The enum values are ordered. Each successive cast kind is more
268+
/// transformative than the last.
269+
///
270+
/// TODO: Distinguish between LayoutEquivalent and LayoutCompatibile.
271+
enum class AccessStorageCast { Identity, Type };
272+
263273
/// The physical representation used to identify access information and common
264274
/// API used by both AccessBase and AccessStorage.
265275
///
@@ -898,6 +908,30 @@ struct AccessStorageWithBase {
898908
void dump() const;
899909
};
900910

911+
/// Extends AccessStorageWithBase by adding information that was obtained while
912+
/// visiting from a particular address, to which an instance of this is
913+
/// relative.
914+
struct RelativeAccessStorageWithBase {
915+
916+
/// Identical to AccessStorageWithBase::compute but preserves information
917+
/// specific to the walk from address;
918+
static RelativeAccessStorageWithBase compute(SILValue address);
919+
920+
/// Identical to AccessStorageWithBase::computeInScope but preserves
921+
/// information specific to the walk from address;
922+
static RelativeAccessStorageWithBase computeInScope(SILValue address);
923+
924+
/// The address to which this RelativeAccessStorageWithBase is relative.
925+
SILValue address;
926+
/// The underlying access storage and base.
927+
AccessStorageWithBase storageWithBase;
928+
/// The most transformative cast that was seen between when walking from
929+
/// address to storage.base;
930+
Optional<AccessStorageCast> cast;
931+
932+
AccessStorage getStorage() const { return storageWithBase.storage; }
933+
};
934+
901935
/// Return an AccessStorage value that identifies formally accessed storage
902936
/// for \p beginAccess, considering any outer access scope as having distinct
903937
/// storage from this access scope. This is useful for exclusivity checking
@@ -1637,8 +1671,9 @@ class AccessUseDefChainVisitor {
16371671
// Result visitBase(SILValue base, AccessStorage::Kind kind);
16381672
// Result visitNonAccess(SILValue base);
16391673
// Result visitPhi(SILPhiArgument *phi);
1640-
// Result visitStorageCast(SingleValueInstruction *cast, Operand *sourceOper);
1641-
// Result visitAccessProjection(SingleValueInstruction *projectedAddr,
1674+
// Result visitStorageCast(SingleValueInstruction *cast, Operand *sourceOper,
1675+
// AccessStorageCast cast); Result
1676+
// visitAccessProjection(SingleValueInstruction *projectedAddr,
16421677
// Operand *sourceOper);
16431678

16441679
Result visit(SILValue sourceAddr);
@@ -1650,8 +1685,12 @@ Result AccessUseDefChainVisitor<Impl, Result>::visit(SILValue sourceAddr) {
16501685
if (auto *projOper = getAccessProjectionOperand(svi))
16511686
return asImpl().visitAccessProjection(svi, projOper);
16521687

1653-
if (isAccessStorageCast(svi))
1654-
return asImpl().visitStorageCast(svi, &svi->getAllOperands()[0]);
1688+
if (isAccessStorageTypeCast(svi))
1689+
return asImpl().visitStorageCast(svi, &svi->getAllOperands()[0],
1690+
AccessStorageCast::Type);
1691+
if (isAccessStorageIdentityCast(svi))
1692+
return asImpl().visitStorageCast(svi, &svi->getAllOperands()[0],
1693+
AccessStorageCast::Identity);
16551694
}
16561695
switch (sourceAddr->getKind()) {
16571696
default:
@@ -1826,7 +1865,8 @@ class AccessUseDefChainCloner
18261865
return SILValue();
18271866
}
18281867

1829-
SILValue visitStorageCast(SingleValueInstruction *cast, Operand *sourceOper) {
1868+
SILValue visitStorageCast(SingleValueInstruction *cast, Operand *sourceOper,
1869+
AccessStorageCast) {
18301870
// The cloner does not currently know how to create compensating
18311871
// end_borrows or fix mark_dependence operands.
18321872
if (isa<BeginBorrowInst>(cast) || isa<MarkDependenceInst>(cast))

lib/SIL/Utils/MemAccessUtils.cpp

+37-11
Original file line numberDiff line numberDiff line change
@@ -125,7 +125,8 @@ class AccessPhiVisitor
125125
phiArg->getIncomingPhiValues(pointerWorklist);
126126
}
127127

128-
void visitStorageCast(SingleValueInstruction *cast, Operand *sourceOper) {
128+
void visitStorageCast(SingleValueInstruction *cast, Operand *sourceOper,
129+
AccessStorageCast) {
129130
// Allow conversions to/from pointers and addresses on disjoint phi paths
130131
// only if the underlying useDefVisitor allows it.
131132
if (storageCastTy == IgnoreStorageCast)
@@ -209,7 +210,8 @@ class FindAccessVisitorImpl : public AccessUseDefChainVisitor<Impl, SILValue> {
209210
return this->asImpl().visitNonAccess(phiArg);
210211
}
211212

212-
SILValue visitStorageCast(SingleValueInstruction *, Operand *sourceAddr) {
213+
SILValue visitStorageCast(SingleValueInstruction *, Operand *sourceAddr,
214+
AccessStorageCast cast) {
213215
assert(storageCastTy == IgnoreStorageCast);
214216
return sourceAddr->get();
215217
}
@@ -331,11 +333,12 @@ class FindAccessBaseVisitor
331333
}
332334

333335
// Override visitStorageCast to avoid seeing through arbitrary address casts.
334-
SILValue visitStorageCast(SingleValueInstruction *cast, Operand *sourceAddr) {
336+
SILValue visitStorageCast(SingleValueInstruction *svi, Operand *sourceAddr,
337+
AccessStorageCast cast) {
335338
if (storageCastTy == StopAtStorageCast)
336-
return visitNonAccess(cast);
339+
return visitNonAccess(svi);
337340

338-
return SuperTy::visitStorageCast(cast, sourceAddr);
341+
return SuperTy::visitStorageCast(svi, sourceAddr, cast);
339342
}
340343
};
341344

@@ -1062,11 +1065,13 @@ namespace {
10621065
// AccessStorage object for all projection paths.
10631066
class FindAccessStorageVisitor
10641067
: public FindAccessVisitorImpl<FindAccessStorageVisitor> {
1068+
using SuperTy = FindAccessVisitorImpl<FindAccessStorageVisitor>;
10651069

10661070
public:
10671071
struct Result {
10681072
Optional<AccessStorage> storage;
10691073
SILValue base;
1074+
Optional<AccessStorageCast> seenCast;
10701075
};
10711076

10721077
private:
@@ -1102,6 +1107,8 @@ class FindAccessStorageVisitor
11021107
// may be multiple global_addr bases for identical storage.
11031108
SILValue getBase() const { return result.base; }
11041109

1110+
Optional<AccessStorageCast> getCast() const { return result.seenCast; }
1111+
11051112
// MARK: AccessPhiVisitor::UseDefVisitor implementation.
11061113

11071114
// A valid result requires valid storage, but not a valid base.
@@ -1128,22 +1135,41 @@ class FindAccessStorageVisitor
11281135
invalidateResult();
11291136
return SILValue();
11301137
}
1138+
1139+
SILValue visitStorageCast(SingleValueInstruction *svi, Operand *sourceOper,
1140+
AccessStorageCast cast) {
1141+
result.seenCast = result.seenCast ? std::max(*result.seenCast, cast) : cast;
1142+
return SuperTy::visitStorageCast(svi, sourceOper, cast);
1143+
}
11311144
};
11321145

11331146
} // end anonymous namespace
11341147

1148+
RelativeAccessStorageWithBase
1149+
RelativeAccessStorageWithBase::compute(SILValue address) {
1150+
FindAccessStorageVisitor visitor(NestedAccessType::IgnoreAccessBegin);
1151+
visitor.findStorage(address);
1152+
return {
1153+
address, {visitor.getStorage(), visitor.getBase()}, visitor.getCast()};
1154+
}
1155+
1156+
RelativeAccessStorageWithBase
1157+
RelativeAccessStorageWithBase::computeInScope(SILValue address) {
1158+
FindAccessStorageVisitor visitor(NestedAccessType::StopAtAccessBegin);
1159+
visitor.findStorage(address);
1160+
return {
1161+
address, {visitor.getStorage(), visitor.getBase()}, visitor.getCast()};
1162+
}
1163+
11351164
AccessStorageWithBase
11361165
AccessStorageWithBase::compute(SILValue sourceAddress) {
1137-
FindAccessStorageVisitor visitor(NestedAccessType::IgnoreAccessBegin);
1138-
visitor.findStorage(sourceAddress);
1139-
return {visitor.getStorage(), visitor.getBase()};
1166+
return RelativeAccessStorageWithBase::compute(sourceAddress).storageWithBase;
11401167
}
11411168

11421169
AccessStorageWithBase
11431170
AccessStorageWithBase::computeInScope(SILValue sourceAddress) {
1144-
FindAccessStorageVisitor visitor(NestedAccessType::StopAtAccessBegin);
1145-
visitor.findStorage(sourceAddress);
1146-
return {visitor.getStorage(), visitor.getBase()};
1171+
return RelativeAccessStorageWithBase::computeInScope(sourceAddress)
1172+
.storageWithBase;
11471173
}
11481174

11491175
AccessStorage AccessStorage::compute(SILValue sourceAddress) {

lib/SILOptimizer/SemanticARC/LoadCopyToLoadBorrowOpt.cpp

+1-1
Original file line numberDiff line numberDiff line change
@@ -264,7 +264,7 @@ class StorageGuaranteesLoadVisitor
264264
}
265265

266266
void visitStorageCast(SingleValueInstruction *projectedAddr,
267-
Operand *parentAddr) {
267+
Operand *parentAddr, AccessStorageCast cast) {
268268
return next(parentAddr->get());
269269
}
270270

0 commit comments

Comments
 (0)