diff options
| author | Anders Carlsson <andersca@mac.com> | 2009-10-03 15:13:22 +0000 |
|---|---|---|
| committer | Anders Carlsson <andersca@mac.com> | 2009-10-03 15:13:22 +0000 |
| commit | 3162e4945a6537356178fe409833dab927b74539 (patch) | |
| tree | a65eb7000d3cddde7346a05c304081f53446ea7f /clang/lib/CodeGen | |
| parent | 128a5d51478da2509c16c1bafdeabfffed82135c (diff) | |
| download | bcm5719-llvm-3162e4945a6537356178fe409833dab927b74539.tar.gz bcm5719-llvm-3162e4945a6537356178fe409833dab927b74539.zip | |
Handle base-to-derived casts of member function pointers in CGExprConstant.cpp
llvm-svn: 83265
Diffstat (limited to 'clang/lib/CodeGen')
| -rw-r--r-- | clang/lib/CodeGen/CGExprConstant.cpp | 37 |
1 files changed, 37 insertions, 0 deletions
diff --git a/clang/lib/CodeGen/CGExprConstant.cpp b/clang/lib/CodeGen/CGExprConstant.cpp index 144fa28e19b..a742355f74d 100644 --- a/clang/lib/CodeGen/CGExprConstant.cpp +++ b/clang/lib/CodeGen/CGExprConstant.cpp @@ -485,6 +485,43 @@ public: } case CastExpr::CK_NullToMemberPointer: return CGM.EmitNullConstant(E->getType()); + + case CastExpr::CK_BaseToDerivedMemberPointer: { + Expr *SubExpr = E->getSubExpr(); + + const MemberPointerType *SrcTy = + SubExpr->getType()->getAs<MemberPointerType>(); + const MemberPointerType *DestTy = + E->getType()->getAs<MemberPointerType>(); + + const CXXRecordDecl *BaseClass = + cast<CXXRecordDecl>(cast<RecordType>(SrcTy->getClass())->getDecl()); + const CXXRecordDecl *DerivedClass = + cast<CXXRecordDecl>(cast<RecordType>(DestTy->getClass())->getDecl()); + + if (SrcTy->getPointeeType()->isFunctionProtoType()) { + llvm::Constant *C = + CGM.EmitConstantExpr(SubExpr, SubExpr->getType(), CGF); + if (!C) + return 0; + + llvm::ConstantStruct *CS = cast<llvm::ConstantStruct>(C); + + // Check if we need to update the adjustment. + if (llvm::Constant *Offset = CGM.GetCXXBaseClassOffset(DerivedClass, + BaseClass)) { + llvm::Constant *Values[2]; + + Values[0] = CS->getOperand(0); + Values[1] = llvm::ConstantExpr::getAdd(CS->getOperand(1), Offset); + return llvm::ConstantStruct::get(CGM.getLLVMContext(), Values, 2, + /*Packed=*/false); + } + + return CS; + } + } + default: { // FIXME: This should be handled by the CK_NoOp cast kind. // Explicit and implicit no-op casts |

