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
00037
00038
00039
00040
00041
00042
00043
00044
00045
00046
00047
00048
00049
00050
00051
00052
00053
00054
00055
00056
00057
00058
00059
00060
00061
00062
00063
00064
00065 #ifdef USE_PCH
00066 #include "opt_pch.h"
00067 #endif // USE_PCH
00068 #pragma hdrstop
00069
00070
00071 #ifdef _KEEP_RCS_ID
00072 #define opt_dom_CXX "opt_dom.cxx"
00073 static char *rcs_id = opt_dom_CXX"$Revision$";
00074 #endif
00075
00076 #include "defs.h"
00077 #include "errors.h"
00078 #include "erglob.h"
00079 #include "mempool.h"
00080 #include "tracing.h"
00081 #include "cxx_memory.h"
00082
00083 #include "opt_sys.h"
00084 #include "opt_cfg.h"
00085 #include "opt_defs.h"
00086 #include "bb_node_set.h"
00087
00088
00089
00090
00091
00092
00093
00094
00095
00096 static void
00097 Dom_print_dom( BB_NODE *bb, BOOL print_dom, FILE *fp )
00098 {
00099 if ( bb != NULL ) {
00100 BOOL print_kid;
00101 BB_LIST_ITER bb_iter;
00102 BB_NODE *bb_kid;
00103
00104 fprintf( fp, "BB:%-5d", bb->Id() );
00105
00106
00107 print_kid = TRUE;
00108 FOR_ALL_ELEM(
00109 bb_kid, bb_iter, Init(print_dom?bb->Dom_bbs():bb->Pdom_bbs()) )
00110 {
00111 if ( print_kid ) {
00112 fprintf( fp, " kids: " );
00113 print_kid = FALSE;
00114 }
00115 if ( bb_kid != NULL ) {
00116 fprintf( fp, "bb:%-5d ", bb_kid->Id() );
00117 }
00118 }
00119
00120 fprintf( fp, "\n" );
00121
00122
00123 FOR_ALL_ELEM(
00124 bb_kid, bb_iter, Init(print_dom?bb->Dom_bbs():bb->Pdom_bbs()) )
00125 {
00126 Dom_print_dom( bb_kid, print_dom, fp );
00127 }
00128 }
00129 else {
00130 fprintf( fp, "<null bb_node>\n" );
00131 }
00132 }
00133
00134 static void
00135 Print_dom( const CFG *cfg, FILE *fp=stderr )
00136 {
00137 fprintf ( fp, "Print_dom\n" );
00138 Dom_print_dom( cfg->Entry_bb(), TRUE, fp );
00139 }
00140
00141 static void
00142 Print_pdom( const CFG *cfg, FILE *fp=stderr )
00143 {
00144 fprintf ( fp, "Print_pdom\n" );
00145 Dom_print_dom( cfg->Exit_bb(), FALSE, fp );
00146 }
00147
00148
00149
00150
00151
00152
00153
00154
00155
00156
00157
00158
00159
00160
00161
00162
00163
00164
00165 class DOM_INFO {
00166 private:
00167
00168 class DOM_REC {
00169 friend class DOM_INFO;
00170 private:
00171 IDTYPE parent;
00172 IDTYPE ancestor;
00173 IDTYPE child;
00174 IDTYPE vertex;
00175 IDTYPE dom;
00176 IDTYPE label;
00177 IDTYPE semi;
00178 IDTYPE size;
00179 BB_NODE_SET *bucket;
00180
00181
00182 DOM_REC(void) {}
00183 DOM_REC(const DOM_REC&);
00184 ~DOM_REC(void) {}
00185 DOM_REC &operator = (const DOM_REC&);
00186
00187 void Init( const CFG *cfg, MEM_POOL *mempool );
00188 };
00189
00190 IDTYPE counter;
00191 DOM_REC *recs;
00192
00193
00194 DOM_INFO(void);
00195 DOM_INFO(const DOM_INFO&);
00196 DOM_INFO& operator = (const DOM_INFO&);
00197
00198
00199 void eDFS( BB_NODE *bbv );
00200
00201 void xDFS( BB_NODE *bbv );
00202
00203 void Compress( IDTYPE v );
00204 IDTYPE Eval( const BB_NODE *bbv );
00205 void Link( BB_NODE *bbv, BB_NODE *bbw );
00206
00207 void Build_dom_tree( CFG *cfg );
00208 void Build_pdom_tree( const CFG *cfg );
00209
00210 public:
00211
00212 DOM_INFO( const CFG *cfg, MEM_POOL *mempool );
00213 ~DOM_INFO( void ) {}
00214
00215 IDTYPE Counter( void ) const { return counter; }
00216 IDTYPE Parent( IDTYPE n ) const { return recs[n].parent; }
00217 IDTYPE Ancestor( IDTYPE n ) const { return recs[n].ancestor; }
00218 IDTYPE Child( IDTYPE n ) const { return recs[n].child; }
00219 IDTYPE Vertex( IDTYPE n ) const { return recs[n].vertex; }
00220 IDTYPE Dom( IDTYPE n ) const { return recs[n].dom; }
00221 IDTYPE Label( IDTYPE n ) const { return recs[n].label; }
00222 IDTYPE Semi( IDTYPE n ) const { return recs[n].semi; }
00223 IDTYPE Size( IDTYPE n ) const { return recs[n].size; }
00224
00225 BB_NODE_SET *Bucket( IDTYPE n ) const { return recs[n].bucket; }
00226
00227 void Set_counter(IDTYPE c) { counter = c; }
00228 void Set_parent(IDTYPE n, IDTYPE p) { recs[n].parent = p; }
00229 void Set_ancestor(IDTYPE n, IDTYPE a) { recs[n].ancestor = a; }
00230 void Set_child(IDTYPE n, IDTYPE c) { recs[n].child = c; }
00231 void Set_vertex(IDTYPE n, IDTYPE v) { recs[n].vertex = v; }
00232 void Set_dom(IDTYPE n, IDTYPE d) { recs[n].dom = d; }
00233 void Set_label(IDTYPE n, IDTYPE l) { recs[n].label = l; }
00234 void Set_semi(IDTYPE n, IDTYPE s) { recs[n].semi = s; }
00235 void Set_size(IDTYPE n, IDTYPE s) { recs[n].size = s; }
00236 void Set_bucket(IDTYPE n, BB_NODE_SET *b){ recs[n].bucket = b; }
00237
00238
00239
00240 void Compute_dom_tree( CFG *cfg, BOOL build_dom );
00241 };
00242
00243
00244
00245
00246
00247
00248
00249
00250
00251 void
00252 DOM_INFO::DOM_REC::Init( const CFG *cfg, MEM_POOL *mempool )
00253 {
00254
00255
00256 IDTYPE bbs = cfg->Last_bb_id()+1;
00257 bucket = CXX_NEW(BB_NODE_SET(bbs,cfg,mempool,BBNS_EMPTY),mempool);
00258 }
00259
00260
00261
00262
00263
00264 DOM_INFO::DOM_INFO( const CFG *cfg, MEM_POOL *mempool )
00265 {
00266 IDTYPE bbs = cfg->Last_bb_id()+1;
00267
00268 counter = 0;
00269 recs = CXX_NEW_ARRAY(DOM_REC, bbs, mempool);
00270
00271
00272
00273
00274 BZERO( recs, bbs*sizeof(recs[0]) );
00275
00276
00277 for ( INT i = 0; i < bbs; i++ ) {
00278 recs[i].Init(cfg,mempool);
00279 }
00280 }
00281
00282
00283
00284
00285
00286
00287
00288 void
00289 DOM_INFO::eDFS( BB_NODE *bbv )
00290 {
00291 IDTYPE v = bbv->Id();
00292 BB_NODE *succ;
00293 BB_LIST_ITER bb_succ_iter;
00294 Is_True(v!=0, ("DOM_INFO::eDFS: Uninitialized BB id, Bad WHIRL input!"));
00295 counter++;
00296 Set_semi(v,counter);
00297 Set_vertex(counter,v);
00298 Set_label(v,v);
00299 Set_ancestor(v,0);
00300 Set_child(v,0);
00301 Set_size(v,1);
00302
00303
00304 bbv->Set_idom(NULL);
00305 bbv->Set_dom_bbs(NULL);
00306
00307 FOR_ALL_ELEM( succ, bb_succ_iter, Init(bbv->Succ()) ) {
00308 IDTYPE succ_id = succ->Id();
00309 if ( Semi(succ_id) == 0 ) {
00310 Set_parent(succ_id,v);
00311 eDFS( succ );
00312 }
00313 }
00314 }
00315
00316
00317
00318
00319
00320
00321
00322 void
00323 DOM_INFO::xDFS( BB_NODE *bbv )
00324 {
00325 IDTYPE v = bbv->Id();
00326 BB_NODE *pred;
00327 BB_LIST_ITER bb_pred_iter;
00328
00329 counter++;
00330 Set_semi(v,counter);
00331 Set_vertex(counter,v);
00332 Set_label(v,v);
00333 Set_ancestor(v,0);
00334 Set_child(v,0);
00335 Set_size(v,1);
00336
00337
00338 bbv->Set_ipdom(NULL);
00339 bbv->Set_pdom_bbs(NULL);
00340
00341 FOR_ALL_ELEM( pred, bb_pred_iter, Init(bbv->Pred()) ) {
00342 IDTYPE pred_id = pred->Id();
00343 if ( Semi(pred_id) == 0 ) {
00344 Set_parent(pred_id,v);
00345 xDFS( pred );
00346 }
00347 }
00348 }
00349
00350
00351
00352
00353
00354 void
00355 DOM_INFO::Compress( IDTYPE v )
00356 {
00357 Assert( Ancestor(v) != 0,
00358 (EC_Unimplemented, "Ancestor(v) is 0") );
00359
00360 if ( Ancestor(Ancestor(v)) != 0 ) {
00361 Compress( Ancestor(v) );
00362
00363 if ( Semi(Label(Ancestor(v))) < Semi(Label(v)) ) {
00364 Set_label(v, Label(Ancestor(v)));
00365 }
00366
00367 Set_ancestor(v, Ancestor(Ancestor(v)));
00368 }
00369 }
00370
00371
00372
00373
00374
00375 IDTYPE
00376 DOM_INFO::Eval( const BB_NODE *bbv )
00377 {
00378 IDTYPE v = bbv->Id();
00379
00380 if ( Ancestor(v) == 0 ) {
00381 return ( Label(v) );
00382 }
00383 else {
00384 Compress( v );
00385
00386 return ( (Semi(Label(Ancestor(v))) >= Semi(Label(v))) ?
00387 Label(v) : Label(Ancestor(v)) );
00388 }
00389 }
00390
00391
00392
00393
00394
00395 void
00396 DOM_INFO::Link( BB_NODE *bbv, BB_NODE *bbw )
00397 {
00398 IDTYPE v = bbv->Id();
00399 IDTYPE w = bbw->Id();
00400 IDTYPE s = w;
00401
00402 while ( Semi(Label(w)) < Semi(Label(Child(s))) ) {
00403 if ( Size(s) + Size(Child(Child(s))) >= 2*Size(Child(s)) ) {
00404 Set_ancestor(Child(s), s);
00405 Set_child(s, Child(Child(s)));
00406 }
00407 else {
00408 Set_size(Child(s), Size(s));
00409 Set_ancestor(s, Child(s));
00410 s = Child(s);
00411 }
00412 }
00413
00414 Set_label(s, Label(w));
00415 Set_size(v, Size(v) + Size(w));
00416
00417 if ( Size(v) < 2*Size(w) ) {
00418
00419 IDTYPE tmps = s;
00420 s = Child(v);
00421 Set_child(v, tmps);
00422 }
00423
00424 while ( s != 0 ) {
00425 Set_ancestor(s, v);
00426 s = Child(s);
00427 }
00428 }
00429
00430
00431
00432
00433
00434 void
00435 DOM_INFO::Build_dom_tree( CFG *cfg )
00436 {
00437
00438
00439
00440
00441
00442 DFSBB_ITER dfs_iter(cfg);
00443 BB_NODE *bb;
00444
00445 FOR_ALL_ELEM( bb, dfs_iter, Init() ) {
00446 IDTYPE bbid = bb->Id();
00447
00448 if ( Dom(bbid) != 0 ) {
00449 BB_NODE *idombb = cfg->Get_bb(Dom(bbid));
00450
00451
00452 bb->Set_idom(idombb);
00453
00454
00455 idombb->Add_dom_bbs(bb, cfg->Mem_pool());
00456 }
00457 else {
00458
00459
00460 FmtAssert( bb == cfg->Entry_bb() ||
00461 bb == cfg->Fake_entry_bb() ||
00462 bb == cfg->Fake_exit_bb(),
00463 ("DOM_INFO::Build_dom_tree: No dom for BB_NODE %d", bbid) );
00464 }
00465 }
00466 }
00467
00468
00469
00470
00471
00472 void
00473 DOM_INFO::Build_pdom_tree( const CFG *cfg )
00474 {
00475 CFG_ITER cfg_iter(cfg);
00476 BB_NODE *bb;
00477
00478 FOR_ALL_NODE( bb, cfg_iter, Init() ) {
00479 IDTYPE bbid = bb->Id();
00480
00481 if ( Dom(bbid) != 0 ) {
00482 BB_NODE *ipdombb = cfg->Get_bb(Dom(bbid));
00483
00484
00485 bb->Set_ipdom(ipdombb);
00486
00487
00488 ipdombb->Add_pdom_bbs(bb, cfg->Mem_pool());
00489 }
00490 else {
00491
00492
00493 FmtAssert( bb == cfg->Exit_bb() ||
00494 bb == cfg->Fake_exit_bb() ||
00495 bb == cfg->Fake_entry_bb(),
00496 ("DOM_INFO::Build_pdom_tree: No post-dom for BB_NODE %d", bbid) );
00497 }
00498 }
00499 }
00500
00501
00502
00503
00504
00505
00506
00507
00508 void
00509 DOM_INFO::Compute_dom_tree( CFG *cfg, BOOL build_dom )
00510 {
00511
00512
00513
00514 if ( build_dom ) {
00515 eDFS( cfg->Entry_bb() );
00516 }
00517 else {
00518 xDFS( cfg->Exit_bb() );
00519 }
00520
00521
00522 Set_size(0,0);
00523 Set_label(0,0);
00524 Set_semi(0,0);
00525
00526 for ( IDTYPE i = Counter(); i >= 2; i-- ) {
00527 IDTYPE w = Vertex(i);
00528 Is_True(w!=0, ("DOM_INFO::Compute_dom_tree, algorithm failed, the input tree is bad!"));
00529 BB_NODE *bbw = cfg->Get_bb(w);
00530 BB_NODE *bbv;
00531
00532
00533 if ( build_dom ) {
00534 BB_LIST_ITER bbw_pred_iter;
00535
00536 FOR_ALL_ELEM( bbv, bbw_pred_iter, Init(bbw->Pred()) ) {
00537 IDTYPE u = Eval( bbv );
00538 if ( Semi(u) < Semi(w) ) {
00539 Set_semi(w, Semi(u));
00540 }
00541 }
00542 }
00543 else {
00544 BB_LIST_ITER bbw_succ_iter;
00545
00546 FOR_ALL_ELEM( bbv, bbw_succ_iter, Init(bbw->Succ()) ) {
00547 IDTYPE u = Eval( bbv );
00548 if ( Semi(u) < Semi(w) ) {
00549 Set_semi(w, Semi(u));
00550 }
00551 }
00552 }
00553
00554
00555 Bucket(Vertex(Semi(w)))->Union1D( bbw );
00556
00557 Link( cfg->Get_bb(Parent(w)), bbw );
00558
00559
00560
00561 {BB_NODE_SET_ITER par_w_iter;
00562 FOR_ALL_ELEM( bbv, par_w_iter, Init(Bucket(Parent(w))) ) {
00563 IDTYPE v = bbv->Id();
00564 IDTYPE u = Eval( bbv );
00565
00566
00567 Bucket(Parent(w))->Difference1D( bbv );
00568
00569 Set_dom(v, (Semi(u) < Semi(v)) ? u : Parent(w));
00570 }}
00571
00572 }
00573
00574
00575 for ( IDTYPE j = 2; j <= Counter(); j++ ) {
00576 IDTYPE w = Vertex(j);
00577 if ( Dom(w) != Vertex(Semi(w)) ) {
00578 Set_dom(w, Dom(Dom(w)));
00579 }
00580 }
00581
00582 if ( build_dom ) {
00583
00584 Set_dom(cfg->Entry_bb()->Id(), 0);
00585
00586
00587 Build_dom_tree( cfg );
00588
00589
00590 if ( Get_Trace( TP_GLOBOPT, DOM_DUMP_FLAG ) ) {
00591 Print_dom( cfg, TFile );
00592 }
00593 }
00594 else {
00595
00596 Set_dom(cfg->Exit_bb()->Id(), 0);
00597
00598
00599 Build_pdom_tree( cfg );
00600
00601
00602 if ( Get_Trace( TP_GLOBOPT, DOM_DUMP_FLAG ) ) {
00603 Print_pdom( cfg, TFile );
00604 }
00605 }
00606
00607 }
00608
00609
00610
00611
00612
00613
00614
00615
00616
00617 static void
00618 Compute_dom_dfs_id( BB_NODE *bb, IDTYPE *id )
00619 {
00620 bb->Set_dom_dfs_id(*id);
00621 ++(*id);
00622
00623 BB_NODE *dom_bb;
00624 BB_LIST_ITER dom_bb_iter;
00625 FOR_ALL_ELEM ( dom_bb, dom_bb_iter, Init(bb->Dom_bbs()) )
00626 Compute_dom_dfs_id(dom_bb, id);
00627
00628 bb->Set_dom_dfs_last(*id - 1);
00629 }
00630
00631 static void
00632 Compute_pdom_dfs_id( BB_NODE *bb, IDTYPE *id )
00633 {
00634 bb->Set_pdom_dfs_id(*id);
00635 ++(*id);
00636
00637 BB_NODE *pdom_bb;
00638 BB_LIST_ITER pdom_bb_iter;
00639 FOR_ALL_ELEM ( pdom_bb, pdom_bb_iter, Init(bb->Pdom_bbs()) )
00640 Compute_pdom_dfs_id(pdom_bb, id);
00641
00642 bb->Set_pdom_dfs_last(*id - 1);
00643 }
00644
00645
00646
00647
00648
00649
00650
00651
00652 void
00653 CFG::Compute_dom_tree( BOOL build_dom )
00654 {
00655 DOM_INFO *dom_info;
00656
00657 OPT_POOL_Push( _loc_pool, DOM_DUMP_FLAG );
00658
00659 dom_info = CXX_NEW(DOM_INFO(this, _loc_pool), _loc_pool);
00660 dom_info->Compute_dom_tree(this, build_dom);
00661
00662 OPT_POOL_Pop( _loc_pool, DOM_DUMP_FLAG );
00663
00664 IDTYPE id = 0;
00665 if (build_dom) {
00666
00667
00668
00669
00670 Compute_dom_dfs_id( Entry_bb(), &id );
00671 } else {
00672
00673
00674
00675
00676 Compute_pdom_dfs_id( Exit_bb(), &id );
00677 }
00678 }
00679
00680
00681
00682
00683
00684
00685
00686
00687
00688
00689
00690
00691
00692
00693
00694
00695 static void
00696 Dom_compute_dom_frontier( const CFG *cfg, BB_NODE *bbx, MEM_POOL *mem_pool )
00697 {
00698 BB_LIST_ITER bbx_iter;
00699 BB_NODE *bby, *bbz;
00700 BB_NODE_SET *dfx;
00701
00702
00703
00704 FOR_ALL_ELEM( bby, bbx_iter, Init(bbx->Dom_bbs()) ) {
00705 Dom_compute_dom_frontier( cfg, bby, mem_pool );
00706 }
00707
00708
00709 dfx = CXX_NEW(BB_NODE_SET(0, cfg, mem_pool, BBNS_EMPTY),
00710 mem_pool);
00711
00712
00713 FOR_ALL_ELEM( bby, bbx_iter, Init(bbx->Succ()) ) {
00714 BB_NODE *idomy = bby->Idom();
00715
00716 if ( idomy != bbx ) {
00717
00718 dfx->Union1D( bby );
00719 }
00720 }
00721
00722
00723 FOR_ALL_ELEM( bbz, bbx_iter, Init(bbx->Dom_bbs()) ) {
00724 BB_NODE_SET *dfz = bbz->Dom_frontier();
00725 BB_NODE_SET_ITER dfz_iter;
00726
00727
00728 FOR_ALL_ELEM( bby, dfz_iter, Init(dfz) ) {
00729 BB_NODE *idomy = bby->Idom();
00730
00731 if ( idomy != bbx ) {
00732
00733 dfx->Union1D( bby );
00734 }
00735 }
00736 }
00737
00738
00739 bbx->Set_dom_frontier( dfx );
00740
00741 if ( Get_Trace( TP_GLOBOPT, DOM_DUMP_FLAG ) ) {
00742 fprintf ( TFile, "DF(BB:%d): ", bbx->Id() );
00743 dfx->Print( TFile );
00744 fprintf ( TFile, "\n" );
00745 }
00746 }
00747
00748
00749
00750
00751
00752 void
00753 CFG::Compute_dom_frontier( void )
00754 {
00755
00756 Assert( Entry_bb()->Dom_bbs() != NULL,
00757 (EC_Unimplemented,"CFG::Compute_dom_frontier: no dominator"));
00758 Dom_compute_dom_frontier( this, Entry_bb(), _mem_pool );
00759 }
00760
00761
00762
00763
00764
00765 static void
00766 Dom_compute_rcfg_dom_frontier( const CFG *cfg, BB_NODE *block, MEM_POOL *mem_pool )
00767 {
00768 BB_LIST_ITER block_iter;
00769 BB_NODE *y, *z;
00770
00771
00772 FOR_ALL_ELEM( y, block_iter, Init(block->Pdom_bbs()) )
00773 {
00774 Dom_compute_rcfg_dom_frontier( cfg, y, mem_pool );
00775 }
00776
00777
00778 block->Set_rcfg_dom_frontier (
00779 CXX_NEW(BB_NODE_SET(0, cfg, mem_pool, BBNS_EMPTY), mem_pool));
00780
00781
00782
00783 FOR_ALL_ELEM( y, block_iter, Init(block->Pred()) ) {
00784 if ( y->Ipdom() != block ) {
00785 block->Rcfg_dom_frontier()->Union1D( y );
00786 }
00787 }
00788
00789
00790 FOR_ALL_ELEM( z, block_iter, Init(block->Pdom_bbs()) ) {
00791 BB_NODE_SET_ITER dfz_iter;
00792 BB_NODE_SET *dfz = z->Rcfg_dom_frontier();
00793
00794
00795 FOR_ALL_ELEM( y, dfz_iter, Init(dfz) ) {
00796 if ( y->Ipdom() != block ) {
00797 block->Rcfg_dom_frontier()->Union1D( y );
00798 }
00799 }
00800 }
00801
00802
00803 if ( Get_Trace( TP_GLOBOPT, DOM_DUMP_FLAG ) ) {
00804 block->Print(TFile);
00805 }
00806 }
00807
00808
00809
00810
00811
00812
00813
00814
00815
00816
00817
00818
00819
00820
00821
00822
00823 void
00824 CFG::Compute_control_dependence( void )
00825 {
00826 CFG_ITER cfg_iter;
00827
00828
00829 Assert( Entry_bb()->Dom_bbs() != NULL,
00830 (EC_Unimplemented,"CFG::Compute_control_dependence: no dominator"));
00831
00832 Dom_compute_rcfg_dom_frontier( this, Exit_bb(), _mem_pool );
00833
00834 }
00835