Skip to content

Commit

Permalink
[Clang] Improve diagnostics when using a concept as template argument
Browse files Browse the repository at this point in the history
When using the name of a template variable or concept in places
where an expression was expected, Clang would drop the cxxscope token
preceeding it, if any.

This leads to subpar diagnostics - complaining about the
identifier being undeclared as clang would not know to look into a
non-global scope.

We make sure the scope is preserved.

When encountering `ns::Concept foo x;`, Clang would also fail
to provide the same quality as it does at global scope.

Reviewed By: aaron.ballman, erichkeane

Differential Revision: https://reviews.llvm.org/D146719
  • Loading branch information
cor3ntin committed Mar 30, 2023
1 parent 5114843 commit abf4a8c
Show file tree
Hide file tree
Showing 4 changed files with 62 additions and 22 deletions.
4 changes: 3 additions & 1 deletion clang/docs/ReleaseNotes.rst
Original file line number Diff line number Diff line change
Expand Up @@ -253,6 +253,8 @@ Bug Fixes in This Version
to limit non-defined/non-member functions as well. Additionally, we now diagnose
requires on lambdas when not allowed, which we previously missed.
(`#61748 <https://github.com/llvm/llvm-project/issues/61748>`_)
- Fix confusing diagnostic for incorrect use of qualified concepts names.


Bug Fixes to Compiler Builtins
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
Expand Down Expand Up @@ -300,7 +302,7 @@ AMDGPU Support
undefined symbols in the created module to be a linker error. To prevent this,
pass ``-Wl,--undefined`` if compiling directly, or ``-Xoffload-linker
--undefined`` if using an offloading language.
- The deprecated ``-mcode-object-v3`` and ``-mno-code-object-v3`` command-line
- The deprecated ``-mcode-object-v3`` and ``-mno-code-object-v3`` command-line
options have been removed.

X86 Support
Expand Down
10 changes: 5 additions & 5 deletions clang/lib/Parse/ParseDecl.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -3448,12 +3448,12 @@ void Parser::ParseDeclarationSpecifiers(
continue;
}

if (TemplateId && TemplateId->Kind == TNK_Concept_template &&
GetLookAheadToken(2).isOneOf(tok::kw_auto, tok::kw_decltype)) {
if (TemplateId && TemplateId->Kind == TNK_Concept_template) {
DS.getTypeSpecScope() = SS;
// This is a qualified placeholder-specifier, e.g., ::C<int> auto ...
// Consume the scope annotation and continue to consume the template-id
// as a placeholder-specifier.
// This is probably a qualified placeholder-specifier, e.g., ::C<int>
// auto ... Consume the scope annotation and continue to consume the
// template-id as a placeholder-specifier. Let the next iteration
// diagnose a missing auto.
ConsumeAnnotationToken();
continue;
}
Expand Down
26 changes: 10 additions & 16 deletions clang/lib/Parse/Parser.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -1883,31 +1883,25 @@ Parser::TryAnnotateName(CorrectionCandidateCallback *CCC,
return ANK_TemplateName;
}
[[fallthrough]];
case Sema::NC_Concept:
case Sema::NC_VarTemplate:
case Sema::NC_FunctionTemplate:
case Sema::NC_UndeclaredTemplate: {
// We have a type, variable or function template followed by '<'.
ConsumeToken();
UnqualifiedId Id;
Id.setIdentifier(Name, NameLoc);
if (AnnotateTemplateIdToken(
TemplateTy::make(Classification.getTemplateName()),
Classification.getTemplateNameKind(), SS, SourceLocation(), Id))
return ANK_Error;
return ANK_Success;
}
case Sema::NC_Concept: {
UnqualifiedId Id;
Id.setIdentifier(Name, NameLoc);
bool IsConceptName = Classification.getKind() == Sema::NC_Concept;
// We have a template name followed by '<'. Consume the identifier token so
// we reach the '<' and annotate it.
if (Next.is(tok::less))
// We have a concept name followed by '<'. Consume the identifier token so
// we reach the '<' and annotate it.
ConsumeToken();
UnqualifiedId Id;
Id.setIdentifier(Name, NameLoc);
if (AnnotateTemplateIdToken(
TemplateTy::make(Classification.getTemplateName()),
Classification.getTemplateNameKind(), SS, SourceLocation(), Id,
/*AllowTypeAnnotation=*/false, /*TypeConstraint=*/true))
/*AllowTypeAnnotation=*/!IsConceptName,
/*TypeConstraint=*/IsConceptName))
return ANK_Error;
if (SS.isNotEmpty())
AnnotateScopeToken(SS, !WasScopeAnnotation);
return ANK_Success;
}
}
Expand Down
44 changes: 44 additions & 0 deletions clang/test/Parser/cxx-template-template-recovery.cpp
Original file line number Diff line number Diff line change
@@ -0,0 +1,44 @@
// RUN: %clang_cc1 -std=c++20 -verify -fsyntax-only %s

namespace a {
template <typename T>
concept C1 = true; // #C1

template <typename T>
auto V1 = true; // #V1

namespace b {
template <typename T>
concept C2 = true; // #C2
template <typename T>
auto V2 = true; // #V2
}
}

template <typename T>
concept C3 = true; // #C3
template <typename T>
auto V3 = true; // #V3
template <template <typename T> typename C>
constexpr bool test = true;

static_assert(test<a::C1>); // expected-error {{too few template arguments for concept 'C1'}} \
// expected-note@#C1 {{here}}
static_assert(test<a::b::C2>); // expected-error {{too few template arguments for concept 'C2'}} \
// expected-note@#C2 {{here}}
static_assert(test<C3>); // expected-error {{too few template arguments for concept 'C3'}} \
// expected-note@#C3 {{here}}

static_assert(test<a::V1>); // expected-error {{use of variable template 'V1' requires template arguments}} \
// expected-note@#V1 {{here}}
static_assert(test<a::b::V2>); // expected-error {{use of variable template 'V2' requires template arguments}} \
// expected-note@#V2 {{here}}
static_assert(test<V3>); // expected-error {{use of variable template 'V3' requires template arguments}} \
// expected-note@#V3 {{here}}


void f() {
C3 t1 = 0; // expected-error {{expected 'auto' or 'decltype(auto)' after concept name}}
a::C1 t2 = 0; // expected-error {{expected 'auto' or 'decltype(auto)' after concept name}}
a::b::C2 t3 = 0; // expected-error {{expected 'auto' or 'decltype(auto)' after concept name}}
}

0 comments on commit abf4a8c

Please sign in to comment.