mirror of
https://github.com/apple/swift.git
synced 2025-12-21 12:14:44 +01:00
The main changes are: *) Rewrite everything in swift. So far, parts of memory-behavior analysis were already implemented in swift. Now everything is done in swift and lives in `AliasAnalysis.swift`. This is a big code simplification. *) Support many more instructions in the memory-behavior analysis - especially OSSA instructions, like `begin_borrow`, `end_borrow`, `store_borrow`, `load_borrow`. The computation of end_borrow effects is now much more precise. Also, partial_apply is now handled more precisely. *) Simplify and reduce type-based alias analysis (TBAA). The complexity of the old TBAA comes from old days where the language and SIL didn't have strict aliasing and exclusivity rules (e.g. for inout arguments). Now TBAA is only needed for code using unsafe pointers. The new TBAA handles this - and not more. Note that TBAA for classes is already done in `AccessBase.isDistinct`. *) Handle aliasing in `begin_access [modify]` scopes. We already supported truly immutable scopes like `begin_access [read]` or `ref_element_addr [immutable]`. For `begin_access [modify]` we know that there are no other reads or writes to the access-address within the scope. *) Don't cache memory-behavior results. It turned out that the hit-miss rate was pretty bad (~ 1:7). The overhead of the cache lookup took as long as recomputing the memory behavior.
108 lines
3.8 KiB
C++
108 lines
3.8 KiB
C++
//===--- BasicCalleeAnalysis.cpp - Determine callees per call site --------===//
|
|
//
|
|
// This source file is part of the Swift.org open source project
|
|
//
|
|
// Copyright (c) 2014 - 2017 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
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#include "swift/SILOptimizer/Analysis/BasicCalleeAnalysis.h"
|
|
|
|
#include "swift/AST/Decl.h"
|
|
#include "swift/AST/ProtocolConformance.h"
|
|
#include "swift/Basic/Statistic.h"
|
|
#include "swift/SIL/MemAccessUtils.h"
|
|
#include "swift/SIL/SILBridging.h"
|
|
#include "swift/SIL/SILModule.h"
|
|
#include "swift/SIL/Test.h"
|
|
#include "swift/SILOptimizer/OptimizerBridging.h"
|
|
#include "swift/SILOptimizer/PassManager/Transforms.h"
|
|
#include "swift/SILOptimizer/Utils/InstOptUtils.h"
|
|
#include "llvm/Support/Compiler.h"
|
|
|
|
#include <algorithm>
|
|
|
|
#define DEBUG_TYPE "BasicCalleeAnalysis"
|
|
|
|
using namespace swift;
|
|
|
|
// TODO: can't be inlined to work around https://github.com/apple/swift/issues/64502
|
|
BasicCalleeAnalysis::~BasicCalleeAnalysis() {
|
|
}
|
|
|
|
void BasicCalleeAnalysis::dump() const {
|
|
print(llvm::errs());
|
|
}
|
|
|
|
void BasicCalleeAnalysis::print(llvm::raw_ostream &os) const {
|
|
if (!Cache) {
|
|
os << "<no cache>\n";
|
|
}
|
|
llvm::DenseSet<SILDeclRef> printed;
|
|
for (auto &VTable : M.getVTables()) {
|
|
for (const SILVTable::Entry &entry : VTable->getEntries()) {
|
|
if (printed.insert(entry.getMethod()).second) {
|
|
os << "callees for " << entry.getMethod() << ":\n";
|
|
Cache->getCalleeList(entry.getMethod()).print(os);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
// Swift Bridging
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
static BridgedCalleeAnalysis::IsDeinitBarrierFn instructionIsDeinitBarrierFunction;
|
|
static BridgedCalleeAnalysis::GetMemBehvaiorFn getMemBehvaiorFunction = nullptr;
|
|
|
|
void BridgedCalleeAnalysis::registerAnalysis(IsDeinitBarrierFn instructionIsDeinitBarrierFn,
|
|
GetMemBehvaiorFn getMemBehvaiorFn) {
|
|
instructionIsDeinitBarrierFunction = instructionIsDeinitBarrierFn;
|
|
getMemBehvaiorFunction = getMemBehvaiorFn;
|
|
}
|
|
|
|
MemoryBehavior BasicCalleeAnalysis::
|
|
getMemoryBehavior(FullApplySite as, bool observeRetains) {
|
|
if (getMemBehvaiorFunction) {
|
|
auto b = getMemBehvaiorFunction({as.getInstruction()->asSILNode()},
|
|
observeRetains,
|
|
{this});
|
|
return (MemoryBehavior)b;
|
|
}
|
|
return MemoryBehavior::MayHaveSideEffects;
|
|
}
|
|
|
|
bool swift::isDeinitBarrier(SILInstruction *const instruction,
|
|
BasicCalleeAnalysis *bca) {
|
|
if (!instructionIsDeinitBarrierFunction) {
|
|
return mayBeDeinitBarrierNotConsideringSideEffects(instruction);
|
|
}
|
|
BridgedInstruction inst = {
|
|
cast<SILNode>(const_cast<SILInstruction *>(instruction))};
|
|
BridgedCalleeAnalysis analysis = {bca};
|
|
return instructionIsDeinitBarrierFunction(inst, analysis);
|
|
}
|
|
|
|
namespace swift::test {
|
|
// Arguments:
|
|
// - instruction
|
|
// Dumps:
|
|
// - instruction
|
|
// - whether it's a deinit barrier
|
|
static FunctionTest IsDeinitBarrierTest("is_deinit_barrier", [](auto &function,
|
|
auto &arguments,
|
|
auto &test) {
|
|
auto *instruction = arguments.takeInstruction();
|
|
auto *analysis = test.template getAnalysis<BasicCalleeAnalysis>();
|
|
auto isBarrier = isDeinitBarrier(instruction, analysis);
|
|
instruction->print(llvm::outs());
|
|
auto *boolString = isBarrier ? "true" : "false";
|
|
llvm::outs() << boolString << "\n";
|
|
});
|
|
} // namespace swift::test
|