-
Notifications
You must be signed in to change notification settings - Fork 258
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
[BUG] Converting assignment generated from non-converting constructor #468
Comments
This comment was marked as resolved.
This comment was marked as resolved.
This comment was marked as resolved.
This comment was marked as resolved.
Besides the function body cases, in my own code I regularly use |
@JohelEGP - thanks! I think this is intended behavior though... given your example type:
The first example is:
This is what I intended... I considered both lines to be equivalently "explicit" syntaxes. My view is that both are explicitly asking
and with this version the My understanding of the danger of implicit conversions is when Cpp1 silently creates a temporary Does that make sense? |
Yes!
The use of the C++ standard term converting constructor in the wiki Now, how does one opt-out of this generation? |
Opt-out is via But I think what you're asking is maybe what if the user writes |
That doesn't work: https://cpp2.godbolt.org/z/zTbj843v7. |
Oh, I see -- you want an opt-out of even generating the assignment? Assuming you want to disable all mutation, not just assignment, one way is to make a If you want to allow mutating operations, and allow construction from
There's a bug here because I should not generate the converting assignment when it's user-written, but this would be the way to do it. But let me ask: Given
I'm trying to imagine a situation where that would be desirable... the reason why Cpp2 generates converting assignment from converting construction is that usually they should be symmetric and it's a surprise when Cpp1 class authors support construction from a different type but not assignment from that same type. Is there a use case you came across? |
The use case is unit/quantity types. |
Thanks. OK, so I see why you want to suppress assignment, and you'll soon be able to do that by writing a private converting assignment operator (that will work as soon as I fix the bug to not generate it if the user wrote it). FWIW, this works nicely if you want the UDL-like suffix syntax, which in Cpp2 works for both literals and non-literals (below, an
Note this way all the cases can use the same consistent syntax (though I'll look at fixing the double generation bug so you can opt out and make |
Thank you. |
[Edited to add "default or converting"] Ah, good point, I should address this phrase:
An explicit converting constructor is still a converting constructor, right? A converting constructor (and, now that we've generalized this with the other Issues in this area #375 #398 #450, any non-copy I've added a note to the
Does that help? |
If it's
then only an
Not really. |
Ah, I didn't notice that "converting constructor" was already a Cpp1 standard term of art. The Cpp1 term seems like it would be better named an "implicitly converting constructor." A Cpp1 In contrast, the Cpp1 standard uses the term "conversion function" whether it's implicit or explicit, then qualifies the term as needed, e.g., to say things like "a conversion operator may be explicit". So restricting "converting constructor" to mean only a non-explicit one seems like inconsistent terminology between converting constructors and conversion operators, as well as inconsistent in that explicit constructors also perform type conversions just in a subset of contexts. My inclination is to keep using the term consistently as I've been doing in Cpp2, but if it's confusing I may need to come up with a different term just to avoid the confusion. |
I actually noticed that term when writing #468 (comment) 2 days ago. |
The opt-out by commit eac30d8 turns out to be rather noisy.
And because the solution only looks at the parameter's type, |
An alternative could be to discard the signature: |
Thanks! Can you elaborate please...
Why isn't
Yes, I'd put it in the initializer. Is there a reason it can't be?
I tried making this code warn using But assuming it was that: Perhaps it'd be useful to support giving a parameter the name |
To prevent it being accidentally invoked because it exists.
None in this case.
See also #305 (comment), where by "discard a parameter's name" I meant not emitting it. |
Here's an example where it's not possible, and actually made me discover a bug in |
In that particular case, now I can just omit the initialization by assigning |
Title: Converting assignment generated from non-converting constructor.
Description:
Relevant extracts from https://github.com/hsutter/cppfront/wiki/Cpp2:-operator=,-this-&-that
always talk about generating converting assignment from a converting constructor,
but
cppfront
currently generate converting assignment from an explicit constructor.Minimal reproducer (https://cpp2.godbolt.org/z/zzK9qc4Gx):
Commands:
cppfront -clean-cpp1 main.cpp2 clang++17 -std=c++2b -stdlib=libc++ -lc++abi -pedantic-errors -Wall -Wextra -Wconversion -I . main.cpp
Expected result:
An error at:
x = 1;
Actual result and error:
Cpp2 lowered to Cpp1.
Output:
The text was updated successfully, but these errors were encountered: