-
Notifications
You must be signed in to change notification settings - Fork 769
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
[class.spaceship] lacks to specify the behavior if the defined expression for synthesized three-way comparison is ill-formed #5338
Comments
Isn't this the case for |
Thanks, that paper has listed a large of particular cases that can expose the problem to which this issue is concerning. From [class.spaceship.note] p1, the standard prefers to consider these synthesized three-way comparisons to be ill-formed, thus I agree with the opinion thereof in that paper: Erring on the side of ill-formed helps programmers catch bugs earlier; rather than just delete the defaulted function instead. |
The proposal in that paper says:
Conversely, the current draft only talks about "is not defined". For the example given in this issue, it has a defined synthesized three-way comparison, merely the defined expression is ill-formed. The case falls into the "would be ill-formed", however, the current draft lacks to specify such a condition in the formal rules. Improvement:
|
Not an improvement. "If ill-formed, then deleted" is too far-reaching to be implementable. See CWG2432. |
The other opinion is, whether the defaulted function is defined as deleted, which is only determined by whether the synthesized three-way comparison is defined; This keeps the same as the current draft did. However, if once the defaulted function is odr-used or its definition is required, the prior synthesized three-way comparison is rechecked to determine whether the program is well-formed as if the synthesized three-way comparison would appear in the definition of the defaulted function. This is consistent with what Clang has done. |
And that's consistent with what we do for defaulted constructors: In a first phase, we check whether the default constructor is deleted; in a second phase (when the definition is needed), we compile the actual definition (which might end up being ill-formed). And this seems to be the intent here, too, and seems to be supported by the existing wording, so maybe it's enough to add a note. |
@jensmaurer For default constructor or other special member functions, we explicitly define the condition that the defaulted default constructor should be defined as deleted. For ill-formed cases, I think they are regulated by [class.base.init] together with [dcl.init]. Such as the default initialization of the member of reference type. we explicitly state such a case will be ill-formed through [dcl.init.general] p10. The difference here is that we didn't explicitly phrase what will happen when evaluating the defaulted three-way comparison except that we may intend to imply this point by specifying the return value of the comparison in [class.spaceship] p3. Maybe, we could explicitly state the "ill-formed" story in [class.spaceship] p3:
It seems that the rule intends to say the evaluation of the default comparison will evaluate the synthesized comparison of the subobjects. However, it's not explicit. Maybe, we could say: if any synthesized three-way comparison of type R between xi and yi results in an invalid expression, the program is ill-formed. I think this sentence is close to what the implementation did. |
The point is still that the constructor descriptions rely on specifications elsewhere to get to the ill-formed cases. I'm not seeing why we need to be normatively different for the comparison operators. I do agree that the "deleted" and "use this expression" cases are a bit more interwoven for the comparison functions. |
@jensmaurer I think we just need to specify what expressions are in the implicit definition of a defaulted comparison operator function, all the issues mentioned above will be resolved. Consider [except.spec] p11
we seem to haven't yet defined the concept for what expressions are in the implicit definition. For a defaulted three-way comparison, the expressions presumably are these synthesized three-way comparisons for the subobjects in the object x of class C of which the defaulted function is a member, which are used to determine the return value. The same is true of defaulted equality operator function. For these defaulted functions, we just specify their return values, we never explicitly phrase that these expressions that are used to determine the return value appear within the implicit definitions. So, I think we lack to define the concept for what expressions are in the implicit definition of a defaulted function. This can clarify [except.spec] p11, and it can naturally interpret why the invocation of the defaulted function or the context where the definition is required will cause the program ill-formed if any expression in the implicit definition is invalid. |
Consider this example:
GCC:
GCC and Clang have a divergence implementation on the point that if the defined expression is ill-formed. Specifically, the expanded list of subobjects for an object x of type C consists of {C::a}, where the synthesized three-way comparison is defined as
static_cast<int>(a<=>a)
as per [class.spaceship] p1.1. Since the operand of the operator<=>
in the defined expression has typeint
, built-in expression interpretation applies to the operator, its type is std::strong_ordering whose prvalue cannot be converted to typeint
. Thus, the defined expressionstatic_cast<int>(a<=>a)
is ill-formed, on this point, GCC thinks the defaulted three-way comparison function should be deleted while Clang thinks it's ok(not be deleted). However, [class.spaceship.note] p1 wants to imply something regarding this case. However, it's not a formal rule and no overload resolution performs in this case too.The text was updated successfully, but these errors were encountered: