Skip to content

Commit 6b6e585

Browse files
committed
[MemAccessUtils] Visit product type tree of addr.
Added a function that visits the leaves of the type/projection tree of the specified address and calls its visitor with the path node to and type of each.
1 parent c2d80cd commit 6b6e585

File tree

2 files changed

+40
-0
lines changed

2 files changed

+40
-0
lines changed

include/swift/SIL/MemAccessUtils.h

+10
Original file line numberDiff line numberDiff line change
@@ -1209,6 +1209,16 @@ struct AccessPathWithBase {
12091209
void dump() const;
12101210
};
12111211

1212+
// Visits all the "product leaves" of the type tree of the specified value and
1213+
// invokes provided visitor, identifying the leaf by its path node and
1214+
// providing its type.
1215+
//
1216+
// The "product leaves" are the leaves obtained by only looking through type
1217+
// products (structs and tuples) and NOT type sums (enums).
1218+
void visitProductLeafAccessPathNodes(
1219+
SILValue address, TypeExpansionContext tec, SILModule &module,
1220+
std::function<void(AccessPath::PathNode, SILType)> visitor);
1221+
12121222
inline AccessPath AccessPath::compute(SILValue address) {
12131223
return AccessPathWithBase::compute(address).accessPath;
12141224
}

lib/SIL/Utils/MemAccessUtils.cpp

+30
Original file line numberDiff line numberDiff line change
@@ -1357,6 +1357,36 @@ AccessPathWithBase AccessPathWithBase::computeInScope(SILValue address) {
13571357
.findAccessPath(address);
13581358
}
13591359

1360+
void swift::visitProductLeafAccessPathNodes(
1361+
SILValue address, TypeExpansionContext tec, SILModule &module,
1362+
std::function<void(AccessPath::PathNode, SILType)> visitor) {
1363+
SmallVector<std::pair<SILType, IndexTrieNode *>, 32> worklist;
1364+
auto rootPath = AccessPath::compute(address);
1365+
auto *node = rootPath.getPathNode().node;
1366+
worklist.push_back({address->getType(), node});
1367+
while (!worklist.empty()) {
1368+
auto pair = worklist.pop_back_val();
1369+
auto silType = pair.first;
1370+
auto *node = pair.second;
1371+
if (auto tupleType = silType.getAs<TupleType>()) {
1372+
for (unsigned index : indices(tupleType->getElements())) {
1373+
auto *elementNode = node->getChild(index);
1374+
worklist.push_back({silType.getTupleElementType(index), elementNode});
1375+
}
1376+
} else if (auto *decl = silType.getStructOrBoundGenericStruct()) {
1377+
unsigned index = 0;
1378+
for (auto *field : decl->getStoredProperties()) {
1379+
auto *fieldNode = node->getChild(index);
1380+
worklist.push_back(
1381+
{silType.getFieldType(field, module, tec), fieldNode});
1382+
++index;
1383+
}
1384+
} else {
1385+
visitor(AccessPath::PathNode(node), silType);
1386+
}
1387+
}
1388+
}
1389+
13601390
void AccessPath::Index::print(raw_ostream &os) const {
13611391
if (isSubObjectProjection())
13621392
os << '#' << getSubObjectIndex();

0 commit comments

Comments
 (0)