Merge pull request #68389 from nate-chandler/rdar115033825

[AccessBase] FindAccessBaseVisitor visits nested accesses.
This commit is contained in:
nate-chandler
2023-09-08 07:04:09 -07:00
committed by GitHub
2 changed files with 112 additions and 1 deletions

View File

@@ -193,13 +193,21 @@ public:
// Override AccessUseDefChainVisitor to ignore access markers and find the // Override AccessUseDefChainVisitor to ignore access markers and find the
// outer access base. // outer access base.
SILValue visitNestedAccess(BeginAccessInst *access) { SILValue visitNestedAccessImpl(BeginAccessInst *access) {
if (nestedAccessTy == NestedAccessType::IgnoreAccessBegin) if (nestedAccessTy == NestedAccessType::IgnoreAccessBegin)
return access->getSource(); return access->getSource();
return SuperTy::visitNestedAccess(access); return SuperTy::visitNestedAccess(access);
} }
SILValue visitNestedAccess(BeginAccessInst *access) {
auto value = visitNestedAccessImpl(access);
if (value) {
reenterUseDef(value);
}
return SILValue();
}
SILValue visitPhi(SILPhiArgument *phiArg) { SILValue visitPhi(SILPhiArgument *phiArg) {
// Cycles involving phis are only handled within AccessPhiVisitor. // Cycles involving phis are only handled within AccessPhiVisitor.
// Path components are not allowed in phi cycles. // Path components are not allowed in phi cycles.
@@ -362,6 +370,18 @@ SILValue swift::getTypedAccessAddress(SILValue address) {
return accessAddress; return accessAddress;
} }
namespace swift::test {
static FunctionTest
GetTypedAccessAddress("get_typed_access_address",
[](auto &function, auto &arguments, auto &test) {
auto address = arguments.takeValue();
function.dump();
llvm::dbgs() << "Address: " << address;
auto access = getTypedAccessAddress(address);
llvm::dbgs() << "Access: " << access;
});
} // end namespace swift::test
// TODO: When the optimizer stops stripping begin_access markers and SILGen // TODO: When the optimizer stops stripping begin_access markers and SILGen
// protects all memory operations with at least an "unsafe" access scope, then // protects all memory operations with at least an "unsafe" access scope, then
// we should be able to assert that this returns a BeginAccessInst. // we should be able to assert that this returns a BeginAccessInst.
@@ -379,6 +399,18 @@ SILValue swift::getAccessBase(SILValue address) {
.findPossibleBaseAddress(address); .findPossibleBaseAddress(address);
} }
namespace swift::test {
static FunctionTest GetAccessBaseTest("get_access_base",
[](auto &function, auto &arguments,
auto &test) {
auto address = arguments.takeValue();
function.dump();
llvm::dbgs() << "Address: " << address;
auto base = getAccessBase(address);
llvm::dbgs() << "Base: " << base;
});
} // end namespace swift::test
static bool isLetForBase(SILValue base) { static bool isLetForBase(SILValue base) {
// Is this an address of a "let" class member? // Is this an address of a "let" class member?
if (auto *rea = dyn_cast<RefElementAddrInst>(base)) if (auto *rea = dyn_cast<RefElementAddrInst>(base))
@@ -1156,6 +1188,18 @@ AccessStorage AccessStorage::compute(SILValue sourceAddress) {
return AccessStorageWithBase::compute(sourceAddress).storage; return AccessStorageWithBase::compute(sourceAddress).storage;
} }
namespace swift::test {
static FunctionTest ComputeAccessStorage("compute_access_storage",
[](auto &function, auto &arguments,
auto &test) {
auto address = arguments.takeValue();
function.dump();
llvm::dbgs() << "Address: " << address;
auto accessStorage = AccessStorage::compute(address);
accessStorage.dump();
});
} // end namespace swift::test
AccessStorage AccessStorage::computeInScope(SILValue sourceAddress) { AccessStorage AccessStorage::computeInScope(SILValue sourceAddress) {
return AccessStorageWithBase::computeInScope(sourceAddress).storage; return AccessStorageWithBase::computeInScope(sourceAddress).storage;
} }

View File

@@ -0,0 +1,67 @@
// RUN: %target-sil-opt -test-runner %s -o /dev/null 2>&1 | %FileCheck %s
import Builtin
class Box<T> {
var value: T
}
class C {}
// CHECK-LABEL: begin running test {{.*}} on test_phi_nested_access: get_access_base with
// CHECK: sil @test_phi_nested_access : {{.*}} {
// CHECK: bb0([[BOX:%[^,]+]] :
// CHECK: [[VALUE_ADDR:%[^,]+]] = ref_element_addr [[BOX]]
// CHECK: [[VALUE_ACCESS:%[^,]+]] = begin_access [read] [unsafe] [[VALUE_ADDR]]
// CHECK: [[VALUE_PTR:%[^,]+]] = address_to_pointer [stack_protection] [[VALUE_ACCESS]]
// CHECK: end_access [[VALUE_ACCESS]]
// CHECK: br bb1([[VALUE_PTR]]
// CHECK: bb1([[VALUE_PHI:%[^,]+]] :
// CHECK: } // end sil function 'test_phi_nested_access'
// CHECK: Address: [[VALUE_PHI]] = argument of bb1
// CHECK: Base: [[VALUE_ADDR]] = ref_element_addr
// CHECK-LABEL: end running test {{.*}} on test_phi_nested_access: get_access_base with
// CHECK-LABEL: begin running test {{.*}} on test_phi_nested_access: compute_access_storage with
// CHECK: Address: [[VALUE_PHI]] = argument of bb1
// CHECK: Class [[BOX]] = argument of bb0
// CHECK: Field: var value: T Index: 0
// CHECK-LABEL: end running test {{.*}} on test_phi_nested_access: compute_access_storage with
// CHECK-LABEL: begin running test 3 of 3 on test_phi_nested_access: accesspath-base
// CHECK: Access path base: %6 = argument of bb1
// CHECK: Exact Use: end_access %2
// CHECK-LABEL: end running test 3 of 3 on test_phi_nested_access: accesspath-base
sil @test_phi_nested_access : $@convention(method) (@guaranteed Box<C>) -> () {
bb1(%box : $Box<C>):
%value_addr = ref_element_addr %box : $Box<C>, #Box.value
%value_access = begin_access [read] [unsafe] %value_addr : $*C
%value_ptr = address_to_pointer [stack_protection] %value_access : $*C to $Builtin.RawPointer
end_access %value_access : $*C
br exit(%value_ptr : $Builtin.RawPointer)
exit(%phi : $Builtin.RawPointer):
test_specification "get_access_base @argument"
test_specification "compute_access_storage @argument"
test_specification "accesspath-base @argument"
%retval = tuple ()
return %retval : $()
}
sil_global @global_c : $C
// CHECK-LABEL: begin running test {{.*}} on test_access_global
// CHECK: sil @test_access_global : {{.*}} {
// CHECK: [[GLOBAL:%[^,]+]] = global_addr @global_c
// CHECK: [[ACCESS:%[^,]+]] = begin_access [read] [dynamic] [[GLOBAL]]
// CHECK: } // end sil function 'test_access_global'
// CHECK: Address: [[ACCESS]] = begin_access [read] [dynamic] [[GLOBAL]]
// CHECK: Access: [[ACCESS]] = begin_access [read] [dynamic] [[GLOBAL]]
// CHECK-LABEL: end running test {{.*}} on test_access_global
sil @test_access_global : $@convention(thin) () -> () {
%15 = global_addr @global_c : $*C
test_specification "get_typed_access_address @instruction"
%16 = begin_access [read] [dynamic] %15 : $*C
end_access %16 : $*C
%retval = tuple ()
return %retval : $()
}