diff --git a/libcxx/include/__chrono/formatter.h b/libcxx/include/__chrono/formatter.h index 226fccbee6d13..e9b81c3de8a70 100644 --- a/libcxx/include/__chrono/formatter.h +++ b/libcxx/include/__chrono/formatter.h @@ -88,6 +88,9 @@ __format_sub_seconds(basic_stringstream<_CharT>& __sstr, const chrono::duration< using __duration = chrono::duration<_Rep, _Period>; auto __fraction = __value - chrono::duration_cast(__value); + // Converts a negative fraction to its positive value. + if (__value < chrono::seconds{0} && __fraction != __duration{0}) + __fraction += chrono::seconds{1}; if constexpr (chrono::treat_as_floating_point_v<_Rep>) // When the floating-point value has digits itself they are ignored based // on the wording in [tab:time.format.spec] diff --git a/libcxx/test/std/time/time.clock/time.clock.file/ostream.pass.cpp b/libcxx/test/std/time/time.clock/time.clock.file/ostream.pass.cpp index 18a4506b91566..11eab1dddfe38 100644 --- a/libcxx/test/std/time/time.clock/time.clock.file/ostream.pass.cpp +++ b/libcxx/test/std/time/time.clock/time.clock.file/ostream.pass.cpp @@ -71,8 +71,45 @@ template static void test_c() { using namespace std::literals::chrono_literals; + assert(stream_c_locale(std::chrono::file_time{-946'688'523'123'456'789ns}) == + SV("1940-01-01 22:57:56.876543211")); + + assert(stream_c_locale(std::chrono::file_time{-946'688'523'123'456us}) == + SV("1940-01-01 22:57:56.876544")); + + assert(stream_c_locale(std::chrono::file_time{-946'688'523'123ms}) == + SV("1940-01-01 22:57:56.877")); + + assert(stream_c_locale(std::chrono::file_time{-1000000000ns}) == + SV("1969-12-31 23:59:59.000000000")); + + assert(stream_c_locale(std::chrono::file_time{-1000000us}) == + SV("1969-12-31 23:59:59.000000")); + + assert(stream_c_locale(std::chrono::file_time{-1000ms}) == + SV("1969-12-31 23:59:59.000")); + + assert(stream_c_locale(std::chrono::file_time{-1ns}) == + SV("1969-12-31 23:59:59.999999999")); + + assert(stream_c_locale(std::chrono::file_time{0ns}) == + SV("1970-01-01 00:00:00.000000000")); + + assert(stream_c_locale(std::chrono::file_time{1ns}) == + SV("1970-01-01 00:00:00.000000001")); + + assert(stream_c_locale(std::chrono::file_time{1000000000ns}) == + SV("1970-01-01 00:00:01.000000000")); + + assert(stream_c_locale(std::chrono::file_time{1000000us}) == + SV("1970-01-01 00:00:01.000000")); + + assert(stream_c_locale(std::chrono::file_time{1000ms}) == + SV("1970-01-01 00:00:01.000")); + assert(stream_c_locale(file_time{946'688'523'123'456'789ns}) == SV("2000-01-01 01:02:03.123456789")); + assert(stream_c_locale(file_time{946'688'523'123'456us}) == SV("2000-01-01 01:02:03.123456")); @@ -107,6 +144,42 @@ template static void test_fr_FR() { using namespace std::literals::chrono_literals; + assert(stream_fr_FR_locale(std::chrono::file_time{-946'688'523'123'456'789ns}) == + SV("1940-01-01 22:57:56,876543211")); + + assert(stream_fr_FR_locale(std::chrono::file_time{-946'688'523'123'456us}) == + SV("1940-01-01 22:57:56,876544")); + + assert(stream_fr_FR_locale(std::chrono::file_time{-946'688'523'123ms}) == + SV("1940-01-01 22:57:56,877")); + + assert(stream_fr_FR_locale(std::chrono::file_time{-1000000000ns}) == + SV("1969-12-31 23:59:59,000000000")); + + assert(stream_fr_FR_locale(std::chrono::file_time{-1000000us}) == + SV("1969-12-31 23:59:59,000000")); + + assert(stream_fr_FR_locale(std::chrono::file_time{-1000ms}) == + SV("1969-12-31 23:59:59,000")); + + assert(stream_fr_FR_locale(std::chrono::file_time{-1ns}) == + SV("1969-12-31 23:59:59,999999999")); + + assert(stream_fr_FR_locale(std::chrono::file_time{0ns}) == + SV("1970-01-01 00:00:00,000000000")); + + assert(stream_fr_FR_locale(std::chrono::file_time{1ns}) == + SV("1970-01-01 00:00:00,000000001")); + + assert(stream_fr_FR_locale(std::chrono::file_time{1000000000ns}) == + SV("1970-01-01 00:00:01,000000000")); + + assert(stream_fr_FR_locale(std::chrono::file_time{1000000us}) == + SV("1970-01-01 00:00:01,000000")); + + assert(stream_fr_FR_locale(std::chrono::file_time{1000ms}) == + SV("1970-01-01 00:00:01,000")); + assert(stream_fr_FR_locale(file_time{946'688'523'123'456'789ns}) == SV("2000-01-01 01:02:03,123456789")); assert(stream_fr_FR_locale(file_time{946'688'523'123'456us}) == @@ -144,6 +217,42 @@ template static void test_ja_JP() { using namespace std::literals::chrono_literals; + assert(stream_ja_JP_locale(std::chrono::file_time{-946'688'523'123'456'789ns}) == + SV("1940-01-01 22:57:56.876543211")); + + assert(stream_ja_JP_locale(std::chrono::file_time{-946'688'523'123'456us}) == + SV("1940-01-01 22:57:56.876544")); + + assert(stream_ja_JP_locale(std::chrono::file_time{-946'688'523'123ms}) == + SV("1940-01-01 22:57:56.877")); + + assert(stream_ja_JP_locale(std::chrono::file_time{-1000000000ns}) == + SV("1969-12-31 23:59:59.000000000")); + + assert(stream_ja_JP_locale(std::chrono::file_time{-1000000us}) == + SV("1969-12-31 23:59:59.000000")); + + assert(stream_ja_JP_locale(std::chrono::file_time{-1000ms}) == + SV("1969-12-31 23:59:59.000")); + + assert(stream_ja_JP_locale(std::chrono::file_time{-1ns}) == + SV("1969-12-31 23:59:59.999999999")); + + assert(stream_ja_JP_locale(std::chrono::file_time{0ns}) == + SV("1970-01-01 00:00:00.000000000")); + + assert(stream_ja_JP_locale(std::chrono::file_time{1ns}) == + SV("1970-01-01 00:00:00.000000001")); + + assert(stream_ja_JP_locale(std::chrono::file_time{1000000000ns}) == + SV("1970-01-01 00:00:01.000000000")); + + assert(stream_ja_JP_locale(std::chrono::file_time{1000000us}) == + SV("1970-01-01 00:00:01.000000")); + + assert(stream_ja_JP_locale(std::chrono::file_time{1000ms}) == + SV("1970-01-01 00:00:01.000")); + assert(stream_ja_JP_locale(file_time{946'688'523'123'456'789ns}) == SV("2000-01-01 01:02:03.123456789")); assert(stream_ja_JP_locale(file_time{946'688'523'123'456us}) == diff --git a/libcxx/test/std/time/time.clock/time.clock.local/ostream.pass.cpp b/libcxx/test/std/time/time.clock/time.clock.local/ostream.pass.cpp index 9fdef8d5adc78..6ec63a14fbbd3 100644 --- a/libcxx/test/std/time/time.clock/time.clock.local/ostream.pass.cpp +++ b/libcxx/test/std/time/time.clock/time.clock.local/ostream.pass.cpp @@ -64,6 +64,24 @@ template static void test_c() { using namespace std::literals::chrono_literals; + assert(stream_c_locale(std::chrono::local_time{-946'688'523'123'456'789ns}) == + SV("1940-01-01 22:57:56.876543211")); + + assert(stream_c_locale(std::chrono::local_time{-946'688'523'123'456us}) == + SV("1940-01-01 22:57:56.876544")); + + assert(stream_c_locale(std::chrono::local_time{-946'688'523'123ms}) == + SV("1940-01-01 22:57:56.877")); + + assert(stream_c_locale(std::chrono::local_time{-1ns}) == + SV("1969-12-31 23:59:59.999999999")); + + assert(stream_c_locale(std::chrono::local_time{0ns}) == + SV("1970-01-01 00:00:00.000000000")); + + assert(stream_c_locale(std::chrono::local_time{1ns}) == + SV("1970-01-01 00:00:00.000000001")); + assert(stream_c_locale(std::chrono::local_time{946'688'523'123'456'789ns}) == SV("2000-01-01 01:02:03.123456789")); assert(stream_c_locale(std::chrono::local_time{946'688'523'123'456us}) == @@ -97,6 +115,24 @@ template static void test_fr_FR() { using namespace std::literals::chrono_literals; + assert(stream_fr_FR_locale(std::chrono::local_time{-946'688'523'123'456'789ns}) == + SV("1940-01-01 22:57:56,876543211")); + + assert(stream_fr_FR_locale(std::chrono::local_time{-946'688'523'123'456us}) == + SV("1940-01-01 22:57:56,876544")); + + assert(stream_fr_FR_locale(std::chrono::local_time{-946'688'523'123ms}) == + SV("1940-01-01 22:57:56,877")); + + assert(stream_fr_FR_locale(std::chrono::local_time{-1ns}) == + SV("1969-12-31 23:59:59,999999999")); + + assert(stream_fr_FR_locale(std::chrono::local_time{0ns}) == + SV("1970-01-01 00:00:00,000000000")); + + assert(stream_fr_FR_locale(std::chrono::local_time{1ns}) == + SV("1970-01-01 00:00:00,000000001")); + assert(stream_fr_FR_locale(std::chrono::local_time{946'688'523'123'456'789ns}) == SV("2000-01-01 01:02:03,123456789")); assert(stream_fr_FR_locale(std::chrono::local_time{946'688'523'123'456us}) == @@ -131,6 +167,24 @@ template static void test_ja_JP() { using namespace std::literals::chrono_literals; + assert(stream_ja_JP_locale(std::chrono::local_time{-946'688'523'123'456'789ns}) == + SV("1940-01-01 22:57:56.876543211")); + + assert(stream_ja_JP_locale(std::chrono::local_time{-946'688'523'123'456us}) == + SV("1940-01-01 22:57:56.876544")); + + assert(stream_ja_JP_locale(std::chrono::local_time{-946'688'523'123ms}) == + SV("1940-01-01 22:57:56.877")); + + assert(stream_ja_JP_locale(std::chrono::local_time{-1ns}) == + SV("1969-12-31 23:59:59.999999999")); + + assert(stream_ja_JP_locale(std::chrono::local_time{0ns}) == + SV("1970-01-01 00:00:00.000000000")); + + assert(stream_ja_JP_locale(std::chrono::local_time{1ns}) == + SV("1970-01-01 00:00:00.000000001")); + assert(stream_ja_JP_locale(std::chrono::local_time{946'688'523'123'456'789ns}) == SV("2000-01-01 01:02:03.123456789")); assert(stream_ja_JP_locale(std::chrono::local_time{946'688'523'123'456us}) == diff --git a/libcxx/test/std/time/time.clock/time.clock.system/sys_time.ostream.pass.cpp b/libcxx/test/std/time/time.clock/time.clock.system/sys_time.ostream.pass.cpp index 78d8da57c150a..e596ddefde51d 100644 --- a/libcxx/test/std/time/time.clock/time.clock.system/sys_time.ostream.pass.cpp +++ b/libcxx/test/std/time/time.clock/time.clock.system/sys_time.ostream.pass.cpp @@ -64,6 +64,24 @@ template static void test_c() { using namespace std::literals::chrono_literals; + assert(stream_c_locale(std::chrono::sys_time{-946'688'523'123'456'789ns}) == + SV("1940-01-01 22:57:56.876543211")); + + assert(stream_c_locale(std::chrono::sys_time{-946'688'523'123'456us}) == + SV("1940-01-01 22:57:56.876544")); + + assert(stream_c_locale(std::chrono::sys_time{-946'688'523'123ms}) == + SV("1940-01-01 22:57:56.877")); + + assert(stream_c_locale(std::chrono::sys_time{-1ns}) == + SV("1969-12-31 23:59:59.999999999")); + + assert(stream_c_locale(std::chrono::sys_time{0ns}) == + SV("1970-01-01 00:00:00.000000000")); + + assert(stream_c_locale(std::chrono::sys_time{1ns}) == + SV("1970-01-01 00:00:00.000000001")); + assert(stream_c_locale(std::chrono::sys_time{946'688'523'123'456'789ns}) == SV("2000-01-01 01:02:03.123456789")); assert(stream_c_locale(std::chrono::sys_time{946'688'523'123'456us}) == @@ -92,6 +110,24 @@ template static void test_fr_FR() { using namespace std::literals::chrono_literals; + assert(stream_fr_FR_locale(std::chrono::sys_time{-946'688'523'123'456'789ns}) == + SV("1940-01-01 22:57:56,876543211")); + + assert(stream_fr_FR_locale(std::chrono::sys_time{-946'688'523'123'456us}) == + SV("1940-01-01 22:57:56,876544")); + + assert(stream_fr_FR_locale(std::chrono::sys_time{-946'688'523'123ms}) == + SV("1940-01-01 22:57:56,877")); + + assert(stream_fr_FR_locale(std::chrono::sys_time{-1ns}) == + SV("1969-12-31 23:59:59,999999999")); + + assert(stream_fr_FR_locale(std::chrono::sys_time{0ns}) == + SV("1970-01-01 00:00:00,000000000")); + + assert(stream_fr_FR_locale(std::chrono::sys_time{1ns}) == + SV("1970-01-01 00:00:00,000000001")); + assert(stream_fr_FR_locale(std::chrono::sys_time{946'688'523'123'456'789ns}) == SV("2000-01-01 01:02:03,123456789")); assert(stream_fr_FR_locale(std::chrono::sys_time{946'688'523'123'456us}) == @@ -120,6 +156,24 @@ template static void test_ja_JP() { using namespace std::literals::chrono_literals; + assert(stream_ja_JP_locale(std::chrono::sys_time{-946'688'523'123'456'789ns}) == + SV("1940-01-01 22:57:56.876543211")); + + assert(stream_ja_JP_locale(std::chrono::sys_time{-946'688'523'123'456us}) == + SV("1940-01-01 22:57:56.876544")); + + assert(stream_ja_JP_locale(std::chrono::sys_time{-946'688'523'123ms}) == + SV("1940-01-01 22:57:56.877")); + + assert(stream_ja_JP_locale(std::chrono::sys_time{-1ns}) == + SV("1969-12-31 23:59:59.999999999")); + + assert(stream_ja_JP_locale(std::chrono::sys_time{0ns}) == + SV("1970-01-01 00:00:00.000000000")); + + assert(stream_ja_JP_locale(std::chrono::sys_time{1ns}) == + SV("1970-01-01 00:00:00.000000001")); + assert(stream_ja_JP_locale(std::chrono::sys_time{946'688'523'123'456'789ns}) == SV("2000-01-01 01:02:03.123456789")); assert(stream_ja_JP_locale(std::chrono::sys_time{946'688'523'123'456us}) ==