* [PATCH] Fix PR77826
@ 2016-10-04 10:34 Richard Biener
2016-10-05 11:38 ` Richard Biener
0 siblings, 1 reply; 13+ messages in thread
From: Richard Biener @ 2016-10-04 10:34 UTC (permalink / raw)
To: gcc-patches
The following will fix PR77826, the issue that in match.pd matching
up two things uses operand_equal_p which is too lax about the type
of the toplevel entity (at least for integer constants).
Bootstrap / regtest pending on x86_64-unknown-linux-gnu.
Richard.
2016-10-04 Richard Biener <rguenther@suse.de>
PR middle-end/77826
* genmatch.c (dt_operand::gen_match_op): Amend operand_equal_p
with types_match to handle type mismatched constants properly.
* gcc.dg/torture/pr77826.c: New testcase.
Index: gcc/genmatch.c
===================================================================
--- gcc/genmatch.c (revision 240739)
+++ gcc/genmatch.c (working copy)
@@ -2593,8 +2601,10 @@ dt_operand::gen_match_op (FILE *f, int i
{
char match_opname[20];
match_dop->get_name (match_opname);
- fprintf_indent (f, indent, "if (%s == %s || operand_equal_p (%s, %s, 0))\n",
- opname, match_opname, opname, match_opname);
+ fprintf_indent (f, indent, "if (%s == %s || (operand_equal_p (%s, %s, 0) "
+ "&& types_match (%s, %s)))\n",
+ opname, match_opname, opname, match_opname,
+ opname, match_opname);
fprintf_indent (f, indent + 2, "{\n");
return 1;
}
Index: gcc/testsuite/gcc.dg/torture/pr77826.c
===================================================================
--- gcc/testsuite/gcc.dg/torture/pr77826.c (revision 0)
+++ gcc/testsuite/gcc.dg/torture/pr77826.c (working copy)
@@ -0,0 +1,12 @@
+/* { dg-do compile } */
+
+void
+fi(unsigned long int *v0, unsigned int ow, int q2)
+{
+ if (ow + q2 != 0)
+ if (q2 == 1)
+ {
+ *v0 |= q2;
+ q2 ^= *v0;
+ }
+}
^ permalink raw reply [flat|nested] 13+ messages in thread
* Re: [PATCH] Fix PR77826
2016-10-04 10:34 [PATCH] Fix PR77826 Richard Biener
@ 2016-10-05 11:38 ` Richard Biener
2016-10-06 18:44 ` Marc Glisse
0 siblings, 1 reply; 13+ messages in thread
From: Richard Biener @ 2016-10-05 11:38 UTC (permalink / raw)
To: gcc-patches
On Tue, 4 Oct 2016, Richard Biener wrote:
>
> The following will fix PR77826, the issue that in match.pd matching
> up two things uses operand_equal_p which is too lax about the type
> of the toplevel entity (at least for integer constants).
>
> Bootstrap / regtest pending on x86_64-unknown-linux-gnu.
The following is what I have applied.
Richard.
2016-10-05 Richard Biener <rguenther@suse.de>
PR middle-end/77826
* genmatch.c (dt_operand::gen_match_op): Amend operand_equal_p
with types_match for GIMPLE code gen to handle type mismatched
constants properly.
(dt_operand::gen): Adjust.
* match.pd ((X /[ex] A) * A -> X): Properly handle converted
and constant A.
* gcc.dg/torture/pr77826.c: New testcase.
Index: gcc/testsuite/gcc.dg/torture/pr77826.c
===================================================================
--- gcc/testsuite/gcc.dg/torture/pr77826.c (revision 0)
+++ gcc/testsuite/gcc.dg/torture/pr77826.c (working copy)
@@ -0,0 +1,12 @@
+/* { dg-do compile } */
+
+void
+fi(unsigned long int *v0, unsigned int ow, int q2)
+{
+ if (ow + q2 != 0)
+ if (q2 == 1)
+ {
+ *v0 |= q2;
+ q2 ^= *v0;
+ }
+}
Index: gcc/genmatch.c
===================================================================
--- gcc/genmatch.c (revision 240744)
+++ gcc/genmatch.c (working copy)
@@ -1562,7 +1562,7 @@ struct dt_operand : public dt_node
void gen (FILE *, int, bool);
unsigned gen_predicate (FILE *, int, const char *, bool);
- unsigned gen_match_op (FILE *, int, const char *);
+ unsigned gen_match_op (FILE *, int, const char *, bool);
unsigned gen_gimple_expr (FILE *, int);
unsigned gen_generic_expr (FILE *, int, const char *);
@@ -2589,12 +2589,18 @@ dt_operand::gen_predicate (FILE *f, int
a capture-match. */
unsigned
-dt_operand::gen_match_op (FILE *f, int indent, const char *opname)
+dt_operand::gen_match_op (FILE *f, int indent, const char *opname, bool gimple)
{
char match_opname[20];
match_dop->get_name (match_opname);
- fprintf_indent (f, indent, "if (%s == %s || operand_equal_p (%s, %s, 0))\n",
- opname, match_opname, opname, match_opname);
+ if (gimple)
+ fprintf_indent (f, indent, "if (%s == %s || (operand_equal_p (%s, %s, 0) "
+ "&& types_match (%s, %s)))\n",
+ opname, match_opname, opname, match_opname,
+ opname, match_opname);
+ else
+ fprintf_indent (f, indent, "if (%s == %s || operand_equal_p (%s, %s, 0))\n",
+ opname, match_opname, opname, match_opname);
fprintf_indent (f, indent + 2, "{\n");
return 1;
}
@@ -2991,7 +2997,7 @@ dt_operand::gen (FILE *f, int indent, bo
else if (type == DT_TRUE)
;
else if (type == DT_MATCH)
- n_braces = gen_match_op (f, indent, opname);
+ n_braces = gen_match_op (f, indent, opname, gimple);
else
gcc_unreachable ();
Index: gcc/match.pd
===================================================================
--- gcc/match.pd (revision 240770)
+++ gcc/match.pd (working copy)
@@ -1773,9 +1803,13 @@ DEFINE_INT_AND_FLOAT_ROUND_FN (RINT)
/* (X /[ex] A) * A -> X. */
(simplify
- (mult (convert? (exact_div @0 @1)) @1)
- /* Look through a sign-changing conversion. */
- (convert @0))
+ (mult (convert1? (exact_div @0 @1)) (convert2? @2))
+ /* We cannot use matching captures here, since in the case of
+ constants we don't see the second conversion. Look through
+ a sign-changing or widening conversions. */
+ (if (operand_equal_p (@1, @2, 0)
+ && element_precision (@0) <= element_precision (type))
+ (convert @0)))
/* Canonicalization of binary operations. */
^ permalink raw reply [flat|nested] 13+ messages in thread
* Re: [PATCH] Fix PR77826
2016-10-05 11:38 ` Richard Biener
@ 2016-10-06 18:44 ` Marc Glisse
2016-10-07 7:36 ` Richard Biener
0 siblings, 1 reply; 13+ messages in thread
From: Marc Glisse @ 2016-10-06 18:44 UTC (permalink / raw)
To: Richard Biener; +Cc: gcc-patches
On Wed, 5 Oct 2016, Richard Biener wrote:
>> The following will fix PR77826, the issue that in match.pd matching
>> up two things uses operand_equal_p which is too lax about the type
>> of the toplevel entity (at least for integer constants).
>>
>> Bootstrap / regtest pending on x86_64-unknown-linux-gnu.
>
> The following is what I have applied.
>
> Richard.
>
> 2016-10-05 Richard Biener <rguenther@suse.de>
>
> PR middle-end/77826
> * genmatch.c (dt_operand::gen_match_op): Amend operand_equal_p
> with types_match for GIMPLE code gen to handle type mismatched
> constants properly.
I don't understand the disparity between generic and gimple here. Why let
(char)1 and (long)1 match in generic but not in gimple? And there are
probably several transformations in match.pd that could do with an update
if constants don't match anymore. Or did I misunderstand what the patch
does?
> * match.pd ((X /[ex] A) * A -> X): Properly handle converted
> and constant A.
This regressed
int f(int*a,int*b){return 4*(int)(b-a);}
--
Marc Glisse
^ permalink raw reply [flat|nested] 13+ messages in thread
* Re: [PATCH] Fix PR77826
2016-10-06 18:44 ` Marc Glisse
@ 2016-10-07 7:36 ` Richard Biener
2016-10-10 8:47 ` Marc Glisse
0 siblings, 1 reply; 13+ messages in thread
From: Richard Biener @ 2016-10-07 7:36 UTC (permalink / raw)
To: gcc-patches
On Thu, 6 Oct 2016, Marc Glisse wrote:
> On Wed, 5 Oct 2016, Richard Biener wrote:
>
> > > The following will fix PR77826, the issue that in match.pd matching
> > > up two things uses operand_equal_p which is too lax about the type
> > > of the toplevel entity (at least for integer constants).
> > >
> > > Bootstrap / regtest pending on x86_64-unknown-linux-gnu.
> >
> > The following is what I have applied.
> >
> > Richard.
> >
> > 2016-10-05 Richard Biener <rguenther@suse.de>
> >
> > PR middle-end/77826
> > * genmatch.c (dt_operand::gen_match_op): Amend operand_equal_p
> > with types_match for GIMPLE code gen to handle type mismatched
> > constants properly.
>
> I don't understand the disparity between generic and gimple here. Why let
> (char)1 and (long)1 match in generic but not in gimple? And there are probably
> several transformations in match.pd that could do with an update if constants
> don't match anymore. Or did I misunderstand what the patch does?
The disparity is mostly that with GENERIC unfolded trees such as (char)1
are a bug while in GIMPLE the fact that the match.pd machinery does
valueization makes those a "feature" we have to deal with. Originally
I've restricted GENERIC as well but with its types_match_p implementation
it resulted in too many missed matches.
I agree that some transforms would need updates - I've actually tried
to implement a warning for genmatch whenever seeing a match with
(T)@0 but there isn't any good existing place to sneak that in.
> > * match.pd ((X /[ex] A) * A -> X): Properly handle converted
> > and constant A.
>
> This regressed
> int f(int*a,int*b){return 4*(int)(b-a);}
This is because (int)(b-a) could be a truncation in which case
multiplying with 4 might not result in the same value as
b-a truncated(?). The comment before the unpatched patterns
said "sign-changing conversions" but nothign actually verified this.
Might be that truncations are indeed ok now that I think about it.
Btw, do you have a better suggestion as to how to handle the original
issue rather than not relying on operand_equal_p for constants?
Richard.
^ permalink raw reply [flat|nested] 13+ messages in thread
* Re: [PATCH] Fix PR77826
2016-10-07 7:36 ` Richard Biener
@ 2016-10-10 8:47 ` Marc Glisse
2016-10-11 9:09 ` Richard Biener
0 siblings, 1 reply; 13+ messages in thread
From: Marc Glisse @ 2016-10-10 8:47 UTC (permalink / raw)
To: Richard Biener; +Cc: gcc-patches
On Fri, 7 Oct 2016, Richard Biener wrote:
> On Thu, 6 Oct 2016, Marc Glisse wrote:
>
>> On Wed, 5 Oct 2016, Richard Biener wrote:
>>
>>>> The following will fix PR77826, the issue that in match.pd matching
>>>> up two things uses operand_equal_p which is too lax about the type
>>>> of the toplevel entity (at least for integer constants).
>>>>
>>>> Bootstrap / regtest pending on x86_64-unknown-linux-gnu.
>>>
>>> The following is what I have applied.
>>>
>>> Richard.
>>>
>>> 2016-10-05 Richard Biener <rguenther@suse.de>
>>>
>>> PR middle-end/77826
>>> * genmatch.c (dt_operand::gen_match_op): Amend operand_equal_p
>>> with types_match for GIMPLE code gen to handle type mismatched
>>> constants properly.
>>
>> I don't understand the disparity between generic and gimple here. Why let
>> (char)1 and (long)1 match in generic but not in gimple? And there are probably
>> several transformations in match.pd that could do with an update if constants
>> don't match anymore. Or did I misunderstand what the patch does?
>
> The disparity is mostly that with GENERIC unfolded trees such as (char)1
> are a bug while in GIMPLE the fact that the match.pd machinery does
> valueization makes those a "feature" we have to deal with. Originally
> I've restricted GENERIC as well but with its types_match_p implementation
> it resulted in too many missed matches.
I shouldn't have written (long)1, I meant the fact that 1 (as a char
constant) and 1 (as a long constant) will now be matching captures in
generic and not in gimple. If we are going in the direction of not
matching constants of different types, I'd rather do it consistently and
update the patterns as needed to avoid the missed optimizations. The
missed matches exist in gimple as well, and generic optimization seems
less important than gimple to me.
An example that regressed at -O (looking at the .optimized dump)
int f(int a, unsigned b){
a &= 1;
b &= 1;
return a&b;
}
If we stick to the old behavior, maybe we could have some genmatch magic
to help with the constant capture weirdness. With matching captures, we
could select which operand (among those supposed to be equivalent) is
actually captured more cleverly, either with an explicit marker, or by
giving priority to the one that is not immediatly below convert? in the
pattern.
And if we move to stricter matching, maybe genmatch could be updated so
convert could also match integer constants in some cases.
> I agree that some transforms would need updates - I've actually tried
> to implement a warning for genmatch whenever seeing a match with
> (T)@0 but there isn't any good existing place to sneak that in.
>>> * match.pd ((X /[ex] A) * A -> X): Properly handle converted
>>> and constant A.
>>
>> This regressed
>> int f(int*a,int*b){return 4*(int)(b-a);}
>
> This is because (int)(b-a) could be a truncation in which case
> multiplying with 4 might not result in the same value as
> b-a truncated(?). The comment before the unpatched patterns
> said "sign-changing conversions" but nothign actually verified this.
> Might be that truncations are indeed ok now that I think about it.
2015-05-22 Marc Glisse <marc.glisse@inria.fr>
PR tree-optimization/63387
* match.pd ((X /[ex] A) * A -> X): Remove unnecessary condition.
Apparently I forgot to remove the comment at that time :-(
> Btw, do you have a better suggestion as to how to handle the original
> issue rather than not relying on operand_equal_p for constants?
In previous cases, in order to get the right version of a matching
capture, we used non-matching captures and an explicit call to
operand_equal_p, for instance:
/* X - (X / Y) * Y is the same as X % Y. */
(simplify
(minus (convert1? @2) (convert2? (mult:c (trunc_div @0 @1) @1)))
/* We cannot use matching captures here, since in the case of
constants we really want the type of @0, not @2. */
(if (operand_equal_p (@0, @2, 0)
&& (INTEGRAL_TYPE_P (type) || VECTOR_INTEGER_TYPE_P (type)))
(convert (trunc_mod @0 @1))))
That seems to match the solutions that Jakub and you were discussing in
the PR, something like it should work (we can discuss the exact code).
I don't know if it is better. The old behavior of matching captures with
inconsistent types was confusing. A behavior with strict matching may
complicate things (will we duplicate patterns, or write operand_equal_p
explicitly to mimic the old behavior?). The recent inconsistency between
generic and gimple doesnt appeal to me much...
--
Marc Glisse
^ permalink raw reply [flat|nested] 13+ messages in thread
* Re: [PATCH] Fix PR77826
2016-10-10 8:47 ` Marc Glisse
@ 2016-10-11 9:09 ` Richard Biener
2016-10-11 20:35 ` Marc Glisse
0 siblings, 1 reply; 13+ messages in thread
From: Richard Biener @ 2016-10-11 9:09 UTC (permalink / raw)
To: gcc-patches
On Mon, 10 Oct 2016, Marc Glisse wrote:
> On Fri, 7 Oct 2016, Richard Biener wrote:
>
> > On Thu, 6 Oct 2016, Marc Glisse wrote:
> >
> > > On Wed, 5 Oct 2016, Richard Biener wrote:
> > >
> > > > > The following will fix PR77826, the issue that in match.pd matching
> > > > > up two things uses operand_equal_p which is too lax about the type
> > > > > of the toplevel entity (at least for integer constants).
> > > > >
> > > > > Bootstrap / regtest pending on x86_64-unknown-linux-gnu.
> > > >
> > > > The following is what I have applied.
> > > >
> > > > Richard.
> > > >
> > > > 2016-10-05 Richard Biener <rguenther@suse.de>
> > > >
> > > > PR middle-end/77826
> > > > * genmatch.c (dt_operand::gen_match_op): Amend operand_equal_p
> > > > with types_match for GIMPLE code gen to handle type mismatched
> > > > constants properly.
> > >
> > > I don't understand the disparity between generic and gimple here. Why let
> > > (char)1 and (long)1 match in generic but not in gimple? And there are
> > > probably
> > > several transformations in match.pd that could do with an update if
> > > constants
> > > don't match anymore. Or did I misunderstand what the patch does?
> >
> > The disparity is mostly that with GENERIC unfolded trees such as (char)1
> > are a bug while in GIMPLE the fact that the match.pd machinery does
> > valueization makes those a "feature" we have to deal with. Originally
> > I've restricted GENERIC as well but with its types_match_p implementation
> > it resulted in too many missed matches.
>
> I shouldn't have written (long)1, I meant the fact that 1 (as a char constant)
> and 1 (as a long constant) will now be matching captures in generic and not in
> gimple. If we are going in the direction of not matching constants of
> different types, I'd rather do it consistently and update the patterns as
> needed to avoid the missed optimizations. The missed matches exist in gimple
> as well, and generic optimization seems less important than gimple to me.
Yes, I agree. I'll see on looking at the GENERIC fallout in more detail
(fixing patterns).
> An example that regressed at -O (looking at the .optimized dump)
>
> int f(int a, unsigned b){
> a &= 1;
> b &= 1;
> return a&b;
> }
Yeah, but it usually also shows a badly written pattern:
/* (X & Y) & (X & Z) -> (X & Y) & Z
(X | Y) | (X | Z) -> (X | Y) | Z */
(for op (bit_and bit_ior)
(simplify
(op:c (convert1?@3 (op:c@4 @0 @1)) (convert2?@5 (op:c@6 @0 @2)))
(if (tree_nop_conversion_p (type, TREE_TYPE (@1))
&& tree_nop_conversion_p (type, TREE_TYPE (@2)))
so how could we ever match the @0s when we have either of the two
conversions not present? We could only do this then facing constants
(due to using operand_equal_p). We for example fail to handle
(X & Y) & (unsigned) ((singed)X & Z)
> If we stick to the old behavior, maybe we could have some genmatch magic to
> help with the constant capture weirdness. With matching captures, we could
> select which operand (among those supposed to be equivalent) is actually
> captured more cleverly, either with an explicit marker, or by giving priority
> to the one that is not immediatly below convert? in the pattern.
This route is a difficult one to take -- what would be possible is to
capture a specific operand. Like allow one to write
(op (op @0@4 @1) (op @0@3 @2))
and thus actually have three "names" for @0. We have this internally
already when you write
(convert?@0 @1)
for the case where there is no conversion. @0 and @1 are the same
in this case.
Not sure if this helps enough cases.
I still think that having two things matched that are not really
the same is werid (and a possible source of errors as in, ICEs,
rather than missed optimizations).
> And if we move to stricter matching, maybe genmatch could be updated so
> convert could also match integer constants in some cases.
You mean when trying to match @0 ... (convert @0) and @0 is an INTEGER_CST
allow then (convert ...) case to match an INTEGER_CST of different type?
That's an interesting idea (not sure how to implement that though).
> > I agree that some transforms would need updates - I've actually tried
> > to implement a warning for genmatch whenever seeing a match with
> > (T)@0 but there isn't any good existing place to sneak that in.
>
>
> > > > * match.pd ((X /[ex] A) * A -> X): Properly handle converted
> > > > and constant A.
> > >
> > > This regressed
> > > int f(int*a,int*b){return 4*(int)(b-a);}
> >
> > This is because (int)(b-a) could be a truncation in which case
> > multiplying with 4 might not result in the same value as
> > b-a truncated(?). The comment before the unpatched patterns
> > said "sign-changing conversions" but nothign actually verified this.
> > Might be that truncations are indeed ok now that I think about it.
>
> 2015-05-22 Marc Glisse <marc.glisse@inria.fr>
>
> PR tree-optimization/63387
> * match.pd ((X /[ex] A) * A -> X): Remove unnecessary condition.
>
> Apparently I forgot to remove the comment at that time :-(
Ok. I'm now testing a patch to remove the restriction again (and adjust
the comment).
> > Btw, do you have a better suggestion as to how to handle the original
> > issue rather than not relying on operand_equal_p for constants?
>
> In previous cases, in order to get the right version of a matching capture, we
> used non-matching captures and an explicit call to operand_equal_p, for
> instance:
>
> /* X - (X / Y) * Y is the same as X % Y. */
> (simplify
> (minus (convert1? @2) (convert2? (mult:c (trunc_div @0 @1) @1)))
> /* We cannot use matching captures here, since in the case of
> constants we really want the type of @0, not @2. */
> (if (operand_equal_p (@0, @2, 0)
> && (INTEGRAL_TYPE_P (type) || VECTOR_INTEGER_TYPE_P (type)))
> (convert (trunc_mod @0 @1))))
>
> That seems to match the solutions that Jakub and you were discussing in the
> PR, something like it should work (we can discuss the exact code).
Yes. As more of those cases popped up I thought this isn't really
a viable way of going forward (to fix the ICEs). So we'll see the
above for re-instantiating missed optimizations instead.
> I don't know if it is better. The old behavior of matching captures with
> inconsistent types was confusing. A behavior with strict matching may
> complicate things (will we duplicate patterns, or write operand_equal_p
> explicitly to mimic the old behavior?). The recent inconsistency between
> generic and gimple doesnt appeal to me much...
Yeah, so as first I'm going to fix that disparity.
Thanks for the comments,
Richard.
^ permalink raw reply [flat|nested] 13+ messages in thread
* Re: [PATCH] Fix PR77826
2016-10-11 9:09 ` Richard Biener
@ 2016-10-11 20:35 ` Marc Glisse
2016-10-12 11:34 ` Richard Biener
0 siblings, 1 reply; 13+ messages in thread
From: Marc Glisse @ 2016-10-11 20:35 UTC (permalink / raw)
To: Richard Biener; +Cc: gcc-patches
On Tue, 11 Oct 2016, Richard Biener wrote:
>> An example that regressed at -O (looking at the .optimized dump)
>>
>> int f(int a, unsigned b){
>> a &= 1;
>> b &= 1;
>> return a&b;
>> }
>
> Yeah, but it usually also shows a badly written pattern:
>
> /* (X & Y) & (X & Z) -> (X & Y) & Z
> (X | Y) | (X | Z) -> (X | Y) | Z */
> (for op (bit_and bit_ior)
> (simplify
> (op:c (convert1?@3 (op:c@4 @0 @1)) (convert2?@5 (op:c@6 @0 @2)))
> (if (tree_nop_conversion_p (type, TREE_TYPE (@1))
> && tree_nop_conversion_p (type, TREE_TYPE (@2)))
>
> so how could we ever match the @0s when we have either of the two
> conversions not present? We could only do this then facing constants
> (due to using operand_equal_p). We for example fail to handle
>
> (X & Y) & (unsigned) ((singed)X & Z)
Indeed... (oups, looks like I wrote that one)
Trying to find other examples
/* Fold A - (A & B) into ~B & A. */
(simplify
(minus (convert? @0) (convert?:s (bit_and:cs @0 @1)))
(if (tree_nop_conversion_p (type, TREE_TYPE (@0))
&& tree_nop_conversion_p (type, TREE_TYPE (@1)))
(convert (bit_and (bit_not @1) @0))))
Hmm, should be convert1/convert2 to handle the case where @0 is a
constant.
/* (X | Y) ^ X -> Y & ~ X*/
(simplify
(bit_xor:c (convert? (bit_ior:c @0 @1)) (convert? @0))
(if (tree_nop_conversion_p (type, TREE_TYPE (@0)))
(convert (bit_and @1 (bit_not @0)))))
Again, will never match when there is a convert and @0 is a constant.
(for op (bit_and bit_ior bit_xor)
rop (bit_ior bit_and bit_and)
(simplify
(op (convert? (rop:c @0 @1)) (convert? (rop:c @0 @2)))
...
Again won't match for constant @0 that has a different type in both parts.
/* (X & Y) & Y -> X & Y
(X | Y) | Y -> X | Y */
(for op (bit_and bit_ior)
(simplify
(op:c (convert?@2 (op:c @0 @1)) (convert? @1))
@2))
Same issue.
Ok, not many transformations are concerned, and most need a rewrite
anyway...
In the other direction, looking at the transformations for which we used
explicitly operand_equal_p as a workaround for lax integer constant
matches, it doesn't look like changing them back to use matching captures
would help, it would require duplicating the pattern for constants.
>> If we stick to the old behavior, maybe we could have some genmatch magic to
>> help with the constant capture weirdness. With matching captures, we could
>> select which operand (among those supposed to be equivalent) is actually
>> captured more cleverly, either with an explicit marker, or by giving priority
>> to the one that is not immediatly below convert? in the pattern.
>
> This route is a difficult one to take
The simplest version I was thinking about was @0 for a true capture, and
@@0 for something that just has to be checked for equality with @0.
> -- what would be possible is to
> capture a specific operand. Like allow one to write
>
> (op (op @0@4 @1) (op @0@3 @2))
>
> and thus actually have three "names" for @0. We have this internally
> already when you write
>
> (convert?@0 @1)
>
> for the case where there is no conversion. @0 and @1 are the same
> in this case.
Looks nice and convenient (assuming lax constant matching).
> Not sure if this helps enough cases.
IIRC, in all cases where we had trouble with operand_equal_p, chosing
which operand to capture would have solved the issue.
> I still think that having two things matched that are not really
> the same is werid (and a possible source of errors as in, ICEs,
> rather than missed optimizations).
Ok. Let's go with the strict matching, it is indeed less confusing.
>> And if we move to stricter matching, maybe genmatch could be updated so
>> convert could also match integer constants in some cases.
>
> You mean when trying to match @0 ... (convert @0) and @0 is an INTEGER_CST
> allow then (convert ...) case to match an INTEGER_CST of different type?
> That's an interesting idea (not sure how to implement that though).
Yes, though I am not sure of the exact semantics that would work best.
On a related note, seeing duplicated patterns for constants, I tried
several variants of
(match (mynot @0)
(bit_not @0))
(match (mynot @0)
INTEGER_CST@0
(if (@0 = wide_int_to_tree (TREE_TYPE (@0), wi::bit_not (@0)))))
(simplify
(minus (bit_and:cs @0 (mynot @1)) (bit_and:cs @0 @1))
(minus (bit_xor @0 @1) @1))
This kind of hack feels wrong, but I don't see a proper way to write it.
>>> I agree that some transforms would need updates - I've actually tried
>>> to implement a warning for genmatch whenever seeing a match with
>>> (T)@0 but there isn't any good existing place to sneak that in.
>>
>>
>>>>> * match.pd ((X /[ex] A) * A -> X): Properly handle converted
>>>>> and constant A.
>>>>
>>>> This regressed
>>>> int f(int*a,int*b){return 4*(int)(b-a);}
>>>
>>> This is because (int)(b-a) could be a truncation in which case
>>> multiplying with 4 might not result in the same value as
>>> b-a truncated(?). The comment before the unpatched patterns
>>> said "sign-changing conversions" but nothign actually verified this.
>>> Might be that truncations are indeed ok now that I think about it.
>>
>> 2015-05-22 Marc Glisse <marc.glisse@inria.fr>
>>
>> PR tree-optimization/63387
>> * match.pd ((X /[ex] A) * A -> X): Remove unnecessary condition.
>>
>> Apparently I forgot to remove the comment at that time :-(
>
> Ok. I'm now testing a patch to remove the restriction again (and adjust
> the comment).
Thanks. (since exact_div is used almost only with constant divisors, the
old pattern was fine before strict matching, but indeed your more general
pattern is better)
--
Marc Glisse
^ permalink raw reply [flat|nested] 13+ messages in thread
* Re: [PATCH] Fix PR77826
2016-10-11 20:35 ` Marc Glisse
@ 2016-10-12 11:34 ` Richard Biener
2016-10-12 12:49 ` Marc Glisse
0 siblings, 1 reply; 13+ messages in thread
From: Richard Biener @ 2016-10-12 11:34 UTC (permalink / raw)
To: gcc-patches
On Tue, 11 Oct 2016, Marc Glisse wrote:
> On Tue, 11 Oct 2016, Richard Biener wrote:
>
> > > An example that regressed at -O (looking at the .optimized dump)
> > >
> > > int f(int a, unsigned b){
> > > a &= 1;
> > > b &= 1;
> > > return a&b;
> > > }
> >
> > Yeah, but it usually also shows a badly written pattern:
> >
> > /* (X & Y) & (X & Z) -> (X & Y) & Z
> > (X | Y) | (X | Z) -> (X | Y) | Z */
> > (for op (bit_and bit_ior)
> > (simplify
> > (op:c (convert1?@3 (op:c@4 @0 @1)) (convert2?@5 (op:c@6 @0 @2)))
> > (if (tree_nop_conversion_p (type, TREE_TYPE (@1))
> > && tree_nop_conversion_p (type, TREE_TYPE (@2)))
> >
> > so how could we ever match the @0s when we have either of the two
> > conversions not present? We could only do this then facing constants
> > (due to using operand_equal_p). We for example fail to handle
> >
> > (X & Y) & (unsigned) ((singed)X & Z)
>
> Indeed... (oups, looks like I wrote that one)
>
> Trying to find other examples
>
> /* Fold A - (A & B) into ~B & A. */
> (simplify
> (minus (convert? @0) (convert?:s (bit_and:cs @0 @1)))
> (if (tree_nop_conversion_p (type, TREE_TYPE (@0))
> && tree_nop_conversion_p (type, TREE_TYPE (@1)))
> (convert (bit_and (bit_not @1) @0))))
>
> Hmm, should be convert1/convert2 to handle the case where @0 is a constant.
>
> /* (X | Y) ^ X -> Y & ~ X*/
> (simplify
> (bit_xor:c (convert? (bit_ior:c @0 @1)) (convert? @0))
> (if (tree_nop_conversion_p (type, TREE_TYPE (@0)))
> (convert (bit_and @1 (bit_not @0)))))
>
> Again, will never match when there is a convert and @0 is a constant.
>
> (for op (bit_and bit_ior bit_xor)
> rop (bit_ior bit_and bit_and)
> (simplify
> (op (convert? (rop:c @0 @1)) (convert? (rop:c @0 @2)))
> ...
>
> Again won't match for constant @0 that has a different type in both parts.
>
> /* (X & Y) & Y -> X & Y
> (X | Y) | Y -> X | Y */
> (for op (bit_and bit_ior)
> (simplify
> (op:c (convert?@2 (op:c @0 @1)) (convert? @1))
> @2))
>
> Same issue.
>
> Ok, not many transformations are concerned, and most need a rewrite anyway...
>
> In the other direction, looking at the transformations for which we used
> explicitly operand_equal_p as a workaround for lax integer constant matches,
> it doesn't look like changing them back to use matching captures would help,
> it would require duplicating the pattern for constants.
So with doing the same on GENERIC I hit
FAIL: g++.dg/cpp1y/constexpr-array4.C -std=c++14 (test for excess errors)
with the pattern
/* (T)(P + A) - (T)P -> (T) A */
(for add (plus pointer_plus)
(simplify
(minus (convert (add @0 @1))
(convert @0))
...
(convert @1))))
no longer applying to
(long int) ((int *) &ar + 12) - (long int) &ar
because while (int *) &ar is equal to (long int) &ar (operand_equal_p
does STRIP_NOPS) they obviously do not have the same type. So on
GENERIC we have to consider that we feed operand_equal_p with
non-ATOMs (arbitrary expressions). The pattern above is "safe" as
it doesn't reference @0 in the result (not sure if we should take
advantage of that in genmatch).
The other FAILs with doing the same on GENERIC are
FAIL: g++.dg/gomp/declare-simd-3.C -std=gnu++11 (test for excess errors)
FAIL: g++.dg/torture/pr71448.C -O0 (test for excess errors)
FAIL: g++.dg/vect/simd-clone-6.cc -std=c++11 (test for excess errors)
the simd ones are 'warning: ignoring large linear step' and the pr71448.C
case is very similar to the above.
> > > If we stick to the old behavior, maybe we could have some genmatch magic
> > > to
> > > help with the constant capture weirdness. With matching captures, we could
> > > select which operand (among those supposed to be equivalent) is actually
> > > captured more cleverly, either with an explicit marker, or by giving
> > > priority
> > > to the one that is not immediatly below convert? in the pattern.
> >
> > This route is a difficult one to take
>
> The simplest version I was thinking about was @0 for a true capture, and @@0
> for something that just has to be checked for equality with @0.
Hmm, ok. So you'd have @@0 having to match @0 and we'd get the @0 for
the result in a reliable way? Sounds like a reasonable idea, I'll see
how that works out (we could auto-detect '@@' if the capture is not
used in the result, see above).
> > -- what would be possible is to
> > capture a specific operand. Like allow one to write
> >
> > (op (op @0@4 @1) (op @0@3 @2))
> >
> > and thus actually have three "names" for @0. We have this internally
> > already when you write
> >
> > (convert?@0 @1)
> >
> > for the case where there is no conversion. @0 and @1 are the same
> > in this case.
>
> Looks nice and convenient (assuming lax constant matching).
Yes, w/o lax matching it has of course little value.
> > Not sure if this helps enough cases.
>
> IIRC, in all cases where we had trouble with operand_equal_p, chosing which
> operand to capture would have solved the issue.
Yes. We'd still need to actually catch all those cases...
> > I still think that having two things matched that are not really
> > the same is werid (and a possible source of errors as in, ICEs,
> > rather than missed optimizations).
>
> Ok. Let's go with the strict matching, it is indeed less confusing.
I'll hold off with the GENERIC strict matching and will investigate
the @@ idea (plus auto-detecting it).
> > > And if we move to stricter matching, maybe genmatch could be updated so
> > > convert could also match integer constants in some cases.
> >
> > You mean when trying to match @0 ... (convert @0) and @0 is an INTEGER_CST
> > allow then (convert ...) case to match an INTEGER_CST of different type?
> > That's an interesting idea (not sure how to implement that though).
>
> Yes, though I am not sure of the exact semantics that would work best.
>
> On a related note, seeing duplicated patterns for constants, I tried several
> variants of
>
> (match (mynot @0)
> (bit_not @0))
> (match (mynot @0)
> INTEGER_CST@0
> (if (@0 = wide_int_to_tree (TREE_TYPE (@0), wi::bit_not (@0)))))
>
> (simplify
> (minus (bit_and:cs @0 (mynot @1)) (bit_and:cs @0 @1))
> (minus (bit_xor @0 @1) @1))
>
> This kind of hack feels wrong, but I don't see a proper way to write it.
Yes. The above is very much a hack. Must have been me inventing it
just to avoid duplicating the pattern.
Richard.
>
> > > > I agree that some transforms would need updates - I've actually tried
> > > > to implement a warning for genmatch whenever seeing a match with
> > > > (T)@0 but there isn't any good existing place to sneak that in.
> > >
> > >
> > > > > > * match.pd ((X /[ex] A) * A -> X): Properly handle converted
> > > > > > and constant A.
> > > > >
> > > > > This regressed
> > > > > int f(int*a,int*b){return 4*(int)(b-a);}
> > > >
> > > > This is because (int)(b-a) could be a truncation in which case
> > > > multiplying with 4 might not result in the same value as
> > > > b-a truncated(?). The comment before the unpatched patterns
> > > > said "sign-changing conversions" but nothign actually verified this.
> > > > Might be that truncations are indeed ok now that I think about it.
> > >
> > > 2015-05-22 Marc Glisse <marc.glisse@inria.fr>
> > >
> > > PR tree-optimization/63387
> > > * match.pd ((X /[ex] A) * A -> X): Remove unnecessary condition.
> > >
> > > Apparently I forgot to remove the comment at that time :-(
> >
> > Ok. I'm now testing a patch to remove the restriction again (and adjust
> > the comment).
>
> Thanks. (since exact_div is used almost only with constant divisors, the old
> pattern was fine before strict matching, but indeed your more general pattern
> is better)
^ permalink raw reply [flat|nested] 13+ messages in thread
* Re: [PATCH] Fix PR77826
2016-10-12 11:34 ` Richard Biener
@ 2016-10-12 12:49 ` Marc Glisse
2016-10-12 13:25 ` Richard Biener
0 siblings, 1 reply; 13+ messages in thread
From: Marc Glisse @ 2016-10-12 12:49 UTC (permalink / raw)
To: Richard Biener; +Cc: gcc-patches
On Wed, 12 Oct 2016, Richard Biener wrote:
> So with doing the same on GENERIC I hit
>
> FAIL: g++.dg/cpp1y/constexpr-array4.C -std=c++14 (test for excess errors)
>
> with the pattern
>
> /* (T)(P + A) - (T)P -> (T) A */
> (for add (plus pointer_plus)
> (simplify
> (minus (convert (add @0 @1))
> (convert @0))
Ah, grep missed that one because it is on 2 lines :-(
> ...
> (convert @1))))
>
>
> no longer applying to
>
> (long int) ((int *) &ar + 12) - (long int) &ar
>
> because while (int *) &ar is equal to (long int) &ar (operand_equal_p
> does STRIP_NOPS) they obviously do not have the same type.
I believe we are comparing (int *) &ar to &ar, not to (long int) &ar. But
yes, indeed.
> So on GENERIC we have to consider that we feed operand_equal_p with
> non-ATOMs (arbitrary expressions). The pattern above is "safe" as it
> doesn't reference @0 in the result (not sure if we should take advantage
> of that in genmatch).
Since we are in the process of defining an
operand_equal_for_(generic|gimple)_match_p, we could tweak it to check the
type only for INTEGER_CST, or to still STRIP_NOPS, or similar.
Or we could remain very strict and refine the pattern, allowing a convert
on the pointer (we might want to split the plus and pointer_plus versions
then, for clarity). There are not many optimizations that are mandated by
front-ends and for which this is an issue.
> The other FAILs with doing the same on GENERIC are
>
> FAIL: g++.dg/gomp/declare-simd-3.C -std=gnu++11 (test for excess errors)
> FAIL: g++.dg/torture/pr71448.C -O0 (test for excess errors)
> FAIL: g++.dg/vect/simd-clone-6.cc -std=c++11 (test for excess errors)
>
> the simd ones are 'warning: ignoring large linear step' and the pr71448.C
> case is very similar to the above.
Yes, I expect they all come from the same 1 or 2 transformations.
>>>> If we stick to the old behavior, maybe we could have some genmatch
>>>> magic to help with the constant capture weirdness. With matching
>>>> captures, we could select which operand (among those supposed to be
>>>> equivalent) is actually captured more cleverly, either with an
>>>> explicit marker, or by giving priority to the one that is not
>>>> immediatly below convert? in the pattern.
>>>
>>> This route is a difficult one to take
>>
>> The simplest version I was thinking about was @0 for a true capture, and @@0
>> for something that just has to be checked for equality with @0.
>
> Hmm, ok. So you'd have @@0 having to match @0 and we'd get the @0 for
> the result in a reliable way? Sounds like a reasonable idea, I'll see
> how that works out (we could auto-detect '@@' if the capture is not
> used in the result, see above).
It probably doesn't bring much compared to the @0@4 syntax you were
suggesting below, so if that is easier...
>>> -- what would be possible is to
>>> capture a specific operand. Like allow one to write
>>>
>>> (op (op @0@4 @1) (op @0@3 @2))
>>>
>>> and thus actually have three "names" for @0. We have this internally
>>> already when you write
>>>
>>> (convert?@0 @1)
>>>
>>> for the case where there is no conversion. @0 and @1 are the same
>>> in this case.
>>
>> Looks nice and convenient (assuming lax constant matching).
>
> Yes, w/o lax matching it has of course little value.
>
>>> Not sure if this helps enough cases.
>>
>> IIRC, in all cases where we had trouble with operand_equal_p, chosing which
>> operand to capture would have solved the issue.
>
> Yes. We'd still need to actually catch all those cases...
Being forced to chose which operand we capture (say with @ vs @@, making 2
occurences of @0 a syntax error) might help, but it would also require
updating many patterns for which this was never an issue (easy but
boring).
>>> I still think that having two things matched that are not really
>>> the same is werid (and a possible source of errors as in, ICEs,
>>> rather than missed optimizations).
>>
>> Ok. Let's go with the strict matching, it is indeed less confusing.
>
> I'll hold off with the GENERIC strict matching and will investigate
> the @@ idea (plus auto-detecting it).
>
>>>> And if we move to stricter matching, maybe genmatch could be updated so
>>>> convert could also match integer constants in some cases.
>>>
>>> You mean when trying to match @0 ... (convert @0) and @0 is an INTEGER_CST
>>> allow then (convert ...) case to match an INTEGER_CST of different type?
>>> That's an interesting idea (not sure how to implement that though).
>>
>> Yes, though I am not sure of the exact semantics that would work best.
>>
>> On a related note, seeing duplicated patterns for constants, I tried several
>> variants of
>>
>> (match (mynot @0)
>> (bit_not @0))
>> (match (mynot @0)
>> INTEGER_CST@0
>> (if (@0 = wide_int_to_tree (TREE_TYPE (@0), wi::bit_not (@0)))))
>>
>> (simplify
>> (minus (bit_and:cs @0 (mynot @1)) (bit_and:cs @0 @1))
>> (minus (bit_xor @0 @1) @1))
>>
>> This kind of hack feels wrong, but I don't see a proper way to write it.
>
> Yes. The above is very much a hack. Must have been me inventing it
> just to avoid duplicating the pattern.
Uh, no, don't worry, we don't have that hack in match.pd, we have 2
patterns. The hack is just me trying to remove the duplication. If you
thought you had written it, that means it may not have been such an absurd
way to go about it ;-)
--
Marc Glisse
^ permalink raw reply [flat|nested] 13+ messages in thread
* Re: [PATCH] Fix PR77826
2016-10-12 12:49 ` Marc Glisse
@ 2016-10-12 13:25 ` Richard Biener
2016-10-13 8:29 ` Richard Biener
0 siblings, 1 reply; 13+ messages in thread
From: Richard Biener @ 2016-10-12 13:25 UTC (permalink / raw)
To: gcc-patches
On Wed, 12 Oct 2016, Marc Glisse wrote:
> On Wed, 12 Oct 2016, Richard Biener wrote:
>
> > So with doing the same on GENERIC I hit
> >
> > FAIL: g++.dg/cpp1y/constexpr-array4.C -std=c++14 (test for excess errors)
> >
> > with the pattern
> >
> > /* (T)(P + A) - (T)P -> (T) A */
> > (for add (plus pointer_plus)
> > (simplify
> > (minus (convert (add @0 @1))
> > (convert @0))
>
> Ah, grep missed that one because it is on 2 lines :-(
>
> > ...
> > (convert @1))))
> >
> >
> > no longer applying to
> >
> > (long int) ((int *) &ar + 12) - (long int) &ar
> >
> > because while (int *) &ar is equal to (long int) &ar (operand_equal_p
> > does STRIP_NOPS) they obviously do not have the same type.
>
> I believe we are comparing (int *) &ar to &ar, not to (long int) &ar. But yes,
> indeed.
>
> > So on GENERIC we have to consider that we feed operand_equal_p with
> > non-ATOMs (arbitrary expressions). The pattern above is "safe" as it
> > doesn't reference @0 in the result (not sure if we should take advantage of
> > that in genmatch).
>
> Since we are in the process of defining an
> operand_equal_for_(generic|gimple)_match_p, we could tweak it to check the
> type only for INTEGER_CST, or to still STRIP_NOPS, or similar.
>
> Or we could remain very strict and refine the pattern, allowing a convert on
> the pointer (we might want to split the plus and pointer_plus versions then,
> for clarity). There are not many optimizations that are mandated by front-ends
> and for which this is an issue.
>
> > The other FAILs with doing the same on GENERIC are
> >
> > FAIL: g++.dg/gomp/declare-simd-3.C -std=gnu++11 (test for excess errors)
> > FAIL: g++.dg/torture/pr71448.C -O0 (test for excess errors)
> > FAIL: g++.dg/vect/simd-clone-6.cc -std=c++11 (test for excess errors)
> >
> > the simd ones are 'warning: ignoring large linear step' and the pr71448.C
> > case is very similar to the above.
>
> Yes, I expect they all come from the same 1 or 2 transformations.
>
> > > > > If we stick to the old behavior, maybe we could have some genmatch
> > > > > magic to help with the constant capture weirdness. With matching
> > > > > captures, we could select which operand (among those supposed to be
> > > > > equivalent) is actually captured more cleverly, either with an
> > > > > explicit marker, or by giving priority to the one that is not
> > > > > immediatly below convert? in the pattern.
> > > >
> > > > This route is a difficult one to take
> > >
> > > The simplest version I was thinking about was @0 for a true capture, and
> > > @@0
> > > for something that just has to be checked for equality with @0.
> >
> > Hmm, ok. So you'd have @@0 having to match @0 and we'd get the @0 for
> > the result in a reliable way? Sounds like a reasonable idea, I'll see
> > how that works out (we could auto-detect '@@' if the capture is not
> > used in the result, see above).
>
> It probably doesn't bring much compared to the @0@4 syntax you were
> suggesting below, so if that is easier...
Will figure that out ...
> > > > -- what would be possible is to
> > > > capture a specific operand. Like allow one to write
> > > >
> > > > (op (op @0@4 @1) (op @0@3 @2))
> > > >
> > > > and thus actually have three "names" for @0. We have this internally
> > > > already when you write
> > > >
> > > > (convert?@0 @1)
> > > >
> > > > for the case where there is no conversion. @0 and @1 are the same
> > > > in this case.
> > >
> > > Looks nice and convenient (assuming lax constant matching).
> >
> > Yes, w/o lax matching it has of course little value.
> >
> > > > Not sure if this helps enough cases.
> > >
> > > IIRC, in all cases where we had trouble with operand_equal_p, chosing
> > > which
> > > operand to capture would have solved the issue.
> >
> > Yes. We'd still need to actually catch all those cases...
>
> Being forced to chose which operand we capture (say with @ vs @@, making 2
> occurences of @0 a syntax error) might help, but it would also require
> updating many patterns for which this was never an issue (easy but boring).
We can even have today
(plus (minus @0 @1) (plus @0 @2) @0)
thus three matching @0. Now if we have @@ telling to use value equality
rather than "node equality" _and_ making the @@ select the canonical
operand to choose then we'd have at most one @@ per ID. Somewhat tricky
but not impossible to implement I think.
> > > > I still think that having two things matched that are not really
> > > > the same is werid (and a possible source of errors as in, ICEs,
> > > > rather than missed optimizations).
> > >
> > > Ok. Let's go with the strict matching, it is indeed less confusing.
> >
> > I'll hold off with the GENERIC strict matching and will investigate
> > the @@ idea (plus auto-detecting it).
> >
> > > > > And if we move to stricter matching, maybe genmatch could be updated
> > > > > so
> > > > > convert could also match integer constants in some cases.
> > > >
> > > > You mean when trying to match @0 ... (convert @0) and @0 is an
> > > > INTEGER_CST
> > > > allow then (convert ...) case to match an INTEGER_CST of different type?
> > > > That's an interesting idea (not sure how to implement that though).
> > >
> > > Yes, though I am not sure of the exact semantics that would work best.
> > >
> > > On a related note, seeing duplicated patterns for constants, I tried
> > > several
> > > variants of
> > >
> > > (match (mynot @0)
> > > (bit_not @0))
> > > (match (mynot @0)
> > > INTEGER_CST@0
> > > (if (@0 = wide_int_to_tree (TREE_TYPE (@0), wi::bit_not (@0)))))
> > >
> > > (simplify
> > > (minus (bit_and:cs @0 (mynot @1)) (bit_and:cs @0 @1))
> > > (minus (bit_xor @0 @1) @1))
> > >
> > > This kind of hack feels wrong, but I don't see a proper way to write it.
> >
> > Yes. The above is very much a hack. Must have been me inventing it
> > just to avoid duplicating the pattern.
>
> Uh, no, don't worry, we don't have that hack in match.pd, we have 2 patterns.
> The hack is just me trying to remove the duplication. If you thought you had
> written it, that means it may not have been such an absurd way to go about it
> ;-)
Ah ...
;)
^ permalink raw reply [flat|nested] 13+ messages in thread
* Re: [PATCH] Fix PR77826
2016-10-12 13:25 ` Richard Biener
@ 2016-10-13 8:29 ` Richard Biener
2016-10-15 21:50 ` Marc Glisse
0 siblings, 1 reply; 13+ messages in thread
From: Richard Biener @ 2016-10-13 8:29 UTC (permalink / raw)
To: gcc-patches
On Wed, 12 Oct 2016, Richard Biener wrote:
> On Wed, 12 Oct 2016, Marc Glisse wrote:
>
> > On Wed, 12 Oct 2016, Richard Biener wrote:
> >
> > > So with doing the same on GENERIC I hit
> > >
> > > FAIL: g++.dg/cpp1y/constexpr-array4.C -std=c++14 (test for excess errors)
> > >
> > > with the pattern
> > >
> > > /* (T)(P + A) - (T)P -> (T) A */
> > > (for add (plus pointer_plus)
> > > (simplify
> > > (minus (convert (add @0 @1))
> > > (convert @0))
> >
> > Ah, grep missed that one because it is on 2 lines :-(
> >
> > > ...
> > > (convert @1))))
> > >
> > >
> > > no longer applying to
> > >
> > > (long int) ((int *) &ar + 12) - (long int) &ar
> > >
> > > because while (int *) &ar is equal to (long int) &ar (operand_equal_p
> > > does STRIP_NOPS) they obviously do not have the same type.
> >
> > I believe we are comparing (int *) &ar to &ar, not to (long int) &ar. But yes,
> > indeed.
> >
> > > So on GENERIC we have to consider that we feed operand_equal_p with
> > > non-ATOMs (arbitrary expressions). The pattern above is "safe" as it
> > > doesn't reference @0 in the result (not sure if we should take advantage of
> > > that in genmatch).
> >
> > Since we are in the process of defining an
> > operand_equal_for_(generic|gimple)_match_p, we could tweak it to check the
> > type only for INTEGER_CST, or to still STRIP_NOPS, or similar.
> >
> > Or we could remain very strict and refine the pattern, allowing a convert on
> > the pointer (we might want to split the plus and pointer_plus versions then,
> > for clarity). There are not many optimizations that are mandated by front-ends
> > and for which this is an issue.
> >
> > > The other FAILs with doing the same on GENERIC are
> > >
> > > FAIL: g++.dg/gomp/declare-simd-3.C -std=gnu++11 (test for excess errors)
> > > FAIL: g++.dg/torture/pr71448.C -O0 (test for excess errors)
> > > FAIL: g++.dg/vect/simd-clone-6.cc -std=c++11 (test for excess errors)
> > >
> > > the simd ones are 'warning: ignoring large linear step' and the pr71448.C
> > > case is very similar to the above.
> >
> > Yes, I expect they all come from the same 1 or 2 transformations.
> >
> > > > > > If we stick to the old behavior, maybe we could have some genmatch
> > > > > > magic to help with the constant capture weirdness. With matching
> > > > > > captures, we could select which operand (among those supposed to be
> > > > > > equivalent) is actually captured more cleverly, either with an
> > > > > > explicit marker, or by giving priority to the one that is not
> > > > > > immediatly below convert? in the pattern.
> > > > >
> > > > > This route is a difficult one to take
> > > >
> > > > The simplest version I was thinking about was @0 for a true capture, and
> > > > @@0
> > > > for something that just has to be checked for equality with @0.
> > >
> > > Hmm, ok. So you'd have @@0 having to match @0 and we'd get the @0 for
> > > the result in a reliable way? Sounds like a reasonable idea, I'll see
> > > how that works out (we could auto-detect '@@' if the capture is not
> > > used in the result, see above).
> >
> > It probably doesn't bring much compared to the @0@4 syntax you were
> > suggesting below, so if that is easier...
>
> Will figure that out ...
>
> > > > > -- what would be possible is to
> > > > > capture a specific operand. Like allow one to write
> > > > >
> > > > > (op (op @0@4 @1) (op @0@3 @2))
> > > > >
> > > > > and thus actually have three "names" for @0. We have this internally
> > > > > already when you write
> > > > >
> > > > > (convert?@0 @1)
> > > > >
> > > > > for the case where there is no conversion. @0 and @1 are the same
> > > > > in this case.
> > > >
> > > > Looks nice and convenient (assuming lax constant matching).
> > >
> > > Yes, w/o lax matching it has of course little value.
> > >
> > > > > Not sure if this helps enough cases.
> > > >
> > > > IIRC, in all cases where we had trouble with operand_equal_p, chosing
> > > > which
> > > > operand to capture would have solved the issue.
> > >
> > > Yes. We'd still need to actually catch all those cases...
> >
> > Being forced to chose which operand we capture (say with @ vs @@, making 2
> > occurences of @0 a syntax error) might help, but it would also require
> > updating many patterns for which this was never an issue (easy but boring).
>
> We can even have today
>
> (plus (minus @0 @1) (plus @0 @2) @0)
>
> thus three matching @0. Now if we have @@ telling to use value equality
> rather than "node equality" _and_ making the @@ select the canonical
> operand to choose then we'd have at most one @@ per ID. Somewhat tricky
> but not impossible to implement I think.
So here it is. It allows us to remove the operand_equal_p hacks:
@@ -334,11 +334,8 @@ DEFINE_INT_AND_FLOAT_ROUND_FN (RINT)
/* X - (X / Y) * Y is the same as X % Y. */
(simplify
- (minus (convert1? @2) (convert2? (mult:c (trunc_div @0 @1) @1)))
- /* We cannot use matching captures here, since in the case of
- constants we really want the type of @0, not @2. */
- (if (operand_equal_p (@0, @2, 0)
- && (INTEGRAL_TYPE_P (type) || VECTOR_INTEGER_TYPE_P (type)))
+ (minus (convert1? @0) (convert2? (mult:c (trunc_div @@0 @@1) @1)))
+ (if (INTEGRAL_TYPE_P (type) || VECTOR_INTEGER_TYPE_P (type))
(convert (trunc_mod @0 @1))))
The patch introduces '@@' captures which do two things - first they
change the comparison used for matching back to operand_equal_p
only (thus perform "value-matching"), second the @@ capture denotes
the specific operand that should be refered to in the result
section (ifs and result expressions).
When we face (plus @0 @1) (convert? @@0) for example this is lowered
to (plus @__2 @1) (convert? @__2@0) marking the @__2 match for
value handling.
I modified the patterns you identified and the ones I did and
removed the operand_equal_p uses where possible.
I did not implement the auto-guessing idea as I usually like
to be more explicit here.
Bootstrap and regtest running on x86_64-unknown-linux-gnu.
Richard.
2016-10-13 Richard Biener <rguenther@suse.de>
PR middle-end/77826
* genmatch.c (struct capture): Add value_match member.
(commutate): Preserve value_match.
(lower_opt_convert): Likewise.
(lower_cond): Likewise.
(replace_id): Likewise.
(struct dt_operand): Add value_match member.
(decision_tree::cmp_node): Compare it.
(decision_tree::insert_operand): Honor it when finding and
when appending a DT_MATCH.
(dt_operand::gen_match_op): Generate a type check after
operand_equal_p if ! value_match for both GENERIC and GIMPLE.
(parser::get_internal_capture_id): New helper.
(parser::finish_match_operand): New function lowering @@<id>.
(parser::parse_capture): Parse @@<id> as value-match.
(parser::parse_expr): Use get_internal_capture_id.
(parser::parse_simplify): Call finish_match_operand.
(walk_captures): New helper.
* match.pd (X - (X / Y) * Y -> X % Y): Use value-matching instead
of operand_equal_p.
((X /[ex] A) * A -> X): Likewise.
((X | Y) ^ X -> Y & ~ X): Handle constants properly by using
convert[12] and value-matching.
((A | B) & (A | C) -> A | (B & C)): Likewise.
((X | Y) | Y -> X | Y): Likewise.
((X ^ Y) ^ Y -> X): Likewise.
(A - (A & B) -> ~B & A): Likewise.
((T)(P + A) - (T)P -> (T) A): Likewise.
((T)P - (T)(P + A) -> -(T) A): Likewise.
((T)(P + A) - (T)(P + B) -> (T)A - (T)B): Likewise.
* doc/match-and-simplify.texi: Amend capture section.
Index: gcc/genmatch.c
===================================================================
--- gcc/genmatch.c (revision 241085)
+++ gcc/genmatch.c (working copy)
@@ -693,10 +693,15 @@ struct c_expr : public operand
struct capture : public operand
{
- capture (source_location loc, unsigned where_, operand *what_)
- : operand (OP_CAPTURE, loc), where (where_), what (what_) {}
+ capture (source_location loc, unsigned where_, operand *what_, bool value_)
+ : operand (OP_CAPTURE, loc), where (where_), value_match (value_),
+ what (what_) {}
/* Identifier index for the value. */
unsigned where;
+ /* Whether in a match of two operands the compare should be for
+ equal values rather than equal atoms (boils down to a type
+ check or not). */
+ bool value_match;
/* The captured value. */
operand *what;
virtual void gen_transform (FILE *f, int, const char *, bool, int,
@@ -895,7 +900,8 @@ commutate (operand *op, vec<vec<user_id
vec<operand *> v = commutate (c->what, for_vec);
for (unsigned i = 0; i < v.length (); ++i)
{
- capture *nc = new capture (c->location, c->where, v[i]);
+ capture *nc = new capture (c->location, c->where, v[i],
+ c->value_match);
ret.safe_push (nc);
}
return ret;
@@ -1013,7 +1019,8 @@ lower_opt_convert (operand *o, enum tree
{
if (c->what)
return new capture (c->location, c->where,
- lower_opt_convert (c->what, oper, to_oper, strip));
+ lower_opt_convert (c->what, oper, to_oper, strip),
+ c->value_match);
else
return c;
}
@@ -1146,7 +1153,8 @@ lower_cond (operand *o)
lop = lower_cond (c->what);
for (unsigned i = 0; i < lop.length (); ++i)
- ro.safe_push (new capture (c->location, c->where, lop[i]));
+ ro.safe_push (new capture (c->location, c->where, lop[i],
+ c->value_match));
return ro;
}
}
@@ -1196,7 +1204,8 @@ lower_cond (operand *o)
for (unsigned j = 0; j < ocmp->ops.length (); ++j)
cmp->append_op (ocmp->ops[j]);
cmp->is_generic = true;
- ne->ops[0] = new capture (c->location, c->where, cmp);
+ ne->ops[0] = new capture (c->location, c->where, cmp,
+ c->value_match);
}
else
{
@@ -1275,7 +1284,7 @@ replace_id (operand *o, user_id *id, id_
if (!c->what)
return c;
return new capture (c->location, c->where,
- replace_id (c->what, id, with));
+ replace_id (c->what, id, with), c->value_match);
}
else if (expr *e = dyn_cast<expr *> (o))
{
@@ -1554,11 +1563,12 @@ struct dt_operand : public dt_node
dt_operand *match_dop;
dt_operand *parent;
unsigned pos;
+ bool value_match;
dt_operand (enum dt_type type, operand *op_, dt_operand *match_dop_,
dt_operand *parent_ = 0, unsigned pos_ = 0)
: dt_node (type), op (op_), match_dop (match_dop_),
- parent (parent_), pos (pos_) {}
+ parent (parent_), pos (pos_), value_match (false) {}
void gen (FILE *, int, bool);
unsigned gen_predicate (FILE *, int, const char *, bool);
@@ -1670,8 +1680,10 @@ decision_tree::cmp_node (dt_node *n1, dt
return cmp_operand ((as_a<dt_operand *> (n1))->op,
(as_a<dt_operand *> (n2))->op);
else if (n1->type == dt_node::DT_MATCH)
- return ((as_a<dt_operand *> (n1))->match_dop
- == (as_a<dt_operand *> (n2))->match_dop);
+ return (((as_a<dt_operand *> (n1))->match_dop
+ == (as_a<dt_operand *> (n2))->match_dop)
+ && ((as_a<dt_operand *> (n1))->value_match
+ == (as_a<dt_operand *> (n2))->value_match));
return false;
}
@@ -1841,6 +1853,7 @@ decision_tree::insert_operand (dt_node *
if (elm == 0)
{
dt_operand temp (dt_node::DT_MATCH, 0, match_op);
+ temp.value_match = c->value_match;
elm = decision_tree::find_node (p->kids, &temp);
}
}
@@ -1860,6 +1873,7 @@ at_assert_elm:
else
{
p = p->append_match_op (indexes[capt_index], parent, pos);
+ as_a <dt_operand *>(p)->value_match = c->value_match;
if (c->what)
return insert_operand (p, c->what, indexes, 0, p);
else
@@ -2591,18 +2613,18 @@ dt_operand::gen_predicate (FILE *f, int
a capture-match. */
unsigned
-dt_operand::gen_match_op (FILE *f, int indent, const char *opname, bool gimple)
+dt_operand::gen_match_op (FILE *f, int indent, const char *opname, bool)
{
char match_opname[20];
match_dop->get_name (match_opname);
- if (gimple)
+ if (value_match)
+ fprintf_indent (f, indent, "if (%s == %s || operand_equal_p (%s, %s, 0))\n",
+ opname, match_opname, opname, match_opname);
+ else
fprintf_indent (f, indent, "if (%s == %s || (operand_equal_p (%s, %s, 0) "
"&& types_match (%s, %s)))\n",
opname, match_opname, opname, match_opname,
opname, match_opname);
- else
- fprintf_indent (f, indent, "if (%s == %s || operand_equal_p (%s, %s, 0))\n",
- opname, match_opname, opname, match_opname);
fprintf_indent (f, indent + 2, "{\n");
return 1;
}
@@ -3731,6 +3767,8 @@ private:
const cpp_token *eat_ident (const char *);
const char *get_number ();
+ unsigned get_internal_capture_id ();
+
id_base *parse_operation ();
operand *parse_capture (operand *, bool);
operand *parse_expr ();
@@ -3750,6 +3788,8 @@ private:
void parse_predicates (source_location);
void parse_operator_list (source_location);
+ void finish_match_operand (operand *);
+
cpp_reader *r;
vec<c_expr *> active_ifs;
vec<vec<user_id *> > active_fors;
@@ -3886,6 +3926,21 @@ parser::get_number ()
return (const char *)token->val.str.text;
}
+/* Return a capture ID that can be used internally. */
+
+unsigned
+parser::get_internal_capture_id ()
+{
+ unsigned newid = capture_ids->elements ();
+ /* Big enough for a 32-bit UINT_MAX plus prefix. */
+ char id[13];
+ bool existed;
+ sprintf (id, "__%u", newid);
+ capture_ids->get_or_insert (xstrdup (id), &existed);
+ if (existed)
+ fatal ("reserved capture id '%s' already used", id);
+ return newid;
+}
/* Record an operator-list use for transparent for handling. */
@@ -3967,6 +4022,16 @@ parser::parse_capture (operand *op, bool
source_location src_loc = eat_token (CPP_ATSIGN)->src_loc;
const cpp_token *token = peek ();
const char *id = NULL;
+ bool value_match = false;
+ /* For matches parse @@ as a value-match denoting the prevailing operand. */
+ if (token->type == CPP_ATSIGN
+ && ! (token->flags & PREV_WHITE)
+ && parsing_match_operand)
+ {
+ eat_token (CPP_ATSIGN);
+ token = peek ();
+ value_match = true;
+ }
if (token->type == CPP_NUMBER)
id = get_number ();
else if (token->type == CPP_NAME)
@@ -3982,7 +4047,7 @@ parser::parse_capture (operand *op, bool
fatal_at (src_loc, "unknown capture id");
num = next_id;
}
- return new capture (src_loc, num, op);
+ return new capture (src_loc, num, op, value_match);
}
/* Parse an expression
@@ -4062,15 +4127,8 @@ parser::parse_expr ()
op = parse_capture (e, false);
else if (force_capture)
{
- unsigned num = capture_ids->elements ();
- /* Big enough for a 32-bit UINT_MAX plus prefix. */
- char id[13];
- bool existed;
- sprintf (id, "__%u", num);
- capture_ids->get_or_insert (xstrdup (id), &existed);
- if (existed)
- fatal_at (token, "reserved capture id '%s' already used", id);
- op = new capture (token->src_loc, num, e);
+ unsigned num = get_internal_capture_id ();
+ op = new capture (token->src_loc, num, e, false);
}
else
op = e;
@@ -4384,6 +4449,7 @@ parser::parse_simplify (simplify::simpli
const cpp_token *loc = peek ();
parsing_match_operand = true;
struct operand *match = parse_op ();
+ finish_match_operand (match);
parsing_match_operand = false;
if (match->type == operand::OP_CAPTURE && !matcher)
fatal_at (loc, "outermost expression cannot be captured");
@@ -4724,6 +4790,69 @@ parser::parse_pattern ()
eat_token (CPP_CLOSE_PAREN);
}
+/* Helper for finish_match_operand, collecting captures of OP in CPTS
+ recursively. */
+
+static void
+walk_captures (operand *op, vec<vec<capture *> > cpts)
+{
+ if (! op)
+ return;
+
+ if (capture *c = dyn_cast <capture *> (op))
+ {
+ cpts[c->where].safe_push (c);
+ walk_captures (c->what, cpts);
+ }
+ else if (expr *e = dyn_cast <expr *> (op))
+ for (unsigned i = 0; i < e->ops.length (); ++i)
+ walk_captures (e->ops[i], cpts);
+}
+
+/* Finish up OP which is a match operand. */
+
+void
+parser::finish_match_operand (operand *op)
+{
+ /* Look for matching captures, diagnose mis-uses of @@ and apply
+ early lowering and distribution of value_match. */
+ auto_vec<vec<capture *> > cpts;
+ cpts.safe_grow_cleared (capture_ids->elements ());
+ walk_captures (op, cpts);
+ for (unsigned i = 0; i < cpts.length (); ++i)
+ {
+ capture *value_match = NULL;
+ for (unsigned j = 0; j < cpts[i].length (); ++j)
+ {
+ if (cpts[i][j]->value_match)
+ {
+ if (value_match)
+ fatal_at (cpts[i][j]->location, "duplicate @@");
+ value_match = cpts[i][j];
+ }
+ }
+ if (cpts[i].length () == 1 && value_match)
+ fatal_at (value_match->location, "@@ without a matching capture");
+ if (value_match)
+ {
+ /* Duplicate prevailing capture with the existing ID, create
+ a fake ID and rewrite all captures to use it. This turns
+ @@1 into @__<newid>@1 and @1 into @__<newid>. */
+ value_match->what = new capture (value_match->location,
+ value_match->where,
+ value_match->what, false);
+ /* Create a fake ID and rewrite all captures to use it. */
+ unsigned newid = get_internal_capture_id ();
+ for (unsigned j = 0; j < cpts[i].length (); ++j)
+ {
+ cpts[i][j]->where = newid;
+ cpts[i][j]->value_match = true;
+ }
+ }
+ cpts[i].release ();
+ }
+}
+
/* Main entry of the parser. Repeatedly parse outer control structures. */
parser::parser (cpp_reader *r_)
Index: gcc/match.pd
===================================================================
--- gcc/match.pd (revision 241085)
+++ gcc/match.pd (working copy)
@@ -334,11 +334,8 @@ DEFINE_INT_AND_FLOAT_ROUND_FN (RINT)
/* X - (X / Y) * Y is the same as X % Y. */
(simplify
- (minus (convert1? @2) (convert2? (mult:c (trunc_div @0 @1) @1)))
- /* We cannot use matching captures here, since in the case of
- constants we really want the type of @0, not @2. */
- (if (operand_equal_p (@0, @2, 0)
- && (INTEGRAL_TYPE_P (type) || VECTOR_INTEGER_TYPE_P (type)))
+ (minus (convert1? @0) (convert2? (mult:c (trunc_div @@0 @@1) @1)))
+ (if (INTEGRAL_TYPE_P (type) || VECTOR_INTEGER_TYPE_P (type))
(convert (trunc_mod @0 @1))))
/* Optimize TRUNC_MOD_EXPR by a power of two into a BIT_AND_EXPR,
@@ -714,7 +711,7 @@ DEFINE_INT_AND_FLOAT_ROUND_FN (RINT)
/* (X | Y) ^ X -> Y & ~ X*/
(simplify
- (bit_xor:c (convert? (bit_ior:c @0 @1)) (convert? @0))
+ (bit_xor:c (convert1? (bit_ior:c @@0 @1)) (convert2? @0))
(if (tree_nop_conversion_p (type, TREE_TYPE (@0)))
(convert (bit_and @1 (bit_not @0)))))
@@ -747,7 +744,7 @@ DEFINE_INT_AND_FLOAT_ROUND_FN (RINT)
(for op (bit_and bit_ior bit_xor)
rop (bit_ior bit_and bit_and)
(simplify
- (op (convert? (rop:c @0 @1)) (convert? (rop:c @0 @2)))
+ (op (convert? (rop:c @@0 @1)) (convert? (rop:c @0 @2)))
(if (tree_nop_conversion_p (type, TREE_TYPE (@1))
&& tree_nop_conversion_p (type, TREE_TYPE (@2)))
(rop (convert @0) (op (convert @1) (convert @2))))))
@@ -757,11 +754,11 @@ DEFINE_INT_AND_FLOAT_ROUND_FN (RINT)
(X | Y) | Y -> X | Y */
(for op (bit_and bit_ior)
(simplify
- (op:c (convert?@2 (op:c @0 @1)) (convert? @1))
+ (op:c (convert1?@2 (op:c @0 @@1)) (convert2? @1))
@2))
/* (X ^ Y) ^ Y -> X */
(simplify
- (bit_xor:c (convert? (bit_xor:c @0 @1)) (convert? @1))
+ (bit_xor:c (convert1? (bit_xor:c @0 @@1)) (convert2? @1))
(convert @0))
/* (X & Y) & (X & Z) -> (X & Y) & Z
(X | Y) | (X | Z) -> (X | Y) | Z */
@@ -991,7 +988,7 @@ DEFINE_INT_AND_FLOAT_ROUND_FN (RINT)
/* Fold A - (A & B) into ~B & A. */
(simplify
- (minus (convert? @0) (convert?:s (bit_and:cs @0 @1)))
+ (minus (convert1? @0) (convert2?:s (bit_and:cs @@0 @1)))
(if (tree_nop_conversion_p (type, TREE_TYPE (@0))
&& tree_nop_conversion_p (type, TREE_TYPE (@1)))
(convert (bit_and (bit_not @1) @0))))
@@ -1206,7 +1203,7 @@ DEFINE_INT_AND_FLOAT_ROUND_FN (RINT)
/* (T)(P + A) - (T)P -> (T) A */
(for add (plus pointer_plus)
(simplify
- (minus (convert (add @0 @1))
+ (minus (convert (add @@0 @1))
(convert @0))
(if (element_precision (type) <= element_precision (TREE_TYPE (@1))
/* For integer types, if A has a smaller type
@@ -1231,7 +1228,7 @@ DEFINE_INT_AND_FLOAT_ROUND_FN (RINT)
(for add (plus pointer_plus)
(simplify
(minus (convert @0)
- (convert (add @0 @1)))
+ (convert (add @@0 @1)))
(if (element_precision (type) <= element_precision (TREE_TYPE (@1))
/* For integer types, if A has a smaller type
than T the result depends on the possible
@@ -1254,7 +1251,7 @@ DEFINE_INT_AND_FLOAT_ROUND_FN (RINT)
/* (T)(P + A) - (T)(P + B) -> (T)A - (T)B */
(for add (plus pointer_plus)
(simplify
- (minus (convert (add @0 @1))
+ (minus (convert (add @@0 @1))
(convert (add @0 @2)))
(if (element_precision (type) <= element_precision (TREE_TYPE (@1))
/* For integer types, if A has a smaller type
@@ -1781,11 +1800,8 @@ DEFINE_INT_AND_FLOAT_ROUND_FN (RINT)
/* (X /[ex] A) * A -> X. */
(simplify
- (mult (convert1? (exact_div @0 @1)) (convert2? @2))
- /* We cannot use matching captures here, since in the case of
- constants we don't see the second conversion. */
- (if (operand_equal_p (@1, @2, 0))
- (convert @0)))
+ (mult (convert1? (exact_div @0 @@1)) (convert2? @1))
+ (convert @0))
/* Canonicalization of binary operations. */
Index: gcc/doc/match-and-simplify.texi
===================================================================
--- gcc/doc/match-and-simplify.texi (revision 241085)
+++ gcc/doc/match-and-simplify.texi (working copy)
@@ -110,7 +110,11 @@ are @code{@@} followed by a number or an
@end smallexample
In this example @code{@@0} is mentioned twice which constrains the matched
-expression to have two equal operands. This example also introduces
+expression to have two equal operands. Usually matches are constraint
+to equal types. If operands may be constants and conversions are involved
+matching by value might be preferred in which case use @code{@@@@0} to
+denote a by value match and the specific operand you want to refer to
+in the result part. This example also introduces
operands written in C code. These can be used in the expression
replacements and are supposed to evaluate to a tree node which has to
be a valid GIMPLE operand (so you cannot generate expressions in C code).
^ permalink raw reply [flat|nested] 13+ messages in thread
* Re: [PATCH] Fix PR77826
2016-10-13 8:29 ` Richard Biener
@ 2016-10-15 21:50 ` Marc Glisse
2016-10-17 7:59 ` Richard Biener
0 siblings, 1 reply; 13+ messages in thread
From: Marc Glisse @ 2016-10-15 21:50 UTC (permalink / raw)
To: Richard Biener; +Cc: gcc-patches
On Thu, 13 Oct 2016, Richard Biener wrote:
> The patch introduces '@@' captures which do two things - first they
> change the comparison used for matching back to operand_equal_p
> only (thus perform "value-matching"), second the @@ capture denotes
> the specific operand that should be refered to in the result
> section (ifs and result expressions).
>
> When we face (plus @0 @1) (convert? @@0) for example this is lowered
> to (plus @__2 @1) (convert? @__2@0) marking the @__2 match for
> value handling.
Funny, I had the opposite convention in mind ;-)
On a completely artificial example:
(minus (plus (plus @0 @@0) @@0) @0)
would correspond to:
(minus (plus (plus @1 @2) @3) @4)
where either @1 or @4 is captured as @0, say @1 for example, @4 is
compared strictly to @1, while @2 and @3 are value-compared to @1 (lax).
This way, when we talk about @0 later in the transformation, we are indeed
talking about the thing that was called @0 in the input pattern, while @@0
is not-quite-@0.
But your version should be fine.
> I modified the patterns you identified and the ones I did and
> removed the operand_equal_p uses where possible.
Thanks. (some can be further generalized, as you explained in an earlier
message, but we can do that later as the need arises)
--
Marc Glisse
^ permalink raw reply [flat|nested] 13+ messages in thread
* Re: [PATCH] Fix PR77826
2016-10-15 21:50 ` Marc Glisse
@ 2016-10-17 7:59 ` Richard Biener
0 siblings, 0 replies; 13+ messages in thread
From: Richard Biener @ 2016-10-17 7:59 UTC (permalink / raw)
To: gcc-patches
On Sat, 15 Oct 2016, Marc Glisse wrote:
> On Thu, 13 Oct 2016, Richard Biener wrote:
>
> > The patch introduces '@@' captures which do two things - first they
> > change the comparison used for matching back to operand_equal_p
> > only (thus perform "value-matching"), second the @@ capture denotes
> > the specific operand that should be refered to in the result
> > section (ifs and result expressions).
> >
> > When we face (plus @0 @1) (convert? @@0) for example this is lowered
> > to (plus @__2 @1) (convert? @__2@0) marking the @__2 match for
> > value handling.
>
> Funny, I had the opposite convention in mind ;-)
>
> On a completely artificial example:
> (minus (plus (plus @0 @@0) @@0) @0)
>
> would correspond to:
> (minus (plus (plus @1 @2) @3) @4)
>
> where either @1 or @4 is captured as @0, say @1 for example, @4 is compared
> strictly to @1, while @2 and @3 are value-compared to @1 (lax).
>
> This way, when we talk about @0 later in the transformation, we are indeed
> talking about the thing that was called @0 in the input pattern, while @@0 is
> not-quite-@0.
>
> But your version should be fine.
Yeah, I tried to avoid any ambiguity if @@ is used and didn't like
to write (minus (plus (plus @0 @@0) @@0) @@0) (too many @@s).
> > I modified the patterns you identified and the ones I did and
> > removed the operand_equal_p uses where possible.
>
> Thanks. (some can be further generalized, as you explained in an earlier
> message, but we can do that later as the need arises)
Yes.
Richard.
^ permalink raw reply [flat|nested] 13+ messages in thread
end of thread, other threads:[~2016-10-17 7:59 UTC | newest]
Thread overview: 13+ messages (download: mbox.gz / follow: Atom feed)
-- links below jump to the message on this page --
2016-10-04 10:34 [PATCH] Fix PR77826 Richard Biener
2016-10-05 11:38 ` Richard Biener
2016-10-06 18:44 ` Marc Glisse
2016-10-07 7:36 ` Richard Biener
2016-10-10 8:47 ` Marc Glisse
2016-10-11 9:09 ` Richard Biener
2016-10-11 20:35 ` Marc Glisse
2016-10-12 11:34 ` Richard Biener
2016-10-12 12:49 ` Marc Glisse
2016-10-12 13:25 ` Richard Biener
2016-10-13 8:29 ` Richard Biener
2016-10-15 21:50 ` Marc Glisse
2016-10-17 7:59 ` Richard Biener
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for read-only IMAP folder(s) and NNTP newsgroup(s).