1 //===- ReduceFunctions.cpp - Specialized Delta Pass -----------------------===//
2 //
3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4 // See https://llvm.org/LICENSE.txt for license information.
5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6 //
7 //===----------------------------------------------------------------------===//
8 //
9 // This file implements a function which calls the Generic Delta pass in order
10 // to reduce functions (and any instruction that calls it) in the provided
11 // Module.
12 //
13 //===----------------------------------------------------------------------===//
14
15 #include "ReduceFunctions.h"
16 #include "Delta.h"
17 #include "llvm/ADT/STLExtras.h"
18 #include "llvm/IR/Instructions.h"
19 #include <iterator>
20 #include <vector>
21
22 using namespace llvm;
23
24 /// Removes all the Defined Functions
25 /// that aren't inside any of the desired Chunks.
extractFunctionsFromModule(const std::vector<Chunk> & ChunksToKeep,Module * Program)26 static void extractFunctionsFromModule(const std::vector<Chunk> &ChunksToKeep,
27 Module *Program) {
28 Oracle O(ChunksToKeep);
29
30 // Record all out-of-chunk functions.
31 std::vector<std::reference_wrapper<Function>> FuncsToRemove;
32 copy_if(Program->functions(), std::back_inserter(FuncsToRemove),
33 [&O](Function &F) {
34 // Intrinsics don't have function bodies that are useful to
35 // reduce. Additionally, intrinsics may have additional operand
36 // constraints.
37 return !F.isIntrinsic() && !O.shouldKeep();
38 });
39
40 // Then, drop body of each of them. We want to batch this and do nothing else
41 // here so that minimal number of remaining exteranal uses will remain.
42 for (Function &F : FuncsToRemove)
43 F.dropAllReferences();
44
45 // And finally, we can actually delete them.
46 for (Function &F : FuncsToRemove) {
47 // Replace all *still* remaining uses with undef.
48 F.replaceAllUsesWith(UndefValue::get(F.getType()));
49 // And finally, fully drop it.
50 F.eraseFromParent();
51 }
52 }
53
54 /// Counts the amount of non-declaration functions and prints their
55 /// respective name & index
countFunctions(Module * Program)56 static int countFunctions(Module *Program) {
57 // TODO: Silence index with --quiet flag
58 errs() << "----------------------------\n";
59 errs() << "Function Index Reference:\n";
60 int FunctionCount = 0;
61 for (auto &F : *Program) {
62 if (F.isIntrinsic())
63 continue;
64
65 errs() << '\t' << ++FunctionCount << ": " << F.getName() << '\n';
66 }
67
68 errs() << "----------------------------\n";
69 return FunctionCount;
70 }
71
reduceFunctionsDeltaPass(TestRunner & Test)72 void llvm::reduceFunctionsDeltaPass(TestRunner &Test) {
73 errs() << "*** Reducing Functions...\n";
74 int Functions = countFunctions(Test.getProgram());
75 runDeltaPass(Test, Functions, extractFunctionsFromModule);
76 errs() << "----------------------------\n";
77 }
78