During Varna (2023-06), [P2741R3] had been adopted into the C++26 working draft, which gave
the ability to accept a user-generated string-like object as the message parameter. This extension allowed the user of
to provide a more precise error message in compile time, thus significantly increasing the user-friendliness of libraries. This proposal, therefore, proposes to unify the language by allowing other constructs in the language that currently accept a message parameter, namely
,
, and
, to also allow a user-generated string-like object as the provided message.
1. Revision History
1.1. R1 (2025-01 pre-Hagenberg Mailing)
-
Add discussion on instantiation timing of attributes, aka if SFINAE happens if the attribute argument clause results in invalid substitution.
-
Add polls section.
-
In Wrocław (2024-11), EWGI sees [P3423R0] and forwarded it to EWG (polls). Audience is altered accordingly.
-
Rebase onto latest draft [N5001].
1.2. R0 (2024-10 pre-Wrocław Mailing)
-
Initial revision.
2. Motivation and History
The first appearance of the compile-time message parameter in the C++ core language is in C++11, where
declaration was introduced into the language by [N1720] with the syntax
. At that time, the message parameter was the first time that users of the C++ language were able to control the generation of diagnostic messages in a dynamic way (excluding the preprocess macro
, which is not controllable by the language itself, let alone dynamically in compile-time programming). Such flexibility has allowed the emergence of user-friendly compile-time libraries, where error messages can directly present to the user what they have done wrong. For instance, the following misuse of
:
#include <print>struct A {}; int main () { A a ; std :: println ( "{}" , a ); }
can directly generate user-friendly diagnostics under clang 19.1.0 with libstdc++:
[...] In file included from < source >: 1 : In file included from / opt / compiler - explorer / gcc -14.2.0 / lib / gcc / x86_64 - linux - gnu / 14.2.0 / .. / .. / .. / .. / include / c ++/ 14.2.0 / : 41 : / opt / compiler - explorer / gcc -14.2.0 / lib / gcc / x86_64 - linux - gnu / 14.2.0 / .. / .. / .. / .. / include / c ++/ 14.2.0 / format : 3732 : 18 : error : static assertion failed due to requirement 'is_default_constructible_v < std :: formatter < A , char >> ': std :: formatter must be specialized for the type of each format arg 3732 | static_assert ( is_default_constructible_v < formatter < _Tq , _CharT >> , | [...]
Although a bit cramped and hidden, the error message directly tells the user that
must be specialized, thus informing them how to fix the problem. Without this feature, the user will only be faced with cryptic error messages such as "
is not default constructible", which requires an expert with a deep understanding of the specification of
to uncover the actual cause of the error.
Over the following years, more and more control over the generation and content of the diagnostic message of the compiler had been opened to the library writer:
-
In C++14, [N3760] introduced the
attribute with an optional message parameter.[[ deprecated ]] -
In C++17, [N3928] made
’s message parameter optional, making the condition expression automatically usable as an error message.static_assert -
In C++20, [P1301R4] introduced an optional message parameter to the
attribute.[[ nodiscard ]] -
In C++23, [P2593R1] (adopted through [CWG2518]) allowed
to appear in dependent contexts, which effectively turnedstatic_assert ( false, "message" )
into a basic compile-time print statement that can be used to emit custom error messages.static_assert -
In C++26, [P2573R2] allowed
to have an optional message parameter.= delete
The key difference between those extensions of the message parameter compared to
is that some of them don’t generate error messages but allow the library writer to control the content of the warnings generated by the compiler. These extensions have allowed more and more misuse of the library to be diagnosed with user-friendly error messages that directly convey the cause and the fix of errors, and contributed significantly to making C++'s error message more approachable and readable.
However, despite these advancements, one key limitation of the message parameter in all these examples remained: the parameter must be an (unevaluated) string literal. This restriction means that no computation can be performed when providing the custom message, preventing library writers from providing more useful diagnostic messages to the user. For instance, with the following misuse of
:
std :: format ( "{:d}" , "not a number" ); // Current error message: (https://godbolt.org/z/enxWj5WY1) error : call to consteval function 'std :: basic_format_string < char , const char ( & )[ 13 ] >:: basic_format_string < char [ 5 ] > 'is not a constant expression // Potential best we can get using static_assert with fixed message parameter: error : static assertion failed due to requirement [...] : wrong formatter specified for std :: format
The above "potential best" message is already much more user-friendly than the status quo but still lacks some crucial information: the position of the invalid argument (the first, the second, etc.), the actual required type of
(integer), and so on. Let’s compare with Rust’s error message:
format!( "{} {:b}" , 2 , "not a number" ); // Error message: error [ E0277 ] :the trait bound `str :Binary `is not satisfied - ->< source > :3 :27 | 3 | format!( "{} {:b}" , 2 , "not a number" ); | ---- ^^^^^^^^^^^^^^ the trait `Binary `is not implemented for `str `, which is required by `& str :Binary `| | | required by a bound introduced by this call | = help :the following other types implement trait `Binary `:& T & mut T NonZero < T > Saturating < T > Wrapping < T > i128 i16 i32 and 9 others [ .. .]
The above error message is clearly a step ahead even compared to C++'s "potential best" message achievable. Not only is the precise wrong argument and format specifier is pointed out, but several possible fixes (correct type for the argument) are suggested, prompting the user that an integer-like argument is required here.
Such an error message is not possible with
with a fixed message parameter. Since no computation is allowed, contextual information, such as the position of the argument, cannot be embedded within the message. In light of this, in the C++17 cycle, [N4433] proposed making
’s message parameter dynamically computed and allowing any expression contextually convertible to
to appear in the message parameter.
However, the consensus at that time was that the C++ language was not yet ready for dynamically computed string literals, since
evaluation was still too limited at that point (for example,
could not be used at compile time at that time). Therefore, the first attempt to introduce dynamically generated diagnostic messages to C++ failed.
The
evaluation had evolved significantly after that point in time. In C++20, we gained the ability to use
and
in compile-time, and the extension of allowing user-defined class types in NTTP allowed strings to be passed to functions as compile-time arguments. Nowadays, users no longer need to write compile-time string concatenations on their own, and compile-time string generation facilities such as
are also on the horizon (implemented by
already).
Finally, in the C++26 cycle, the matter of user-generated
messages is revisited, and [P2741R3] allowed the use of custom-generated string-like parameters as the second message parameter of
. This relaxation had allowed for significantly better error messages that embed contextual information to appear in library code, thus allowing the above
misuse example to output an error message similar to:
error : static assertion failed due to requirement [...] : For argument #1 , the d specifier is only valid for integers , but a argument with type const char ( & )[ 13 ] was provided .
Clearly, this new "potential best" error message is much superior to the previous best possible message, thus marking another big step forward for the user-friendliness of C++ error messages. To further expand this advantage, this proposal, therefore, proposes to allow the other three appearances of the message parameter in the core language to also accept user-generated string-like approaches. Such expansion not only unifies the language but also provides more opportunities for more friendly error/warning messages to be provided. Several examples of the possible diagnostics that will be allowed by this proposal can be seen in the Usage Examples section below.
3. Usage Example
3.1. Standard Library
This part contains some concrete examples of how functions and classes in the standard library can benefit from the new flexibility of message parameters. All examples are based on [N5001], and also assumes the existence of
([P3391R0]) and Reflection ([P2996R7]).
One thing to note here is that
is never required on the deprecated features in the standard library. Similarly,
is regarded as QoI after the adoption of [P2422R1] in St. Louis (2024-06) and no longer appears in the standard library’s specification. Therefore, all the following examples only resemble some possible use of the attributes by a standard library vendor. All the examples below that used
were marked as so in the specification before [P2422R1] landed.
// [memory.syn] namespace std { // ... template < class T , class ... Args > // T is not array constexpr unique_ptr < T > make_unique ( Args && ... args ); template < class T > // T is U[] constexpr unique_ptr < T > make_unique ( size_t n ); template < class T , class ... Args > // T is U[N] unspecified make_unique ( Args && ...) = delete ( format ( "make_unique<U[{}]>(...) is not supported; perhaps you mean make_unique<U[]>({}) instead?" , extent_v < T > , extent_v < T > )); template < class T , class ... Args > // T is not array constexpr unique_ptr < T > make_unique_for_overwrite ( Args && ... args ); template < class T > // T is U[] constexpr unique_ptr < T > make_unique_for_overwrite ( size_t n ); template < class T , class ... Args > // T is U[N] unspecified make_unique_for_overwrite ( Args && ...) = delete ( format ( "make_unique_for_overwrite<U[{}]>(...) is not supported; perhaps you mean make_unique_for_overwrite<U[]>({}) instead?" , extent_v < T > , extent_v < T > )); } // [new.syn] namespace std { // [ptr.launder], pointer optimization barrier template < class T > [[ nodiscard ( format ( "The resulting {}* pointer returned is discarded, which defeats the purpose of std::launder." , meta :: identifier_of ( ^^ T ) ))]] constexpr T * launder ( T * p ) noexcept ; } // [future.syn] namespace std { // [futures.async], function template async template < class F , class ... Args > [[ nodiscard ( format ( "The returning future<{}> is discarded, which makes the given function of type {} always executes synchronously." , meta :: identifier_of ( ^^ invoke_result_t < decay_t < F > , decay_t < Args > ... > ), meta :: identifier_of ( ^^ F ) ))]] future < invoke_result_t < decay_t < F > , decay_t < Args > ... >> async ( F && f , Args && ... args ); template < class F , class ... Args > [[ nodiscard ( format ( "The returning future<{}> is discarded, which makes the given function of type {} always executes synchronously." , meta :: identifier_of ( ^^ invoke_result_t < decay_t < F > , decay_t < Args > ... > ), meta :: identifier_of ( ^^ F ) ))]] future < invoke_result_t < decay_t < F > , decay_t < Args > ... >> async ( launch policy , F && f , Args && ... args ); } // [depr.relops] namespace std :: rel_ops { template < class T > [[ deprecated ( format ( "operator!= will be automatically generated for class {}" , meta :: identifier_of ( ^^ T ) ))]] bool operator != ( const T & , const T & ); } // [depr.meta.types] namespace std { // ... template < size_t Len , size_t Align = default - alignment > // see below struct [[ deprecated ( format ( "use alignas({}) std::byte[{}] instead." , Align , Len ))]] aligned_storage ; template < size_t Len , size_t Align = default - alignment > // see below using aligned_storage_t = typename aligned_storage < Len , Align >:: type ; template < size_t Len , class ... Types > struct [[ deprecated ( format ( "use alignas({}) std::byte[std::max({{{}}})] instead." , string_view { vector { meta :: identifier_of ( ^^ Types )...} | views :: join_with ( ", " )}, string_view { vector { format ( "sizeof({})" , meta :: identifier_of ( ^^ Types ))...} | views :: join_with ( ", " )} ))]] aligned_union ; template < size_t Len , class ... Types > using aligned_union_t = typename aligned_union < Len , Types ... >:: type ; }
Some of the examples above are particularly notable:
-
For
, the dynamically generated message will embed the corresponding type parameter, which makes it easier for users to pinpoint which class still has relation operators generated by the deprecatedstd :: rel_ops
namespace helper.rel_ops -
For
, the dynamically generated message will directly tell users about the correct classes and specific invocations to use:std :: aligned_union
// Usage: std :: aligned_union_t < 0 , A , B > t_buff : // Diagnostics: warning : 'std :: aligned_union 'is deprecated : use alignas ( A , B ) std :: byte [ std :: max ({ sizeof ( A ), sizeof ( B )})] instead .
-
All of these examples are just a proof of concept on how user-friendly diagnostics is possible; in reality, due to the cost of pulling in the dependency on
andstd :: format
, such a message is probably not feasible for most vendors.< ranges >
3.2. Third-Party Libraries
A potential use case of the new flexibility to third-party library developers is to include the library name in every diagnostic emitted by the types and functions within the library. Although this is already achievable with the current fixed message parameter, adding the name to each message parameter is error-prone. A better approach is something like (assuming that
([P3094R5]) exists):
namespace my_lib { constexpr std :: fixed_string lib_name = "myLib" ; [[ nodiscard ( lib_name + ": api()'s return value should not be discarded because ..." )]] void api (); struct [[ deprecated ( lib_name + ": class S is deprecated because ..." )]] S ; }
This design allows the library name to be changed easily by altering
and also reduces the possibility of misspelling the library name across different message parameters. Furthermore, a similar design can be applied to functions within a class, with the class name factored out into a
static variable.
4. Design
The general design of this proposal is to allow the following constructs:
= delete ( message ); [[ nodiscard ( message )]] [[ deprecated ( message )]]
where
can be a user-provided string-like parameter with the requirements documented below.
4.1. What Is A String?
An immediately obvious question is: What constitute as a string that can be provided as the message parameter? Obviously, we don’t want these core language parameters to be tied to a concrete library type such as
.
Fortunately, [P2741R3] already solved this problem for
. A compatible string-like type that can be used as the message parameter to
has the following requirements:
-
Has a
member function that produces a type implicitly convertible tosize () std :: size_t -
Has a
member function that produces a type implicitly convertible todata () const char * -
The elements in the range
are valid integer constant expression[ data (), data () + size ())
This proposal also proposes the same requirement for the message parameter in
,
, and
.
4.2. Design Decisions
The following design decisions were made by [P2741R3] and are also proposed by this proposal. Please see [P2741R3] for detailed motivations of these decisions:
-
Only contiguous ranges are supported. Thus
anddata ()
are used instead ofsize ()
andbegin ()
.end () -
The message-producing expression is always instantiated but only evaluated if the corresponding declaration fires:
template < int N > void bad () { static_assert ( N > 0 , "fires" ); } [[ nodiscard ( bad < 0 > ())]] int fun () { return 2 ; } // although fun() is never called, bad<0> is instantiated and the static_assert is fired.
-
Only ranges of
are supported; no support forchar
,wchar_t
,char8_t
, orchar16_t
is proposed following SG16 guidance to [P2741R3].char32_t -
Diagnostic encoding is used, and the provided message parameter needs to be converted to that encoding.
-
Null-terminated strings are not supported (
\
will always be printed in the message).0
4.3. Attribute Expression Argument
One thing that is novel in this proposal is the fact that arbitrary expression is allowed as arguments to
and
attributes. This is not completely unheard of, as, in C++23, the
attribute was adopted via [P1774R8], which also accepts arbitrary expression as an argument. However, the processing of expression in these two cases is different:
-
For
,[[ assume ( expr )]]
is allowed to be any expression, which is unevaluated but ODR-used (i.e. potentially evaluated).expr -
For
and[[ nodiscard ( message )]]
,[[ deprecated ( message )]]
must be a constant expression, which is treated as an unevaluated operand if the applied entity is not used.message
Here, potentially evaluated and unevaluated operands are complete inversions of each other ([basic.def.odr]/3). The most prominent distinction between those two treatments is the requirement on the definition of the function used:
int some_fact (); // no definition consteval std :: array < char , 5 > message (); // no definition yet // For ODR-use, all used functions must have a definition ([basic.def.odr]/12) // For unevaluated operand, function definitions are not required at the point of use template < typename T > void fun () { [[ assume ( some_fact () > 42 )]]; // IFNDR static_assert ( false, message ()); // okay } consteval std :: array < char , 5 > message () { return { 'H' , 'e' , 'l' , 'l' , 'o' }; } int main () { fun < int > (); // Diagnostics: Hello }
Although as of Oct 2024, the latest released versions of GCC and Clang both do not diagnose the above example (Compiler Explorer) and instead just silently ignore the assumption.
This can also affect the ABI of a class, as demonstrated in [P1774R8]:
constexpr auto f ( int i ) { return sizeof ( [ = ] { [[ assume ( i == 0 )]]; } ); } constexpr auto f2 ( int i ) { // nothing is actually captured here return sizeof ( [ = ] { static_assert ( true, std :: array < char , 1 > { "Hello" [ sizeof ( i )]}); } ); } struct X { char data [ f ( 0 )]; }; struct X2 { char data [ f2 ( 0 )]; }; // sizeof(X) == 4, sizeof(X2) == 1
Overall, the author doesn’t think that this discrepancy has that much of an effect since
and
’s message parameter is required to be a constant expression, and no runtime behavior should be affected.
4.4. Attribute Placement and Name Visibility
An unfortunate consequence of the grammar of attributes is that, as demonstrated in the Usage Example section, the placement of
and
on class declarations must be after the
:
[[ nodiscard ( message )]] struct S ; // Wrong, attribute ignored struct [[ nodiscard ( message )]] S ; // Right
Fortunately, most compilers will emit a warning that the specified attribute is ignored in this case.
Another problem caused by the placement of attribute and
is the name visibility problem. Since
is a kind of definition and attributes must appear before the parameter list, the former can reference the parameters to the function:
template < typename T > std :: fixed_string message (); template < typename T , typename U > void fun ( const T & t , const U & u ) = delete ( message < decltype ( t + u ) > ()); template < typename T , typename U > [[ nodiscard ( message < decltype ( std :: declval < T > () + std :: declval < U > ()) > ())]] // cannot reference t and u here int fun2 ( const T & t , const U & u ); // Placement: [[ deprecated ]] void fun (); // okay void fun [[ deprecated ]] (); // okay, equivalent void fun () [[ deprecated ]]; // attribute here applies to the function type, which is invalid and is silently ignored/cause error in major implementations
This proposal does not propose to change any grammar of attribute placement, and these unfortunate consequences are only documented here for information.
4.5. What If The Message Parameter Is Invalid?
As discussed before, the message parameter is intended to always be instantiated but only evaluated on triggering:
template < auto N > consteval std :: string_view oups () { static_assert ( N , "this always fires" ); return "oups!" ; } void f () = delete ( oups < false> ()); // static_assert fires even though f() is not called [[ deprecated ( oups < false> ())]] void f2 (); // ditto
However, when templates are involved, the situation is less obvious.
As pointed out in an issue submitted to CWG GitHub, it is currently unclear if the attribute-specifier-seq that appears in a function declaration is in the immediate context and therefore contributes to overload resolution. This part of the function declaration is not in the deduction substitution loci ([temp.deduct.general]/7), so the standard currently does not specify the answer to this question. With the addition of user-provided argument to
and
, we have to answer this question.
Consider:
template < bool B > consteval std :: string_view get_message () { static_assert ( B ); return "message" ; } template < bool B > void f1 (...) {} // 1A template < bool B > [[ deprecated ( get_message < B > ())]] void f1 ( int ) {} // 1B void test_f1 () { f1 < false> ( 1 ); // #1 } template < bool B > [[ deprecated ( get_message < B > ())]] void f2 (...) {} // 2A template < bool B > void f2 ( int ) {} // 2B void test_f2 () { f2 < false> ( 2 ); // #2 } template < typename T > void f3 (...) {} // 3A template < typename T > [[ deprecated ( get_message < T :: value > ())]] void f3 ( int ) {} // 3B void test_f3 () { f3 < int > ( 3 ); // #3 }
The author think there are two approaches in general to determine the well-formedness of these three examples.
The first approach is to make attribute-specifier-seq attached to a function part of its deduction substitution loci (i.e. in the immediate context). This makes them equivalent to return types (equivalent Compiler Explorer), and thus:
-
#1 is ill-formed, since substitution into 1B instantiates
.get_message < false> () -
#2 is ill-formed, since substitution into 2A still instantiates
.get_message < false> () -
#3 is well-formed, since substituting into 3B will cause substitution to fail, and thus 3B will SFINAE-away and 3A is chosen by overload resolution.
In other words, substitution into attribute (and
) happens before overload resolution, and any substitution failure is an error that cause the overload to drop out of overload resolution.
Another approach is to make attributes and
to only be instantiated when overload resolution selects that function overload (i.e. not in the immediate context). This makes them equivalent to
(or, equivalently,
with message in the body, equivalent Compiler Explorer), and thus:
-
#1 is ill-formed, since overload resolution selects 1B and instantiates
.get_message < false> () -
#2 is well-formed, since overload resolution does not select 2B and
is not instantiated.get_message -
#3 is ill-formed, since substitution failure after overload resolution finishes is an immediate hard error.
The author thinks that the second approach is more sensible, since
is conceptually a kind of function body, and the idea that
affects overload resolution is a bit nonsensical. The original response to the issue agrees with this decision.
4.6. Alternative Syntax Considered
One possible alternative syntax proposed by [P1267R0] is to use a
attribute to express the message uniformly. Such a design will look like:
[[ deprecated , reason ( message )]] void fun (); [[ nodiscard , reason ( message )]] int fun2 (); [[ reason ( message )]] void fun3 () = delete ;
However,
,
, and
already accept a message parameter today, so the author thinks this ship has already been sunk.
4.7. Claim of Syntax Space
During the EWGI review of [P2573R2], some people expressed concern about the possibility of that proposal "claiming the syntax space" of possible further enhancement to
. The same concern may apply to this proposal; namely, it is possible for a future proposal to propose "conditional
":
template < typename T > [[ nodiscard ( sizeof ( T ) > 10 )]] int some_fun ();
While this is certainly a possible future direction, the author doesn’t think this proposal claims any syntax space in this regard. Similar to
, the only modification needed for conditional
to still be possible is to require two arguments in that case:
template < typename T > [[ nodiscard ( sizeof ( T ) > 10 , "reason" )]] int some_fun ();
This does sacrifice some convenience, but if desired, the distinction can be made such that for
, if
is convertible to
, then conditional semantic is used; if
satisfies the string-like requirements, the message semantic is used. A tie-breaker between these two is obviously needed for types that satisfy both, but the author doesn’t think this is a huge conflict that forms an obstacle to this proposal.
4.8. Interaction with Reflection
Reflection ([P2996R7]) is great and will probably come to C++ in the near future. It will also greatly aid in generating friendly diagnostic messages that display the template type argument’s name, as demonstrated in the Usage Examples section above.
However, a natural question arises: should there be a reflection facility to get the message specified to
and attributes?
In the current reflection proposal ([P2996R7]), there is one metafunction that queries whether the function is deleted:
. No attribute-reflecting mechanisms are proposed in [P2996R7], but there is ongoing work on reflecting attributes first by [P1887R1] and currently by [P3385R2]. Should these proposals also propose the ability to query the message provided to the attributes and
?
The author thinks that they probably should (for instance, by letting
return the message or add a new
), but this is certainly outside the scope of this proposal. The message parameter already exists in these attributes, and this proposal does not add any new parameters to them, so the difficulty of supporting them should not change drastically.
4.9. Proposal Scope
This proposal is a pure language extension proposal with no library changes involved. It is also a pure addition, so no breaking changes are involved.
It is intended practice that exchanging
for
will have no breakage and no user-noticeable effect except for the diagnostic messages.
There has been sustained strong opposition to the idea of mandating diagnostic texts in the standard library specifications, and in my opinion, such complaints are justified. This paper does not change this by only proposing to make passing a user-generated string-like object as a message parameter possible and nothing more.
This paper encourages vendors to apply any message they see fit for the purpose as a non-breaking QoI feature.
4.10. Target Vehicle
This proposal targets C++26.
4.11. Feature Test Macro
This proposal bumps the feature test macro for
(
) and the value reported by
for
and
.
However, one thing to note here is that
will report
for
and
if the implementation cannot issue a warning that includes the text of the message provided. This proposal intends for the following rules to be used:
-
If an implementation cannot issue warnings that include the message (even a fixed message) at all,
should report__has_cpp_attribute
.0 -
If an implementation can issue warnings that include a fixed (string literal) message supplied but cannot support arbitrary string-like constant parameter as a message,
should report their current value (__has_cpp_attribute
for201907L
and[[ nodiscard ]]
for201309L
as of Oct 2024).[[ deprecated ]] -
If an implementation can issue warnings that include user-generated strings,
should report the new values modified by this proposal.__has_cpp_attribute
Such rules are consistent with how other feature test macros worked.
5. Implementation Experience
An experimental implementation of the proposed feature is located in my Clang fork at [clang-implementation], which is capable of handling
template < std :: fixed_string str > consteval std :: fixed_string < str . size () + 2 > fun () { return "^" + str + "$" ; } void fun1 () = delete ( fun < "Test" > ()); [[ nodiscard ( fun < "Test2" > ())]] int fun2 () { return 2 ; } [[ deprecated ( fun < "Test3" > ())]] void fun3 () {}
and outputting the following diagnostic messages (close to what the author intended; of course, there are other formats such as put reason on separate lines):
propose . cpp : 56 : 5 : error : call to deleted function 'fun1 ': ^ Test$ 56 | fun1 (); fun2 (); fun3 (); | ^~~~ propose . cpp : 50 : 6 : note : candidate function has been explicitly deleted 50 | void fun1 () = delete ( fun < "Test" > ()); | ^ propose . cpp : 56 : 13 : warning : ignoring return value of function declared with 'nodiscard 'attribute : ^ Test2$ [ - Wunused - result ] 56 | fun1 (); fun2 (); fun3 (); | ^~~~ propose . cpp : 56 : 21 : warning : 'fun3 'is deprecated : ^ Test3$ [ - Wdeprecated - declarations ] 56 | fun1 (); fun2 (); fun3 (); | ^ propose . cpp : 52 : 3 : note : 'fun3 'has been explicitly marked deprecated here 52 | [[ deprecated ( fun < "Test3" > ())]] void fun3 () {} | ^ 2 warnings and 1 error generated .
The implementation is very incomplete (no feature-test macro, only a few tests, etc.), so it is only aimed at proving that the vendors can support the proposed feature relatively easily. No significant obstacles are observed during the implementation.
6. Wording
The wording below is based on [N5001].
Wording notes for CWG and editor:
-
The wording below renames static_assert-message to diagnostic-message, which is now used as the grammar for the general diagnostic message parameter. (Should we actually move diagnostic-message to another place?)
-
The phrase "text of the diagnostic-message" is treated as a word of power, and is referenced by other sections defining
and attributes to simplify the wording.= delete -
The wording for attributes that makes them not a part of a function’s immediate context comes in two parts: Note in [temp.deduct.general]/7 and normative wording in [temp.inst]/14 are both adjusted to give attribute specifiers the same instantiation timing with
specifiers. This part of the wording is inspired by similar wording in [P2900R12] that makes contract specifiers not part of immediate context.noexcept -
is not worded since that is considered a kind of function body and thus follows the same rule already.= delete
-
6.1. 7.7 Constant expressions [expr.const]
[Note 12: Except for a
static_assertdiagnostic-message, a manifestly constant-evaluated expression is evaluated even in an unevaluated operand ([expr.context]). — end note]
6.2. 9.1 Preamble [dcl.pre]
Paragraph 1:
Declarations generally specify how names are to be interpreted. Declarations have the form
[...]static_assertdiagnostic-message: unevaluated-string constant-expression static_assert-declaration: static_assert ( constant-expression ) ; static_assert ( constant-expression ,static_assertdiagnostic-message ) ; [...]
Paragraph 12:
If a
static_assertdiagnostic-message matches the syntactic requirements of unevaluated-string, it is an unevaluated-string and the text of thestatic_assertdiagnostic-message is the text of the unevaluated-string. Otherwise, astatic_assertdiagnostic-message shall be an expressionsuch that [...]
M
Paragraph 13:
In a static_assert-declaration, the constant-expression
is contextually converted to
E and the converted expression shall be a constant expression ([expr.const]). If the value of the expression
bool when so converted is
E true
or the expression is evaluated in the context of a template definition, the declaration has no effect and thestatic_assertdiagnostic-message is an unevaluated operand ([expr.context]). Otherwise, the static_assert-declaration fails and
the program is ill-formed, and
if the
static_assertdiagnostic-message is a constant-expression,
M
shall be a converted constant expression of type
M . size () and let
std :: size_t denote the value of that expression,
N
, implicitly converted to the type "pointer to
M . data () ", shall be a core constant expression and let
const char denote the converted expression,
D for each
where
i ,
0 ≤i < N shall be an integral constant expression, and
D [ i ] the text of the
static_assertdiagnostic-message is formed by the sequence ofcode units, starting at
N , of the ordinary literal encoding ([lex.charset]).
D
Paragraph 14:
Recommended practice: When a static_assert-declaration fails, the resulting diagnostic message should include the text of the
static_assertdiagnostic-message, if one is supplied.
6.3. 9.5 Function definitions [dcl.fct.def]
6.3.1. 9.5.1 General [dcl.fct.def.general]
Paragraph 1:
Function definitions have the form
function-definition: attribute-specifier-seqopt decl-specifier-seqopt declarator virt-specifier-seqopt function-body attribute-specifier-seqopt decl-specifier-seqopt declarator requires-clause function-body function-body: ctor-initializeropt compound-statement function-try-block = default ; deleted-function-body deleted-function-body: = delete ; = delete (unevaluated-stringdiagnostic-message ) ;
6.3.2. 9.5.3 Deleted definitions [dcl.fct.def.delete]
Paragraph 2:
A program that refers to a deleted function implicitly or explicitly, other than to declare it, is ill-formed.
Recommended practice: The resulting diagnostic message should include the text of the
unevaluated-stringdiagnostic-message ([dcl.pre]) , if one is supplied.
[Note 1: This includes calling the function implicitly or explicitly and forming a pointer or pointer-to-member to the function. It applies even for references in expressions that are not potentially-evaluated. For an overload set, only the function selected by overload resolution is referenced. The implicit odr-use ([basic.def.odr]) of a virtual function does not, by itself, constitute a reference. The
unevaluated-stringdiagnostic-message, if present, can be used to explain the rationale for deletion and/or to suggest an alternative. — end note]
6.4. 9.12 Attributes [dcl.attr]
6.4.1. 9.12.1 Attribute syntax and semantics [dcl.attr.grammar]
Paragraph 5-6:
Each attribute-specifier-seq is said to appertain to some entity or statement, identified by the syntactic context where it appears ([stmt.stmt], [dcl.dcl], [dcl.decl]). If an attribute-specifier-seq that appertains to some entity or statement contains an attribute or alignment-specifier that is not allowed to apply to that entity or statement, the program is ill-formed. If an attribute-specifier-seq appertains to a friend declaration ([class.friend]), that declaration shall be a definition.
[Note 3: An attribute-specifier-seq cannot appertain to an explicit instantiation ([temp.explicit]). — end note]
The attribute-specifier-seqs appertaining to a function are considered to be needed ([temp.inst]) when
the function is odr-used ([basic.def.odr]), or
the function is defined.
The attribute-specifier-seqs appertaining to a templated function is instantiated only when needed.
For an attribute-token (including an attribute-scoped-token) not specified in this document, the behavior is implementation-defined; any such attribute-token that is not recognized by the implementation is ignored.
6.4.2. 9.12.5 Deprecated attribute [dcl.attr.deprecated]
Paragraph 1:
The attribute-token
can be used to mark names and entities whose use is still allowed, but is discouraged for some reason.
deprecated
[Note 1: In particular,
is appropriate for names and entities that are deemed obsolescent or unsafe. — end note]
deprecated
An attribute-argument-clause may be present and, if present, it shall have the form:
(unevaluated-stringdiagnostic-message )
[Note 2: The
unevaluated-stringdiagnostic-message in the attribute-argument-clause can be used to explain the rationale for deprecation and/or to suggest a replacing entity. — end note]
Paragraph 4:
Recommended practice: Implementations should use the
attribute to produce a diagnostic message in case the program refers to a name or entity other than to declare it, after a declaration that specifies the attribute. The diagnostic message should include the text of the diagnostic-message ([dcl.pre]) provided within the attribute-argument-clause of any
deprecated attribute applied to the name or entity. The value of a has-attribute-expression for the
deprecated attribute should be
deprecated unless the implementation can issue such diagnostic messages.
0
6.4.3. 9.12.10 Nodiscard attribute [dcl.attr.nodiscard]
Paragraph 1:
The attribute-token
may be applied to a function or a lambda call operator or to the declaration of a class or enumeration. An attribute-argument-clause may be present and, if present, it shall have the form:
nodiscard
(unevaluated-stringdiagnostic-message )
Paragraph 4:
Recommended practice: Appearance of a
call as a potentially-evaluated discarded-value expression ([expr.prop]) is discouraged unless explicitly cast to
nodiscard . Implementations should issue a warning in such cases. The value of a has-attribute-expression for the
void attribute should be
nodiscard unless the implementation can issue such warnings.
0
[Note 2: This is typically because discarding the return value of a
call has surprising consequences. — end note]
nodiscard
The
unevaluated-stringtext of the diagnostic-message ([dcl.pre]) in aattribute-argument-clause should be used in the message of the warning as the rationale for why the result should not be discarded.
nodiscard
6.5. 13.9.2 Implicit instantiation [temp.inst]
Paragraph 14:
The noexcept-specifier and attribute-specifier-seqs of a function template specialization
isare not instantiated along with the function declaration;it isthey are instantiated when needed ([except.spec] , [dcl.attr.grammar] ). If such anoexcept-specifierspecifier is needed but has not yet been instantiated, the dependent names are looked up, the semantics constraints are checked, and the instantiation of any template used in thenoexcept-specifierspecifier is done as if it were being done as part of instantiating the declaration of the specialization at that point.
6.6. 13.10.3 Template argument deduction [temp.deduct]
6.6.1. 13.10.3.1 General [temp.deduct.general]
Paragraph 7:
[Note 4: The equivalent substitution in exception and attribute specifications is done only when the noexcept-specifier or attribute-specifier-seq, respectively, is instantiated, at which point a program is ill-formed if the substitution results in an invalid type or expression. — end note]
6.7. 15.2 Conditional inclusion [cpp.cond]
In [tab:cpp.cond.ha], modify the rows as indicated, and substituting
by the date of adoption.
Attribute | Value |
|
|
|
|
6.8. 15.11 Predefined macro names [cpp.predefined]
In [tab:cpp.predefined.ft], modify the rows as indicated, and substituting
by the date of adoption.
Macro name | Value |
|
|
7. Poll Results
7.1. EWGI Review, Wrocław (2024-11)
Poll: [P3423R0] Extending User-Generated Diagnostic Messages: Forward to EWG.
Strongly Favor | Favor | Neutral | Against | Strongly Against |
1 | 8 | 0 | 0 | 0 |
-
Outcome: Consensus.