210 lines
5.2 KiB
C
210 lines
5.2 KiB
C
#include "mcg.h"
|
|
|
|
struct graph_data cfg;
|
|
struct dominance_data dominance;
|
|
|
|
static ARRAYOF(struct basicblock) pending;
|
|
|
|
static bool collect_outputs_cb(struct ir* ir, void* user)
|
|
{
|
|
struct basicblock* caller = user;
|
|
|
|
if (ir->opcode == IR_BLOCK)
|
|
{
|
|
array_appendu(&caller->nexts, ir->u.bvalue);
|
|
array_appendu(&ir->u.bvalue->prevs, caller);
|
|
pmap_add(&cfg.graph, caller, ir->u.bvalue);
|
|
}
|
|
return false;
|
|
}
|
|
|
|
static void update_block_pointers_from_ir(void)
|
|
{
|
|
int i, j;
|
|
|
|
for (i=0; i<current_proc->blocks.count; i++)
|
|
{
|
|
struct basicblock* bb = current_proc->blocks.item[i];
|
|
bb->prevs.count = bb->nexts.count = 0;
|
|
}
|
|
|
|
for (i=0; i<current_proc->blocks.count; i++)
|
|
{
|
|
struct basicblock* bb = current_proc->blocks.item[i];
|
|
for (j=0; j<bb->irs.count; j++)
|
|
ir_walk(bb->irs.item[j], collect_outputs_cb, bb);
|
|
}
|
|
|
|
for (i=0; i<current_proc->blocks.count; i++)
|
|
{
|
|
struct basicblock* bb = current_proc->blocks.item[i];
|
|
|
|
for (j=0; j<bb->nexts.count; j++)
|
|
{
|
|
tracef('D', "D: cfg graph %s -> %s\n",
|
|
bb->name,
|
|
bb->nexts.item[j]->name);
|
|
}
|
|
}
|
|
}
|
|
|
|
static void recursively_walk_cfg_graph(struct basicblock* bb)
|
|
{
|
|
int i;
|
|
|
|
if (array_contains(&cfg.postorder, bb) || array_contains(&pending, bb))
|
|
return;
|
|
|
|
array_appendu(&cfg.preorder, bb);
|
|
array_appendu(&pending, bb);
|
|
|
|
for (i=0; i<bb->nexts.count; i++)
|
|
recursively_walk_cfg_graph(bb->nexts.item[i]);
|
|
|
|
array_remove(&pending, bb);
|
|
bb->order = cfg.postorder.count;
|
|
array_appendu(&cfg.postorder, bb);
|
|
}
|
|
|
|
static void walk_cfg_graph(void)
|
|
{
|
|
int i;
|
|
|
|
cfg.preorder.count = 0;
|
|
cfg.postorder.count = 0;
|
|
pending.count = 0;
|
|
recursively_walk_cfg_graph(cfg.entry);
|
|
|
|
for (i=0; i<cfg.preorder.count; i++)
|
|
{
|
|
tracef('G', "G: cfg preorder: %s\n",
|
|
cfg.preorder.item[i]->name);
|
|
}
|
|
}
|
|
|
|
static struct basicblock* intersect(struct basicblock* p1, struct basicblock* p2)
|
|
{
|
|
while (p1 != p2)
|
|
{
|
|
while (p1->order < p2->order)
|
|
p1 = pmap_findleft(&dominance.graph, p1);
|
|
|
|
while (p2->order < p1->order)
|
|
p2 = pmap_findleft(&dominance.graph, p2);
|
|
}
|
|
|
|
return p1;
|
|
}
|
|
|
|
static void calculate_dominance_graph(void)
|
|
{
|
|
/* This is the algorithm described here:
|
|
*
|
|
* Cooper, Keith D., Timothy J. Harvey, and Ken Kennedy.
|
|
* "A simple, fast dominance algorithm."
|
|
* Software Practice & Experience 4.1-10 (2001): 1-8.
|
|
*
|
|
* https://www.cs.rice.edu/~keith/EMBED/dom.pdf
|
|
*/
|
|
|
|
int i, j;
|
|
bool changed;
|
|
|
|
dominance.graph.count = 0;
|
|
|
|
/* The entry block dominates itself. */
|
|
|
|
pmap_put(&dominance.graph, cfg.entry, cfg.entry);
|
|
|
|
do
|
|
{
|
|
changed = false;
|
|
|
|
for (i = cfg.postorder.count-2; i >= 0; i--)
|
|
{
|
|
struct basicblock* b = cfg.postorder.item[i];
|
|
struct basicblock* new_idom = NULL;
|
|
for (j=0; j<b->prevs.count; j++)
|
|
{
|
|
struct basicblock* p = b->prevs.item[j];
|
|
|
|
/* Skip unprocessed blocks. */
|
|
if (!pmap_findleft(&dominance.graph, p))
|
|
continue;
|
|
|
|
if (!new_idom)
|
|
new_idom = p;
|
|
else if (pmap_findleft(&dominance.graph, p))
|
|
new_idom = intersect(p, new_idom);
|
|
}
|
|
|
|
if (pmap_findleft(&dominance.graph, b) != new_idom)
|
|
{
|
|
pmap_put(&dominance.graph, b, new_idom);
|
|
changed = true;
|
|
}
|
|
}
|
|
}
|
|
while (changed);
|
|
|
|
for (i=0; i<dominance.graph.count; i++)
|
|
{
|
|
tracef('G', "G: dominance graph: %s -> %s\n",
|
|
dominance.graph.item[i].right->name,
|
|
dominance.graph.item[i].left->name);
|
|
}
|
|
}
|
|
|
|
static void recursively_walk_dominance_graph(struct basicblock* bb)
|
|
{
|
|
int i;
|
|
|
|
if (array_contains(&dominance.postorder, bb) || array_contains(&pending, bb))
|
|
return;
|
|
|
|
array_appendu(&dominance.preorder, bb);
|
|
array_appendu(&pending, bb);
|
|
|
|
for (i=0; i<dominance.graph.count; i++)
|
|
if (dominance.graph.item[i].right == bb)
|
|
recursively_walk_dominance_graph(dominance.graph.item[i].left);
|
|
|
|
array_remove(&pending, bb);
|
|
array_appendu(&dominance.postorder, bb);
|
|
}
|
|
|
|
static void walk_dominance_graph(void)
|
|
{
|
|
int i;
|
|
|
|
dominance.preorder.count = 0;
|
|
dominance.postorder.count = 0;
|
|
pending.count = 0;
|
|
recursively_walk_dominance_graph(cfg.entry);
|
|
|
|
for (i=0; i<dominance.preorder.count; i++)
|
|
{
|
|
tracef('G', "G: dominance preorder: %s\n",
|
|
dominance.preorder.item[i]->name);
|
|
}
|
|
}
|
|
|
|
void update_graph_data(void)
|
|
{
|
|
cfg.entry = current_proc->blocks.item[0];
|
|
cfg.graph.count = 0;
|
|
update_block_pointers_from_ir();
|
|
|
|
walk_cfg_graph();
|
|
assert(cfg.postorder.count == current_proc->blocks.count);
|
|
assert(cfg.preorder.count == current_proc->blocks.count);
|
|
|
|
calculate_dominance_graph();
|
|
|
|
walk_dominance_graph();
|
|
assert(dominance.postorder.count == dominance.graph.count);
|
|
assert(dominance.preorder.count == dominance.graph.count);
|
|
}
|
|
|
|
/* vim: set sw=4 ts=4 expandtab : */
|
|
|