]> git.ipfire.org Git - thirdparty/gcc.git/commitdiff
libstdc++: Fix incorrect use of abs and log10 in std::format [PR110860]
authorJonathan Wakely <jwakely@redhat.com>
Mon, 7 Aug 2023 14:30:03 +0000 (15:30 +0100)
committerJonathan Wakely <jwakely@redhat.com>
Mon, 7 Aug 2023 21:09:11 +0000 (22:09 +0100)
The std::formatter implementation for floating-point types uses
__builtin_abs and __builtin_log10 to avoid including all of <cmath>, but
those functions are not generic. The result of abs(2e304) is -INT_MIN
which is undefined, and then log10(INT_MIN) is NaN. As well as being
undefined, we fail to grow the buffer correctly, and then loop more
times than needed to allocate a buffer and try formatting the value into
it again.

We can use if-constexpr to choose the correct form of log10 to use for
the type, and avoid using abs entirely. This avoids the undefined
behaviour and should mean we only reallocate and retry std::to_chars
once.

libstdc++-v3/ChangeLog:

PR libstdc++/110860
* include/std/format (__formatter_fp::format): Do not use
__builtin_abs and __builtin_log10 with arbitrary floating-point
types.

libstdc++-v3/include/std/format

index 60e53642b11957310503a59a67e91b6df1e241e9..f68308e721058e3580717e1e323c5bad372abbab 100644 (file)
@@ -1487,9 +1487,20 @@ namespace __format
            {
              // If the buffer is too small it's probably because of a large
              // precision, or a very large value in fixed format.
-             size_t __guess =  __prec + sizeof(__buf);
-             if (__fmt == chars_format::fixed)
-               __guess += max((int)__builtin_log10(__builtin_abs(__v)) / 2, 1);
+             size_t __guess = 8 + __prec;
+             if (__fmt == chars_format::fixed) // +ddd.prec
+               {
+                 if constexpr (is_same_v<_Fp, float>)
+                   __guess += __builtin_log10f(__v < 0.0f ? -__v : __v);
+                 else if constexpr (is_same_v<_Fp, double>)
+                   __guess += __builtin_log10(__v < 0.0 ? -__v : __v);
+                 else if constexpr (is_same_v<_Fp, long double>)
+                   __guess += __builtin_log10l(__v < 0.0l ? -__v : __v);
+                 else
+                   __guess += numeric_limits<_Fp>::max_exponent10;
+               }
+             if (__guess <= sizeof(__buf)) [[unlikely]]
+               __guess = sizeof(__buf) * 2;
              __dynbuf.reserve(__guess);
 
              do