diff options
author | Fariborz Jahanian <fjahanian@apple.com> | 2009-07-28 17:38:28 +0000 |
---|---|---|
committer | Fariborz Jahanian <fjahanian@apple.com> | 2009-07-28 17:38:28 +0000 |
commit | 9e809e7da2448c08aa11f15be4680226754678ce (patch) | |
tree | 80da38d384657efa4f61e14563f8296541f7999d /lib/CodeGen | |
parent | 227aad58cc46c15dfc3fee2244d4a56d8c003d8f (diff) |
More work toward data member access ir-gen.
git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@77332 91177308-0d34-0410-b5e6-96231b3b80d8
Diffstat (limited to 'lib/CodeGen')
-rw-r--r-- | lib/CodeGen/CGCXX.cpp | 36 | ||||
-rw-r--r-- | lib/CodeGen/CGExpr.cpp | 10 | ||||
-rw-r--r-- | lib/CodeGen/CodeGenFunction.h | 8 |
3 files changed, 54 insertions, 0 deletions
diff --git a/lib/CodeGen/CGCXX.cpp b/lib/CodeGen/CGCXX.cpp index dd32cad21b..a0aaa82b4b 100644 --- a/lib/CodeGen/CGCXX.cpp +++ b/lib/CodeGen/CGCXX.cpp @@ -155,6 +155,42 @@ llvm::Value *CodeGenFunction::LoadCXXThis() { return Builder.CreateLoad(LocalDeclMap[CXXThisDecl], "this"); } +llvm::Value *CodeGenFunction::AddressCXXOfBaseClass(llvm::Value *BaseValue, + CXXRecordDecl *ClassDecl, + CXXRecordDecl *BaseClassDecl) { + if (ClassDecl == BaseClassDecl) + return BaseValue; + + // Accessing a member of the base class. Must add delata to + // the load of 'this'. + // FIXME. Once type layout is complete, this will probably change. + const ASTRecordLayout &Layout = + getContext().getASTRecordLayout(ClassDecl); + llvm::Type *I8Ptr = VMContext.getPointerTypeUnqual(llvm::Type::Int8Ty); + unsigned Idx = 0; + for (CXXRecordDecl::base_class_const_iterator i = + ClassDecl->bases_begin(), + e = ClassDecl->bases_end(); i != e; ++i, ++Idx) { + if (!i->isVirtual()) { + const CXXRecordDecl *Base = + cast<CXXRecordDecl>(i->getType()->getAsRecordType()->getDecl()); + if (Base == BaseClassDecl) + break; + } + } + uint64_t Offset = Layout.getFieldOffset(Idx) / 8; + llvm::Value *OffsetVal = llvm::ConstantInt::get(llvm::Type::Int32Ty, Offset); + BaseValue = Builder.CreateBitCast(BaseValue, I8Ptr); + BaseValue = Builder.CreateGEP(BaseValue, OffsetVal, "add.ptr"); + QualType BTy = + getContext().getCanonicalType( + getContext().getTypeDeclType(BaseClassDecl)); + const llvm::Type *BasePtr = ConvertType(BTy); + BasePtr = VMContext.getPointerTypeUnqual(BasePtr); + BaseValue = Builder.CreateBitCast(BaseValue, BasePtr); + return BaseValue; +} + void CodeGenFunction::EmitCXXConstructorCall(const CXXConstructorDecl *D, CXXCtorType Type, diff --git a/lib/CodeGen/CGExpr.cpp b/lib/CodeGen/CGExpr.cpp index 48f69e4ca4..b0e091bd0a 100644 --- a/lib/CodeGen/CGExpr.cpp +++ b/lib/CodeGen/CGExpr.cpp @@ -993,6 +993,16 @@ LValue CodeGenFunction::EmitMemberExpr(const MemberExpr *E) { if (PTy->getPointeeType()->isUnionType()) isUnion = true; CVRQualifiers = PTy->getPointeeType().getCVRQualifiers(); + if (CXXThisExpr *ThisExpr = dyn_cast<CXXThisExpr>(BaseExpr)) { + QualType ClassTy = ThisExpr->getType(); + ClassTy = ClassTy->getPointeeType(); + CXXRecordDecl *ClassDecl = + cast<CXXRecordDecl>(ClassTy->getAsRecordType()->getDecl()); + FieldDecl *Field = dyn_cast<FieldDecl>(E->getMemberDecl()); + CXXRecordDecl *BaseClassDecl = + cast<CXXRecordDecl>(Field->getDeclContext()); + BaseValue = AddressCXXOfBaseClass(BaseValue, ClassDecl, BaseClassDecl); + } } else if (isa<ObjCPropertyRefExpr>(BaseExpr) || isa<ObjCKVCRefExpr>(BaseExpr)) { RValue RV = EmitObjCPropertyGet(BaseExpr); diff --git a/lib/CodeGen/CodeGenFunction.h b/lib/CodeGen/CodeGenFunction.h index 6e905e48ac..06265196f9 100644 --- a/lib/CodeGen/CodeGenFunction.h +++ b/lib/CodeGen/CodeGenFunction.h @@ -528,6 +528,14 @@ public: /// generating code for an C++ member function. llvm::Value *LoadCXXThis(); + /// AddressCXXOfBaseClass - This function will add the necessary delta + /// to the load of 'this' and returns address of the base class. + // FIXME. This currently only does a derived to non-virtual base conversion. + // Other kinds of conversions will come later. + llvm::Value *AddressCXXOfBaseClass(llvm::Value *ThisValue, + CXXRecordDecl *ClassDecl, + CXXRecordDecl *BaseClassDecl); + void EmitCXXConstructorCall(const CXXConstructorDecl *D, CXXCtorType Type, llvm::Value *This, CallExpr::const_arg_iterator ArgBeg, |