forked from OSchip/llvm-project
Add an extension to avoid an error when a global template has the same name as
a member template, and you try to call the member template with an explicit template argument. See PR7247 For example, this downgrades the error to a warning in: template<typename T> struct set{}; struct Value { template<typename T> void set(T value) { } }; void foo() { Value v; v.set<double>(3.2); // Warning here. } llvm-svn: 105518
This commit is contained in:
parent
0501b03242
commit
2f96e9f5c9
|
@ -386,6 +386,30 @@ by commenting them out.</p>
|
||||||
and <a href="">-Wbaz</a>.</p>
|
and <a href="">-Wbaz</a>.</p>
|
||||||
</dd>
|
</dd>
|
||||||
|
|
||||||
|
<!-- - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -->
|
||||||
|
<dt id="opt_Wambiguous-member-template"><b>-Wambiguous-member-template</b>:
|
||||||
|
Warn about unqualified uses of a member template whose name resolves
|
||||||
|
to another template at the location of the use.</dt>
|
||||||
|
<dd>This option, which defaults to on, enables a warning in the
|
||||||
|
following code:</p>
|
||||||
|
|
||||||
|
<pre>
|
||||||
|
template<typename T> struct set{};
|
||||||
|
template<typename T> struct trait { typedef const T& type; };
|
||||||
|
struct Value {
|
||||||
|
template<typename T> void set(typename trait<T>::type value) {}
|
||||||
|
};
|
||||||
|
void foo() {
|
||||||
|
Value v;
|
||||||
|
v.set<double>(3.2);
|
||||||
|
}
|
||||||
|
</pre>
|
||||||
|
|
||||||
|
<p>C++ [basic.lookup.classref] requires this to be an error, but,
|
||||||
|
because it's hard to work around, Clang downgrades it to a warning as
|
||||||
|
an extension.</p>
|
||||||
|
</dd>
|
||||||
|
|
||||||
</dl>
|
</dl>
|
||||||
|
|
||||||
<!-- ======================================================================= -->
|
<!-- ======================================================================= -->
|
||||||
|
|
|
@ -20,6 +20,7 @@ def Implicit : DiagGroup<"implicit", [
|
||||||
def : DiagGroup<"address">;
|
def : DiagGroup<"address">;
|
||||||
def AddressOfTemporary : DiagGroup<"address-of-temporary">;
|
def AddressOfTemporary : DiagGroup<"address-of-temporary">;
|
||||||
def : DiagGroup<"aggregate-return">;
|
def : DiagGroup<"aggregate-return">;
|
||||||
|
def AmbigMemberTemplate : DiagGroup<"ambiguous-member-template">;
|
||||||
def : DiagGroup<"attributes">;
|
def : DiagGroup<"attributes">;
|
||||||
def : DiagGroup<"bad-function-cast">;
|
def : DiagGroup<"bad-function-cast">;
|
||||||
def : DiagGroup<"c++-compat">;
|
def : DiagGroup<"c++-compat">;
|
||||||
|
|
|
@ -549,6 +549,9 @@ def err_dependent_nested_name_spec : Error<
|
||||||
"parameter">;
|
"parameter">;
|
||||||
def err_nested_name_member_ref_lookup_ambiguous : Error<
|
def err_nested_name_member_ref_lookup_ambiguous : Error<
|
||||||
"lookup of %0 in member access expression is ambiguous">;
|
"lookup of %0 in member access expression is ambiguous">;
|
||||||
|
def ext_nested_name_member_ref_lookup_ambiguous : ExtWarn<
|
||||||
|
"lookup of %0 in member access expression is ambiguous; using member of %1">,
|
||||||
|
InGroup<AmbigMemberTemplate>;
|
||||||
def note_ambig_member_ref_object_type : Note<
|
def note_ambig_member_ref_object_type : Note<
|
||||||
"lookup in the object type %0 refers here">;
|
"lookup in the object type %0 refers here">;
|
||||||
def note_ambig_member_ref_scope : Note<
|
def note_ambig_member_ref_scope : Note<
|
||||||
|
|
|
@ -311,8 +311,9 @@ void Sema::LookupTemplateName(LookupResult &Found,
|
||||||
Found.getFoundDecl()->getCanonicalDecl()
|
Found.getFoundDecl()->getCanonicalDecl()
|
||||||
!= FoundOuter.getFoundDecl()->getCanonicalDecl()) {
|
!= FoundOuter.getFoundDecl()->getCanonicalDecl()) {
|
||||||
Diag(Found.getNameLoc(),
|
Diag(Found.getNameLoc(),
|
||||||
diag::err_nested_name_member_ref_lookup_ambiguous)
|
diag::ext_nested_name_member_ref_lookup_ambiguous)
|
||||||
<< Found.getLookupName();
|
<< Found.getLookupName()
|
||||||
|
<< ObjectType;
|
||||||
Diag(Found.getRepresentativeDecl()->getLocation(),
|
Diag(Found.getRepresentativeDecl()->getLocation(),
|
||||||
diag::note_ambig_member_ref_object_type)
|
diag::note_ambig_member_ref_object_type)
|
||||||
<< ObjectType;
|
<< ObjectType;
|
||||||
|
|
|
@ -0,0 +1,46 @@
|
||||||
|
// RUN: %clang_cc1 -fsyntax-only -fdiagnostics-show-option -verify %s
|
||||||
|
|
||||||
|
// C++98 [basic.lookup.classref]p1:
|
||||||
|
// In a class member access expression (5.2.5), if the . or -> token is
|
||||||
|
// immediately followed by an identifier followed by a <, the identifier must
|
||||||
|
// be looked up to determine whether the < is the beginning of a template
|
||||||
|
// argument list (14.2) or a less-than operator. The identifier is first
|
||||||
|
// looked up in the class of the object expression. If the identifier is not
|
||||||
|
// found, it is then looked up in the context of the entire postfix-expression
|
||||||
|
// and shall name a class or function template. If the lookup in the class of
|
||||||
|
// the object expression finds a template, the name is also looked up in the
|
||||||
|
// context of the entire postfix-expression and
|
||||||
|
// -- if the name is not found, the name found in the class of the object
|
||||||
|
// expression is used, otherwise
|
||||||
|
// -- if the name is found in the context of the entire postfix-expression
|
||||||
|
// and does not name a class template, the name found in the class of the
|
||||||
|
// object expression is used, otherwise
|
||||||
|
// -- if the name found is a class template, it must refer to the same
|
||||||
|
// entity as the one found in the class of the object expression,
|
||||||
|
// otherwise the program is ill-formed.
|
||||||
|
|
||||||
|
// From PR 7247
|
||||||
|
template<typename T>
|
||||||
|
struct set{}; // expected-note{{lookup from the current scope refers here}}
|
||||||
|
struct Value {
|
||||||
|
template<typename T>
|
||||||
|
void set(T value) {} // expected-note{{lookup in the object type 'Value' refers here}}
|
||||||
|
|
||||||
|
void resolves_to_same() {
|
||||||
|
Value v;
|
||||||
|
v.set<double>(3.2);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
void resolves_to_different() {
|
||||||
|
{
|
||||||
|
Value v;
|
||||||
|
// The fact that the next line is a warning rather than an error is an
|
||||||
|
// extension.
|
||||||
|
v.set<double>(3.2); // expected-warning{{lookup of 'set' in member access expression is ambiguous; using member of 'Value' [-Wambiguous-member-template]}}
|
||||||
|
}
|
||||||
|
{
|
||||||
|
int set; // Non-template.
|
||||||
|
Value v;
|
||||||
|
v.set<double>(3.2);
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in New Issue