blob: c43ec33283728c27d5f9480c42a2fbb05c7bcb24 [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"
39#include "isl/set.h"
40#include "isl/map.h"
41#include "isl/union_map.h"
Tobias Grosser1d348672012-01-31 14:00:27 +000042
Tobias Grosser1d348672012-01-31 14:00:27 +000043using namespace llvm;
44using namespace polly;
45
46namespace {
Tobias Grosser37eb4222014-02-20 21:43:54 +000047
Tobias Grosser817d51d2014-02-21 20:51:46 +000048cl::opt<int> DCEPreciseSteps(
49 "polly-dce-precise-steps",
Tobias Grosser356faa82014-02-24 08:52:20 +000050 cl::desc("The number of precise steps between two approximating "
51 "iterations. (A value of -1 schedules another approximation stage "
52 "before the actual dead code elimination."),
Tobias Grosser64e8e372014-03-13 23:37:43 +000053 cl::ZeroOrMore, cl::init(-1));
Tobias Grosser1d348672012-01-31 14:00:27 +000054
Tobias Grosser184e2fc2013-01-08 08:27:46 +000055class DeadCodeElim : public ScopPass {
Tobias Grosser184e2fc2013-01-08 08:27:46 +000056public:
57 static char ID;
Tobias Grosserc30d9cc2013-03-23 00:16:05 +000058 explicit DeadCodeElim() : ScopPass(ID) {}
Tobias Grosser1d348672012-01-31 14:00:27 +000059
Tobias Grosser184e2fc2013-01-08 08:27:46 +000060 virtual bool runOnScop(Scop &S);
Tobias Grosser37eb4222014-02-20 21:43:54 +000061
Tobias Grosser184e2fc2013-01-08 08:27:46 +000062 void printScop(llvm::raw_ostream &OS) const;
63 void getAnalysisUsage(AnalysisUsage &AU) const;
Tobias Grosser37eb4222014-02-20 21:43:54 +000064
65private:
66 isl_union_set *getLastWrites(isl_union_map *Writes, isl_union_map *Schedule);
Tobias Grosser817d51d2014-02-21 20:51:46 +000067 bool eliminateDeadCode(Scop &S, int PreciseSteps);
Tobias Grosser184e2fc2013-01-08 08:27:46 +000068};
Tobias Grosser1d348672012-01-31 14:00:27 +000069}
70
71char DeadCodeElim::ID = 0;
72
Tobias Grosser37eb4222014-02-20 21:43:54 +000073/// Return the set of iterations that contains the last write for each location.
74isl_union_set *DeadCodeElim::getLastWrites(__isl_take isl_union_map *Writes,
75 __isl_take isl_union_map *Schedule) {
76 isl_union_map *WriteIterations = isl_union_map_reverse(Writes);
77 isl_union_map *WriteTimes =
78 isl_union_map_apply_range(WriteIterations, isl_union_map_copy(Schedule));
Tobias Grosser1d348672012-01-31 14:00:27 +000079
Tobias Grosser37eb4222014-02-20 21:43:54 +000080 isl_union_map *LastWriteTimes = isl_union_map_lexmax(WriteTimes);
81 isl_union_map *LastWriteIterations = isl_union_map_apply_range(
82 LastWriteTimes, isl_union_map_reverse(Schedule));
Tobias Grosser1d348672012-01-31 14:00:27 +000083
Tobias Grosser37eb4222014-02-20 21:43:54 +000084 isl_union_set *Live = isl_union_map_range(LastWriteIterations);
85 return isl_union_set_coalesce(Live);
Tobias Grosser1d348672012-01-31 14:00:27 +000086}
87
Tobias Grosser37eb4222014-02-20 21:43:54 +000088/// Performs polyhedral dead iteration elimination by:
89/// o Assuming that the last write to each location is live.
90/// o Following each RAW dependency from a live iteration backwards and adding
91/// that iteration to the live set.
Tobias Grosser817d51d2014-02-21 20:51:46 +000092///
93/// To ensure the set of live iterations does not get too complex we always
94/// combine a certain number of precise steps with one approximating step that
95/// simplifies the life set with an affine hull.
96bool DeadCodeElim::eliminateDeadCode(Scop &S, int PreciseSteps) {
Tobias Grosser37eb4222014-02-20 21:43:54 +000097 Dependences *D = &getAnalysis<Dependences>();
Tobias Grosser38c36ea2014-02-23 15:15:44 +000098
99 if (!D->hasValidDependences())
100 return false;
101
102 isl_union_set *Live = this->getLastWrites(S.getWrites(), S.getSchedule());
Johannes Doerfertf1906132014-06-20 16:37:11 +0000103 isl_union_map *Dep =
104 D->getDependences(Dependences::TYPE_RAW | Dependences::TYPE_RED);
Tobias Grosser37eb4222014-02-20 21:43:54 +0000105 Dep = isl_union_map_reverse(Dep);
106
Tobias Grosser356faa82014-02-24 08:52:20 +0000107 if (PreciseSteps == -1)
108 Live = isl_union_set_affine_hull(Live);
109
Tobias Grosser37eb4222014-02-20 21:43:54 +0000110 isl_union_set *OriginalDomain = S.getDomains();
Tobias Grosser817d51d2014-02-21 20:51:46 +0000111 int Steps = 0;
Tobias Grosser0bc0ad02014-02-21 21:06:08 +0000112 while (true) {
Tobias Grosser37eb4222014-02-20 21:43:54 +0000113 isl_union_set *Extra;
Tobias Grosser817d51d2014-02-21 20:51:46 +0000114 Steps++;
Tobias Grosser37eb4222014-02-20 21:43:54 +0000115
116 Extra =
117 isl_union_set_apply(isl_union_set_copy(Live), isl_union_map_copy(Dep));
118
119 if (isl_union_set_is_subset(Extra, Live)) {
120 isl_union_set_free(Extra);
121 break;
122 }
123
124 Live = isl_union_set_union(Live, Extra);
Tobias Grosser817d51d2014-02-21 20:51:46 +0000125
126 if (Steps > PreciseSteps) {
127 Steps = 0;
Tobias Grosser37eb4222014-02-20 21:43:54 +0000128 Live = isl_union_set_affine_hull(Live);
Tobias Grosser817d51d2014-02-21 20:51:46 +0000129 }
130
Tobias Grosser37eb4222014-02-20 21:43:54 +0000131 Live = isl_union_set_intersect(Live, isl_union_set_copy(OriginalDomain));
132 }
133 isl_union_map_free(Dep);
134 isl_union_set_free(OriginalDomain);
135
136 return S.restrictDomains(isl_union_set_coalesce(Live));
137}
138
Tobias Grosser817d51d2014-02-21 20:51:46 +0000139bool DeadCodeElim::runOnScop(Scop &S) {
140 return eliminateDeadCode(S, DCEPreciseSteps);
141}
Tobias Grosser37eb4222014-02-20 21:43:54 +0000142
Tobias Grosserc30d9cc2013-03-23 00:16:05 +0000143void DeadCodeElim::printScop(raw_ostream &OS) const {}
Tobias Grosser1d348672012-01-31 14:00:27 +0000144
145void DeadCodeElim::getAnalysisUsage(AnalysisUsage &AU) const {
146 ScopPass::getAnalysisUsage(AU);
147 AU.addRequired<Dependences>();
148}
149
Tobias Grosserc30d9cc2013-03-23 00:16:05 +0000150Pass *polly::createDeadCodeElimPass() { return new DeadCodeElim(); }
Tobias Grosser1d348672012-01-31 14:00:27 +0000151
Tobias Grosserc30d9cc2013-03-23 00:16:05 +0000152INITIALIZE_PASS_BEGIN(DeadCodeElim, "polly-dce",
Tobias Grosserd7e58642013-04-10 06:55:45 +0000153 "Polly - Remove dead iterations", false, false)
154INITIALIZE_PASS_DEPENDENCY(Dependences)
155INITIALIZE_PASS_DEPENDENCY(ScopInfo)
Tobias Grosserc30d9cc2013-03-23 00:16:05 +0000156INITIALIZE_PASS_END(DeadCodeElim, "polly-dce", "Polly - Remove dead iterations",
157 false, false)