On Sat, 18 Jul 2020, Jakub Jelinek wrote:

> Hi!
> 
> When working on __builtin_bit_cast that needs to handle bitfields too,
> I've made the following change to handle at least some bitfields in
> native_encode_initializer (those that have integral representative).
> 
> Bootstrapped/regtested on {x86_64,i686,powerpc64{,le}}-linux, ok for trunk?

OK.

Thanks,
Richard.

> 2020-07-18  Jakub Jelinek  <ja...@redhat.com>
> 
>       PR libstdc++/93121
>       * fold-const.c (native_encode_initializer): Handle bit-fields.
> 
>       * gcc.dg/tree-ssa/pr93121-1.c: New test.
> 
> --- gcc/fold-const.c.jj       2020-06-24 10:39:48.995469213 +0200
> +++ gcc/fold-const.c  2020-07-17 13:51:53.181392890 +0200
> @@ -8047,6 +8047,7 @@ native_encode_initializer (tree init, un
>             tree field = ce->index;
>             tree val = ce->value;
>             HOST_WIDE_INT pos, fieldsize;
> +           unsigned HOST_WIDE_INT bpos = 0, epos = 0;
>  
>             if (field == NULL_TREE)
>               return 0;
> @@ -8066,15 +8067,122 @@ native_encode_initializer (tree init, un
>             if (fieldsize == 0)
>               continue;
>  
> +           if (DECL_BIT_FIELD (field))
> +             {
> +               if (!tree_fits_uhwi_p (DECL_FIELD_BIT_OFFSET (field)))
> +                 return 0;
> +               fieldsize = TYPE_PRECISION (TREE_TYPE (field));
> +               bpos = tree_to_uhwi (DECL_FIELD_BIT_OFFSET (field));
> +               if (bpos % BITS_PER_UNIT)
> +                 bpos %= BITS_PER_UNIT;
> +               else
> +                 bpos = 0;
> +               fieldsize += bpos;
> +               epos = fieldsize % BITS_PER_UNIT;
> +               fieldsize += BITS_PER_UNIT - 1;
> +               fieldsize /= BITS_PER_UNIT;
> +             }
> +
>             if (off != -1 && pos + fieldsize <= off)
>               continue;
>  
> -           if (DECL_BIT_FIELD (field))
> -             return 0;
> -
>             if (val == NULL_TREE)
>               continue;
>  
> +           if (DECL_BIT_FIELD (field))
> +             {
> +               /* FIXME: Handle PDP endian.  */
> +               if (BYTES_BIG_ENDIAN != WORDS_BIG_ENDIAN)
> +                 return 0;
> +
> +               tree repr = DECL_BIT_FIELD_REPRESENTATIVE (field);
> +               if (repr == NULL_TREE
> +                   || TREE_CODE (val) != INTEGER_CST
> +                   || !INTEGRAL_TYPE_P (TREE_TYPE (repr)))
> +                 return 0;
> +
> +               HOST_WIDE_INT rpos = int_byte_position (repr);
> +               if (rpos > pos)
> +                 return 0;
> +               wide_int w = wi::to_wide (val,
> +                                         TYPE_PRECISION (TREE_TYPE (repr)));
> +               int diff = (TYPE_PRECISION (TREE_TYPE (repr))
> +                           - TYPE_PRECISION (TREE_TYPE (field)));
> +               HOST_WIDE_INT bitoff = (pos - rpos) * BITS_PER_UNIT + bpos;
> +               if (!BYTES_BIG_ENDIAN)
> +                 w = wi::lshift (w, bitoff);
> +               else
> +                 w = wi::lshift (w, diff - bitoff);
> +               val = wide_int_to_tree (TREE_TYPE (repr), w);
> +
> +               unsigned char buf[MAX_BITSIZE_MODE_ANY_INT
> +                                 / BITS_PER_UNIT + 1];
> +               int l = native_encode_int (val, buf, sizeof buf, 0);
> +               if (l * BITS_PER_UNIT != TYPE_PRECISION (TREE_TYPE (repr)))
> +                 return 0;
> +
> +               if (ptr == NULL)
> +                 continue;
> +
> +               /* If the bitfield does not start at byte boundary, handle
> +                  the partial byte at the start.  */
> +               if (bpos
> +                   && (off == -1 || (pos >= off && len >= 1)))
> +                 {
> +                   if (!BYTES_BIG_ENDIAN)
> +                     {
> +                       int mask = (1 << bpos) - 1;
> +                       buf[pos - rpos] &= ~mask;
> +                       buf[pos - rpos] |= ptr[pos - o] & mask;
> +                     }
> +                   else
> +                     {
> +                       int mask = (1 << (BITS_PER_UNIT - bpos)) - 1;
> +                       buf[pos - rpos] &= mask;
> +                       buf[pos - rpos] |= ptr[pos - o] & ~mask;
> +                     }
> +                 }
> +               /* If the bitfield does not end at byte boundary, handle
> +                  the partial byte at the end.  */
> +               if (epos
> +                   && (off == -1
> +                       || pos + fieldsize <= (HOST_WIDE_INT) off + len))
> +                 {
> +                   if (!BYTES_BIG_ENDIAN)
> +                     {
> +                       int mask = (1 << epos) - 1;
> +                       buf[pos - rpos + fieldsize - 1] &= mask;
> +                       buf[pos - rpos + fieldsize - 1]
> +                         |= ptr[pos + fieldsize - 1 - o] & ~mask;
> +                     }
> +                    else
> +                     {
> +                       int mask = (1 << (BITS_PER_UNIT - epos)) - 1;
> +                       buf[pos - rpos + fieldsize - 1] &= ~mask;
> +                       buf[pos - rpos + fieldsize - 1]
> +                         |= ptr[pos + fieldsize - 1 - o] & mask;
> +                     }
> +                 }
> +               if (off == -1
> +                   || (pos >= off
> +                       && (pos + fieldsize <= (HOST_WIDE_INT) off + len)))
> +                 memcpy (ptr + pos - o, buf + (pos - rpos), fieldsize);
> +               else
> +                 {
> +                   /* Partial overlap.  */
> +                   HOST_WIDE_INT fsz = fieldsize;
> +                   if (pos < off)
> +                     {
> +                       fsz -= (off - pos);
> +                       pos = off;
> +                     }
> +                   if (pos + fsz > (HOST_WIDE_INT) off + len)
> +                     fsz = (HOST_WIDE_INT) off + len - pos;
> +                   memcpy (ptr + pos - off, buf + (pos - rpos), fsz);
> +                 }
> +               continue;
> +             }
> +
>             if (off == -1
>                 || (pos >= off
>                     && (pos + fieldsize <= (HOST_WIDE_INT) off + len)))
> --- gcc/testsuite/gcc.dg/tree-ssa/pr93121-1.c.jj      2020-07-17 
> 13:33:47.106539060 +0200
> +++ gcc/testsuite/gcc.dg/tree-ssa/pr93121-1.c 2020-07-17 13:54:46.786813165 
> +0200
> @@ -0,0 +1,56 @@
> +/* PR libstdc++/93121 */
> +/* { dg-do compile { target { ilp32 || lp64 } } } */
> +/* { dg-options "-O2 -fdump-tree-optimized" } */
> +
> +union U { int a[3]; short c[6]; struct S { int d; int a : 2; int f : 1; int 
> b : 24; int c : 5; int e; } b; };
> +const union U u = { .b = { 0x7efa3412, 3, 0, 0x50eca8, 0xb, 0x1eeffeed } };
> +const union U v = { .b = { 0x7efa3412, 1, 1, 0x7feedb, 0x5, 0x1eeffeed } };
> +union W { struct T { long long int a, b : 11, c : 3, d : 37, e : 1, f : 10, 
> g : 2, h; } a; int b[6]; short c[12]; long long d[3]; };
> +const union W w = { .a = { 0x0feedbacdeadbeefLL, -1011, 2, -0xbacdeadbeLL, 
> -1, 721, 1, 0x0feedbacdeadbeefLL } };
> +int a, b, c, d, e, f, g, h, i, j, k, l;
> +long long m;
> +
> +void
> +foo ()
> +{
> +  a = u.a[1];
> +  b = v.a[1];
> +  c = u.c[2];
> +  d = u.c[3];
> +  e = v.c[2];
> +  f = v.c[3];
> +  g = w.b[2];
> +  h = w.b[3];
> +  i = w.c[4];
> +  j = w.c[5];
> +  k = w.c[6];
> +  l = w.c[7];
> +  m = w.d[1];
> +}
> +
> +/* { dg-final { scan-tree-dump-times "a = 1518822723;" 1 "optimized" { 
> target le } } } */
> +/* { dg-final { scan-tree-dump-times "b = 738162397;" 1 "optimized" { target 
> le } } } */
> +/* { dg-final { scan-tree-dump-times "c = 25923;" 1 "optimized" { target le 
> } } } */
> +/* { dg-final { scan-tree-dump-times "d = 23175;" 1 "optimized" { target le 
> } } } */
> +/* { dg-final { scan-tree-dump-times "e = 30429;" 1 "optimized" { target le 
> } } } */
> +/* { dg-final { scan-tree-dump-times "f = 11263;" 1 "optimized" { target le 
> } } } */
> +/* { dg-final { scan-tree-dump-times "g = 1418761229;" 1 "optimized" { 
> target le } } } */
> +/* { dg-final { scan-tree-dump-times "h = 1830622408;" 1 "optimized" { 
> target le } } } */
> +/* { dg-final { scan-tree-dump-times "i = -27635;" 1 "optimized" { target le 
> } } } */
> +/* { dg-final { scan-tree-dump-times "j = 21648;" 1 "optimized" { target le 
> } } } */
> +/* { dg-final { scan-tree-dump-times "k = 5320;" 1 "optimized" { target le } 
> } } */
> +/* { dg-final { scan-tree-dump-times "l = 27933;" 1 "optimized" { target le 
> } } } */
> +/* { dg-final { scan-tree-dump-times "m = 7862463375103529997;" 1 
> "optimized" { target le } } } */
> +/* { dg-final { scan-tree-dump-times "a = -904030965;" 1 "optimized" { 
> target be } } } */
> +/* { dg-final { scan-tree-dump-times "b = 1878907749;" 1 "optimized" { 
> target be } } } */
> +/* { dg-final { scan-tree-dump-times "c = -13795;" 1 "optimized" { target be 
> } } } */
> +/* { dg-final { scan-tree-dump-times "d = -27381;" 1 "optimized" { target be 
> } } } */
> +/* { dg-final { scan-tree-dump-times "e = 28669;" 1 "optimized" { target be 
> } } } */
> +/* { dg-final { scan-tree-dump-times "f = -9371;" 1 "optimized" { target be 
> } } } */
> +/* { dg-final { scan-tree-dump-times "g = -2119529884;" 1 "optimized" { 
> target be } } } */
> +/* { dg-final { scan-tree-dump-times "h = 709385029;" 1 "optimized" { target 
> be } } } */
> +/* { dg-final { scan-tree-dump-times "i = -32342;" 1 "optimized" { target be 
> } } } */
> +/* { dg-final { scan-tree-dump-times "j = -30108;" 1 "optimized" { target be 
> } } } */
> +/* { dg-final { scan-tree-dump-times "k = 10824;" 1 "optimized" { target be 
> } } } */
> +/* { dg-final { scan-tree-dump-times "l = 23365;" 1 "optimized" { target be 
> } } } */
> +/* { dg-final { scan-tree-dump-times "m = -9103311533965288635;" 1 
> "optimized" { target be } } } */
> 
>       Jakub
> 
> 

-- 
Richard Biener <rguent...@suse.de>
SUSE Software Solutions Germany GmbH, Maxfeldstrasse 5, 90409 Nuernberg,
Germany; GF: Felix Imendörffer; HRB 36809 (AG Nuernberg)

Reply via email to