mirror of
https://github.com/apple/swift.git
synced 2025-12-21 12:14:44 +01:00
MemoryBehaviorVisitor - Also, compute use points for destroy_value - Cleanup explicit checks for refcount instructions in RLE
110 lines
3.7 KiB
C++
110 lines
3.7 KiB
C++
//===--- EscapeAnalysisDumper.cpp - Dumps the escape analysis -------------===//
|
|
//
|
|
// 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
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#define DEBUG_TYPE "dump-ea"
|
|
#include "swift/SILOptimizer/Analysis/EscapeAnalysis.h"
|
|
#include "swift/SILOptimizer/PassManager/Passes.h"
|
|
#include "swift/SILOptimizer/PassManager/Transforms.h"
|
|
#include "llvm/Support/CommandLine.h"
|
|
|
|
using namespace swift;
|
|
|
|
// For manual debugging, dump graphs in DOT format.
|
|
static llvm::cl::opt<bool> EnableGraphWriter(
|
|
"escapes-enable-graphwriter", llvm::cl::init(false),
|
|
llvm::cl::desc("With -escapes-dump, also write .dot files."));
|
|
|
|
namespace {
|
|
|
|
static bool gatherValues(EscapeAnalysis *EA, SILFunction &Fn,
|
|
std::vector<SILValue> &Values) {
|
|
for (auto &BB : Fn) {
|
|
for (auto *Arg : BB.getArguments()) {
|
|
if (EA->isPointer(Arg))
|
|
Values.push_back(SILValue(Arg));
|
|
}
|
|
for (auto &II : BB) {
|
|
for (auto result : II.getResults()) {
|
|
if (EA->isPointer(result))
|
|
Values.push_back(result);
|
|
}
|
|
}
|
|
}
|
|
return Values.size() > 1;
|
|
}
|
|
|
|
/// Dumps the escape information of all functions in the module.
|
|
/// Only dumps if the compiler is built with assertions.
|
|
/// For details see EscapeAnalysis.
|
|
class EscapeAnalysisDumper : public SILModuleTransform {
|
|
|
|
void run() override {
|
|
LLVM_DEBUG(llvm::dbgs() << "** EscapeAnalysisDumper **\n");
|
|
|
|
#ifndef NDEBUG
|
|
auto *EA = PM->getAnalysis<EscapeAnalysis>();
|
|
|
|
llvm::outs() << "Escape information of module\n";
|
|
for (auto &F : *getModule()) {
|
|
if (!F.isExternalDeclaration()) {
|
|
auto *ConnectionGraph = EA->getConnectionGraph(&F);
|
|
ConnectionGraph->print(llvm::outs());
|
|
if (EnableGraphWriter)
|
|
ConnectionGraph->dumpCG();
|
|
|
|
// Gather up all Values in Fn.
|
|
std::vector<SILValue> Values;
|
|
if (!gatherValues(EA, F, Values))
|
|
continue;
|
|
|
|
// Emit the N^2 escape analysis evaluation of the values.
|
|
for (auto &bb : F) {
|
|
for (auto &ii : bb) {
|
|
if (auto fas = FullApplySite::isa(&ii)) {
|
|
for (unsigned i = 0, e = Values.size(); i != e; ++i) {
|
|
SILValue val = Values[i];
|
|
bool escape = EA->canEscapeTo(val, fas);
|
|
llvm::outs() << (escape ? "May" : "No") << "Escape: " << val
|
|
<< " to " << ii;
|
|
}
|
|
}
|
|
if (RefCountingInst *rci = dyn_cast<RefCountingInst>(&ii)) {
|
|
for (unsigned i = 0, e = Values.size(); i != e; ++i) {
|
|
SILValue val = Values[i];
|
|
bool escape = EA->canEscapeTo(val, rci);
|
|
llvm::outs() << (escape ? "May" : "No") << "Escape: " << val
|
|
<< " to " << ii;
|
|
}
|
|
}
|
|
if (DestroyValueInst *dvi = dyn_cast<DestroyValueInst>(&ii)) {
|
|
for (unsigned i = 0, e = Values.size(); i != e; ++i) {
|
|
SILValue val = Values[i];
|
|
bool escape = EA->canEscapeTo(val, dvi);
|
|
llvm::outs() << (escape ? "May" : "No") << "Escape: " << val
|
|
<< " to " << ii;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
#endif
|
|
}
|
|
|
|
};
|
|
|
|
} // end anonymous namespace
|
|
|
|
SILTransform *swift::createEscapeAnalysisDumper() {
|
|
return new EscapeAnalysisDumper();
|
|
}
|