1 //===- DeadArgumentElimination.h - Eliminate Dead Args ----------*- C++ -*-===//
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
7 //===----------------------------------------------------------------------===//
9 // This pass deletes dead arguments from internal functions. Dead argument
10 // elimination removes arguments which are directly dead, as well as arguments
11 // only passed into function calls as dead arguments of other functions. This
12 // pass also deletes dead return values in a similar way.
14 // This pass is often useful as a cleanup pass to run after aggressive
15 // interprocedural passes, which add possibly-dead arguments or return values.
17 //===----------------------------------------------------------------------===//
19 #ifndef LLVM_TRANSFORMS_IPO_DEADARGUMENTELIMINATION_H
20 #define LLVM_TRANSFORMS_IPO_DEADARGUMENTELIMINATION_H
22 #include "llvm/ADT/SmallVector.h"
23 #include "llvm/ADT/Twine.h"
24 #include "llvm/IR/Function.h"
25 #include "llvm/IR/PassManager.h"
37 /// Eliminate dead arguments (and return values) from functions.
38 class DeadArgumentEliminationPass
39 : public PassInfoMixin
<DeadArgumentEliminationPass
> {
41 /// Struct that represents (part of) either a return value or a function
42 /// argument. Used so that arguments and return values can be used
49 RetOrArg(const Function
*F
, unsigned Idx
, bool IsArg
)
50 : F(F
), Idx(Idx
), IsArg(IsArg
) {}
52 /// Make RetOrArg comparable, so we can put it into a map.
53 bool operator<(const RetOrArg
&O
) const {
54 return std::tie(F
, Idx
, IsArg
) < std::tie(O
.F
, O
.Idx
, O
.IsArg
);
57 /// Make RetOrArg comparable, so we can easily iterate the multimap.
58 bool operator==(const RetOrArg
&O
) const {
59 return F
== O
.F
&& Idx
== O
.Idx
&& IsArg
== O
.IsArg
;
62 std::string
getDescription() const {
63 return (Twine(IsArg
? "Argument #" : "Return value #") + Twine(Idx
) +
64 " of function " + F
->getName())
69 /// Liveness enum - During our initial pass over the program, we determine
70 /// that things are either alive or maybe alive. We don't mark anything
71 /// explicitly dead (even if we know they are), since anything not alive
72 /// with no registered uses (in Uses) will never be marked alive and will
73 /// thus become dead in the end.
74 enum Liveness
{ Live
, MaybeLive
};
76 DeadArgumentEliminationPass(bool ShouldHackArguments_
= false)
77 : ShouldHackArguments(ShouldHackArguments_
) {}
79 PreservedAnalyses
run(Module
&M
, ModuleAnalysisManager
&);
81 /// Convenience wrapper
82 RetOrArg
CreateRet(const Function
*F
, unsigned Idx
) {
83 return RetOrArg(F
, Idx
, false);
86 /// Convenience wrapper
87 RetOrArg
CreateArg(const Function
*F
, unsigned Idx
) {
88 return RetOrArg(F
, Idx
, true);
91 using UseMap
= std::multimap
<RetOrArg
, RetOrArg
>;
93 /// This maps a return value or argument to any MaybeLive return values or
94 /// arguments it uses. This allows the MaybeLive values to be marked live
95 /// when any of its users is marked live.
96 /// For example (indices are left out for clarity):
97 /// - Uses[ret F] = ret G
98 /// This means that F calls G, and F returns the value returned by G.
99 /// - Uses[arg F] = ret G
100 /// This means that some function calls G and passes its result as an
102 /// - Uses[ret F] = arg F
103 /// This means that F returns one of its own arguments.
104 /// - Uses[arg F] = arg G
105 /// This means that G calls F and passes one of its own (G's) arguments
109 using LiveSet
= std::set
<RetOrArg
>;
110 using LiveFuncSet
= std::set
<const Function
*>;
112 /// This set contains all values that have been determined to be live.
115 /// This set contains all values that are cannot be changed in any way.
116 LiveFuncSet LiveFunctions
;
118 using UseVector
= SmallVector
<RetOrArg
, 5>;
120 /// This allows this pass to do double-duty as the dead arg hacking pass
121 /// (used only by bugpoint).
122 bool ShouldHackArguments
= false;
125 Liveness
MarkIfNotLive(RetOrArg Use
, UseVector
&MaybeLiveUses
);
126 Liveness
SurveyUse(const Use
*U
, UseVector
&MaybeLiveUses
,
127 unsigned RetValNum
= -1U);
128 Liveness
SurveyUses(const Value
*V
, UseVector
&MaybeLiveUses
);
130 void SurveyFunction(const Function
&F
);
131 void MarkValue(const RetOrArg
&RA
, Liveness L
,
132 const UseVector
&MaybeLiveUses
);
133 void MarkLive(const RetOrArg
&RA
);
134 void MarkLive(const Function
&F
);
135 void PropagateLiveness(const RetOrArg
&RA
);
136 bool RemoveDeadStuffFromFunction(Function
*F
);
137 bool DeleteDeadVarargs(Function
&Fn
);
138 bool RemoveDeadArgumentsFromCallers(Function
&Fn
);
141 } // end namespace llvm
143 #endif // LLVM_TRANSFORMS_IPO_DEADARGUMENTELIMINATION_H