blob: 3e58b22aafc8dcc3f5b1942b6f82c859676ff86b [file] [log] [blame]
Chris Lattnerf97fe382007-05-24 06:29:05 +00001//===--- ModuleBuilder.cpp - Emit LLVM Code from ASTs ---------------------===//
2//
3// The LLVM Compiler Infrastructure
4//
Chris Lattner5b12ab82007-12-29 19:59:25 +00005// This file is distributed under the University of Illinois Open Source
6// License. See LICENSE.TXT for details.
Chris Lattnerf97fe382007-05-24 06:29:05 +00007//
8//===----------------------------------------------------------------------===//
9//
10// This builds an AST and converts it to LLVM Code.
11//
12//===----------------------------------------------------------------------===//
13
14#include "clang/CodeGen/ModuleBuilder.h"
David Blaikie48ad6dc2013-07-13 21:08:14 +000015#include "CGDebugInfo.h"
Chandler Carruth5553d0d2014-01-07 11:51:46 +000016#include "CodeGenModule.h"
Chris Lattneradf1f512008-02-06 02:01:47 +000017#include "clang/AST/ASTContext.h"
Daniel Dunbar221fa942008-08-11 04:54:23 +000018#include "clang/AST/DeclObjC.h"
19#include "clang/AST/Expr.h"
Chris Lattneradf1f512008-02-06 02:01:47 +000020#include "clang/Basic/Diagnostic.h"
21#include "clang/Basic/TargetInfo.h"
Chandler Carruth3a022472012-12-04 09:13:33 +000022#include "clang/Frontend/CodeGenOptions.h"
23#include "llvm/ADT/OwningPtr.h"
Reid Klecknere43f0fe2013-05-08 13:44:39 +000024#include "llvm/ADT/StringRef.h"
Chandler Carruthffd55512013-01-02 11:45:17 +000025#include "llvm/IR/DataLayout.h"
26#include "llvm/IR/LLVMContext.h"
27#include "llvm/IR/Module.h"
Chris Lattner984fac52009-03-26 05:00:52 +000028using namespace clang;
Ted Kremenek2c674f62008-08-05 18:50:11 +000029
Chris Lattneradf1f512008-02-06 02:01:47 +000030namespace {
Benjamin Kramer337e3a52009-11-28 19:45:26 +000031 class CodeGeneratorImpl : public CodeGenerator {
David Blaikie9c902b52011-09-25 23:23:43 +000032 DiagnosticsEngine &Diags;
Micah Villmowdd31ca12012-10-08 16:25:52 +000033 OwningPtr<const llvm::DataLayout> TD;
Chris Lattneradf1f512008-02-06 02:01:47 +000034 ASTContext *Ctx;
Chandler Carruthbc55fe22009-11-12 17:24:48 +000035 const CodeGenOptions CodeGenOpts; // Intentionally copied in.
Chris Lattneradf1f512008-02-06 02:01:47 +000036 protected:
Dylan Noblesmithe2778992012-02-05 02:12:40 +000037 OwningPtr<llvm::Module> M;
38 OwningPtr<CodeGen::CodeGenModule> Builder;
Chris Lattneradf1f512008-02-06 02:01:47 +000039 public:
David Blaikie9c902b52011-09-25 23:23:43 +000040 CodeGeneratorImpl(DiagnosticsEngine &diags, const std::string& ModuleName,
John McCall568d4102013-04-16 22:48:20 +000041 const CodeGenOptions &CGO, llvm::LLVMContext& C)
42 : Diags(diags), CodeGenOpts(CGO),
Bill Wendlingc86a2f32013-02-14 08:09:20 +000043 M(new llvm::Module(ModuleName, C)) {}
Mike Stump11289f42009-09-09 15:08:12 +000044
Ted Kremenek2c674f62008-08-05 18:50:11 +000045 virtual ~CodeGeneratorImpl() {}
Mike Stump11289f42009-09-09 15:08:12 +000046
Daniel Dunbar30c514e2008-10-21 19:55:09 +000047 virtual llvm::Module* GetModule() {
48 return M.get();
49 }
Mike Stump11289f42009-09-09 15:08:12 +000050
Ted Kremenek7db4f602008-08-07 19:47:41 +000051 virtual llvm::Module* ReleaseModule() {
Ted Kremenek2c674f62008-08-05 18:50:11 +000052 return M.take();
Chris Lattneradf1f512008-02-06 02:01:47 +000053 }
Mike Stump11289f42009-09-09 15:08:12 +000054
Chris Lattneradf1f512008-02-06 02:01:47 +000055 virtual void Initialize(ASTContext &Context) {
56 Ctx = &Context;
Mike Stump11289f42009-09-09 15:08:12 +000057
Douglas Gregore8bbc122011-09-02 00:18:52 +000058 M->setTargetTriple(Ctx->getTargetInfo().getTriple().getTriple());
59 M->setDataLayout(Ctx->getTargetInfo().getTargetDescription());
Micah Villmowdd31ca12012-10-08 16:25:52 +000060 TD.reset(new llvm::DataLayout(Ctx->getTargetInfo().getTargetDescription()));
John McCall568d4102013-04-16 22:48:20 +000061 Builder.reset(new CodeGen::CodeGenModule(Context, CodeGenOpts, *M, *TD,
62 Diags));
Hans Wennborg75958c42013-08-08 00:17:41 +000063
64 for (size_t i = 0, e = CodeGenOpts.DependentLibraries.size(); i < e; ++i)
65 HandleDependentLibrary(CodeGenOpts.DependentLibraries[i]);
Chris Lattneradf1f512008-02-06 02:01:47 +000066 }
Mike Stump11289f42009-09-09 15:08:12 +000067
Rafael Espindoladf88f6f2012-03-08 15:51:03 +000068 virtual void HandleCXXStaticMemberVarInstantiation(VarDecl *VD) {
David Blaikie4a9ec7b2013-08-19 21:02:26 +000069 if (Diags.hasErrorOccurred())
70 return;
71
Rafael Espindoladf88f6f2012-03-08 15:51:03 +000072 Builder->HandleCXXStaticMemberVarInstantiation(VD);
Rafael Espindola189fa742012-03-05 10:54:55 +000073 }
74
Argyrios Kyrtzidis841dd882011-11-18 00:26:59 +000075 virtual bool HandleTopLevelDecl(DeclGroupRef DG) {
David Blaikie4a9ec7b2013-08-19 21:02:26 +000076 if (Diags.hasErrorOccurred())
77 return true;
78
Douglas Gregor6e6ad602009-01-20 01:17:11 +000079 // Make sure to emit all elements of a Decl.
Chris Lattner5bbb3c82009-03-29 16:50:03 +000080 for (DeclGroupRef::iterator I = DG.begin(), E = DG.end(); I != E; ++I)
81 Builder->EmitTopLevelDecl(*I);
Argyrios Kyrtzidis841dd882011-11-18 00:26:59 +000082 return true;
Chris Lattneradf1f512008-02-06 02:01:47 +000083 }
Daniel Dunbarfce4be82008-08-15 23:26:23 +000084
Chris Lattnera5e4d302008-02-06 04:51:19 +000085 /// HandleTagDeclDefinition - This callback is invoked each time a TagDecl
Chris Lattner5bbb3c82009-03-29 16:50:03 +000086 /// to (e.g. struct, union, enum, class) is completed. This allows the
87 /// client hack on the type, which can occur at any point in the file
88 /// (because these can be defined in declspecs).
Chris Lattnera5e4d302008-02-06 04:51:19 +000089 virtual void HandleTagDeclDefinition(TagDecl *D) {
David Blaikie4a9ec7b2013-08-19 21:02:26 +000090 if (Diags.hasErrorOccurred())
91 return;
92
Chris Lattner68be6062008-02-06 05:08:19 +000093 Builder->UpdateCompletedType(D);
Douglas Gregore0610152011-02-15 18:11:42 +000094
95 // In C++, we may have member functions that need to be emitted at this
96 // point.
David Blaikiebbafb8a2012-03-11 07:00:24 +000097 if (Ctx->getLangOpts().CPlusPlus && !D->isDependentContext()) {
Douglas Gregore0610152011-02-15 18:11:42 +000098 for (DeclContext::decl_iterator M = D->decls_begin(),
99 MEnd = D->decls_end();
100 M != MEnd; ++M)
101 if (CXXMethodDecl *Method = dyn_cast<CXXMethodDecl>(*M))
Alexis Hunt4a8ea102011-05-06 20:44:56 +0000102 if (Method->doesThisDeclarationHaveABody() &&
Douglas Gregord66828d2011-02-19 21:54:50 +0000103 (Method->hasAttr<UsedAttr>() ||
104 Method->hasAttr<ConstructorAttr>()))
Douglas Gregore0610152011-02-15 18:11:42 +0000105 Builder->EmitTopLevelDecl(Method);
106 }
Chris Lattnera5e4d302008-02-06 04:51:19 +0000107 }
Ted Kremenek7db4f602008-08-07 19:47:41 +0000108
David Blaikie48ad6dc2013-07-13 21:08:14 +0000109 virtual void HandleTagDeclRequiredDefinition(const TagDecl *D) LLVM_OVERRIDE {
David Blaikie4a9ec7b2013-08-19 21:02:26 +0000110 if (Diags.hasErrorOccurred())
111 return;
112
David Blaikie48ad6dc2013-07-13 21:08:14 +0000113 if (CodeGen::CGDebugInfo *DI = Builder->getModuleDebugInfo())
114 if (const RecordDecl *RD = dyn_cast<RecordDecl>(D))
David Blaikieb2e86eb2013-08-15 20:49:17 +0000115 DI->completeRequiredType(RD);
David Blaikie48ad6dc2013-07-13 21:08:14 +0000116 }
117
Chris Lattnercf169832009-03-28 04:11:33 +0000118 virtual void HandleTranslationUnit(ASTContext &Ctx) {
Ted Kremenek7db4f602008-08-07 19:47:41 +0000119 if (Diags.hasErrorOccurred()) {
Rafael Espindolac0ff7442013-12-09 14:59:08 +0000120 if (Builder)
121 Builder->clear();
Ted Kremenek7db4f602008-08-07 19:47:41 +0000122 M.reset();
123 return;
124 }
125
126 if (Builder)
127 Builder->Release();
Daniel Dunbare017ecc2009-12-19 17:50:07 +0000128 }
Douglas Gregorbeecd582009-04-21 17:11:58 +0000129
130 virtual void CompleteTentativeDefinition(VarDecl *D) {
131 if (Diags.hasErrorOccurred())
132 return;
133
134 Builder->EmitTentativeDefinition(D);
135 }
Douglas Gregor88d292c2010-05-13 16:44:06 +0000136
137 virtual void HandleVTable(CXXRecordDecl *RD, bool DefinitionRequired) {
138 if (Diags.hasErrorOccurred())
139 return;
140
141 Builder->EmitVTable(RD, DefinitionRequired);
142 }
Reid Klecknere43f0fe2013-05-08 13:44:39 +0000143
144 virtual void HandleLinkerOptionPragma(llvm::StringRef Opts) {
145 Builder->AppendLinkerOptions(Opts);
146 }
147
Aaron Ballman5d041be2013-06-04 02:07:14 +0000148 virtual void HandleDetectMismatch(llvm::StringRef Name,
149 llvm::StringRef Value) {
150 Builder->AddDetectMismatch(Name, Value);
151 }
152
Reid Klecknere43f0fe2013-05-08 13:44:39 +0000153 virtual void HandleDependentLibrary(llvm::StringRef Lib) {
154 Builder->AddDependentLib(Lib);
155 }
Chris Lattneradf1f512008-02-06 02:01:47 +0000156 };
Chris Lattnerf97fe382007-05-24 06:29:05 +0000157}
158
David Blaikie68e081d2011-12-20 02:48:34 +0000159void CodeGenerator::anchor() { }
160
David Blaikie9c902b52011-09-25 23:23:43 +0000161CodeGenerator *clang::CreateLLVMCodeGen(DiagnosticsEngine &Diags,
Ted Kremenek2c674f62008-08-05 18:50:11 +0000162 const std::string& ModuleName,
Chandler Carruthbc55fe22009-11-12 17:24:48 +0000163 const CodeGenOptions &CGO,
John McCall568d4102013-04-16 22:48:20 +0000164 const TargetOptions &/*TO*/,
Owen Andersonecaeaa82009-07-01 23:14:14 +0000165 llvm::LLVMContext& C) {
John McCall568d4102013-04-16 22:48:20 +0000166 return new CodeGeneratorImpl(Diags, ModuleName, CGO, C);
Chris Lattnerf97fe382007-05-24 06:29:05 +0000167}