On Tue, 17 Oct 2023, Jakub Jelinek wrote:

> Hi!
> 
> The following patch implements printing of wide_int/widest_int numbers
> decimally when asked for that using print_dec{,s,u}, even if they have
> precision larger than 64 and get_len () above 1 (right now we printed
> them hexadecimally and even negative numbers as huge positive hexadecimal).
> 
> In order to avoid the expensive division/modulo by 10^19 twice, once to
> estimate how many will be needed and another to actually print it, the
> patch prints the 19 digit chunks in reverse order (from least significant
> to most significant) and then reorders those with linear complexity to form
> the right printed number.
> Tested with printing both 256 and 320 bit numbers (first as an example
> of even number of 19 digit chunks plus one shorter above it, the second
> as an example of odd number of 19 digit chunks plus one shorter above it).
> 
> The l * HOST_BITS_PER_WIDE_INT / 3 + 3 estimatition thinking about it now
> is one byte too much (one byte for -, one for '\0') and too conservative,
> so we could go with l * HOST_BITS_PER_WIDE_INT / 3 + 2 as well, or e.g.
> l * HOST_BITS_PER_WIDE_INT * 10 / 33 + 3 as even less conservative
> estimation (though more expensive to compute in inline code).
> But that l * HOST_BITS_PER_WIDE_INT / 4 + 4; is likely one byte too much
> as well, 2 bytes for 0x, one byte for '\0' and where does the 4th one come
> from?  Of course all of these assuming HOST_BITS_PER_WIDE_INT is a multiple
> of 64...
> 
> Bootstrapped/regtested on x86_64-linux and i686-linux, ok for trunk?

OK.

Thanks,
Richard.

