// { dg-do run { target c++20 } } #include #include template bool is_format_string_for(const char* str, Args&&... args) { try { (void) std::vformat(str, std::make_format_args(args...)); return true; } catch (const std::format_error&) { return false; } } template bool is_format_string_for(const wchar_t* str, Args&&... args) { try { (void) std::vformat(str, std::make_wformat_args(args...)); return true; } catch (const std::format_error&) { return false; } } void test_no_args() { VERIFY( is_format_string_for("") ); VERIFY( is_format_string_for("chars") ); VERIFY( is_format_string_for(" The Great Escape {{}} ") ); VERIFY( ! is_format_string_for("{") ); VERIFY( ! is_format_string_for("}") ); VERIFY( ! is_format_string_for("}{") ); VERIFY( ! is_format_string_for("{{}") ); VERIFY( ! is_format_string_for("{{{") ); VERIFY( ! is_format_string_for("{{{{{") ); } void test_indexing() { VERIFY( is_format_string_for("{} to {}", "a", "b") ); // automatic indexing VERIFY( is_format_string_for("{1} to {0}", "a", "b") ); // manual indexing VERIFY( ! is_format_string_for("{0} to {}", "a", "b") ); // mixed indexing VERIFY( ! is_format_string_for("{} to {1}", "a", "b") ); // mixed indexing VERIFY( is_format_string_for("{} {} {}", 1, 2, 3) ); VERIFY( is_format_string_for("{} {} {}", 1, 2, 3, 4) ); VERIFY( is_format_string_for("{0} {1} {2}", 1, 2, 3, 4) ); VERIFY( is_format_string_for("{1} {2} {3}", 1, 2, 3, 4) ); VERIFY( is_format_string_for("{3} {3} {3}", 1, 2, 3, 4) ); VERIFY( ! is_format_string_for("{2}", 1, 2) ); VERIFY( ! is_format_string_for("{0} {}", 1) ); VERIFY( ! is_format_string_for("{} {0}", 1) ); } #if __cpp_lib_format_ranges constexpr bool escaped_strings_supported = true; #else constexpr bool escaped_strings_supported = false; #endif void test_format_spec() { VERIFY( is_format_string_for("{:}", 1) ); VERIFY( is_format_string_for("{0:}", 1) ); VERIFY( is_format_string_for("{2:}", 1, 2, 3) ); VERIFY( is_format_string_for("{0:s} {0:}", "str") ); VERIFY( is_format_string_for("{0:} {0:c}", 'c') ); VERIFY( is_format_string_for("{0:p} {0:}", nullptr) ); VERIFY( is_format_string_for("{:d} {:+d}", true, true) ); VERIFY( is_format_string_for("{:0<-#03Ld}", 1) ); VERIFY( is_format_string_for("{1:0<-#03.4Lf}", 1, 2.3) ); VERIFY( is_format_string_for("{1:3.3f}", 1, 2.3) ); VERIFY( is_format_string_for("{:#d}", 'c') ); VERIFY( is_format_string_for("{:#d}", true) ); VERIFY( is_format_string_for("{0:s} {0:?}", "str") == escaped_strings_supported ); VERIFY( is_format_string_for("{0:} {0:?}", 'c') == escaped_strings_supported ); // Invalid sign options. VERIFY( ! is_format_string_for("{:+}", "str") ); VERIFY( ! is_format_string_for("{:+s}", "str") ); VERIFY( ! is_format_string_for("{:+}", 'c') ); VERIFY( ! is_format_string_for("{:+c}", 'c') ); VERIFY( ! is_format_string_for("{:+p}", nullptr) ); VERIFY( ! is_format_string_for("{:+}", true) ); VERIFY( ! is_format_string_for("{:+s}", true) ); VERIFY( ! is_format_string_for("{:+?}", "str") ); VERIFY( ! is_format_string_for("{:+?}", 'c') ); // Invalid alternate forms. VERIFY( ! is_format_string_for("{:#}", "str") ); VERIFY( ! is_format_string_for("{:#s}", "str") ); VERIFY( ! is_format_string_for("{:#}", 'c') ); VERIFY( ! is_format_string_for("{:#c}", 'c') ); VERIFY( ! is_format_string_for("{:#}", true) ); VERIFY( ! is_format_string_for("{:#s}", true) ); VERIFY( ! is_format_string_for("{:#}", nullptr) ); VERIFY( ! is_format_string_for("{:#p}", nullptr) ); VERIFY( ! is_format_string_for("{:#?}", "str") ); VERIFY( ! is_format_string_for("{:#?}", 'c') ); VERIFY( ! is_format_string_for("{:0c}", 'c') ); VERIFY( ! is_format_string_for("{:0s}", true) ); // Precision only valid for string and floating-point types. VERIFY( ! is_format_string_for("{:.3d}", 1) ); VERIFY( ! is_format_string_for("{:3.3d}", 1) ); VERIFY( is_format_string_for("{:3.3s}", "str") ); VERIFY( ! is_format_string_for("{:3.3s}", 'c') ); VERIFY( ! is_format_string_for("{:3.3p}", nullptr) ); // Invalid presentation types for integers. VERIFY( ! is_format_string_for("{:f}", 1) ); VERIFY( ! is_format_string_for("{:s}", 1) ); VERIFY( ! is_format_string_for("{:g}", 1) ); VERIFY( ! is_format_string_for("{:E}", 1) ); VERIFY( ! is_format_string_for("{:D}", 1) ); // Invalid presentation types for floating-point types. VERIFY( ! is_format_string_for("{:d}", 1.2) ); VERIFY( ! is_format_string_for("{:b}", 1.2) ); VERIFY( ! is_format_string_for("{:x}", 1.2) ); VERIFY( ! is_format_string_for("{:s}", 1.2) ); // Invalid presentation types for strings. VERIFY( ! is_format_string_for("{:S}", "str") ); VERIFY( ! is_format_string_for("{:d}", "str") ); // Maximum integer value supported for widths and precisions is USHRT_MAX. VERIFY( is_format_string_for("{:65535}", 1) ); VERIFY( is_format_string_for(L"{:65535}", 1) ); VERIFY( ! is_format_string_for("{:65536}", 1) ); VERIFY( ! is_format_string_for(L"{:65536}", 1) ); VERIFY( ! is_format_string_for("{:9999999}", 1) ); VERIFY( ! is_format_string_for(L"{:9999999}", 1) ); } void test_pr110862() { try { int i = 1; // PR libstdc++/110862 out-of-bounds read on invalid format string (void) std::vformat("{0:{0}", std::make_format_args(i)); VERIFY( false ); } catch (const std::format_error& e) { std::string_view what = e.what(); VERIFY( what.find("unmatched '{'") != what.npos ); } } void test_pr110974() { try { double d = 1.0; int i = 1; // PR libstdc++/110974 out of bounds read on invalid format string "{:{}." std::string_view fmt{"{:{}.0", 5}; // "0" is not part of the format string. (void) std::vformat(fmt, std::make_format_args(d, i)); VERIFY( false ); } catch (const std::format_error& e) { std::string_view what = e.what(); // GCC 13.2 throws "invalid width or precision in format-spec" after // trying to parse the "0" past-the-end of the format string. // There should be an exception before even trying that: VERIFY( what.find("missing precision after '.'") != what.npos ); } } int main() { test_no_args(); test_indexing(); test_format_spec(); test_pr110862(); test_pr110974(); }