Skip to content

Commit 7ea3363

Browse files
committed
[NFC] Port isDeinitBarrier to Swift.
Added new C++-to-Swift callback for isDeinitBarrier. And pass it CalleeAnalysis so it can depend on function effects. For now, the argument is ignored. And, all callers just pass nullptr. Promoted to API the mayAccessPointer component predicate of isDeinitBarrier which needs to remain in C++. That predicate will also depends on function effects. For that reason, it too is now passed a BasicCalleeAnalysis and is moved into SILOptimizer. Also, added more conservative versions of isDeinitBarrier and maySynchronize which will never consider side-effects.
1 parent 8535e80 commit 7ea3363

17 files changed

+148
-48
lines changed

SwiftCompilerSources/Sources/Optimizer/Analysis/CalleeAnalysis.swift

+33
Original file line numberDiff line numberDiff line change
@@ -16,6 +16,15 @@ import SIL
1616
public struct CalleeAnalysis {
1717
let bridged: BridgedCalleeAnalysis
1818

19+
static func register() {
20+
CalleeAnalysis_register(
21+
// isDeinitBarrierFn:
22+
{ (inst : BridgedInstruction, bca: BridgedCalleeAnalysis) -> Bool in
23+
return inst.instruction.isDeinitBarrier(bca.analysis)
24+
}
25+
)
26+
}
27+
1928
public func getCallees(callee: Value) -> FunctionArray? {
2029
let bridgedFuncs = CalleeAnalysis_getCallees(bridged, callee.bridged)
2130
if bridgedFuncs.incomplete != 0 {
@@ -45,6 +54,24 @@ public struct CalleeAnalysis {
4554
}
4655
}
4756

57+
extension Instruction {
58+
public final func maySynchronize(_ analysis: CalleeAnalysis) -> Bool {
59+
if let site = self as? FullApplySite {
60+
return true
61+
}
62+
return maySynchronizeNotConsideringSideEffects
63+
}
64+
65+
/// Whether lifetime ends of lexical values may safely be hoisted over this
66+
/// instruction.
67+
///
68+
/// Deinitialization barriers constrain variable lifetimes. Lexical
69+
/// end_borrow, destroy_value, and destroy_addr cannot be hoisted above them.
70+
public final func isDeinitBarrier(_ analysis: CalleeAnalysis) -> Bool {
71+
return mayAccessPointer || mayLoadWeakOrUnowned || maySynchronize(analysis)
72+
}
73+
}
74+
4875
public struct FunctionArray : RandomAccessCollection, FormattedLikeArray {
4976
fileprivate let bridged: BridgedCalleeList
5077

@@ -55,3 +82,9 @@ public struct FunctionArray : RandomAccessCollection, FormattedLikeArray {
5582
return BridgedFunctionArray_get(bridged, index).function
5683
}
5784
}
85+
// Bridging utilities
86+
87+
extension BridgedCalleeAnalysis {
88+
public var analysis: CalleeAnalysis { .init(bridged: self) }
89+
}
90+

SwiftCompilerSources/Sources/Optimizer/PassManager/PassRegistration.swift

+5
Original file line numberDiff line numberDiff line change
@@ -17,6 +17,7 @@ import Parse
1717
@_cdecl("initializeSwiftModules")
1818
public func initializeSwiftModules() {
1919
registerSILClasses()
20+
registerSwiftAnalyses()
2021
registerSwiftPasses()
2122
initializeSwiftParseModules()
2223
}
@@ -75,3 +76,7 @@ private func registerSwiftPasses() {
7576
registerPass(rangeDumper, { rangeDumper.run($0) })
7677
registerPass(runUnitTests, { runUnitTests.run($0) })
7778
}
79+
80+
private func registerSwiftAnalyses() {
81+
CalleeAnalysis.register()
82+
}

SwiftCompilerSources/Sources/SIL/Instruction.swift

+39
Original file line numberDiff line numberDiff line change
@@ -108,6 +108,41 @@ public class Instruction : ListNode, CustomStringConvertible, Hashable {
108108
return SILInstruction_hasUnspecifiedSideEffects(bridged)
109109
}
110110

111+
public final var mayAccessPointer: Bool {
112+
return swift_mayAccessPointer(bridged)
113+
}
114+
115+
/// Whether this instruction loads or copies a value whose storage does not
116+
/// increment the stored value's reference count.
117+
public final var mayLoadWeakOrUnowned: Bool {
118+
switch self {
119+
case is LoadWeakInst, is LoadUnownedInst, is StrongCopyUnownedValueInst, is StrongCopyUnmanagedValueInst:
120+
return true
121+
default:
122+
return false
123+
}
124+
}
125+
126+
/// Conservatively, whether this instruction could involve a synchronization
127+
/// point like a memory barrier, lock or syscall.
128+
public final var maySynchronizeNotConsideringSideEffects: Bool {
129+
switch self {
130+
case is FullApplySite, is EndApplyInst, is AbortApplyInst:
131+
return true
132+
default:
133+
return false
134+
}
135+
}
136+
137+
/// Conservatively, whether this instruction could be a barrier to hoisting
138+
/// destroys.
139+
///
140+
/// Does not consider function so effects, so every apply is treated as a
141+
/// barrier.
142+
public final var mayBeDeinitBarrierNotConsideringSideEffects: Bool {
143+
return mayAccessPointer || mayLoadWeakOrUnowned || maySynchronizeNotConsideringSideEffects
144+
}
145+
111146
public func visitReferencedFunctions(_ cl: (Function) -> ()) {
112147
}
113148

@@ -618,6 +653,10 @@ final public class ProjectBoxInst : SingleValueInstruction, UnaryInstruction {
618653

619654
final public class CopyValueInst : SingleValueInstruction, UnaryInstruction {}
620655

656+
final public class StrongCopyUnownedValueInst : SingleValueInstruction, UnaryInstruction {}
657+
658+
final public class StrongCopyUnmanagedValueInst : SingleValueInstruction, UnaryInstruction {}
659+
621660
final public class EndCOWMutationInst : SingleValueInstruction, UnaryInstruction {}
622661

623662
final public

SwiftCompilerSources/Sources/SIL/Registration.swift

+2
Original file line numberDiff line numberDiff line change
@@ -60,6 +60,8 @@ public func registerSILClasses() {
6060
register(ReleaseValueInst.self)
6161
register(DestroyValueInst.self)
6262
register(DestroyAddrInst.self)
63+
register(StrongCopyUnownedValueInst.self)
64+
register(StrongCopyUnmanagedValueInst.self)
6365
register(InjectEnumAddrInst.self)
6466
register(LoadInst.self)
6567
register(LoadWeakInst.self)

docs/SIL.rst

+4-4
Original file line numberDiff line numberDiff line change
@@ -2444,10 +2444,10 @@ required.
24442444
Deinit Barriers
24452445
```````````````
24462446

2447-
Deinit barriers (see swift::isDeinitBarrier) are instructions which would be
2448-
affected by the side effects of deinitializers. To maintain the order of
2449-
effects that is visible to the programmer, destroys of lexical values cannot be
2450-
reordered with respect to them. There are three kinds:
2447+
Deinit barriers (see Instruction.isDeinitBarrier(_:)) are instructions which
2448+
would be affected by the side effects of deinitializers. To maintain the order
2449+
of effects that is visible to the programmer, destroys of lexical values cannot
2450+
be reordered with respect to them. There are three kinds:
24512451

24522452
1. synchronization points (locks, memory barriers, syscalls, etc.)
24532453
2. loads of weak or unowned values

include/swift/SIL/MemAccessUtils.h

+3-5
Original file line numberDiff line numberDiff line change
@@ -232,11 +232,9 @@ inline bool accessKindMayConflict(SILAccessKind a, SILAccessKind b) {
232232
return !(a == SILAccessKind::Read && b == SILAccessKind::Read);
233233
}
234234

235-
/// Return true if \p instruction is a deinitialization barrier.
236-
///
237-
/// Deinitialization barriers constrain variable lifetimes. Lexical end_borrow,
238-
/// destroy_value, and destroy_addr cannot be hoisted above them.
239-
bool isDeinitBarrier(SILInstruction *instruction);
235+
/// Whether \p instruction accesses storage whose representation is unidentified
236+
/// such as by reading a pointer.
237+
bool mayAccessPointer(SILInstruction *instruction);
240238

241239
} // end namespace swift
242240

include/swift/SIL/SILBridging.h

+1
Original file line numberDiff line numberDiff line change
@@ -364,6 +364,7 @@ swift::SILDebugLocation SILInstruction_getLocation(BridgedInstruction inst);
364364
BridgedMemoryBehavior SILInstruction_getMemBehavior(BridgedInstruction inst);
365365
bool SILInstruction_mayRelease(BridgedInstruction inst);
366366
bool SILInstruction_hasUnspecifiedSideEffects(BridgedInstruction inst);
367+
bool swift_mayAccessPointer(BridgedInstruction inst);
367368

368369
BridgedInstruction MultiValueInstResult_getParent(BridgedMultiValueResult result);
369370
SwiftInt MultiValueInstResult_getIndex(BridgedMultiValueResult result);

include/swift/SIL/SILInstruction.h

-6
Original file line numberDiff line numberDiff line change
@@ -649,12 +649,6 @@ class SILInstruction : public llvm::ilist_node<SILInstruction> {
649649
/// Can this instruction abort the program in some manner?
650650
bool mayTrap() const;
651651

652-
/// Involves a synchronization point like a memory barrier, lock or syscall.
653-
///
654-
/// TODO: We need side-effect analysis and library annotation for this to be
655-
/// a reasonable API. For now, this is just a placeholder.
656-
bool maySynchronize() const;
657-
658652
/// Returns true if the given instruction is completely identical to RHS.
659653
bool isIdenticalTo(const SILInstruction *RHS) const {
660654
return isIdenticalTo(RHS,

include/swift/SILOptimizer/Analysis/BasicCalleeAnalysis.h

+3
Original file line numberDiff line numberDiff line change
@@ -230,6 +230,9 @@ class BasicCalleeAnalysis : public SILAnalysis {
230230
}
231231
};
232232

233+
bool isDeinitBarrier(SILInstruction *const instruction,
234+
BasicCalleeAnalysis *bca);
235+
233236
} // end namespace swift
234237

235238
#endif

include/swift/SILOptimizer/OptimizerBridging.h

+4
Original file line numberDiff line numberDiff line change
@@ -40,6 +40,10 @@ typedef struct {
4040
void * _Nullable bca;
4141
} BridgedCalleeAnalysis;
4242

43+
typedef bool (* _Nonnull InstructionIsDeinitBarrierFn)(BridgedInstruction, BridgedCalleeAnalysis bca);
44+
45+
void CalleeAnalysis_register(InstructionIsDeinitBarrierFn isDeinitBarrierFn);
46+
4347
typedef struct {
4448
void * _Nullable dea;
4549
} BridgedDeadEndBlocksAnalysis;

include/swift/SILOptimizer/Utils/InstOptUtils.h

+1
Original file line numberDiff line numberDiff line change
@@ -34,6 +34,7 @@ namespace swift {
3434

3535
class DominanceInfo;
3636
class DeadEndBlocks;
37+
class BasicCalleeAnalysis;
3738
template <class T> class NullablePtr;
3839

3940
/// Transform a Use Range (Operand*) into a User Range (SILInstruction *)

lib/SIL/IR/SILInstruction.cpp

-7
Original file line numberDiff line numberDiff line change
@@ -1358,13 +1358,6 @@ bool SILInstruction::mayTrap() const {
13581358
}
13591359
}
13601360

1361-
bool SILInstruction::maySynchronize() const {
1362-
// TODO: We need side-effect analysis and library annotation for this to be
1363-
// a reasonable API. For now, this is just a placeholder.
1364-
return isa<FullApplySite>(this) || isa<EndApplyInst>(this) ||
1365-
isa<AbortApplyInst>(this);
1366-
}
1367-
13681361
bool SILInstruction::isMetaInstruction() const {
13691362
// Every instruction that implements getVarInfo() should be in this list.
13701363
switch (getKind()) {

lib/SIL/Utils/MemAccessUtils.cpp

+5-22
Original file line numberDiff line numberDiff line change
@@ -16,6 +16,8 @@
1616
#include "swift/Basic/GraphNodeWorklist.h"
1717
#include "swift/SIL/Consumption.h"
1818
#include "swift/SIL/DynamicCasts.h"
19+
#include "swift/SIL/SILBridging.h"
20+
#include "swift/SIL/SILBridgingUtils.h"
1921
#include "swift/SIL/SILInstruction.h"
2022
#include "swift/SIL/SILModule.h"
2123
#include "swift/SIL/SILUndef.h"
@@ -400,12 +402,7 @@ bool swift::isLetAddress(SILValue address) {
400402
// MARK: Deinitialization barriers.
401403
//===----------------------------------------------------------------------===//
402404

403-
static bool isBarrierApply(FullApplySite) {
404-
// TODO: check side effect analysis
405-
return true;
406-
}
407-
408-
static bool mayAccessPointer(SILInstruction *instruction) {
405+
bool swift::mayAccessPointer(SILInstruction *instruction) {
409406
if (!instruction->mayReadOrWriteMemory())
410407
return false;
411408
bool isUnidentified = false;
@@ -417,22 +414,8 @@ static bool mayAccessPointer(SILInstruction *instruction) {
417414
return isUnidentified;
418415
}
419416

420-
static bool mayLoadWeakOrUnowned(SILInstruction *instruction) {
421-
// TODO: It is possible to do better here by looking at the address that is
422-
// being loaded.
423-
return isa<LoadWeakInst>(instruction) || isa<LoadUnownedInst>(instruction)
424-
|| isa<StrongCopyUnownedValueInst>(instruction)
425-
|| isa<StrongCopyUnmanagedValueInst>(instruction);
426-
}
427-
428-
bool swift::isDeinitBarrier(SILInstruction *instruction) {
429-
if (instruction->maySynchronize()) {
430-
if (auto apply = FullApplySite::isa(instruction)) {
431-
return isBarrierApply(apply);
432-
}
433-
return true;
434-
}
435-
return mayLoadWeakOrUnowned(instruction) || mayAccessPointer(instruction);
417+
bool swift_mayAccessPointer(BridgedInstruction inst) {
418+
return mayAccessPointer(castToInst(inst));
436419
}
437420

438421
//===----------------------------------------------------------------------===//

lib/SILOptimizer/Analysis/BasicCalleeAnalysis.cpp

+41-1
Original file line numberDiff line numberDiff line change
@@ -15,8 +15,10 @@
1515
#include "swift/AST/Decl.h"
1616
#include "swift/AST/ProtocolConformance.h"
1717
#include "swift/Basic/Statistic.h"
18-
#include "swift/SIL/SILModule.h"
18+
#include "swift/SIL/MemAccessUtils.h"
19+
#include "swift/SIL/SILBridging.h"
1920
#include "swift/SIL/SILBridgingUtils.h"
21+
#include "swift/SIL/SILModule.h"
2022
#include "swift/SILOptimizer/OptimizerBridging.h"
2123
#include "swift/SILOptimizer/Utils/InstOptUtils.h"
2224
#include "llvm/Support/Compiler.h"
@@ -363,3 +365,41 @@ BridgedFunction BridgedFunctionArray_get(BridgedCalleeList callees,
363365
assert(index >= 0 && iter < cl.end());
364366
return {*iter};
365367
}
368+
369+
static InstructionIsDeinitBarrierFn instructionIsDeinitBarrierFunction;
370+
371+
void CalleeAnalysis_register(
372+
InstructionIsDeinitBarrierFn instructionIsDeinitBarrierFn) {
373+
instructionIsDeinitBarrierFunction = instructionIsDeinitBarrierFn;
374+
}
375+
376+
/// Implementation of mayBeDeinitBarrierNotConsideringSideEffects for use only
377+
/// during bootstrapping or in compilers that lack Swift sources.
378+
///
379+
/// TODO: Remove.
380+
static bool
381+
isDeinitBarrierWithoutEffectsCpp(SILInstruction *const instruction) {
382+
auto mayLoadWeakOrUnowned = [](SILInstruction *const instruction) {
383+
return isa<LoadWeakInst>(instruction) ||
384+
isa<LoadUnownedInst>(instruction) ||
385+
isa<StrongCopyUnownedValueInst>(instruction) ||
386+
isa<StrongCopyUnmanagedValueInst>(instruction);
387+
};
388+
auto maySynchronize = [](SILInstruction *const instruction) {
389+
return isa<FullApplySite>(instruction) || isa<EndApplyInst>(instruction) ||
390+
isa<AbortApplyInst>(instruction);
391+
};
392+
return mayAccessPointer(instruction) || mayLoadWeakOrUnowned(instruction) ||
393+
maySynchronize(instruction);
394+
}
395+
396+
bool swift::isDeinitBarrier(SILInstruction *const instruction,
397+
BasicCalleeAnalysis *bca) {
398+
if (!instructionIsDeinitBarrierFunction) {
399+
return isDeinitBarrierWithoutEffectsCpp(instruction);
400+
}
401+
BridgedInstruction inst = {
402+
cast<SILNode>(const_cast<SILInstruction *>(instruction))};
403+
BridgedCalleeAnalysis analysis = {bca};
404+
return instructionIsDeinitBarrierFunction(inst, analysis);
405+
}

lib/SILOptimizer/Transforms/SSADestroyHoisting.cpp

+3-1
Original file line numberDiff line numberDiff line change
@@ -97,9 +97,11 @@
9797
#include "swift/SIL/SILBasicBlock.h"
9898
#include "swift/SIL/SILBuilder.h"
9999
#include "swift/SIL/SILInstruction.h"
100+
#include "swift/SILOptimizer/Analysis/BasicCalleeAnalysis.h"
100101
#include "swift/SILOptimizer/Analysis/Reachability.h"
101102
#include "swift/SILOptimizer/Analysis/VisitBarrierAccessScopes.h"
102103
#include "swift/SILOptimizer/PassManager/Transforms.h"
104+
#include "swift/SILOptimizer/Utils/InstOptUtils.h"
103105
#include "swift/SILOptimizer/Utils/InstructionDeleter.h"
104106

105107
using namespace swift;
@@ -339,7 +341,7 @@ DeinitBarriers::classifyInstruction(SILInstruction *inst) const {
339341
if (knownUses.storageUsers.contains(inst)) {
340342
return Classification::Barrier;
341343
}
342-
if (!ignoreDeinitBarriers && isDeinitBarrier(inst)) {
344+
if (!ignoreDeinitBarriers && isDeinitBarrier(inst, nullptr)) {
343345
return Classification::Barrier;
344346
}
345347
if (auto *eai = dyn_cast<EndAccessInst>(inst)) {

lib/SILOptimizer/Utils/LexicalDestroyHoisting.cpp

+2-1
Original file line numberDiff line numberDiff line change
@@ -19,6 +19,7 @@
1919
#include "swift/SIL/SILBasicBlock.h"
2020
#include "swift/SIL/SILInstruction.h"
2121
#include "swift/SIL/SILValue.h"
22+
#include "swift/SILOptimizer/Analysis/BasicCalleeAnalysis.h"
2223
#include "swift/SILOptimizer/Analysis/Reachability.h"
2324
#include "swift/SILOptimizer/Analysis/VisitBarrierAccessScopes.h"
2425
#include "swift/SILOptimizer/Utils/CanonicalizeBorrowScope.h"
@@ -194,7 +195,7 @@ Dataflow::classifyInstruction(SILInstruction *instruction) {
194195
? Classification::Barrier
195196
: Classification::Other;
196197
}
197-
if (isDeinitBarrier(instruction)) {
198+
if (isDeinitBarrier(instruction, nullptr)) {
198199
return Classification::Barrier;
199200
}
200201
return Classification::Other;

lib/SILOptimizer/Utils/ShrinkBorrowScope.cpp

+2-1
Original file line numberDiff line numberDiff line change
@@ -20,6 +20,7 @@
2020
#include "swift/SIL/OwnershipUtils.h"
2121
#include "swift/SIL/SILBasicBlock.h"
2222
#include "swift/SIL/SILInstruction.h"
23+
#include "swift/SILOptimizer/Analysis/BasicCalleeAnalysis.h"
2324
#include "swift/SILOptimizer/Analysis/Reachability.h"
2425
#include "swift/SILOptimizer/Analysis/VisitBarrierAccessScopes.h"
2526
#include "swift/SILOptimizer/Utils/CanonicalizeBorrowScope.h"
@@ -244,7 +245,7 @@ Dataflow::classifyInstruction(SILInstruction *instruction) {
244245
? Classification::Barrier
245246
: Classification::Other;
246247
}
247-
if (isDeinitBarrier(instruction)) {
248+
if (isDeinitBarrier(instruction, nullptr)) {
248249
return Classification::Barrier;
249250
}
250251
return Classification::Other;

0 commit comments

Comments
 (0)