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
|
//===--- DiagnosticDeadFunctionElimination.cpp ----------------------------===//
//
// 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
//
//===----------------------------------------------------------------------===//
///
/// Delete functions that early diagnostic specialization passes mark as being
/// able to be DCE-ed if there are no further uses. This prevents later
/// diagnostic passes from emitting diagnostics both on the original function
/// and the diagnostic function.
///
//===----------------------------------------------------------------------===//
#define DEBUG_TYPE "sil-diagnostic-dead-function-eliminator"
#include "swift/AST/SemanticAttrs.h"
#include "swift/SIL/SILBuilder.h"
#include "swift/SILOptimizer/PassManager/Passes.h"
#include "swift/SILOptimizer/PassManager/Transforms.h"
#include "llvm/Support/Debug.h"
using namespace swift;
//===----------------------------------------------------------------------===//
// MARK: Top Level Entrypoint
//===----------------------------------------------------------------------===//
namespace {
struct DiagnosticDeadFunctionEliminator : SILFunctionTransform {
void run() override {
auto *fn = getFunction();
// If an earlier pass asked us to eliminate the function body if it's
// unused, and the function is in fact unused, do that now.
if (!fn->hasSemanticsAttr(semantics::DELETE_IF_UNUSED) ||
fn->getRefCount() != 0 ||
isPossiblyUsedExternally(fn->getLinkage(),
fn->getModule().isWholeModule())) {
return;
}
LLVM_DEBUG(llvm::dbgs()
<< "===> Stubbifying unused function " << fn->getName()
<< "'s body that was marked for deletion\n");
// Remove all non-entry blocks.
auto entryBB = fn->begin();
auto nextBB = std::next(entryBB);
while (nextBB != fn->end()) {
auto thisBB = nextBB;
++nextBB;
thisBB->eraseFromParent();
}
// Rewrite the entry block to only contain an unreachable.
auto loc = entryBB->begin()->getLoc();
entryBB->eraseAllInstructions(fn->getModule());
{
SILBuilder b(&*entryBB);
b.createUnreachable(loc);
}
// If the function has shared linkage, reduce this version to private
// linkage, because we don't want the deleted-body form to win in any
// ODR shootouts.
if (fn->getLinkage() == SILLinkage::Shared) {
fn->setLinkage(SILLinkage::Private);
fn->setSerializedKind(IsNotSerialized);
}
invalidateAnalysis(SILAnalysis::InvalidationKind::FunctionBody);
}
};
} // namespace
SILTransform *swift::createDiagnosticDeadFunctionElimination() {
return new DiagnosticDeadFunctionEliminator();
}
|