Skip to content

Commit 32aa7d8

Browse files
authored
[clang] Fix CTAD not work for C++ explicit type conversion (functional annotation). (#75779)
This fixes #64347. The CTAD for an aggregate class is missing to handle the explicit type conversion case, e.g. `TemplateFooClass(1, 2);`. Per C++ expr.type.conv p1, the deduced type is the return type of the deduction guide selected by the CTAD for the reminder. In the deduction implementation `DeduceTemplateSpecializationFromInitializer`, the parenthesized express-list case relies on the `ParenListExpr` parameter (default is nullptr), the AST `ParenListExpr` node is not built for all variant initializer cases (`BuildCXXTypeConstructorExpr`, `BuildCXXNew` etc), thus the deduction doesn't perform for these cases. This patch fixes it by removing the `ParenListExpr` and using the `Inits` instead (which also simplifies the interface and implementation).
1 parent 0d3d445 commit 32aa7d8

File tree

5 files changed

+31
-7
lines changed

5 files changed

+31
-7
lines changed

clang/docs/ReleaseNotes.rst

+3
Original file line numberDiff line numberDiff line change
@@ -686,6 +686,9 @@ Bug Fixes in This Version
686686
- Fix an issue where clang doesn't respect detault template arguments that
687687
are added in a later redeclaration for CTAD.
688688
Fixes (#69987 <https://github.com/llvm/llvm-project/issues/69987>`_)
689+
- Fix an issue where CTAD fails for explicit type conversion.
690+
Fixes (#64347 <https://github.com/llvm/llvm-project/issues/64347>`_)
691+
689692

690693
Bug Fixes to Compiler Builtins
691694
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^

clang/include/clang/Sema/Sema.h

+1-2
Original file line numberDiff line numberDiff line change
@@ -9354,8 +9354,7 @@ class Sema final {
93549354

93559355
QualType DeduceTemplateSpecializationFromInitializer(
93569356
TypeSourceInfo *TInfo, const InitializedEntity &Entity,
9357-
const InitializationKind &Kind, MultiExprArg Init,
9358-
ParenListExpr *PL = nullptr);
9357+
const InitializationKind &Kind, MultiExprArg Init);
93599358

93609359
QualType deduceVarTypeFromInitializer(VarDecl *VDecl, DeclarationName Name,
93619360
QualType Type, TypeSourceInfo *TSI,

clang/lib/Sema/SemaDecl.cpp

+1-1
Original file line numberDiff line numberDiff line change
@@ -12954,7 +12954,7 @@ QualType Sema::deduceVarTypeFromInitializer(VarDecl *VDecl,
1295412954
// FIXME: Initialization should not be taking a mutable list of inits.
1295512955
SmallVector<Expr*, 8> InitsCopy(DeduceInits.begin(), DeduceInits.end());
1295612956
return DeduceTemplateSpecializationFromInitializer(TSI, Entity, Kind,
12957-
InitsCopy, PL);
12957+
InitsCopy);
1295812958
}
1295912959

1296012960
if (DirectInit) {

clang/lib/Sema/SemaInit.cpp

+7-4
Original file line numberDiff line numberDiff line change
@@ -10561,7 +10561,7 @@ static bool isOrIsDerivedFromSpecializationOf(CXXRecordDecl *RD,
1056110561

1056210562
QualType Sema::DeduceTemplateSpecializationFromInitializer(
1056310563
TypeSourceInfo *TSInfo, const InitializedEntity &Entity,
10564-
const InitializationKind &Kind, MultiExprArg Inits, ParenListExpr *PL) {
10564+
const InitializationKind &Kind, MultiExprArg Inits) {
1056510565
auto *DeducedTST = dyn_cast<DeducedTemplateSpecializationType>(
1056610566
TSInfo->getType()->getContainedDeducedType());
1056710567
assert(DeducedTST && "not a deduced template specialization type");
@@ -10792,9 +10792,12 @@ QualType Sema::DeduceTemplateSpecializationFromInitializer(
1079210792
if (getLangOpts().CPlusPlus20 && !HasAnyDeductionGuide) {
1079310793
if (ListInit && ListInit->getNumInits()) {
1079410794
SynthesizeAggrGuide(ListInit);
10795-
} else if (PL && PL->getNumExprs()) {
10796-
InitListExpr TempListInit(getASTContext(), PL->getLParenLoc(),
10797-
PL->exprs(), PL->getRParenLoc());
10795+
} else if (Inits.size()) { // parenthesized expression-list
10796+
// Inits are expressions inside the parentheses. We don't have
10797+
// the parentheses source locations, use the begin/end of Inits as the
10798+
// best heuristic.
10799+
InitListExpr TempListInit(getASTContext(), Inits.front()->getBeginLoc(),
10800+
Inits, Inits.back()->getEndLoc());
1079810801
SynthesizeAggrGuide(&TempListInit);
1079910802
}
1080010803
}

clang/test/SemaCXX/ctad.cpp

+19
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,19 @@
1+
// RUN: %clang_cc1 -fsyntax-only -verify -Wno-unused-value -std=c++20 %s
2+
// expected-no-diagnostics
3+
4+
namespace GH64347 {
5+
6+
template<typename X, typename Y> struct A { X x; Y y;};
7+
void test() {
8+
A(1, 2);
9+
new A(1, 2);
10+
}
11+
12+
template<A a>
13+
void f() { (void)a; }
14+
void k() {
15+
// Test CTAD works for non-type template arguments.
16+
f<A(0, 0)>();
17+
}
18+
19+
} // namespace GH64347

0 commit comments

Comments
 (0)