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
|
//===--- DifferentiabilityWitnessDevirtualizer.cpp ------------------------===//
//
// This source file is part of the Swift.org open source project
//
// Copyright (c) 2020 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
//
//===----------------------------------------------------------------------===//
//
// Devirtualizes `differentiability_witness_function` instructions into
// `function_ref` instructions for differentiability witness definitions.
//
//===----------------------------------------------------------------------===//
#include "swift/SIL/SILBuilder.h"
#include "swift/SIL/SILFunction.h"
#include "swift/SIL/SILInstruction.h"
#include "swift/SILOptimizer/PassManager/Transforms.h"
using namespace swift;
namespace {
class DifferentiabilityWitnessDevirtualizer : public SILFunctionTransform {
/// Returns true if any changes were made.
bool devirtualizeDifferentiabilityWitnessesInFunction(SILFunction &f);
/// The entry point to the transformation.
void run() override {
if (devirtualizeDifferentiabilityWitnessesInFunction(*getFunction()))
invalidateAnalysis(SILAnalysis::InvalidationKind::CallsAndInstructions);
}
};
} // end anonymous namespace
bool DifferentiabilityWitnessDevirtualizer::
devirtualizeDifferentiabilityWitnessesInFunction(SILFunction &f) {
bool changed = false;
llvm::SmallVector<DifferentiabilityWitnessFunctionInst *, 8> insts;
for (auto &bb : f) {
for (auto &inst : bb) {
auto *dfwi = dyn_cast<DifferentiabilityWitnessFunctionInst>(&inst);
if (!dfwi)
continue;
insts.push_back(dfwi);
}
}
for (auto *inst : insts) {
auto *witness = inst->getWitness();
if (witness->isDeclaration())
f.getModule().loadDifferentiabilityWitness(witness);
if (witness->isDeclaration())
continue;
changed = true;
SILBuilderWithScope builder(inst);
auto kind = inst->getWitnessKind().getAsDerivativeFunctionKind();
assert(kind.has_value());
auto *newInst = builder.createFunctionRefFor(inst->getLoc(),
witness->getDerivative(*kind));
inst->replaceAllUsesWith(newInst);
inst->getParent()->erase(inst);
}
return changed;
}
SILTransform *swift::createDifferentiabilityWitnessDevirtualizer() {
return new DifferentiabilityWitnessDevirtualizer();
}
|