blob: 9a935ca53f4a045672f12537b8763eb80157e79d [file] [log] [blame]
Krzysztof Parzyszek12798812016-01-12 19:09:01 +00001//===--- HexagonRDFOpt.cpp ------------------------------------------------===//
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
10#include "HexagonInstrInfo.h"
11#include "HexagonRDF.h"
12#include "HexagonSubtarget.h"
13#include "RDFCopy.h"
14#include "RDFDeadCode.h"
15#include "RDFGraph.h"
16#include "RDFLiveness.h"
17#include "llvm/ADT/SetVector.h"
18#include "llvm/CodeGen/MachineBasicBlock.h"
19#include "llvm/CodeGen/MachineDominanceFrontier.h"
20#include "llvm/CodeGen/MachineDominators.h"
21#include "llvm/CodeGen/MachineFunction.h"
22#include "llvm/CodeGen/MachineFunctionPass.h"
23#include "llvm/CodeGen/MachineRegisterInfo.h"
24#include "llvm/Support/CommandLine.h"
25#include "llvm/Support/Format.h"
26#include "llvm/Target/TargetInstrInfo.h"
27#include "llvm/Target/TargetRegisterInfo.h"
28
29using namespace llvm;
30using namespace rdf;
31
32namespace llvm {
33 void initializeHexagonRDFOptPass(PassRegistry&);
34 FunctionPass *createHexagonRDFOpt();
35}
36
37namespace {
Krzysztof Parzyszek12798812016-01-12 19:09:01 +000038 unsigned RDFCount = 0;
Krzysztof Parzyszek7aae9b32016-01-18 20:45:51 +000039 cl::opt<unsigned> RDFLimit("rdf-limit", cl::init(UINT_MAX));
Krzysztof Parzyszek12798812016-01-12 19:09:01 +000040 cl::opt<bool> RDFDump("rdf-dump", cl::init(false));
41
42 class HexagonRDFOpt : public MachineFunctionPass {
43 public:
44 HexagonRDFOpt() : MachineFunctionPass(ID) {
45 initializeHexagonRDFOptPass(*PassRegistry::getPassRegistry());
46 }
47 void getAnalysisUsage(AnalysisUsage &AU) const override {
48 AU.addRequired<MachineDominatorTree>();
49 AU.addRequired<MachineDominanceFrontier>();
50 AU.setPreservesAll();
51 MachineFunctionPass::getAnalysisUsage(AU);
52 }
53 const char *getPassName() const override {
54 return "Hexagon RDF optimizations";
55 }
56 bool runOnMachineFunction(MachineFunction &MF) override;
57
58 static char ID;
59
60 private:
61 MachineDominatorTree *MDT;
62 MachineRegisterInfo *MRI;
63 };
64
65 char HexagonRDFOpt::ID = 0;
66}
67
68INITIALIZE_PASS_BEGIN(HexagonRDFOpt, "rdfopt", "Hexagon RDF opt", false, false)
69INITIALIZE_PASS_DEPENDENCY(MachineDominatorTree)
70INITIALIZE_PASS_DEPENDENCY(MachineDominanceFrontier)
71INITIALIZE_PASS_END(HexagonRDFOpt, "rdfopt", "Hexagon RDF opt", false, false)
72
73
Krzysztof Parzyszek7aae9b32016-01-18 20:45:51 +000074struct HexagonCP : public CopyPropagation {
75 HexagonCP(DataFlowGraph &G) : CopyPropagation(G) {}
76 bool interpretAsCopy(const MachineInstr *MI, EqualityMap &EM) override;
77};
78
79
Krzysztof Parzyszek12798812016-01-12 19:09:01 +000080struct HexagonDCE : public DeadCodeElimination {
Krzysztof Parzyszekf62d44b2016-01-12 19:27:59 +000081 HexagonDCE(DataFlowGraph &G, MachineRegisterInfo &MRI)
82 : DeadCodeElimination(G, MRI) {}
Krzysztof Parzyszek12798812016-01-12 19:09:01 +000083 bool rewrite(NodeAddr<InstrNode*> IA, SetVector<NodeId> &Remove);
84 void removeOperand(NodeAddr<InstrNode*> IA, unsigned OpNum);
85
86 bool run();
87};
88
89
Krzysztof Parzyszek7aae9b32016-01-18 20:45:51 +000090bool HexagonCP::interpretAsCopy(const MachineInstr *MI, EqualityMap &EM) {
91 auto mapRegs = [MI,&EM] (RegisterRef DstR, RegisterRef SrcR) -> void {
92 EM.insert(std::make_pair(DstR, SrcR));
93 };
94
95 unsigned Opc = MI->getOpcode();
96 switch (Opc) {
97 case Hexagon::A2_combinew: {
98 const MachineOperand &DstOp = MI->getOperand(0);
99 const MachineOperand &HiOp = MI->getOperand(1);
100 const MachineOperand &LoOp = MI->getOperand(2);
101 assert(DstOp.getSubReg() == 0 && "Unexpected subregister");
102 mapRegs({ DstOp.getReg(), Hexagon::subreg_hireg },
103 { HiOp.getReg(), HiOp.getSubReg() });
104 mapRegs({ DstOp.getReg(), Hexagon::subreg_loreg },
105 { LoOp.getReg(), LoOp.getSubReg() });
106 return true;
107 }
108 case Hexagon::A2_addi: {
109 const MachineOperand &A = MI->getOperand(2);
110 if (!A.isImm() || A.getImm() != 0)
111 return false;
112 }
113 // Fall through.
114 case Hexagon::A2_tfr: {
115 const MachineOperand &DstOp = MI->getOperand(0);
116 const MachineOperand &SrcOp = MI->getOperand(1);
117 mapRegs({ DstOp.getReg(), DstOp.getSubReg() },
118 { SrcOp.getReg(), SrcOp.getSubReg() });
119 return true;
120 }
121 }
122
123 return CopyPropagation::interpretAsCopy(MI, EM);
124}
125
126
Krzysztof Parzyszek12798812016-01-12 19:09:01 +0000127bool HexagonDCE::run() {
128 bool Collected = collect();
129 if (!Collected)
130 return false;
131
132 const SetVector<NodeId> &DeadNodes = getDeadNodes();
133 const SetVector<NodeId> &DeadInstrs = getDeadInstrs();
134
135 typedef DenseMap<NodeId,NodeId> RefToInstrMap;
136 RefToInstrMap R2I;
137 SetVector<NodeId> PartlyDead;
138 DataFlowGraph &DFG = getDFG();
139
140 for (NodeAddr<BlockNode*> BA : DFG.getFunc().Addr->members(DFG)) {
141 for (auto TA : BA.Addr->members_if(DFG.IsCode<NodeAttrs::Stmt>, DFG)) {
142 NodeAddr<StmtNode*> SA = TA;
143 for (NodeAddr<RefNode*> RA : SA.Addr->members(DFG)) {
144 R2I.insert(std::make_pair(RA.Id, SA.Id));
145 if (DFG.IsDef(RA) && DeadNodes.count(RA.Id))
146 if (!DeadInstrs.count(SA.Id))
147 PartlyDead.insert(SA.Id);
148 }
149 }
150 }
151
Krzysztof Parzyszek7aae9b32016-01-18 20:45:51 +0000152
Krzysztof Parzyszek12798812016-01-12 19:09:01 +0000153 // Nodes to remove.
154 SetVector<NodeId> Remove = DeadInstrs;
155
156 bool Changed = false;
157 for (NodeId N : PartlyDead) {
158 auto SA = DFG.addr<StmtNode*>(N);
159 if (trace())
160 dbgs() << "Partly dead: " << *SA.Addr->getCode();
161 Changed |= rewrite(SA, Remove);
162 }
163
164 return erase(Remove) || Changed;
165}
166
167
168void HexagonDCE::removeOperand(NodeAddr<InstrNode*> IA, unsigned OpNum) {
169 MachineInstr *MI = NodeAddr<StmtNode*>(IA).Addr->getCode();
170
171 auto getOpNum = [MI] (MachineOperand &Op) -> unsigned {
172 for (unsigned i = 0, n = MI->getNumOperands(); i != n; ++i)
173 if (&MI->getOperand(i) == &Op)
174 return i;
175 llvm_unreachable("Invalid operand");
176 };
177 DenseMap<NodeId,unsigned> OpMap;
178 NodeList Refs = IA.Addr->members(getDFG());
179 for (NodeAddr<RefNode*> RA : Refs)
180 OpMap.insert(std::make_pair(RA.Id, getOpNum(RA.Addr->getOp())));
181
182 MI->RemoveOperand(OpNum);
183
184 for (NodeAddr<RefNode*> RA : Refs) {
185 unsigned N = OpMap[RA.Id];
186 if (N < OpNum)
187 RA.Addr->setRegRef(&MI->getOperand(N));
188 else if (N > OpNum)
189 RA.Addr->setRegRef(&MI->getOperand(N-1));
190 }
191}
192
193
194bool HexagonDCE::rewrite(NodeAddr<InstrNode*> IA, SetVector<NodeId> &Remove) {
195 if (!getDFG().IsCode<NodeAttrs::Stmt>(IA))
196 return false;
197 DataFlowGraph &DFG = getDFG();
198 MachineInstr *MI = NodeAddr<StmtNode*>(IA).Addr->getCode();
199 auto &HII = static_cast<const HexagonInstrInfo&>(DFG.getTII());
200 if (HII.getAddrMode(MI) != HexagonII::PostInc)
201 return false;
202 unsigned Opc = MI->getOpcode();
203 unsigned OpNum, NewOpc;
204 switch (Opc) {
205 case Hexagon::L2_loadri_pi:
206 NewOpc = Hexagon::L2_loadri_io;
207 OpNum = 1;
208 break;
209 case Hexagon::L2_loadrd_pi:
210 NewOpc = Hexagon::L2_loadrd_io;
211 OpNum = 1;
212 break;
213 case Hexagon::V6_vL32b_pi:
214 NewOpc = Hexagon::V6_vL32b_ai;
215 OpNum = 1;
216 break;
217 case Hexagon::S2_storeri_pi:
218 NewOpc = Hexagon::S2_storeri_io;
219 OpNum = 0;
220 break;
221 case Hexagon::S2_storerd_pi:
222 NewOpc = Hexagon::S2_storerd_io;
223 OpNum = 0;
224 break;
225 case Hexagon::V6_vS32b_pi:
226 NewOpc = Hexagon::V6_vS32b_ai;
227 OpNum = 0;
228 break;
229 default:
230 return false;
231 }
232 auto IsDead = [this] (NodeAddr<DefNode*> DA) -> bool {
233 return getDeadNodes().count(DA.Id);
234 };
235 NodeList Defs;
236 MachineOperand &Op = MI->getOperand(OpNum);
237 for (NodeAddr<DefNode*> DA : IA.Addr->members_if(DFG.IsDef, DFG)) {
238 if (&DA.Addr->getOp() != &Op)
239 continue;
240 Defs = DFG.getRelatedRefs(IA, DA);
241 if (!std::all_of(Defs.begin(), Defs.end(), IsDead))
242 return false;
243 break;
244 }
245
246 // Mark all nodes in Defs for removal.
247 for (auto D : Defs)
248 Remove.insert(D.Id);
249
250 if (trace())
251 dbgs() << "Rewriting: " << *MI;
252 MI->setDesc(HII.get(NewOpc));
253 MI->getOperand(OpNum+2).setImm(0);
254 removeOperand(IA, OpNum);
255 if (trace())
256 dbgs() << " to: " << *MI;
257
258 return true;
259}
260
261
262bool HexagonRDFOpt::runOnMachineFunction(MachineFunction &MF) {
263 if (RDFLimit.getPosition()) {
264 if (RDFCount >= RDFLimit)
265 return false;
266 RDFCount++;
267 }
268
269 MDT = &getAnalysis<MachineDominatorTree>();
270 const auto &MDF = getAnalysis<MachineDominanceFrontier>();
271 const auto &HII = *MF.getSubtarget<HexagonSubtarget>().getInstrInfo();
272 const auto &HRI = *MF.getSubtarget<HexagonSubtarget>().getRegisterInfo();
273 MRI = &MF.getRegInfo();
Krzysztof Parzyszek7aae9b32016-01-18 20:45:51 +0000274 bool Changed;
Krzysztof Parzyszek12798812016-01-12 19:09:01 +0000275
276 if (RDFDump)
277 MF.print(dbgs() << "Before " << getPassName() << "\n", nullptr);
Krzysztof Parzyszek7aae9b32016-01-18 20:45:51 +0000278
279 HexagonRegisterAliasInfo HAI(HRI);
280 TargetOperandInfo TOI(HII);
Krzysztof Parzyszek12798812016-01-12 19:09:01 +0000281 DataFlowGraph G(MF, HII, HRI, *MDT, MDF, HAI, TOI);
282 G.build();
Krzysztof Parzyszek12798812016-01-12 19:09:01 +0000283
Krzysztof Parzyszek7aae9b32016-01-18 20:45:51 +0000284 if (RDFDump)
285 dbgs() << "Starting copy propagation on: " << MF.getName() << '\n'
286 << PrintNode<FuncNode*>(G.getFunc(), G) << '\n';
287 HexagonCP CP(G);
Krzysztof Parzyszek12798812016-01-12 19:09:01 +0000288 CP.trace(RDFDump);
289 Changed = CP.run();
Krzysztof Parzyszek12798812016-01-12 19:09:01 +0000290
Krzysztof Parzyszek7aae9b32016-01-18 20:45:51 +0000291 if (RDFDump)
292 dbgs() << "Starting dead code elimination on: " << MF.getName() << '\n'
293 << PrintNode<FuncNode*>(G.getFunc(), G) << '\n';
Krzysztof Parzyszek12798812016-01-12 19:09:01 +0000294 HexagonDCE DCE(G, *MRI);
295 DCE.trace(RDFDump);
296 Changed |= DCE.run();
297
298 if (Changed) {
Krzysztof Parzyszek7aae9b32016-01-18 20:45:51 +0000299 if (RDFDump)
300 dbgs() << "Starting liveness recomputation on: " << MF.getName() << '\n';
Krzysztof Parzyszek12798812016-01-12 19:09:01 +0000301 Liveness LV(*MRI, G);
302 LV.trace(RDFDump);
303 LV.computeLiveIns();
304 LV.resetLiveIns();
305 LV.resetKills();
306 }
307
308 if (RDFDump)
309 MF.print(dbgs() << "After " << getPassName() << "\n", nullptr);
Krzysztof Parzyszek7aae9b32016-01-18 20:45:51 +0000310
Krzysztof Parzyszek12798812016-01-12 19:09:01 +0000311 return false;
312}
313
314
315FunctionPass *llvm::createHexagonRDFOpt() {
316 return new HexagonRDFOpt();
317}