mirror of git://gcc.gnu.org/git/gcc.git
re PR target/50123 (cmpxchg generated for atomic and with zero/or with -1)
2011-11-29 Andrew MacLeod <amacleod@redhat.com> PR target/50123 * optabs.c (maybe_optimize_fetch_op): New. Look for more optimal instructions for a FECTH_OP or OP_FECTH sequence. (expand_atomic_fetch_op): Call maybe_optimize_fetch_op. * testsuite/gcc.dg/atomic-op-optimize.c: New. Test for optimizations. From-SVN: r181815
This commit is contained in:
parent
421ecf992e
commit
91f59d8bf4
|
@ -1,3 +1,10 @@
|
||||||
|
2011-11-29 Andrew MacLeod <amacleod@redhat.com>
|
||||||
|
|
||||||
|
PR target/50123
|
||||||
|
* optabs.c (maybe_optimize_fetch_op): New. Look for more optimal
|
||||||
|
instructions for a FECTH_OP or OP_FECTH sequence.
|
||||||
|
(expand_atomic_fetch_op): Call maybe_optimize_fetch_op.
|
||||||
|
|
||||||
2011-11-29 Uros Bizjak <ubizjak@gmail.com>
|
2011-11-29 Uros Bizjak <ubizjak@gmail.com>
|
||||||
|
|
||||||
* config/i386/sync.md (UNSPEC_LDA, UNSPEC_STA): New unspecs.
|
* config/i386/sync.md (UNSPEC_LDA, UNSPEC_STA): New unspecs.
|
||||||
|
|
40
gcc/optabs.c
40
gcc/optabs.c
|
@ -7943,6 +7943,41 @@ get_atomic_op_for_code (struct atomic_op_functions *op, enum rtx_code code)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* See if there is a more optimal way to implement the operation "*MEM CODE VAL"
|
||||||
|
using memory order MODEL. If AFTER is true the operation needs to return
|
||||||
|
the value of *MEM after the operation, otherwise the previous value.
|
||||||
|
TARGET is an optional place to place the result. The result is unused if
|
||||||
|
it is const0_rtx.
|
||||||
|
Return the result if there is a better sequence, otherwise NULL_RTX. */
|
||||||
|
|
||||||
|
static rtx
|
||||||
|
maybe_optimize_fetch_op (rtx target, rtx mem, rtx val, enum rtx_code code,
|
||||||
|
enum memmodel model, bool after)
|
||||||
|
{
|
||||||
|
/* If the value is prefetched, or not used, it may be possible to replace
|
||||||
|
the sequence with a native exchange operation. */
|
||||||
|
if (!after || target == const0_rtx)
|
||||||
|
{
|
||||||
|
/* fetch_and (&x, 0, m) can be replaced with exchange (&x, 0, m). */
|
||||||
|
if (code == AND && val == const0_rtx)
|
||||||
|
{
|
||||||
|
if (target == const0_rtx)
|
||||||
|
target = gen_reg_rtx (GET_MODE (mem));
|
||||||
|
return maybe_emit_atomic_exchange (target, mem, val, model);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* fetch_or (&x, -1, m) can be replaced with exchange (&x, -1, m). */
|
||||||
|
if (code == IOR && val == constm1_rtx)
|
||||||
|
{
|
||||||
|
if (target == const0_rtx)
|
||||||
|
target = gen_reg_rtx (GET_MODE (mem));
|
||||||
|
return maybe_emit_atomic_exchange (target, mem, val, model);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return NULL_RTX;
|
||||||
|
}
|
||||||
|
|
||||||
/* Try to emit an instruction for a specific operation varaition.
|
/* Try to emit an instruction for a specific operation varaition.
|
||||||
OPTAB contains the OP functions.
|
OPTAB contains the OP functions.
|
||||||
TARGET is an optional place to return the result. const0_rtx means unused.
|
TARGET is an optional place to return the result. const0_rtx means unused.
|
||||||
|
@ -8028,6 +8063,11 @@ expand_atomic_fetch_op (rtx target, rtx mem, rtx val, enum rtx_code code,
|
||||||
|
|
||||||
get_atomic_op_for_code (&optab, code);
|
get_atomic_op_for_code (&optab, code);
|
||||||
|
|
||||||
|
/* Check to see if there are any better instructions. */
|
||||||
|
result = maybe_optimize_fetch_op (target, mem, val, code, model, after);
|
||||||
|
if (result)
|
||||||
|
return result;
|
||||||
|
|
||||||
/* Check for the case where the result isn't used and try those patterns. */
|
/* Check for the case where the result isn't used and try those patterns. */
|
||||||
if (unused_result)
|
if (unused_result)
|
||||||
{
|
{
|
||||||
|
|
|
@ -1,3 +1,8 @@
|
||||||
|
2011-11-29 Andrew MacLeod <amacleod@redhat.com>
|
||||||
|
|
||||||
|
PR target/50123
|
||||||
|
* gcc.dg/atomic-op-optimize.c: New. Test for optimizations.
|
||||||
|
|
||||||
2011-11-29 Michael Meissner <meissner@linux.vnet.ibm.com>
|
2011-11-29 Michael Meissner <meissner@linux.vnet.ibm.com>
|
||||||
|
|
||||||
* gcc.target/powerpc/ppc-target-4.c: New file to test target
|
* gcc.target/powerpc/ppc-target-4.c: New file to test target
|
||||||
|
|
|
@ -0,0 +1,20 @@
|
||||||
|
/* Both these atomic operations should be optimized to an exchange operation.
|
||||||
|
Test that it at happens on x86 by making sure there are 2 xchg's and no
|
||||||
|
compare_exchange loop. */
|
||||||
|
|
||||||
|
/* { dg-require-effective-target sync_int_long } */
|
||||||
|
/* { dg-do compile { target { i?86-*-* x86_64-*-* } } } */
|
||||||
|
/* { dg-final { scan-assembler-times "cmpxchg" 0 } } */
|
||||||
|
/* { dg-final { scan-assembler-times "xchg" 2 } } */
|
||||||
|
|
||||||
|
int x;
|
||||||
|
|
||||||
|
int f()
|
||||||
|
{
|
||||||
|
return __atomic_fetch_and (&x, 0, __ATOMIC_RELAXED);
|
||||||
|
}
|
||||||
|
|
||||||
|
int g()
|
||||||
|
{
|
||||||
|
return __atomic_fetch_or (&x, -1, __ATOMIC_RELAXED);
|
||||||
|
}
|
Loading…
Reference in New Issue