From: Anders Carlsson Date: Tue, 13 Oct 2009 17:41:28 +0000 (+0000) Subject: Don't assume that the LHS and RHS of a member pointer expression is a DeclRefExpr... X-Git-Url: https://granicus.if.org/sourcecode?a=commitdiff_plain;h=3eea6350412eb7db13f0e2791c118920b36130a4;p=clang Don't assume that the LHS and RHS of a member pointer expression is a DeclRefExpr. Fixes PR5177. git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@83986 91177308-0d34-0410-b5e6-96231b3b80d8 --- diff --git a/lib/CodeGen/CGCXX.cpp b/lib/CodeGen/CGCXX.cpp index ff56c8c85a..3960cf5186 100644 --- a/lib/CodeGen/CGCXX.cpp +++ b/lib/CodeGen/CGCXX.cpp @@ -278,10 +278,11 @@ RValue CodeGenFunction::EmitCXXMemberCallExpr(const CXXMemberCallExpr *CE) { RValue CodeGenFunction::EmitCXXMemberPointerCallExpr(const CXXMemberCallExpr *E) { const BinaryOperator *BO = cast(E->getCallee()); - const DeclRefExpr *BaseExpr = cast(BO->getLHS()); - const DeclRefExpr *MemFn = cast(BO->getRHS()); + const Expr *BaseExpr = BO->getLHS(); + const Expr *MemFnExpr = BO->getRHS(); - const MemberPointerType *MPT = MemFn->getType()->getAs(); + const MemberPointerType *MPT = + MemFnExpr->getType()->getAs(); const FunctionProtoType *FPT = MPT->getPointeeType()->getAs(); const CXXRecordDecl *RD = @@ -296,8 +297,8 @@ CodeGenFunction::EmitCXXMemberPointerCallExpr(const CXXMemberCallExpr *E) { // Get the member function pointer. llvm::Value *MemFnPtr = - CreateTempAlloca(ConvertType(MemFn->getType()), "mem.fn"); - EmitAggExpr(MemFn, MemFnPtr, /*VolatileDest=*/false); + CreateTempAlloca(ConvertType(MemFnExpr->getType()), "mem.fn"); + EmitAggExpr(MemFnExpr, MemFnPtr, /*VolatileDest=*/false); // Emit the 'this' pointer. llvm::Value *This; diff --git a/test/CodeGenCXX/member-function-pointers.cpp b/test/CodeGenCXX/member-function-pointers.cpp index 9727a9dabd..13f7de5a63 100644 --- a/test/CodeGenCXX/member-function-pointers.cpp +++ b/test/CodeGenCXX/member-function-pointers.cpp @@ -54,3 +54,20 @@ void f3(A *a, A &ar) { (a->*pa)(); (ar.*pa)(); } + +// PR5177 +namespace PR5177 { + struct A { + bool foo(int*) const; + } a; + + struct B1 { + bool (A::*pmf)(int*) const; + const A* pa; + + B1() : pmf(&A::foo), pa(&a) {} + bool operator()() const { return (pa->*pmf)(new int); } + }; + + void bar(B1 b2) { while (b2()) ; } +}