On Fri, 27 Jun 2025 at 10:10, Tomasz Kaminski <tkami...@redhat.com> wrote:
>
>
>
> On Fri, Jun 27, 2025 at 10:31 AM Tomasz Kamiński <tkami...@redhat.com> wrote:
>>
>> This patch adjust all internal std::format call inside of __formatter_chrono,
>> to use runtime format stirng and thus avoid compile time checking of validity
>> of the format string. Majority of cases are covered by calling newly 
>> introduced
>> _S_empty_fs() function that returns __Runtime_format_string containing
>> _S_empty_spec, instead of passing later directly.
>>
>> In case of _M_j we use _S_str_d3 function (extracted from _S_str_d2), 
>> eliminating
>> call to std::format outside of unlikely scenario in which day of year is 
>> greater
>> than 1000 (this may happen for year_month_day with month greater than 12). In
>> consequence, outside of handling subseconds, we no longer delegate to 
>> std::format
>> or construct temporary strings, when formatting chrono types with ok() 
>> values.
>>
>>         PR libstdc++/110739
>>
>> libstdc++-v3/ChangeLog:
>>
>>         * include/bits/chrono_io.h (__formatter_chrono::_S_empty_fs): Define.
>>         (__formatter_chrono::_S_str_d2): Use _S_str_d3 for 3+ digits.
>>         (__formatter_chrono::_S_str_d3): Extracted from _S_str_d2.
>>         (__formatter_chrono::_M_H_I, __formatter_chrono::_M_R_X): Replace
>>         _S_empty_spec with _S_empty_fs().
>>         (__formatter_chrono::_M_j): Likewise and use _S_str_d3 in common
>>         case.
>> ---
>>
>>
>
> I do not think this buys us much, but I think it is worth doing anyway.

I agree, I've never really liked doing recursive std::format calls
inside formatters (obviously sometimes it's unavoidable, e.g. for
custom duration rep types).

> It also finishes my side goal, of getting rid of temporary strings, and using 
> local buffers,
> that I applied to other specifiers in previous commits.
> Tested on x86_64-linux locally. The std/time* test passed with
> -D_GLIBCXX_USE_CXX11_ABI=0 and -D_GLIBCXX_DEBUG.

Just to be sure, it passes without those options too, right? :-)

> OK for trunk?

Please put the always_inline attribute after the comment on _S_str_d3.

OK for trunk with that change, thanks.


>
>>
>>  libstdc++-v3/include/bits/chrono_io.h | 27 ++++++++++++++++++++++-----
>>  1 file changed, 22 insertions(+), 5 deletions(-)
>>
>> diff --git a/libstdc++-v3/include/bits/chrono_io.h 
>> b/libstdc++-v3/include/bits/chrono_io.h
>> index bcfd51b9866..d6bc6c7cf2a 100644
>> --- a/libstdc++-v3/include/bits/chrono_io.h
>> +++ b/libstdc++-v3/include/bits/chrono_io.h
>> @@ -873,6 +873,11 @@ namespace __format
>>        static constexpr const _CharT* _S_minus_empty_spec = _S_chars + 17;
>>        static constexpr const _CharT* _S_empty_spec = _S_chars + 18;
>>
>> +      [[__gnu__::__always_inline__]]
>> +      static _Runtime_format_string<_CharT>
>> +      _S_empty_fs()
>> +      { return _Runtime_format_string<_CharT>(_S_empty_spec); }
>> +
>>        // Return the formatting locale.
>>        template<typename _FormatContext>
>>         std::locale
>> @@ -1411,7 +1416,7 @@ namespace __format
>>                 __i = 12;
>>             }
>>           else if (__i >= 100) [[unlikely]]
>> -           return std::format_to(std::move(__out), _S_empty_spec, __i);
>> +           return std::format_to(std::move(__out), _S_empty_fs(), __i);
>>
>>           return __format::__write(std::move(__out), _S_two_digits(__i));
>>         }
>> @@ -1425,11 +1430,15 @@ namespace __format
>>           {
>>             // Decimal number of days, without padding.
>>             auto __d = chrono::floor<chrono::days>(__t._M_hours).count();
>> -           return std::format_to(std::move(__out), _S_empty_spec, __d);
>> +           return std::format_to(std::move(__out), _S_empty_fs(), __d);
>>           }
>>
>> -         return std::format_to(std::move(__out), _GLIBCXX_WIDEN("{:03d}"),
>> -                               __t._M_day_of_year.count());
>> +         auto __d = __t._M_day_of_year.count();
>> +         if (__d >= 1000) [[unlikely]]
>> +           return std::format_to(std::move(__out), _S_empty_fs(), __d);
>> +
>> +         _CharT __buf[3];
>> +         return __format::__write(std::move(__out), _S_str_d3(__buf, __d));
>>         }
>>
>>        template<typename _FormatContext>
>> @@ -1534,7 +1543,7 @@ namespace __format
>>
>>           if (__hi >= 100) [[unlikely]]
>>             {
>> -             __out = std::format_to(std::move(__out), _S_empty_spec, __hi);
>> +             __out = std::format_to(std::move(__out), _S_empty_fs(), __hi);
>>               __sv.remove_prefix(2);
>>             }
>>           else
>> @@ -1772,7 +1781,15 @@ namespace __format
>>        {
>>         if (__n < 100) [[likely]]
>>           return _S_two_digits(__n);
>> +        return _S_str_d3(__buf, __n);
>> +      }
>>
>> +      [[__gnu__::__always_inline__]]
>> +      // Returns decimal representation of __n, padded to 3 digits.
>> +      // Returned string_view points to __buf.
>> +      static basic_string_view<_CharT>
>> +      _S_str_d3(span<_CharT, 3> __buf, unsigned __n)
>> +      {
>>         _S_fill_two_digits(__buf.data(), __n / 10);
>>         __buf[2] = _S_chars[__n % 10];
>>         return __string_view(__buf.data(), 3);
>> --
>> 2.49.0
>>

Reply via email to