1e5d2d3eaSMark de Wever //===----------------------------------------------------------------------===// 2*6a54dfbfSLouis Dionne // 3e5d2d3eaSMark de Wever // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. 4e5d2d3eaSMark de Wever // See https://llvm.org/LICENSE.txt for license information. 5e5d2d3eaSMark de Wever // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception 6e5d2d3eaSMark de Wever // 7e5d2d3eaSMark de Wever //===----------------------------------------------------------------------===// 8e5d2d3eaSMark de Wever 9e5d2d3eaSMark de Wever // UNSUPPORTED: c++03, c++11, c++14, c++17 10d529e811SLouis Dionne // UNSUPPORTED: no-localization 11520c7fbbSLouis Dionne // UNSUPPORTED: GCC-ALWAYS_INLINE-FIXME 12e5d2d3eaSMark de Wever 133d334df5SLouis Dionne // TODO FMT This test should not require std::to_chars(floating-point) 14f0fc8c48SLouis Dionne // XFAIL: availability-fp_to_chars-missing 153d334df5SLouis Dionne 161403080aSMark de Wever // TODO FMT Investigate Windows issues. 171403080aSMark de Wever // XFAIL: msvc 181403080aSMark de Wever 19e5d2d3eaSMark de Wever // REQUIRES: locale.fr_FR.UTF-8 20e5d2d3eaSMark de Wever // REQUIRES: locale.ja_JP.UTF-8 21e5d2d3eaSMark de Wever 22e5d2d3eaSMark de Wever // <chrono> 23e5d2d3eaSMark de Wever 24e5d2d3eaSMark de Wever // template<class charT> struct formatter<chrono::day, charT>; 25e5d2d3eaSMark de Wever 26e5d2d3eaSMark de Wever #include <chrono> 27e5d2d3eaSMark de Wever #include <format> 28e5d2d3eaSMark de Wever 29e5d2d3eaSMark de Wever #include <cassert> 30e5d2d3eaSMark de Wever #include <concepts> 31e5d2d3eaSMark de Wever #include <locale> 32e5d2d3eaSMark de Wever #include <iostream> 33e5d2d3eaSMark de Wever #include <type_traits> 34e5d2d3eaSMark de Wever 35e5d2d3eaSMark de Wever #include "formatter_tests.h" 36e5d2d3eaSMark de Wever #include "make_string.h" 37e5d2d3eaSMark de Wever #include "platform_support.h" // locale name macros 38e5d2d3eaSMark de Wever #include "string_literal.h" 39e5d2d3eaSMark de Wever #include "test_macros.h" 40e5d2d3eaSMark de Wever 41e5d2d3eaSMark de Wever template <class CharT> 42e5d2d3eaSMark de Wever static void test_no_chrono_specs() { 43e5d2d3eaSMark de Wever using namespace std::literals::chrono_literals; 44e5d2d3eaSMark de Wever 45e5d2d3eaSMark de Wever // Valid day 46d23f609dSMark de Wever check(SV("01"), SV("{}"), 1d); 47d23f609dSMark de Wever check(SV("*01*"), SV("{:*^4}"), 1d); 48d23f609dSMark de Wever check(SV("*01"), SV("{:*>3}"), 1d); 49e5d2d3eaSMark de Wever 50e5d2d3eaSMark de Wever // Invalid day 51d23f609dSMark de Wever check(SV("00 is not a valid day"), SV("{}"), 0d); 52d23f609dSMark de Wever check(SV("*00 is not a valid day*"), SV("{:*^23}"), 0d); 53e5d2d3eaSMark de Wever } 54e5d2d3eaSMark de Wever 55e5d2d3eaSMark de Wever template <class CharT> 56e5d2d3eaSMark de Wever static void test_valid_values() { 57e5d2d3eaSMark de Wever using namespace std::literals::chrono_literals; 58e5d2d3eaSMark de Wever 59d23f609dSMark de Wever constexpr std::basic_string_view<CharT> fmt = SV("{:%%d='%d'%t%%Od='%Od'%t%%e='%e'%t%%Oe='%Oe'%n}"); 60d23f609dSMark de Wever constexpr std::basic_string_view<CharT> lfmt = SV("{:L%%d='%d'%t%%Od='%Od'%t%%e='%e'%t%%Oe='%Oe'%n}"); 61e5d2d3eaSMark de Wever 62e5d2d3eaSMark de Wever const std::locale loc(LOCALE_ja_JP_UTF_8); 63e5d2d3eaSMark de Wever std::locale::global(std::locale(LOCALE_fr_FR_UTF_8)); 64e5d2d3eaSMark de Wever 65e5d2d3eaSMark de Wever // Non localized output using C-locale 661403080aSMark de Wever #if defined(_WIN32) 671403080aSMark de Wever check(SV("%d=''\t%Od=''\t%e=''\t%Oe=''\n"), fmt, 0d); 681403080aSMark de Wever #else 69d23f609dSMark de Wever check(SV("%d='00'\t%Od='00'\t%e=' 0'\t%Oe=' 0'\n"), fmt, 0d); 701403080aSMark de Wever #endif 71d23f609dSMark de Wever check(SV("%d='01'\t%Od='01'\t%e=' 1'\t%Oe=' 1'\n"), fmt, 1d); 72d23f609dSMark de Wever check(SV("%d='31'\t%Od='31'\t%e='31'\t%Oe='31'\n"), fmt, 31d); 731403080aSMark de Wever #if defined(_WIN32) 741403080aSMark de Wever check(SV("%d=''\t%Od=''\t%e=''\t%Oe=''\n"), fmt, 0d); 751403080aSMark de Wever #elif defined(_AIX) 76d23f609dSMark de Wever check(SV("%d='55'\t%Od='55'\t%e='55'\t%Oe='55'\n"), fmt, 255d); 77e5d2d3eaSMark de Wever #else 78d23f609dSMark de Wever check(SV("%d='255'\t%Od='255'\t%e='255'\t%Oe='255'\n"), fmt, 255d); 79e5d2d3eaSMark de Wever #endif 80e5d2d3eaSMark de Wever 81e5d2d3eaSMark de Wever // Use the global locale (fr_FR) 821403080aSMark de Wever #if defined(_WIN32) 831403080aSMark de Wever check(SV("%d=''\t%Od=''\t%e=''\t%Oe=''\n"), lfmt, 0d); 841403080aSMark de Wever #else 85d23f609dSMark de Wever check(SV("%d='00'\t%Od='00'\t%e=' 0'\t%Oe=' 0'\n"), lfmt, 0d); 861403080aSMark de Wever #endif 87d23f609dSMark de Wever check(SV("%d='01'\t%Od='01'\t%e=' 1'\t%Oe=' 1'\n"), lfmt, 1d); 88d23f609dSMark de Wever check(SV("%d='31'\t%Od='31'\t%e='31'\t%Oe='31'\n"), lfmt, 31d); 891403080aSMark de Wever #if defined(_WIN32) 901403080aSMark de Wever check(SV("%d=''\t%Od=''\t%e=''\t%Oe=''\n"), lfmt, 255d); 911403080aSMark de Wever #elif defined(_AIX) 92d23f609dSMark de Wever check(SV("%d='55'\t%Od='55'\t%e='55'\t%Oe='55'\n"), lfmt, 255d); 93e5d2d3eaSMark de Wever #else 94d23f609dSMark de Wever check(SV("%d='255'\t%Od='255'\t%e='255'\t%Oe='255'\n"), lfmt, 255d); 95e5d2d3eaSMark de Wever #endif 96e5d2d3eaSMark de Wever 97e5d2d3eaSMark de Wever // Use supplied locale (ja_JP). This locale has a different alternate on some platforms. 988f01029bSMark de Wever #if defined(__APPLE__) || defined(_AIX) || defined(_WIN32) || defined(__FreeBSD__) 991403080aSMark de Wever # if defined(_WIN32) 1001403080aSMark de Wever check(loc, SV("%d=''\t%Od=''\t%e=''\t%Oe=''\n"), lfmt, 0d); 1011403080aSMark de Wever # else 102d23f609dSMark de Wever check(loc, SV("%d='00'\t%Od='00'\t%e=' 0'\t%Oe=' 0'\n"), lfmt, 0d); 1031403080aSMark de Wever # endif 104d23f609dSMark de Wever check(loc, SV("%d='01'\t%Od='01'\t%e=' 1'\t%Oe=' 1'\n"), lfmt, 1d); 105d23f609dSMark de Wever check(loc, SV("%d='31'\t%Od='31'\t%e='31'\t%Oe='31'\n"), lfmt, 31d); 1061403080aSMark de Wever # if defined(_WIN32) 1071403080aSMark de Wever check(SV("%d=''\t%Od=''\t%e=''\t%Oe=''\n"), fmt, 255d); 1081403080aSMark de Wever # elif defined(_AIX) 109d23f609dSMark de Wever check(SV("%d='55'\t%Od='55'\t%e='55'\t%Oe='55'\n"), fmt, 255d); 110e5d2d3eaSMark de Wever # else 111d23f609dSMark de Wever check(SV("%d='255'\t%Od='255'\t%e='255'\t%Oe='255'\n"), fmt, 255d); 112e5d2d3eaSMark de Wever # endif 1138f01029bSMark de Wever #else // defined(__APPLE__) || defined(_AIX) || defined(_WIN32) || defined(__FreeBSD__) 114d23f609dSMark de Wever check(loc, SV("%d='00'\t%Od='〇'\t%e=' 0'\t%Oe='〇'\n"), lfmt, 0d); 115d23f609dSMark de Wever check(loc, SV("%d='01'\t%Od='一'\t%e=' 1'\t%Oe='一'\n"), lfmt, 1d); 116d23f609dSMark de Wever check(loc, SV("%d='31'\t%Od='三十一'\t%e='31'\t%Oe='三十一'\n"), lfmt, 31d); 117d23f609dSMark de Wever check(loc, SV("%d='255'\t%Od='255'\t%e='255'\t%Oe='255'\n"), lfmt, 255d); 1188f01029bSMark de Wever #endif // defined(__APPLE__) || defined(_AIX) || defined(_WIN32) || defined(__FreeBSD__) 119e5d2d3eaSMark de Wever 120e5d2d3eaSMark de Wever std::locale::global(std::locale::classic()); 121e5d2d3eaSMark de Wever } 122e5d2d3eaSMark de Wever 123e5d2d3eaSMark de Wever template <class CharT> 124e5d2d3eaSMark de Wever static void test() { 125e5d2d3eaSMark de Wever using namespace std::literals::chrono_literals; 126e5d2d3eaSMark de Wever 127e5d2d3eaSMark de Wever test_no_chrono_specs<CharT>(); 128e5d2d3eaSMark de Wever test_valid_values<CharT>(); 129e5d2d3eaSMark de Wever check_invalid_types<CharT>({SV("d"), SV("e"), SV("Od"), SV("Oe")}, 0d); 130e5d2d3eaSMark de Wever 131402eb2efSMark de Wever check_exception("The format specifier expects a '%' or a '}'", SV("{:A"), 0d); 132402eb2efSMark de Wever check_exception("The chrono specifiers contain a '{'", SV("{:%%{"), 0d); 133402eb2efSMark de Wever check_exception("End of input while parsing a conversion specifier", SV("{:%"), 0d); 134e5d2d3eaSMark de Wever check_exception("End of input while parsing the modifier E", SV("{:%E"), 0d); 135e5d2d3eaSMark de Wever check_exception("End of input while parsing the modifier O", SV("{:%O"), 0d); 136e5d2d3eaSMark de Wever 137e5d2d3eaSMark de Wever // Precision not allowed 138402eb2efSMark de Wever check_exception("The format specifier expects a '%' or a '}'", SV("{:.3}"), 0d); 139e5d2d3eaSMark de Wever } 140e5d2d3eaSMark de Wever 141e5d2d3eaSMark de Wever int main(int, char**) { 142e5d2d3eaSMark de Wever test<char>(); 143e5d2d3eaSMark de Wever 144e5d2d3eaSMark de Wever #ifndef TEST_HAS_NO_WIDE_CHARACTERS 145e5d2d3eaSMark de Wever test<wchar_t>(); 146e5d2d3eaSMark de Wever #endif 147e5d2d3eaSMark de Wever 148e5d2d3eaSMark de Wever return 0; 149e5d2d3eaSMark de Wever } 150