On Mon, 16 Feb 2015, Thomas Preud'homme wrote:

> Hi,
> 
> The RTL cprop pass in GCC operates by doing a local constant/copy 
> propagation first and then a global one. In the local one, if a constant 
> cannot be propagated (eg. due to constraints of the destination 
> instruction) a copy propagation is done instead. However, at the global 
> level copy propagation is only tried if no constant can be propagated, 
> ie. if a constant can be propagated but the constraints of the 
> destination instruction forbids it, no copy propagation will be tried. 
> This patch fixes this issue. This solves the redundant ldr problem in 
> GCC32RM-439.

I think Steven is more familiar with this code.

Richard.

> ChangeLog entries are as follows:
> 
> *** gcc/ChangeLog ***
> 
> 2015-01-21 Thomas Preud'homme thomas.preudho...@arm.com
> 
>     * cprop.c (find_avail_set): Return up to two sets, one whose source is
>     a register and one whose source is a constant.  Sets are returned in
>     an array passed as parameter rather than as a return value.
>     (cprop_insn): Use a do while loop rather than a goto.  Try each of the
>     sets returned by find_avail_set, starting with the one whose source is
>     a constant.
> 
> 
> *** gcc/testsuite/ChangeLog ***
> 
> 2015-01-21 Thomas Preud'homme thomas.preudho...@arm.com
> 
>     * gcc.target/arm/pr64616.c: New file.
> 
> Following testing was done:
> 
>     Bootstrapped on x86_64 and ran the testsuite without regression
>     Build an arm-none-eabi cross-compilers and ran the testsuite without 
> regression with QEMU emulating a Cortex-M3
>     Compiled CSiBE targeting x86_64 and Cortex-M3 arm-none-eabi without 
> regression
> 
> diff --git a/gcc/cprop.c b/gcc/cprop.c
> index 4538291..c246d4b 100644
> --- a/gcc/cprop.c
> +++ b/gcc/cprop.c
> @@ -815,15 +815,15 @@ try_replace_reg (rtx from, rtx to, rtx_insn *insn)
>    return success;
>  }
>  
> -/* Find a set of REGNOs that are available on entry to INSN's block.  Return
> -   NULL no such set is found.  */
> +/* Find a set of REGNOs that are available on entry to INSN's block.  If 
> found,
> +   SET_RET[0] will be assigned a set with a register source and SET_RET[1] a
> +   set with a constant source.  If not found the corresponding entry is set 
> to
> +   NULL.  */
>  
> -static struct cprop_expr *
> -find_avail_set (int regno, rtx_insn *insn)
> +static void
> +find_avail_set (int regno, rtx_insn *insn, struct cprop_expr *set_ret[2])
>  {
> -  /* SET1 contains the last set found that can be returned to the caller for
> -     use in a substitution.  */
> -  struct cprop_expr *set1 = 0;
> +  set_ret[0] = set_ret[1] = NULL;
>  
>    /* Loops are not possible here.  To get a loop we would need two sets
>       available at the start of the block containing INSN.  i.e. we would
> @@ -863,8 +863,10 @@ find_avail_set (int regno, rtx_insn *insn)
>           If the source operand changed, we may still use it for the next
>           iteration of this loop, but we may not use it for substitutions.  */
>  
> -      if (cprop_constant_p (src) || reg_not_set_p (src, insn))
> -     set1 = set;
> +      if (cprop_constant_p (src))
> +     set_ret[1] = set;
> +      else if (reg_not_set_p (src, insn))
> +     set_ret[0] = set;
>  
>        /* If the source of the set is anything except a register, then
>        we have reached the end of the copy chain.  */
> @@ -875,10 +877,6 @@ find_avail_set (int regno, rtx_insn *insn)
>        and see if we have an available copy into SRC.  */
>        regno = REGNO (src);
>      }
> -
> -  /* SET1 holds the last set that was available and anticipatable at
> -     INSN.  */
> -  return set1;
>  }
>  
>  /* Subroutine of cprop_insn that tries to propagate constants into
> @@ -1044,40 +1042,41 @@ cprop_insn (rtx_insn *insn)
>    int changed = 0, changed_this_round;
>    rtx note;
>  
> -retry:
> -  changed_this_round = 0;
> -  reg_use_count = 0;
> -  note_uses (&PATTERN (insn), find_used_regs, NULL);
> +  do
> +    {
> +      changed_this_round = 0;
> +      reg_use_count = 0;
> +      note_uses (&PATTERN (insn), find_used_regs, NULL);
>  
> -  /* We may win even when propagating constants into notes.  */
> -  note = find_reg_equal_equiv_note (insn);
> -  if (note)
> -    find_used_regs (&XEXP (note, 0), NULL);
> +      /* We may win even when propagating constants into notes.  */
> +      note = find_reg_equal_equiv_note (insn);
> +      if (note)
> +     find_used_regs (&XEXP (note, 0), NULL);
>  
> -  for (i = 0; i < reg_use_count; i++)
> -    {
> -      rtx reg_used = reg_use_table[i];
> -      unsigned int regno = REGNO (reg_used);
> -      rtx src;
> -      struct cprop_expr *set;
> +      for (i = 0; i < reg_use_count; i++)
> +     {
> +       rtx reg_used = reg_use_table[i];
> +       unsigned int regno = REGNO (reg_used);
> +       rtx src_cst = NULL, src_reg = NULL;
> +       struct cprop_expr *set[2];
>  
> -      /* If the register has already been set in this block, there's
> -      nothing we can do.  */
> -      if (! reg_not_set_p (reg_used, insn))
> -     continue;
> +       /* If the register has already been set in this block, there's
> +          nothing we can do.  */
> +       if (! reg_not_set_p (reg_used, insn))
> +         continue;
>  
> -      /* Find an assignment that sets reg_used and is available
> -      at the start of the block.  */
> -      set = find_avail_set (regno, insn);
> -      if (! set)
> -     continue;
> +       /* Find an assignment that sets reg_used and is available
> +          at the start of the block.  */
> +       find_avail_set (regno, insn, set);
>  
> -      src = set->src;
> +       if (set[0])
> +         src_reg = set[0]->src;
> +       if (set[1])
> +         src_cst = set[1]->src;
>  
> -      /* Constant propagation.  */
> -      if (cprop_constant_p (src))
> -     {
> -          if (constprop_register (reg_used, src, insn))
> +       /* Constant propagation.  */
> +       if (src_cst && cprop_constant_p (src_cst)
> +           && constprop_register (reg_used, src_cst, insn))
>           {
>             changed_this_round = changed = 1;
>             global_const_prop_count++;
> @@ -1087,18 +1086,16 @@ retry:
>                          "GLOBAL CONST-PROP: Replacing reg %d in ", regno);
>                 fprintf (dump_file, "insn %d with constant ",
>                          INSN_UID (insn));
> -               print_rtl (dump_file, src);
> +               print_rtl (dump_file, src_cst);
>                 fprintf (dump_file, "\n");
>               }
>             if (insn->deleted ())
>               return 1;
>           }
> -     }
> -      else if (REG_P (src)
> -            && REGNO (src) >= FIRST_PSEUDO_REGISTER
> -            && REGNO (src) != regno)
> -     {
> -       if (try_replace_reg (reg_used, src, insn))
> +       else if (src_reg && REG_P (src_reg)
> +                && REGNO (src_reg) >= FIRST_PSEUDO_REGISTER
> +                && REGNO (src_reg) != regno
> +                && try_replace_reg (reg_used, src_reg, insn))
>           {
>             changed_this_round = changed = 1;
>             global_copy_prop_count++;
> @@ -1107,22 +1104,20 @@ retry:
>                 fprintf (dump_file,
>                          "GLOBAL COPY-PROP: Replacing reg %d in insn %d",
>                          regno, INSN_UID (insn));
> -               fprintf (dump_file, " with reg %d\n", REGNO (src));
> +               fprintf (dump_file, " with reg %d\n", REGNO (src_reg));
>               }
>  
>             /* The original insn setting reg_used may or may not now be
>                deletable.  We leave the deletion to DCE.  */
> -           /* FIXME: If it turns out that the insn isn't deletable,
> -              then we may have unnecessarily extended register lifetimes
> +           /* FIXME: If it turns out that the insn isn't deletable, then we
> +              then we may have unnecessarily extended register lifetimes and
>                and made things worse.  */
>           }
>       }
> -
> -      /* If try_replace_reg simplified the insn, the regs found
> -      by find_used_regs may not be valid anymore.  Start over.  */
> -      if (changed_this_round)
> -     goto retry;
>      }
> +  /* If try_replace_reg simplified the insn, the regs found
> +     by find_used_regs may not be valid anymore.  Start over.  */
> +  while (changed_this_round);
>  
>    if (changed && DEBUG_INSN_P (insn))
>      return 0;
> diff --git a/gcc/testsuite/gcc.target/arm/pr64616.c 
> b/gcc/testsuite/gcc.target/arm/pr64616.c
> new file mode 100644
> index 0000000..c686ffa
> --- /dev/null
> +++ b/gcc/testsuite/gcc.target/arm/pr64616.c
> @@ -0,0 +1,14 @@
> +/* { dg-do compile } */
> +/* { dg-options "-O2" } */
> +
> +int f (int);
> +unsigned int glob;
> +
> +void
> +g ()
> +{
> +  while (f (glob));
> +  glob = 5;
> +}
> +
> +/* { dg-final { scan-assembler-times "ldr" 2 } } */
> 
> Is this ok for stage1?
> 
> Best regards,
> 
> Thomas
> 
> 
> 
> 

-- 
Richard Biener <rguent...@suse.de>
SUSE LINUX GmbH, GF: Felix Imendoerffer, Jane Smithard, Jennifer Guild,
Dilip Upmanyu, Graham Norton HRB 21284 (AG Nuernberg)

Reply via email to