1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111
|
//===--- IRGenPrepare.cpp -------------------------------------------------===//
//
// This source file is part of the Swift.org open source project
//
// Copyright (c) 2014 - 2018 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
//
//===----------------------------------------------------------------------===//
///
/// \file
///
/// Cleanup SIL to make it suitable for IRGen.
///
/// We perform the following canonicalizations:
///
/// 1. We remove calls to Builtin.poundAssert() and Builtin.staticReport(),
/// which are not needed post SIL.
///
//===----------------------------------------------------------------------===//
#define DEBUG_TYPE "sil-cleanup"
#include "swift/SIL/SILBuilder.h"
#include "swift/SIL/SILFunction.h"
#include "swift/SIL/SILInstruction.h"
#include "swift/SIL/SILModule.h"
#include "swift/SILOptimizer/PassManager/Passes.h"
#include "swift/SILOptimizer/PassManager/Transforms.h"
#include "swift/SILOptimizer/Utils/InstructionDeleter.h"
#include "swift/Strings.h"
using namespace swift;
static bool cleanFunction(SILFunction &fn) {
bool madeChange = false;
for (auto &bb : fn) {
for (auto i = bb.begin(), e = bb.end(); i != e;) {
// Make sure there is no iterator invalidation if the inspected
// instruction gets removed from the block.
SILInstruction *inst = &*i;
++i;
// Remove calls to Builtin.poundAssert() and Builtin.staticReport().
auto *bi = dyn_cast<BuiltinInst>(inst);
if (!bi) {
continue;
}
switch (bi->getBuiltinInfo().ID) {
case BuiltinValueKind::CondFailMessage: {
SILBuilderWithScope Builder(bi);
Builder.createCondFail(bi->getLoc(), bi->getOperand(0),
"unknown program error");
LLVM_FALLTHROUGH;
}
case BuiltinValueKind::PoundAssert:
case BuiltinValueKind::StaticReport: {
// The call to the builtin should get removed before we reach
// IRGen.
InstructionDeleter deleter;
deleter.forceDelete(bi);
// StaticReport only takes trivial operands, and therefore doesn't
// require fixing the lifetime of its operands.
deleter.cleanupDeadInstructions();
madeChange = true;
break;
}
default:
break;
}
}
}
return madeChange;
}
//===----------------------------------------------------------------------===//
// Top Level Entrypoint
//===----------------------------------------------------------------------===//
namespace {
class IRGenPrepare : public SILFunctionTransform {
void run() override {
SILFunction *F = getFunction();
if (getOptions().EmbeddedSwift) {
// In embedded swift all the code is generated in the top-level module.
// Even de-serialized functions must be code-gen'd.
SILLinkage linkage = F->getLinkage();
if (isAvailableExternally(linkage)) {
F->setLinkage(stripExternalFromLinkage(linkage));
}
}
bool shouldInvalidate = cleanFunction(*F);
if (shouldInvalidate)
invalidateAnalysis(SILAnalysis::InvalidationKind::Instructions);
}
};
} // end anonymous namespace
SILTransform *swift::createIRGenPrepare() {
return new IRGenPrepare();
}
|