00001
00002
00003
00004
00005
00006
00007
00008
00009
00010
00011
00012
00013
00014
00015
00016
00017
00018
00019
00020
00021
00022
00023
00024
00025
00026
00027
00028
00029
00030
00031
00032
00033
00034
00035
00036 #include "config.h"
00037 #include "system.h"
00038 #include "coretypes.h"
00039 #include "tm.h"
00040 #include "rtl.h"
00041 #include "hard-reg-set.h"
00042 #include "obstack.h"
00043 #include "basic-block.h"
00044 #include "toplev.h"
00045 #include "et-forest.h"
00046 #include "timevar.h"
00047
00048
00049 enum dom_state dom_computed[2];
00050
00051
00052
00053
00054
00055
00056
00057
00058 typedef unsigned int TBB;
00059
00060
00061
00062
00063
00064
00065 struct dom_info
00066 {
00067
00068 TBB *dfs_parent;
00069
00070
00071
00072 TBB *key;
00073
00074
00075 TBB *path_min;
00076
00077 TBB *bucket;
00078
00079 TBB *next_bucket;
00080
00081
00082 TBB *dom;
00083
00084
00085
00086
00087
00088 TBB *set_chain;
00089
00090 unsigned int *set_size;
00091
00092
00093 TBB *set_child;
00094
00095
00096
00097
00098 TBB *dfs_order;
00099
00100
00101
00102
00103 basic_block *dfs_to_bb;
00104
00105
00106 unsigned int dfsnum;
00107
00108 unsigned int nodes;
00109
00110
00111
00112 bitmap fake_exit_edge;
00113 };
00114
00115 static void init_dom_info (struct dom_info *, enum cdi_direction);
00116 static void free_dom_info (struct dom_info *);
00117 static void calc_dfs_tree_nonrec (struct dom_info *, basic_block,
00118 enum cdi_direction);
00119 static void calc_dfs_tree (struct dom_info *, enum cdi_direction);
00120 static void compress (struct dom_info *, TBB);
00121 static TBB eval (struct dom_info *, TBB);
00122 static void link_roots (struct dom_info *, TBB, TBB);
00123 static void calc_idoms (struct dom_info *, enum cdi_direction);
00124 void debug_dominance_info (enum cdi_direction);
00125
00126
00127 static unsigned n_bbs_in_dom_tree[2];
00128
00129
00130
00131 #define init_ar(var, type, num, content) \
00132 do \
00133 { \
00134 unsigned int i = 1; \
00135 if (! (content)) \
00136 (var) = XCNEWVEC (type, num); \
00137 else \
00138 { \
00139 (var) = XNEWVEC (type, (num)); \
00140 for (i = 0; i < num; i++) \
00141 (var)[i] = (content); \
00142 } \
00143 } \
00144 while (0)
00145
00146
00147
00148
00149 static void
00150 init_dom_info (struct dom_info *di, enum cdi_direction dir)
00151 {
00152 unsigned int num = n_basic_blocks;
00153 init_ar (di->dfs_parent, TBB, num, 0);
00154 init_ar (di->path_min, TBB, num, i);
00155 init_ar (di->key, TBB, num, i);
00156 init_ar (di->dom, TBB, num, 0);
00157
00158 init_ar (di->bucket, TBB, num, 0);
00159 init_ar (di->next_bucket, TBB, num, 0);
00160
00161 init_ar (di->set_chain, TBB, num, 0);
00162 init_ar (di->set_size, unsigned int, num, 1);
00163 init_ar (di->set_child, TBB, num, 0);
00164
00165 init_ar (di->dfs_order, TBB, (unsigned int) last_basic_block + 1, 0);
00166 init_ar (di->dfs_to_bb, basic_block, num, 0);
00167
00168 di->dfsnum = 1;
00169 di->nodes = 0;
00170
00171 di->fake_exit_edge = dir ? BITMAP_ALLOC (NULL) : NULL;
00172 }
00173
00174 #undef init_ar
00175
00176
00177
00178 static void
00179 free_dom_info (struct dom_info *di)
00180 {
00181 free (di->dfs_parent);
00182 free (di->path_min);
00183 free (di->key);
00184 free (di->dom);
00185 free (di->bucket);
00186 free (di->next_bucket);
00187 free (di->set_chain);
00188 free (di->set_size);
00189 free (di->set_child);
00190 free (di->dfs_order);
00191 free (di->dfs_to_bb);
00192 BITMAP_FREE (di->fake_exit_edge);
00193 }
00194
00195
00196
00197
00198
00199
00200
00201 static void
00202 calc_dfs_tree_nonrec (struct dom_info *di, basic_block bb,
00203 enum cdi_direction reverse)
00204 {
00205
00206 edge e;
00207 TBB child_i, my_i = 0;
00208 edge_iterator *stack;
00209 edge_iterator ei, einext;
00210 int sp;
00211
00212
00213 basic_block en_block;
00214
00215 basic_block ex_block;
00216
00217 stack = XNEWVEC (edge_iterator, n_basic_blocks + 1);
00218 sp = 0;
00219
00220
00221 if (reverse)
00222 {
00223 ei = ei_start (bb->preds);
00224 en_block = EXIT_BLOCK_PTR;
00225 ex_block = ENTRY_BLOCK_PTR;
00226 }
00227 else
00228 {
00229 ei = ei_start (bb->succs);
00230 en_block = ENTRY_BLOCK_PTR;
00231 ex_block = EXIT_BLOCK_PTR;
00232 }
00233
00234
00235 while (1)
00236 {
00237 basic_block bn;
00238
00239
00240
00241 while (!ei_end_p (ei))
00242 {
00243 e = ei_edge (ei);
00244
00245
00246
00247 if (reverse)
00248 {
00249 bn = e->src;
00250
00251
00252
00253
00254 if (bn == ex_block || di->dfs_order[bn->index])
00255 {
00256 ei_next (&ei);
00257 continue;
00258 }
00259 bb = e->dest;
00260 einext = ei_start (bn->preds);
00261 }
00262 else
00263 {
00264 bn = e->dest;
00265 if (bn == ex_block || di->dfs_order[bn->index])
00266 {
00267 ei_next (&ei);
00268 continue;
00269 }
00270 bb = e->src;
00271 einext = ei_start (bn->succs);
00272 }
00273
00274 gcc_assert (bn != en_block);
00275
00276
00277 if (bb != en_block)
00278 my_i = di->dfs_order[bb->index];
00279 else
00280 my_i = di->dfs_order[last_basic_block];
00281 child_i = di->dfs_order[bn->index] = di->dfsnum++;
00282 di->dfs_to_bb[child_i] = bn;
00283 di->dfs_parent[child_i] = my_i;
00284
00285
00286 stack[sp++] = ei;
00287 ei = einext;
00288 }
00289
00290 if (!sp)
00291 break;
00292 ei = stack[--sp];
00293
00294
00295
00296
00297
00298
00299
00300
00301
00302
00303 ei_next (&ei);
00304 }
00305 free (stack);
00306 }
00307
00308
00309
00310
00311
00312
00313 static void
00314 calc_dfs_tree (struct dom_info *di, enum cdi_direction reverse)
00315 {
00316
00317 basic_block begin = reverse ? EXIT_BLOCK_PTR : ENTRY_BLOCK_PTR;
00318 di->dfs_order[last_basic_block] = di->dfsnum;
00319 di->dfs_to_bb[di->dfsnum] = begin;
00320 di->dfsnum++;
00321
00322 calc_dfs_tree_nonrec (di, begin, reverse);
00323
00324 if (reverse)
00325 {
00326
00327
00328
00329
00330
00331
00332
00333
00334
00335
00336 basic_block b;
00337 bool saw_unconnected = false;
00338
00339 FOR_EACH_BB_REVERSE (b)
00340 {
00341 if (EDGE_COUNT (b->succs) > 0)
00342 {
00343 if (di->dfs_order[b->index] == 0)
00344 saw_unconnected = true;
00345 continue;
00346 }
00347 bitmap_set_bit (di->fake_exit_edge, b->index);
00348 di->dfs_order[b->index] = di->dfsnum;
00349 di->dfs_to_bb[di->dfsnum] = b;
00350 di->dfs_parent[di->dfsnum] = di->dfs_order[last_basic_block];
00351 di->dfsnum++;
00352 calc_dfs_tree_nonrec (di, b, reverse);
00353 }
00354
00355 if (saw_unconnected)
00356 {
00357 FOR_EACH_BB_REVERSE (b)
00358 {
00359 if (di->dfs_order[b->index])
00360 continue;
00361 bitmap_set_bit (di->fake_exit_edge, b->index);
00362 di->dfs_order[b->index] = di->dfsnum;
00363 di->dfs_to_bb[di->dfsnum] = b;
00364 di->dfs_parent[di->dfsnum] = di->dfs_order[last_basic_block];
00365 di->dfsnum++;
00366 calc_dfs_tree_nonrec (di, b, reverse);
00367 }
00368 }
00369 }
00370
00371 di->nodes = di->dfsnum - 1;
00372
00373
00374 gcc_assert (di->nodes == (unsigned int) n_basic_blocks - 1);
00375 }
00376
00377
00378
00379
00380
00381
00382 static void
00383 compress (struct dom_info *di, TBB v)
00384 {
00385
00386
00387
00388 TBB parent = di->set_chain[v];
00389 if (di->set_chain[parent])
00390 {
00391 compress (di, parent);
00392 if (di->key[di->path_min[parent]] < di->key[di->path_min[v]])
00393 di->path_min[v] = di->path_min[parent];
00394 di->set_chain[v] = di->set_chain[parent];
00395 }
00396 }
00397
00398
00399
00400
00401
00402 static inline TBB
00403 eval (struct dom_info *di, TBB v)
00404 {
00405
00406
00407 TBB rep = di->set_chain[v];
00408
00409
00410 if (!rep)
00411 return di->path_min[v];
00412
00413
00414 if (di->set_chain[rep])
00415 {
00416 compress (di, v);
00417 rep = di->set_chain[v];
00418 }
00419
00420 if (di->key[di->path_min[rep]] >= di->key[di->path_min[v]])
00421 return di->path_min[v];
00422 else
00423 return di->path_min[rep];
00424 }
00425
00426
00427
00428
00429
00430
00431 static void
00432 link_roots (struct dom_info *di, TBB v, TBB w)
00433 {
00434 TBB s = w;
00435
00436
00437 while (di->key[di->path_min[w]] < di->key[di->path_min[di->set_child[s]]])
00438 {
00439 if (di->set_size[s] + di->set_size[di->set_child[di->set_child[s]]]
00440 >= 2 * di->set_size[di->set_child[s]])
00441 {
00442 di->set_chain[di->set_child[s]] = s;
00443 di->set_child[s] = di->set_child[di->set_child[s]];
00444 }
00445 else
00446 {
00447 di->set_size[di->set_child[s]] = di->set_size[s];
00448 s = di->set_chain[s] = di->set_child[s];
00449 }
00450 }
00451
00452 di->path_min[s] = di->path_min[w];
00453 di->set_size[v] += di->set_size[w];
00454 if (di->set_size[v] < 2 * di->set_size[w])
00455 {
00456 TBB tmp = s;
00457 s = di->set_child[v];
00458 di->set_child[v] = tmp;
00459 }
00460
00461
00462 while (s)
00463 {
00464 di->set_chain[s] = v;
00465 s = di->set_child[s];
00466 }
00467 }
00468
00469
00470
00471
00472
00473 static void
00474 calc_idoms (struct dom_info *di, enum cdi_direction reverse)
00475 {
00476 TBB v, w, k, par;
00477 basic_block en_block;
00478 edge_iterator ei, einext;
00479
00480 if (reverse)
00481 en_block = EXIT_BLOCK_PTR;
00482 else
00483 en_block = ENTRY_BLOCK_PTR;
00484
00485
00486 v = di->nodes;
00487 while (v > 1)
00488 {
00489 basic_block bb = di->dfs_to_bb[v];
00490 edge e;
00491
00492 par = di->dfs_parent[v];
00493 k = v;
00494
00495 ei = (reverse) ? ei_start (bb->succs) : ei_start (bb->preds);
00496
00497 if (reverse)
00498 {
00499
00500 if (bitmap_bit_p (di->fake_exit_edge, bb->index))
00501 {
00502 einext = ei;
00503 einext.index = 0;
00504 goto do_fake_exit_edge;
00505 }
00506 }
00507
00508
00509
00510
00511
00512 while (!ei_end_p (ei))
00513 {
00514 TBB k1;
00515 basic_block b;
00516
00517 e = ei_edge (ei);
00518 b = (reverse) ? e->dest : e->src;
00519 einext = ei;
00520 ei_next (&einext);
00521
00522 if (b == en_block)
00523 {
00524 do_fake_exit_edge:
00525 k1 = di->dfs_order[last_basic_block];
00526 }
00527 else
00528 k1 = di->dfs_order[b->index];
00529
00530
00531
00532 if (k1 > v)
00533 k1 = di->key[eval (di, k1)];
00534 if (k1 < k)
00535 k = k1;
00536
00537 ei = einext;
00538 }
00539
00540 di->key[v] = k;
00541 link_roots (di, par, v);
00542 di->next_bucket[v] = di->bucket[k];
00543 di->bucket[k] = v;
00544
00545
00546 for (w = di->bucket[par]; w; w = di->next_bucket[w])
00547 {
00548 k = eval (di, w);
00549 if (di->key[k] < di->key[w])
00550 di->dom[w] = k;
00551 else
00552 di->dom[w] = par;
00553 }
00554
00555 di->bucket[par] = 0;
00556 v--;
00557 }
00558
00559
00560 di->dom[1] = 0;
00561 for (v = 2; v <= di->nodes; v++)
00562 if (di->dom[v] != di->key[v])
00563 di->dom[v] = di->dom[di->dom[v]];
00564 }
00565
00566
00567
00568 static void
00569 assign_dfs_numbers (struct et_node *node, int *num)
00570 {
00571 struct et_node *son;
00572
00573 node->dfs_num_in = (*num)++;
00574
00575 if (node->son)
00576 {
00577 assign_dfs_numbers (node->son, num);
00578 for (son = node->son->right; son != node->son; son = son->right)
00579 assign_dfs_numbers (son, num);
00580 }
00581
00582 node->dfs_num_out = (*num)++;
00583 }
00584
00585
00586
00587
00588 static void
00589 compute_dom_fast_query (enum cdi_direction dir)
00590 {
00591 int num = 0;
00592 basic_block bb;
00593
00594 gcc_assert (dom_info_available_p (dir));
00595
00596 if (dom_computed[dir] == DOM_OK)
00597 return;
00598
00599 FOR_ALL_BB (bb)
00600 {
00601 if (!bb->dom[dir]->father)
00602 assign_dfs_numbers (bb->dom[dir], &num);
00603 }
00604
00605 dom_computed[dir] = DOM_OK;
00606 }
00607
00608
00609
00610
00611 void
00612 calculate_dominance_info (enum cdi_direction dir)
00613 {
00614 struct dom_info di;
00615 basic_block b;
00616
00617 if (dom_computed[dir] == DOM_OK)
00618 return;
00619
00620 timevar_push (TV_DOMINANCE);
00621 if (!dom_info_available_p (dir))
00622 {
00623 gcc_assert (!n_bbs_in_dom_tree[dir]);
00624
00625 FOR_ALL_BB (b)
00626 {
00627 b->dom[dir] = et_new_tree (b);
00628 }
00629 n_bbs_in_dom_tree[dir] = n_basic_blocks;
00630
00631 init_dom_info (&di, dir);
00632 calc_dfs_tree (&di, dir);
00633 calc_idoms (&di, dir);
00634
00635 FOR_EACH_BB (b)
00636 {
00637 TBB d = di.dom[di.dfs_order[b->index]];
00638
00639 if (di.dfs_to_bb[d])
00640 et_set_father (b->dom[dir], di.dfs_to_bb[d]->dom[dir]);
00641 }
00642
00643 free_dom_info (&di);
00644 dom_computed[dir] = DOM_NO_FAST_QUERY;
00645 }
00646
00647 compute_dom_fast_query (dir);
00648
00649 timevar_pop (TV_DOMINANCE);
00650 }
00651
00652
00653 void
00654 free_dominance_info (enum cdi_direction dir)
00655 {
00656 basic_block bb;
00657
00658 if (!dom_info_available_p (dir))
00659 return;
00660
00661 FOR_ALL_BB (bb)
00662 {
00663 et_free_tree_force (bb->dom[dir]);
00664 bb->dom[dir] = NULL;
00665 }
00666 et_free_pools ();
00667
00668 n_bbs_in_dom_tree[dir] = 0;
00669
00670 dom_computed[dir] = DOM_NONE;
00671 }
00672
00673
00674 basic_block
00675 get_immediate_dominator (enum cdi_direction dir, basic_block bb)
00676 {
00677 struct et_node *node = bb->dom[dir];
00678
00679 gcc_assert (dom_computed[dir]);
00680
00681 if (!node->father)
00682 return NULL;
00683
00684 return node->father->data;
00685 }
00686
00687
00688
00689 inline void
00690 set_immediate_dominator (enum cdi_direction dir, basic_block bb,
00691 basic_block dominated_by)
00692 {
00693 struct et_node *node = bb->dom[dir];
00694
00695 gcc_assert (dom_computed[dir]);
00696
00697 if (node->father)
00698 {
00699 if (node->father->data == dominated_by)
00700 return;
00701 et_split (node);
00702 }
00703
00704 if (dominated_by)
00705 et_set_father (node, dominated_by->dom[dir]);
00706
00707 if (dom_computed[dir] == DOM_OK)
00708 dom_computed[dir] = DOM_NO_FAST_QUERY;
00709 }
00710
00711
00712
00713 int
00714 get_dominated_by (enum cdi_direction dir, basic_block bb, basic_block **bbs)
00715 {
00716 int n;
00717 struct et_node *node = bb->dom[dir], *son = node->son, *ason;
00718
00719 gcc_assert (dom_computed[dir]);
00720
00721 if (!son)
00722 {
00723 *bbs = NULL;
00724 return 0;
00725 }
00726
00727 for (ason = son->right, n = 1; ason != son; ason = ason->right)
00728 n++;
00729
00730 *bbs = XNEWVEC (basic_block, n);
00731 (*bbs)[0] = son->data;
00732 for (ason = son->right, n = 1; ason != son; ason = ason->right)
00733 (*bbs)[n++] = ason->data;
00734
00735 return n;
00736 }
00737
00738
00739
00740
00741
00742
00743 unsigned
00744 get_dominated_by_region (enum cdi_direction dir, basic_block *region,
00745 unsigned n_region, basic_block *doms)
00746 {
00747 unsigned n_doms = 0, i;
00748 basic_block dom;
00749
00750 for (i = 0; i < n_region; i++)
00751 region[i]->flags |= BB_DUPLICATED;
00752 for (i = 0; i < n_region; i++)
00753 for (dom = first_dom_son (dir, region[i]);
00754 dom;
00755 dom = next_dom_son (dir, dom))
00756 if (!(dom->flags & BB_DUPLICATED))
00757 doms[n_doms++] = dom;
00758 for (i = 0; i < n_region; i++)
00759 region[i]->flags &= ~BB_DUPLICATED;
00760
00761 return n_doms;
00762 }
00763
00764
00765 void
00766 redirect_immediate_dominators (enum cdi_direction dir, basic_block bb,
00767 basic_block to)
00768 {
00769 struct et_node *bb_node = bb->dom[dir], *to_node = to->dom[dir], *son;
00770
00771 gcc_assert (dom_computed[dir]);
00772
00773 if (!bb_node->son)
00774 return;
00775
00776 while (bb_node->son)
00777 {
00778 son = bb_node->son;
00779
00780 et_split (son);
00781 et_set_father (son, to_node);
00782 }
00783
00784 if (dom_computed[dir] == DOM_OK)
00785 dom_computed[dir] = DOM_NO_FAST_QUERY;
00786 }
00787
00788
00789 basic_block
00790 nearest_common_dominator (enum cdi_direction dir, basic_block bb1, basic_block bb2)
00791 {
00792 gcc_assert (dom_computed[dir]);
00793
00794 if (!bb1)
00795 return bb2;
00796 if (!bb2)
00797 return bb1;
00798
00799 return et_nca (bb1->dom[dir], bb2->dom[dir])->data;
00800 }
00801
00802
00803
00804
00805
00806 basic_block
00807 nearest_common_dominator_for_set (enum cdi_direction dir, bitmap blocks)
00808 {
00809 unsigned i, first;
00810 bitmap_iterator bi;
00811 basic_block dom;
00812
00813 first = bitmap_first_set_bit (blocks);
00814 dom = BASIC_BLOCK (first);
00815 EXECUTE_IF_SET_IN_BITMAP (blocks, 0, i, bi)
00816 if (dom != BASIC_BLOCK (i))
00817 dom = nearest_common_dominator (dir, dom, BASIC_BLOCK (i));
00818
00819 return dom;
00820 }
00821
00822
00823
00824
00825
00826
00827
00828
00829
00830
00831
00832
00833
00834
00835
00836
00837
00838
00839
00840
00841
00842
00843
00844
00845
00846
00847
00848
00849
00850
00851
00852
00853
00854
00855
00856
00857
00858
00859
00860
00861
00862
00863
00864
00865
00866
00867
00868
00869
00870
00871
00872
00873
00874
00875
00876
00877
00878
00879
00880
00881
00882
00883
00884
00885
00886
00887
00888
00889
00890
00891
00892
00893
00894
00895
00896
00897
00898 bool
00899 dominated_by_p (enum cdi_direction dir, basic_block bb1, basic_block bb2)
00900 {
00901 struct et_node *n1 = bb1->dom[dir], *n2 = bb2->dom[dir];
00902
00903 gcc_assert (dom_computed[dir]);
00904
00905 if (dom_computed[dir] == DOM_OK)
00906 return (n1->dfs_num_in >= n2->dfs_num_in
00907 && n1->dfs_num_out <= n2->dfs_num_out);
00908
00909 return et_below (n1, n2);
00910 }
00911
00912
00913
00914 unsigned
00915 bb_dom_dfs_in (enum cdi_direction dir, basic_block bb)
00916 {
00917 struct et_node *n = bb->dom[dir];
00918
00919 gcc_assert (dom_computed[dir] == DOM_OK);
00920 return n->dfs_num_in;
00921 }
00922
00923
00924
00925 unsigned
00926 bb_dom_dfs_out (enum cdi_direction dir, basic_block bb)
00927 {
00928 struct et_node *n = bb->dom[dir];
00929
00930 gcc_assert (dom_computed[dir] == DOM_OK);
00931 return n->dfs_num_out;
00932 }
00933
00934
00935 void
00936 verify_dominators (enum cdi_direction dir)
00937 {
00938 int err = 0;
00939 basic_block bb;
00940
00941 gcc_assert (dom_info_available_p (dir));
00942
00943 FOR_EACH_BB (bb)
00944 {
00945 basic_block dom_bb;
00946 basic_block imm_bb;
00947
00948 dom_bb = recount_dominator (dir, bb);
00949 imm_bb = get_immediate_dominator (dir, bb);
00950 if (dom_bb != imm_bb)
00951 {
00952 if ((dom_bb == NULL) || (imm_bb == NULL))
00953 error ("dominator of %d status unknown", bb->index);
00954 else
00955 error ("dominator of %d should be %d, not %d",
00956 bb->index, dom_bb->index, imm_bb->index);
00957 err = 1;
00958 }
00959 }
00960
00961 if (dir == CDI_DOMINATORS)
00962 {
00963 FOR_EACH_BB (bb)
00964 {
00965 if (!dominated_by_p (dir, bb, ENTRY_BLOCK_PTR))
00966 {
00967 error ("ENTRY does not dominate bb %d", bb->index);
00968 err = 1;
00969 }
00970 }
00971 }
00972
00973 gcc_assert (!err);
00974 }
00975
00976
00977
00978
00979
00980
00981 basic_block
00982 recount_dominator (enum cdi_direction dir, basic_block bb)
00983 {
00984 basic_block dom_bb = NULL;
00985 edge e;
00986 edge_iterator ei;
00987
00988 gcc_assert (dom_computed[dir]);
00989
00990 if (dir == CDI_DOMINATORS)
00991 {
00992 FOR_EACH_EDGE (e, ei, bb->preds)
00993 {
00994
00995
00996 if (!dominated_by_p (dir, e->src, ENTRY_BLOCK_PTR))
00997 continue;
00998
00999 if (!dominated_by_p (dir, e->src, bb))
01000 dom_bb = nearest_common_dominator (dir, dom_bb, e->src);
01001 }
01002 }
01003 else
01004 {
01005 FOR_EACH_EDGE (e, ei, bb->succs)
01006 {
01007 if (!dominated_by_p (dir, e->dest, bb))
01008 dom_bb = nearest_common_dominator (dir, dom_bb, e->dest);
01009 }
01010 }
01011
01012 return dom_bb;
01013 }
01014
01015
01016
01017 void
01018 iterate_fix_dominators (enum cdi_direction dir, basic_block *bbs, int n)
01019 {
01020 int i, changed = 1;
01021 basic_block old_dom, new_dom;
01022
01023 gcc_assert (dom_computed[dir]);
01024
01025 for (i = 0; i < n; i++)
01026 set_immediate_dominator (dir, bbs[i], NULL);
01027
01028 while (changed)
01029 {
01030 changed = 0;
01031 for (i = 0; i < n; i++)
01032 {
01033 old_dom = get_immediate_dominator (dir, bbs[i]);
01034 new_dom = recount_dominator (dir, bbs[i]);
01035 if (old_dom != new_dom)
01036 {
01037 changed = 1;
01038 set_immediate_dominator (dir, bbs[i], new_dom);
01039 }
01040 }
01041 }
01042
01043 for (i = 0; i < n; i++)
01044 gcc_assert (get_immediate_dominator (dir, bbs[i]));
01045 }
01046
01047 void
01048 add_to_dominance_info (enum cdi_direction dir, basic_block bb)
01049 {
01050 gcc_assert (dom_computed[dir]);
01051 gcc_assert (!bb->dom[dir]);
01052
01053 n_bbs_in_dom_tree[dir]++;
01054
01055 bb->dom[dir] = et_new_tree (bb);
01056
01057 if (dom_computed[dir] == DOM_OK)
01058 dom_computed[dir] = DOM_NO_FAST_QUERY;
01059 }
01060
01061 void
01062 delete_from_dominance_info (enum cdi_direction dir, basic_block bb)
01063 {
01064 gcc_assert (dom_computed[dir]);
01065
01066 et_free_tree (bb->dom[dir]);
01067 bb->dom[dir] = NULL;
01068 n_bbs_in_dom_tree[dir]--;
01069
01070 if (dom_computed[dir] == DOM_OK)
01071 dom_computed[dir] = DOM_NO_FAST_QUERY;
01072 }
01073
01074
01075
01076
01077 basic_block
01078 first_dom_son (enum cdi_direction dir, basic_block bb)
01079 {
01080 struct et_node *son = bb->dom[dir]->son;
01081
01082 return son ? son->data : NULL;
01083 }
01084
01085
01086
01087
01088 basic_block
01089 next_dom_son (enum cdi_direction dir, basic_block bb)
01090 {
01091 struct et_node *next = bb->dom[dir]->right;
01092
01093 return next->father->son == next ? NULL : next->data;
01094 }
01095
01096
01097
01098 bool
01099 dom_info_available_p (enum cdi_direction dir)
01100 {
01101 return dom_computed[dir] != DOM_NONE;
01102 }
01103
01104 void
01105 debug_dominance_info (enum cdi_direction dir)
01106 {
01107 basic_block bb, bb2;
01108 FOR_EACH_BB (bb)
01109 if ((bb2 = get_immediate_dominator (dir, bb)))
01110 fprintf (stderr, "%i %i\n", bb->index, bb2->index);
01111 }