mirror of
https://github.com/apple/swift.git
synced 2025-12-21 12:14:44 +01:00
Swift Optimizer: add the SSAUpdater utility
This commit is contained in:
@@ -145,17 +145,15 @@ extension MutatingContext {
|
|||||||
SubstitutionMap(_bridged.getContextSubstitutionMap(type.bridged))
|
SubstitutionMap(_bridged.getContextSubstitutionMap(type.bridged))
|
||||||
}
|
}
|
||||||
|
|
||||||
// Private utilities
|
func notifyInstructionsChanged() {
|
||||||
|
|
||||||
fileprivate func notifyInstructionsChanged() {
|
|
||||||
_bridged.asNotificationHandler().notifyChanges(.instructionsChanged)
|
_bridged.asNotificationHandler().notifyChanges(.instructionsChanged)
|
||||||
}
|
}
|
||||||
|
|
||||||
fileprivate func notifyCallsChanged() {
|
func notifyCallsChanged() {
|
||||||
_bridged.asNotificationHandler().notifyChanges(.callsChanged)
|
_bridged.asNotificationHandler().notifyChanges(.callsChanged)
|
||||||
}
|
}
|
||||||
|
|
||||||
fileprivate func notifyBranchesChanged() {
|
func notifyBranchesChanged() {
|
||||||
_bridged.asNotificationHandler().notifyChanges(.branchesChanged)
|
_bridged.asNotificationHandler().notifyChanges(.branchesChanged)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -11,5 +11,6 @@ swift_compiler_sources(Optimizer
|
|||||||
OptUtils.swift
|
OptUtils.swift
|
||||||
WalkUtils.swift
|
WalkUtils.swift
|
||||||
AccessUtils.swift
|
AccessUtils.swift
|
||||||
|
SSAUpdater.swift
|
||||||
StaticInitCloner.swift
|
StaticInitCloner.swift
|
||||||
)
|
)
|
||||||
|
|||||||
@@ -0,0 +1,51 @@
|
|||||||
|
//===--- SSAUpdater.swift -------------------------------------------------===//
|
||||||
|
//
|
||||||
|
// This source file is part of the Swift.org open source project
|
||||||
|
//
|
||||||
|
// Copyright (c) 2014 - 2023 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
|
||||||
|
//
|
||||||
|
//===----------------------------------------------------------------------===//
|
||||||
|
|
||||||
|
import SIL
|
||||||
|
import OptimizerBridging
|
||||||
|
|
||||||
|
/// Utility for updating SSA for a set of SIL instructions defined in multiple blocks.
|
||||||
|
struct SSAUpdater<Context: MutatingContext> {
|
||||||
|
let context: Context
|
||||||
|
|
||||||
|
init(type: Type, ownership: Ownership, _ context: Context) {
|
||||||
|
self.context = context
|
||||||
|
context._bridged.SSAUpdater_initialize(type.bridged, ownership._bridged)
|
||||||
|
}
|
||||||
|
|
||||||
|
mutating func addAvailableValue(_ value: Value, in block: BasicBlock) {
|
||||||
|
context._bridged.SSAUpdater_addAvailableValue(block.bridged, value.bridged)
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Construct SSA for a value that is live at the *end* of a basic block.
|
||||||
|
mutating func getValue(atEndOf block: BasicBlock) -> Value {
|
||||||
|
context.notifyInstructionsChanged()
|
||||||
|
return context._bridged.SSAUpdater_getValueAtEndOfBlock(block.bridged).value
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Construct SSA for a value that is live in the *middle* of a block.
|
||||||
|
/// This handles the case where the use is before a definition of the value in the same block.
|
||||||
|
///
|
||||||
|
/// bb1:
|
||||||
|
/// %1 = def
|
||||||
|
/// br bb2
|
||||||
|
/// bb2:
|
||||||
|
/// = use(?)
|
||||||
|
/// %2 = def
|
||||||
|
/// cond_br bb2, bb3
|
||||||
|
///
|
||||||
|
/// In this case we need to insert a phi argument in bb2, merging %1 and %2.
|
||||||
|
mutating func getValue(inMiddleOf block: BasicBlock) -> Value {
|
||||||
|
context.notifyInstructionsChanged()
|
||||||
|
return context._bridged.SSAUpdater_getValueInMiddleOfBlock(block.bridged).value
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -452,6 +452,26 @@ struct BridgedPassContext {
|
|||||||
invocation->getTransform());
|
invocation->getTransform());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// SSAUpdater
|
||||||
|
|
||||||
|
void SSAUpdater_initialize(swift::SILType type, BridgedValue::Ownership ownership) const {
|
||||||
|
invocation->initializeSSAUpdater(type, castToOwnership(ownership));
|
||||||
|
}
|
||||||
|
|
||||||
|
void SSAUpdater_addAvailableValue(BridgedBasicBlock block, BridgedValue value) const {
|
||||||
|
invocation->getSSAUpdater()->addAvailableValue(block.getBlock(), value.getSILValue());
|
||||||
|
}
|
||||||
|
|
||||||
|
SWIFT_IMPORT_UNSAFE
|
||||||
|
BridgedValue SSAUpdater_getValueAtEndOfBlock(BridgedBasicBlock block) const {
|
||||||
|
return {invocation->getSSAUpdater()->getValueAtEndOfBlock(block.getBlock())};
|
||||||
|
}
|
||||||
|
|
||||||
|
SWIFT_IMPORT_UNSAFE
|
||||||
|
BridgedValue SSAUpdater_getValueInMiddleOfBlock(BridgedBasicBlock block) const {
|
||||||
|
return {invocation->getSSAUpdater()->getValueInMiddleOfBlock(block.getBlock())};
|
||||||
|
}
|
||||||
|
|
||||||
// Options
|
// Options
|
||||||
|
|
||||||
bool enableStackProtection() const {
|
bool enableStackProtection() const {
|
||||||
|
|||||||
@@ -17,6 +17,7 @@
|
|||||||
#include "swift/SILOptimizer/Analysis/Analysis.h"
|
#include "swift/SILOptimizer/Analysis/Analysis.h"
|
||||||
#include "swift/SILOptimizer/PassManager/PassPipeline.h"
|
#include "swift/SILOptimizer/PassManager/PassPipeline.h"
|
||||||
#include "swift/SILOptimizer/PassManager/Passes.h"
|
#include "swift/SILOptimizer/PassManager/Passes.h"
|
||||||
|
#include "swift/SILOptimizer/Utils/SILSSAUpdater.h"
|
||||||
#include "llvm/ADT/DenseMap.h"
|
#include "llvm/ADT/DenseMap.h"
|
||||||
#include "llvm/ADT/SmallVector.h"
|
#include "llvm/ADT/SmallVector.h"
|
||||||
#include "llvm/Support/Casting.h"
|
#include "llvm/Support/Casting.h"
|
||||||
@@ -68,6 +69,8 @@ class SwiftPassInvocation {
|
|||||||
/// All slabs, allocated by the pass.
|
/// All slabs, allocated by the pass.
|
||||||
SILModule::SlabList allocatedSlabs;
|
SILModule::SlabList allocatedSlabs;
|
||||||
|
|
||||||
|
SILSSAUpdater *ssaUpdater = nullptr;
|
||||||
|
|
||||||
static constexpr int BlockSetCapacity = 8;
|
static constexpr int BlockSetCapacity = 8;
|
||||||
char blockSetStorage[sizeof(BasicBlockSet) * BlockSetCapacity];
|
char blockSetStorage[sizeof(BasicBlockSet) * BlockSetCapacity];
|
||||||
bool aliveBlockSets[BlockSetCapacity];
|
bool aliveBlockSets[BlockSetCapacity];
|
||||||
@@ -82,7 +85,7 @@ class SwiftPassInvocation {
|
|||||||
|
|
||||||
bool needFixStackNesting = false;
|
bool needFixStackNesting = false;
|
||||||
|
|
||||||
void endPassRunChecks();
|
void endPass();
|
||||||
|
|
||||||
public:
|
public:
|
||||||
SwiftPassInvocation(SILPassManager *passManager, SILFunction *function,
|
SwiftPassInvocation(SILPassManager *passManager, SILFunction *function,
|
||||||
@@ -143,6 +146,17 @@ public:
|
|||||||
|
|
||||||
void setNeedFixStackNesting(bool newValue) { needFixStackNesting = newValue; }
|
void setNeedFixStackNesting(bool newValue) { needFixStackNesting = newValue; }
|
||||||
bool getNeedFixStackNesting() const { return needFixStackNesting; }
|
bool getNeedFixStackNesting() const { return needFixStackNesting; }
|
||||||
|
|
||||||
|
void initializeSSAUpdater(SILType type, ValueOwnershipKind ownership) {
|
||||||
|
if (!ssaUpdater)
|
||||||
|
ssaUpdater = new SILSSAUpdater;
|
||||||
|
ssaUpdater->initialize(type, ownership);
|
||||||
|
}
|
||||||
|
|
||||||
|
SILSSAUpdater *getSSAUpdater() const {
|
||||||
|
assert(ssaUpdater && "SSAUpdater not initialized");
|
||||||
|
return ssaUpdater;
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
/// The SIL pass manager.
|
/// The SIL pass manager.
|
||||||
|
|||||||
@@ -1360,7 +1360,7 @@ void SwiftPassInvocation::startInstructionPassRun(SILInstruction *inst) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void SwiftPassInvocation::finishedModulePassRun() {
|
void SwiftPassInvocation::finishedModulePassRun() {
|
||||||
endPassRunChecks();
|
endPass();
|
||||||
assert(!function && transform && "not running a pass");
|
assert(!function && transform && "not running a pass");
|
||||||
assert(changeNotifications == SILAnalysis::InvalidationKind::Nothing
|
assert(changeNotifications == SILAnalysis::InvalidationKind::Nothing
|
||||||
&& "unhandled change notifications at end of module pass");
|
&& "unhandled change notifications at end of module pass");
|
||||||
@@ -1368,22 +1368,26 @@ void SwiftPassInvocation::finishedModulePassRun() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void SwiftPassInvocation::finishedFunctionPassRun() {
|
void SwiftPassInvocation::finishedFunctionPassRun() {
|
||||||
endPassRunChecks();
|
endPass();
|
||||||
endTransformFunction();
|
endTransformFunction();
|
||||||
assert(allocatedSlabs.empty() && "StackList is leaking slabs");
|
assert(allocatedSlabs.empty() && "StackList is leaking slabs");
|
||||||
transform = nullptr;
|
transform = nullptr;
|
||||||
}
|
}
|
||||||
|
|
||||||
void SwiftPassInvocation::finishedInstructionPassRun() {
|
void SwiftPassInvocation::finishedInstructionPassRun() {
|
||||||
endPassRunChecks();
|
endPass();
|
||||||
}
|
}
|
||||||
|
|
||||||
void SwiftPassInvocation::endPassRunChecks() {
|
void SwiftPassInvocation::endPass() {
|
||||||
assert(allocatedSlabs.empty() && "StackList is leaking slabs");
|
assert(allocatedSlabs.empty() && "StackList is leaking slabs");
|
||||||
assert(numBlockSetsAllocated == 0 && "Not all BasicBlockSets deallocated");
|
assert(numBlockSetsAllocated == 0 && "Not all BasicBlockSets deallocated");
|
||||||
assert(numNodeSetsAllocated == 0 && "Not all NodeSets deallocated");
|
assert(numNodeSetsAllocated == 0 && "Not all NodeSets deallocated");
|
||||||
assert(numClonersAllocated == 0 && "Not all cloners deallocated");
|
assert(numClonersAllocated == 0 && "Not all cloners deallocated");
|
||||||
assert(!needFixStackNesting && "Stack nesting not fixed");
|
assert(!needFixStackNesting && "Stack nesting not fixed");
|
||||||
|
if (ssaUpdater) {
|
||||||
|
delete ssaUpdater;
|
||||||
|
ssaUpdater = nullptr;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void SwiftPassInvocation::beginTransformFunction(SILFunction *function) {
|
void SwiftPassInvocation::beginTransformFunction(SILFunction *function) {
|
||||||
|
|||||||
Reference in New Issue
Block a user