diff --git a/include/fmt/compile.h b/include/fmt/compile.h index 9c0ff9bd10f23..cfca194981523 100644 --- a/include/fmt/compile.h +++ b/include/fmt/compile.h @@ -194,24 +194,10 @@ constexpr const auto& get([[maybe_unused]] const T& first, return get(rest...); } -constexpr int invalid_arg_index = -1; - -template -constexpr int get_arg_index_by_name(basic_string_view name) { - if constexpr (detail::is_statically_named_arg()) { - if (name == T::name) return N; - } - if constexpr (sizeof...(Args) == 0) { - return invalid_arg_index; - } else { - return get_arg_index_by_name(name); - } -} - template constexpr int get_arg_index_by_name(basic_string_view name, type_list) { - return get_arg_index_by_name<0, Args...>(name); + return get_arg_index_by_name(name); } template struct get_type_impl; diff --git a/include/fmt/core.h b/include/fmt/core.h index d2a6e92c67b0c..4e27edebb3adb 100644 --- a/include/fmt/core.h +++ b/include/fmt/core.h @@ -281,6 +281,16 @@ # define FMT_COMPILE_TIME_CHECKS 0 #endif +#ifndef FMT_USE_NONTYPE_TEMPLATE_PARAMETERS +# if defined(__cpp_nontype_template_args) && \ + ((FMT_GCC_VERSION >= 903 && __cplusplus >= 201709L) || \ + __cpp_nontype_template_args >= 201911L) +# define FMT_USE_NONTYPE_TEMPLATE_PARAMETERS 1 +# else +# define FMT_USE_NONTYPE_TEMPLATE_PARAMETERS 0 +# endif +#endif + // Enable minimal optimizations for more compact code in debug mode. FMT_GCC_PRAGMA("GCC push_options") #ifndef __OPTIMIZE__ @@ -991,6 +1001,7 @@ template inline void init_named_args(named_arg_info*, int, int) {} template struct is_named_arg : std::false_type {}; +template struct is_statically_named_arg : std::false_type {}; template struct is_named_arg> : std::true_type {}; @@ -2244,21 +2255,87 @@ class compile_parse_context using base::check_arg_id; }; +template constexpr bool check_named_argument() { + return !detail::is_named_arg() || + (detail::is_named_arg() && detail::is_statically_named_arg()); +} + +template +constexpr bool check_named_arguments() { + return true; +} + +template constexpr bool check_named_arguments() { + return check_named_argument() && check_named_arguments(); +} + +constexpr int invalid_arg_index = -1; + +#if FMT_USE_NONTYPE_TEMPLATE_PARAMETERS +template +constexpr int get_arg_index_by_name(basic_string_view name) { + if constexpr (detail::is_statically_named_arg()) { + if (name == T::name) return N; + } + if constexpr (sizeof...(Args) == 0) { + return invalid_arg_index; + } else { + return get_arg_index_by_name(name); + } +} + +template +constexpr int get_arg_index_by_name(basic_string_view name) { + if constexpr (sizeof...(Args) == 0) { + return invalid_arg_index; + } else { + return get_arg_index_by_name<0, Args...>(name); + } +} +#else +template +constexpr int get_arg_index_by_name(basic_string_view) { + return invalid_arg_index; +} +#endif + template class format_string_checker { public: explicit FMT_CONSTEXPR format_string_checker( basic_string_view format_str, ErrorHandler eh) : context_(format_str, num_args, eh), - parse_funcs_{&parse_format_specs...} {} + parse_funcs_{&parse_format_specs...} { + static_assert( + check_named_arguments(), +#if FMT_USE_NONTYPE_TEMPLATE_PARAMETERS + "use statically named arguments with compile-time format strings: " + "fmt::arg(\"name\", value) -> \"name\"_a = value" +#else + "compile-time checks don't support named arguments, " + "please use C++20 or/and a newer compiler" +#endif + ); + } FMT_CONSTEXPR void on_text(const Char*, const Char*) {} FMT_CONSTEXPR int on_arg_id() { return context_.next_arg_id(); } FMT_CONSTEXPR int on_arg_id(int id) { return context_.check_arg_id(id), id; } - FMT_CONSTEXPR int on_arg_id(basic_string_view) { - on_error("compile-time checks don't support named arguments"); + FMT_CONSTEXPR int on_arg_id(basic_string_view id) { +#if FMT_USE_NONTYPE_TEMPLATE_PARAMETERS + auto index = get_arg_index_by_name(id); + if (index == invalid_arg_index) { + on_error("named argument is not found"); + } + return context_.check_arg_id(index), index; +#else + (void)id; + on_error( + "compile-time checks don't support named arguments, " + "please use C++20 or/and a newer compiler"); return 0; +#endif } FMT_CONSTEXPR void on_replacement_field(int, const Char*) {} diff --git a/include/fmt/format.h b/include/fmt/format.h index 089f042de5ae4..c1cf15c249af4 100644 --- a/include/fmt/format.h +++ b/include/fmt/format.h @@ -255,16 +255,6 @@ inline int ctzll(uint64_t x) { FMT_END_NAMESPACE #endif -#ifndef FMT_USE_NONTYPE_TEMPLATE_PARAMETERS -# if defined(__cpp_nontype_template_args) && \ - ((FMT_GCC_VERSION >= 903 && __cplusplus >= 201709L) || \ - __cpp_nontype_template_args >= 201911L) -# define FMT_USE_NONTYPE_TEMPLATE_PARAMETERS 1 -# else -# define FMT_USE_NONTYPE_TEMPLATE_PARAMETERS 0 -# endif -#endif - FMT_BEGIN_NAMESPACE namespace detail { @@ -3408,9 +3398,6 @@ template struct udl_formatter { } }; -template -struct is_statically_named_arg : std::false_type {}; - # if FMT_USE_NONTYPE_TEMPLATE_PARAMETERS template Str> struct statically_named_arg : view { diff --git a/test/format-test.cc b/test/format-test.cc index 50e96ef53b3e8..0946ea40833d6 100644 --- a/test/format-test.cc +++ b/test/format-test.cc @@ -1694,6 +1694,14 @@ TEST(format_test, compile_time_string) { EXPECT_EQ(L"42", fmt::format(FMT_STRING(L"{}"), 42)); EXPECT_EQ("foo", fmt::format(FMT_STRING("{}"), string_like())); +#if FMT_USE_NONTYPE_TEMPLATE_PARAMETERS + using namespace fmt::literals; + EXPECT_EQ("foobar", fmt::format(FMT_STRING("{foo}{bar}"), "bar"_a = "bar", + "foo"_a = "foo")); + EXPECT_EQ("", fmt::format(FMT_STRING(""))); + EXPECT_EQ("", fmt::format(FMT_STRING(""), "arg"_a = 42)); +#endif + (void)static_with_null; (void)static_with_null_wide; (void)static_no_null; @@ -2339,8 +2347,14 @@ TEST(format_test, format_string_errors) { # else fmt::print("warning: constexpr is broken in this version of MSVC\n"); # endif - EXPECT_ERROR("{foo", "compile-time checks don't support named arguments", +# if FMT_USE_NONTYPE_TEMPLATE_PARAMETERS + EXPECT_ERROR("{foo}", "named argument is not found", decltype("bar"_a = 42)); +# else + EXPECT_ERROR("{foo}", + "compile-time checks don't support named arguments, " + "please use C++20 or/and a newer compiler", int); +# endif EXPECT_ERROR_NOARGS("{10000000000}", "number is too big"); EXPECT_ERROR_NOARGS("{0x}", "invalid format string"); EXPECT_ERROR_NOARGS("{-}", "invalid format string");