• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright 2011 Christoph Bumiller
3  *
4  * Permission is hereby granted, free of charge, to any person obtaining a
5  * copy of this software and associated documentation files (the "Software"),
6  * to deal in the Software without restriction, including without limitation
7  * the rights to use, copy, modify, merge, publish, distribute, sublicense,
8  * and/or sell copies of the Software, and to permit persons to whom the
9  * Software is furnished to do so, subject to the following conditions:
10  *
11  * The above copyright notice and this permission notice shall be included in
12  * all copies or substantial portions of the Software.
13  *
14  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
15  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
16  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL
17  * THE AUTHORS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
18  * WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF
19  * OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
20  * SOFTWARE.
21  */
22 
23 #include "nv50_ir.h"
24 #include "nv50_ir_target.h"
25 
26 namespace nv50_ir {
27 
28 // Converts nv50 IR generated from TGSI to SSA form.
29 
30 // DominatorTree implements an algorithm for finding immediate dominators,
31 // as described by T. Lengauer & R. Tarjan.
32 class DominatorTree : public Graph
33 {
34 public:
35    DominatorTree(Graph *cfg);
~DominatorTree()36    ~DominatorTree() { }
37 
38    bool dominates(BasicBlock *, BasicBlock *);
39 
40    void findDominanceFrontiers();
41 
42 private:
43    void build();
44    void buildDFS(Node *);
45 
46    void squash(int);
47    inline void link(int, int);
48    inline int eval(int);
49 
50    void debugPrint();
51 
52    Graph *cfg;
53 
54    Node **vert;
55    int *data;
56    const int count;
57 
58    #define SEMI(i)     (data[(i) + 0 * count])
59    #define ANCESTOR(i) (data[(i) + 1 * count])
60    #define PARENT(i)   (data[(i) + 2 * count])
61    #define LABEL(i)    (data[(i) + 3 * count])
62    #define DOM(i)      (data[(i) + 4 * count])
63 };
64 
debugPrint()65 void DominatorTree::debugPrint()
66 {
67    for (int i = 0; i < count; ++i) {
68       INFO("SEMI(%i) = %i\n", i, SEMI(i));
69       INFO("ANCESTOR(%i) = %i\n", i, ANCESTOR(i));
70       INFO("PARENT(%i) = %i\n", i, PARENT(i));
71       INFO("LABEL(%i) = %i\n", i, LABEL(i));
72       INFO("DOM(%i) = %i\n", i, DOM(i));
73    }
74 }
75 
DominatorTree(Graph * cfgraph)76 DominatorTree::DominatorTree(Graph *cfgraph) : cfg(cfgraph),
77                                                count(cfg->getSize())
78 {
79    int i = 0;
80 
81    vert = new Node * [count];
82    data = new int[5 * count];
83 
84    for (IteratorRef it = cfg->iteratorDFS(true); !it->end(); it->next(), ++i) {
85       vert[i] = reinterpret_cast<Node *>(it->get());
86       vert[i]->tag = i;
87       LABEL(i) = i;
88       SEMI(i) = ANCESTOR(i) = -1;
89    }
90 
91    build();
92 
93    delete[] vert;
94    delete[] data;
95 }
96 
buildDFS(Graph::Node * node)97 void DominatorTree::buildDFS(Graph::Node *node)
98 {
99    SEMI(node->tag) = node->tag;
100 
101    for (Graph::EdgeIterator ei = node->outgoing(); !ei.end(); ei.next()) {
102       if (SEMI(ei.getNode()->tag) < 0) {
103          buildDFS(ei.getNode());
104          PARENT(ei.getNode()->tag) = node->tag;
105       }
106    }
107 }
108 
squash(int v)109 void DominatorTree::squash(int v)
110 {
111    if (ANCESTOR(ANCESTOR(v)) >= 0) {
112       squash(ANCESTOR(v));
113 
114       if (SEMI(LABEL(ANCESTOR(v))) < SEMI(LABEL(v)))
115          LABEL(v) = LABEL(ANCESTOR(v));
116       ANCESTOR(v) = ANCESTOR(ANCESTOR(v));
117    }
118 }
119 
eval(int v)120 int DominatorTree::eval(int v)
121 {
122    if (ANCESTOR(v) < 0)
123       return v;
124    squash(v);
125    return LABEL(v);
126 }
127 
link(int v,int w)128 void DominatorTree::link(int v, int w)
129 {
130    ANCESTOR(w) = v;
131 }
132 
build()133 void DominatorTree::build()
134 {
135    DLList *bucket = new DLList[count];
136    Node *nv, *nw;
137    int p, u, v, w;
138 
139    buildDFS(cfg->getRoot());
140 
141    for (w = count - 1; w >= 1; --w) {
142       nw = vert[w];
143       assert(nw->tag == w);
144       for (Graph::EdgeIterator ei = nw->incident(); !ei.end(); ei.next()) {
145          nv = ei.getNode();
146          v = nv->tag;
147          u = eval(v);
148          if (SEMI(u) < SEMI(w))
149             SEMI(w) = SEMI(u);
150       }
151       p = PARENT(w);
152       bucket[SEMI(w)].insert(nw);
153       link(p, w);
154 
155       for (DLList::Iterator it = bucket[p].iterator(); !it.end(); it.erase()) {
156          v = reinterpret_cast<Node *>(it.get())->tag;
157          u = eval(v);
158          DOM(v) = (SEMI(u) < SEMI(v)) ? u : p;
159       }
160    }
161    for (w = 1; w < count; ++w) {
162       if (DOM(w) != SEMI(w))
163          DOM(w) = DOM(DOM(w));
164    }
165    DOM(0) = 0;
166 
167    insert(&BasicBlock::get(cfg->getRoot())->dom);
168    do {
169       p = 0;
170       for (v = 1; v < count; ++v) {
171          nw = &BasicBlock::get(vert[DOM(v)])->dom;;
172          nv = &BasicBlock::get(vert[v])->dom;
173          if (nw->getGraph() && !nv->getGraph()) {
174             ++p;
175             nw->attach(nv, Graph::Edge::TREE);
176          }
177       }
178    } while (p);
179 
180    delete[] bucket;
181 }
182 
183 #undef SEMI
184 #undef ANCESTOR
185 #undef PARENT
186 #undef LABEL
187 #undef DOM
188 
findDominanceFrontiers()189 void DominatorTree::findDominanceFrontiers()
190 {
191    BasicBlock *bb;
192 
193    for (IteratorRef dtIt = iteratorDFS(false); !dtIt->end(); dtIt->next()) {
194       EdgeIterator succIt, chldIt;
195 
196       bb = BasicBlock::get(reinterpret_cast<Node *>(dtIt->get()));
197       bb->getDF().clear();
198 
199       for (succIt = bb->cfg.outgoing(); !succIt.end(); succIt.next()) {
200          BasicBlock *dfLocal = BasicBlock::get(succIt.getNode());
201          if (dfLocal->idom() != bb)
202             bb->getDF().insert(dfLocal);
203       }
204 
205       for (chldIt = bb->dom.outgoing(); !chldIt.end(); chldIt.next()) {
206          BasicBlock *cb = BasicBlock::get(chldIt.getNode());
207 
208          DLList::Iterator dfIt = cb->getDF().iterator();
209          for (; !dfIt.end(); dfIt.next()) {
210             BasicBlock *dfUp = BasicBlock::get(dfIt);
211             if (dfUp->idom() != bb)
212                bb->getDF().insert(dfUp);
213          }
214       }
215    }
216 }
217 
218 // liveIn(bb) = usedBeforeAssigned(bb) U (liveOut(bb) - assigned(bb))
219 void
buildLiveSetsPreSSA(BasicBlock * bb,const int seq)220 Function::buildLiveSetsPreSSA(BasicBlock *bb, const int seq)
221 {
222    Function *f = bb->getFunction();
223    BitSet usedBeforeAssigned(allLValues.getSize(), true);
224    BitSet assigned(allLValues.getSize(), true);
225 
226    bb->liveSet.allocate(allLValues.getSize(), false);
227 
228    int n = 0;
229    for (Graph::EdgeIterator ei = bb->cfg.outgoing(); !ei.end(); ei.next()) {
230       BasicBlock *out = BasicBlock::get(ei.getNode());
231       if (out == bb)
232          continue;
233       if (out->cfg.visit(seq))
234          buildLiveSetsPreSSA(out, seq);
235       if (!n++)
236          bb->liveSet = out->liveSet;
237       else
238          bb->liveSet |= out->liveSet;
239    }
240    if (!n && !bb->liveSet.marker)
241       bb->liveSet.fill(0);
242    bb->liveSet.marker = true;
243 
244    for (Instruction *i = bb->getEntry(); i; i = i->next) {
245       for (int s = 0; i->srcExists(s); ++s)
246          if (i->getSrc(s)->asLValue() && !assigned.test(i->getSrc(s)->id))
247             usedBeforeAssigned.set(i->getSrc(s)->id);
248       for (int d = 0; i->defExists(d); ++d)
249          assigned.set(i->getDef(d)->id);
250    }
251 
252    if (bb == BasicBlock::get(f->cfgExit)) {
253       for (std::deque<ValueRef>::iterator it = f->outs.begin();
254            it != f->outs.end(); ++it) {
255          if (!assigned.test(it->get()->id))
256             usedBeforeAssigned.set(it->get()->id);
257       }
258    }
259 
260    bb->liveSet.andNot(assigned);
261    bb->liveSet |= usedBeforeAssigned;
262 }
263 
264 void
buildDefSetsPreSSA(BasicBlock * bb,const int seq)265 Function::buildDefSetsPreSSA(BasicBlock *bb, const int seq)
266 {
267    bb->defSet.allocate(allLValues.getSize(), !bb->liveSet.marker);
268    bb->liveSet.marker = true;
269 
270    for (Graph::EdgeIterator ei = bb->cfg.incident(); !ei.end(); ei.next()) {
271       BasicBlock *in = BasicBlock::get(ei.getNode());
272 
273       if (in->cfg.visit(seq))
274          buildDefSetsPreSSA(in, seq);
275 
276       bb->defSet |= in->defSet;
277    }
278 
279    for (Instruction *i = bb->getEntry(); i; i = i->next) {
280       for (int d = 0; i->defExists(d); ++d)
281          bb->defSet.set(i->getDef(d)->id);
282    }
283 }
284 
285 class RenamePass
286 {
287 public:
288    RenamePass(Function *);
289    ~RenamePass();
290 
291    bool run();
292    void search(BasicBlock *);
293 
294    inline LValue *getStackTop(Value *);
295 
296    LValue *mkUndefined(Value *);
297 
298 private:
299    Stack *stack;
300    Function *func;
301    Program *prog;
302 };
303 
304 bool
convertToSSA()305 Program::convertToSSA()
306 {
307    for (ArrayList::Iterator fi = allFuncs.iterator(); !fi.end(); fi.next()) {
308       Function *fn = reinterpret_cast<Function *>(fi.get());
309       if (!fn->convertToSSA())
310          return false;
311    }
312    return true;
313 }
314 
315 // XXX: add edge from entry to exit ?
316 
317 // Efficiently Computing Static Single Assignment Form and
318 //  the Control Dependence Graph,
319 // R. Cytron, J. Ferrante, B. K. Rosen, M. N. Wegman, F. K. Zadeck
320 bool
convertToSSA()321 Function::convertToSSA()
322 {
323    // 0. calculate live in variables (for pruned SSA)
324    buildLiveSets();
325 
326    // 1. create the dominator tree
327    domTree = new DominatorTree(&cfg);
328    reinterpret_cast<DominatorTree *>(domTree)->findDominanceFrontiers();
329 
330    // 2. insert PHI functions
331    DLList workList;
332    LValue *lval;
333    BasicBlock *bb;
334    int var;
335    int iterCount = 0;
336    int *hasAlready = new int[allBBlocks.getSize() * 2];
337    int *work = &hasAlready[allBBlocks.getSize()];
338 
339    memset(hasAlready, 0, allBBlocks.getSize() * 2 * sizeof(int));
340 
341    // for each variable
342    for (var = 0; var < allLValues.getSize(); ++var) {
343       if (!allLValues.get(var))
344          continue;
345       lval = reinterpret_cast<Value *>(allLValues.get(var))->asLValue();
346       if (!lval || lval->defs.empty())
347          continue;
348       ++iterCount;
349 
350       // TODO: don't add phi functions for values that aren't used outside
351       //  the BB they're defined in
352 
353       // gather blocks with assignments to lval in workList
354       for (Value::DefIterator d = lval->defs.begin();
355            d != lval->defs.end(); ++d) {
356          bb = ((*d)->getInsn() ? (*d)->getInsn()->bb : NULL);
357          if (!bb)
358             continue; // instruction likely been removed but not XXX deleted
359 
360          if (work[bb->getId()] == iterCount)
361             continue;
362          work[bb->getId()] = iterCount;
363          workList.insert(bb);
364       }
365 
366       // for each block in workList, insert a phi for lval in the block's
367       //  dominance frontier (if we haven't already done so)
368       for (DLList::Iterator wI = workList.iterator(); !wI.end(); wI.erase()) {
369          bb = BasicBlock::get(wI);
370 
371          DLList::Iterator dfIter = bb->getDF().iterator();
372          for (; !dfIter.end(); dfIter.next()) {
373             Instruction *phi;
374             BasicBlock *dfBB = BasicBlock::get(dfIter);
375 
376             if (hasAlready[dfBB->getId()] >= iterCount)
377                continue;
378             hasAlready[dfBB->getId()] = iterCount;
379 
380             // pruned SSA: don't need a phi if the value is not live-in
381             if (!dfBB->liveSet.test(lval->id))
382                continue;
383 
384             phi = new_Instruction(this, OP_PHI, typeOfSize(lval->reg.size));
385             dfBB->insertTail(phi);
386 
387             phi->setDef(0, lval);
388             for (int s = 0; s < dfBB->cfg.incidentCount(); ++s)
389                phi->setSrc(s, lval);
390 
391             if (work[dfBB->getId()] < iterCount) {
392                work[dfBB->getId()] = iterCount;
393                wI.insert(dfBB);
394             }
395          }
396       }
397    }
398    delete[] hasAlready;
399 
400    RenamePass rename(this);
401    return rename.run();
402 }
403 
RenamePass(Function * fn)404 RenamePass::RenamePass(Function *fn) : func(fn), prog(fn->getProgram())
405 {
406    stack = new Stack[func->allLValues.getSize()];
407 }
408 
~RenamePass()409 RenamePass::~RenamePass()
410 {
411    if (stack)
412       delete[] stack;
413 }
414 
415 LValue *
getStackTop(Value * val)416 RenamePass::getStackTop(Value *val)
417 {
418    if (!stack[val->id].getSize())
419       return 0;
420    return reinterpret_cast<LValue *>(stack[val->id].peek().u.p);
421 }
422 
423 LValue *
mkUndefined(Value * val)424 RenamePass::mkUndefined(Value *val)
425 {
426    LValue *lval = val->asLValue();
427    assert(lval);
428    LValue *ud = new_LValue(func, lval);
429    Instruction *nop = new_Instruction(func, OP_NOP, typeOfSize(lval->reg.size));
430    nop->setDef(0, ud);
431    BasicBlock::get(func->cfg.getRoot())->insertHead(nop);
432    return ud;
433 }
434 
run()435 bool RenamePass::run()
436 {
437    if (!stack)
438       return false;
439    search(BasicBlock::get(func->domTree->getRoot()));
440 
441    return true;
442 }
443 
search(BasicBlock * bb)444 void RenamePass::search(BasicBlock *bb)
445 {
446    LValue *lval, *ssa;
447    int d, s;
448    const Target *targ = prog->getTarget();
449 
450    if (bb == BasicBlock::get(func->cfg.getRoot())) {
451       for (std::deque<ValueDef>::iterator it = func->ins.begin();
452            it != func->ins.end(); ++it) {
453          lval = it->get()->asLValue();
454          assert(lval);
455 
456          ssa = new_LValue(func, targ->nativeFile(lval->reg.file));
457          ssa->reg.size = lval->reg.size;
458          ssa->reg.data.id = lval->reg.data.id;
459 
460          it->setSSA(ssa);
461          stack[lval->id].push(ssa);
462       }
463    }
464 
465    for (Instruction *stmt = bb->getFirst(); stmt; stmt = stmt->next) {
466       if (stmt->op != OP_PHI) {
467          for (s = 0; stmt->srcExists(s); ++s) {
468             lval = stmt->getSrc(s)->asLValue();
469             if (!lval)
470                continue;
471             lval = getStackTop(lval);
472             if (!lval)
473                lval = mkUndefined(stmt->getSrc(s));
474             stmt->setSrc(s, lval);
475          }
476       }
477       for (d = 0; stmt->defExists(d); ++d) {
478          lval = stmt->def(d).get()->asLValue();
479          assert(lval);
480          stmt->def(d).setSSA(
481             new_LValue(func, targ->nativeFile(lval->reg.file)));
482          stmt->def(d).get()->reg.size = lval->reg.size;
483          stmt->def(d).get()->reg.data.id = lval->reg.data.id;
484          stack[lval->id].push(stmt->def(d).get());
485       }
486    }
487 
488    for (Graph::EdgeIterator ei = bb->cfg.outgoing(); !ei.end(); ei.next()) {
489       Instruction *phi;
490       int p = 0;
491       BasicBlock *sb = BasicBlock::get(ei.getNode());
492 
493       // which predecessor of sb is bb ?
494       for (Graph::EdgeIterator ei = sb->cfg.incident(); !ei.end(); ei.next()) {
495          if (ei.getNode() == &bb->cfg)
496             break;
497          ++p;
498       }
499       assert(p < sb->cfg.incidentCount());
500 
501       for (phi = sb->getPhi(); phi && phi->op == OP_PHI; phi = phi->next) {
502          lval = getStackTop(phi->getSrc(p));
503          if (!lval)
504             lval = mkUndefined(phi->getSrc(p));
505          phi->setSrc(p, lval);
506       }
507    }
508 
509    for (Graph::EdgeIterator ei = bb->dom.outgoing(); !ei.end(); ei.next())
510       search(BasicBlock::get(ei.getNode()));
511 
512    if (bb == BasicBlock::get(func->cfgExit)) {
513       for (std::deque<ValueRef>::iterator it = func->outs.begin();
514            it != func->outs.end(); ++it) {
515          lval = it->get()->asLValue();
516          if (!lval)
517             continue;
518          lval = getStackTop(lval);
519          if (!lval)
520             lval = mkUndefined(it->get());
521          it->set(lval);
522       }
523    }
524 
525    for (Instruction *stmt = bb->getFirst(); stmt; stmt = stmt->next) {
526       if (stmt->op == OP_NOP)
527          continue;
528       for (d = 0; stmt->defExists(d); ++d)
529          stack[stmt->def(d).preSSA()->id].pop();
530    }
531 }
532 
533 } // namespace nv50_ir
534