diff options
author | Anders Carlsson <andersca@mac.com> | 2009-09-29 02:09:01 +0000 |
---|---|---|
committer | Anders Carlsson <andersca@mac.com> | 2009-09-29 02:09:01 +0000 |
commit | b05a3e551b9a2f2f8df16c03a995853662b3895b (patch) | |
tree | 242286c2323bc94b816b919c76210f65f80a05e9 /clang/lib/CodeGen | |
parent | 1fb7ae9e3ca14afc6621f65d7a6b0909ddfa4a4d (diff) | |
download | bcm5719-llvm-b05a3e551b9a2f2f8df16c03a995853662b3895b.tar.gz bcm5719-llvm-b05a3e551b9a2f2f8df16c03a995853662b3895b.zip |
Improve support for member function pointers.
llvm-svn: 83039
Diffstat (limited to 'clang/lib/CodeGen')
-rw-r--r-- | clang/lib/CodeGen/CGExprAgg.cpp | 15 | ||||
-rw-r--r-- | clang/lib/CodeGen/CGExprConstant.cpp | 11 | ||||
-rw-r--r-- | clang/lib/CodeGen/CodeGenFunction.cpp | 3 |
3 files changed, 26 insertions, 3 deletions
diff --git a/clang/lib/CodeGen/CGExprAgg.cpp b/clang/lib/CodeGen/CGExprAgg.cpp index f710b036a64..3bc8c129746 100644 --- a/clang/lib/CodeGen/CGExprAgg.cpp +++ b/clang/lib/CodeGen/CGExprAgg.cpp @@ -198,6 +198,21 @@ void AggExprEmitter::VisitCastExpr(CastExpr *E) { "Implicit cast types must be compatible"); Visit(E->getSubExpr()); break; + + case CastExpr::CK_NullToMemberPointer: { + QualType T = E->getType(); + const llvm::Type *PtrDiffTy = + CGF.ConvertType(CGF.getContext().getPointerDiffType()); + + llvm::Value *NullValue = llvm::Constant::getNullValue(PtrDiffTy); + llvm::Value *Ptr = Builder.CreateStructGEP(DestPtr, 0, "ptr"); + Builder.CreateStore(NullValue, Ptr, VolatileDest); + + llvm::Value *Adj = Builder.CreateStructGEP(DestPtr, 1, "adj"); + Builder.CreateStore(NullValue, Adj, VolatileDest); + + break; + } } } diff --git a/clang/lib/CodeGen/CGExprConstant.cpp b/clang/lib/CodeGen/CGExprConstant.cpp index d1713626ef7..3f010b17b0b 100644 --- a/clang/lib/CodeGen/CGExprConstant.cpp +++ b/clang/lib/CodeGen/CGExprConstant.cpp @@ -785,6 +785,13 @@ llvm::Constant *CodeGenModule::EmitConstantExpr(const Expr *E, return C; } +static inline bool isDataMemberPointerType(QualType T) { + if (const MemberPointerType *MPT = T->getAs<MemberPointerType>()) + return !MPT->getPointeeType()->isFunctionType(); + + return false; +} + llvm::Constant *CodeGenModule::EmitNullConstant(QualType T) { // No need to check for member pointers when not compiling C++. if (!getContext().getLangOptions().CPlusPlus) @@ -795,7 +802,7 @@ llvm::Constant *CodeGenModule::EmitNullConstant(QualType T) { QualType ElementTy = CAT->getElementType(); // FIXME: Handle arrays of structs that contain member pointers. - if (Context.getBaseElementType(ElementTy)->isMemberPointerType()) { + if (isDataMemberPointerType(Context.getBaseElementType(ElementTy))) { llvm::Constant *Element = EmitNullConstant(ElementTy); uint64_t NumElements = CAT->getSize().getZExtValue(); std::vector<llvm::Constant *> Array(NumElements); @@ -821,7 +828,7 @@ llvm::Constant *CodeGenModule::EmitNullConstant(QualType T) { } // FIXME: Handle structs that contain member pointers. - if (T->isMemberPointerType()) + if (isDataMemberPointerType(T)) return llvm::Constant::getAllOnesValue(getTypes().ConvertTypeForMem(T)); return llvm::Constant::getNullValue(getTypes().ConvertTypeForMem(T)); diff --git a/clang/lib/CodeGen/CodeGenFunction.cpp b/clang/lib/CodeGen/CodeGenFunction.cpp index fff1a84f702..72009daef68 100644 --- a/clang/lib/CodeGen/CodeGenFunction.cpp +++ b/clang/lib/CodeGen/CodeGenFunction.cpp @@ -66,7 +66,8 @@ const llvm::Type *CodeGenFunction::ConvertType(QualType T) { } bool CodeGenFunction::hasAggregateLLVMType(QualType T) { - return T->isRecordType() || T->isArrayType() || T->isAnyComplexType(); + return T->isRecordType() || T->isArrayType() || T->isAnyComplexType() || + T->isMemberFunctionPointerType(); } void CodeGenFunction::EmitReturnBlock() { |