forked from OSchip/llvm-project
Catch the case of trying to turn '&(X::a)' into a member pointer as well.
llvm-svn: 111997
This commit is contained in:
parent
98f0ea6ade
commit
7d46051eea
|
@ -6216,9 +6216,24 @@ Sema::ResolveAddressOfOverloadedFunction(Expr *From, QualType ToType,
|
||||||
// C++ [over.over]p1:
|
// C++ [over.over]p1:
|
||||||
// [...] The overloaded function name can be preceded by the &
|
// [...] The overloaded function name can be preceded by the &
|
||||||
// operator.
|
// operator.
|
||||||
llvm::PointerIntPair<OverloadExpr*,1> Ovl = OverloadExpr::find(From);
|
// However, remember whether the expression has member-pointer form:
|
||||||
OverloadExpr *OvlExpr = Ovl.getPointer();
|
// C++ [expr.unary.op]p4:
|
||||||
|
// A pointer to member is only formed when an explicit & is used
|
||||||
|
// and its operand is a qualified-id not enclosed in
|
||||||
|
// parentheses.
|
||||||
|
bool HasFormOfMemberPointer = false;
|
||||||
|
OverloadExpr *OvlExpr;
|
||||||
|
{
|
||||||
|
Expr *Tmp = From->IgnoreParens();
|
||||||
|
if (isa<UnaryOperator>(Tmp)) {
|
||||||
|
Tmp = cast<UnaryOperator>(Tmp)->getSubExpr();
|
||||||
|
OvlExpr = cast<OverloadExpr>(Tmp->IgnoreParens());
|
||||||
|
HasFormOfMemberPointer = (Tmp == OvlExpr && OvlExpr->getQualifier());
|
||||||
|
} else {
|
||||||
|
OvlExpr = cast<OverloadExpr>(Tmp);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// We expect a pointer or reference to function, or a function pointer.
|
// We expect a pointer or reference to function, or a function pointer.
|
||||||
FunctionType = Context.getCanonicalType(FunctionType).getUnqualifiedType();
|
FunctionType = Context.getCanonicalType(FunctionType).getUnqualifiedType();
|
||||||
if (!FunctionType->isFunctionType()) {
|
if (!FunctionType->isFunctionType()) {
|
||||||
|
@ -6230,13 +6245,8 @@ Sema::ResolveAddressOfOverloadedFunction(Expr *From, QualType ToType,
|
||||||
}
|
}
|
||||||
|
|
||||||
// If the overload expression doesn't have the form of a pointer to
|
// If the overload expression doesn't have the form of a pointer to
|
||||||
// member, don't try to convert it to a pointer-to-member type:
|
// member, don't try to convert it to a pointer-to-member type.
|
||||||
// C++ [expr.unary.op]p4:
|
if (IsMember && !HasFormOfMemberPointer) {
|
||||||
// A pointer to member is only formed when an explicit & is used
|
|
||||||
// and its operand is a qualified-id not enclosed in
|
|
||||||
// parentheses.
|
|
||||||
// We don't diagnose the parentheses here, though. Should we?
|
|
||||||
if (IsMember && !(Ovl.getInt() && OvlExpr->getQualifier())) {
|
|
||||||
if (!Complain) return 0;
|
if (!Complain) return 0;
|
||||||
|
|
||||||
// TODO: Should we condition this on whether any functions might
|
// TODO: Should we condition this on whether any functions might
|
||||||
|
|
|
@ -91,28 +91,3 @@ namespace test2 {
|
||||||
a.test3(); // expected-note {{in instantiation}}
|
a.test3(); // expected-note {{in instantiation}}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
namespace test3 {
|
|
||||||
struct A {
|
|
||||||
void foo(void (A::*)(int)); // expected-note {{passing argument to parameter here}}
|
|
||||||
template<typename T> void g(T);
|
|
||||||
|
|
||||||
void test() {
|
|
||||||
foo(&g<int>); // expected-error {{cannot initialize a parameter}}
|
|
||||||
}
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
// This should succeed.
|
|
||||||
namespace test4 {
|
|
||||||
struct A {
|
|
||||||
static void f(void (A::*)());
|
|
||||||
static void f(void (*)(int));
|
|
||||||
void g();
|
|
||||||
static void g(int);
|
|
||||||
|
|
||||||
void test() {
|
|
||||||
f(&g);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
|
@ -0,0 +1,45 @@
|
||||||
|
// RUN: %clang_cc1 -fsyntax-only -verify %s
|
||||||
|
|
||||||
|
// rdar://problem/8347416
|
||||||
|
namespace test0 {
|
||||||
|
struct A {
|
||||||
|
void foo(void (A::*)(int)); // expected-note {{passing argument to parameter here}}
|
||||||
|
template<typename T> void g(T);
|
||||||
|
|
||||||
|
void test() {
|
||||||
|
// FIXME: this diagnostic is terrible
|
||||||
|
foo(&g<int>); // expected-error {{cannot initialize a parameter of type 'void (test0::A::*)(int)' with an rvalue of type '<overloaded function type>'}}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
// This should succeed.
|
||||||
|
namespace test1 {
|
||||||
|
struct A {
|
||||||
|
static void f(void (A::*)());
|
||||||
|
static void f(void (*)(int));
|
||||||
|
void g();
|
||||||
|
static void g(int);
|
||||||
|
|
||||||
|
void test() {
|
||||||
|
f(&g);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
// Also rdar://problem/8347416
|
||||||
|
namespace test2 {
|
||||||
|
struct A {
|
||||||
|
static int foo(short);
|
||||||
|
static int foo(float);
|
||||||
|
int foo(int);
|
||||||
|
int foo(double);
|
||||||
|
|
||||||
|
void test();
|
||||||
|
};
|
||||||
|
|
||||||
|
void A::test() {
|
||||||
|
// FIXME: This diagnostic is terrible.
|
||||||
|
int (A::*ptr)(int) = &(A::foo); // expected-error {{cannot initialize a variable of type 'int (test2::A::*)(int)' with an rvalue of type '<overloaded function type>'}}
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in New Issue