2 * dominators.c: Dominator computation on the control flow graph
5 * Dietmar Maurer (dietmar@ximian.com)
6 * Paolo Molaro (lupus@ximian.com)
8 * (C) 2003 Ximian, Inc.
11 #include <mono/metadata/debug-helpers.h>
12 #include <mono/metadata/mempool.h>
13 #include <mono/metadata/mempool-internals.h>
19 //#define DEBUG_DOMINATORS
22 * bb->dfn == 0 means either the bblock is ignored by the dfn calculation, or
23 * it is the entry bblock.
25 #define HAS_DFN(bb, entry) ((bb)->dfn || ((bb) == entry))
28 * Compute dominators and immediate dominators using the algorithm in the
29 * paper "A Simple, Fast Dominance Algorithm" by Keith D. Cooper,
30 * Timothy J. Harvey, and Ken Kennedy:
31 * http://citeseer.ist.psu.edu/cooper01simple.html
34 compute_dominators (MonoCompile
*cfg
)
36 int bindex
, i
, bitsize
;
38 MonoBasicBlock
*entry
;
39 MonoBasicBlock
**doms
;
42 g_assert (!(cfg
->comp_done
& MONO_COMP_DOM
));
44 bitsize
= mono_bitset_alloc_size (cfg
->num_bblocks
, 0);
46 mem
= mono_mempool_alloc0 (cfg
->mempool
, bitsize
* cfg
->num_bblocks
);
48 entry
= cfg
->bblocks
[0];
50 doms
= g_new0 (MonoBasicBlock
*, cfg
->num_bblocks
);
51 doms
[entry
->dfn
] = entry
;
53 #ifdef DEBUG_DOMINATORS
54 for (i
= 0; i
< cfg
->num_bblocks
; ++i
) {
55 MonoBasicBlock
*bb
= cfg
->bblocks
[i
];
57 printf ("BB%d IN: ", bb
->block_num
);
58 for (j
= 0; j
< bb
->in_count
; ++j
)
59 printf ("%d ", bb
->in_bb
[j
]->block_num
);
68 for (bindex
= 0; bindex
< cfg
->num_bblocks
; ++bindex
) {
69 MonoBasicBlock
*bb
= cfg
->bblocks
[bindex
];
73 for (i
= 0; i
< bb
->in_count
; ++i
) {
74 MonoBasicBlock
*in_bb
= bb
->in_bb
[i
];
75 if ((in_bb
!= bb
) && doms
[in_bb
->dfn
]) {
80 if (bb
!= cfg
->bblocks
[0])
83 while (i
< bb
->in_count
) {
84 MonoBasicBlock
*in_bb
= bb
->in_bb
[i
];
86 if (HAS_DFN (in_bb
, entry
) && doms
[in_bb
->dfn
]) {
88 MonoBasicBlock
*f1
= idom
;
89 MonoBasicBlock
*f2
= in_bb
;
92 if (f1
->dfn
< f2
->dfn
)
103 if (idom
!= doms
[bb
->dfn
]) {
104 if (bb
== cfg
->bblocks
[0])
107 doms
[bb
->dfn
] = idom
;
111 //printf ("A: bb=%d dfn=%d dom:%d\n", bb->block_num, bb->dfn, doms [bb->dfn]->block_num);
116 /* Compute bb->dominators for each bblock */
117 for (i
= 0; i
< cfg
->num_bblocks
; ++i
) {
118 MonoBasicBlock
*bb
= cfg
->bblocks
[i
];
120 MonoBitSet
*dominators
;
122 bb
->dominators
= dominators
= mono_bitset_mem_new (mem
, cfg
->num_bblocks
, 0);
125 mono_bitset_set_fast (dominators
, bb
->dfn
);
128 for (cbb
= doms
[bb
->dfn
]; cbb
->dfn
; cbb
= doms
[cbb
->dfn
])
129 mono_bitset_set_fast (dominators
, cbb
->dfn
);
131 bb
->idom
= doms
[bb
->dfn
];
133 bb
->idom
->dominated
= g_slist_prepend_mempool (cfg
->mempool
, bb
->idom
->dominated
, bb
);
137 mono_bitset_set_fast (dominators
, 0);
142 cfg
->comp_done
|= MONO_COMP_DOM
| MONO_COMP_IDOM
;
144 #ifdef DEBUG_DOMINATORS
145 printf ("DTREE %s %d\n", mono_method_full_name (cfg
->method
, TRUE
),
146 mono_method_get_header (cfg
->method
)->num_clauses
);
147 for (i
= 0; i
< cfg
->num_bblocks
; ++i
) {
148 MonoBasicBlock
*bb
= cfg
->bblocks
[i
];
149 printf ("BB%d(dfn=%d) (IDOM=BB%d): ", bb
->block_num
, bb
->dfn
, bb
->idom
? bb
->idom
->block_num
: -1);
150 mono_blockset_print (cfg
, bb
->dominators
, NULL
, -1);
158 check_dominance_frontier (MonoBasicBlock
*x
, MonoBasicBlock
*t
)
162 t
->flags
|= BB_VISITED
;
164 if (mono_bitset_test_fast (t
->dominators
, x
->dfn
)) {
165 for (i
= 0; i
< t
->out_count
; ++i
) {
166 if (!(t
->flags
& BB_VISITED
)) {
168 check_dominance_frontier (x
, t
->out_bb
[i
]);
170 for (j
= 0; j
< t
->out_bb
[i
]->in_count
; j
++) {
171 if (t
->out_bb
[i
]->in_bb
[j
] == t
)
178 if (!mono_bitset_test_fast (x
->dfrontier
, t
->dfn
)) {
179 printf ("BB%d not in frontier of BB%d\n", t
->block_num
, x
->block_num
);
180 g_assert_not_reached ();
188 * Compute dominance frontiers using the algorithm from the same paper.
191 compute_dominance_frontier (MonoCompile
*cfg
)
196 g_assert (!(cfg
->comp_done
& MONO_COMP_DFRONTIER
));
198 for (i
= 0; i
< cfg
->num_bblocks
; ++i
)
199 cfg
->bblocks
[i
]->flags
&= ~BB_VISITED
;
201 bitsize
= mono_bitset_alloc_size (cfg
->num_bblocks
, 0);
202 mem
= mono_mempool_alloc0 (cfg
->mempool
, bitsize
* cfg
->num_bblocks
);
204 for (i
= 0; i
< cfg
->num_bblocks
; ++i
) {
205 MonoBasicBlock
*bb
= cfg
->bblocks
[i
];
206 bb
->dfrontier
= mono_bitset_mem_new (mem
, cfg
->num_bblocks
, 0);
210 for (i
= 0; i
< cfg
->num_bblocks
; ++i
) {
211 MonoBasicBlock
*bb
= cfg
->bblocks
[i
];
213 if (bb
->in_count
> 1) {
214 for (j
= 0; j
< bb
->in_count
; ++j
) {
215 MonoBasicBlock
*p
= bb
->in_bb
[j
];
217 if (p
->dfn
|| (p
== cfg
->bblocks
[0])) {
218 while (p
!= bb
->idom
) {
219 mono_bitset_set_fast (p
->dfrontier
, bb
->dfn
);
228 for (i
= 0; i
< cfg
->num_bblocks
; ++i
) {
229 MonoBasicBlock
*bb
= cfg
->bblocks
[i
];
231 printf ("DFRONT %s BB%d: ", mono_method_full_name (cfg
->method
, TRUE
), bb
->block_num
);
232 mono_blockset_print (cfg
, bb
->dfrontier
, NULL
, -1);
237 /* this is a check for the dominator frontier */
238 for (i
= 0; i
< m
->num_bblocks
; ++i
) {
239 MonoBasicBlock
*x
= m
->bblocks
[i
];
241 mono_bitset_foreach_bit ((x
->dfrontier
), j
, (m
->num_bblocks
)) {
242 MonoBasicBlock
*w
= m
->bblocks
[j
];
244 /* x must not strictly dominates w */
245 if (mono_bitset_test_fast (w
->dominators
, x
->dfn
) && w
!= x
)
246 g_assert_not_reached ();
248 for (k
= 0; k
< m
->num_bblocks
; ++k
)
249 m
->bblocks
[k
]->flags
&= ~BB_VISITED
;
251 check_dominance_frontier (x
, x
);
256 cfg
->comp_done
|= MONO_COMP_DFRONTIER
;
260 df_set (MonoCompile
*m
, MonoBitSet
* dest
, MonoBitSet
*set
)
264 mono_bitset_foreach_bit (set
, i
, m
->num_bblocks
) {
265 mono_bitset_union_fast (dest
, m
->bblocks
[i
]->dfrontier
);
270 mono_compile_iterated_dfrontier (MonoCompile
*m
, MonoBitSet
*set
)
273 int bitsize
, count1
, count2
;
275 bitsize
= mono_bitset_alloc_size (m
->num_bblocks
, 0);
276 result
= mono_bitset_mem_new (mono_mempool_alloc0 (m
->mempool
, bitsize
), m
->num_bblocks
, 0);
278 df_set (m
, result
, set
);
279 count2
= mono_bitset_count (result
);
282 df_set (m
, result
, result
);
283 count2
= mono_bitset_count (result
);
284 } while (count2
> count1
);
290 mono_compile_dominator_info (MonoCompile
*cfg
, int dom_flags
)
292 if ((dom_flags
& MONO_COMP_DOM
) && !(cfg
->comp_done
& MONO_COMP_DOM
))
293 compute_dominators (cfg
);
294 if ((dom_flags
& MONO_COMP_DFRONTIER
) && !(cfg
->comp_done
& MONO_COMP_DFRONTIER
))
295 compute_dominance_frontier (cfg
);
298 //#define DEBUG_NATURAL_LOOPS
301 * code to detect loops and loop nesting level
304 mono_compute_natural_loops (MonoCompile
*cfg
)
307 MonoBitSet
*in_loop_blocks
;
310 g_assert (!(cfg
->comp_done
& MONO_COMP_LOOPS
));
312 in_loop_blocks
= mono_bitset_new (cfg
->num_bblocks
+ 1, 0);
313 for (i
= 0; i
< cfg
->num_bblocks
; ++i
) {
314 MonoBasicBlock
*n
= cfg
->bblocks
[i
];
316 for (j
= 0; j
< n
->out_count
; j
++) {
317 MonoBasicBlock
*h
= n
->out_bb
[j
];
318 /* check for back-edge from n to h */
319 if (n
!= h
&& mono_bitset_test_fast (n
->dominators
, h
->dfn
)) {
322 /* already in loop_blocks? */
323 if (h
->loop_blocks
&& g_list_find (h
->loop_blocks
, n
)) {
327 mono_bitset_clear_all (in_loop_blocks
);
328 if (h
->loop_blocks
) {
331 for (l
= h
->loop_blocks
; l
; l
= l
->next
) {
332 MonoBasicBlock
*b
= l
->data
;
334 mono_bitset_set_fast (in_loop_blocks
, b
->dfn
);
337 todo
= g_slist_prepend (NULL
, n
);
340 MonoBasicBlock
*cb
= (MonoBasicBlock
*)todo
->data
;
341 todo
= g_slist_delete_link (todo
, todo
);
343 if ((cb
->dfn
&& mono_bitset_test_fast (in_loop_blocks
, cb
->dfn
)) || (!cb
->dfn
&& g_list_find (h
->loop_blocks
, cb
)))
346 h
->loop_blocks
= g_list_prepend_mempool (cfg
->mempool
, h
->loop_blocks
, cb
);
349 mono_bitset_set_fast (in_loop_blocks
, cb
->dfn
);
351 for (k
= 0; k
< cb
->in_count
; k
++) {
352 MonoBasicBlock
*prev
= cb
->in_bb
[k
];
353 /* add all previous blocks */
354 if (prev
!= h
&& !((prev
->dfn
&& mono_bitset_test_fast (in_loop_blocks
, prev
->dfn
)) || (!prev
->dfn
&& g_list_find (h
->loop_blocks
, prev
)))) {
355 todo
= g_slist_prepend (todo
, prev
);
360 /* add the header if not already there */
361 if (!((h
->dfn
&& mono_bitset_test_fast (in_loop_blocks
, h
->dfn
)) || (!h
->dfn
&& g_list_find (h
->loop_blocks
, h
)))) {
362 h
->loop_blocks
= g_list_prepend_mempool (cfg
->mempool
, h
->loop_blocks
, h
);
368 mono_bitset_free (in_loop_blocks
);
370 cfg
->comp_done
|= MONO_COMP_LOOPS
;
372 /* Compute loop_body_start for each loop */
373 bb_indexes
= g_new0 (int, cfg
->num_bblocks
);
377 for (i
= 0, bb
= cfg
->bb_entry
; bb
; i
++, bb
= bb
->next_bb
) {
379 bb_indexes
[bb
->dfn
] = i
;
382 for (i
= 0; i
< cfg
->num_bblocks
; ++i
) {
383 if (cfg
->bblocks
[i
]->loop_blocks
) {
384 /* The loop body start is the first bblock in the order they will be emitted */
385 MonoBasicBlock
*h
= cfg
->bblocks
[i
];
386 MonoBasicBlock
*body_start
= h
;
389 for (l
= h
->loop_blocks
; l
; l
= l
->next
) {
390 MonoBasicBlock
*cb
= (MonoBasicBlock
*)l
->data
;
392 if (cb
->dfn
&& bb_indexes
[cb
->dfn
] < bb_indexes
[body_start
->dfn
]) {
397 body_start
->loop_body_start
= 1;
402 #ifdef DEBUG_NATURAL_LOOPS
403 for (i
= 0; i
< cfg
->num_bblocks
; ++i
) {
404 if (cfg
->bblocks
[i
]->loop_blocks
) {
405 MonoBasicBlock
*h
= (MonoBasicBlock
*)cfg
->bblocks
[i
]->loop_blocks
->data
;
407 printf ("LOOP START %d\n", h
->block_num
);
408 for (l
= h
->loop_blocks
; l
; l
= l
->next
) {
409 MonoBasicBlock
*cb
= (MonoBasicBlock
*)l
->data
;
410 printf (" BB%d %d %p\n", cb
->block_num
, cb
->nesting
, cb
->loop_blocks
);
419 clear_idominators (MonoCompile
*cfg
)
423 for (i
= 0; i
< cfg
->num_bblocks
; ++i
) {
424 if (cfg
->bblocks
[i
]->dominated
) {
425 cfg
->bblocks
[i
]->dominated
= NULL
;
429 cfg
->comp_done
&= ~MONO_COMP_IDOM
;
433 clear_loops (MonoCompile
*cfg
)
437 for (i
= 0; i
< cfg
->num_bblocks
; ++i
) {
438 cfg
->bblocks
[i
]->nesting
= 0;
439 cfg
->bblocks
[i
]->loop_blocks
= NULL
;
442 cfg
->comp_done
&= ~MONO_COMP_LOOPS
;
446 mono_free_loop_info (MonoCompile
*cfg
)
448 if (cfg
->comp_done
& MONO_COMP_IDOM
)
449 clear_idominators (cfg
);
450 if (cfg
->comp_done
& MONO_COMP_LOOPS
)
454 #else /* DISABLE_JIT */
457 mono_free_loop_info (MonoCompile
*cfg
)
461 #endif /* DISABLE_JIT */