diff options
author | danglin <danglin@138bc75d-0d04-0410-961f-82ee72b054a4> | 2014-01-26 16:30:27 +0000 |
---|---|---|
committer | danglin <danglin@138bc75d-0d04-0410-961f-82ee72b054a4> | 2014-01-26 16:30:27 +0000 |
commit | 68385542f9a41c400c6ef16e749e3a655a3077d4 (patch) | |
tree | 898a7858166664e9711e645bc93fd20f2a0b02d9 /gcc/config/pa | |
parent | 1688a966ed46deda9b998a52c618313930eed5bf (diff) | |
download | gcc-68385542f9a41c400c6ef16e749e3a655a3077d4.tar.gz |
* config/pa/pa.md (call): Generate indirect long calls to non-local
functions when outputing 32-bit code.
(call_value): Likewise except for special call to buggy powf function.
git-svn-id: svn+ssh://gcc.gnu.org/svn/gcc/trunk@207121 138bc75d-0d04-0410-961f-82ee72b054a4
Diffstat (limited to 'gcc/config/pa')
-rw-r--r-- | gcc/config/pa/pa.md | 43 |
1 files changed, 33 insertions, 10 deletions
diff --git a/gcc/config/pa/pa.md b/gcc/config/pa/pa.md index 94ba162a43e..e55d0b86b90 100644 --- a/gcc/config/pa/pa.md +++ b/gcc/config/pa/pa.md @@ -7087,7 +7087,17 @@ add,l %2,%3,%3\;bv,n %%r0(%3)" if (TARGET_PORTABLE_RUNTIME) op = force_reg (SImode, XEXP (operands[0], 0)); else - op = XEXP (operands[0], 0); + { + op = XEXP (operands[0], 0); + + /* Generate indirect long calls to non-local functions. */ + if (!TARGET_64BIT && TARGET_LONG_CALLS && GET_CODE (op) == SYMBOL_REF) + { + tree call_decl = SYMBOL_REF_DECL (op); + if (!(call_decl && targetm.binds_local_p (call_decl))) + op = force_reg (word_mode, op); + } + } if (TARGET_64BIT) { @@ -7575,11 +7585,29 @@ add,l %2,%3,%3\;bv,n %%r0(%3)" rtx op; rtx dst = operands[0]; rtx nb = operands[2]; + bool call_powf = false; if (TARGET_PORTABLE_RUNTIME) op = force_reg (SImode, XEXP (operands[1], 0)); else - op = XEXP (operands[1], 0); + { + op = XEXP (operands[1], 0); + if (GET_CODE (op) == SYMBOL_REF) + { + /* Handle special call to buggy powf function. */ + if (TARGET_HPUX && !TARGET_DISABLE_FPREGS && !TARGET_SOFT_FLOAT + && !strcmp (targetm.strip_name_encoding (XSTR (op, 0)), "powf")) + call_powf = true; + + /* Generate indirect long calls to non-local functions. */ + else if (!TARGET_64BIT && TARGET_LONG_CALLS) + { + tree call_decl = SYMBOL_REF_DECL (op); + if (!(call_decl && targetm.binds_local_p (call_decl))) + op = force_reg (word_mode, op); + } + } + } if (TARGET_64BIT) { @@ -7639,8 +7667,7 @@ add,l %2,%3,%3\;bv,n %%r0(%3)" rtx r4 = gen_rtx_REG (word_mode, 4); if (GET_CODE (op) == SYMBOL_REF) { - if (TARGET_HPUX && !TARGET_DISABLE_FPREGS && !TARGET_SOFT_FLOAT - && !strcmp (targetm.strip_name_encoding (XSTR (op, 0)), "powf")) + if (call_powf) emit_call_insn (gen_call_val_powf_64bit (dst, op, nb, r4)); else emit_call_insn (gen_call_val_symref_64bit (dst, op, nb, r4)); @@ -7659,18 +7686,14 @@ add,l %2,%3,%3\;bv,n %%r0(%3)" { rtx r4 = gen_rtx_REG (word_mode, 4); - if (TARGET_HPUX && !TARGET_DISABLE_FPREGS && !TARGET_SOFT_FLOAT - && !strcmp (targetm.strip_name_encoding (XSTR (op, 0)), - "powf")) + if (call_powf) emit_call_insn (gen_call_val_powf_pic (dst, op, nb, r4)); else emit_call_insn (gen_call_val_symref_pic (dst, op, nb, r4)); } else { - if (TARGET_HPUX && !TARGET_DISABLE_FPREGS && !TARGET_SOFT_FLOAT - && !strcmp (targetm.strip_name_encoding (XSTR (op, 0)), - "powf")) + if (call_powf) emit_call_insn (gen_call_val_powf (dst, op, nb)); else emit_call_insn (gen_call_val_symref (dst, op, nb)); |