diff options
author | Zack Weinberg <zack@gcc.gnu.org> | 2003-10-04 21:36:08 +0000 |
---|---|---|
committer | Zack Weinberg <zack@gcc.gnu.org> | 2003-10-04 21:36:08 +0000 |
commit | c9034561031a91eb1bb00c94c4d3d1b9bdf5d7e4 (patch) | |
tree | f6324710f59b8320a13b6d0d5d1869cf5e937d30 /gcc/optabs.c | |
parent | ce59b97aeeec8d9cc9754a29b88081b8db4323dc (diff) | |
download | gcc-c9034561031a91eb1bb00c94c4d3d1b9bdf5d7e4.zip gcc-c9034561031a91eb1bb00c94c4d3d1b9bdf5d7e4.tar.gz gcc-c9034561031a91eb1bb00c94c4d3d1b9bdf5d7e4.tar.bz2 |
* libfuncs.h
(LTI_eqhf2, LTI_nehf2, LTI_gthf2, LTI_gehf2, LTI_lthf2)
(LTI_lehf2, LTI_unordhf2, LTI_eqsf2, LTI_nesf2, LTI_gtsf2)
(LTI_gesf2, LTI_ltsf2, LTI_lesf2, LTI_unordsf2, LTI_eqdf2)
(LTI_nedf2, LTI_gtdf2, LTI_gedf2, LTI_ltdf2, LTI_ledf2)
(LTI_unorddf2, LTI_eqxf2, LTI_nexf2, LTI_gtxf2, LTI_gexf2)
(LTI_ltxf2, LTI_lexf2, LTI_unordxf2, LTI_eqtf2, LTI_netf2)
(LTI_gttf2, LTI_getf2, LTI_lttf2, LTI_letf2, LTI_unordtf2)
(eqhf2_libfunc, nehf2_libfunc, gthf2_libfunc, gehf2_libfunc)
(lthf2_libfunc, lehf2_libfunc, unordhf2_libfunc, eqsf2_libfunc)
(nesf2_libfunc, gtsf2_libfunc, gesf2_libfunc, ltsf2_libfunc)
(lesf2_libfunc, unordsf2_libfunc eqdf2_libfunc, nedf2_libfunc)
(gtdf2_libfunc, gedf2_libfunc, ltdf2_libfunc, ledf2_libfunc)
(unorddf2_libfunc eqxf2_libfunc, nexf2_libfunc, gtxf2_libfunc)
(gexf2_libfunc, ltxf2_libfunc, lexf2_libfunc, unordxf2_libfunc
(eqtf2_libfunc, netf2_libfunc, gttf2_libfunc, getf2_libfunc)
(lttf2_libfunc, letf2_libfunc, unordtf2_libfunc):
Delete.
* optabs.h (OTI_eq, OTI_ne, OTI_gt, OTI_ge, OTI_lt, OTI_le)
(OTI_unord, eq_optab, ne_optab, gt_optab, ge_optab, lt_optab)
(le_optab, unord_optab): New.
* optabs.c (prepare_float_lib_cmp): Rewrite. Get the libfuncs
from the code_to_optab table, not a giant switch; use
swap_condition; do widening only if a comparison function that
we can call exists in a wider mode, not if a cmp_optab insn or
libfunc exists in a wider mode; call protect_from_queue
exactly once on each operand.
(init_optabs): Initialize the new optabs, not the deleted libfuncs.
* config/gofast.h, config/ia64/ia64.c, config/mips/mips.c
* config/pa/pa.c, config/rs6000/rs6000.c, config/sparc/sparc.c:
Set floating point comparison libfuncs using set_optab_libfunc
on the appropriate optab.
* config/ia64/ia64.c (ia64_hpux_init_libfuncs): Fix typo.
* config/rs6000/rs6000.c (rs6000_init_libfuncs): Correct ABI
selector conditionals.
From-SVN: r72101
Diffstat (limited to 'gcc/optabs.c')
-rw-r--r-- | gcc/optabs.c | 397 |
1 files changed, 43 insertions, 354 deletions
diff --git a/gcc/optabs.c b/gcc/optabs.c index bd1b42a..bd9306d 100644 --- a/gcc/optabs.c +++ b/gcc/optabs.c @@ -3872,336 +3872,47 @@ prepare_float_lib_cmp (rtx *px, rtx *py, enum rtx_code *pcomparison, enum machine_mode *pmode, int *punsignedp) { enum rtx_code comparison = *pcomparison; - rtx tmp; - rtx x = *px = protect_from_queue (*px, 0); - rtx y = *py = protect_from_queue (*py, 0); - enum machine_mode mode = GET_MODE (x); + enum rtx_code swapped = swap_condition (comparison); + rtx x = protect_from_queue (*px, 0); + rtx y = protect_from_queue (*py, 0); + enum machine_mode orig_mode = GET_MODE (x); + enum machine_mode mode; rtx libfunc = 0; rtx result; - if (mode == HFmode) - switch (comparison) - { - case EQ: - libfunc = eqhf2_libfunc; - break; - - case NE: - libfunc = nehf2_libfunc; - break; - - case GT: - libfunc = gthf2_libfunc; - if (libfunc == NULL_RTX) - { - tmp = x; x = y; y = tmp; - *pcomparison = LT; - libfunc = lthf2_libfunc; - } - break; - - case GE: - libfunc = gehf2_libfunc; - if (libfunc == NULL_RTX) - { - tmp = x; x = y; y = tmp; - *pcomparison = LE; - libfunc = lehf2_libfunc; - } - break; - - case LT: - libfunc = lthf2_libfunc; - if (libfunc == NULL_RTX) - { - tmp = x; x = y; y = tmp; - *pcomparison = GT; - libfunc = gthf2_libfunc; - } - break; - - case LE: - libfunc = lehf2_libfunc; - if (libfunc == NULL_RTX) - { - tmp = x; x = y; y = tmp; - *pcomparison = GE; - libfunc = gehf2_libfunc; - } - break; - - case UNORDERED: - libfunc = unordhf2_libfunc; - break; - - default: - break; - } - else if (mode == SFmode) - switch (comparison) - { - case EQ: - libfunc = eqsf2_libfunc; - break; - - case NE: - libfunc = nesf2_libfunc; - break; - - case GT: - libfunc = gtsf2_libfunc; - if (libfunc == NULL_RTX) - { - tmp = x; x = y; y = tmp; - *pcomparison = LT; - libfunc = ltsf2_libfunc; - } - break; - - case GE: - libfunc = gesf2_libfunc; - if (libfunc == NULL_RTX) - { - tmp = x; x = y; y = tmp; - *pcomparison = LE; - libfunc = lesf2_libfunc; - } - break; - - case LT: - libfunc = ltsf2_libfunc; - if (libfunc == NULL_RTX) - { - tmp = x; x = y; y = tmp; - *pcomparison = GT; - libfunc = gtsf2_libfunc; - } - break; - - case LE: - libfunc = lesf2_libfunc; - if (libfunc == NULL_RTX) - { - tmp = x; x = y; y = tmp; - *pcomparison = GE; - libfunc = gesf2_libfunc; - } - break; - - case UNORDERED: - libfunc = unordsf2_libfunc; - break; - - default: - break; - } - else if (mode == DFmode) - switch (comparison) - { - case EQ: - libfunc = eqdf2_libfunc; - break; - - case NE: - libfunc = nedf2_libfunc; - break; - - case GT: - libfunc = gtdf2_libfunc; - if (libfunc == NULL_RTX) - { - tmp = x; x = y; y = tmp; - *pcomparison = LT; - libfunc = ltdf2_libfunc; - } - break; - - case GE: - libfunc = gedf2_libfunc; - if (libfunc == NULL_RTX) - { - tmp = x; x = y; y = tmp; - *pcomparison = LE; - libfunc = ledf2_libfunc; - } - break; - - case LT: - libfunc = ltdf2_libfunc; - if (libfunc == NULL_RTX) - { - tmp = x; x = y; y = tmp; - *pcomparison = GT; - libfunc = gtdf2_libfunc; - } - break; - - case LE: - libfunc = ledf2_libfunc; - if (libfunc == NULL_RTX) - { - tmp = x; x = y; y = tmp; - *pcomparison = GE; - libfunc = gedf2_libfunc; - } - break; - - case UNORDERED: - libfunc = unorddf2_libfunc; - break; - - default: - break; - } - else if (mode == XFmode) - switch (comparison) - { - case EQ: - libfunc = eqxf2_libfunc; - break; - - case NE: - libfunc = nexf2_libfunc; - break; - - case GT: - libfunc = gtxf2_libfunc; - if (libfunc == NULL_RTX) - { - tmp = x; x = y; y = tmp; - *pcomparison = LT; - libfunc = ltxf2_libfunc; - } - break; - - case GE: - libfunc = gexf2_libfunc; - if (libfunc == NULL_RTX) - { - tmp = x; x = y; y = tmp; - *pcomparison = LE; - libfunc = lexf2_libfunc; - } - break; - - case LT: - libfunc = ltxf2_libfunc; - if (libfunc == NULL_RTX) - { - tmp = x; x = y; y = tmp; - *pcomparison = GT; - libfunc = gtxf2_libfunc; - } - break; - - case LE: - libfunc = lexf2_libfunc; - if (libfunc == NULL_RTX) - { - tmp = x; x = y; y = tmp; - *pcomparison = GE; - libfunc = gexf2_libfunc; - } - break; - - case UNORDERED: - libfunc = unordxf2_libfunc; - break; - - default: - break; - } - else if (mode == TFmode) - switch (comparison) - { - case EQ: - libfunc = eqtf2_libfunc; - break; - - case NE: - libfunc = netf2_libfunc; - break; - - case GT: - libfunc = gttf2_libfunc; - if (libfunc == NULL_RTX) - { - tmp = x; x = y; y = tmp; - *pcomparison = LT; - libfunc = lttf2_libfunc; - } - break; - - case GE: - libfunc = getf2_libfunc; - if (libfunc == NULL_RTX) - { - tmp = x; x = y; y = tmp; - *pcomparison = LE; - libfunc = letf2_libfunc; - } - break; - - case LT: - libfunc = lttf2_libfunc; - if (libfunc == NULL_RTX) - { - tmp = x; x = y; y = tmp; - *pcomparison = GT; - libfunc = gttf2_libfunc; - } - break; - - case LE: - libfunc = letf2_libfunc; - if (libfunc == NULL_RTX) - { - tmp = x; x = y; y = tmp; - *pcomparison = GE; - libfunc = getf2_libfunc; - } - break; - - case UNORDERED: - libfunc = unordtf2_libfunc; - break; - - default: - break; - } - else + for (mode = orig_mode; mode != VOIDmode; mode = GET_MODE_WIDER_MODE (mode)) { - enum machine_mode wider_mode; + if ((libfunc = code_to_optab[comparison]->handlers[mode].libfunc)) + break; - for (wider_mode = GET_MODE_WIDER_MODE (mode); wider_mode != VOIDmode; - wider_mode = GET_MODE_WIDER_MODE (wider_mode)) + if ((libfunc = code_to_optab[swapped]->handlers[mode].libfunc)) { - if ((cmp_optab->handlers[(int) wider_mode].insn_code - != CODE_FOR_nothing) - || (cmp_optab->handlers[(int) wider_mode].libfunc != 0)) - { - x = protect_from_queue (x, 0); - y = protect_from_queue (y, 0); - *px = convert_to_mode (wider_mode, x, 0); - *py = convert_to_mode (wider_mode, y, 0); - prepare_float_lib_cmp (px, py, pcomparison, pmode, punsignedp); - return; - } + rtx tmp; + tmp = x; x = y; y = tmp; + comparison = swapped; + break; } - abort (); } - if (libfunc == 0) + if (mode == VOIDmode) abort (); + if (mode != orig_mode) + { + x = convert_to_mode (mode, x, 0); + y = convert_to_mode (mode, y, 0); + } + + if (comparison == UNORDERED + || FLOAT_LIB_COMPARE_RETURNS_BOOL (mode, comparison)) + comparison = NE; + result = emit_library_call_value (libfunc, NULL_RTX, LCT_CONST_MAKE_BLOCK, word_mode, 2, x, mode, y, mode); *px = result; *py = const0_rtx; *pmode = word_mode; - if (comparison == UNORDERED - || FLOAT_LIB_COMPARE_RETURNS_BOOL (mode, comparison)) - *pcomparison = NE; + *pcomparison = comparison; *punsignedp = 0; } @@ -5430,6 +5141,15 @@ init_optabs (void) ucmp_optab = init_optab (UNKNOWN); tst_optab = init_optab (UNKNOWN); + + eq_optab = init_optab (EQ); + ne_optab = init_optab (NE); + gt_optab = init_optab (GT); + ge_optab = init_optab (GE); + lt_optab = init_optab (LT); + le_optab = init_optab (LE); + unord_optab = init_optab (UNORDERED); + neg_optab = init_optab (NEG); negv_optab = init_optabv (NEG); abs_optab = init_optab (ABS); @@ -5530,6 +5250,15 @@ init_optabs (void) init_integral_libfuncs (ucmp_optab, "ucmp", '2'); init_floating_libfuncs (cmp_optab, "cmp", '2'); + /* EQ etc are floating point only. */ + init_floating_libfuncs (eq_optab, "eq", '2'); + init_floating_libfuncs (ne_optab, "ne", '2'); + init_floating_libfuncs (gt_optab, "gt", '2'); + init_floating_libfuncs (ge_optab, "ge", '2'); + init_floating_libfuncs (lt_optab, "lt", '2'); + init_floating_libfuncs (le_optab, "le", '2'); + init_floating_libfuncs (unord_optab, "unord", '2'); + /* Use cabs for DC complex abs, since systems generally have cabs. Don't define any libcall for SCmode, so that cabs will be used. */ abs_optab->handlers[(int) DCmode].libfunc @@ -5575,46 +5304,6 @@ init_optabs (void) unwind_sjlj_unregister_libfunc = init_one_libfunc ("_Unwind_SjLj_Unregister"); - eqhf2_libfunc = init_one_libfunc ("__eqhf2"); - nehf2_libfunc = init_one_libfunc ("__nehf2"); - gthf2_libfunc = init_one_libfunc ("__gthf2"); - gehf2_libfunc = init_one_libfunc ("__gehf2"); - lthf2_libfunc = init_one_libfunc ("__lthf2"); - lehf2_libfunc = init_one_libfunc ("__lehf2"); - unordhf2_libfunc = init_one_libfunc ("__unordhf2"); - - eqsf2_libfunc = init_one_libfunc ("__eqsf2"); - nesf2_libfunc = init_one_libfunc ("__nesf2"); - gtsf2_libfunc = init_one_libfunc ("__gtsf2"); - gesf2_libfunc = init_one_libfunc ("__gesf2"); - ltsf2_libfunc = init_one_libfunc ("__ltsf2"); - lesf2_libfunc = init_one_libfunc ("__lesf2"); - unordsf2_libfunc = init_one_libfunc ("__unordsf2"); - - eqdf2_libfunc = init_one_libfunc ("__eqdf2"); - nedf2_libfunc = init_one_libfunc ("__nedf2"); - gtdf2_libfunc = init_one_libfunc ("__gtdf2"); - gedf2_libfunc = init_one_libfunc ("__gedf2"); - ltdf2_libfunc = init_one_libfunc ("__ltdf2"); - ledf2_libfunc = init_one_libfunc ("__ledf2"); - unorddf2_libfunc = init_one_libfunc ("__unorddf2"); - - eqxf2_libfunc = init_one_libfunc ("__eqxf2"); - nexf2_libfunc = init_one_libfunc ("__nexf2"); - gtxf2_libfunc = init_one_libfunc ("__gtxf2"); - gexf2_libfunc = init_one_libfunc ("__gexf2"); - ltxf2_libfunc = init_one_libfunc ("__ltxf2"); - lexf2_libfunc = init_one_libfunc ("__lexf2"); - unordxf2_libfunc = init_one_libfunc ("__unordxf2"); - - eqtf2_libfunc = init_one_libfunc ("__eqtf2"); - netf2_libfunc = init_one_libfunc ("__netf2"); - gttf2_libfunc = init_one_libfunc ("__gttf2"); - getf2_libfunc = init_one_libfunc ("__getf2"); - lttf2_libfunc = init_one_libfunc ("__lttf2"); - letf2_libfunc = init_one_libfunc ("__letf2"); - unordtf2_libfunc = init_one_libfunc ("__unordtf2"); - floatsisf_libfunc = init_one_libfunc ("__floatsisf"); floatdisf_libfunc = init_one_libfunc ("__floatdisf"); floattisf_libfunc = init_one_libfunc ("__floattisf"); |