diff options
Diffstat (limited to 'lib/CodeGen/CodeGenFunction.cpp')
-rw-r--r-- | lib/CodeGen/CodeGenFunction.cpp | 153 |
1 files changed, 20 insertions, 133 deletions
diff --git a/lib/CodeGen/CodeGenFunction.cpp b/lib/CodeGen/CodeGenFunction.cpp index c8ef8fcd33..ebc1ff500d 100644 --- a/lib/CodeGen/CodeGenFunction.cpp +++ b/lib/CodeGen/CodeGenFunction.cpp @@ -27,7 +27,10 @@ using namespace CodeGen; CodeGenFunction::CodeGenFunction(CodeGenModule &cgm) : CGM(cgm), Target(CGM.getContext().Target), SwitchInsn(NULL), - CaseRangeBlock(NULL) {} + CaseRangeBlock(NULL) { + LLVMIntTy = ConvertType(getContext().IntTy); + LLVMPointerWidth = Target.getPointerWidth(0); +} ASTContext &CodeGenFunction::getContext() const { return CGM.getContext(); @@ -51,82 +54,20 @@ const llvm::Type *CodeGenFunction::ConvertType(QualType T) { return CGM.getTypes().ConvertType(T); } -bool CodeGenFunction::hasAggregateLLVMType(QualType T) { - return !T->isRealType() && !T->isPointerLikeType() && - !T->isVoidType() && !T->isVectorType() && !T->isFunctionType(); +bool CodeGenFunction::isObjCPointerType(QualType T) { + // All Objective-C types are pointers. + return T->isObjCInterfaceType() || + T->isObjCQualifiedInterfaceType() || T->isObjCQualifiedIdType(); } -/// Generate an Objective-C method. An Objective-C method is a C function with -/// its pointer, name, and types registered in the class struture. -// FIXME: This method contains a lot of code copied and pasted from -// GenerateCode. This should be factored out. -void CodeGenFunction::GenerateObjCMethod(const ObjCMethodDecl *OMD) { - llvm::SmallVector<const llvm::Type *, 16> ParamTypes; - for (unsigned i=0 ; i<OMD->param_size() ; i++) { - ParamTypes.push_back(ConvertType(OMD->getParamDecl(i)->getType())); - } - std::string CategoryName = ""; - if (ObjCCategoryImplDecl *OCD = - dyn_cast<ObjCCategoryImplDecl>(OMD->getMethodContext())) { - CategoryName = OCD->getName(); - } - - CurFn =CGM.getObjCRuntime()->MethodPreamble( - OMD->getClassInterface()->getName(), - CategoryName, - OMD->getSelector().getName(), - ConvertType(OMD->getResultType()), - llvm::PointerType::getUnqual(llvm::Type::Int32Ty), - ParamTypes.begin(), - OMD->param_size(), - !OMD->isInstance(), - OMD->isVariadic()); - llvm::BasicBlock *EntryBB = llvm::BasicBlock::Create("entry", CurFn); - - // Create a marker to make it easy to insert allocas into the entryblock - // later. Don't create this with the builder, because we don't want it - // folded. - llvm::Value *Undef = llvm::UndefValue::get(llvm::Type::Int32Ty); - AllocaInsertPt = new llvm::BitCastInst(Undef, llvm::Type::Int32Ty, "allocapt", - EntryBB); - - FnRetTy = OMD->getResultType(); - - Builder.SetInsertPoint(EntryBB); - - // Emit allocs for param decls. Give the LLVM Argument nodes names. - llvm::Function::arg_iterator AI = CurFn->arg_begin(); - - // Name the struct return argument. - // FIXME: Probably should be in the runtime, or it will trample the other - // hidden arguments. - if (hasAggregateLLVMType(OMD->getResultType())) { - AI->setName("agg.result"); - ++AI; - } - - // Add implicit parameters to the decl map. - // TODO: Add something to AST to let the runtime specify the names and types - // of these. - llvm::Value *&DMEntry = LocalDeclMap[&(*OMD->getSelfDecl())]; - const llvm::Type *SelfTy = AI->getType(); - llvm::Value *DeclPtr = new llvm::AllocaInst(SelfTy, 0, "self.addr", - AllocaInsertPt); - - // Store the initial value into the alloca. - // FIXME: volatility - Builder.CreateStore(AI, DeclPtr); - DMEntry = DeclPtr; - ++AI; ++AI; - +bool CodeGenFunction::hasAggregateLLVMType(QualType T) { + return !isObjCPointerType(T) &&!T->isRealType() && !T->isPointerLikeType() && + !T->isVoidType() && !T->isVectorType() && !T->isFunctionType(); +} - for (unsigned i = 0, e = OMD->getNumParams(); i != e; ++i, ++AI) { - assert(AI != CurFn->arg_end() && "Argument mismatch!"); - EmitParmDecl(*OMD->getParamDecl(i), AI); - } - +void CodeGenFunction::GenerateFunction(const Stmt *Body) { // Emit the function body. - EmitStmt(OMD->getBody()); + EmitStmt(Body); // Emit a return for code that falls off the end. If insert point // is a dummy block with no predecessors then remove the block itself. @@ -134,6 +75,7 @@ void CodeGenFunction::GenerateObjCMethod(const ObjCMethodDecl *OMD) { if (isDummyBlock(BB)) BB->eraseFromParent(); else { + // FIXME: if this is C++ main, this should return 0. if (CurFn->getReturnType() == llvm::Type::VoidTy) Builder.CreateRetVoid(); else @@ -145,28 +87,14 @@ void CodeGenFunction::GenerateObjCMethod(const ObjCMethodDecl *OMD) { // Remove the AllocaInsertPt instruction, which is just a convenience for us. AllocaInsertPt->eraseFromParent(); AllocaInsertPt = 0; + // Verify that the function is well formed. - assert(!verifyFunction(*CurFn) && "Generated method is not well formed."); -} - -llvm::Value *CodeGenFunction::LoadObjCSelf(void) -{ - if(const ObjCMethodDecl *OMD = dyn_cast<ObjCMethodDecl>(CurFuncDecl)) { - llvm::Value *SelfPtr = LocalDeclMap[&(*OMD->getSelfDecl())]; - // FIXME: Volatility - return Builder.CreateLoad(SelfPtr, "self"); - } - return NULL; + assert(!verifyFunction(*CurFn) && "Generated function is not well formed."); } void CodeGenFunction::GenerateCode(const FunctionDecl *FD) { - LLVMIntTy = ConvertType(getContext().IntTy); - LLVMPointerWidth = static_cast<unsigned>( - getContext().getTypeSize(getContext().getPointerType(getContext().VoidTy))); - CurFuncDecl = FD; - FnRetTy = FD->getType()->getAsFunctionType()->getResultType(); - + FnRetTy = FD->getResultType(); CurFn = cast<llvm::Function>(CGM.GetAddrOfFunctionDecl(FD, true)); assert(CurFn->isDeclaration() && "Function already has body?"); @@ -180,16 +108,7 @@ void CodeGenFunction::GenerateCode(const FunctionDecl *FD) { EntryBB); Builder.SetInsertPoint(EntryBB); - - CGDebugInfo *DI = CGM.getDebugInfo(); - if (DI) { - CompoundStmt* body = cast<CompoundStmt>(CurFuncDecl->getBody()); - if (body->getLBracLoc().isValid()) { - DI->setLocation(body->getLBracLoc()); - } - DI->EmitFunctionStart(FD, CurFn, Builder); - } - + // Emit allocs for param decls. Give the LLVM Argument nodes names. llvm::Function::arg_iterator AI = CurFn->arg_begin(); @@ -203,39 +122,7 @@ void CodeGenFunction::GenerateCode(const FunctionDecl *FD) { assert(AI != CurFn->arg_end() && "Argument mismatch!"); EmitParmDecl(*FD->getParamDecl(i), AI); } - - // Emit the function body. - EmitStmt(FD->getBody()); - - if (DI) { - CompoundStmt* body = cast<CompoundStmt>(CurFuncDecl->getBody()); - if (body->getRBracLoc().isValid()) { - DI->setLocation(body->getRBracLoc()); - } - DI->EmitRegionEnd(CurFn, Builder); - } - - // Emit a return for code that falls off the end. If insert point - // is a dummy block with no predecessors then remove the block itself. - llvm::BasicBlock *BB = Builder.GetInsertBlock(); - if (isDummyBlock(BB)) - BB->eraseFromParent(); - else { - // FIXME: if this is C++ main, this should return 0. - if (CurFn->getReturnType() == llvm::Type::VoidTy) - Builder.CreateRetVoid(); - else - Builder.CreateRet(llvm::UndefValue::get(CurFn->getReturnType())); - } - assert(BreakContinueStack.empty() && - "mismatched push/pop in break/continue stack!"); - - // Remove the AllocaInsertPt instruction, which is just a convenience for us. - AllocaInsertPt->eraseFromParent(); - AllocaInsertPt = 0; - - // Verify that the function is well formed. - assert(!verifyFunction(*CurFn) && "Generated function is not well formed."); + GenerateFunction(FD->getBody()); } /// isDummyBlock - Return true if BB is an empty basic block |