forked from OSchip/llvm-project
[Sema] Teach overload resolution about unaddressable functions.
Given an expression like `(&Foo)();`, we perform overload resolution as if we are calling `Foo` directly. This causes problems if `Foo` is a function that can't have its address taken. This patch teaches overload resolution to ignore functions that can't have their address taken in such cases. Differential Revision: http://reviews.llvm.org/D15590 llvm-svn: 257016
This commit is contained in:
parent
103d2381d6
commit
7204ed97dd
|
@ -570,8 +570,8 @@ namespace clang {
|
||||||
/// This conversion candidate is not viable because its result
|
/// This conversion candidate is not viable because its result
|
||||||
/// type is not implicitly convertible to the desired type.
|
/// type is not implicitly convertible to the desired type.
|
||||||
ovl_fail_bad_final_conversion,
|
ovl_fail_bad_final_conversion,
|
||||||
|
|
||||||
/// This conversion function template specialization candidate is not
|
/// This conversion function template specialization candidate is not
|
||||||
/// viable because the final conversion was not an exact match.
|
/// viable because the final conversion was not an exact match.
|
||||||
ovl_fail_final_conversion_not_exact,
|
ovl_fail_final_conversion_not_exact,
|
||||||
|
|
||||||
|
@ -582,7 +582,10 @@ namespace clang {
|
||||||
|
|
||||||
/// This candidate function was not viable because an enable_if
|
/// This candidate function was not viable because an enable_if
|
||||||
/// attribute disabled it.
|
/// attribute disabled it.
|
||||||
ovl_fail_enable_if
|
ovl_fail_enable_if,
|
||||||
|
|
||||||
|
/// This candidate was not viable because its address could not be taken.
|
||||||
|
ovl_fail_addr_not_available
|
||||||
};
|
};
|
||||||
|
|
||||||
/// OverloadCandidate - A single candidate in an overload set (C++ 13.3).
|
/// OverloadCandidate - A single candidate in an overload set (C++ 13.3).
|
||||||
|
|
|
@ -2548,7 +2548,8 @@ public:
|
||||||
MultiExprArg Args,
|
MultiExprArg Args,
|
||||||
SourceLocation RParenLoc,
|
SourceLocation RParenLoc,
|
||||||
Expr *ExecConfig,
|
Expr *ExecConfig,
|
||||||
bool AllowTypoCorrection=true);
|
bool AllowTypoCorrection=true,
|
||||||
|
bool CalleesAddressIsTaken=false);
|
||||||
|
|
||||||
bool buildOverloadedCallSet(Scope *S, Expr *Fn, UnresolvedLookupExpr *ULE,
|
bool buildOverloadedCallSet(Scope *S, Expr *Fn, UnresolvedLookupExpr *ULE,
|
||||||
MultiExprArg Args, SourceLocation RParenLoc,
|
MultiExprArg Args, SourceLocation RParenLoc,
|
||||||
|
|
|
@ -4935,7 +4935,9 @@ Sema::ActOnCallExpr(Scope *S, Expr *Fn, SourceLocation LParenLoc,
|
||||||
OverloadExpr *ovl = find.Expression;
|
OverloadExpr *ovl = find.Expression;
|
||||||
if (UnresolvedLookupExpr *ULE = dyn_cast<UnresolvedLookupExpr>(ovl))
|
if (UnresolvedLookupExpr *ULE = dyn_cast<UnresolvedLookupExpr>(ovl))
|
||||||
return BuildOverloadedCallExpr(S, Fn, ULE, LParenLoc, ArgExprs,
|
return BuildOverloadedCallExpr(S, Fn, ULE, LParenLoc, ArgExprs,
|
||||||
RParenLoc, ExecConfig);
|
RParenLoc, ExecConfig,
|
||||||
|
/*AllowTypoCorrection=*/true,
|
||||||
|
find.IsAddressOfOperand);
|
||||||
return BuildCallToMemberFunction(S, Fn, LParenLoc, ArgExprs, RParenLoc);
|
return BuildCallToMemberFunction(S, Fn, LParenLoc, ArgExprs, RParenLoc);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -4949,10 +4951,14 @@ Sema::ActOnCallExpr(Scope *S, Expr *Fn, SourceLocation LParenLoc,
|
||||||
|
|
||||||
Expr *NakedFn = Fn->IgnoreParens();
|
Expr *NakedFn = Fn->IgnoreParens();
|
||||||
|
|
||||||
|
bool CallingNDeclIndirectly = false;
|
||||||
NamedDecl *NDecl = nullptr;
|
NamedDecl *NDecl = nullptr;
|
||||||
if (UnaryOperator *UnOp = dyn_cast<UnaryOperator>(NakedFn))
|
if (UnaryOperator *UnOp = dyn_cast<UnaryOperator>(NakedFn)) {
|
||||||
if (UnOp->getOpcode() == UO_AddrOf)
|
if (UnOp->getOpcode() == UO_AddrOf) {
|
||||||
|
CallingNDeclIndirectly = true;
|
||||||
NakedFn = UnOp->getSubExpr()->IgnoreParens();
|
NakedFn = UnOp->getSubExpr()->IgnoreParens();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (isa<DeclRefExpr>(NakedFn)) {
|
if (isa<DeclRefExpr>(NakedFn)) {
|
||||||
NDecl = cast<DeclRefExpr>(NakedFn)->getDecl();
|
NDecl = cast<DeclRefExpr>(NakedFn)->getDecl();
|
||||||
|
@ -4974,6 +4980,11 @@ Sema::ActOnCallExpr(Scope *S, Expr *Fn, SourceLocation LParenLoc,
|
||||||
NDecl = cast<MemberExpr>(NakedFn)->getMemberDecl();
|
NDecl = cast<MemberExpr>(NakedFn)->getMemberDecl();
|
||||||
|
|
||||||
if (FunctionDecl *FD = dyn_cast_or_null<FunctionDecl>(NDecl)) {
|
if (FunctionDecl *FD = dyn_cast_or_null<FunctionDecl>(NDecl)) {
|
||||||
|
if (CallingNDeclIndirectly &&
|
||||||
|
!checkAddressOfFunctionIsAvailable(FD, /*Complain=*/true,
|
||||||
|
Fn->getLocStart()))
|
||||||
|
return ExprError();
|
||||||
|
|
||||||
if (FD->hasAttr<EnableIfAttr>()) {
|
if (FD->hasAttr<EnableIfAttr>()) {
|
||||||
if (const EnableIfAttr *Attr = CheckEnableIf(FD, ArgExprs, true)) {
|
if (const EnableIfAttr *Attr = CheckEnableIf(FD, ArgExprs, true)) {
|
||||||
Diag(Fn->getLocStart(),
|
Diag(Fn->getLocStart(),
|
||||||
|
|
|
@ -9643,6 +9643,13 @@ static void NoteFunctionCandidate(Sema &S, OverloadCandidate *Cand,
|
||||||
|
|
||||||
case ovl_fail_enable_if:
|
case ovl_fail_enable_if:
|
||||||
return DiagnoseFailedEnableIfAttr(S, Cand);
|
return DiagnoseFailedEnableIfAttr(S, Cand);
|
||||||
|
|
||||||
|
case ovl_fail_addr_not_available: {
|
||||||
|
bool Available = checkAddressOfCandidateIsAvailable(S, Cand->Function);
|
||||||
|
(void)Available;
|
||||||
|
assert(!Available);
|
||||||
|
break;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -11245,6 +11252,17 @@ static ExprResult FinishOverloadedCallExpr(Sema &SemaRef, Scope *S, Expr *Fn,
|
||||||
return ExprError();
|
return ExprError();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void markUnaddressableCandidatesUnviable(Sema &S,
|
||||||
|
OverloadCandidateSet &CS) {
|
||||||
|
for (auto I = CS.begin(), E = CS.end(); I != E; ++I) {
|
||||||
|
if (I->Viable &&
|
||||||
|
!S.checkAddressOfFunctionIsAvailable(I->Function, /*Complain=*/false)) {
|
||||||
|
I->Viable = false;
|
||||||
|
I->FailureKind = ovl_fail_addr_not_available;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/// BuildOverloadedCallExpr - Given the call expression that calls Fn
|
/// BuildOverloadedCallExpr - Given the call expression that calls Fn
|
||||||
/// (which eventually refers to the declaration Func) and the call
|
/// (which eventually refers to the declaration Func) and the call
|
||||||
/// arguments Args/NumArgs, attempt to resolve the function call down
|
/// arguments Args/NumArgs, attempt to resolve the function call down
|
||||||
|
@ -11257,7 +11275,8 @@ ExprResult Sema::BuildOverloadedCallExpr(Scope *S, Expr *Fn,
|
||||||
MultiExprArg Args,
|
MultiExprArg Args,
|
||||||
SourceLocation RParenLoc,
|
SourceLocation RParenLoc,
|
||||||
Expr *ExecConfig,
|
Expr *ExecConfig,
|
||||||
bool AllowTypoCorrection) {
|
bool AllowTypoCorrection,
|
||||||
|
bool CalleesAddressIsTaken) {
|
||||||
OverloadCandidateSet CandidateSet(Fn->getExprLoc(),
|
OverloadCandidateSet CandidateSet(Fn->getExprLoc(),
|
||||||
OverloadCandidateSet::CSK_Normal);
|
OverloadCandidateSet::CSK_Normal);
|
||||||
ExprResult result;
|
ExprResult result;
|
||||||
|
@ -11266,6 +11285,11 @@ ExprResult Sema::BuildOverloadedCallExpr(Scope *S, Expr *Fn,
|
||||||
&result))
|
&result))
|
||||||
return result;
|
return result;
|
||||||
|
|
||||||
|
// If the user handed us something like `(&Foo)(Bar)`, we need to ensure that
|
||||||
|
// functions that aren't addressible are considered unviable.
|
||||||
|
if (CalleesAddressIsTaken)
|
||||||
|
markUnaddressableCandidatesUnviable(*this, CandidateSet);
|
||||||
|
|
||||||
OverloadCandidateSet::iterator Best;
|
OverloadCandidateSet::iterator Best;
|
||||||
OverloadingResult OverloadResult =
|
OverloadingResult OverloadResult =
|
||||||
CandidateSet.BestViableFunction(*this, Fn->getLocStart(), Best);
|
CandidateSet.BestViableFunction(*this, Fn->getLocStart(), Best);
|
||||||
|
|
|
@ -25,3 +25,21 @@ void Lambdas(char *ptr) {
|
||||||
// CHECK-DAG: define internal i64 @"_ZZN7lambdas7LambdasEPcENK3$_1clEPvU17pass_object_size0"
|
// CHECK-DAG: define internal i64 @"_ZZN7lambdas7LambdasEPcENK3$_1clEPvU17pass_object_size0"
|
||||||
// CHECK-NOT: call i64 @llvm.objectsize
|
// CHECK-NOT: call i64 @llvm.objectsize
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// This is here instead of in Sema/ because we need to check to make sure the
|
||||||
|
// proper function is called. If it's not, we'll end up with assertion errors.
|
||||||
|
namespace addrof {
|
||||||
|
void OvlFoo(void *const __attribute__((pass_object_size(0)))) {}
|
||||||
|
void OvlFoo(int *const) {}
|
||||||
|
|
||||||
|
// CHECK: define void @_ZN6addrof4TestEv
|
||||||
|
void Test() {
|
||||||
|
// Treating parens-only calls as though they were direct is consistent with
|
||||||
|
// how we handle other implicitly unaddressable functions (e.g. builtins).
|
||||||
|
// CHECK: call void @_ZN6addrof6OvlFooEPvU17pass_object_size0
|
||||||
|
(OvlFoo)(nullptr);
|
||||||
|
|
||||||
|
// CHECK: call void @_ZN6addrof6OvlFooEPi
|
||||||
|
(&OvlFoo)(nullptr);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
@ -33,7 +33,7 @@ void TakeFnOvl(void (*)(int *)) overloaded;
|
||||||
|
|
||||||
void NotOverloaded(void *p PS(0));
|
void NotOverloaded(void *p PS(0));
|
||||||
void IsOverloaded(void *p PS(0)) overloaded;
|
void IsOverloaded(void *p PS(0)) overloaded;
|
||||||
void IsOverloaded(char *p) overloaded;
|
void IsOverloaded(char *p) overloaded; // char* inestead of void* is intentional
|
||||||
void FunctionPtrs() {
|
void FunctionPtrs() {
|
||||||
void (*p)(void *) = NotOverloaded; //expected-error{{cannot take address of function 'NotOverloaded' because parameter 1 has pass_object_size attribute}}
|
void (*p)(void *) = NotOverloaded; //expected-error{{cannot take address of function 'NotOverloaded' because parameter 1 has pass_object_size attribute}}
|
||||||
void (*p2)(void *) = &NotOverloaded; //expected-error{{cannot take address of function 'NotOverloaded' because parameter 1 has pass_object_size attribute}}
|
void (*p2)(void *) = &NotOverloaded; //expected-error{{cannot take address of function 'NotOverloaded' because parameter 1 has pass_object_size attribute}}
|
||||||
|
@ -49,4 +49,8 @@ void FunctionPtrs() {
|
||||||
|
|
||||||
TakeFnOvl(NotOverloaded); //expected-error{{cannot take address of function 'NotOverloaded' because parameter 1 has pass_object_size attribute}}
|
TakeFnOvl(NotOverloaded); //expected-error{{cannot take address of function 'NotOverloaded' because parameter 1 has pass_object_size attribute}}
|
||||||
TakeFnOvl(&NotOverloaded); //expected-error{{cannot take address of function 'NotOverloaded' because parameter 1 has pass_object_size attribute}}
|
TakeFnOvl(&NotOverloaded); //expected-error{{cannot take address of function 'NotOverloaded' because parameter 1 has pass_object_size attribute}}
|
||||||
|
|
||||||
|
int P;
|
||||||
|
(&NotOverloaded)(&P); //expected-error{{cannot take address of function 'NotOverloaded' because parameter 1 has pass_object_size attribute}}
|
||||||
|
(&IsOverloaded)(&P); //expected-error{{no matching function}} expected-note@35{{candidate address cannot be taken because parameter 1 has pass_object_size attribute}} expected-note@36{{candidate function not viable: no known conversion from 'int *' to 'char *' for 1st argument}}
|
||||||
}
|
}
|
||||||
|
|
|
@ -120,3 +120,17 @@ void Bar() {
|
||||||
(void)+[](void *const p __attribute__((pass_object_size(0)))) {}; //expected-error-re{{invalid argument type '(lambda at {{.*}})' to unary expression}}
|
(void)+[](void *const p __attribute__((pass_object_size(0)))) {}; //expected-error-re{{invalid argument type '(lambda at {{.*}})' to unary expression}}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
namespace ovlbug {
|
||||||
|
// Directly calling an address-of function expression (e.g. in (&foo)(args...))
|
||||||
|
// doesn't go through regular address-of-overload logic. This caused the above
|
||||||
|
// code to generate an ICE.
|
||||||
|
void DirectAddrOf(void *__attribute__((pass_object_size(0))));
|
||||||
|
void DirectAddrOfOvl(void *__attribute__((pass_object_size(0))));
|
||||||
|
void DirectAddrOfOvl(int *);
|
||||||
|
|
||||||
|
void Test() {
|
||||||
|
(&DirectAddrOf)(nullptr); //expected-error{{cannot take address of function 'DirectAddrOf' because parameter 1 has pass_object_size attribute}}
|
||||||
|
(&DirectAddrOfOvl)((char*)nullptr); //expected-error{{no matching function}} expected-note@129{{candidate address cannot be taken because parameter 1 has pass_object_size attribute}} expected-note@130{{candidate function not viable: no known conversion from 'char *' to 'int *' for 1st argument}}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
Loading…
Reference in New Issue