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 #ifdef USE_PCH
00060 #include "be_com_pch.h"
00061 #endif
00062 #pragma hdrstop
00063 #ifdef _KEEP_RCS_ID
00064 #define opt_alias_mgr_CXX "opt_alias_mgr.cxx"
00065 static char *rcs_id = opt_alias_mgr_CXX"$Revision: 1.7 $";
00066 #endif
00067
00068 #include "string.h"
00069
00070 #include "defs.h"
00071 #include "config.h"
00072 #include "opt_config.h"
00073 #include "errors.h"
00074 #include "erglob.h"
00075 #include "tracing.h"
00076 #include "cxx_base.h"
00077 #include "cxx_template.h"
00078 #include "wn.h"
00079 #include "wn_map.h"
00080 #include "wn_util.h"
00081 #include "ir_reader.h"
00082 #include "stblock.h"
00083 #include "opt_defs.h"
00084 #include "opt_base.h"
00085 #include "opt_alias_mgr.h"
00086 #include "opt_alias_class.h"
00087 #include "opt_alias_rule.h"
00088 #include "opt_points_to.h"
00089 #include "region_util.h"
00090 #include "targ_sim.h"
00091
00092 #define ALIAS_TRACE_FLAG 0x1000000
00093 #define ALIAS_DUMP_FLAG 0x0800
00094
00095
00096
00097
00098
00099 class RESTRICTED_MAP {
00100 struct INFO {
00101 UINT _restricted:1;
00102 UINT _unique_pt:1;
00103 UINT _depth:3;
00104 ST *_based_sym;
00105 };
00106 ALIAS_MANAGER *_am;
00107 WN_MAP _map;
00108 MEM_POOL *_pu_pool;
00109 vector<const ST *,
00110 mempool_allocator<const ST *> > _invalid_based_syms;
00111
00112 public:
00113 #ifdef Is_True_On
00114 BOOL _saved;
00115 #endif
00116
00117 RESTRICTED_MAP(MEM_POOL *pu_pool) :
00118 _invalid_based_syms(pu_pool),
00119 _am(NULL),
00120 #if Is_True_On
00121 _saved(FALSE),
00122 #endif
00123 _pu_pool(pu_pool),
00124 _map(WN_MAP_Create(pu_pool))
00125 { WN_MAP_Set_dont_copy(_map, TRUE);}
00126
00127 ~RESTRICTED_MAP(void) {
00128 WN_MAP_Delete(_map);
00129 }
00130
00131 void Note_invalid_based_symbol(const ST *st) {
00132 _invalid_based_syms.push_back(st);
00133 }
00134
00135 void Save_info(WN *);
00136 BOOL Restore_info(WN *, POINTS_TO *);
00137 void Verify_info(const WN *, const POINTS_TO *);
00138 void Set_alias_mgr(ALIAS_MANAGER *am) { _am = am; }
00139 void Erase_entry_if_invalid(WN *);
00140 void Remove_info(WN *);
00141 MEM_POOL *Pu_pool(void) { return _pu_pool; }
00142 };
00143
00144 static RESTRICTED_MAP *Restricted_map = NULL;
00145
00146
00147
00148 void
00149 RESTRICTED_MAP::Save_info(WN *wn)
00150 {
00151 if (wn == NULL)
00152 return;
00153
00154 if (OPERATOR_is_scalar_iload (WN_operator (wn)) ||
00155 OPERATOR_is_scalar_istore (WN_operator (wn)) ||
00156 WN_operator(wn) == OPR_MLOAD ||
00157 WN_operator(wn) == OPR_MSTORE ||
00158 WN_operator(wn) == OPR_PARM) {
00159 IDTYPE id = _am->Id(wn);
00160 if (id > _am->Preg_id()) {
00161 POINTS_TO *pt = _am->Pt(id);
00162 if (pt != NULL && pt->Based_sym() != NULL) {
00163 struct INFO *pinfo = CXX_NEW(struct INFO, Pu_pool());
00164 pinfo->_based_sym = pt->Based_sym();
00165 pinfo->_depth = pt->Based_sym_depth();
00166 pinfo->_unique_pt = (pt->Unique_pt() != 0);
00167 pinfo->_restricted = (pt->Restricted() != 0);
00168 WN_MAP_Set(_map, wn, (void *) pinfo);
00169 if (Get_Trace( TP_GLOBOPT, ALIAS_TRACE_FLAG)) {
00170 fprintf(TFile, "--- RESTRICTED_MAP::Save_info WN %d ST %s(%d) %s %s\n", WN_map_id(wn),
00171 ST_name(pt->Based_sym()), pt->Based_sym_depth(),
00172 pinfo->_restricted ? "restricted" : "",
00173 pinfo->_unique_pt ? "unique_pt" : "");
00174 fdump_tree(TFile, wn);
00175 }
00176 }
00177 }
00178 }
00179
00180
00181 if (WN_opcode(wn) == OPC_REGION) {
00182 RID *rid = REGION_get_rid(wn);
00183 Is_True(rid != NULL,("RESTRICTED_MAP::Save_info, NULL rid"));
00184 if (RID_level(rid) >= RL_RBI)
00185 return;
00186 }
00187
00188 WN *stmt;
00189 if (WN_opcode(wn) == OPC_BLOCK)
00190 for (stmt = WN_first(wn); stmt != NULL; stmt = WN_next(stmt))
00191 Save_info(stmt);
00192 else if ( !OPCODE_is_black_box( WN_opcode(wn) ) ) {
00193 for (INT i = 0; i < WN_kid_count(wn); i++)
00194 Save_info(WN_kid(wn,i));
00195 }
00196 }
00197
00198 void
00199 RESTRICTED_MAP::Erase_entry_if_invalid(WN *wn)
00200 {
00201 struct INFO *pinfo = (struct INFO *) WN_MAP_Get(_map, wn);
00202 if (pinfo != NULL) {
00203 Is_True(pinfo->_based_sym != NULL,
00204 ("Restricted map entry must have non-NULL based_sym"));
00205 if (find(_invalid_based_syms.begin(), _invalid_based_syms.end(),
00206 pinfo->_based_sym) != _invalid_based_syms.end()) {
00207 Remove_info(wn);
00208 }
00209 }
00210 }
00211
00212
00213
00214 void
00215 RESTRICTED_MAP::Remove_info(WN *wn)
00216 {
00217 WN_MAP_Set(_map, wn, NULL);
00218 }
00219
00220 BOOL
00221 RESTRICTED_MAP::Restore_info(WN *wn, POINTS_TO *pt)
00222 {
00223 struct INFO *pinfo = (struct INFO *) WN_MAP_Get(_map, wn);
00224 if (pinfo != NULL) {
00225 pt->Set_expr_kind(EXPR_IS_ADDR);
00226 if (pinfo->_unique_pt)
00227 pt->Set_unique_pt();
00228 else
00229 pt->Reset_unique_pt();
00230 if (pinfo->_restricted)
00231 pt->Set_restricted();
00232 else
00233 pt->Reset_restricted();
00234 pt->Set_based_sym(pinfo->_based_sym);
00235 pt->Set_based_sym_depth(pinfo->_depth);
00236 if (Get_Trace( TP_GLOBOPT, ALIAS_TRACE_FLAG)) {
00237 fprintf(TFile, "--- RESTRICTED_MAP::Restore_info WN %d ST %s(%d) %s %s\n", WN_map_id(wn),
00238 ST_name(pt->Based_sym()), pt->Based_sym_depth(),
00239 pt->Restricted() ? "restricted" : "",
00240 pt->Unique_pt() ? "unique_pt" : "");
00241 fdump_tree_no_st(TFile, wn);
00242 }
00243 return TRUE;
00244 }
00245 return FALSE;
00246 }
00247
00248 BOOL Update_From_Restricted_Map(WN *const wn, POINTS_TO *const pt)
00249 {
00250 if (WOPT_Enable_Restricted_Map && Restricted_map != NULL) {
00251 return Restricted_map->Restore_info(wn, pt);
00252 }
00253 return FALSE;
00254 }
00255
00256 void
00257 RESTRICTED_MAP::Verify_info(const WN *const wn,
00258 const POINTS_TO *const pt)
00259 {
00260 #if Is_True_On
00261 struct INFO *pinfo = (struct INFO *) WN_MAP_Get(_map, wn);
00262 if (pinfo != NULL) {
00263 Is_True(_saved, ("RESTRICTED_MAP: Map must be saved"));
00264 Is_True(pt->Expr_kind() == EXPR_IS_ADDR,
00265 ("RESTRICTED_MAP: Wrong Expr_kind()"));
00266 if (pinfo->_unique_pt) {
00267 Is_True(pt->Unique_pt(),
00268 ("RESTRICTED_MAP: Lost Unique_pt()"));
00269 }
00270 else {
00271 Is_True(!pt->Unique_pt(),
00272 ("RESTRICTED_MAP: Spurious Unique_pt()"));
00273 }
00274 if (pinfo->_restricted) {
00275 Is_True(pt->Restricted(),
00276 ("RESTRICTED_MAP: Lost Restricted()"));
00277 }
00278 else {
00279 Is_True(!pt->Restricted(),
00280 ("RESTRICTED_MAP: Spurious Restricted()"));
00281 }
00282 Is_True(pt->Based_sym() == pinfo->_based_sym,
00283 ("RESTRICTED_MAP: Incorrect Based_sym()"));
00284 Is_True(pt->Based_sym_depth() == pinfo->_depth,
00285 ("RESTRICTED_MAP: Incorrect Based_sym_depth()"));
00286 }
00287 else {
00288 if (_saved) {
00289
00290
00291
00292
00293 #if 0
00294 Is_True(!pt->Unique_pt(),
00295 ("RESTRICTED_MAP: Spurious Unique_pt()"));
00296 Is_True(!pt->Restricted(),
00297 ("RESTRICTED_MAP: Spurious Restricted()"));
00298 #endif
00299 }
00300 }
00301 #endif
00302 }
00303
00304 void
00305 Verify_Restricted_Map(const WN *const wn, const POINTS_TO *const pt)
00306 {
00307 if (WOPT_Enable_Restricted_Map && Restricted_map != NULL) {
00308 Restricted_map->Verify_info(wn, pt);
00309 }
00310 }
00311
00312
00313 void Create_Restricted_Map(MEM_POOL *pu_pool)
00314 {
00315 if (WOPT_Enable_Restricted_Map) {
00316 Restricted_map = CXX_NEW(RESTRICTED_MAP(pu_pool), pu_pool);
00317
00318 }
00319
00320 }
00321
00322
00323 void Copy_Restricted_Map(WN *tree, ALIAS_MANAGER *am)
00324 {
00325 if (WOPT_Enable_Restricted_Map) {
00326 Restricted_map->Set_alias_mgr(am);
00327 Restricted_map->Save_info(tree);
00328 #if Is_True_On
00329 Restricted_map->_saved = TRUE;
00330 #endif
00331 }
00332 }
00333
00334
00335 void Delete_Restricted_Map(void)
00336 {
00337 if (WOPT_Enable_Restricted_Map && Restricted_map != NULL) {
00338 MEM_POOL *pu_pool = Restricted_map->Pu_pool();
00339 CXX_DELETE(Restricted_map, pu_pool);
00340 Restricted_map = NULL;
00341 }
00342 }
00343
00344
00345 void Erase_Restricted_Mapping(WN *wn)
00346 {
00347 if (WOPT_Enable_Restricted_Map && Restricted_map != NULL) {
00348
00349
00350
00351 if (WN_map_id(wn) == WN_MAP_UNDEFINED)
00352 return;
00353 Restricted_map->Remove_info(wn);
00354 }
00355 }
00356
00357 void Note_Invalid_Based_Symbol(const ST *st)
00358 {
00359 if (WOPT_Enable_Restricted_Map && Restricted_map != NULL) {
00360 Restricted_map->Note_invalid_based_symbol(st);
00361 }
00362 }
00363
00364
00365
00366
00367
00368 ALIAS_MANAGER::ALIAS_MANAGER(void)
00369 {
00370 MEM_POOL_Initialize(&_mem_pool, "ALIAS_pool", FALSE);
00371 MEM_POOL_Push(&_mem_pool);
00372
00373 #if 0
00374 _invalid_ip_alias_classes =
00375 CXX_NEW(vector<IDTYPE, mempool_allocator<IDTYPE> > (&_mem_pool),
00376 &mem_pool);
00377 #else
00378
00379
00380
00381
00382
00383
00384 typedef vector<IDTYPE, mempool_allocator<IDTYPE> > STUPID_COMPILER;
00385 _invalid_ip_alias_classes =
00386 CXX_NEW(STUPID_COMPILER(&_mem_pool), &_mem_pool);
00387 #endif
00388
00389
00390 ALIAS_CONTEXT ac = (DEFAULT_COMMON_RULES | DEFAULT_ANALYSIS_RULES | DEFAULT_COMPATIABILITY_RULES);
00391
00392 switch (PU_src_lang(Get_Current_PU())) {
00393 case PU_F77_LANG:
00394 ac |= DEFAULT_F_RULES;
00395 if (Alias_Pointer_Parms)
00396 ac |= F_PARM_RULE;
00397 if (Alias_Pointer_Cray)
00398 ac |= F_CRAY_POINTER_RULE;
00399 break;
00400 case PU_F90_LANG:
00401 ac |= DEFAULT_F_RULES;
00402 if (Alias_Pointer_Parms)
00403 ac |= F_PARM_RULE;
00404 if (Alias_Pointer_Cray)
00405 ac |= F_CRAY_POINTER_RULE;
00406 ac |= DEFAULT_F90_RULES;
00407 break;
00408 case PU_C_LANG:
00409 ac |= DEFAULT_C_RULES;
00410 if (Alias_Pointer_Types)
00411 ac |= C_ANSI_RULE;
00412 if (Alias_Pointer_Strongly_Typed)
00413 ac |= C_STRONGLY_TYPED_RULE;
00414 if (Alias_Pointer_Named_Data)
00415 ac |= RAG_UNNAMED_RULE;
00416 if (Alias_Pointer_Restricted)
00417 ac |= RAG_RESTRICTED_RULE;
00418 if (Alias_Pointer_Disjoint)
00419 ac |= IBM_DISJOINT_RULE;
00420 break;
00421 case PU_CXX_LANG:
00422 ac |= DEFAULT_C_RULES;
00423 ac |= DEFAULT_CXX_RULES;
00424 if (Alias_Pointer_Types)
00425 ac |= C_ANSI_RULE;
00426 if (Alias_Pointer_Strongly_Typed)
00427 ac |= C_STRONGLY_TYPED_RULE;
00428 if (Alias_Pointer_Named_Data)
00429 ac |= RAG_UNNAMED_RULE;
00430 if (Alias_Pointer_Restricted)
00431 ac |= RAG_RESTRICTED_RULE;
00432 if (Alias_Pointer_Disjoint)
00433 ac |= IBM_DISJOINT_RULE;
00434
00435 break;
00436 default:
00437 Is_True(FALSE, ("Language is unknown; mixed-language inlining illegal."));
00438 }
00439
00440 Set_pu_context(ac);
00441 _rule = CXX_NEW(ALIAS_RULE(ac), &_mem_pool);
00442
00443
00444 _trace = Get_Trace(TP_GLOBOPT, ALIAS_DUMP_FLAG);
00445
00446
00447
00448
00449
00450
00451
00452 _vec = CXX_NEW(DYN_ARRAY<POINTS_TO*>(&_mem_pool), &_mem_pool);
00453 _vec->Initidx(_preg_id);
00454 _map = WN_MAP32_Create(&_mem_pool);
00455 _homing_map = WN_MAP32_Create(&_mem_pool);
00456 _last_alias_id = _preg_id;
00457 _no_alias_info_id = New_alias_id();
00458
00459
00460 POINTS_TO *npt = Pt(_no_alias_info_id);
00461 npt->Set_id(_no_alias_info_id);
00462 npt->Init();
00463 npt->Set_expr_kind(EXPR_IS_ADDR);
00464 }
00465
00466
00467
00468
00469 ALIAS_MANAGER::~ALIAS_MANAGER(void)
00470 {
00471 CXX_DELETE(_invalid_ip_alias_classes, &_mem_pool);
00472 WN_MAP_Delete(_map);
00473 WN_MAP_Delete(_homing_map);
00474 MEM_POOL_Pop(&_mem_pool);
00475 MEM_POOL_Delete(&_mem_pool);
00476 }
00477
00478
00479 IDTYPE
00480 ALIAS_MANAGER::Id(const WN *wn) const
00481 {
00482 return WN_MAP32_Get(_map, wn);
00483 }
00484
00485
00486 inline void
00487 ALIAS_MANAGER::Set_id(WN *wn, IDTYPE id) const
00488 {
00489 WN_MAP32_Set(_map, wn, id);
00490 }
00491
00492
00493 inline void
00494 ALIAS_MANAGER::Set_context(ALIAS_CONTEXT c)
00495 {
00496 _rule->Set_context(c);
00497 }
00498
00499
00500
00501
00502 inline IDTYPE
00503 ALIAS_MANAGER::New_alias_id(void)
00504 {
00505 IDTYPE id = ++_last_alias_id;
00506 Vec()->Newidx();
00507 POINTS_TO *pt = CXX_NEW(POINTS_TO, &_mem_pool);
00508 pt->Init();
00509 (*_vec)[id] = pt;
00510 pt->Set_id(id);
00511 return id;
00512 }
00513
00514 inline POINTS_TO *
00515 ALIAS_MANAGER::New_points_to(WN *wn)
00516 {
00517 IDTYPE id = ++_last_alias_id;
00518 Vec()->Newidx();
00519 POINTS_TO *pt = CXX_NEW(POINTS_TO, &_mem_pool);
00520 pt->Init();
00521 (*_vec)[id] = pt;
00522 pt->Set_id(id);
00523 Set_id(wn,id);
00524 return pt;
00525 }
00526
00527
00528 #ifdef Is_True_On
00529 static void
00530 Check_self_alias(ALIAS_MANAGER *am, WN *wn)
00531 {
00532
00533
00534 FmtAssert(Aliased(am, wn, wn), ("WN is not aliased to itself."));
00535
00536 OPERATOR opr = WN_operator(wn);
00537 IDTYPE id = am->Id(wn);
00538 POINTS_TO *pt = (id != 0) ? am->Pt(id) : NULL;
00539
00540
00541
00542 if ((OPERATOR_is_scalar_iload (opr) || OPERATOR_is_scalar_istore (opr)) &&
00543 !TY_is_const(WN_object_ty(wn)) &&
00544 !(pt != NULL && pt->Const()))
00545 FmtAssert(Overlapped_base(am, wn, wn), ("Indirect is not aliased to itself."));
00546 }
00547 #else
00548 #define Check_self_alias(am,wn)
00549 #endif
00550
00551
00552 inline static bool Is_PREG_ldst(WN *wn)
00553 {
00554 OPERATOR opr = WN_operator(wn);
00555 return ((OPERATOR_is_scalar_load (opr) || OPERATOR_is_scalar_store (opr)) &&
00556 ST_sclass(WN_st(wn)) == SCLASS_REG);
00557 }
00558
00559
00560
00561
00562 void
00563 ALIAS_MANAGER::Gen_alias_id(WN *wn, POINTS_TO *pt)
00564 {
00565 if (pt != NULL) {
00566 WN_MAP32_Set(WN_MAP_ALIAS_CLASS, wn, pt->Ip_alias_class());
00567 }
00568
00569 if (!WOPT_Enable_CG_Alias) {
00570 Set_id(wn, No_alias_info_id());
00571 return;
00572 }
00573
00574 if (Is_PREG_ldst(wn)) {
00575 Set_id(wn, Preg_id());
00576 return;
00577 }
00578
00579 IDTYPE id;
00580 Is_True(pt != NULL && pt->Expr_kind() != EXPR_IS_INVALID, ("POINTS_TO is invalid."));
00581 if ((id = pt->Id()) == 0) {
00582
00583 id = New_alias_id();
00584 POINTS_TO *npt = Pt(id);
00585 npt->Copy_fully(pt);
00586 pt->Set_id(id);
00587 npt->Set_id(id);
00588 Set_id(wn, id);
00589 if (_trace) {
00590 fprintf(TFile, "gen_alias_id<%d(map %d)>\n", id, WN_map_id(wn));
00591 pt->Print(TFile);
00592 fdump_tree(TFile, wn);
00593 fprintf(TFile,"aliased_with<%d,{",id);
00594 for (INT32 oldid = Preg_id() + 1; oldid <= id; oldid++) {
00595 if (Rule()->Aliased_Memop(Pt(oldid), Pt(id), Pt(oldid)->Ty(), Pt(id)->Ty()))
00596 fprintf(TFile,"%d ", oldid);
00597 }
00598 fprintf(TFile, "}>\n");
00599 }
00600 } else {
00601 Set_id(wn, id);
00602 if (_trace) {
00603 fprintf(TFile, "set_alias_id<%d(map %d)>\n", id, WN_map_id(wn));
00604 pt->Print(TFile);
00605 fdump_tree(TFile, wn);
00606 }
00607 }
00608
00609 Check_self_alias(this,wn);
00610 }
00611
00612
00613
00614
00615 void ALIAS_MANAGER::Gen_alias_id_list(WN *wn, POINTS_TO_LIST *ptl)
00616 {
00617 if (ptl == NULL) {
00618 Set_id(wn,0);
00619 return;
00620 }
00621 POINTS_TO_ITER def_iter;
00622 POINTS_TO_NODE *defn;
00623 IDTYPE first_id = 0;
00624 POINTS_TO *npt = NULL;
00625 FOR_ALL_NODE( defn, def_iter, Init(ptl)) {
00626 POINTS_TO *pt = defn->Pt();
00627 IDTYPE id = New_alias_id();
00628 if (first_id == 0) first_id = id;
00629 npt = Pt(id);
00630 npt->Copy_fully(pt);
00631 npt->Set_id(id);
00632 npt->Set_extended();
00633 }
00634 Set_id(wn, first_id);
00635 if (npt) npt->Reset_extended();
00636 }
00637
00638
00639 void
00640 ALIAS_MANAGER::Dup_tree_alias_id( const WN *old_wn, WN *new_wn )
00641 {
00642 const OPCODE opc = WN_opcode(old_wn);
00643
00644 Is_True( opc == WN_opcode(new_wn),
00645 ("ALIAS_MANAGER::Dup_tree_alias_id: non-matching trees") );
00646
00647
00648 IDTYPE old_id = Id(old_wn);
00649 if ( old_id != 0 ) {
00650 Set_id( new_wn, old_id );
00651 }
00652
00653 IDTYPE ip_alias_class = WN_MAP32_Get(WN_MAP_ALIAS_CLASS, old_wn);
00654 if (ip_alias_class != 0) {
00655 WN_MAP32_Set(WN_MAP_ALIAS_CLASS, new_wn, ip_alias_class);
00656 }
00657
00658
00659 if ( opc == OPC_BLOCK ) {
00660 WN *old_bwn, *new_bwn;
00661 for ( old_bwn = WN_first(old_wn), new_bwn = WN_first(new_wn);
00662 old_bwn != NULL;
00663 old_bwn = WN_next(old_bwn), new_bwn = WN_next(new_bwn) )
00664 {
00665 Dup_tree_alias_id( old_bwn, new_bwn );
00666 }
00667 }
00668 else {
00669
00670 if ( OPCODE_is_load(opc) ) {
00671 Set_homing_load( new_wn, Homing_load(old_wn) );
00672 }
00673 else if ( OPCODE_is_store(opc) ) {
00674 Set_homing_store( new_wn, Homing_store(old_wn) );
00675 }
00676
00677 for ( INT ikid = 0; ikid < WN_kid_count(old_wn); ikid++ ) {
00678 Dup_tree_alias_id( WN_kid(old_wn,ikid), WN_kid(new_wn,ikid) );
00679 }
00680 }
00681 }
00682
00683
00684
00685
00686 void
00687 ALIAS_MANAGER::Gen_black_box_alias(WN *wn)
00688 {
00689 const OPCODE opc = WN_opcode(wn);
00690 if (OPCODE_is_load(opc) || OPCODE_is_store(opc))
00691 Set_id(wn, No_alias_info_id());
00692
00693 for (INT i = 0; i < WN_kid_count(wn); i++) {
00694 Gen_black_box_alias( WN_kid(wn,i));
00695 }
00696 }
00697
00698
00699
00700
00701
00702
00703
00704
00705
00706
00707
00708
00709
00710
00711
00712
00713
00714 IDTYPE
00715 ALIAS_MANAGER::Cross_dso_new_alias_id(void)
00716 {
00717 return New_alias_id();
00718 }
00719
00720 void
00721 ALIAS_MANAGER::Cross_dso_set_id(WN *wn, IDTYPE id) const
00722 {
00723 Set_id(wn, id);
00724 }
00725
00726
00727
00728
00729
00730
00731 ALIAS_MANAGER *Create_Alias_Manager(MEM_POOL *pu_pool)
00732 {
00733 return CXX_NEW(ALIAS_MANAGER(), pu_pool);
00734 }
00735
00736
00737 void Delete_Alias_Manager(ALIAS_MANAGER *am, MEM_POOL *pu_pool)
00738 {
00739 CXX_DELETE (am, pu_pool);
00740 }
00741
00742
00743
00744
00745 ALIAS_CONTEXT Get_Default_Alias_Context(ALIAS_MANAGER *am)
00746 {
00747 return am->Pu_context();
00748 }
00749
00750
00751 void Set_Alias_Context(ALIAS_MANAGER *am, ALIAS_CONTEXT context)
00752 {
00753 am->Set_context(context);
00754 }
00755
00756
00757 void Reset_Alias_Context(ALIAS_MANAGER *am)
00758 {
00759 am->Set_context(am->Pu_context());
00760 }
00761
00762
00763
00764
00765 void Create_alias(ALIAS_MANAGER *am, WN *wn)
00766 {
00767 if (Is_PREG_ldst(wn)) {
00768 am->Set_id(wn, am->Preg_id());
00769 } else {
00770 POINTS_TO *pt = am->New_points_to(wn);
00771 pt->Analyze_WN_expr(wn);
00772 pt->Set_ty (WN_object_ty(wn));
00773 }
00774 }
00775
00776
00777
00778
00779
00780 void Create_local_alias(ALIAS_MANAGER *am, WN *wn)
00781 {
00782 if (Is_PREG_ldst(wn)) {
00783 am->Set_id(wn, am->Preg_id());
00784 } else {
00785 POINTS_TO *pt = am->New_points_to(wn);
00786 pt->Analyze_WN_expr(wn);
00787 Is_True(pt->Local(), ("Create local alias: not a local variable."));
00788 }
00789 }
00790
00791
00792
00793
00794
00795 void Create_global_alias(ALIAS_MANAGER *am, ST *st, WN *ldid, WN *iload)
00796 {
00797 if (ldid != NULL) {
00798 Is_True(st == WN_st(ldid),
00799 ("Create_global_alias: ST and LDID does not match."));
00800 POINTS_TO *pt = am->New_points_to(ldid);
00801 pt->Analyze_WN_expr(ldid);
00802 pt->Set_not_addr_saved();
00803 pt->Set_not_addr_passed();
00804 }
00805 if (iload != NULL) {
00806 POINTS_TO *pt = am->New_points_to(iload);
00807 pt->Analyze_WN_expr(iload);
00808
00809 if (WOPT_Enable_Restricted_Map && Restricted_map != NULL) {
00810 Restricted_map->Remove_info(iload);
00811 }
00812 }
00813 }
00814
00815
00816
00817
00818 void Create_formal_alias(ALIAS_MANAGER *am, ST *st, WN *formal_addr, WN *formal)
00819 {
00820 Is_True(ST_sclass(st) == SCLASS_FORMAL,
00821 ("ST is not a formal parameter."));
00822
00823 if (formal_addr != NULL) {
00824 Is_True(st == WN_st(formal_addr),
00825 ("Create_formal_alias: ST and LDID does not match."));
00826 POINTS_TO *pt = am->New_points_to(formal_addr);
00827 pt->Analyze_WN_expr(formal_addr);
00828 pt->Set_not_addr_saved();
00829 pt->Set_not_addr_passed();
00830 Is_True(pt->Local(), ("Create formal alias: not an addr of formal variable."));
00831 }
00832
00833 if (formal != NULL) {
00834 POINTS_TO *pt = am->New_points_to(formal);
00835 pt->Analyze_WN_expr(formal);
00836 Is_True(pt->F_param(), ("Create formal alias: not a formal variable."));
00837 }
00838 }
00839
00840
00841
00842
00843
00844 void Create_unique_pointer_alias(ALIAS_MANAGER *am, ST *st, WN *ldid, WN *iload)
00845 {
00846 Is_True( ST_pt_to_unique_mem(st), ("Create_pointer: ST is not PT_TO_UNIQUE_MEM."));
00847
00848 if (ldid != NULL) {
00849 POINTS_TO *pt = am->New_points_to(ldid);
00850 pt->Analyze_WN_expr(ldid);
00851 Is_True(pt->Local(), ("Create unique pointer alias: not a local variable."));
00852 }
00853
00854 if (iload != NULL) {
00855 POINTS_TO *pt = am->New_points_to(iload);
00856 pt->Analyze_WN_expr(iload);
00857 Is_True(pt->Unique_pt() && pt->Based_sym() != NULL,
00858 ("Create unique pointer alias: not a local variable."));
00859
00860 if (WOPT_Enable_Restricted_Map && Restricted_map != NULL) {
00861 Restricted_map->Remove_info(iload);
00862 }
00863 }
00864 }
00865
00866
00867
00868
00869 void Create_lda_array_alias(ALIAS_MANAGER *am, WN *lda, WN *iload)
00870 {
00871 Is_True( WN_operator(lda) == OPR_LDA, ("Create_lda_array_alias: need a LDA node."));
00872
00873 if (iload != NULL) {
00874 ST *st = WN_st(lda);
00875 POINTS_TO *pt = am->New_points_to(iload);
00876 pt->Analyze_WN_expr(iload);
00877 pt->Set_byte_size(TY_size(ST_type(st)));
00878 if (TY_size(ST_type(st)) == 0) pt->Set_ofst_kind(OFST_IS_UNKNOWN);
00879 Is_True(pt->Base_kind() == BASE_IS_FIXED,
00880 ("Create_lda_array_alias: base is not fixed."));
00881
00882 if (WOPT_Enable_Restricted_Map && Restricted_map != NULL) {
00883 Restricted_map->Remove_info(iload);
00884 }
00885 }
00886 }
00887
00888
00889
00890
00891 BOOL No_alias(const ALIAS_MANAGER *am, WN *wn)
00892 {
00893 IDTYPE id = am->Id(wn);
00894 if (id == 0)
00895 return FALSE;
00896 if (id == am->Preg_id())
00897 return TRUE;
00898 POINTS_TO *pt = am->Pt(id);
00899 return pt->No_alias();
00900 }
00901
00902
00903
00904
00905 BOOL Valid_alias(const ALIAS_MANAGER *am, WN *wn)
00906 {
00907 if (WN_map_id(wn) != -1)
00908 {
00909 IDTYPE id = am->Id(wn);
00910
00911 return (id == 0) ? FALSE : TRUE;
00912 }
00913 return FALSE;
00914 }
00915
00916
00917
00918
00919 ALIAS_RESULT Aliased(const ALIAS_MANAGER *am, WN *wn1, WN *wn2,
00920 BOOL ignore_loop_carried)
00921 {
00922 IDTYPE id1 = am->Id(wn1);
00923 IDTYPE id2 = am->Id(wn2);
00924
00925
00926 if (id1 == 0 && Is_PREG_ldst(wn1))
00927 am->Set_id(wn1, id1 = am->Preg_id());
00928
00929 if (id2 == 0 && Is_PREG_ldst(wn2))
00930 am->Set_id(wn2, id2 = am->Preg_id());
00931
00932
00933 if (id1 == 0 || id2 == 0)
00934 return POSSIBLY_ALIASED;
00935
00936
00937
00938 if (id1 == am->Preg_id() && id2 == am->Preg_id())
00939 return (WN_offset(wn1) != WN_offset(wn2)) ? NOT_ALIASED : SAME_LOCATION;
00940
00941
00942
00943 if ((id1 == am->Preg_id() && id2 != am->Preg_id()) ||
00944 (id1 != am->Preg_id() && id2 == am->Preg_id()))
00945 return NOT_ALIASED;
00946
00947 POINTS_TO *pt1 = am->Pt(id1);
00948 POINTS_TO *pt2 = am->Pt(id2);
00949
00950 Is_True(pt1 != NULL && pt2 != NULL, ("Aliased: null points to"));
00951
00952
00953
00954 if (id1 == id2)
00955 return am->Rule()->Same_location(wn1, wn2, pt1, pt2) ? SAME_LOCATION : POSSIBLY_ALIASED;
00956
00957 if (Alias_Pointer_Strongly_Typed &&
00958 (OPERATOR_is_scalar_iload (WN_operator(wn1)) ||
00959 OPERATOR_is_scalar_istore (WN_operator(wn1))) &&
00960 (OPERATOR_is_scalar_iload (WN_operator(wn2)) ||
00961 OPERATOR_is_scalar_istore (WN_operator(wn2)))) {
00962 TY_IDX ty1 = OPERATOR_is_load (WN_operator(wn1)) ?
00963 WN_load_addr_ty(wn1) : WN_ty(wn1);
00964 TY_IDX ty2 = OPERATOR_is_load (WN_operator(wn2)) ?
00965 WN_load_addr_ty(wn2) : WN_ty(wn2);
00966 if (ty1 != (TY_IDX) NULL && TY_kind(ty1) == KIND_POINTER &&
00967 ty2 != (TY_IDX) NULL && TY_kind(ty2) == KIND_POINTER &&
00968 !am->Rule()->Aliased_Strongly_Typed_Rule(TY_pointed(ty1), TY_pointed(ty2))) {
00969 return NOT_ALIASED;
00970 }
00971 }
00972
00973 if (OPERATOR_is_store(WN_operator(wn1)) && OPERATOR_is_load(WN_operator(wn2)) ||
00974 OPERATOR_is_store(WN_operator(wn2)) && OPERATOR_is_load(WN_operator(wn1))) {
00975 if (am->Rule()->Aliased_Memop(pt1, pt2, ignore_loop_carried))
00976 return POSSIBLY_ALIASED;
00977 } else {
00978
00979 if (am->Rule()->Aliased_Memop(pt1, pt2, (TY_IDX)NULL, (TY_IDX)NULL,
00980 ignore_loop_carried))
00981 return POSSIBLY_ALIASED;
00982 }
00983 return NOT_ALIASED;
00984 }
00985
00986
00987 ALIAS_RESULT
00988 ALIAS_MANAGER::Aliased(const POINTS_TO *pt1, const POINTS_TO *pt2,
00989 BOOL ignore_loop_carried)
00990 {
00991 if (Rule()->Aliased_Memop(pt1, pt2, (TY_IDX) NULL, (TY_IDX) NULL,
00992 ignore_loop_carried))
00993 return POSSIBLY_ALIASED;
00994 return NOT_ALIASED;
00995 }
00996
00997 ALIAS_RESULT
00998 ALIAS_MANAGER::Aliased(WN *wn, const POINTS_TO *pt2,
00999 BOOL ignore_loop_carried)
01000 {
01001 IDTYPE id = Id(wn);
01002
01003
01004 if (id == 0 && Is_PREG_ldst(wn))
01005 Set_id(wn, id = Preg_id());
01006
01007 if (id == Preg_id() &&
01008 pt2->Base_is_fixed() &&
01009 ST_sclass(pt2->Base()) == SCLASS_REG) {
01010 return (WN_offset(wn) != pt2->Byte_Ofst()) ? NOT_ALIASED : SAME_LOCATION;
01011 }
01012 return Aliased(Pt(id), pt2, ignore_loop_carried);
01013 }
01014
01015 ALIAS_RESULT
01016 ALIAS_MANAGER::Aliased(const POINTS_TO *pt1, WN *wn,
01017 BOOL ignore_loop_carried)
01018 {
01019 return Aliased(wn, pt1, ignore_loop_carried);
01020 }
01021
01022 void
01023 ALIAS_MANAGER::Note_invalid_ip_alias_class(const WN *wn)
01024 {
01025 IDTYPE alias_class = WN_MAP32_Get(WN_MAP_ALIAS_CLASS, wn);
01026 if (alias_class != PESSIMISTIC_AC_ID &&
01027 alias_class != OPTIMISTIC_AC_ID) {
01028 if (Get_Trace( TP_GLOBOPT, ALIAS_TRACE_FLAG)) {
01029 fprintf(TFile, "Alias manager: Noting IP alias class %d invalid\n",
01030 alias_class);
01031 }
01032 _invalid_ip_alias_classes->push_back(alias_class);
01033 }
01034 }
01035
01036 void
01037 ALIAS_MANAGER::Erase_ip_alias_class_if_invalid(WN *wn)
01038 {
01039 IDTYPE alias_class = WN_MAP32_Get(WN_MAP_ALIAS_CLASS, wn);
01040 if (find(_invalid_ip_alias_classes->begin(),
01041 _invalid_ip_alias_classes->end(),
01042 alias_class) != _invalid_ip_alias_classes->end()) {
01043 if (Get_Trace( TP_GLOBOPT, ALIAS_TRACE_FLAG)) {
01044 fprintf(TFile, "Alias manager: Invalidating IP alias class info on\n");
01045 fdump_tree(TFile, wn);
01046 }
01047 WN_MAP32_Set(WN_MAP_ALIAS_CLASS, wn, PESSIMISTIC_AC_ID);
01048 }
01049 }
01050
01051
01052
01053 ALIAS_RESULT Aliased_with_region(const ALIAS_MANAGER *am, const WN *wn, const WN *region_or_call, READ_WRITE how)
01054 {
01055 IDTYPE id = am->Id(wn);
01056 if (id == 0) return POSSIBLY_ALIASED;
01057
01058 if (id == am->Preg_id()) {
01059 if (OPERATOR_is_scalar_store (WN_operator(wn)) ||
01060 OPERATOR_is_scalar_load (WN_operator(wn))) {
01061 INT32 reg = WN_offset(wn);
01062 if (reg > Last_Dedicated_Preg_Offset)
01063 return NOT_ALIASED;
01064 }
01065 return POSSIBLY_ALIASED;
01066 }
01067
01068 POINTS_TO *pt = am->Pt(id);
01069 if (OPERATOR_is_call(WN_operator(region_or_call))) {
01070
01071 if (Is_nested_call(region_or_call))
01072 return POSSIBLY_ALIASED;
01073
01074 ST *call_st = NULL;
01075 if (WN_operator(region_or_call) == OPR_CALL)
01076 call_st = WN_st(region_or_call);
01077 else if (WN_operator(region_or_call) == OPR_PICCALL)
01078 call_st = WN_st(region_or_call);
01079
01080 if (call_st == NULL)
01081 return POSSIBLY_ALIASED;
01082
01083
01084 READ_WRITE how_aliased =
01085 am->Rule()->Aliased_with_Call(call_st, WN_call_flag(region_or_call), pt);
01086 if (how_aliased != NO_READ_NO_WRITE &&
01087 (how == READ_AND_WRITE || how == how_aliased))
01088 return POSSIBLY_ALIASED;
01089
01090
01091 for (INT32 i = 0; i < WN_kid_count(region_or_call); i++) {
01092 WN *wn = WN_kid(region_or_call,i);
01093 #if defined(TARG_SL)
01094 if (WN_operator(wn) == OPR_PARM &&
01095 (WN_Parm_By_Reference(wn)||WN_Parm_Dereference(wn))) {
01096 #else
01097 if (WN_operator(wn) == OPR_PARM &&
01098 WN_Parm_By_Reference(wn)) {
01099 #endif
01100
01101 IDTYPE id2 = am->Id(wn);
01102 if (id2 == 0) return POSSIBLY_ALIASED;
01103 POINTS_TO *pt2 = am->Pt(id2);
01104 if (am->Rule()->Aliased_Memop(pt, pt2, (TY_IDX) NULL, (TY_IDX) NULL))
01105 return POSSIBLY_ALIASED;
01106 }
01107 }
01108 return NOT_ALIASED;
01109 }
01110
01111 OPERATOR opr = WN_operator(region_or_call);
01112
01113 if (opr == OPR_DEALLOCA && pt->Not_alloca_mem())
01114 return NOT_ALIASED;
01115
01116 if (opr == OPR_FORWARD_BARRIER ||
01117 opr == OPR_BACKWARD_BARRIER ||
01118 opr == OPR_DEALLOCA) {
01119 IDTYPE id = am->Id(wn);
01120 if (id != 0) {
01121 POINTS_TO *pt2;
01122 do {
01123 pt2 = am->Pt(id++);
01124 if (am->Rule()->Aliased_Memop(pt, pt2, (TY_IDX) NULL, (TY_IDX) NULL))
01125 return POSSIBLY_ALIASED;
01126 } while (pt2->Extended());
01127 return NOT_ALIASED;
01128 }
01129 }
01130
01131 if (opr == OPR_ASM_STMT) {
01132 if (am->Rule()->Aliased_with_Asm(region_or_call, pt) == NO_READ_NO_WRITE)
01133 return NOT_ALIASED;
01134 }
01135
01136 return POSSIBLY_ALIASED;
01137 }
01138
01139
01140
01141
01142 ALIAS_RESULT Aliased_with_intr_op(const ALIAS_MANAGER *am, const WN *intr_op, const WN *mem_op)
01143 {
01144 Is_True(WN_operator(intr_op) == OPR_INTRINSIC_OP,
01145 ("Aliased_with_intr_op: expecting OPR_INTRINSIC_OP."));
01146
01147 IDTYPE id1 = am->Id(mem_op);
01148 if (id1 == am->Preg_id())
01149 return NOT_ALIASED;
01150
01151 POINTS_TO *pt1 = am->Pt(id1);
01152
01153
01154 for (INT32 i = 0; i < WN_kid_count(intr_op); i++) {
01155 WN *wn = WN_kid(intr_op,i);
01156 #if defined(TARG_SL)
01157 if (WN_Parm_By_Reference(wn) || WN_Parm_Dereference(wn)) {
01158 #else
01159 if (WN_Parm_By_Reference(wn)) {
01160 #endif
01161
01162 IDTYPE id2 = am->Id(wn);
01163 if (id2 == 0) return POSSIBLY_ALIASED;
01164 POINTS_TO *pt2 = am->Pt(id2);
01165 if (am->Rule()->Aliased_Memop(pt1, pt2, (TY_IDX) NULL, (TY_IDX) NULL))
01166 return POSSIBLY_ALIASED;
01167 }
01168 }
01169 return NOT_ALIASED;
01170 }
01171
01172
01173
01174
01175
01176
01177
01178
01179
01180
01181
01182
01183 static void
01184 Check_range(POINTS_TO *pt, const WN *wn)
01185 {
01186 if (OPERATOR_is_scalar_iload (WN_operator(wn)) ||
01187 OPERATOR_is_scalar_istore (WN_operator(wn))) {
01188 WN *wn_tmp = OPERATOR_is_load (WN_operator(wn)) ?
01189 WN_kid0(wn) : WN_kid1(wn);
01190 WN *wn_lda = NULL;
01191 if (WN_operator(wn_tmp) == OPR_ARRAY) {
01192 if (WN_operator(WN_kid0(wn_tmp)) == OPR_LDA)
01193 wn_lda = WN_kid0(wn_tmp);
01194 } else if (WN_operator(wn_tmp) == OPR_LDA)
01195 wn_lda = wn_tmp;
01196
01197 if (wn_lda != NULL) {
01198 mINT64 ofst = 0;
01199 ST *st = WN_st(wn_lda);
01200 ST *base = st;
01201 Expand_ST_into_base_and_ofst(st, 0, &base, &ofst);
01202
01203
01204
01205
01206
01207
01208
01209 if (ST_class(st) != CLASS_BLOCK && TY_size(ST_type(st)) > 0) {
01210 pt->Set_ofst_kind(OFST_IS_FIXED);
01211 pt->Set_byte_ofst(ofst);
01212 pt->Set_byte_size(TY_size(ST_type(st)));
01213 }
01214 }
01215 } else if (OPERATOR_is_scalar_store (WN_operator(wn)) ||
01216 OPERATOR_is_scalar_load (WN_operator(wn))) {
01217 mINT64 ofst = 0;
01218 ST *st = WN_st(wn);
01219 ST *base = st;
01220 Expand_ST_into_base_and_ofst(st, 0, &base, &ofst);
01221
01222 if (ST_class(st) != CLASS_BLOCK && TY_size(ST_type(st)) > 0) {
01223 pt->Set_ofst_kind(OFST_IS_FIXED);
01224 pt->Set_byte_ofst(ofst);
01225 pt->Set_byte_size(TY_size(ST_type(st)));
01226 }
01227 }
01228 }
01229
01230
01231
01232
01233 ALIAS_RESULT Overlapped_base(const ALIAS_MANAGER *am, const WN *wn1, const WN *wn2)
01234 {
01235 IDTYPE id1 = am->Id(wn1);
01236 IDTYPE id2 = am->Id(wn2);
01237 if (id1 == 0 || id2 == 0)
01238 return POSSIBLY_ALIASED;
01239
01240 if (id1 == am->Preg_id() || id2 == am->Preg_id())
01241 return POSSIBLY_ALIASED;
01242
01243 POINTS_TO pt1, pt2;
01244 pt1.Copy_fully(am->Pt(id1));
01245 pt2.Copy_fully(am->Pt(id2));
01246
01247
01248
01249 if (pt1.Base_kind() == BASE_IS_DYNAMIC)
01250 pt1.Set_base_kind(BASE_IS_UNKNOWN);
01251 if (pt2.Base_kind() == BASE_IS_DYNAMIC)
01252 pt2.Set_base_kind(BASE_IS_UNKNOWN);
01253
01254
01255
01256
01257
01258 pt1.Set_ofst_kind(OFST_IS_UNKNOWN);
01259
01260 pt2.Set_ofst_kind(OFST_IS_UNKNOWN);
01261
01262 if (am->Rule()->Aliased_Memop(&pt1, &pt2, WN_object_ty(wn1), WN_object_ty(wn2))) {
01263 if (pt1.Same_base(&pt2)) {
01264
01265
01266 Check_range(&pt1, wn1);
01267 Check_range(&pt2, wn2);
01268
01269
01270
01271
01272
01273
01274 if (pt1.Base() != pt2.Base())
01275 return POSSIBLY_ALIASED;
01276
01277 if (pt1.Overlap(&pt2))
01278 return POSSIBLY_ALIASED;
01279 else
01280 return NOT_ALIASED;
01281 }
01282 return POSSIBLY_ALIASED;
01283 } else
01284 return NOT_ALIASED;
01285 }
01286
01287
01288
01289
01290 void Copy_alias_info(const ALIAS_MANAGER *am, WN *wn1, WN *wn2)
01291 {
01292 const OPCODE opc1 = WN_opcode(wn1);
01293 const OPCODE opc2 = WN_opcode(wn2);
01294
01295 WN_MAP32_Set(WN_MAP_ALIAS_CLASS, wn2,
01296 WN_MAP32_Get(WN_MAP_ALIAS_CLASS, wn1));
01297
01298 IDTYPE id = am->Id(wn1);
01299 if (id == 0) {
01300 OPERATOR opr = OPCODE_operator(opc1);
01301 if ((OPERATOR_is_scalar_load (opr) ||
01302 OPERATOR_is_scalar_store (opr)) &&
01303 ST_sclass(WN_st(wn1)) == SCLASS_REG) {
01304 id = am->Preg_id();
01305 am->Set_id(wn1, id);
01306 #if defined(TARG_SL)
01307 } else if (opr == OPR_PARM && !WN_Parm_By_Reference(wn1) && !WN_Parm_Dereference(wn1)) {
01308 #else
01309 } else if (opr == OPR_PARM && !WN_Parm_By_Reference(wn1)) {
01310 #endif
01311
01312 am->Set_id(wn2, 0);
01313 return;
01314 } else {
01315 am->Set_id(wn2, 0);
01316 return;
01317 }
01318 }
01319 am->Set_id(wn2, id);
01320
01321
01322 if ( OPCODE_is_load(opc1) && OPCODE_is_load(opc2) ) {
01323 am->Set_homing_load( wn2, am->Homing_load(wn1) );
01324 }
01325 else if ( OPCODE_is_store(opc1) && OPCODE_is_store(opc2) ) {
01326 am->Set_homing_store( wn2, am->Homing_store(wn1) );
01327 }
01328 }
01329
01330
01331
01332
01333 void Duplicate_alias_info(ALIAS_MANAGER *am, WN *wn1, WN *wn2)
01334 {
01335 const OPCODE opc1 = WN_opcode(wn1);
01336 const OPCODE opc2 = WN_opcode(wn2);
01337
01338 WN_MAP32_Set(WN_MAP_ALIAS_CLASS, wn2,
01339 WN_MAP32_Get(WN_MAP_ALIAS_CLASS, wn1));
01340
01341
01342 if ( OPCODE_is_load(opc1) && OPCODE_is_load(opc2) ) {
01343 am->Set_homing_load( wn2, am->Homing_load(wn1) );
01344 }
01345 else if ( OPCODE_is_store(opc1) && OPCODE_is_store(opc2) ) {
01346 am->Set_homing_store( wn2, am->Homing_store(wn1) );
01347 }
01348
01349 IDTYPE id = am->Id(wn1);
01350 if (id == 0) {
01351 OPERATOR opr = OPCODE_operator(opc1);
01352 if ((OPERATOR_is_scalar_load (opr) ||
01353 OPERATOR_is_scalar_store (opr)) &&
01354 ST_sclass(WN_st(wn1)) == SCLASS_REG) {
01355 id = am->Preg_id();
01356 am->Set_id(wn1, id);
01357 #if defined(TARG_SL)
01358 } else if (opr == OPR_PARM && !WN_Parm_By_Reference(wn1) && !WN_Parm_Dereference(wn1)) {
01359 #else
01360 } else if (opr == OPR_PARM && !WN_Parm_By_Reference(wn1)) {
01361 #endif
01362
01363 am->Set_id(wn2, 0);
01364 return;
01365 } else {
01366 am->Set_id(wn2, 0);
01367 return;
01368 }
01369 }
01370 if (id == am->Preg_id()) {
01371 am->Set_id(wn2, id);
01372 } else {
01373 POINTS_TO *pt = am->Pt(id);
01374 if (pt) {
01375 IDTYPE new_id = am->New_alias_id();
01376 POINTS_TO *npt = am->Pt(new_id);
01377 am->Set_id(wn2, new_id);
01378 npt->Copy_fully(pt);
01379 npt->Set_id(new_id);
01380 if (!npt->Base_is_fixed()) {
01381 npt->Set_base_kind(BASE_IS_UNKNOWN);
01382 }
01383 }
01384 }
01385 }
01386
01387
01388 void Note_Invalid_IP_Alias_Class(ALIAS_MANAGER *am, const WN *wn)
01389 {
01390 am->Note_invalid_ip_alias_class(wn);
01391 }
01392
01393 void Invalidate_Persistent_Alias_Info(ALIAS_MANAGER *am, WN *wn)
01394 {
01395 am->Erase_ip_alias_class_if_invalid(wn);
01396 if (WOPT_Enable_Restricted_Map && Restricted_map != NULL) {
01397 Restricted_map->Erase_entry_if_invalid(wn);
01398 }
01399 if (WN_opcode(wn) == OPC_BLOCK) {
01400 WN *wn2;
01401 for (wn2 = WN_first(wn);
01402 wn2 != NULL;
01403 wn2 = WN_next(wn2)) {
01404 Invalidate_Persistent_Alias_Info(am, wn2);
01405 }
01406 }
01407 else {
01408 for (INT i = 0; i < WN_kid_count(wn); i++) {
01409 Invalidate_Persistent_Alias_Info(am, WN_kid(wn, i));
01410 }
01411 }
01412 }
01413
01414
01415 void
01416 Create_vector_alias(ALIAS_MANAGER *am, WN *wn1, WN *wn2)
01417 {
01418 IDTYPE id1 = am->Id(wn1);
01419 if (id1 == 0) {
01420 am->Set_id(wn2, 0);
01421 return;
01422 }
01423 POINTS_TO *pt1 = am->Pt(id1);
01424 POINTS_TO *pt2 = am->New_points_to(wn2);
01425 pt2->Copy_fully(pt1);
01426 pt2->Set_ofst_kind(OFST_IS_UNKNOWN);
01427 }
01428
01429
01430
01431 void
01432 ALIAS_MANAGER::Print( const WN *wn, FILE *fp ) const
01433 {
01434 INT i;
01435 fprintf( fp, "\nFINAL WHIRL TREE DUMP WITH ALIAS ID:\n");
01436 fdump_tree_with_alias( fp, wn, _map, this );
01437
01438 fprintf( fp, "\nPOINTS_TO TABLE DUMP:\n");
01439 for ( i = Preg_id()+1; i <= Vec()->Lastidx(); i++) {
01440 fprintf( fp, " %d: ", i );
01441 Pt(i)->Print(fp);
01442 }
01443
01444 fprintf( fp, "\nALIAS ARC DUMP:\n");
01445 for ( i = Preg_id()+1; i <= Vec()->Lastidx(); i++) {
01446 fprintf( fp, "aliased_with<%d,{", i);
01447 for (INT32 oldid = Preg_id() + 1; oldid <= i; oldid++) {
01448 if (Rule()->Aliased_Memop(Pt(oldid), Pt(i), Pt(oldid)->Ty(), Pt(i)->Ty(), TRUE))
01449 fprintf( fp,"%d ", oldid);
01450 }
01451 fprintf( fp, "}>\n");
01452 }
01453
01454 if (WN_opcode(wn) == OPC_REGION) {
01455 RID *rid = REGION_get_rid(wn);
01456 Is_True(rid != NULL, ("ALIAS_MANAGER::Print, NULL RID"));
01457 if (!RID_TYPE_mp(rid) && !RID_TYPE_eh(rid)) {
01458 fprintf(fp, "\nREGION BOUNDARY SETS:\n");
01459 RID_set_print(fp, rid);
01460 }
01461 }
01462 }
01463
01464
01465
01466
01467 void Dump_alias_mgr(const struct ALIAS_MANAGER *am, const WN *tree, FILE *fp)
01468 {
01469 am->Print(tree, fp);
01470 }
01471
01472
01473
01474
01475 void Print_alias_info(char *buf, const ALIAS_MANAGER *am, const WN *wn)
01476 {
01477 IDTYPE alias_id = am->Id(wn);
01478
01479
01480 if (alias_id == 0) {
01481 buf[0] ='\0';
01482 return;
01483 }
01484
01485 POINTS_TO *pt = am->Pt(alias_id);
01486
01487 if (pt->Expr_kind() == EXPR_IS_ADDR && pt->Base_kind() == BASE_IS_FIXED) {
01488 if (pt->Ofst_kind() == OFST_IS_FIXED)
01489 sprintf(buf, "id:%d %s+0x%llx",
01490 alias_id,
01491 ST_class(pt->Base())==CLASS_VAR ? ST_name(pt->Base()) : "not_variable",
01492 pt->Byte_Ofst());
01493 else
01494 sprintf(buf, "id:%d %s", alias_id, ST_name(pt->Base()));
01495 } else if (pt->F_param() && pt->Based_sym() != NULL) {
01496 sprintf(buf, "id:%d parm:%s", alias_id, ST_name(pt->Based_sym()));
01497 } else if (pt->Unique_pt() && pt->Based_sym() != NULL) {
01498 sprintf(buf, "id:%d uniq:%s", alias_id, ST_name(pt->Based_sym()));
01499 } else
01500 sprintf(buf, "id:%d", alias_id);
01501 }
01502
01503
01504
01505
01506
01507
01508 BOOL ALIAS_MANAGER::Safe_to_speculate(const WN *wn) const
01509 {
01510 IDTYPE id = Id(wn);
01511
01512
01513 if (id == 0) {
01514 return FALSE;
01515 }
01516
01517 if (id == Preg_id()) {
01518 return TRUE;
01519 }
01520
01521
01522
01523
01524 POINTS_TO *pt = Pt(id);
01525 if (pt->Expr_kind() == EXPR_IS_ADDR &&
01526 pt->Base_kind() == BASE_IS_FIXED &&
01527 pt->Ofst_kind() == OFST_IS_FIXED &&
01528 pt->Safe_to_speculate()) {
01529 return TRUE;
01530 }
01531 return FALSE;
01532 }
01533
01534
01535 BOOL ALIAS_MANAGER::May_refer_to_alloca_mem(const WN *wn) const
01536 {
01537 IDTYPE alias_id = Id(wn);
01538 #ifdef KEY
01539
01540 if (alias_id == 0) {
01541 return FALSE;
01542 }
01543 #endif
01544 POINTS_TO *pt = Pt(alias_id);
01545 if (_trace) {
01546 fprintf(TFile, "--- Checking for pointing to alloca memory:\n");
01547 fdump_tree_with_alias(TFile, wn, _map, this);
01548 fprintf(TFile, " %s point to alloca mem\n",
01549 pt->Not_alloca_mem() ? "does not" : "may");
01550 }
01551 return !pt->Not_alloca_mem();
01552 }
01553
01554 BOOL May_refer_to_alloca_mem(const struct ALIAS_MANAGER *am, const WN *wn)
01555 {
01556 return am->May_refer_to_alloca_mem(wn);
01557 }
01558
01559
01560
01561
01562
01563
01564 extern "C"
01565 BOOL Homing_Load( const ALIAS_MANAGER *am, const WN *load_wn )
01566 {
01567 return am->Homing_load(load_wn);
01568 }
01569
01570 extern "C"
01571 void Set_Homing_Load( ALIAS_MANAGER *am, WN *load_wn )
01572 {
01573 am->Set_homing_load(load_wn,TRUE);
01574 }
01575
01576 extern "C"
01577 void Reset_Homing_Load( ALIAS_MANAGER *am, WN *load_wn )
01578 {
01579 am->Set_homing_load(load_wn,FALSE);
01580 }
01581
01582 extern "C"
01583 BOOL Homing_Store( const ALIAS_MANAGER *am, const WN *store_wn )
01584 {
01585 return am->Homing_store(store_wn);
01586 }
01587
01588 extern "C"
01589 void Set_Homing_Store( ALIAS_MANAGER *am, WN *store_wn )
01590 {
01591 am->Set_homing_store(store_wn,TRUE);
01592 }
01593
01594 extern "C"
01595 void Reset_Homing_Store( ALIAS_MANAGER *am, WN *store_wn )
01596 {
01597 am->Set_homing_store(store_wn,FALSE);
01598 }
01599
01600
01601
01602
01603
01604
01605
01606 extern "C"
01607 POINTS_TO *Points_to(ALIAS_MANAGER *am, WN *wn)
01608 {
01609 IDTYPE id = am->Id(wn);
01610 return am->Pt(id);
01611 }
01612
01613
01614
01615 extern "C"
01616 POINTS_TO *Points_to_copy(POINTS_TO *pt, MEM_POOL *rpool)
01617 {
01618 POINTS_TO *tmp = CXX_NEW(POINTS_TO, rpool);
01619 tmp->Copy_fully(pt);
01620 return tmp;
01621 }