<table border="1" cellspacing="0" cellpadding="8">
<tr>
<th>Issue</th>
<td>
<a href=https://github.com/llvm/llvm-project/issues/92495>92495</a>
</td>
</tr>
<tr>
<th>Summary</th>
<td>
clang: calls redundant move-constructor when explicit constructor of a nested object is called in an aggregate
</td>
</tr>
<tr>
<th>Labels</th>
<td>
clang
</td>
</tr>
<tr>
<th>Assignees</th>
<td>
</td>
</tr>
<tr>
<th>Reporter</th>
<td>
samolisov
</td>
</tr>
</table>
<pre>
**To be short**
When clang compiles the following minimum demonstration with an aggregate with a base class where the base class has an explicit constructor and therefore double `{}` cannot be used to make the aggregate's instances, a code to create a temporary object of the base class is generated by clang and then the move constructor is invoked redundancy to move the temporary into its correct place within the aggregate. I compared the code with the one generated by MSVC and g++, they **do not** insert a call for the move constructor (but to be clear, I've found a comment in the corresponding code of clang what unsure me who does the correct thing in this situation).
The code:
```c++
#include <cstdio>
struct A {
/*explicit */A(const char *) { printf("A(const char *)\n"); }
A(const A &) { printf("A(const A &)\n"); }
A(A&&) { printf("A(A&&)\n"); }
A &operator=(const A &) { printf("operator=(const A&)\n"); return *this; }
A &operator=(A &&) { printf("operator=(A&&)\n"); return *this; }
~A() { printf("~A()\n"); }
};
struct B : A {};
B rvo() {
return B{{"Hi"}};
}
B norvo() {
return B{A{"Hi2"}};
}
A rvoa() {
return {"hi"};
}
A norvoa() {
return A{"hi2"};
}
int main() {
{
printf("rvo:\n");
B b = rvo();
A a = rvoa();
}
{
printf("no rvo:\n");
B b = norvo();
A a = norvoa();
}
return 0;
}
```
So, when an object of type `B` is created using `B{{"Hi"}}`, everything is fine and no move-constructor for `A` is invoked, but when the object of `B` is created using `B{A{"Hi2"}}`, the move constructor for `A` is invoked.
** Analysis **
The following AST was generated for the first case (`B{{"Hi"}}`):
```
|-FunctionDecl 0x28607286238 <line:14:1, line:16:1> line:14:3 used rvo 'B ()'
| `-CompoundStmt 0x28607286620 <col:9, line:16:1>
| `-ReturnStmt 0x28607286610 <line:15:3, col:18>
| `-ExprWithCleanups 0x286072865f8 <col:10, col:18> 'B':'B'
| `-CXXFunctionalCastExpr 0x286072865d0 <col:10, col:18> 'B':'B' functional cast to B <NoOp>
| `-CXXBindTemporaryExpr 0x286072865b0 <col:11, col:18> 'B':'B' (CXXTemporary 0x286072865b0)
| `-InitListExpr 0x28607286418 <col:11, col:18> 'B':'B'
| `-CXXConstructExpr 0x28607286568 <col:12, col:17> 'A':'A' 'void (const char *)' list
| `-ImplicitCastExpr 0x28607286460 <col:13> 'const char *' <ArrayToPointerDecay>
| `-StringLiteral 0x28607286368 <col:13> 'const char[3]' lvalue "Hi"
```
while the AST for the second case (`B{A{"Hi2"}}`) contains a `MaterializeTemporaryExpr` expression:
```
|-FunctionDecl 0x28607286658 <line:18:1, line:20:1> line:18:3 used norvo 'B ()'
| `-CompoundStmt 0x286072869c8 <col:11, line:20:1>
| `-ReturnStmt 0x286072869b8 <line:19:3, col:20>
| `-ExprWithCleanups 0x286072869a0 <col:10, col:20> 'B':'B'
| `-CXXFunctionalCastExpr 0x28607286978 <col:10, col:20> 'B':'B' functional cast to B <NoOp>
| `-CXXBindTemporaryExpr 0x28607286958 <col:11, col:20> 'B':'B' (CXXTemporary 0x28607286958)
| `-InitListExpr 0x286072868c0 <col:11, col:20> 'B':'B'
| `-CXXConstructExpr 0x28607286920 <col:12, col:19> 'A':'A' 'void (A &&)'
| `-MaterializeTemporaryExpr 0x28607286908 <col:12, col:19> 'A':'A' xvalue
| `-CXXBindTemporaryExpr 0x28607286858 <col:12, col:19> 'A':'A' (CXXTemporary 0x28607286858)
| `-CXXTemporaryObjectExpr 0x28607286818 <col:12, col:19> 'A':'A' 'void (const char *)' list
| `-ImplicitCastExpr 0x28607286800 <col:14> 'const char *' <ArrayToPointerDecay>
| `-StringLiteral 0x28607286798 <col:14> 'const char[4]' lvalue "Hi2"
```
After some debugging, I've found the following code in the [SemaInit.cpp](https://github.com/llvm/llvm-project/blob/main/clang/lib/Sema/SemaInit.cpp#L4264) file, which is responsible for adding a `QualificationConversion` step into the sequence of initialization actions:
```c++
// C++17 [dcl.init]p17:
// - If the initializer expression is a prvalue and the cv-unqualified
// version of the source type is the same class as the class of the
// destination, the initializer expression is used to initialize the
// destination object.
// Per DR (no number yet), this does not apply when initializing a base
// class or delegating to another constructor from a mem-initializer.
// ObjC++: Lambda captured by the block in the lambda to block conversion
// should avoid copy elision.
if (S.getLangOpts().CPlusPlus17 && !RequireActualConstructor &&
UnwrappedArgs.size() == 1 && UnwrappedArgs[0]->isPRValue() &&
S.Context.hasSameUnqualifiedType(UnwrappedArgs[0]->getType(), DestType)) {
// Convert qualifications if necessary.
Sequence.AddQualificationConversionStep(DestType, VK_PRValue);
...
return;
}
...
```
But this code works only for the first case because only in this case the `RequireActualConstructor` variable is set to `false`. In the second case, with an explicit constructor call, the variable is set to `true` and no-early return occurs, the rest of the `void TryConstructorInitialization()` function is executed what ends to the adding a `ConstructorInitialization` step into the sequence of initialization actions:
```c++
// Add the constructor initialization step. Any cv-qualification conversion is
// subsumed by the initialization.
Sequence.AddConstructorInitializationStep(
Best->FoundDecl, CtorDecl, DestArrayType, HadMultipleCandidates,
IsListInit | IsInitListCopy, AsInitializerList);
```
The `RequireActualConstructor` variable is set by the following expression:
```c++
bool RequireActualConstructor =
!(Entity.getKind() != InitializedEntity::EK_Base &&
Entity.getKind() != InitializedEntity::EK_Delegating &&
Entity.getKind() !=
InitializedEntity::EK_LambdaToBlockConversionBlockElement);
```
In this case, we have `Entity.getKind()` equals to `InitializedEntity::EK_Base`. When the decision for the `RequireActualConstructor` variable is changed (this comparison was excluded), the observed behavior is changed too and no calling of the move constructor is generated anymore.
This may look as a fix until we see the second part of the comment on line 4264:
```
// Per DR (no number yet), this does not apply when initializing a base
// class or delegating to another constructor from a mem-initializer.
```
I didn't manage to find the corresponding DR, as I see this is now (hm, now, I mean 7 years ago?) a part of the C++17 standard as well as all the subsequent versions and before changing anything in the code the question about legality of such change should be considered. But the difference in the behavior between clang and MSVC + gcc makes me wonder.
</pre>
<img width="1px" height="1px" alt="" src="http://email.email.llvm.org/o/eJzUWl1z2zaz_jXwzY41FGlJ1IUvKCmeepqe5MRum7sOSK4kNCDAAKBs9eL89jMLkhIpk3bct714M44tkeCzH1g8u1iQWyt2CvGWzVZstrnildtrc2t5oaWw-nCV6vx4y8KEhcmjhhTB7rVx9QUWbFjQ_P59jwoyydUOMl2UQqIFt0fYain1k1A7KIQSRVVAjoVW1hnuhFbwJNweuAK-2xnccYfNFUi5RQK0Fp72aNCjdS7uuaXn8LmUIhMOMg9aZU4b4Cqn4Qa32iDkukolApsHbLFiiw2bB5BxpbQjeyqLOTgNBf9WyzhpwsKFBaGs4ypDy8I1cMh0jjQ6M0i6cnBYlNpwcwSd_omZA7291FRY2KFCwx3mkB4bLzU6Kj-60AfsWSBI8kF_wxwM5pXKucqOXk0aSY-cBQvlNAhnIdPGkAql5FntR6H6Jk3g3k8PN-il1_Z4j9M3rbCv6i8Pv629pjsWrvzPmgYeoQ6AXIPSTTCQp9A48hGXErbaDFvGwjitHJmSkoOQGwK9Z-HiQNFSqdy7uShQOWj094bZUqtc-PjKkdxc-_Fpzx1UylYGoUB42mvIdRN8rUPIEbsaTFiwwlU--Fi4nHRj-LFxCIt6oU2B43-yxgn11TASKpNVjsCidWZdLjSLPnQfrG2GBCju_BUAFt6xMDlFrffdXcLC2PsIsj039cUlPQWlEcptWRizMBwcxWZrxcKQPkUroOiuBZ0HJ8DC-et47ZAxMI-W-DGjSOfb4ygkRpcUXtqwaPO2ioODh6QYdJVR5BOa4rflNhJ_QOiYXa9K_L_EQ73Ebm-M-Ig-RKuBGFoBi5Imkl6MWYE56LPARolGwZV_ZMXC8CdBEhebHsBJcoOk9BBWFyxp0cI34RJSjI-i1UD7Vq1REK_TOEzS4oSvAwnloOBCvQQ6f-zOFTkiSnoz1T4AsIIUWLQ5e753NwHe3uUvbl-o1VUEegooDT-kQ2fOBrXo-u8VPVp_BgMObCmw-8CDJuZ-ohzGVTf9HUufbVeUaIVtcmUOlSUW9jeGQnIeEBwe0BwbvrawFQp9_lF15rvuZhJKMWzulevkSwKhBPPU5tazYm_pNBTZtVaDmWxYfi-jNKkxUVwerbDwsm567NVIycMjPPFuvdDm0a0wRPxUVdBEvubE5Vj-aid1fX1XqYwy4AYzCcFzGM-DRRjPwyimZCaFohw4vaFfZH57Ye4vRB-gOyKqSyhz0MDCxQqaOAsXJ3Hkpeu1LkrK7g-ucB2R8zDw-VNLFiXLIWFnGPBAX3yUXsJMg67mM9KLwGrcadzDIZQPz6X5Xbj9WiJXVWk7WLNtfFZpGlzAeCPJuihpP_Xt_Pq19S6Xa24dSeqi58G70GF7QqPp95UTZYP1_-hP5YV76n-NFiuh8se2SLxUIu0qMX1bCRbG669fT3B9KJruF1rUetwr4T6KF064mcbvkj8EfzJ03a7JSyPnXSFhR8iiEZKchCS1kYuDFjkM1lnhAqSwrj_Z90Vdxw3M88286-KokXiJu6AxiTH8-Kg_a6Ecmg1m_Dg4sV27H5wRavdRODS8u4SjntEvxbLZKmKzjTfnwGVFdNIwyDjTP-2FrHcdRFAtJVnMtMovOWmMRJfEnY4LZSkrzYNfuEMjuBR_YS9KiU3xuTRoLZXof5fK5rMelcUXVBYGl1QWn6nM58u_Q2bL7EVYX8r7ITZbpj3ll302I6wfZ7MlH-EbD_OfstlyMcKVI-j_CpstZyNsMqbEOJstZ_H72SzORtj0TRe_g82W3UTZY7Plm2zW2e28mOGxddgVHYwR6YjoZ88sLyLptTmMZ--UMT6H8egcXrj69PAnXyJeajR9t0b_YPqIg-503_yT6ePtJLJYxq8KZ7PVzUASCV_PIsnWoQGrC4Qc02q3E2r3ovfT7xn6Vk_TBGKz1QMWnNbgJCtLLz_eO1daPwd3LLzbCbev0kmmCxbeSXlo_1yXRtMMs_AulTpl4V29D7zzLSQaJOgiwTd_TlLC6ONNOL-h_LUVEusdj8j2VPHXPSkrUok-KfLc96d8evvfikuxFZlvNK21OqDx-WwegHVY1k27Oot-r1BlvqMllHB-Kda9Ue6Z0v5QO6rpLN3Bur46XZDD8kxOCJTNNuV0cQI6DaZ_13BftyxP0tF0MjAZyqE09Tw3XUvIDteV-l7biPkAKkBjcdsQtboyGdabQ1G35ywv2h4pbxp2_kv9xCBojtYJ1XTv1m-o3bZ2zyN-BLfZME4uxn1GA5svtLiVBlUVKRo4ovNre103Fn3fUWkHvCzlsd6BnmTXkZFye6lAY7OBHCXuuKOBTgNX2u3R9HecRhfAocDiumP1paaf0j-bIGBRAh95keYcMl66ytRdXd-fljr71i4tWY9xurmcneO1D233upI5cM9zmS6PgFLQuJMOYksuepjs0H3kavepdLauoSbrz7Ky9J9C06cjYOH0C36vhMEkcxWX616f2GesUzMD4Ff1ZHhZYp6YnZ1Y8Re2nZxow6INTFvY3kA2WwVstrlm0QdhP3_5zWem5rmuiIfJWiuHz26y5_aBF_jrOb4fjyU9M4a7Q9eMaKJhg7a5srzoNJ2XqXexg-9dnrDkPoUZWsvNsfXpQ8MQkyTPR2jlwWHJwvgsdw2__fzHydp-W2gymfRahEMtoc6oCy5fVa6O9vroQJtvFrSSx6FORYoZryzW99v2u7_jCX0ejM0-0eSBG8GJWoUFi75SZPNgy6VFNg8mcK8udyGenJvzpMFzoYxL2dLGMLwzFaE3Padr5EYe276YzrLK2PZ5g_Z02MPmgV8Rj-bYMeK-x-ZNdMyDUwlMkvEZs8phXh9joMotNHmhm0zGQf_1dJLk7UFR53Cqj0sKTCBRR8oKvXjuEAkIe8klVWqr4sxIfdSh4B91QxP-XbJYoXW0Nu-oqqC9Ic3b2mnTfqa1UldOzYL5iee_VNKJUuKaq1zk3PlTvy7qvaXyn4QDVVb3tt0RrHV5JJDE3p-JmW50F99gVfT4_pXQOOxcJr29X-5Nbqq1hHHijVoOYOGUhfEH5YQ7EqH_LFR-os4pUe7Z2LweRvKj5MPPf6zqrkCHYP8ezuacFH8UrRkxjlmnxEe9omR3JlH_9YPEAtVr03bf4THPOAh7fvCTOKSU72fQqrANxbzuM09tv7e96xwzn15P5PqOSMn2XO3Qb0cavi5KboTVyreY8dkfXuanAgZBpxbNgVYk7vlB1MfQLYzTuu3EE43ShDTsN3R4fW5gc3UstMGLg1ZhoeBHkFp_o9qPw1Y8Q6WckORPi9jl9pKbE9W2J8Na-dYK-OL81SbRf08FN8gP95CLXLFw4aDgiu_86wdb0dbhvZPxzRf_loKF-8aFwr99oPQTWb0v6K7ST37TBQVyBQs4IjcW-E6z6I7WEO-5-7ybsI6rnJuc4J9QSj9tUtbzVKV13nFt1W99rKT1Cxg-hLz31LFzFN-8fUAfvldUf1OOSnXlgLwohTuSFrbK9k0QtuVnWoebyNFgPoG6IkHIxXaLxie_Bv8Uxym6Jzy9oEKq-TcbWLiCXZb5dz-sf3tAqxzN5Cq_jfJltORXeDtdTGfLaTwPo6v9LZ_GcbZcpOkivoljPp8uslmaznE7y4JZmm-vxG0YhDfBbLoIboLoJpoEyONZmi7zYIrBNozYTYAFF3JC-9KJNrsrYW2Ft8vwZjm7kjxFaf3LOGHYbE5DNttcmVu_j02rnWU3gRTW2TOCE07ibT08SvzytKfXRtzLgzMf1oMFkt4CB4WWFm5zbuaZTkrMyandF3WuKiNv373_9tZaFt55g_8_AAD__xeIy4A">