From: Martin Sebor Date: Mon, 13 Jan 2020 13:20:41 +0000 (+0000) Subject: PR tree-optimization/93213 - wrong code with -Og -foptimize-strlen X-Git-Url: https://git.libre-soc.org/?a=commitdiff_plain;h=e13f37d9f7600d337f8a4589b5f985ded9e05852;p=gcc.git PR tree-optimization/93213 - wrong code with -Og -foptimize-strlen gcc/testsuite/ChangeLog: PR tree-optimization/93213 * gcc.c-torture/execute/pr93213.c: New test. gcc/ChangeLog: PR tree-optimization/93213 * tree-ssa-strlen.c (handle_store): Only allow single-byte nul-over-nul stores to be eliminated. --- diff --git a/gcc/ChangeLog b/gcc/ChangeLog index fc25469452c..2a4b03a32d3 100644 --- a/gcc/ChangeLog +++ b/gcc/ChangeLog @@ -1,3 +1,9 @@ +2020-01-13 Martin Sebor + + PR tree-optimization/93213 + * tree-ssa-strlen.c (handle_store): Only allow single-byte nul-over-nul + stores to be eliminated. + 2020-01-13 Martin Liska * opts.c (print_help): Do not print CL_PARAM diff --git a/gcc/testsuite/ChangeLog b/gcc/testsuite/ChangeLog index cccc2853ed5..f14638c5079 100644 --- a/gcc/testsuite/ChangeLog +++ b/gcc/testsuite/ChangeLog @@ -1,3 +1,8 @@ +2020-01-13 Martin Sebor + + PR tree-optimization/93213 + * gcc.c-torture/execute/pr93213.c: New test. + 2020-01-10 Thomas Schwinge * c-c++-common/goacc/host_data-1.c: Extend. diff --git a/gcc/testsuite/gcc.c-torture/execute/pr93213.c b/gcc/testsuite/gcc.c-torture/execute/pr93213.c new file mode 100644 index 00000000000..86d0465bf67 --- /dev/null +++ b/gcc/testsuite/gcc.c-torture/execute/pr93213.c @@ -0,0 +1,50 @@ +/* PR tree-optimization/93213 - wrong code on a multibyte store with + -Og -foptimize-strlen + { dg-require-effective-target int128 } + { dg-additional-options "-Og -foptimize-strlen" } */ + +typedef unsigned __INT16_TYPE__ u16; +typedef unsigned __INT32_TYPE__ u32; +typedef unsigned __int128 u128; + +static inline u128 +foo (u16 u16_1, u32 u32_1, u128 u128_1) +{ +#if __BYTE_ORDER__ == __ORDER_LITTLE_ENDIAN__ + u128 u128_0 = 0; + u128_1 -= __builtin_mul_overflow (u32_1, u16_1, &u32_1); + __builtin_memmove (&u16_1, &u128_0, 2); + __builtin_memmove (&u16_1, &u128_1, 1); + return u16_1; +#else + return 0xff; +#endif +} + +__attribute__ ((noipa)) void +bar (void) +{ + char a[] = { 1, 2 }; + const char b[] = { 0, 0 }; + const char c[] = { 2 }; + __builtin_memcpy (a, b, 2); + // The above is transformed into + // MEM [(char * {ref-all})&a] = 0; + // which was then dropped because of the non-nul store below. + __builtin_memcpy (a, c, 1); + + volatile char *p = a; + if (p[0] != 2 || p[1] != 0) + __builtin_abort (); +} + +int +main (void) +{ + u16 x = foo (-1, -1, 0); + if (x != 0xff) + __builtin_abort (); + + bar (); + return 0; +} diff --git a/gcc/tree-ssa-strlen.c b/gcc/tree-ssa-strlen.c index 4f2a0ad6a19..ad9e98973b1 100644 --- a/gcc/tree-ssa-strlen.c +++ b/gcc/tree-ssa-strlen.c @@ -5239,10 +5239,10 @@ handle_store (gimple_stmt_iterator *gsi, bool *zero_write, } } - if (si != NULL && offset == 0 && storing_all_zeros_p) + if (si != NULL && offset == 0 && storing_all_zeros_p && lenrange[2] == 1) { - /* Allow adjust_last_stmt to remove it if the stored '\0' - is immediately overwritten. */ + /* For single-byte stores only, allow adjust_last_stmt to remove + the statement if the stored '\0' is immediately overwritten. */ laststmt.stmt = stmt; laststmt.len = build_int_cst (size_type_node, 1); laststmt.stridx = si->idx;