On Fri, Aug 9, 2024 at 2:58 PM Juergen Christ <jchr...@linux.ibm.com> wrote:
>
> Am Thu, Aug 08, 2024 at 02:06:44PM +0200 schrieb Richard Biener:
> > On Mon, Aug 5, 2024 at 4:02 PM Juergen Christ <jchr...@linux.ibm.com> wrote:
> > >
> > > Am Mon, Aug 05, 2024 at 01:00:31PM +0200 schrieb Richard Biener:
> > > > On Fri, Aug 2, 2024 at 2:43 PM Juergen Christ <jchr...@linux.ibm.com> 
> > > > wrote:
> > > > >
> > > > > Do not convert floats to ints in multiple step if trapping math is
> > > > > enabled.  This might hide some inexact signals.
> > > > >
> > > > > Also use correct sign (the sign of the target integer type) for the
> > > > > intermediate steps.  This only affects undefined behaviour (casting
> > > > > floats to unsigned datatype where the float is negative).
> > > > >
> > > > > gcc/ChangeLog:
> > > > >
> > > > >         * tree-vect-stmts.cc (vectorizable_conversion): multi-step
> > > > >           float to int conversion only with trapping math and correct
> > > > >           sign.
> > > > >
> > > > > Signed-off-by: Juergen Christ <jchr...@linux.ibm.com>
> > > > >
> > > > > Bootstrapped and tested on x84 and s390.  Ok for trunk?
> > > > >
> > > > > ---
> > > > >  gcc/tree-vect-stmts.cc | 8 +++++---
> > > > >  1 file changed, 5 insertions(+), 3 deletions(-)
> > > > >
> > > > > diff --git a/gcc/tree-vect-stmts.cc b/gcc/tree-vect-stmts.cc
> > > > > index fdcda0d2abae..2ddd13383193 100644
> > > > > --- a/gcc/tree-vect-stmts.cc
> > > > > +++ b/gcc/tree-vect-stmts.cc
> > > > > @@ -5448,7 +5448,8 @@ vectorizable_conversion (vec_info *vinfo,
> > > > >             break;
> > > > >
> > > > >           cvt_type
> > > > > -           = build_nonstandard_integer_type (GET_MODE_BITSIZE 
> > > > > (rhs_mode), 0);
> > > > > +           = build_nonstandard_integer_type (GET_MODE_BITSIZE 
> > > > > (rhs_mode),
> > > > > +                                             TYPE_UNSIGNED 
> > > > > (lhs_type));
> > > >
> > > > But lhs_type should be a float type here, the idea that for a
> > > > FLOAT_EXPR (int -> float)
> > > > a signed integer type is the natural one to use - as it's 2x wider
> > > > than the original
> > > > RHS type it's signedness doesn't matter.  Note all float types should be
> > > > !TYPE_UNSIGNED so this hunk is a no-op but still less clear on the 
> > > > intent IMO.
> > > >
> > > > Please drop it.
> > >
> > > Will do.  Sorry about that.
> > >
> > > > >           cvt_type = get_same_sized_vectype (cvt_type, vectype_in);
> > > > >           if (cvt_type == NULL_TREE)
> > > > >             goto unsupported;
> > > > > @@ -5505,10 +5506,11 @@ vectorizable_conversion (vec_info *vinfo,
> > > > >        if (GET_MODE_SIZE (lhs_mode) >= GET_MODE_SIZE (rhs_mode))
> > > > >         goto unsupported;
> > > > >
> > > > > -      if (code == FIX_TRUNC_EXPR)
> > > > > +      if (code == FIX_TRUNC_EXPR && !flag_trapping_math)
> > > > >         {
> > > > >           cvt_type
> > > > > -           = build_nonstandard_integer_type (GET_MODE_BITSIZE 
> > > > > (rhs_mode), 0);
> > > > > +           = build_nonstandard_integer_type (GET_MODE_BITSIZE 
> > > > > (rhs_mode),
> > > > > +                                             TYPE_UNSIGNED 
> > > > > (lhs_type));
> > > >
> > > > Here it might be relevant for correctness - we have to choose between
> > > > sfix and ufix for the float -> [u]int conversion.
> > > >
> > > > Do  you have a testcase?  Shouldn't the exactness be independent of the 
> > > > integer
> > > > type we convert to?
> > >
> > > I was looking at this little program which contains undefined behaviour:
> > >
> > > #include <stdio.h>
> > >
> > > __attribute__((noinline,noclone,noipa))
> > > void
> > > vec_pack_ufix_trunc_v2df (double *in, unsigned int *out);
> > >
> > > void
> > > vec_pack_ufix_trunc_v2df (double *in, unsigned int *out)
> > > {
> > >         out[0] = in[0];
> > >         out[1] = in[1];
> > >         out[2] = in[2];
> > >         out[3] = in[3];
> > > }
> > >
> > > int main()
> > > {
> > >         double in[] = {-1,-2,-3,-4};
> > >         unsigned int out[4];
> > >
> > >         vec_pack_ufix_trunc_v2df (in, out);
> > >         for (int i = 0; i < 4; ++i)
> > >                 printf("out[%d] = %u\n", i, out[i]);
> > >         return 0;
> > > }
> > >
> > > On s390x, I get different results after vectorization:
> > >
> > > out[0] = 4294967295
> > > out[1] = 4294967294
> > > out[2] = 4294967293
> > > out[3] = 4294967292
> > >
> > > than without vectorization:
> > >
> > > out[0] = 0
> > > out[1] = 0
> > > out[2] = 0
> > > out[3] = 0
> > >
> > > Even if this is undefined behaviour, I think it would be nice to have
> > > consistent results here.
> > >
> > > Also, while I added an expander to circumvent this problem in a
> > > previous patch, reviewers requested to hide this behind trapping math.
> > > Thus, I looked into this.
> > >
> > > Seeing the result from the CI for aarch64, I guess there are some
> > > tests that actually expect this vectorization to always happen even
> > > though it might not be save w.r.t. trapping math.
> >
> > I do remember this was extensively discussed (but we might have missed
> > something) and one argument indeed was that when it's undefined behavior
> > we can do the vectorization given the actual values might be in-bound.
>
> Okay.  Would you be fine with the patch to only vectorize when
> trapping math is disabled?  I still could take care of the rest on
> s390 backend side by defining the appropriate expanders.  Still think
> it is weird though that we might produce different results on
> vectorization than without vectorization.  Yes, that is what
> "undefined behaviour" is all about, but we can simply fix this here.
> Nevertheless, how about just adding the trapping math check?

So to summarize - the problem is different result when vectorizing
when the scalar code invokes undefined behavior?  I think there is
nothing to fix and we shouldn't pessimize code not invoking undefined
behavior by adding a trapping math check.

Or did I misunderstand things?

Thanks,
Richard.

Reply via email to