From: Anders Carlsson Date: Tue, 13 Oct 2009 22:43:21 +0000 (+0000) Subject: Check the return type of binary operators and the arrow operator. X-Git-Url: https://granicus.if.org/sourcecode?a=commitdiff_plain;h=15ea378083ee4d9e4838076e4a310177da3d46c4;p=clang Check the return type of binary operators and the arrow operator. git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@84043 91177308-0d34-0410-b5e6-96231b3b80d8 --- diff --git a/lib/Sema/SemaExprCXX.cpp b/lib/Sema/SemaExprCXX.cpp index 5a40fc83c5..8a67d47954 100644 --- a/lib/Sema/SemaExprCXX.cpp +++ b/lib/Sema/SemaExprCXX.cpp @@ -1993,7 +1993,7 @@ Sema::ActOnStartCXXMemberReference(Scope *S, ExprArg Base, SourceLocation OpLoc, CTypes.insert(Context.getCanonicalType(BaseType)); while (BaseType->isRecordType()) { - Base = BuildOverloadedArrowExpr(S, move(Base), BaseExpr->getExprLoc()); + Base = BuildOverloadedArrowExpr(S, move(Base), OpLoc); BaseExpr = (Expr*)Base.get(); if (BaseExpr == NULL) return ExprError(); diff --git a/lib/Sema/SemaOverload.cpp b/lib/Sema/SemaOverload.cpp index 7b79fa5f4a..8c697cc868 100644 --- a/lib/Sema/SemaOverload.cpp +++ b/lib/Sema/SemaOverload.cpp @@ -4698,9 +4698,16 @@ Sema::CreateOverloadedBinOp(SourceLocation OpLoc, OpLoc); UsualUnaryConversions(FnExpr); - Expr *CE = new (Context) CXXOperatorCallExpr(Context, Op, FnExpr, - Args, 2, ResultTy, OpLoc); - return MaybeBindToTemporary(CE); + ExprOwningPtr + TheCall(this, new (Context) CXXOperatorCallExpr(Context, Op, FnExpr, + Args, 2, ResultTy, + OpLoc)); + + if (CheckCallReturnType(FnDecl->getResultType(), OpLoc, TheCall.get(), + FnDecl)) + return ExprError(); + + return MaybeBindToTemporary(TheCall.release()); } else { // We matched a built-in operator. Convert the arguments, then // break out so that we will build the appropriate built-in @@ -5171,10 +5178,16 @@ Sema::BuildOverloadedArrowExpr(Scope *S, ExprArg BaseIn, SourceLocation OpLoc) { Expr *FnExpr = new (Context) DeclRefExpr(Method, Method->getType(), SourceLocation()); UsualUnaryConversions(FnExpr); - Base = new (Context) CXXOperatorCallExpr(Context, OO_Arrow, FnExpr, &Base, 1, - Method->getResultType().getNonReferenceType(), - Method->getLocation()); - return Owned(Base); + + QualType ResultTy = Method->getResultType().getNonReferenceType(); + ExprOwningPtr + TheCall(this, new (Context) CXXOperatorCallExpr(Context, OO_Arrow, FnExpr, + &Base, 1, ResultTy, OpLoc)); + + if (CheckCallReturnType(Method->getResultType(), OpLoc, TheCall.get(), + Method)) + return ExprError(); + return move(TheCall); } /// FixOverloadedFunctionReference - E is an expression that refers to diff --git a/test/SemaCXX/incomplete-call.cpp b/test/SemaCXX/incomplete-call.cpp index 4decd86483..6134189f0a 100644 --- a/test/SemaCXX/incomplete-call.cpp +++ b/test/SemaCXX/incomplete-call.cpp @@ -1,5 +1,5 @@ // RUN: clang-cc -fsyntax-only -verify %s -struct A; // expected-note 11 {{forward declaration of 'struct A'}} +struct A; // expected-note 13 {{forward declaration of 'struct A'}} A f(); // expected-note {{note: 'f' declared here}} @@ -10,6 +10,8 @@ struct B { A operator!(); // expected-note 2 {{'operator!' declared here}} A operator++(int); // expected-note {{'operator++' declared here}} A operator[](int); // expected-note {{'operator[]' declared here}} + A operator+(int); // expected-note {{'operator+' declared here}} + A operator->(); // expected-note {{'operator->' declared here}} }; void g() { @@ -31,4 +33,6 @@ void g() { b(); // expected-error {{calling 'operator()' with incomplete return type 'struct A'}} b++; // expected-error {{calling 'operator++' with incomplete return type 'struct A'}} b[0]; // expected-error {{calling 'operator[]' with incomplete return type 'struct A'}} + b + 1; // expected-error {{calling 'operator+' with incomplete return type 'struct A'}} + b->f(); // expected-error {{calling 'operator->' with incomplete return type 'struct A'}} }