diff --git a/libcxx/benchmarks/CMakeLists.txt b/libcxx/benchmarks/CMakeLists.txt --- a/libcxx/benchmarks/CMakeLists.txt +++ b/libcxx/benchmarks/CMakeLists.txt @@ -145,8 +145,8 @@ RUNTIME_OUTPUT_DIRECTORY "${BENCHMARK_OUTPUT_DIR}" COMPILE_FLAGS "${BENCHMARK_TEST_LIBCXX_COMPILE_FLAGS}" LINK_FLAGS "${BENCHMARK_TEST_LIBCXX_LINK_FLAGS}" - CXX_STANDARD 17 - CXX_STANDARD_REQUIRED YES + CXX_STANDARD 20 + CXX_STANDARD_REQUIRED NO CXX_EXTENSIONS NO) cxx_link_system_libraries(${libcxx_target}) if (LIBCXX_BENCHMARK_NATIVE_STDLIB) @@ -175,8 +175,8 @@ INCLUDE_DIRECTORIES "" COMPILE_FLAGS "${BENCHMARK_TEST_NATIVE_COMPILE_FLAGS}" LINK_FLAGS "${BENCHMARK_TEST_NATIVE_LINK_FLAGS}" - CXX_STANDARD 17 - CXX_STANDARD_REQUIRED YES + CXX_STANDARD 20 + CXX_STANDARD_REQUIRED NO CXX_EXTENSIONS NO) endif() endfunction() diff --git a/libcxx/benchmarks/std_format_spec_string_unicode.bench.cpp b/libcxx/benchmarks/std_format_spec_string_unicode.bench.cpp new file mode 100644 --- /dev/null +++ b/libcxx/benchmarks/std_format_spec_string_unicode.bench.cpp @@ -0,0 +1,225 @@ +// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. +// See https://llvm.org/LICENSE.txt for license information. +// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception +// +//===----------------------------------------------------------------------===// + +// TODO FMT Remove once libc++ requires C++20 support. +#if __cplusplus > 201703L && !defined(_LIBCPP_HAS_NO_UNICODE) + +#include +#include + +#include "benchmark/benchmark.h" + +#include "test_macros.h" + +#ifdef _LIBCPP_HAS_NO_UNICODE +#error The benchmark requires Unicode support enabled. +#endif + +// Always enable asserts since they are used compile-time not run-time +#ifdef NDEBUG +#undef NDEBUG +#endif +#include + +using namespace std::__format_spec; + +template +class tester { + static constexpr size_t size_ = N - 1; + std::array data_; + + constexpr void validate(auto it) const noexcept { + assert(it == end()); + auto result = + __get_format_string_traits(begin(), end(), 1'000'000, 1'000'000); + assert( + result.__last == end() && + static_cast(result.__size) == + __detail::__estimate_column_width(begin(), end(), -1).first && + result.__align); + } + +public: + explicit constexpr tester(const CharT (&input)[N]) { + auto it = data_.begin(); + for (int i = 0; i < 100; ++i) + it = std::copy_n(input, size_, it); + + validate(it); + } + + constexpr size_t size() const noexcept { return data_.size(); } + constexpr const CharT* begin() const noexcept { return data_.begin(); } + constexpr const CharT* end() const noexcept { return data_.end(); } + + void test(benchmark::State& state) const { + for (auto _ : state) + benchmark::DoNotOptimize( + __get_format_string_traits(begin(), end(), 1'000'000, 1'000'000)); + state.SetItemsProcessed(state.iterations() * size()); + } +}; + +#define TEST(u8) \ + if constexpr (std::same_as) { \ + constexpr auto p = tester{u8}; \ + p.test(state); \ + } else if constexpr (std::same_as) { \ + constexpr auto p = tester{TEST_CONCAT(u, u8)}; \ + p.test(state); \ + } else { \ + constexpr auto p = tester{TEST_CONCAT(U, u8)}; \ + p.test(state); \ + } + +template +static void BM_EstimateLengthNoMultiByte(benchmark::State& state) { + TEST("The quick brown fox jumps over the lazy dog"); +} + +template +static void BM_EstimateLengthTwoByteDE(benchmark::State& state) { + static_assert( + sizeof( + "Victor jagt zwölf Boxkämpfer quer über den großen Sylter Deich") == + 67); + + // https://en.wikipedia.org/wiki/Pangram + TEST("Victor jagt zwölf Boxkämpfer quer über den großen Sylter Deich"); +} + +template +static void BM_EstimateLengthTwoBytePL(benchmark::State& state) { + static_assert(sizeof("Stróż pchnął kość w quiz gędźb vel fax myjń") == 53); + + // https://en.wikipedia.org/wiki/Pangram + TEST("Stróż pchnął kość w quiz gędźb vel fax myjń"); +} + +// All values below 1100, which is is the first multibyte sequence +template +static void BM_EstimateLengthThreeByteSingleColumnLow(benchmark::State& state) { + static_assert(sizeof("\u0800\u0801\u0802\u0803\u0804\u0805\u0806\u0807" + "\u0808\u0809\u080a\u080b\u080c\u080d\u080e\u080f") == + 49); + + TEST("\u0800\u0801\u0802\u0803\u0804\u0805\u0806\u0807" + "\u0808\u0809\u080a\u080b\u080c\u080d\u080e\u080f"); +} + +template +static void +BM_EstimateLengthThreeByteSingleColumnHigh(benchmark::State& state) { + static_assert(sizeof("\u1800\u1801\u1802\u1803\u1804\u1805\u1806\u1807" + "\u1808\u1809\u180a\u180b\u180c\u180d\u180e\u180f") == + 49); + + TEST("\u1800\u1801\u1802\u1803\u1804\u1805\u1806\u1807" + "\u1808\u1809\u180a\u180b\u180c\u180d\u180e\u180f"); +} + +template +static void BM_EstimateLengthThreeByteDoubleColumn(benchmark::State& state) { + static_assert(sizeof("\u1100\u0801\u0802\u0803\u0804\u0805\u0806\u0807" + "\u1108\u0809\u080a\u080b\u080c\u080d\u080e\u080f") == + 49); + + TEST("\u1100\u0801\u0802\u0803\u0804\u0805\u0806\u0807" + "\u1108\u0809\u080a\u080b\u080c\u080d\u080e\u080f"); +} + +template +static void BM_EstimateLengthThreeByte(benchmark::State& state) { + static_assert(sizeof("\u1400\u1501\ubbbb\uff00\u0800\u4099\uabcd\u4000" + "\u8ead\ubeef\u1111\u4987\u4321\uffff\u357a\ud50e") == + 49); + + TEST("\u1400\u1501\ubbbb\uff00\u0800\u4099\uabcd\u4000" + "\u8ead\ubeef\u1111\u4987\u4321\uffff\u357a\ud50e"); +} + +template +static void BM_EstimateLengthFourByteSingleColumn(benchmark::State& state) { + static_assert(sizeof("\U00010000\U00010001\U00010002\U00010003" + "\U00010004\U00010005\U00010006\U00010007" + "\U00010008\U00010009\U0001000a\U0001000b" + "\U0001000c\U0001000d\U0001000e\U0001000f") == 65); + + TEST("\U00010000\U00010001\U00010002\U00010003" + "\U00010004\U00010005\U00010006\U00010007" + "\U00010008\U00010009\U0001000a\U0001000b" + "\U0001000c\U0001000d\U0001000e\U0001000f"); +} + +template +static void BM_EstimateLengthFourByteDoubleColumn(benchmark::State& state) { + static_assert(sizeof("\U00020000\U00020002\U00020002\U00020003" + "\U00020004\U00020005\U00020006\U00020007" + "\U00020008\U00020009\U0002000a\U0002000b" + "\U0002000c\U0002000d\U0002000e\U0002000f") == 65); + + TEST("\U00020000\U00020002\U00020002\U00020003" + "\U00020004\U00020005\U00020006\U00020007" + "\U00020008\U00020009\U0002000a\U0002000b" + "\U0002000c\U0002000d\U0002000e\U0002000f"); +} + +template +static void BM_EstimateLengthFourByte(benchmark::State& state) { + static_assert(sizeof("\U00010000\U00010001\U00010002\U00010003" + "\U00020004\U00020005\U00020006\U00020007" + "\U00010008\U00010009\U0001000a\U0001000b" + "\U0002000c\U0002000d\U0002000e\U0002000f") == 65); + + TEST("\U00010000\U00010001\U00010002\U00010003" + "\U00020004\U00020005\U00020006\U00020007" + "\U00010008\U00010009\U0001000a\U0001000b" + "\U0002000c\U0002000d\U0002000e\U0002000f"); +} + +BENCHMARK_TEMPLATE(BM_EstimateLengthNoMultiByte, char); +BENCHMARK_TEMPLATE(BM_EstimateLengthTwoByteDE, char); +BENCHMARK_TEMPLATE(BM_EstimateLengthTwoBytePL, char); +BENCHMARK_TEMPLATE(BM_EstimateLengthThreeByteSingleColumnLow, char); +BENCHMARK_TEMPLATE(BM_EstimateLengthThreeByteSingleColumnHigh, char); +BENCHMARK_TEMPLATE(BM_EstimateLengthThreeByteDoubleColumn, char); +BENCHMARK_TEMPLATE(BM_EstimateLengthThreeByte, char); +BENCHMARK_TEMPLATE(BM_EstimateLengthFourByteSingleColumn, char); +BENCHMARK_TEMPLATE(BM_EstimateLengthFourByteDoubleColumn, char); +BENCHMARK_TEMPLATE(BM_EstimateLengthFourByte, char); + +BENCHMARK_TEMPLATE(BM_EstimateLengthNoMultiByte, char16_t); +BENCHMARK_TEMPLATE(BM_EstimateLengthTwoByteDE, char16_t); +BENCHMARK_TEMPLATE(BM_EstimateLengthTwoBytePL, char16_t); +BENCHMARK_TEMPLATE(BM_EstimateLengthThreeByteSingleColumnLow, char16_t); +BENCHMARK_TEMPLATE(BM_EstimateLengthThreeByteSingleColumnHigh, char16_t); +BENCHMARK_TEMPLATE(BM_EstimateLengthThreeByteDoubleColumn, char16_t); +BENCHMARK_TEMPLATE(BM_EstimateLengthThreeByte, char16_t); +BENCHMARK_TEMPLATE(BM_EstimateLengthFourByteSingleColumn, char16_t); +BENCHMARK_TEMPLATE(BM_EstimateLengthFourByteDoubleColumn, char16_t); +BENCHMARK_TEMPLATE(BM_EstimateLengthFourByte, char16_t); + +BENCHMARK_TEMPLATE(BM_EstimateLengthNoMultiByte, char32_t); +BENCHMARK_TEMPLATE(BM_EstimateLengthTwoByteDE, char32_t); +BENCHMARK_TEMPLATE(BM_EstimateLengthTwoBytePL, char32_t); +BENCHMARK_TEMPLATE(BM_EstimateLengthThreeByteSingleColumnLow, char32_t); +BENCHMARK_TEMPLATE(BM_EstimateLengthThreeByteSingleColumnHigh, char32_t); +BENCHMARK_TEMPLATE(BM_EstimateLengthThreeByteDoubleColumn, char32_t); +BENCHMARK_TEMPLATE(BM_EstimateLengthThreeByte, char32_t); +BENCHMARK_TEMPLATE(BM_EstimateLengthFourByteSingleColumn, char32_t); +BENCHMARK_TEMPLATE(BM_EstimateLengthFourByteDoubleColumn, char32_t); +BENCHMARK_TEMPLATE(BM_EstimateLengthFourByte, char32_t); + +int main(int argc, char** argv) { + benchmark::Initialize(&argc, argv); + if (benchmark::ReportUnrecognizedArguments(argc, argv)) + return 1; + + benchmark::RunSpecifiedBenchmarks(); +} +#else +int main(int, char**) { return 0; } +#endif diff --git a/libcxx/docs/Cxx2aStatusPaperStatus.csv b/libcxx/docs/Cxx2aStatusPaperStatus.csv --- a/libcxx/docs/Cxx2aStatusPaperStatus.csv +++ b/libcxx/docs/Cxx2aStatusPaperStatus.csv @@ -171,7 +171,7 @@ "`P1460 `__","LWG","Mandating the Standard Library: Clause 20 - Utilities library","Prague","* *","" "`P1739 `__","LWG","Avoid template bloat for safe_ranges in combination with ""subrange-y"" view adaptors","Prague","* *","" "`P1831 `__","LWG","Deprecating volatile: library","Prague","* *","" -"`P1868 `__","LWG","width: clarifying units of width and precision in std::format","Prague","* *","" +"`P1868 `__","LWG","width: clarifying units of width and precision in std::format","Prague","|In Progress|","" "`P1908 `__","CWG","Reserving Attribute Namespaces for Future Use","Prague","* *","" "`P1937 `__","CWG","Fixing inconsistencies between constexpr and consteval functions","Prague","* *","" "`P1956 `__","LWG","On the names of low-level bit manipulation functions","Prague","|Complete|","12.0" diff --git a/libcxx/include/__format/parser_std_format_spec.h b/libcxx/include/__format/parser_std_format_spec.h --- a/libcxx/include/__format/parser_std_format_spec.h +++ b/libcxx/include/__format/parser_std_format_spec.h @@ -14,7 +14,9 @@ #include <__debug> #include <__format/format_error.h> #include <__format/format_string.h> +#include #include +#include #if !defined(_LIBCPP_HAS_NO_PRAGMA_SYSTEM_HEADER) #pragma GCC system_header @@ -627,6 +629,444 @@ } }; +/** Helper struct returned from @ref __get_format_string_traits. */ +template +struct _LIBCPP_TEMPLATE_VIS __format_string_traits { + /** Points beyond the last character to write to the output. */ + const _CharT* __last; + /** + * The estimated number of columns in the output or 0. + * + * Only when the output needs to be aligned it's required to know the exact + * nuber of columns in the output. So if the formatted output has only a + * minimum width the exact size isn't important. It's only important to know + * the minimum has been reached. + * + * So if: + * * @ref __align == @c true the @ref __size is the estimated number of + * columns required. + * * @ref __align == @c false the @ref __size is the estimated number of + * columns required or 0 when the estimation algorithm stopped prematurely. + */ + ptrdiff_t __size; + /** + * Does the output need to be aligned. + * + * When alignment is needed the output algorithm needs to add the proper + * padding. Else the output algorithm just needs to copy the input up to + * @ref __last. + */ + bool __align; +}; + +#ifndef _LIBCPP_HAS_NO_UNICODE +namespace __detail { + +/** + * Unicode column width estimates. + * + * Unicode can be store in several formats UTF-8, UTF-16, and UTF-32. Depending + * on format the relation between the number of characters stored and the + * number of monospace columns differs. The first relation is the number of + * characters to a code point. (The text assumes the characters are unsigned.) + * - UTF-8 The number of character is between 1 and 4. The first 127 Unicode + * code points match the ASCII character set. When the highest bit is set it + * means the code point has more than one character. + * - UTF-16: The number of character is between 1 and 2. The range + * [0xd800,0xdfff) indicated two characters are used. + * - UTF-16: The number of character is always 1. + * + * The code point to the number of columns isn't well defined. The code uses the + * estimation as defined in [format.string.std]/11. This list may change in the + * future. + * + * The algorithm of @ref __get_format_string_traits uses two different scanners: + * - The simple scanner @ref __estimate_column_width_fast. This scanner assumes + * 1 character is 1 column. This scanner stops when it can't be sure the + * assumption is valid: + * - UTF-8 when the code point is encoded in more than 1 character. + * - UTF-16 and UTF-32 when the first multi-column character is encountered. + * (The character's value is lower than 0xd800 so the 2 character encoding + * is irrelevant for this scanner.) + * Due to these assumptions the scanner is faster than the full scanner. It + * can process all text only containing ASCII. For UTF-16/32 it can process + * most (all?) European languages. (Note the set it can process might be + * reduced in the future, due to updates in the scanning rules.) + * - The full scanner @ref __estimate_column_width. This scanner, if needed, + * turn multiple characters into one code point then converts the code point + * to a column width. + * + * See also: + * - [format.string.general]/11 + * - https://en.wikipedia.org/wiki/UTF-8#Encoding + * - https://en.wikipedia.org/wiki/UTF-16#U+D800_to_U+DFFF + */ + +/** + * The first 2 column code point. + * + * This is the point where the fast UTF-16/32 scanner needs to stop processing. + */ +inline constexpr uint32_t __two_column_code_point = 0x1100; + +/** Helper concept for an UTF-8 character type. */ +template +concept __utf8 = same_as<_CharT, char> || same_as<_CharT, char8_t>; + +/** Helper concept for an UTF-16 character type. */ +template +concept __utf16 = (same_as<_CharT, wchar_t> && sizeof(wchar_t) == 2) || + same_as<_CharT, char16_t>; + +/** Helper concept for an UTF-32 character type. */ +template +concept __utf32 = (same_as<_CharT, wchar_t> && sizeof(wchar_t) == 4) || + same_as<_CharT, char32_t>; + +/** Helper concept for an UTF-16 or UTF-32 character type. */ +template +concept __utf16_32 = __utf16<_CharT> || __utf32<_CharT>; + +/** + * Converts a code point to the column width. + * + * The estimations are conforming to [format.string.general]/11 + * + * This version expects a value less than 0x1'0000, which is a 3-byte UTF-8 + * character. + */ +[[nodiscard]] _LIBCPP_INLINE_VISIBILITY inline constexpr int +__column_width_3(uint32_t __c) noexcept { + _LIBCPP_ASSERT(__c < 0x1'0000, + "Use __column_width_4 or __column_width for larger values"); + return 1 + (__c >= 0x1100 && + (__c <= 0x115f || + (__c >= 0x2100 && + (__c <= 0x232a || + (__c >= 0x2e80 && + (__c <= 0x303e || + (__c >= 0x3040 && + (__c <= 0xa4cf || + (__c >= 0x2e80 && + (__c <= 0x303e || + (__c >= 0x3040 && + (__c <= 0xa4cf || + (__c >= 0xac00 && + (__c <= 0xd7a3 || + (__c >= 0xf900 && + (__c <= 0xfaff || + (__c >= 0xfe10 && + (__c <= 0xfe19 || + (__c >= 0xfe30 && + (__c <= 0xfe6f || + (__c >= 0xff00 && + (__c <= 0xff60 || + (__c >= 0xffe0 && + (__c <= 0xffe6)))))))))))))))))))))))); +} + +/** + * @overload + * + * This version expects a value greater than or equal to 0x1'0000, which is a + * 4-byte UTF-8 character. + */ +[[nodiscard]] _LIBCPP_INLINE_VISIBILITY inline constexpr int +__column_width_4(uint32_t __c) noexcept { + _LIBCPP_ASSERT(__c >= 0x1'0000, + "Use __column_width_3 or __column_width for smaller values"); + return 1 + (__c >= 0x1f300 && + (__c <= 0x1f64f || + (__c >= 0x1f900 && + (__c <= 0x1f9ff || + (__c >= 0x20000 && + (__c <= 0x2fffd || (__c >= 0x30000 && (__c <= 0x3fffd)))))))); +} + +/** + * @overload + * + * The general case, accepting all values. + */ +[[nodiscard]] _LIBCPP_INLINE_VISIBILITY inline constexpr int +__column_width(uint32_t __c) noexcept { + if (__c < 0x1'0000) + return __column_width_3(__c); + + return __column_width_4(__c); +} + +/** + * Estimate the column width for the UTF-8 sequence using the fast algorithm. + */ +template <__utf8 _CharT> +[[nodiscard]] _LIBCPP_INLINE_VISIBILITY constexpr const _CharT* +__estimate_column_width_fast(const _CharT* __first, + const _CharT* __last) noexcept { + return _VSTD::find_if(__first, __last, + [](unsigned char __c) { return __c & 0x80; }); +} + +/** + * @overload + * + * The implementation for UTF-16/32. + */ +template <__utf16_32 _CharT> +[[nodiscard]] _LIBCPP_INLINE_VISIBILITY constexpr const _CharT* +__estimate_column_width_fast(const _CharT* __first, + const _CharT* __last) noexcept { + return _VSTD::find_if(__first, __last, + [](uint32_t __c) { return __c >= 0x1100; }); +} + +/** + * Determines the number of output columns are needed to render the input. + * + * The number of columns will never return more than @a __threshold. + * + * @note The scanner will stop when it encounters malformed Unicode. Since our + * scanner isn't a full blown Unicode parser it stops instead of throwing an + * exception. + * + * @tparam __truncate When the last parsed code point is a multi-column output + * it may exceed the @a __threshold by one. The behaviour + * depends on the use case: + * - @c true The scanner is used for the precision. The + * @a __threshold may not be exceeded and current column is + * truncated. The returned size is less than the @a + * __threshold. + * - @c false The scanner is used for the width. The + * @a __threshold is no limit, but an optimization + * termination point. Truncating would give the wrong + * result. In this case the returned size exceeds the + * @a __threshold. + * + * @param __first Points to the first element of the input range. + * @param __last Points beyond the last element of the input range. + * @param __threshold The limit of the number of output columns. The exact + * handling of the @a __threshold is determined by + * @a __truncate. + * + * @returns A pair: + * - Number of columns. + * - Last parsed element. This limits the original output to fit in the wanted + * maximum number of columns. + */ +template +[[nodiscard]] _LIBCPP_INLINE_VISIBILITY constexpr pair +__estimate_column_width(const _CharT* __first, const _CharT* __last, + size_t __threshold) noexcept { + size_t __result = 0; + + while (__first != __last) { + // Based on the number of leading 1 characters the number of bytes in the + // code point can be determined. See + // https://en.wikipedia.org/wiki/UTF-8#Encoding + switch (_VSTD::countl_one(static_cast(*__first))) { + case 0: // 1-byte encoding: all 1 column + ++__result; + ++__first; + break; + + case 2: // 2-byte encoding: all 1 column + // Malformed Unicode, discard last part. + if (__last - __first < 2) [[unlikely]] + return {__result, __first}; + __first += 2; + ++__result; + break; + + case 3: // 3-byte encoding: either 1 or 2 columns + // Malformed Unicode, discard last part. + if (__last - __first < 3) [[unlikely]] + return {__result, __first}; + { + uint32_t __c = static_cast(*__first++) & 0x0f; + __c <<= 6; + __c |= static_cast(*__first++) & 0x3f; + __c <<= 6; + __c |= static_cast(*__first++) & 0x3f; + __result += __column_width_3(__c); + if constexpr (__truncate) + if (__result > __threshold) + return {__result - 2, __first - 3}; + } + break; + case 4: // 4-byte encoding: either 1 or 2 columns + // Malformed Unicode, discard last part. + if (__last - __first < 4) [[unlikely]] + return {__result, __first}; + { + uint32_t __c = static_cast(*__first++) & 0x07; + __c <<= 6; + __c |= static_cast(*__first++) & 0x3f; + __c <<= 6; + __c |= static_cast(*__first++) & 0x3f; + __c <<= 6; + __c |= static_cast(*__first++) & 0x3f; + __result += __column_width_4(__c); + if constexpr (__truncate) + if (__result > __threshold) + return {__result - 2, __first - 4}; + } + break; + default: + // Malformed Unicode, discard last part. + return {__result, __first}; + } + + if (__result >= __threshold) + return {__result, __first}; + } + return {__result, __first}; +} + +template +[[nodiscard]] _LIBCPP_INLINE_VISIBILITY constexpr pair +__estimate_column_width(const _CharT* __first, const _CharT* __last, + size_t __threshold) noexcept { + size_t __result = 0; + + while (__first != __last) { + uint32_t __c = *__first; + // Is the character part of a surrogate pair? See + // https://en.wikipedia.org/wiki/UTF-16#U+D800_to_U+DFFF + if (__c >= 0xd800 && __c <= 0xDfff) { + // Malformed Unicode, discard last part. + if (__last - __first < 2) [[unlikely]] + return {__result, __first}; + + __c -= 0xd800; + __c <<= 10; + __c += (*(__first + 1) - 0xdc00); + __c += 0x10'000; + + __result += __column_width_4(__c); + if constexpr (__truncate) + if (__result > __threshold) + return {__result - 2, __first}; + __first += 2; + } else { + __result += __column_width_3(__c); + if constexpr (__truncate) + if (__result > __threshold) + return {__result - 2, __first}; + ++__first; + } + + if (__result >= __threshold) + return {__result, __first}; + } + + return {__result, __first}; +} + +template +[[nodiscard]] _LIBCPP_INLINE_VISIBILITY constexpr pair +__estimate_column_width(const _CharT* __first, const _CharT* __last, + size_t __threshold) noexcept { + size_t __result = 0; + + while (__first != __last) { + wchar_t __c = *__first; + __result += __column_width(__c); + + if constexpr (__truncate) + if (__result > __threshold) + return {__result - 2, __first}; + + ++__first; + if (__result >= __threshold) + return {__result, __first}; + } + + return {__result, __first}; +} + +} // namespace __detail + +template +[[nodiscard]] _LIBCPP_INLINE_VISIBILITY constexpr __format_string_traits<_CharT> +__get_format_string_traits(const _CharT* __first, const _CharT* __last, + ptrdiff_t __width, ptrdiff_t __precision) noexcept { + + _LIBCPP_ASSERT(__width != 0 || __precision != __format::__number_max, + "The function has no effect and shouldn't be used"); + _LIBCPP_ASSERT(__width <= __precision, + "The caller should validate this invariant"); + + // TODO FMT There might be more optimizations possible: + // If __precision == __format::__number_max and the encoding is: + // * UTF-8 : 4 * (__last - __first) >= __width + // * UTF-16 : 2 * (__last - __first) >= __width + // * UTF-32 : (__last - __first) >= __width + // In these cases it's certain the output is at least the requested width. + // It's unknown how often this happens in practice. For now the improvement + // isn't implemented. + + /* + * First assume there are no special Unicode characters in the input. + * - Apply the precision (this may reduce the size of the input). + * - Scan for special characters in the input. + * If our assumption was correct the __pos will be at the end of the input. + */ + const _CharT* __limit = __first + _VSTD::min(__last - __first, __precision); + ptrdiff_t __size = __limit - __first; + const _CharT* __pos = + __detail::__estimate_column_width_fast(__first, __limit); + + if (__pos == __limit) + return {__limit, __size, __size < __width}; + + /* + * Our assumption was wrong, there are special Unicode characters. + * The range [__first, __pos) contains a set of characters with the + * following property: + * Every _CharT in the range will be rendered in 1 column. + * + * If there's no maximum width and the parsed size already exceeds the + * minimum required width. The real size isn't important. So bail out. + */ + if (__precision == __format::__number_max && (__pos - __first) >= __width) + return {__last, 0, false}; + + /* If there's a __precision, truncate the output to that width. */ + ptrdiff_t __prefix = __pos - __first; + if (__precision != __format::__number_max) { + _LIBCPP_ASSERT(__precision > __prefix, "Logic error."); + auto __lengh_info = __detail::__estimate_column_width( + __pos, __last, __precision - __prefix); + __size = __lengh_info.first + __prefix; + return {__lengh_info.second, __size, __size < __width}; + } + + /* Else use __width to determine the number of required padding characters. */ + _LIBCPP_ASSERT(__width > __prefix, "Logic error."); + auto __lengh_info = __detail::__estimate_column_width( + __pos, __last, __width - __prefix); + if (__lengh_info.second != __last) { + // Consumed the width number of characters. The exact size of the string + // is unknown. We only know we don't need to align the output. + _LIBCPP_ASSERT(static_cast(__lengh_info.first + __prefix) >= + __width, + "Logic error"); + return {__last, 0, false}; + } + + __size = __lengh_info.first + __prefix; + return {__last, __size, __size < __width}; +} +#else // _LIBCPP_HAS_NO_UNICODE +template +[[nodiscard]] _LIBCPP_INLINE_VISIBILITY constexpr __format_string_traits<_CharT> +__get_format_string_traits(const _CharT* __first, const _CharT* __last, + ptrdiff_t __width, ptrdiff_t __precision) { + ptrdiff_t __size = _VSTD::min(__last - __first, __precision); + return {__first + __size, __size, __size < __width}; +} +#endif // _LIBCPP_HAS_NO_UNICODE + } // namespace __format_spec #endif // !defined(_LIBCPP_HAS_NO_CONCEPTS) && !defined(_LIBCPP_HAS_NO_BUILTIN_IS_CONSTANT_EVALUATED) diff --git a/libcxx/test/std/utilities/format/format.string/format.string.std/std_format_spec_string_unicode.pass.cpp b/libcxx/test/std/utilities/format/format.string/format.string.std/std_format_spec_string_unicode.pass.cpp new file mode 100644 --- /dev/null +++ b/libcxx/test/std/utilities/format/format.string/format.string.std/std_format_spec_string_unicode.pass.cpp @@ -0,0 +1,304 @@ +//===----------------------------------------------------------------------===// +// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. +// See https://llvm.org/LICENSE.txt for license information. +// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception +// +//===----------------------------------------------------------------------===// + +// UNSUPPORTED: c++03, c++11, c++14, c++17 +// UNSUPPORTED: libcpp-no-concepts + +// + +// Tests the Unicode width support of the standard format specifiers. +// It tests [format.string.std]/8 - 11: +// - Properly determining the estimated with of a unicode string. +// - Properly truncating to the wanted maximum width. + +#include +#include + +#include "test_macros.h" +#include "make_string.h" + +#define CSTR(S) MAKE_CSTRING(CharT, S) + +using namespace std::__format_spec; + +template +constexpr bool operator==(const __format_string_traits& lhs, + const __format_string_traits& rhs) noexcept { + return lhs.__last == rhs.__last && lhs.__size == rhs.__size && + lhs.__align == rhs.__align; +} + +template +constexpr void get_format_string_traits(size_t offset, ptrdiff_t size, + bool align, const CharT* str, + size_t width, size_t precision) { + std::basic_string_view sv{str}; + __format_string_traits expected{sv.begin() + offset, size, align}; + __format_string_traits traits = + __get_format_string_traits(sv.begin(), sv.end(), width, precision); + assert(traits == expected); +} + +#ifndef _LIBCPP_HAS_NO_UNICODE +template +constexpr void estimate_column_width_fast(size_t expected, const CharT* str) { + std::basic_string_view sv{str}; + const CharT* out = + __detail::__estimate_column_width_fast(sv.begin(), sv.end()); + assert(out == sv.begin() + expected); +} + +template +constexpr void estimate_column_width_fast() { + + // No unicode + estimate_column_width_fast(3, CSTR("abc")); + estimate_column_width_fast(3, CSTR("a\u007fc")); + + if constexpr (sizeof(CharT) == 1) { + // UTF-8 stop at the first multi-byte character. + estimate_column_width_fast(0, CSTR("\u0080bc")); + estimate_column_width_fast(1, CSTR("a\u0080c")); + estimate_column_width_fast(2, CSTR("ab\u0080")); + estimate_column_width_fast(1, CSTR("aßc")); + + estimate_column_width_fast(1, CSTR("a\u07ffc")); + estimate_column_width_fast(1, CSTR("a\u0800c")); + + estimate_column_width_fast(1, CSTR("a\u10ffc")); + } else { + // UTF-16/32 stop at the first multi-column character. + estimate_column_width_fast(3, CSTR("\u0080bc")); + estimate_column_width_fast(3, CSTR("a\u0080c")); + estimate_column_width_fast(3, CSTR("ab\u0080")); + estimate_column_width_fast(3, CSTR("aßc")); + + estimate_column_width_fast(3, CSTR("a\u07ffc")); + estimate_column_width_fast(3, CSTR("a\u0800c")); + + estimate_column_width_fast(3, CSTR("a\u10ffc")); + } + // First 2-column character + estimate_column_width_fast(1, CSTR("a\u1100c")); + + estimate_column_width_fast(1, CSTR("a\U0000ffffc")); + estimate_column_width_fast(1, CSTR("a\U00010000c")); + estimate_column_width_fast(1, CSTR("a\U0010FFFFc")); +} + +template +constexpr void estimate_column_width(size_t expected, const CharT* str) { + std::basic_string_view sv{str}; + std::pair length_info = + __detail::__estimate_column_width(sv.begin(), sv.end(), -1); + assert(length_info.first == expected); +} + +template +constexpr void estimate_column_width() { + //*** 1-byte code points *** + estimate_column_width(1, CSTR(" ")); + estimate_column_width(1, CSTR("~")); + + //*** 2-byte code points *** + estimate_column_width(1, CSTR("\u00a1")); // INVERTED EXCLAMATION MARK + estimate_column_width(1, CSTR("\u07ff")); // NKO TAMAN SIGN + + //*** 3-byte code points *** + estimate_column_width(1, CSTR("\u0800")); // SAMARITAN LETTER ALAF + estimate_column_width(1, CSTR("\ufffd")); // REPLACEMENT CHARACTER + + // 2 column ranges + estimate_column_width(2, CSTR("\u1100")); // HANGUL CHOSEONG KIYEOK + estimate_column_width(2, CSTR("\u115f")); // HANGUL CHOSEONG FILLER + + estimate_column_width(2, CSTR("\u2329")); // LEFT-POINTING ANGLE BRACKET + estimate_column_width(2, CSTR("\u232a")); // RIGHT-POINTING ANGLE BRACKET + + estimate_column_width(2, CSTR("\u2e80")); // CJK RADICAL REPEAT + estimate_column_width(2, CSTR("\u303e")); // IDEOGRAPHIC VARIATION INDICATOR + + estimate_column_width(2, CSTR("\u3040")); // U+3041 HIRAGANA LETTER SMALL A + estimate_column_width(2, CSTR("\ua4cf")); // U+A4D0 LISU LETTER BA + + estimate_column_width(2, CSTR("\uac00")); // + estimate_column_width(2, CSTR("\ud7a3")); // Hangul Syllable Hih + + estimate_column_width(2, CSTR("\uf900")); // CJK COMPATIBILITY IDEOGRAPH-F900 + estimate_column_width(2, CSTR("\ufaff")); // U+FB00 LATIN SMALL LIGATURE FF + + estimate_column_width(2, + CSTR("\ufe10")); // PRESENTATION FORM FOR VERTICAL COMMA + estimate_column_width( + 2, CSTR("\ufe19")); // PRESENTATION FORM FOR VERTICAL HORIZONTAL ELLIPSIS + + estimate_column_width( + 2, CSTR("\ufe30")); // PRESENTATION FORM FOR VERTICAL TWO DOT LEADER + estimate_column_width(2, + CSTR("\ufe6f")); // U+FE70 ARABIC FATHATAN ISOLATED FORM + + estimate_column_width(2, CSTR("\uff00")); // U+FF01 FULLWIDTH EXCLAMATION MARK + estimate_column_width(2, CSTR("\uff60")); // FULLWIDTH RIGHT WHITE PARENTHESIS + + estimate_column_width(2, CSTR("\uffe0")); // FULLWIDTH CENT SIGN + estimate_column_width(2, CSTR("\uffe6")); // FULLWIDTH WON SIGN + + //*** 4-byte code points *** + estimate_column_width(1, CSTR("\U00010000")); // LINEAR B SYLLABLE B008 A + estimate_column_width(1, CSTR("\U0010FFFF")); // Undefined Character + + // 2 column ranges + estimate_column_width(2, CSTR("\U0001f300")); // CYCLONE + estimate_column_width(2, CSTR("\U0001f64f")); // PERSON WITH FOLDED HANDS + estimate_column_width( + 2, CSTR("\U0001f900")); // CIRCLED CROSS FORMEE WITH FOUR DOTS + estimate_column_width(2, CSTR("\U0001f9ff")); // NAZAR AMULET + estimate_column_width( + 2, CSTR("\U00020000")); // + estimate_column_width(2, CSTR("\U0002fffd")); // Undefined Character + estimate_column_width( + 2, CSTR("\U00030000")); // + estimate_column_width(2, CSTR("\U0003fffd")); // Undefined Character +} + +template +constexpr void get_format_string_traits() { + // Truncate the input. + get_format_string_traits(2, 2, false, CSTR("abc"), 0, 2); + + // The 2-column character gets entirely rejected. + get_format_string_traits(1, 1, false, CSTR("a\u115f"), 0, 2); + + // Due to the requested width extra alignment is required. + get_format_string_traits(1, 1, true, CSTR("a\u115f"), 2, 2); + + // Same but for a 2-column 4-byte UTF-8 sequence + get_format_string_traits(1, 1, false, CSTR("a\U0001f300"), 0, 2); + get_format_string_traits(1, 1, true, CSTR("a\U0001f300"), 2, 2); + + // No alignment required. + get_format_string_traits(3, 3, false, CSTR("abc"), 2, + std::__format::__number_max); + get_format_string_traits(3, 3, false, CSTR("abc"), 3, + std::__format::__number_max); + + // Special case, we have a special character already parsed and have enough + // withd to satisfy the minumum required width. + get_format_string_traits(3 + 2 * (sizeof(CharT) == 1), 0, false, + CSTR("ab\u1111"), 2, std::__format::__number_max); + + // Doesn't evaluate 'c' so size -> 0 + get_format_string_traits(3 + 2 * (sizeof(CharT) == 1), 0, false, + CSTR("a\u115fc") /* 2-column character */, 3, + std::__format::__number_max); + // Evaluates all so size ->4 + get_format_string_traits(3 + 2 * (sizeof(CharT) == 1), 4, false, + CSTR("a\u115fc") /* 2-column character */, 4, + std::__format::__number_max); + + // Extend width + get_format_string_traits(3, 3, true, CSTR("abc"), 4, + std::__format::__number_max); + get_format_string_traits(3 + 2 * (sizeof(CharT) == 1), 3, true, + CSTR("a\u1160c") /* 1-column character */, 4, + std::__format::__number_max); + + // In this case the threshold where the width is still determined. + get_format_string_traits(2 + 2 * (sizeof(CharT) == 1), 3, false, + CSTR("i\u1110"), 2, std::__format::__number_max); + + // The width is no longer exactly determined. + get_format_string_traits(2 + 2 * (sizeof(CharT) == 1), 0, false, + CSTR("i\u1110"), 1, std::__format::__number_max); + + if constexpr (sizeof(CharT) == 1) { + // Corrupt UTF-8 sequence. + get_format_string_traits(1, 1, false, CSTR("a\xc0"), 0, 3); + get_format_string_traits(1, 1, false, CSTR("a\xe0"), 0, 3); + get_format_string_traits(1, 1, false, CSTR("a\xf0"), 0, 3); + } else if constexpr (sizeof(CharT) == 2) { + // Corrupt UTF-16 sequence. + if constexpr (std::same_as) + get_format_string_traits(1, 1, false, u"a\xdddd", 0, 3); + else + // Corrupt UTF-16 wchar_t seqence. + get_format_string_traits(1, 1, false, L"a\xdddd", 0, 3); + } + // UTF-32 doesn't combine characters, thus no corruption tests. +} + +template +constexpr void test() { + estimate_column_width_fast(); + estimate_column_width(); + get_format_string_traits(); +} +#else // _LIBCPP_HAS_NO_UNICODE +template +constexpr void get_format_string_traits() { + // Truncate the input. + get_format_string_traits(2, 2, false, CSTR("abc"), 0, 2); + + // The 2-column character gets half accepted. + get_format_string_traits(2, 2, false, CSTR("a\u115f"), 0, 2); + + // No alignment since the number of characters fits. + get_format_string_traits(2, 2, false, CSTR("a\u115f"), 2, 2); + + // Same but for a 2-column 4-byte UTF-8 sequence + get_format_string_traits(2, 2, false, CSTR("a\U0001f300"), 0, 2); + get_format_string_traits(2, 2, false, CSTR("a\U0001f300"), 2, 2); + + // No alignment required. + get_format_string_traits(3, 3, false, CSTR("abc"), 2, + std::__format::__number_max); + get_format_string_traits(3, 3, false, CSTR("abc"), 3, + std::__format::__number_max); + + get_format_string_traits(3 + 2 * (sizeof(CharT) == 1), + 3 + 2 * (sizeof(CharT) == 1), false, + CSTR("ab\u1111"), 2, std::__format::__number_max); + + // Doesn't evaluate 'c' so size -> 0 + get_format_string_traits(3 + 2 * (sizeof(CharT) == 1), + 3 + 2 * (sizeof(CharT) == 1), false, + CSTR("a\u115fc") /* 2-column character */, 3, + std::__format::__number_max); + // Extend width + get_format_string_traits(3, 3, true, CSTR("abc"), 4, + std::__format::__number_max); + get_format_string_traits(3 + 2 * (sizeof(CharT) == 1), + 3 + 2 * (sizeof(CharT) == 1), true, + CSTR("a\u1160c") /* 1-column character */, 6, + std::__format::__number_max); +} + +template +constexpr void test() { + get_format_string_traits(); +} +#endif // _LIBCPP_HAS_NO_UNICODE + +constexpr bool test() { + test(); + test(); +#ifndef _LIBCPP_HAS_NO_CHAR8_T + test(); +#endif +#ifndef _LIBCPP_HAS_NO_UNICODE_CHARS + test(); + test(); +#endif + return true; +} + +int main(int, char**) { + test(); + static_assert(test()); + + return 0; +}