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
00066
00067
00068
00069
00070
00071
00072
00073
00074
00075
00076
00077
00078
00079
00080
00081
00082
00083
00084
00085
00086
00087
00088
00089
00090
00091
00092
00093
00094
00095
00096
00097
00098
00099
00100
00101
00102
00103
00104
00105
00106
00107
00108
00109
00110
00111
00112
00113
00114
00115
00116
00117
00118
00119
00120
00121
00122
00123
00124
00125
00126
00127
00128
00129
00130
00131
00132
00133
00134
00135
00136
00137
00138
00139
00140
00141
00142
00143
00144
00145
00146
00147
00148
00149
00150
00151
00152
00153
00154
00155
00156
00157
00158
00159
00160
00161
00162
00163
00164
00165
00166
00167
00168
00169
00170
00171
00172
00173
00174
00175
00176
00177
00178
00179
00180
00181
00182
00183
00184
00185
00186
00187
00188
00189
00190
00191
00192
00193
00194
00195
00196
00197
00198
00199
00200
00201
00202
00203
00204
00205
00206
00207
00208
00209
00210
00211
00212
00213
00214
00215
00216
00217
00218
00219
00220
00221
00222
00223
00224
00225
00226
00227
00228
00229
00230
00231
00232
00233
00234
00235
00236
00237
00238
00239
00240
00241
00242
00243
00244
00245
00246
00247
00248
00249
00250
00251
00252
00253
00254
00255
00256
00257
00258
00259
00260
00261
00262
00263
00264
00265
00266
00267
00268
00269
00270
00271
00272
00273
00274
00275
00276
00277
00278
00279
00280
00281
00282
00283
00284
00285
00286
00287
00288
00289
00290
00291
00292
00293
00294
00295
00296
00297
00298
00299
00300
00301
00302
00303
00304
00305
00306
00307
00308
00309
00310
00311
00312
00313
00314
00315
00316
00317
00318
00319
00320
00321
00322
00323
00324
00325
00326
00327
00328
00329
00330
00331
00332
00333
00334
00335
00336
00337
00338
00339
00340
00341
00342
00343
00344
00345
00346
00347
00348
00349
00350
00351
00352
00353
00354
00355
00356
00357
00358
00359
00360
00361
00362
00363
00364
00365
00366
00367
00368
00369
00370
00371
00372
00373
00374
00375
00376
00377
00378
00379
00380
00381
00382
00383
00384
00385
00386
00387
00388
00389
00390
00391
00392
00393
00394
00395
00396
00397
00398
00399
00400
00401
00402
00403
00404
00405
00406
00407
00408
00409
00410
00411
00412
00413
00414
00415
00416
00417
00418
00419
00420
00421
00422
00423
00424
00425
00426
00427
00428
00429
00430
00431
00432
00433
00434
00435
00436
00437
00438
00439
00440
00441
00442
00443
00444
00445
00446
00447
00448
00449
00450
00451
00452
00453
00454
00455
00456
00457
00458
00459
00460
00461
00462
00463
00464
00465 #include "config.h"
00466 #include "system.h"
00467 #include "coretypes.h"
00468 #include "tm.h"
00469
00470 #include "obstack.h"
00471 #include "rtl.h"
00472 #include "output.h"
00473 #include "df.h"
00474 #include "insn-config.h"
00475 #include "recog.h"
00476 #include "expr.h"
00477 #include "splay-tree.h"
00478 #include "hashtab.h"
00479 #include "regs.h"
00480 #include "timevar.h"
00481 #include "tree-pass.h"
00482
00483
00484 enum entry_type {
00485 NOT_RELEVANT,
00486 SIGN_EXTENDED_DEF,
00487 ZERO_EXTENDED_DEF,
00488 EXTENDED_DEF,
00489 RELEVANT_USE
00490 };
00491
00492
00493 enum extension_type {
00494 DEF_EXTENSION,
00495 EXPLICIT_DEF_EXTENSION,
00496 IMPLICIT_DEF_EXTENSION,
00497 USE_EXTENSION
00498 };
00499
00500
00501
00502
00503
00504
00505
00506 struct see_entry_extra_info
00507 {
00508
00509 enum entry_type relevancy;
00510
00511
00512 enum entry_type local_relevancy;
00513
00514 enum machine_mode source_mode;
00515
00516
00517 enum machine_mode local_source_mode;
00518
00519
00520 enum machine_mode source_mode_signed;
00521
00522
00523 enum machine_mode source_mode_unsigned;
00524 };
00525
00526
00527
00528
00529 struct see_ref_s
00530 {
00531
00532 unsigned int luid;
00533
00534 rtx insn;
00535
00536
00537 rtx merged_insn;
00538
00539
00540 htab_t unmerged_def_se_hash;
00541
00542
00543 htab_t merged_def_se_hash;
00544
00545 htab_t use_se_hash;
00546 };
00547
00548
00549
00550
00551 struct see_register_properties
00552 {
00553
00554 unsigned int regno;
00555
00556
00557 int last_def;
00558
00559
00560 int first_se_before_any_def;
00561
00562
00563 int first_se_after_last_def;
00564 };
00565
00566
00567
00568
00569 struct see_occr
00570 {
00571
00572 struct see_occr *next;
00573
00574 rtx insn;
00575 int block_num;
00576 };
00577
00578
00579
00580
00581 struct see_pre_extension_expr
00582 {
00583
00584 rtx se_insn;
00585
00586 int bitmap_index;
00587
00588
00589
00590
00591
00592 struct see_occr *antic_occr;
00593
00594
00595
00596
00597 struct see_occr *avail_occr;
00598 };
00599
00600
00601 struct see_replace_data
00602 {
00603 rtx from;
00604 rtx to;
00605 };
00606
00607
00608 struct see_mentioned_reg_data
00609 {
00610 rtx reg;
00611 bool mentioned;
00612 };
00613
00614
00615
00616 static struct df *df = NULL;
00617
00618
00619 static struct web_entry *def_entry = NULL;
00620
00621
00622 static struct web_entry *use_entry = NULL;
00623
00624
00625
00626
00627
00628 static splay_tree *see_bb_splay_ar = NULL;
00629
00630
00631
00632 static htab_t *see_bb_hash_ar = NULL;
00633
00634
00635 static htab_t see_pre_extension_hash = NULL;
00636
00637
00638
00639 static sbitmap *transp = NULL;
00640
00641 static sbitmap *comp = NULL;
00642
00643 static sbitmap *antloc = NULL;
00644
00645 static sbitmap *ae_kill = NULL;
00646
00647 static sbitmap *pre_insert_map = NULL;
00648
00649 static sbitmap *pre_delete_map = NULL;
00650
00651 static struct edge_list *edge_list = NULL;
00652
00653 static int last_bb;
00654
00655 static unsigned int uses_num;
00656
00657 static unsigned int defs_num;
00658
00659 #define ENTRY_EI(ENTRY) ((struct see_entry_extra_info *) (ENTRY)->extra_info)
00660
00661
00662
00663
00664
00665
00666
00667
00668
00669
00670 static rtx
00671 see_get_extension_reg (rtx extension, bool return_dest_reg)
00672 {
00673 rtx set, rhs, lhs;
00674 rtx reg1 = NULL;
00675 rtx reg2 = NULL;
00676
00677
00678 if (GET_CODE (PATTERN (extension)) == PARALLEL)
00679 return NULL;
00680
00681 set = single_set (extension);
00682 if (!set)
00683 return NULL;
00684 lhs = SET_DEST (set);
00685 rhs = SET_SRC (set);
00686
00687 if (REG_P (lhs))
00688 reg1 = lhs;
00689 else if (REG_P (SUBREG_REG (lhs)))
00690 reg1 = SUBREG_REG (lhs);
00691 else
00692 return NULL;
00693
00694 if (GET_CODE (rhs) != SIGN_EXTEND && GET_CODE (rhs) != ZERO_EXTEND)
00695 return NULL;
00696
00697 rhs = XEXP (rhs, 0);
00698 if (REG_P (rhs))
00699 reg2 = rhs;
00700 else if (REG_P (SUBREG_REG (rhs)))
00701 reg2 = SUBREG_REG (rhs);
00702 else
00703 return NULL;
00704
00705 if (return_dest_reg)
00706 return reg1;
00707 return reg2;
00708 }
00709
00710
00711
00712
00713
00714
00715
00716
00717
00718 static enum rtx_code
00719 see_get_extension_data (rtx extension, enum machine_mode *source_mode)
00720 {
00721 rtx rhs, lhs, set;
00722
00723 if (!extension || !INSN_P (extension))
00724 return UNKNOWN;
00725
00726
00727 if (GET_CODE (PATTERN (extension)) == PARALLEL)
00728 return NOT_RELEVANT;
00729
00730 set = single_set (extension);
00731 if (!set)
00732 return NOT_RELEVANT;
00733 rhs = SET_SRC (set);
00734 lhs = SET_DEST (set);
00735
00736
00737
00738 if (!REG_P (lhs) && !SUBREG_REG (lhs))
00739 return UNKNOWN;
00740
00741 if (GET_CODE (rhs) != SIGN_EXTEND && GET_CODE (rhs) != ZERO_EXTEND)
00742 return UNKNOWN;
00743
00744 if (!REG_P (XEXP (rhs, 0))
00745 && !(GET_CODE (XEXP (rhs, 0)) == SUBREG
00746 && REG_P (SUBREG_REG (XEXP (rhs, 0)))))
00747 return UNKNOWN;
00748
00749 *source_mode = GET_MODE (XEXP (rhs, 0));
00750
00751 if (GET_CODE (rhs) == SIGN_EXTEND)
00752 return SIGN_EXTEND;
00753 return ZERO_EXTEND;
00754 }
00755
00756
00757
00758
00759
00760
00761
00762
00763
00764
00765 static rtx
00766 see_gen_normalized_extension (rtx reg, enum rtx_code extension_code,
00767 enum machine_mode mode)
00768 {
00769 rtx subreg, insn;
00770 rtx extension = NULL;
00771
00772 if (!reg
00773 || !REG_P (reg)
00774 || (extension_code != SIGN_EXTEND && extension_code != ZERO_EXTEND))
00775 return NULL;
00776
00777 subreg = gen_lowpart_SUBREG (mode, reg);
00778 if (extension_code == SIGN_EXTEND)
00779 extension = gen_rtx_SIGN_EXTEND (GET_MODE (reg), subreg);
00780 else
00781 extension = gen_rtx_ZERO_EXTEND (GET_MODE (reg), subreg);
00782
00783 start_sequence ();
00784 emit_insn (gen_rtx_SET (VOIDmode, reg, extension));
00785 insn = get_insns ();
00786 end_sequence ();
00787
00788 if (insn_invalid_p (insn))
00789
00790
00791 return NULL;
00792 return insn;
00793 }
00794
00795
00796
00797
00798
00799
00800
00801
00802
00803
00804
00805
00806
00807
00808
00809
00810 static int
00811 eq_descriptor_pre_extension (const void *p1, const void *p2)
00812 {
00813 const struct see_pre_extension_expr *extension1 = p1;
00814 const struct see_pre_extension_expr *extension2 = p2;
00815 rtx set1 = single_set (extension1->se_insn);
00816 rtx set2 = single_set (extension2->se_insn);
00817 rtx rhs1, rhs2;
00818
00819 gcc_assert (set1 && set2);
00820 rhs1 = SET_SRC (set1);
00821 rhs2 = SET_SRC (set2);
00822
00823 return rtx_equal_p (rhs1, rhs2);
00824 }
00825
00826
00827
00828
00829
00830
00831 static hashval_t
00832 hash_descriptor_pre_extension (const void *p)
00833 {
00834 const struct see_pre_extension_expr *extension = p;
00835 rtx set = single_set (extension->se_insn);
00836 rtx rhs;
00837
00838 gcc_assert (set);
00839 rhs = SET_SRC (set);
00840
00841 return hash_rtx (rhs, GET_MODE (rhs), 0, NULL, 0);
00842 }
00843
00844
00845
00846
00847
00848
00849 static void
00850 hash_del_pre_extension (void *p)
00851 {
00852 struct see_pre_extension_expr *extension = p;
00853 struct see_occr *curr_occr = extension->antic_occr;
00854 struct see_occr *next_occr = NULL;
00855
00856
00857 while (curr_occr)
00858 {
00859 next_occr = curr_occr->next;
00860 free (curr_occr);
00861 curr_occr = next_occr;
00862 }
00863
00864
00865 curr_occr = extension->avail_occr;
00866 while (curr_occr)
00867 {
00868 next_occr = curr_occr->next;
00869 free (curr_occr);
00870 curr_occr = next_occr;
00871 }
00872
00873
00874 free (extension);
00875 }
00876
00877
00878
00879
00880
00881
00882
00883
00884
00885
00886
00887 static int
00888 eq_descriptor_properties (const void *p1, const void *p2)
00889 {
00890 const struct see_register_properties *curr_prop1 = p1;
00891 const struct see_register_properties *curr_prop2 = p2;
00892
00893 return curr_prop1->regno == curr_prop2->regno;
00894 }
00895
00896
00897
00898
00899
00900 static hashval_t
00901 hash_descriptor_properties (const void *p)
00902 {
00903 const struct see_register_properties *curr_prop = p;
00904 return curr_prop->regno;
00905 }
00906
00907
00908
00909 static void
00910 hash_del_properties (void *p)
00911 {
00912 struct see_register_properties *curr_prop = p;
00913 free (curr_prop);
00914 }
00915
00916
00917
00918
00919
00920
00921
00922
00923
00924
00925
00926
00927
00928
00929
00930
00931
00932
00933
00934
00935 static int
00936 eq_descriptor_extension (const void *p1, const void *p2)
00937 {
00938 const rtx insn = (rtx) p1;
00939 const rtx element = (rtx) p2;
00940 rtx set1 = single_set (insn);
00941 rtx dest_reg1;
00942 rtx set2 = NULL;
00943 rtx dest_reg2 = NULL;
00944
00945 gcc_assert (set1 && element && (REG_P (element) || INSN_P (element)));
00946
00947 dest_reg1 = SET_DEST (set1);
00948
00949 if (INSN_P (element))
00950 {
00951 set2 = single_set (element);
00952 dest_reg2 = SET_DEST (set2);
00953 }
00954 else
00955 dest_reg2 = element;
00956
00957 return REGNO (dest_reg1) == REGNO (dest_reg2);
00958 }
00959
00960
00961
00962
00963
00964 static hashval_t
00965 hash_descriptor_extension (const void *p)
00966 {
00967 const rtx r = (rtx) p;
00968 rtx set, lhs;
00969
00970 if (r && REG_P (r))
00971 return REGNO (r);
00972
00973 gcc_assert (r && INSN_P (r));
00974 set = single_set (r);
00975 gcc_assert (set);
00976 lhs = SET_DEST (set);
00977 return REGNO (lhs);
00978 }
00979
00980
00981
00982
00983
00984
00985
00986 static void
00987 see_free_ref_s (splay_tree_value value)
00988 {
00989 struct see_ref_s *ref_s = (struct see_ref_s *)value;
00990
00991 if (ref_s->unmerged_def_se_hash)
00992 htab_delete (ref_s->unmerged_def_se_hash);
00993 if (ref_s->merged_def_se_hash)
00994 htab_delete (ref_s->merged_def_se_hash);
00995 if (ref_s->use_se_hash)
00996 htab_delete (ref_s->use_se_hash);
00997 free (ref_s);
00998 }
00999
01000
01001
01002
01003
01004
01005
01006
01007
01008
01009
01010
01011
01012 static struct see_pre_extension_expr *
01013 see_seek_pre_extension_expr (rtx extension, enum extension_type type)
01014 {
01015 struct see_pre_extension_expr **slot_pre_exp, temp_pre_exp;
01016 rtx dest_extension_reg = see_get_extension_reg (extension, 1);
01017 enum rtx_code extension_code;
01018 enum machine_mode source_extension_mode;
01019
01020 if (type == DEF_EXTENSION)
01021 {
01022 extension_code = see_get_extension_data (extension,
01023 &source_extension_mode);
01024 gcc_assert (extension_code != UNKNOWN);
01025 extension =
01026 see_gen_normalized_extension (dest_extension_reg, extension_code,
01027 source_extension_mode);
01028 }
01029 temp_pre_exp.se_insn = extension;
01030 slot_pre_exp =
01031 (struct see_pre_extension_expr **) htab_find_slot (see_pre_extension_hash,
01032 &temp_pre_exp, INSERT);
01033 if (*slot_pre_exp == NULL)
01034
01035
01036 {
01037 (*slot_pre_exp) = xmalloc (sizeof (struct see_pre_extension_expr));
01038 (*slot_pre_exp)->se_insn = extension;
01039 (*slot_pre_exp)->bitmap_index =
01040 (htab_elements (see_pre_extension_hash) - 1);
01041 (*slot_pre_exp)->antic_occr = NULL;
01042 (*slot_pre_exp)->avail_occr = NULL;
01043 return NULL;
01044 }
01045 return *slot_pre_exp;
01046 }
01047
01048
01049
01050
01051
01052
01053
01054
01055
01056
01057
01058
01059 static bool
01060 see_update_leader_extra_info (struct web_entry *first, struct web_entry *second)
01061 {
01062 struct see_entry_extra_info *first_ei, *second_ei;
01063
01064 first = unionfind_root (first);
01065 second = unionfind_root (second);
01066
01067 if (unionfind_union (first, second))
01068 return true;
01069
01070 first_ei = (struct see_entry_extra_info *) first->extra_info;
01071 second_ei = (struct see_entry_extra_info *) second->extra_info;
01072
01073 gcc_assert (first_ei && second_ei);
01074
01075 if (second_ei->relevancy == NOT_RELEVANT)
01076 {
01077 first_ei->relevancy = NOT_RELEVANT;
01078 return false;
01079 }
01080 switch (first_ei->relevancy)
01081 {
01082 case NOT_RELEVANT:
01083 break;
01084 case RELEVANT_USE:
01085 switch (second_ei->relevancy)
01086 {
01087 case RELEVANT_USE:
01088 break;
01089 case EXTENDED_DEF:
01090 first_ei->relevancy = second_ei->relevancy;
01091 first_ei->source_mode_signed = second_ei->source_mode_signed;
01092 first_ei->source_mode_unsigned = second_ei->source_mode_unsigned;
01093 break;
01094 case SIGN_EXTENDED_DEF:
01095 case ZERO_EXTENDED_DEF:
01096 first_ei->relevancy = second_ei->relevancy;
01097 first_ei->source_mode = second_ei->source_mode;
01098 break;
01099 default:
01100 gcc_unreachable ();
01101 }
01102 break;
01103 case SIGN_EXTENDED_DEF:
01104 switch (second_ei->relevancy)
01105 {
01106 case SIGN_EXTENDED_DEF:
01107
01108 first_ei->source_mode =
01109 (first_ei->source_mode > second_ei->source_mode) ?
01110 first_ei->source_mode : second_ei->source_mode;
01111 break;
01112 case RELEVANT_USE:
01113 break;
01114 case ZERO_EXTENDED_DEF:
01115
01116 first_ei->relevancy = NOT_RELEVANT;
01117 break;
01118 case EXTENDED_DEF:
01119 if (second_ei->source_mode_signed == MAX_MACHINE_MODE)
01120 first_ei->relevancy = NOT_RELEVANT;
01121 else
01122
01123 first_ei->source_mode =
01124 (first_ei->source_mode > second_ei->source_mode_signed) ?
01125 first_ei->source_mode : second_ei->source_mode_signed;
01126 break;
01127 default:
01128 gcc_unreachable ();
01129 }
01130 break;
01131
01132 case ZERO_EXTENDED_DEF:
01133 switch (second_ei->relevancy)
01134 {
01135 case SIGN_EXTENDED_DEF:
01136
01137 first_ei->relevancy = NOT_RELEVANT;
01138 break;
01139 case RELEVANT_USE:
01140 break;
01141 case ZERO_EXTENDED_DEF:
01142
01143 first_ei->source_mode =
01144 (first_ei->source_mode > second_ei->source_mode) ?
01145 first_ei->source_mode : second_ei->source_mode;
01146 break;
01147 case EXTENDED_DEF:
01148 if (second_ei->source_mode_unsigned == MAX_MACHINE_MODE)
01149 first_ei->relevancy = NOT_RELEVANT;
01150 else
01151
01152 first_ei->source_mode =
01153 (first_ei->source_mode > second_ei->source_mode_unsigned) ?
01154 first_ei->source_mode : second_ei->source_mode_unsigned;
01155 break;
01156 default:
01157 gcc_unreachable ();
01158 }
01159 break;
01160 case EXTENDED_DEF:
01161 if (first_ei->source_mode_signed != MAX_MACHINE_MODE
01162 && first_ei->source_mode_unsigned != MAX_MACHINE_MODE)
01163 {
01164 switch (second_ei->relevancy)
01165 {
01166 case SIGN_EXTENDED_DEF:
01167 first_ei->relevancy = SIGN_EXTENDED_DEF;
01168 first_ei->source_mode =
01169 (first_ei->source_mode_signed > second_ei->source_mode) ?
01170 first_ei->source_mode_signed : second_ei->source_mode;
01171 break;
01172 case RELEVANT_USE:
01173 break;
01174 case ZERO_EXTENDED_DEF:
01175 first_ei->relevancy = ZERO_EXTENDED_DEF;
01176 first_ei->source_mode =
01177 (first_ei->source_mode_unsigned > second_ei->source_mode) ?
01178 first_ei->source_mode_unsigned : second_ei->source_mode;
01179 break;
01180 case EXTENDED_DEF:
01181 if (second_ei->source_mode_unsigned != MAX_MACHINE_MODE)
01182 first_ei->source_mode_unsigned =
01183 (first_ei->source_mode_unsigned >
01184 second_ei->source_mode_unsigned) ?
01185 first_ei->source_mode_unsigned :
01186 second_ei->source_mode_unsigned;
01187 if (second_ei->source_mode_signed != MAX_MACHINE_MODE)
01188 first_ei->source_mode_signed =
01189 (first_ei->source_mode_signed >
01190 second_ei->source_mode_signed) ?
01191 first_ei->source_mode_signed : second_ei->source_mode_signed;
01192 break;
01193 default:
01194 gcc_unreachable ();
01195 }
01196 }
01197 else if (first_ei->source_mode_signed == MAX_MACHINE_MODE)
01198 {
01199 gcc_assert (first_ei->source_mode_unsigned != MAX_MACHINE_MODE);
01200 switch (second_ei->relevancy)
01201 {
01202 case SIGN_EXTENDED_DEF:
01203 first_ei->relevancy = NOT_RELEVANT;
01204 break;
01205 case RELEVANT_USE:
01206 break;
01207 case ZERO_EXTENDED_DEF:
01208 first_ei->relevancy = ZERO_EXTENDED_DEF;
01209 first_ei->source_mode =
01210 (first_ei->source_mode_unsigned > second_ei->source_mode) ?
01211 first_ei->source_mode_unsigned : second_ei->source_mode;
01212 break;
01213 case EXTENDED_DEF:
01214 if (second_ei->source_mode_unsigned == MAX_MACHINE_MODE)
01215 first_ei->relevancy = NOT_RELEVANT;
01216 else
01217 first_ei->source_mode_unsigned =
01218 (first_ei->source_mode_unsigned >
01219 second_ei->source_mode_unsigned) ?
01220 first_ei->source_mode_unsigned :
01221 second_ei->source_mode_unsigned;
01222 break;
01223 default:
01224 gcc_unreachable ();
01225 }
01226 }
01227 else
01228 {
01229 gcc_assert (first_ei->source_mode_unsigned == MAX_MACHINE_MODE);
01230 gcc_assert (first_ei->source_mode_signed != MAX_MACHINE_MODE);
01231 switch (second_ei->relevancy)
01232 {
01233 case SIGN_EXTENDED_DEF:
01234 first_ei->relevancy = SIGN_EXTENDED_DEF;
01235 first_ei->source_mode =
01236 (first_ei->source_mode_signed > second_ei->source_mode) ?
01237 first_ei->source_mode_signed : second_ei->source_mode;
01238 break;
01239 case RELEVANT_USE:
01240 break;
01241 case ZERO_EXTENDED_DEF:
01242 first_ei->relevancy = NOT_RELEVANT;
01243 break;
01244 case EXTENDED_DEF:
01245 if (second_ei->source_mode_signed == MAX_MACHINE_MODE)
01246 first_ei->relevancy = NOT_RELEVANT;
01247 else
01248 first_ei->source_mode_signed =
01249 (first_ei->source_mode_signed >
01250 second_ei->source_mode_signed) ?
01251 first_ei->source_mode_signed : second_ei->source_mode_signed;
01252 break;
01253 default:
01254 gcc_unreachable ();
01255 }
01256 }
01257 break;
01258 default:
01259
01260 gcc_unreachable ();
01261 }
01262
01263 return false;
01264 }
01265
01266
01267
01268
01269 static void
01270 see_free_data_structures (void)
01271 {
01272 int i;
01273 unsigned int j;
01274
01275
01276 if (transp)
01277 {
01278 sbitmap_vector_free (transp);
01279 transp = NULL;
01280 sbitmap_vector_free (comp);
01281 comp = NULL;
01282 sbitmap_vector_free (antloc);
01283 antloc = NULL;
01284 sbitmap_vector_free (ae_kill);
01285 ae_kill = NULL;
01286 }
01287 if (pre_insert_map)
01288 {
01289 sbitmap_vector_free (pre_insert_map);
01290 pre_insert_map = NULL;
01291 }
01292 if (pre_delete_map)
01293 {
01294 sbitmap_vector_free (pre_delete_map);
01295 pre_delete_map = NULL;
01296 }
01297 if (edge_list)
01298 {
01299 free_edge_list (edge_list);
01300 edge_list = NULL;
01301 }
01302
01303
01304 htab_delete (see_pre_extension_hash);
01305
01306
01307 for (i = 0; i < last_bb; i++)
01308 if (see_bb_hash_ar[i])
01309 htab_delete (see_bb_hash_ar[i]);
01310 free (see_bb_hash_ar);
01311
01312
01313 for (i = 0; i < last_bb; i++)
01314 if (see_bb_splay_ar[i])
01315 splay_tree_delete (see_bb_splay_ar[i]);
01316 free (see_bb_splay_ar);
01317
01318
01319 for (j = 0; j < defs_num; j++)
01320 free (def_entry[j].extra_info);
01321 free (def_entry);
01322 for (j = 0; j < uses_num; j++)
01323 free (use_entry[j].extra_info);
01324 free (use_entry);
01325 }
01326
01327
01328
01329
01330 static void
01331 see_initialize_data_structures (void)
01332 {
01333
01334 df = df_init (DF_HARD_REGS | DF_EQUIV_NOTES | DF_SUBREGS);
01335 df_rd_add_problem (df, 0);
01336 df_chain_add_problem (df, DF_DU_CHAIN | DF_UD_CHAIN);
01337 df_analyze (df);
01338
01339 if (dump_file)
01340 df_dump (df, dump_file);
01341
01342
01343 last_bb = last_basic_block;
01344
01345 uses_num = DF_USES_SIZE (df);
01346
01347 defs_num = DF_DEFS_SIZE (df);
01348
01349
01350 def_entry = xcalloc (defs_num, sizeof (struct web_entry));
01351 use_entry = xcalloc (uses_num, sizeof (struct web_entry));
01352
01353
01354
01355 see_bb_splay_ar = xcalloc (last_bb, sizeof (splay_tree));
01356
01357
01358
01359 see_bb_hash_ar = xcalloc (last_bb, sizeof (htab_t));
01360
01361
01362
01363 see_pre_extension_hash = htab_create (10,
01364 hash_descriptor_pre_extension,
01365 eq_descriptor_pre_extension,
01366 hash_del_pre_extension);
01367 }
01368
01369
01370
01371
01372
01373
01374
01375
01376
01377 static void
01378 see_mentioned_reg (rtx *x, void *data)
01379 {
01380 struct see_mentioned_reg_data *d
01381 = (struct see_mentioned_reg_data *) data;
01382
01383 if (reg_mentioned_p (d->reg, *x))
01384 d->mentioned = true;
01385 }
01386
01387
01388
01389
01390
01391
01392
01393
01394
01395
01396
01397
01398
01399
01400 static bool
01401 see_want_to_be_merged_with_extension (rtx ref, rtx extension,
01402 enum extension_type type)
01403 {
01404 rtx pat;
01405 rtx dest_extension_reg = see_get_extension_reg (extension, 1);
01406 rtx source_extension_reg = see_get_extension_reg (extension, 0);
01407 enum rtx_code code;
01408 struct see_mentioned_reg_data d;
01409 int i;
01410
01411 pat = PATTERN (ref);
01412 code = GET_CODE (pat);
01413
01414 if (code == PARALLEL)
01415 {
01416 for (i = 0; i < XVECLEN (pat, 0); i++)
01417 {
01418 rtx sub = XVECEXP (pat, 0, i);
01419
01420 if (GET_CODE (sub) == SET
01421 && (REG_P (SET_DEST (sub))
01422 || (GET_CODE (SET_DEST (sub)) == SUBREG
01423 && REG_P (SUBREG_REG (SET_DEST (sub)))))
01424 && (REG_P (SET_SRC (sub))
01425 || (GET_CODE (SET_SRC (sub)) == SUBREG
01426 && REG_P (SUBREG_REG (SET_SRC (sub))))))
01427 {
01428
01429 if (type == DEF_EXTENSION
01430 && reg_mentioned_p (source_extension_reg, SET_DEST (sub)))
01431 return false;
01432 }
01433 else
01434 {
01435
01436
01437 if (type == USE_EXTENSION)
01438 {
01439 d.reg = dest_extension_reg;
01440 d.mentioned = false;
01441 note_uses (&sub, see_mentioned_reg, &d);
01442 if (d.mentioned)
01443 return true;
01444 }
01445 }
01446 }
01447 if (type == USE_EXTENSION)
01448 return false;
01449 }
01450 else
01451 {
01452 if (code == SET
01453 && (REG_P (SET_DEST (pat))
01454 || (GET_CODE (SET_DEST (pat)) == SUBREG
01455 && REG_P (SUBREG_REG (SET_DEST (pat)))))
01456 && (REG_P (SET_SRC (pat))
01457 || (GET_CODE (SET_SRC (pat)) == SUBREG
01458 && REG_P (SUBREG_REG (SET_SRC (pat))))))
01459
01460 return false;
01461 }
01462
01463 return true;
01464 }
01465
01466
01467
01468
01469
01470
01471
01472
01473 static int
01474 see_print_register_properties (void **slot, void *b ATTRIBUTE_UNUSED)
01475 {
01476 struct see_register_properties *prop = *slot;
01477
01478 gcc_assert (prop);
01479 fprintf (dump_file, "Property found for register %d\n", prop->regno);
01480 return 1;
01481 }
01482
01483
01484
01485
01486
01487
01488
01489
01490 static int
01491 see_print_pre_extension_expr (void **slot, void *b ATTRIBUTE_UNUSED)
01492 {
01493 struct see_pre_extension_expr *pre_extension = *slot;
01494
01495 gcc_assert (pre_extension
01496 && pre_extension->se_insn
01497 && INSN_P (pre_extension->se_insn));
01498
01499 fprintf (dump_file, "Index %d for:\n", pre_extension->bitmap_index);
01500 print_rtl_single (dump_file, pre_extension->se_insn);
01501
01502 return 1;
01503 }
01504
01505
01506
01507
01508
01509
01510
01511
01512
01513
01514
01515 static int
01516 see_delete_merged_def_extension (void **slot, void *b ATTRIBUTE_UNUSED)
01517 {
01518 rtx def_se = *slot;
01519
01520 if (dump_file)
01521 {
01522 fprintf (dump_file, "Deleting merged def extension:\n");
01523 print_rtl_single (dump_file, def_se);
01524 }
01525
01526 if (INSN_DELETED_P (def_se))
01527
01528
01529 return 1;
01530
01531 delete_insn (def_se);
01532 return 1;
01533 }
01534
01535
01536
01537
01538
01539
01540
01541
01542
01543 static int
01544 see_delete_unmerged_def_extension (void **slot, void *b ATTRIBUTE_UNUSED)
01545 {
01546 rtx def_se = *slot;
01547
01548 if (dump_file)
01549 {
01550 fprintf (dump_file, "Deleting unmerged def extension:\n");
01551 print_rtl_single (dump_file, def_se);
01552 }
01553
01554 delete_insn (def_se);
01555 return 1;
01556 }
01557
01558
01559
01560
01561
01562
01563
01564
01565
01566 static int
01567 see_emit_use_extension (void **slot, void *b)
01568 {
01569 rtx use_se = *slot;
01570 struct see_ref_s *curr_ref_s = (struct see_ref_s *) b;
01571
01572 if (INSN_DELETED_P (use_se))
01573
01574
01575 return 1;
01576
01577 if (dump_file)
01578 {
01579 fprintf (dump_file, "Inserting use extension:\n");
01580 print_rtl_single (dump_file, use_se);
01581 }
01582
01583 add_insn_before (use_se, curr_ref_s->insn);
01584
01585 return 1;
01586 }
01587
01588
01589
01590
01591
01592
01593
01594
01595
01596
01597
01598
01599
01600 static int
01601 see_commit_ref_changes (splay_tree_node stn,
01602 void *data ATTRIBUTE_UNUSED)
01603 {
01604 htab_t use_se_hash = ((struct see_ref_s *) (stn->value))->use_se_hash;
01605 htab_t unmerged_def_se_hash =
01606 ((struct see_ref_s *) (stn->value))->unmerged_def_se_hash;
01607 htab_t merged_def_se_hash =
01608 ((struct see_ref_s *) (stn->value))->merged_def_se_hash;
01609 rtx ref = ((struct see_ref_s *) (stn->value))->insn;
01610 rtx merged_ref = ((struct see_ref_s *) (stn->value))->merged_insn;
01611
01612
01613 if (use_se_hash)
01614 htab_traverse_noresize (use_se_hash, see_emit_use_extension,
01615 (PTR) (stn->value));
01616
01617
01618 if (unmerged_def_se_hash)
01619 htab_traverse (unmerged_def_se_hash, see_delete_unmerged_def_extension,
01620 (PTR) (stn->value));
01621
01622 if (merged_def_se_hash)
01623 htab_traverse (merged_def_se_hash, see_delete_merged_def_extension,
01624 (PTR) (stn->value));
01625
01626
01627
01628 if (merged_ref && !INSN_DELETED_P (ref))
01629 {
01630 if (dump_file)
01631 {
01632 fprintf (dump_file, "Replacing orig reference:\n");
01633 print_rtl_single (dump_file, ref);
01634 fprintf (dump_file, "With merged reference:\n");
01635 print_rtl_single (dump_file, merged_ref);
01636 }
01637 emit_insn_after (merged_ref, ref);
01638 delete_insn (ref);
01639 }
01640
01641
01642 return 0;
01643 }
01644
01645
01646
01647
01648
01649
01650
01651
01652
01653 static bool
01654 see_pre_insert_extensions (struct see_pre_extension_expr **index_map)
01655 {
01656 int num_edges = NUM_EDGES (edge_list);
01657 int set_size = pre_insert_map[0]->size;
01658 size_t pre_extension_num = htab_elements (see_pre_extension_hash);
01659
01660 int did_insert = 0;
01661 int e;
01662 int i;
01663 int j;
01664
01665 for (e = 0; e < num_edges; e++)
01666 {
01667 int indx;
01668 basic_block bb = INDEX_EDGE_PRED_BB (edge_list, e);
01669
01670 for (i = indx = 0; i < set_size; i++, indx += SBITMAP_ELT_BITS)
01671 {
01672 SBITMAP_ELT_TYPE insert = pre_insert_map[e]->elms[i];
01673
01674 for (j = indx; insert && j < (int) pre_extension_num;
01675 j++, insert >>= 1)
01676 if (insert & 1)
01677 {
01678 struct see_pre_extension_expr *expr = index_map[j];
01679 int idx = expr->bitmap_index;
01680 rtx se_insn = NULL;
01681 edge eg = INDEX_EDGE (edge_list, e);
01682
01683 start_sequence ();
01684 emit_insn (PATTERN (expr->se_insn));
01685 se_insn = get_insns ();
01686 end_sequence ();
01687
01688 if (eg->flags & EDGE_ABNORMAL)
01689 {
01690 rtx new_insn = NULL;
01691
01692 new_insn = insert_insn_end_bb_new (se_insn, bb);
01693 gcc_assert (new_insn && INSN_P (new_insn));
01694
01695 if (dump_file)
01696 {
01697 fprintf (dump_file,
01698 "PRE: end of bb %d, insn %d, ",
01699 bb->index, INSN_UID (new_insn));
01700 fprintf (dump_file,
01701 "inserting expression %d\n", idx);
01702 }
01703 }
01704 else
01705 {
01706 insert_insn_on_edge (se_insn, eg);
01707
01708 if (dump_file)
01709 {
01710 fprintf (dump_file, "PRE: edge (%d,%d), ",
01711 bb->index,
01712 INDEX_EDGE_SUCC_BB (edge_list, e)->index);
01713 fprintf (dump_file, "inserting expression %d\n", idx);
01714 }
01715 }
01716 did_insert = true;
01717 }
01718 }
01719 }
01720 return did_insert;
01721 }
01722
01723
01724
01725
01726
01727
01728
01729
01730
01731
01732 static int
01733 see_pre_delete_extension (void **slot, void *b ATTRIBUTE_UNUSED)
01734 {
01735 struct see_pre_extension_expr *expr = *slot;
01736 struct see_occr *occr;
01737 int indx = expr->bitmap_index;
01738
01739 for (occr = expr->antic_occr; occr != NULL; occr = occr->next)
01740 {
01741 if (TEST_BIT (pre_delete_map[occr->block_num], indx))
01742 {
01743
01744 INSN_DELETED_P (occr->insn) = 1;
01745 if (dump_file)
01746 {
01747 fprintf (dump_file,"Redundant extension deleted:\n");
01748 print_rtl_single (dump_file, occr->insn);
01749 }
01750 }
01751 }
01752 return 1;
01753 }
01754
01755
01756
01757
01758
01759
01760
01761
01762
01763 static int
01764 see_map_extension (void **slot, void *b)
01765 {
01766 struct see_pre_extension_expr *expr = *slot;
01767 struct see_pre_extension_expr **index_map =
01768 (struct see_pre_extension_expr **) b;
01769
01770 index_map[expr->bitmap_index] = expr;
01771
01772 return 1;
01773 }
01774
01775
01776
01777
01778
01779
01780
01781
01782 static void
01783 see_commit_changes (void)
01784 {
01785 struct see_pre_extension_expr **index_map;
01786 size_t pre_extension_num = htab_elements (see_pre_extension_hash);
01787 bool did_insert = false;
01788 int i;
01789
01790 index_map = xcalloc (pre_extension_num,
01791 sizeof (struct see_pre_extension_expr *));
01792
01793 if (dump_file)
01794 fprintf (dump_file,
01795 "* Phase 4: Commit changes to the insn stream. *\n");
01796
01797
01798 htab_traverse (see_pre_extension_hash, see_map_extension, (PTR) index_map);
01799
01800
01801
01802 htab_traverse (see_pre_extension_hash, see_pre_delete_extension, NULL);
01803
01804
01805
01806 df_finish (df);
01807 df = NULL;
01808
01809
01810 did_insert = see_pre_insert_extensions (index_map);
01811
01812 if (did_insert)
01813 commit_edge_insertions ();
01814
01815
01816 for (i = 0; i < last_bb; i++)
01817 {
01818 if (see_bb_splay_ar[i])
01819 {
01820
01821
01822 splay_tree_foreach (see_bb_splay_ar[i],
01823 see_commit_ref_changes, NULL);
01824 }
01825 }
01826
01827 free (index_map);
01828 }
01829
01830
01831
01832
01833
01834
01835
01836
01837
01838
01839
01840
01841
01842 static int
01843 see_analyze_merged_def_local_prop (void **slot, void *b)
01844 {
01845 rtx def_se = *slot;
01846 struct see_ref_s *curr_ref_s = (struct see_ref_s *) b;
01847 rtx ref = curr_ref_s->insn;
01848 struct see_pre_extension_expr *extension_expr;
01849 int indx;
01850 int bb_num = BLOCK_NUM (ref);
01851 htab_t curr_bb_hash;
01852 struct see_register_properties *curr_prop, **slot_prop;
01853 struct see_register_properties temp_prop;
01854 rtx dest_extension_reg = see_get_extension_reg (def_se, 1);
01855 struct see_occr *curr_occr = NULL;
01856 struct see_occr *tmp_occr = NULL;
01857
01858 extension_expr = see_seek_pre_extension_expr (def_se, DEF_EXTENSION);
01859
01860 gcc_assert (extension_expr);
01861
01862 curr_bb_hash = see_bb_hash_ar[bb_num];
01863 gcc_assert (curr_bb_hash);
01864 temp_prop.regno = REGNO (dest_extension_reg);
01865 slot_prop =
01866 (struct see_register_properties **) htab_find_slot (curr_bb_hash,
01867 &temp_prop, INSERT);
01868 curr_prop = *slot_prop;
01869 gcc_assert (curr_prop);
01870
01871 indx = extension_expr->bitmap_index;
01872
01873
01874 RESET_BIT (transp[bb_num], indx);
01875
01876 RESET_BIT (ae_kill[bb_num], indx);
01877
01878 if (curr_prop->first_se_after_last_def == DF_INSN_LUID (df, ref))
01879 {
01880
01881 SET_BIT (comp[bb_num], indx);
01882
01883 curr_occr = xmalloc (sizeof (struct see_occr));
01884 curr_occr->next = NULL;
01885 curr_occr->insn = def_se;
01886 curr_occr->block_num = bb_num;
01887 tmp_occr = extension_expr->avail_occr;
01888 if (!tmp_occr)
01889 extension_expr->avail_occr = curr_occr;
01890 else
01891 {
01892 while (tmp_occr->next)
01893 tmp_occr = tmp_occr->next;
01894 tmp_occr->next = curr_occr;
01895 }
01896 }
01897
01898 return 1;
01899 }
01900
01901
01902
01903
01904
01905
01906
01907
01908
01909
01910 static int
01911 see_analyze_unmerged_def_local_prop (void **slot, void *b)
01912 {
01913 rtx def_se = *slot;
01914 struct see_ref_s *curr_ref_s = (struct see_ref_s *) b;
01915 rtx ref = curr_ref_s->insn;
01916 struct see_pre_extension_expr *extension_expr;
01917 int indx;
01918 int bb_num = BLOCK_NUM (ref);
01919 htab_t curr_bb_hash;
01920 struct see_register_properties *curr_prop, **slot_prop;
01921 struct see_register_properties temp_prop;
01922 rtx dest_extension_reg = see_get_extension_reg (def_se, 1);
01923
01924 extension_expr = see_seek_pre_extension_expr (def_se, DEF_EXTENSION);
01925
01926 gcc_assert (extension_expr);
01927
01928 curr_bb_hash = see_bb_hash_ar[bb_num];
01929 gcc_assert (curr_bb_hash);
01930 temp_prop.regno = REGNO (dest_extension_reg);
01931 slot_prop =
01932 (struct see_register_properties **) htab_find_slot (curr_bb_hash,
01933 &temp_prop, INSERT);
01934 curr_prop = *slot_prop;
01935 gcc_assert (curr_prop);
01936
01937 indx = extension_expr->bitmap_index;
01938
01939
01940 RESET_BIT (transp[bb_num], indx);
01941
01942 SET_BIT (ae_kill[bb_num], indx);
01943
01944 return 1;
01945 }
01946
01947
01948
01949
01950
01951
01952
01953
01954
01955
01956
01957 static int
01958 see_analyze_use_local_prop (void **slot, void *b)
01959 {
01960 struct see_ref_s *curr_ref_s = (struct see_ref_s *) b;
01961 rtx use_se = *slot;
01962 rtx ref = curr_ref_s->insn;
01963 rtx dest_extension_reg = see_get_extension_reg (use_se, 1);
01964 struct see_pre_extension_expr *extension_expr;
01965 struct see_register_properties *curr_prop, **slot_prop;
01966 struct see_register_properties temp_prop;
01967 struct see_occr *curr_occr = NULL;
01968 struct see_occr *tmp_occr = NULL;
01969 htab_t curr_bb_hash;
01970 int indx;
01971 int bb_num = BLOCK_NUM (ref);
01972
01973 extension_expr = see_seek_pre_extension_expr (use_se, USE_EXTENSION);
01974
01975 gcc_assert (extension_expr);
01976
01977 curr_bb_hash = see_bb_hash_ar[bb_num];
01978 gcc_assert (curr_bb_hash);
01979 temp_prop.regno = REGNO (dest_extension_reg);
01980 slot_prop =
01981 (struct see_register_properties **) htab_find_slot (curr_bb_hash,
01982 &temp_prop, INSERT);
01983 curr_prop = *slot_prop;
01984 gcc_assert (curr_prop);
01985
01986 indx = extension_expr->bitmap_index;
01987
01988 if (curr_prop->first_se_before_any_def == DF_INSN_LUID (df, ref))
01989 {
01990
01991 SET_BIT (antloc[bb_num], indx);
01992
01993 curr_occr = xmalloc (sizeof (struct see_occr));
01994 curr_occr->next = NULL;
01995 curr_occr->insn = use_se;
01996 curr_occr->block_num = bb_num;
01997 tmp_occr = extension_expr->antic_occr;
01998 if (!tmp_occr)
01999 extension_expr->antic_occr = curr_occr;
02000 else
02001 {
02002 while (tmp_occr->next)
02003 tmp_occr = tmp_occr->next;
02004 tmp_occr->next = curr_occr;
02005 }
02006 if (curr_prop->last_def < 0)
02007 {
02008
02009 SET_BIT (comp[bb_num], indx);
02010
02011 curr_occr = xmalloc (sizeof (struct see_occr));
02012 curr_occr->next = NULL;
02013 curr_occr->insn = use_se;
02014 curr_occr->block_num = bb_num;
02015 tmp_occr = extension_expr->avail_occr;
02016 if (!tmp_occr)
02017 extension_expr->avail_occr = curr_occr;
02018 else
02019 {
02020 while (tmp_occr->next)
02021 tmp_occr = tmp_occr->next;
02022 tmp_occr->next = curr_occr;
02023 }
02024 }
02025
02026
02027 }
02028 else if (curr_prop->first_se_after_last_def == DF_INSN_LUID (df, ref))
02029 {
02030
02031 SET_BIT (comp[bb_num], indx);
02032
02033 RESET_BIT (ae_kill[bb_num], indx);
02034
02035 curr_occr = xmalloc (sizeof (struct see_occr));
02036 curr_occr->next = NULL;
02037 curr_occr->insn = use_se;
02038 curr_occr->block_num = bb_num;
02039 tmp_occr = extension_expr->avail_occr;
02040 if (!tmp_occr)
02041 extension_expr->avail_occr = curr_occr;
02042 else
02043 {
02044 while (tmp_occr->next)
02045 tmp_occr = tmp_occr->next;
02046 tmp_occr->next = curr_occr;
02047 }
02048 }
02049 return 1;
02050 }
02051
02052
02053
02054
02055
02056
02057
02058
02059
02060
02061 static int
02062 see_analyze_ref_local_prop (splay_tree_node stn,
02063 void *data ATTRIBUTE_UNUSED)
02064 {
02065 htab_t use_se_hash = ((struct see_ref_s *) (stn->value))->use_se_hash;
02066 htab_t unmerged_def_se_hash =
02067 ((struct see_ref_s *) (stn->value))->unmerged_def_se_hash;
02068 htab_t merged_def_se_hash =
02069 ((struct see_ref_s *) (stn->value))->merged_def_se_hash;
02070
02071
02072 if (use_se_hash)
02073 htab_traverse_noresize (use_se_hash, see_analyze_use_local_prop,
02074 (PTR) (stn->value));
02075
02076
02077 if (unmerged_def_se_hash)
02078 htab_traverse (unmerged_def_se_hash, see_analyze_unmerged_def_local_prop,
02079 (PTR) (stn->value));
02080
02081
02082 if (merged_def_se_hash)
02083 htab_traverse (merged_def_se_hash, see_analyze_merged_def_local_prop,
02084 (PTR) (stn->value));
02085
02086
02087 return 0;
02088 }
02089
02090
02091
02092
02093
02094
02095
02096 static void
02097 see_execute_LCM (void)
02098 {
02099 size_t pre_extension_num = htab_elements (see_pre_extension_hash);
02100 int i = 0;
02101
02102 if (dump_file)
02103 fprintf (dump_file,
02104 "* Phase 3: Eliminate globally redundant extensions. *\n");
02105
02106
02107 transp = sbitmap_vector_alloc (last_bb, pre_extension_num);
02108 comp = sbitmap_vector_alloc (last_bb, pre_extension_num);
02109 antloc = sbitmap_vector_alloc (last_bb, pre_extension_num);
02110 ae_kill = sbitmap_vector_alloc (last_bb, pre_extension_num);
02111 sbitmap_vector_ones (transp, last_bb);
02112 sbitmap_vector_zero (comp, last_bb);
02113 sbitmap_vector_zero (antloc, last_bb);
02114 sbitmap_vector_zero (ae_kill, last_bb);
02115
02116
02117 for (i = 0; i < last_bb; i++)
02118 {
02119 if (see_bb_splay_ar[i])
02120 {
02121
02122
02123 splay_tree_foreach (see_bb_splay_ar[i],
02124 see_analyze_ref_local_prop, NULL);
02125 }
02126 }
02127
02128
02129 add_noreturn_fake_exit_edges ();
02130
02131
02132 edge_list = pre_edge_lcm (pre_extension_num, transp, comp, antloc,
02133 ae_kill, &pre_insert_map, &pre_delete_map);
02134
02135
02136 remove_fake_exit_edges ();
02137 }
02138
02139
02140
02141
02142
02143
02144
02145
02146
02147
02148
02149
02150
02151
02152
02153
02154
02155 static int
02156 see_set_prop_merged_def (void **slot, void *b)
02157 {
02158 rtx def_se = *slot;
02159 struct see_ref_s *curr_ref_s = (struct see_ref_s *) b;
02160 rtx insn = curr_ref_s->insn;
02161 rtx dest_extension_reg = see_get_extension_reg (def_se, 1);
02162 htab_t curr_bb_hash;
02163 struct see_register_properties *curr_prop = NULL;
02164 struct see_register_properties **slot_prop;
02165 struct see_register_properties temp_prop;
02166 int ref_luid = DF_INSN_LUID (df, insn);
02167
02168 curr_bb_hash = see_bb_hash_ar[BLOCK_NUM (curr_ref_s->insn)];
02169 if (!curr_bb_hash)
02170 {
02171
02172 curr_bb_hash = htab_create (10,
02173 hash_descriptor_properties,
02174 eq_descriptor_properties,
02175 hash_del_properties);
02176 see_bb_hash_ar[BLOCK_NUM (curr_ref_s->insn)] = curr_bb_hash;
02177 }
02178
02179
02180 temp_prop.regno = REGNO (dest_extension_reg);
02181 slot_prop =
02182 (struct see_register_properties **) htab_find_slot (curr_bb_hash,
02183 &temp_prop, INSERT);
02184
02185 if (slot_prop && *slot_prop != NULL)
02186 {
02187
02188 curr_prop = *slot_prop;
02189 gcc_assert (curr_prop->regno == REGNO (dest_extension_reg));
02190
02191 curr_prop->last_def = ref_luid;
02192 curr_prop->first_se_after_last_def = ref_luid;
02193 }
02194 else
02195 {
02196
02197 curr_prop = xmalloc (sizeof (struct see_register_properties));
02198 curr_prop->regno = REGNO (dest_extension_reg);
02199 curr_prop->last_def = ref_luid;
02200 curr_prop->first_se_before_any_def = -1;
02201 curr_prop->first_se_after_last_def = ref_luid;
02202 *slot_prop = curr_prop;
02203 }
02204
02205
02206
02207 see_seek_pre_extension_expr (def_se, DEF_EXTENSION);
02208
02209 return 1;
02210 }
02211
02212
02213
02214
02215
02216
02217
02218
02219
02220
02221
02222
02223
02224
02225
02226 static int
02227 see_set_prop_unmerged_def (void **slot, void *b)
02228 {
02229 rtx def_se = *slot;
02230 struct see_ref_s *curr_ref_s = (struct see_ref_s *) b;
02231 rtx insn = curr_ref_s->insn;
02232 rtx dest_extension_reg = see_get_extension_reg (def_se, 1);
02233 htab_t curr_bb_hash;
02234 struct see_register_properties *curr_prop = NULL;
02235 struct see_register_properties **slot_prop;
02236 struct see_register_properties temp_prop;
02237 int ref_luid = DF_INSN_LUID (df, insn);
02238
02239 curr_bb_hash = see_bb_hash_ar[BLOCK_NUM (curr_ref_s->insn)];
02240 if (!curr_bb_hash)
02241 {
02242
02243 curr_bb_hash = htab_create (10,
02244 hash_descriptor_properties,
02245 eq_descriptor_properties,
02246 hash_del_properties);
02247 see_bb_hash_ar[BLOCK_NUM (curr_ref_s->insn)] = curr_bb_hash;
02248 }
02249
02250
02251 temp_prop.regno = REGNO (dest_extension_reg);
02252 slot_prop =
02253 (struct see_register_properties **) htab_find_slot (curr_bb_hash,
02254 &temp_prop, INSERT);
02255
02256 if (slot_prop && *slot_prop != NULL)
02257 {
02258
02259 curr_prop = *slot_prop;
02260 gcc_assert (curr_prop->regno == REGNO (dest_extension_reg));
02261
02262 curr_prop->last_def = ref_luid;
02263 curr_prop->first_se_after_last_def = -1;
02264 }
02265 else
02266 {
02267
02268 curr_prop = xmalloc (sizeof (struct see_register_properties));
02269 curr_prop->regno = REGNO (dest_extension_reg);
02270 curr_prop->last_def = ref_luid;
02271 curr_prop->first_se_before_any_def = -1;
02272 curr_prop->first_se_after_last_def = -1;
02273 *slot_prop = curr_prop;
02274 }
02275
02276
02277
02278 see_seek_pre_extension_expr (def_se, DEF_EXTENSION);
02279
02280 return 1;
02281 }
02282
02283
02284
02285
02286
02287
02288
02289
02290
02291
02292
02293
02294
02295
02296
02297
02298
02299 static int
02300 see_set_prop_unmerged_use (void **slot, void *b)
02301 {
02302 rtx use_se = *slot;
02303 struct see_ref_s *curr_ref_s = (struct see_ref_s *) b;
02304 rtx insn = curr_ref_s->insn;
02305 rtx dest_extension_reg = see_get_extension_reg (use_se, 1);
02306 htab_t curr_bb_hash;
02307 struct see_register_properties *curr_prop = NULL;
02308 struct see_register_properties **slot_prop;
02309 struct see_register_properties temp_prop;
02310 bool locally_redundant = false;
02311 int ref_luid = DF_INSN_LUID (df, insn);
02312
02313 curr_bb_hash = see_bb_hash_ar[BLOCK_NUM (curr_ref_s->insn)];
02314 if (!curr_bb_hash)
02315 {
02316
02317 curr_bb_hash = htab_create (10,
02318 hash_descriptor_properties,
02319 eq_descriptor_properties,
02320 hash_del_properties);
02321 see_bb_hash_ar[BLOCK_NUM (curr_ref_s->insn)] = curr_bb_hash;
02322 }
02323
02324
02325 temp_prop.regno = REGNO (dest_extension_reg);
02326 slot_prop =
02327 (struct see_register_properties **) htab_find_slot (curr_bb_hash,
02328 &temp_prop, INSERT);
02329
02330 if (slot_prop && *slot_prop != NULL)
02331 {
02332
02333 curr_prop = *slot_prop;
02334 gcc_assert (curr_prop->regno == REGNO (dest_extension_reg));
02335
02336
02337 if (curr_prop->last_def < 0 && curr_prop->first_se_before_any_def < 0)
02338 curr_prop->first_se_before_any_def = ref_luid;
02339 else if (curr_prop->last_def < 0
02340 && curr_prop->first_se_before_any_def >= 0)
02341 {
02342
02343 htab_clear_slot (curr_ref_s->use_se_hash, (PTR *)slot);
02344 locally_redundant = true;
02345 }
02346 else if (curr_prop->last_def >= 0
02347 && curr_prop->first_se_after_last_def < 0)
02348 curr_prop->first_se_after_last_def = ref_luid;
02349 else if (curr_prop->last_def >= 0
02350 && curr_prop->first_se_after_last_def >= 0)
02351 {
02352
02353 htab_clear_slot (curr_ref_s->use_se_hash, (PTR *)slot);
02354 locally_redundant = true;
02355 }
02356 else
02357 gcc_unreachable ();
02358 }
02359 else
02360 {
02361
02362 curr_prop = xmalloc (sizeof (struct see_register_properties));
02363 curr_prop->regno = REGNO (dest_extension_reg);
02364 curr_prop->last_def = -1;
02365 curr_prop->first_se_before_any_def = ref_luid;
02366 curr_prop->first_se_after_last_def = -1;
02367 *slot_prop = curr_prop;
02368 }
02369
02370
02371
02372 if (!locally_redundant)
02373 see_seek_pre_extension_expr (use_se, USE_EXTENSION);
02374 if (locally_redundant && dump_file)
02375 {
02376 fprintf (dump_file, "Locally redundant extension:\n");
02377 print_rtl_single (dump_file, use_se);
02378 }
02379 return 1;
02380 }
02381
02382
02383
02384
02385
02386
02387
02388
02389 static int
02390 see_print_one_extension (void **slot, void *b ATTRIBUTE_UNUSED)
02391 {
02392 rtx def_se = *slot;
02393
02394 gcc_assert (def_se && INSN_P (def_se));
02395 print_rtl_single (dump_file, def_se);
02396
02397 return 1;
02398 }
02399
02400
02401
02402
02403
02404
02405 static void
02406 see_replace_src (rtx *x, void *data)
02407 {
02408 struct see_replace_data *d
02409 = (struct see_replace_data *) data;
02410
02411 *x = replace_rtx (*x, d->from, d->to);
02412 }
02413
02414
02415
02416
02417
02418
02419
02420
02421
02422
02423
02424
02425
02426
02427
02428
02429
02430
02431
02432
02433
02434
02435
02436
02437
02438
02439
02440 static void
02441 see_def_extension_not_merged (struct see_ref_s *curr_ref_s, rtx def_se)
02442 {
02443 struct see_replace_data d;
02444
02445
02446 rtx ref = (curr_ref_s->merged_insn) ? curr_ref_s->merged_insn :
02447 curr_ref_s->insn;
02448 rtx merged_ref_next = (curr_ref_s->merged_insn) ?
02449 NEXT_INSN (curr_ref_s->merged_insn): NULL_RTX;
02450 rtx ref_copy = copy_rtx (ref);
02451 rtx source_extension_reg = see_get_extension_reg (def_se, 0);
02452 rtx dest_extension_reg = see_get_extension_reg (def_se, 1);
02453 rtx move_insn = NULL;
02454 rtx set, rhs;
02455 rtx dest_reg, dest_real_reg;
02456 rtx new_pseudo_reg, subreg;
02457 enum machine_mode source_extension_mode = GET_MODE (source_extension_reg);
02458 enum machine_mode dest_mode;
02459
02460 set = single_set (def_se);
02461 gcc_assert (set);
02462 rhs = SET_SRC (set);
02463 gcc_assert (GET_CODE (rhs) == SIGN_EXTEND
02464 || GET_CODE (rhs) == ZERO_EXTEND);
02465 dest_reg = XEXP (rhs, 0);
02466 gcc_assert (REG_P (dest_reg)
02467 || (GET_CODE (dest_reg) == SUBREG
02468 && REG_P (SUBREG_REG (dest_reg))));
02469 dest_real_reg = REG_P (dest_reg) ? dest_reg : SUBREG_REG (dest_reg);
02470 dest_mode = GET_MODE (dest_reg);
02471
02472 subreg = gen_lowpart_SUBREG (dest_mode, dest_extension_reg);
02473 new_pseudo_reg = gen_reg_rtx (source_extension_mode);
02474
02475
02476 d.from = dest_real_reg;
02477 d.to = new_pseudo_reg;
02478 note_uses (&PATTERN (ref_copy), see_replace_src, &d);
02479
02480 ref_copy = replace_rtx (ref_copy, dest_reg, subreg);
02481
02482
02483
02484 d.from = new_pseudo_reg;
02485 d.to = dest_real_reg;
02486 note_uses (&PATTERN (ref_copy), see_replace_src, &d);
02487
02488 if (rtx_equal_p (PATTERN (ref), PATTERN (ref_copy))
02489 || insn_invalid_p (ref_copy))
02490 {
02491
02492
02493
02494 ref_copy = copy_rtx (ref);
02495
02496
02497 start_sequence ();
02498 emit_move_insn (subreg, dest_reg);
02499 move_insn = get_insns ();
02500 end_sequence ();
02501
02502
02503
02504 PREV_INSN (ref_copy) = NULL_RTX;
02505 NEXT_INSN (ref_copy) = move_insn;
02506 PREV_INSN (move_insn) = ref_copy;
02507 NEXT_INSN (move_insn) = merged_ref_next;
02508 if (merged_ref_next != NULL_RTX)
02509 PREV_INSN (merged_ref_next) = move_insn;
02510 curr_ref_s->merged_insn = ref_copy;
02511
02512 if (dump_file)
02513 {
02514 fprintf (dump_file, "Following def merge failure a move ");
02515 fprintf (dump_file, "insn was added after the ref.\n");
02516 fprintf (dump_file, "Original ref:\n");
02517 print_rtl_single (dump_file, ref);
02518 fprintf (dump_file, "Move insn that was added:\n");
02519 print_rtl_single (dump_file, move_insn);
02520 }
02521 return;
02522 }
02523
02524
02525
02526
02527 validate_simplify_insn (ref_copy);
02528
02529
02530 start_sequence ();
02531 emit_move_insn (dest_reg, subreg);
02532 move_insn = get_insns ();
02533 end_sequence ();
02534
02535
02536
02537 PREV_INSN (ref_copy) = NULL_RTX;
02538 NEXT_INSN (ref_copy) = move_insn;
02539 PREV_INSN (move_insn) = ref_copy;
02540 NEXT_INSN (move_insn) = merged_ref_next;
02541 if (merged_ref_next != NULL_RTX)
02542 PREV_INSN (merged_ref_next) = move_insn;
02543 curr_ref_s->merged_insn = ref_copy;
02544
02545 if (dump_file)
02546 {
02547 fprintf (dump_file, "Following merge failure the ref was transformed!\n");
02548 fprintf (dump_file, "Original ref:\n");
02549 print_rtl_single (dump_file, ref);
02550 fprintf (dump_file, "Transformed ref:\n");
02551 print_rtl_single (dump_file, ref_copy);
02552 fprintf (dump_file, "Move insn that was added:\n");
02553 print_rtl_single (dump_file, move_insn);
02554 }
02555 }
02556
02557
02558
02559
02560
02561
02562
02563
02564
02565
02566
02567
02568
02569
02570
02571
02572
02573
02574
02575
02576
02577
02578
02579
02580 static int
02581 see_merge_one_use_extension (void **slot, void *b)
02582 {
02583 struct see_ref_s *curr_ref_s = (struct see_ref_s *) b;
02584 rtx use_se = *slot;
02585 rtx ref = (curr_ref_s->merged_insn) ? curr_ref_s->merged_insn :
02586 curr_ref_s->insn;
02587 rtx merged_ref_next = (curr_ref_s->merged_insn) ?
02588 NEXT_INSN (curr_ref_s->merged_insn): NULL_RTX;
02589 rtx ref_copy = copy_rtx (ref);
02590 rtx extension_set = single_set (use_se);
02591 rtx extension_rhs = NULL;
02592 rtx dest_extension_reg = see_get_extension_reg (use_se, 1);
02593 rtx note = NULL;
02594 rtx simplified_note = NULL;
02595
02596 gcc_assert (use_se && curr_ref_s && extension_set);
02597
02598 extension_rhs = SET_SRC (extension_set);
02599
02600
02601
02602
02603
02604
02605 note = find_reg_equal_equiv_note (ref_copy);
02606 if (note)
02607 {
02608 simplified_note = simplify_replace_rtx (XEXP (note, 0),
02609 dest_extension_reg,
02610 extension_rhs);
02611 if (rtx_equal_p (XEXP (note, 0), simplified_note))
02612
02613 remove_note (ref_copy, note);
02614 else
02615 XEXP (note, 0) = simplified_note;
02616 }
02617
02618 if (!see_want_to_be_merged_with_extension (ref, use_se, USE_EXTENSION))
02619 {
02620
02621 if (dump_file)
02622 {
02623 fprintf (dump_file, "Use merge skipped!\n");
02624 fprintf (dump_file, "Original instructions:\n");
02625 print_rtl_single (dump_file, use_se);
02626 print_rtl_single (dump_file, ref);
02627 }
02628
02629
02630 return 1;
02631 }
02632
02633 validate_replace_src_group (dest_extension_reg, extension_rhs, ref_copy);
02634
02635 if (!num_changes_pending ())
02636
02637
02638
02639 {
02640 if (dump_file)
02641 {
02642 fprintf (dump_file, "Use extension not necessary before:\n");
02643 print_rtl_single (dump_file, ref);
02644 }
02645 htab_clear_slot (curr_ref_s->use_se_hash, (PTR *)slot);
02646 PREV_INSN (ref_copy) = NULL_RTX;
02647 NEXT_INSN (ref_copy) = merged_ref_next;
02648 if (merged_ref_next != NULL_RTX)
02649 PREV_INSN (merged_ref_next) = ref_copy;
02650 curr_ref_s->merged_insn = ref_copy;
02651 return 1;
02652 }
02653
02654 if (!apply_change_group ())
02655 {
02656
02657 if (dump_file)
02658 {
02659 fprintf (dump_file, "Use merge failed!\n");
02660 fprintf (dump_file, "Original instructions:\n");
02661 print_rtl_single (dump_file, use_se);
02662 print_rtl_single (dump_file, ref);
02663 }
02664
02665
02666 return 1;
02667 }
02668
02669
02670
02671
02672 validate_simplify_insn (ref_copy);
02673
02674 PREV_INSN (ref_copy) = NULL_RTX;
02675 NEXT_INSN (ref_copy) = merged_ref_next;
02676 if (merged_ref_next != NULL_RTX)
02677 PREV_INSN (merged_ref_next) = ref_copy;
02678 curr_ref_s->merged_insn = ref_copy;
02679
02680 if (dump_file)
02681 {
02682 fprintf (dump_file, "Use merge succeeded!\n");
02683 fprintf (dump_file, "Original instructions:\n");
02684 print_rtl_single (dump_file, use_se);
02685 print_rtl_single (dump_file, ref);
02686 fprintf (dump_file, "Merged instruction:\n");
02687 print_rtl_single (dump_file, ref_copy);
02688 }
02689
02690
02691
02692 htab_clear_slot (curr_ref_s->use_se_hash, (PTR *)slot);
02693 return 1;
02694 }
02695
02696
02697
02698
02699
02700
02701
02702
02703
02704
02705
02706
02707
02708
02709
02710
02711
02712
02713
02714
02715
02716
02717
02718
02719
02720
02721 static int
02722 see_merge_one_def_extension (void **slot, void *b)
02723 {
02724 struct see_ref_s *curr_ref_s = (struct see_ref_s *) b;
02725 rtx def_se = *slot;
02726
02727
02728 rtx ref = (curr_ref_s->merged_insn) ? curr_ref_s->merged_insn :
02729 curr_ref_s->insn;
02730 rtx merged_ref_next = (curr_ref_s->merged_insn) ?
02731 NEXT_INSN (curr_ref_s->merged_insn): NULL_RTX;
02732 rtx ref_copy = copy_rtx (ref);
02733 rtx new_set = NULL;
02734 rtx source_extension_reg = see_get_extension_reg (def_se, 0);
02735 rtx dest_extension_reg = see_get_extension_reg (def_se, 1);
02736 rtx move_insn, *rtx_slot, subreg;
02737 rtx temp_extension = NULL;
02738 rtx simplified_temp_extension = NULL;
02739 rtx *pat;
02740 enum rtx_code code;
02741 enum rtx_code extension_code;
02742 enum machine_mode source_extension_mode;
02743 enum machine_mode source_mode;
02744 enum machine_mode dest_extension_mode;
02745 bool merge_success = false;
02746 int i;
02747
02748 gcc_assert (def_se
02749 && INSN_P (def_se)
02750 && curr_ref_s
02751 && ref
02752 && INSN_P (ref));
02753
02754 if (!see_want_to_be_merged_with_extension (ref, def_se, DEF_EXTENSION))
02755 {
02756
02757 if (dump_file)
02758 {
02759 fprintf (dump_file, "Def merge skipped!\n");
02760 fprintf (dump_file, "Original instructions:\n");
02761 print_rtl_single (dump_file, ref);
02762 print_rtl_single (dump_file, def_se);
02763 }
02764
02765 see_def_extension_not_merged (curr_ref_s, def_se);
02766
02767 return 1;
02768 }
02769
02770 extension_code = see_get_extension_data (def_se, &source_mode);
02771
02772
02773 source_extension_mode = GET_MODE (source_extension_reg);
02774 dest_extension_mode = GET_MODE (dest_extension_reg);
02775
02776 pat = &PATTERN (ref_copy);
02777 code = GET_CODE (*pat);
02778
02779 if (code == PARALLEL)
02780 {
02781 bool need_to_apply_change = false;
02782
02783 for (i = 0; i < XVECLEN (*pat, 0); i++)
02784 {
02785 rtx *sub = &XVECEXP (*pat, 0, i);
02786
02787 if (GET_CODE (*sub) == SET
02788 && GET_MODE (SET_SRC (*sub)) != VOIDmode
02789 && GET_MODE (SET_DEST (*sub)) == source_mode
02790 && ((REG_P (SET_DEST (*sub))
02791 && REGNO (SET_DEST (*sub)) == REGNO (source_extension_reg))
02792 || (GET_CODE (SET_DEST (*sub)) == SUBREG
02793 && REG_P (SUBREG_REG (SET_DEST (*sub)))
02794 && (REGNO (SUBREG_REG (SET_DEST (*sub))) ==
02795 REGNO (source_extension_reg)))))
02796 {
02797 rtx orig_src = SET_SRC (*sub);
02798
02799 if (extension_code == SIGN_EXTEND)
02800 temp_extension = gen_rtx_SIGN_EXTEND (dest_extension_mode,
02801 orig_src);
02802 else
02803 temp_extension = gen_rtx_ZERO_EXTEND (dest_extension_mode,
02804 orig_src);
02805 simplified_temp_extension = simplify_rtx (temp_extension);
02806 temp_extension =
02807 (simplified_temp_extension) ? simplified_temp_extension :
02808 temp_extension;
02809 new_set = gen_rtx_SET (VOIDmode, dest_extension_reg,
02810 temp_extension);
02811 validate_change (ref_copy, sub, new_set, 1);
02812 need_to_apply_change = true;
02813 }
02814 }
02815 if (need_to_apply_change)
02816 if (apply_change_group ())
02817 merge_success = true;
02818 }
02819 else if (code == SET
02820 && GET_MODE (SET_SRC (*pat)) != VOIDmode
02821 && GET_MODE (SET_DEST (*pat)) == source_mode
02822 && ((REG_P (SET_DEST (*pat))
02823 && REGNO (SET_DEST (*pat)) == REGNO (source_extension_reg))
02824 || (GET_CODE (SET_DEST (*pat)) == SUBREG
02825 && REG_P (SUBREG_REG (SET_DEST (*pat)))
02826 && (REGNO (SUBREG_REG (SET_DEST (*pat))) ==
02827 REGNO (source_extension_reg)))))
02828 {
02829 rtx orig_src = SET_SRC (*pat);
02830
02831 if (extension_code == SIGN_EXTEND)
02832 temp_extension = gen_rtx_SIGN_EXTEND (dest_extension_mode, orig_src);
02833 else
02834 temp_extension = gen_rtx_ZERO_EXTEND (dest_extension_mode, orig_src);
02835 simplified_temp_extension = simplify_rtx (temp_extension);
02836 temp_extension = (simplified_temp_extension) ? simplified_temp_extension :
02837 temp_extension;
02838 new_set = gen_rtx_SET (VOIDmode, dest_extension_reg, temp_extension);
02839 if (validate_change (ref_copy, pat, new_set, 0))
02840 merge_success = true;
02841 }
02842 if (!merge_success)
02843 {
02844
02845 if (dump_file)
02846 {
02847 fprintf (dump_file, "Def merge failed!\n");
02848 fprintf (dump_file, "Original instructions:\n");
02849 print_rtl_single (dump_file, ref);
02850 print_rtl_single (dump_file, def_se);
02851 }
02852
02853 see_def_extension_not_merged (curr_ref_s, def_se);
02854
02855 return 1;
02856 }
02857
02858
02859
02860
02861 subreg = gen_lowpart_SUBREG (source_extension_mode, dest_extension_reg);
02862 start_sequence ();
02863 emit_move_insn (source_extension_reg, subreg);
02864 move_insn = get_insns ();
02865 end_sequence ();
02866
02867
02868
02869 PREV_INSN (ref_copy) = NULL_RTX;
02870 NEXT_INSN (ref_copy) = move_insn;
02871 PREV_INSN (move_insn) = ref_copy;
02872 NEXT_INSN (move_insn) = merged_ref_next;
02873 if (merged_ref_next != NULL_RTX)
02874 PREV_INSN (merged_ref_next) = move_insn;
02875 curr_ref_s->merged_insn = ref_copy;
02876
02877 if (dump_file)
02878 {
02879 fprintf (dump_file, "Def merge succeeded!\n");
02880 fprintf (dump_file, "Original instructions:\n");
02881 print_rtl_single (dump_file, ref);
02882 print_rtl_single (dump_file, def_se);
02883 fprintf (dump_file, "Merged instruction:\n");
02884 print_rtl_single (dump_file, ref_copy);
02885 fprintf (dump_file, "Move instruction that was added:\n");
02886 print_rtl_single (dump_file, move_insn);
02887 }
02888
02889
02890
02891 htab_clear_slot (curr_ref_s->unmerged_def_se_hash, (PTR *)slot);
02892 if (!curr_ref_s->merged_def_se_hash)
02893 curr_ref_s->merged_def_se_hash = htab_create (10,
02894 hash_descriptor_extension,
02895 eq_descriptor_extension,
02896 NULL);
02897 rtx_slot = (rtx *) htab_find_slot (curr_ref_s->merged_def_se_hash,
02898 dest_extension_reg, INSERT);
02899 gcc_assert (*rtx_slot == NULL);
02900 *rtx_slot = def_se;
02901
02902 return 1;
02903 }
02904
02905
02906
02907
02908
02909
02910
02911
02912
02913
02914
02915
02916
02917
02918
02919
02920
02921
02922
02923 static int
02924 see_handle_extensions_for_one_ref (splay_tree_node stn,
02925 void *data ATTRIBUTE_UNUSED)
02926 {
02927 htab_t use_se_hash = ((struct see_ref_s *) (stn->value))->use_se_hash;
02928 htab_t unmerged_def_se_hash =
02929 ((struct see_ref_s *) (stn->value))->unmerged_def_se_hash;
02930 htab_t merged_def_se_hash;
02931 rtx ref = ((struct see_ref_s *) (stn->value))->insn;
02932
02933 if (dump_file)
02934 {
02935 fprintf (dump_file, "Handling ref:\n");
02936 print_rtl_single (dump_file, ref);
02937 }
02938
02939
02940 if (unmerged_def_se_hash)
02941 htab_traverse_noresize (unmerged_def_se_hash, see_merge_one_def_extension,
02942 (PTR) (stn->value));
02943
02944 if (use_se_hash)
02945
02946 htab_traverse_noresize (use_se_hash, see_merge_one_use_extension,
02947 (PTR) (stn->value));
02948
02949 merged_def_se_hash = ((struct see_ref_s *) (stn->value))->merged_def_se_hash;
02950
02951 if (dump_file)
02952 {
02953 fprintf (dump_file, "The hashes of the current reference:\n");
02954 if (unmerged_def_se_hash)
02955 {
02956 fprintf (dump_file, "unmerged_def_se_hash:\n");
02957 htab_traverse (unmerged_def_se_hash, see_print_one_extension, NULL);
02958 }
02959 if (merged_def_se_hash)
02960 {
02961 fprintf (dump_file, "merged_def_se_hash:\n");
02962 htab_traverse (merged_def_se_hash, see_print_one_extension, NULL);
02963 }
02964 if (use_se_hash)
02965 {
02966 fprintf (dump_file, "use_se_hash:\n");
02967 htab_traverse (use_se_hash, see_print_one_extension, NULL);
02968 }
02969 }
02970
02971
02972
02973
02974
02975
02976 if (use_se_hash)
02977 htab_traverse_noresize (use_se_hash, see_set_prop_unmerged_use,
02978 (PTR) (stn->value));
02979
02980 if (unmerged_def_se_hash)
02981 htab_traverse (unmerged_def_se_hash, see_set_prop_unmerged_def,
02982 (PTR) (stn->value));
02983
02984 if (merged_def_se_hash)
02985 htab_traverse (merged_def_se_hash, see_set_prop_merged_def,
02986 (PTR) (stn->value));
02987
02988
02989 return 0;
02990 }
02991
02992
02993
02994
02995
02996
02997
02998 static void
02999 see_merge_and_eliminate_extensions (void)
03000 {
03001 int i = 0;
03002
03003 if (dump_file)
03004 fprintf (dump_file,
03005 "* Phase 2: Merge and eliminate locally redundant extensions. *\n");
03006
03007
03008 for (i = 0; i < last_bb; i++)
03009 {
03010 if (see_bb_splay_ar[i])
03011 {
03012 if (dump_file)
03013 fprintf (dump_file, "Handling references for bb %d\n", i);
03014
03015
03016 splay_tree_foreach (see_bb_splay_ar[i],
03017 see_handle_extensions_for_one_ref, NULL);
03018 }
03019 }
03020 }
03021
03022
03023
03024
03025
03026
03027
03028
03029
03030
03031 static bool
03032 see_store_reference_and_extension (rtx ref_insn, rtx se_insn,
03033 enum extension_type type)
03034 {
03035 rtx *rtx_slot;
03036 int curr_bb_num;
03037 splay_tree_node stn = NULL;
03038 htab_t se_hash = NULL;
03039 struct see_ref_s *ref_s = NULL;
03040
03041
03042 gcc_assert (ref_insn && se_insn);
03043 if (!see_bb_splay_ar)
03044 return false;
03045
03046 curr_bb_num = BLOCK_NUM (ref_insn);
03047 gcc_assert (curr_bb_num < last_bb && curr_bb_num >= 0);
03048
03049
03050 if (!see_bb_splay_ar[curr_bb_num])
03051
03052 see_bb_splay_ar[curr_bb_num] = splay_tree_new (splay_tree_compare_ints,
03053 NULL, see_free_ref_s);
03054 else
03055
03056
03057 {
03058 stn = splay_tree_lookup (see_bb_splay_ar[curr_bb_num],
03059 DF_INSN_LUID (df, ref_insn));
03060 if (stn)
03061 switch (type)
03062 {
03063 case EXPLICIT_DEF_EXTENSION:
03064 se_hash =
03065 ((struct see_ref_s *) (stn->value))->unmerged_def_se_hash;
03066 if (!se_hash)
03067 {
03068 se_hash = htab_create (10,
03069 hash_descriptor_extension,
03070 eq_descriptor_extension,
03071 NULL);
03072 ((struct see_ref_s *) (stn->value))->unmerged_def_se_hash =
03073 se_hash;
03074 }
03075 break;
03076 case IMPLICIT_DEF_EXTENSION:
03077 se_hash = ((struct see_ref_s *) (stn->value))->merged_def_se_hash;
03078 if (!se_hash)
03079 {
03080 se_hash = htab_create (10,
03081 hash_descriptor_extension,
03082 eq_descriptor_extension,
03083 NULL);
03084 ((struct see_ref_s *) (stn->value))->merged_def_se_hash =
03085 se_hash;
03086 }
03087 break;
03088 case USE_EXTENSION:
03089 se_hash = ((struct see_ref_s *) (stn->value))->use_se_hash;
03090 if (!se_hash)
03091 {
03092 se_hash = htab_create (10,
03093 hash_descriptor_extension,
03094 eq_descriptor_extension,
03095 NULL);
03096 ((struct see_ref_s *) (stn->value))->use_se_hash = se_hash;
03097 }
03098 break;
03099 default:
03100 gcc_unreachable ();
03101 }
03102 }
03103
03104
03105
03106 if (!stn)
03107 {
03108 ref_s = xmalloc (sizeof (struct see_ref_s));
03109 ref_s->luid = DF_INSN_LUID (df, ref_insn);
03110 ref_s->insn = ref_insn;
03111 ref_s->merged_insn = NULL;
03112
03113
03114 switch (type)
03115 {
03116 case EXPLICIT_DEF_EXTENSION:
03117 ref_s->unmerged_def_se_hash = htab_create (10,
03118 hash_descriptor_extension,
03119 eq_descriptor_extension,
03120 NULL);
03121 se_hash = ref_s->unmerged_def_se_hash;
03122 ref_s->merged_def_se_hash = NULL;
03123 ref_s->use_se_hash = NULL;
03124 break;
03125 case IMPLICIT_DEF_EXTENSION:
03126 ref_s->merged_def_se_hash = htab_create (10,
03127 hash_descriptor_extension,
03128 eq_descriptor_extension,
03129 NULL);
03130 se_hash = ref_s->merged_def_se_hash;
03131 ref_s->unmerged_def_se_hash = NULL;
03132 ref_s->use_se_hash = NULL;
03133 break;
03134 case USE_EXTENSION:
03135 ref_s->use_se_hash = htab_create (10,
03136 hash_descriptor_extension,
03137 eq_descriptor_extension,
03138 NULL);
03139 se_hash = ref_s->use_se_hash;
03140 ref_s->unmerged_def_se_hash = NULL;
03141 ref_s->merged_def_se_hash = NULL;
03142 break;
03143 default:
03144 gcc_unreachable ();
03145 }
03146 }
03147
03148
03149
03150 rtx_slot = (rtx *) htab_find_slot (se_hash, se_insn, INSERT);
03151 if (*rtx_slot != NULL)
03152 {
03153 gcc_assert (type == USE_EXTENSION);
03154 gcc_assert (rtx_equal_p (PATTERN (*rtx_slot), PATTERN (se_insn)));
03155 }
03156 else
03157 *rtx_slot = se_insn;
03158
03159
03160 if (!stn)
03161 splay_tree_insert (see_bb_splay_ar[curr_bb_num],
03162 DF_INSN_LUID (df, ref_insn), (splay_tree_value) ref_s);
03163 return true;
03164 }
03165
03166
03167
03168
03169
03170
03171
03172
03173
03174
03175
03176
03177 static int
03178 see_handle_relevant_defs (void)
03179 {
03180 rtx insn = NULL;
03181 rtx se_insn = NULL;
03182 rtx reg = NULL;
03183 rtx ref_insn = NULL;
03184 struct web_entry *root_entry = NULL;
03185 unsigned int i;
03186 int num_relevant_defs = 0;
03187 enum rtx_code extension_code;
03188
03189 for (i = 0; i < defs_num; i++)
03190 {
03191 insn = DF_REF_INSN (DF_DEFS_GET (df, i));
03192 reg = DF_REF_REAL_REG (DF_DEFS_GET (df, i));
03193
03194 if (!insn)
03195 continue;
03196
03197 if (!INSN_P (insn))
03198 continue;
03199
03200 root_entry = unionfind_root (&def_entry[i]);
03201
03202 if (ENTRY_EI (root_entry)->relevancy != SIGN_EXTENDED_DEF
03203 && ENTRY_EI (root_entry)->relevancy != ZERO_EXTENDED_DEF)
03204
03205 continue;
03206
03207 if (root_entry->reg)
03208
03209
03210 gcc_assert (rtx_equal_p (root_entry->reg, reg));
03211 else
03212 root_entry->reg = reg;
03213
03214
03215
03216
03217
03218 if (ENTRY_EI (&def_entry[i])->local_relevancy == EXTENDED_DEF
03219 || (ENTRY_EI (&def_entry[i])->local_source_mode <
03220 ENTRY_EI (root_entry)->source_mode))
03221 {
03222 num_relevant_defs++;
03223
03224 if (ENTRY_EI (root_entry)->relevancy == SIGN_EXTENDED_DEF)
03225 extension_code = SIGN_EXTEND;
03226 else
03227 extension_code = ZERO_EXTEND;
03228
03229 se_insn =
03230 see_gen_normalized_extension (reg, extension_code,
03231 ENTRY_EI (root_entry)->source_mode);
03232
03233
03234 INSN_DELETED_P (se_insn) = 1;
03235
03236 if (!see_store_reference_and_extension (insn, se_insn,
03237 IMPLICIT_DEF_EXTENSION))
03238
03239 return -1;
03240 continue;
03241 }
03242
03243 ref_insn = PREV_INSN (insn);
03244 gcc_assert (BLOCK_NUM (ref_insn) == BLOCK_NUM (insn));
03245
03246 num_relevant_defs++;
03247
03248 if (!see_store_reference_and_extension (ref_insn, insn,
03249 EXPLICIT_DEF_EXTENSION))
03250
03251 return -1;
03252 }
03253 return num_relevant_defs;
03254 }
03255
03256
03257
03258
03259
03260
03261
03262
03263 static int
03264 see_handle_relevant_uses (void)
03265 {
03266 rtx insn = NULL;
03267 rtx reg = NULL;
03268 struct web_entry *root_entry = NULL;
03269 rtx se_insn = NULL;
03270 unsigned int i;
03271 int num_relevant_uses = 0;
03272 enum rtx_code extension_code;
03273
03274 for (i = 0; i < uses_num; i++)
03275 {
03276 insn = DF_REF_INSN (DF_USES_GET (df, i));
03277 reg = DF_REF_REAL_REG (DF_USES_GET (df, i));
03278
03279 if (!insn)
03280 continue;
03281
03282 if (!INSN_P (insn))
03283 continue;
03284
03285 root_entry = unionfind_root (&use_entry[i]);
03286
03287 if (ENTRY_EI (root_entry)->relevancy != SIGN_EXTENDED_DEF
03288 && ENTRY_EI (root_entry)->relevancy != ZERO_EXTENDED_DEF)
03289
03290 continue;
03291
03292 if (root_entry->reg)
03293
03294
03295 gcc_assert (rtx_equal_p (root_entry->reg, reg));
03296 else
03297 root_entry->reg = reg;
03298
03299
03300 if (ENTRY_EI (root_entry)->relevancy == SIGN_EXTENDED_DEF)
03301 extension_code = SIGN_EXTEND;
03302 else
03303 extension_code = ZERO_EXTEND;
03304
03305 se_insn =
03306 see_gen_normalized_extension (reg, extension_code,
03307 ENTRY_EI (root_entry)->source_mode);
03308 if (!se_insn)
03309
03310 return -1;
03311
03312 num_relevant_uses++;
03313
03314 if (!see_store_reference_and_extension (insn, se_insn,
03315 USE_EXTENSION))
03316
03317 return -1;
03318 }
03319
03320 return num_relevant_uses;
03321 }
03322
03323
03324
03325
03326
03327
03328
03329 static void
03330 see_update_uses_relevancy (void)
03331 {
03332 rtx insn = NULL;
03333 rtx reg = NULL;
03334 struct see_entry_extra_info *curr_entry_extra_info;
03335 enum entry_type et;
03336 unsigned int i;
03337
03338 if (!df || !use_entry)
03339 return;
03340
03341 for (i = 0; i < uses_num; i++)
03342 {
03343
03344 insn = DF_REF_INSN (DF_USES_GET (df, i));
03345 reg = DF_REF_REAL_REG (DF_USES_GET (df, i));
03346
03347 et = RELEVANT_USE;
03348
03349 if (insn)
03350 {
03351 if (!INSN_P (insn))
03352 et = NOT_RELEVANT;
03353 if (insn && find_reg_note (insn, REG_LIBCALL, NULL_RTX))
03354 et = NOT_RELEVANT;
03355 if (find_reg_note (insn, REG_RETVAL, NULL_RTX))
03356 et = NOT_RELEVANT;
03357 }
03358 else
03359 et = NOT_RELEVANT;
03360
03361 if (dump_file)
03362 {
03363 fprintf (dump_file, "u%i insn %i reg %i ",
03364 i, (insn ? INSN_UID (insn) : -1), REGNO (reg));
03365 if (et == NOT_RELEVANT)
03366 fprintf (dump_file, "NOT RELEVANT \n");
03367 else
03368 fprintf (dump_file, "RELEVANT USE \n");
03369 }
03370
03371 curr_entry_extra_info = xmalloc (sizeof (struct see_entry_extra_info));
03372 curr_entry_extra_info->relevancy = et;
03373 curr_entry_extra_info->local_relevancy = et;
03374 use_entry[i].extra_info = curr_entry_extra_info;
03375 use_entry[i].reg = NULL;
03376 use_entry[i].pred = NULL;
03377 }
03378 }
03379
03380
03381
03382
03383
03384
03385
03386
03387
03388
03389
03390
03391
03392
03393
03394
03395
03396
03397
03398
03399
03400
03401
03402
03403
03404
03405
03406
03407
03408
03409
03410
03411
03412
03413
03414
03415
03416 static enum entry_type
03417 see_analyze_one_def (rtx insn, enum machine_mode *source_mode,
03418 enum machine_mode *source_mode_unsigned)
03419 {
03420 enum rtx_code extension_code;
03421 rtx rhs = NULL;
03422 rtx lhs = NULL;
03423 rtx set = NULL;
03424 rtx source_register = NULL;
03425 rtx prev_insn = NULL;
03426 rtx next_insn = NULL;
03427 enum machine_mode mode;
03428 enum machine_mode next_source_mode;
03429 HOST_WIDE_INT val = 0;
03430 HOST_WIDE_INT val2 = 0;
03431 int i = 0;
03432
03433 *source_mode = MAX_MACHINE_MODE;
03434 *source_mode_unsigned = MAX_MACHINE_MODE;
03435
03436 if (!insn)
03437 return NOT_RELEVANT;
03438
03439 if (!INSN_P (insn))
03440 return NOT_RELEVANT;
03441
03442 extension_code = see_get_extension_data (insn, source_mode);
03443 switch (extension_code)
03444 {
03445 case SIGN_EXTEND:
03446 case ZERO_EXTEND:
03447 source_register = see_get_extension_reg (insn, 0);
03448
03449
03450
03451 prev_insn = PREV_INSN (insn);
03452 if (!prev_insn || !INSN_P (prev_insn))
03453 return NOT_RELEVANT;
03454
03455 if (!reg_set_between_p (source_register, PREV_INSN (prev_insn), insn))
03456 return NOT_RELEVANT;
03457
03458 if (find_reg_note (prev_insn, REG_LIBCALL, NULL_RTX))
03459 return NOT_RELEVANT;
03460
03461 if (find_reg_note (prev_insn, REG_RETVAL, NULL_RTX))
03462 return NOT_RELEVANT;
03463
03464
03465 if (GET_CODE (PATTERN (prev_insn)) == PARALLEL
03466 && asm_noperands (PATTERN (prev_insn)) >= 0)
03467 return NOT_RELEVANT;
03468
03469
03470
03471
03472 next_insn = NEXT_INSN (insn);
03473 if (next_insn
03474 && INSN_P (next_insn)
03475 && (see_get_extension_data (next_insn, &next_source_mode) !=
03476 NOT_RELEVANT))
03477 {
03478 rtx curr_dest_register = see_get_extension_reg (insn, 1);
03479 rtx next_source_register = see_get_extension_reg (next_insn, 0);
03480
03481 if (REGNO (curr_dest_register) == REGNO (next_source_register))
03482 return NOT_RELEVANT;
03483 }
03484
03485 if (extension_code == SIGN_EXTEND)
03486 return SIGN_EXTENDED_DEF;
03487 else
03488 return ZERO_EXTENDED_DEF;
03489
03490 case UNKNOWN:
03491
03492
03493
03494
03495 set = single_set (insn);
03496 if (!set)
03497 return NOT_RELEVANT;
03498 rhs = SET_SRC (set);
03499 lhs = SET_DEST (set);
03500
03501
03502
03503 if (!REG_P (lhs) && !SUBREG_REG (lhs))
03504 return NOT_RELEVANT;
03505
03506 switch (GET_CODE (rhs))
03507 {
03508 case SIGN_EXTEND:
03509 *source_mode = GET_MODE (XEXP (rhs, 0));
03510 *source_mode_unsigned = MAX_MACHINE_MODE;
03511 return EXTENDED_DEF;
03512 case ZERO_EXTEND:
03513 *source_mode = MAX_MACHINE_MODE;
03514 *source_mode_unsigned = GET_MODE (XEXP (rhs, 0));
03515 return EXTENDED_DEF;
03516 case CONST_INT:
03517
03518 val = INTVAL (rhs);
03519
03520
03521 for (mode = GET_CLASS_NARROWEST_MODE (MODE_INT), i = 0;
03522 GET_MODE_BITSIZE (mode) < BITS_PER_WORD;
03523 mode = GET_MODE_WIDER_MODE (mode), i++)
03524 {
03525 val2 = trunc_int_for_mode (val, mode);
03526 if (val2 == val && *source_mode == MAX_MACHINE_MODE)
03527 *source_mode = mode;
03528 if (val == (val & (HOST_WIDE_INT)GET_MODE_MASK (mode))
03529 && *source_mode_unsigned == MAX_MACHINE_MODE)
03530 *source_mode_unsigned = mode;
03531 if (*source_mode != MAX_MACHINE_MODE
03532 && *source_mode_unsigned !=MAX_MACHINE_MODE)
03533 return EXTENDED_DEF;
03534 }
03535 if (*source_mode != MAX_MACHINE_MODE
03536 || *source_mode_unsigned !=MAX_MACHINE_MODE)
03537 return EXTENDED_DEF;
03538 return NOT_RELEVANT;
03539 default:
03540 return NOT_RELEVANT;
03541 }
03542 default:
03543 gcc_unreachable ();
03544 }
03545 }
03546
03547
03548
03549
03550
03551 static void
03552 see_update_defs_relevancy (void)
03553 {
03554 struct see_entry_extra_info *curr_entry_extra_info;
03555 unsigned int i;
03556 rtx insn = NULL;
03557 rtx reg = NULL;
03558 enum entry_type et;
03559 enum machine_mode source_mode;
03560 enum machine_mode source_mode_unsigned;
03561
03562 if (!df || !def_entry)
03563 return;
03564
03565 for (i = 0; i < defs_num; i++)
03566 {
03567 insn = DF_REF_INSN (DF_DEFS_GET (df, i));
03568 reg = DF_REF_REAL_REG (DF_DEFS_GET (df, i));
03569
03570 et = see_analyze_one_def (insn, &source_mode, &source_mode_unsigned);
03571
03572 curr_entry_extra_info = xmalloc (sizeof (struct see_entry_extra_info));
03573 curr_entry_extra_info->relevancy = et;
03574 curr_entry_extra_info->local_relevancy = et;
03575 if (et != EXTENDED_DEF)
03576 {
03577 curr_entry_extra_info->source_mode = source_mode;
03578 curr_entry_extra_info->local_source_mode = source_mode;
03579 }
03580 else
03581 {
03582 curr_entry_extra_info->source_mode_signed = source_mode;
03583 curr_entry_extra_info->source_mode_unsigned = source_mode_unsigned;
03584 }
03585 def_entry[i].extra_info = curr_entry_extra_info;
03586 def_entry[i].reg = NULL;
03587 def_entry[i].pred = NULL;
03588
03589 if (dump_file)
03590 {
03591 if (et == NOT_RELEVANT)
03592 {
03593 fprintf (dump_file, "d%i insn %i reg %i ",
03594 i, (insn ? INSN_UID (insn) : -1), REGNO (reg));
03595 fprintf (dump_file, "NOT RELEVANT \n");
03596 }
03597 else
03598 {
03599 fprintf (dump_file, "d%i insn %i reg %i ",
03600 i ,INSN_UID (insn), REGNO (reg));
03601 fprintf (dump_file, "RELEVANT - ");
03602 switch (et)
03603 {
03604 case SIGN_EXTENDED_DEF :
03605 fprintf (dump_file, "SIGN_EXTENDED_DEF, source_mode = %s\n",
03606 GET_MODE_NAME (source_mode));
03607 break;
03608 case ZERO_EXTENDED_DEF :
03609 fprintf (dump_file, "ZERO_EXTENDED_DEF, source_mode = %s\n",
03610 GET_MODE_NAME (source_mode));
03611 break;
03612 case EXTENDED_DEF :
03613 fprintf (dump_file, "EXTENDED_DEF, ");
03614 if (source_mode != MAX_MACHINE_MODE
03615 && source_mode_unsigned != MAX_MACHINE_MODE)
03616 {
03617 fprintf (dump_file, "positive const, ");
03618 fprintf (dump_file, "source_mode_signed = %s, ",
03619 GET_MODE_NAME (source_mode));
03620 fprintf (dump_file, "source_mode_unsigned = %s\n",
03621 GET_MODE_NAME (source_mode_unsigned));
03622 }
03623 else if (source_mode != MAX_MACHINE_MODE)
03624 fprintf (dump_file, "source_mode_signed = %s\n",
03625 GET_MODE_NAME (source_mode));
03626 else
03627 fprintf (dump_file, "source_mode_unsigned = %s\n",
03628 GET_MODE_NAME (source_mode_unsigned));
03629 break;
03630 default :
03631 gcc_unreachable ();
03632 }
03633 }
03634 }
03635 }
03636 }
03637
03638
03639
03640
03641
03642
03643
03644
03645
03646
03647 static bool
03648 see_propagate_extensions_to_uses (void)
03649 {
03650 unsigned int i = 0;
03651 int num_relevant_uses;
03652 int num_relevant_defs;
03653
03654 if (dump_file)
03655 fprintf (dump_file,
03656 "* Phase 1: Propagate extensions to uses. *\n");
03657
03658
03659 see_update_defs_relevancy ();
03660 see_update_uses_relevancy ();
03661
03662
03663
03664
03665
03666 for (i = 0; i < uses_num; i++)
03667 union_defs (df, DF_USES_GET (df, i), def_entry, use_entry,
03668 see_update_leader_extra_info);
03669
03670
03671
03672 num_relevant_uses = see_handle_relevant_uses ();
03673
03674 if (num_relevant_uses < 0)
03675 return false;
03676
03677
03678
03679 num_relevant_defs = see_handle_relevant_defs ();
03680
03681 if (num_relevant_defs < 0)
03682 return false;
03683
03684 return num_relevant_uses > 0 || num_relevant_defs > 0;
03685 }
03686
03687
03688
03689
03690 static void
03691 see_main (void)
03692 {
03693 bool cont = false;
03694 int i = 0;
03695
03696
03697 see_initialize_data_structures ();
03698
03699
03700 cont = see_propagate_extensions_to_uses ();
03701
03702 if (cont)
03703 {
03704 init_recog ();
03705
03706
03707 see_merge_and_eliminate_extensions ();
03708
03709
03710 see_execute_LCM ();
03711
03712
03713 see_commit_changes ();
03714
03715 if (dump_file)
03716 {
03717
03718 fprintf (dump_file, "see_pre_extension_hash:\n");
03719 htab_traverse (see_pre_extension_hash, see_print_pre_extension_expr,
03720 NULL);
03721
03722 for (i = 0; i < last_bb; i++)
03723 {
03724 if (see_bb_hash_ar[i])
03725
03726
03727 {
03728 fprintf (dump_file,
03729 "Searching register properties in bb %d\n", i);
03730 htab_traverse (see_bb_hash_ar[i],
03731 see_print_register_properties, NULL);
03732 }
03733 }
03734 }
03735 }
03736
03737
03738 see_free_data_structures ();
03739 }
03740
03741
03742 static bool
03743 gate_handle_see (void)
03744 {
03745 return optimize > 1 && flag_see;
03746 }
03747
03748 static unsigned int
03749 rest_of_handle_see (void)
03750 {
03751 int no_new_pseudos_bcp = no_new_pseudos;
03752
03753 no_new_pseudos = 0;
03754 see_main ();
03755 no_new_pseudos = no_new_pseudos_bcp;
03756
03757 delete_trivially_dead_insns (get_insns (), max_reg_num ());
03758 update_life_info_in_dirty_blocks (UPDATE_LIFE_GLOBAL_RM_NOTES,
03759 (PROP_DEATH_NOTES));
03760 cleanup_cfg (CLEANUP_EXPENSIVE);
03761 reg_scan (get_insns (), max_reg_num ());
03762
03763 return 0;
03764 }
03765
03766 struct tree_opt_pass pass_see =
03767 {
03768 "see",
03769 gate_handle_see,
03770 rest_of_handle_see,
03771 NULL,
03772 NULL,
03773 0,
03774 TV_SEE,
03775 0,
03776 0,
03777 0,
03778 0,
03779 TODO_dump_func,
03780 'u'
03781 };
03782