> 2023-10-17  Jakub Jelinek  <ja...@redhat.com>
> 
>       * wide-int-print.h (print_dec_buf_size): For length, divide number
>       of bits by 3 and add 3 instead of division by 4 and adding 4.
>       * wide-int-print.cc (print_decs): Remove superfluous ()s.  Don't call
>       print_hex, instead call print_decu on either negated value after
>       printing - or on wi itself.
>       (print_decu): Don't call print_hex, instead print even large numbers
>       decimally.
>       (pp_wide_int_large): Assume len from print_dec_buf_size is big enough
>       even if it returns false.
>       * pretty-print.h (pp_wide_int): Use print_dec_buf_size to check if
>       pp_wide_int_large should be used.
>       * tree-pretty-print.cc (dump_generic_node): Use print_hex_buf_size
>       to compute needed buffer size.
> 
> --- gcc/wide-int-print.h.jj   2023-10-15 23:04:06.195422820 +0200
> +++ gcc/wide-int-print.h      2023-10-16 10:14:41.327401697 +0200
> @@ -42,7 +42,7 @@ print_dec_buf_size (const wide_int_ref &
>    unsigned int l = wi.get_len ();
>    if ((l != 1 || sgn == UNSIGNED) && wi::neg_p (wi))
>      l = WIDE_INT_MAX_HWIS (wi.get_precision ());
> -  l = l * HOST_BITS_PER_WIDE_INT / 4 + 4;
> +  l = l * HOST_BITS_PER_WIDE_INT / 3 + 3;
>    *len = l;
>    return UNLIKELY (l > WIDE_INT_PRINT_BUFFER_SIZE);
>  }
> --- gcc/wide-int-print.cc.jj  2023-10-15 23:04:06.195422820 +0200
> +++ gcc/wide-int-print.cc     2023-10-16 11:20:30.662174735 +0200
> @@ -49,14 +49,12 @@ print_dec (const wide_int_ref &wi, FILE
>  }
>  
>  
> -/* Try to print the signed self in decimal to BUF if the number fits
> -   in a HWI.  Other print in hex.  */
> +/* Try to print the signed self in decimal to BUF.  */
>  
>  void
>  print_decs (const wide_int_ref &wi, char *buf)
>  {
> -  if ((wi.get_precision () <= HOST_BITS_PER_WIDE_INT)
> -      || (wi.get_len () == 1))
> +  if (wi.get_precision () <= HOST_BITS_PER_WIDE_INT || wi.get_len () == 1)
>      {
>        if (wi::neg_p (wi))
>               sprintf (buf, "-" HOST_WIDE_INT_PRINT_UNSIGNED,
> @@ -64,12 +62,17 @@ print_decs (const wide_int_ref &wi, char
>        else
>       sprintf (buf, HOST_WIDE_INT_PRINT_DEC, wi.to_shwi ());
>      }
> +  else if (wi::neg_p (wi))
> +    {
> +      widest2_int w = widest2_int::from (wi, SIGNED);
> +      *buf = '-';
> +      print_decu (-w, buf + 1);
> +    }
>    else
> -    print_hex (wi, buf);
> +    print_decu (wi, buf);
>  }
>  
> -/* Try to print the signed self in decimal to FILE if the number fits
> -   in a HWI.  Other print in hex.  */
> +/* Try to print the signed self in decimal to FILE.  */
>  
>  void
>  print_decs (const wide_int_ref &wi, FILE *file)
> @@ -82,8 +85,7 @@ print_decs (const wide_int_ref &wi, FILE
>    fputs (p, file);
>  }
>  
> -/* Try to print the unsigned self in decimal to BUF if the number fits
> -   in a HWI.  Other print in hex.  */
> +/* Try to print the unsigned self in decimal to BUF.  */
>  
>  void
>  print_decu (const wide_int_ref &wi, char *buf)
> @@ -92,11 +94,37 @@ print_decu (const wide_int_ref &wi, char
>        || (wi.get_len () == 1 && !wi::neg_p (wi)))
>      sprintf (buf, HOST_WIDE_INT_PRINT_UNSIGNED, wi.to_uhwi ());
>    else
> -    print_hex (wi, buf);
> +    {
> +      widest2_int w = widest2_int::from (wi, UNSIGNED), r;
> +      widest2_int ten19 = HOST_WIDE_INT_UC (10000000000000000000);
> +      char buf2[20], next1[19], next2[19];
> +      size_t l, c = 0, i;
> +      /* In order to avoid dividing this twice, print the 19 decimal
> +      digit chunks in reverse order into buffer and then reorder
> +      them in-place.  */
> +      while (wi::gtu_p (w, ten19))
> +     {
> +       w = wi::divmod_trunc (w, ten19, UNSIGNED, &r);
> +       sprintf (buf + c * 19, "%019" PRIu64, r.to_uhwi ());
> +       ++c;
> +     }
> +      l = sprintf (buf2, HOST_WIDE_INT_PRINT_UNSIGNED, w.to_uhwi ());
> +      buf[c * 19 + l] = '\0';
> +      memcpy (next1, buf, 19);
> +      memcpy (buf, buf2, l);
> +      for (i = 0; i < c / 2; ++i)
> +     {
> +       memcpy (next2, buf + (c - i - 1) * 19, 19);
> +       memcpy (buf + l + (c - i - 1) * 19, next1, 19);
> +       memcpy (next1, buf + (i + 1) * 19, 19);
> +       memcpy (buf + l + i * 19, next2, 19);
> +     }
> +      if (c & 1)
> +     memcpy (buf + l + i * 19, next1, 19);
> +    }
>  }
>  
> -/* Try to print the signed self in decimal to FILE if the number fits
> -   in a HWI.  Other print in hex.  */
> +/* Try to print the signed self in decimal to FILE.  */
>  
>  void
>  print_decu (const wide_int_ref &wi, FILE *file)
> @@ -155,8 +183,7 @@ void
>  pp_wide_int_large (pretty_printer *pp, const wide_int_ref &w, signop sgn)
>  {
>    unsigned int len;
> -  if (!print_dec_buf_size (w, sgn, &len))
> -    len = WIDE_INT_PRINT_BUFFER_SIZE;
> +  print_dec_buf_size (w, sgn, &len);
>    char *buf = XALLOCAVEC (char, len);
>    print_dec (w, buf, sgn);
>    pp_string (pp, buf);
> --- gcc/pretty-print.h.jj     2023-10-15 23:04:06.095422965 +0200
> +++ gcc/pretty-print.h        2023-10-16 10:51:56.053529117 +0200
> @@ -448,8 +448,9 @@ pp_wide_integer (pretty_printer *pp, HOS
>  inline void
>  pp_wide_int (pretty_printer *pp, const wide_int_ref &w, signop sgn)
>  {
> -  unsigned int prec = w.get_precision ();
> -  if (UNLIKELY ((prec + 3) / 4 > sizeof (pp_buffer (pp)->digit_buffer) - 3))
> +  unsigned int len;
> +  print_dec_buf_size (w, sgn, &len);
> +  if (UNLIKELY (len > sizeof (pp_buffer (pp)->digit_buffer)))
>      pp_wide_int_large (pp, w, sgn);
>    else
>      {
> --- gcc/tree-pretty-print.cc.jj       2023-09-21 20:02:53.467522151 +0200
> +++ gcc/tree-pretty-print.cc  2023-10-16 11:05:51.131997367 +0200
> @@ -2248,10 +2248,11 @@ dump_generic_node (pretty_printer *pp, t
>             pp_minus (pp);
>             val = -val;
>           }
> -       unsigned int prec = val.get_precision ();
> -       if ((prec + 3) / 4 > sizeof (pp_buffer (pp)->digit_buffer) - 3)
> +       unsigned int len;
> +       print_hex_buf_size (val, &len);
> +       if (UNLIKELY (len > sizeof (pp_buffer (pp)->digit_buffer)))
>           {
> -           char *buf = XALLOCAVEC (char, (prec + 3) / 4 + 3);
> +           char *buf = XALLOCAVEC (char, len);
>             print_hex (val, buf);
>             pp_string (pp, buf);
>           }
> 
>       Jakub
> 
> 

-- 
Richard Biener <rguent...@suse.de>
SUSE Software Solutions Germany GmbH,
Frankenstrasse 146, 90461 Nuernberg, Germany;
GF: Ivo Totev, Andrew McDonald, Werner Knoblich; (HRB 36809, AG Nuernberg)

Reply via email to