blob: 2662dc9c0dc7bc79fe66525d1e2f067edf3810e4 [file] [log] [blame]
Tobias Grosser1d348672012-01-31 14:00:27 +00001//===- DeadCodeElimination.cpp - Eliminate dead iteration ----------------===//
2//
3// The LLVM Compiler Infrastructure
4//
5// This file is distributed under the University of Illinois Open Source
6// License. See LICENSE.TXT for details.
7//
8//===----------------------------------------------------------------------===//
9//
Tobias Grosserecae17f2014-02-21 20:51:40 +000010// The polyhedral dead code elimination pass analyses a SCoP to eliminate
11// statement instances that can be proven dead.
12// As a consequence, the code generated for this SCoP may execute a statement
13// less often. This means, a statement may be executed only in certain loop
14// iterations or it may not even be part of the generated code at all.
Tobias Grosserc30d9cc2013-03-23 00:16:05 +000015//
Tobias Grosserecae17f2014-02-21 20:51:40 +000016// This code:
Tobias Grosserc30d9cc2013-03-23 00:16:05 +000017//
Tobias Grosserecae17f2014-02-21 20:51:40 +000018// for (i = 0; i < N; i++)
19// arr[i] = 0;
20// for (i = 0; i < N; i++)
21// arr[i] = 10;
22// for (i = 0; i < N; i++)
23// arr[i] = i;
24//
25// is e.g. simplified to:
26//
27// for (i = 0; i < N; i++)
28// arr[i] = i;
29//
30// The idea and the algorithm used was first implemented by Sven Verdoolaege in
31// the 'ppcg' tool.
Tobias Grosserc30d9cc2013-03-23 00:16:05 +000032//
Tobias Grosser1d348672012-01-31 14:00:27 +000033//===----------------------------------------------------------------------===//
34
35#include "polly/Dependences.h"
36#include "polly/LinkAllPasses.h"
37#include "polly/ScopInfo.h"
Tobias Grosser37eb4222014-02-20 21:43:54 +000038#include "llvm/Support/CommandLine.h"
Tobias Grosser780ce0f2014-07-11 07:12:10 +000039#include "isl/flow.h"
Tobias Grosser37eb4222014-02-20 21:43:54 +000040#include "isl/set.h"
41#include "isl/map.h"
42#include "isl/union_map.h"
Tobias Grosser1d348672012-01-31 14:00:27 +000043
Tobias Grosser1d348672012-01-31 14:00:27 +000044using namespace llvm;
45using namespace polly;
46
47namespace {
Tobias Grosser37eb4222014-02-20 21:43:54 +000048
Tobias Grosser817d51d2014-02-21 20:51:46 +000049cl::opt<int> DCEPreciseSteps(
50 "polly-dce-precise-steps",
Tobias Grosser356faa82014-02-24 08:52:20 +000051 cl::desc("The number of precise steps between two approximating "
52 "iterations. (A value of -1 schedules another approximation stage "
53 "before the actual dead code elimination."),
Tobias Grosser64e8e372014-03-13 23:37:43 +000054 cl::ZeroOrMore, cl::init(-1));
Tobias Grosser1d348672012-01-31 14:00:27 +000055
Tobias Grosser184e2fc2013-01-08 08:27:46 +000056class DeadCodeElim : public ScopPass {
Tobias Grosser184e2fc2013-01-08 08:27:46 +000057public:
58 static char ID;
Tobias Grosserc30d9cc2013-03-23 00:16:05 +000059 explicit DeadCodeElim() : ScopPass(ID) {}
Tobias Grosser1d348672012-01-31 14:00:27 +000060
Tobias Grosser184e2fc2013-01-08 08:27:46 +000061 virtual bool runOnScop(Scop &S);
Tobias Grosser37eb4222014-02-20 21:43:54 +000062
Tobias Grosser184e2fc2013-01-08 08:27:46 +000063 void printScop(llvm::raw_ostream &OS) const;
64 void getAnalysisUsage(AnalysisUsage &AU) const;
Tobias Grosser37eb4222014-02-20 21:43:54 +000065
66private:
Tobias Grosser780ce0f2014-07-11 07:12:10 +000067 /// @brief Return the set of live iterations.
68 ///
69 /// The set of live iterations are all iterations that write to memory and for
70 /// which we can not prove that there will be a later write that _must_
71 /// overwrite the same memory location and is consequently the only one that
72 /// is visible after the execution of the SCoP.
73 ///
74 isl_union_set *getLiveOut(Scop &S);
Tobias Grosser817d51d2014-02-21 20:51:46 +000075 bool eliminateDeadCode(Scop &S, int PreciseSteps);
Tobias Grosser184e2fc2013-01-08 08:27:46 +000076};
Tobias Grosser1d348672012-01-31 14:00:27 +000077}
78
79char DeadCodeElim::ID = 0;
80
Tobias Grosser780ce0f2014-07-11 07:12:10 +000081// To compute the live outs, we first assume all must and may-writes are exposed
82// and then subtract the set of statements that are definitely overwritten.
83isl_union_set *DeadCodeElim::getLiveOut(Scop &S) {
84 __isl_take isl_union_map *Kills = S.getMustWrites();
85 __isl_take isl_union_map *Empty = isl_union_map_empty(S.getParamSpace());
Tobias Grosser1d348672012-01-31 14:00:27 +000086
Tobias Grosser780ce0f2014-07-11 07:12:10 +000087 isl_union_map *Covering;
88 isl_union_map *Writes = S.getWrites();
89 isl_union_map_compute_flow(Kills, Empty, isl_union_map_copy(Writes),
90 S.getSchedule(), NULL, &Covering, NULL, NULL);
Tobias Grosser1d348672012-01-31 14:00:27 +000091
Tobias Grosser780ce0f2014-07-11 07:12:10 +000092 isl_union_map *Exposed = Writes;
93 Exposed =
94 isl_union_map_subtract_domain(Exposed, isl_union_map_domain(Covering));
95 return isl_union_map_domain(Exposed);
Tobias Grosser1d348672012-01-31 14:00:27 +000096}
97
Tobias Grosser37eb4222014-02-20 21:43:54 +000098/// Performs polyhedral dead iteration elimination by:
99/// o Assuming that the last write to each location is live.
100/// o Following each RAW dependency from a live iteration backwards and adding
101/// that iteration to the live set.
Tobias Grosser817d51d2014-02-21 20:51:46 +0000102///
103/// To ensure the set of live iterations does not get too complex we always
104/// combine a certain number of precise steps with one approximating step that
105/// simplifies the life set with an affine hull.
106bool DeadCodeElim::eliminateDeadCode(Scop &S, int PreciseSteps) {
Tobias Grosser37eb4222014-02-20 21:43:54 +0000107 Dependences *D = &getAnalysis<Dependences>();
Tobias Grosser38c36ea2014-02-23 15:15:44 +0000108
109 if (!D->hasValidDependences())
110 return false;
111
Tobias Grosser780ce0f2014-07-11 07:12:10 +0000112 isl_union_set *Live = getLiveOut(S);
Johannes Doerfertf1906132014-06-20 16:37:11 +0000113 isl_union_map *Dep =
114 D->getDependences(Dependences::TYPE_RAW | Dependences::TYPE_RED);
Tobias Grosser37eb4222014-02-20 21:43:54 +0000115 Dep = isl_union_map_reverse(Dep);
116
Tobias Grosser356faa82014-02-24 08:52:20 +0000117 if (PreciseSteps == -1)
118 Live = isl_union_set_affine_hull(Live);
119
Tobias Grosser37eb4222014-02-20 21:43:54 +0000120 isl_union_set *OriginalDomain = S.getDomains();
Tobias Grosser817d51d2014-02-21 20:51:46 +0000121 int Steps = 0;
Tobias Grosser0bc0ad02014-02-21 21:06:08 +0000122 while (true) {
Tobias Grosser37eb4222014-02-20 21:43:54 +0000123 isl_union_set *Extra;
Tobias Grosser817d51d2014-02-21 20:51:46 +0000124 Steps++;
Tobias Grosser37eb4222014-02-20 21:43:54 +0000125
126 Extra =
127 isl_union_set_apply(isl_union_set_copy(Live), isl_union_map_copy(Dep));
128
129 if (isl_union_set_is_subset(Extra, Live)) {
130 isl_union_set_free(Extra);
131 break;
132 }
133
134 Live = isl_union_set_union(Live, Extra);
Tobias Grosser817d51d2014-02-21 20:51:46 +0000135
136 if (Steps > PreciseSteps) {
137 Steps = 0;
Tobias Grosser37eb4222014-02-20 21:43:54 +0000138 Live = isl_union_set_affine_hull(Live);
Tobias Grosser817d51d2014-02-21 20:51:46 +0000139 }
140
Tobias Grosser37eb4222014-02-20 21:43:54 +0000141 Live = isl_union_set_intersect(Live, isl_union_set_copy(OriginalDomain));
142 }
143 isl_union_map_free(Dep);
144 isl_union_set_free(OriginalDomain);
145
146 return S.restrictDomains(isl_union_set_coalesce(Live));
147}
148
Tobias Grosser817d51d2014-02-21 20:51:46 +0000149bool DeadCodeElim::runOnScop(Scop &S) {
150 return eliminateDeadCode(S, DCEPreciseSteps);
151}
Tobias Grosser37eb4222014-02-20 21:43:54 +0000152
Tobias Grosserc30d9cc2013-03-23 00:16:05 +0000153void DeadCodeElim::printScop(raw_ostream &OS) const {}
Tobias Grosser1d348672012-01-31 14:00:27 +0000154
155void DeadCodeElim::getAnalysisUsage(AnalysisUsage &AU) const {
156 ScopPass::getAnalysisUsage(AU);
157 AU.addRequired<Dependences>();
158}
159
Tobias Grosserc30d9cc2013-03-23 00:16:05 +0000160Pass *polly::createDeadCodeElimPass() { return new DeadCodeElim(); }
Tobias Grosser1d348672012-01-31 14:00:27 +0000161
Tobias Grosserc30d9cc2013-03-23 00:16:05 +0000162INITIALIZE_PASS_BEGIN(DeadCodeElim, "polly-dce",
Tobias Grosserd7e58642013-04-10 06:55:45 +0000163 "Polly - Remove dead iterations", false, false)
164INITIALIZE_PASS_DEPENDENCY(Dependences)
165INITIALIZE_PASS_DEPENDENCY(ScopInfo)
Tobias Grosserc30d9cc2013-03-23 00:16:05 +0000166INITIALIZE_PASS_END(DeadCodeElim, "polly-dce", "Polly - Remove dead iterations",
167 false, false)