blob: c4a0e5c063d461978131372ae90f8724fecae42f [file] [log] [blame]
Reid Spencer5f016e22007-07-11 17:01:13 +00001//===--- ModuleBuilder.cpp - Emit LLVM Code from ASTs ---------------------===//
2//
3// The LLVM Compiler Infrastructure
4//
Chris Lattner0bc735f2007-12-29 19:59:25 +00005// This file is distributed under the University of Illinois Open Source
6// License. See LICENSE.TXT for details.
Reid Spencer5f016e22007-07-11 17:01:13 +00007//
8//===----------------------------------------------------------------------===//
9//
10// This builds an AST and converts it to LLVM Code.
11//
12//===----------------------------------------------------------------------===//
13
14#include "clang/CodeGen/ModuleBuilder.h"
David Blaikie658cd2c2013-07-13 21:08:14 +000015#include "CGDebugInfo.h"
Stephen Hines651f13c2014-04-23 16:59:28 -070016#include "CodeGenModule.h"
Chris Lattner8ee3c032008-02-06 02:01:47 +000017#include "clang/AST/ASTContext.h"
Daniel Dunbare91593e2008-08-11 04:54:23 +000018#include "clang/AST/DeclObjC.h"
19#include "clang/AST/Expr.h"
Chris Lattner8ee3c032008-02-06 02:01:47 +000020#include "clang/Basic/Diagnostic.h"
21#include "clang/Basic/TargetInfo.h"
Chandler Carruth55fc8732012-12-04 09:13:33 +000022#include "clang/Frontend/CodeGenOptions.h"
Reid Kleckner3190ca92013-05-08 13:44:39 +000023#include "llvm/ADT/StringRef.h"
Chandler Carruth3b844ba2013-01-02 11:45:17 +000024#include "llvm/IR/DataLayout.h"
25#include "llvm/IR/LLVMContext.h"
26#include "llvm/IR/Module.h"
Stephen Hines651f13c2014-04-23 16:59:28 -070027#include <memory>
Chris Lattnerbd360642009-03-26 05:00:52 +000028using namespace clang;
Ted Kremenek815c78f2008-08-05 18:50:11 +000029
Chris Lattner8ee3c032008-02-06 02:01:47 +000030namespace {
Benjamin Kramer85b45212009-11-28 19:45:26 +000031 class CodeGeneratorImpl : public CodeGenerator {
David Blaikied6471f72011-09-25 23:23:43 +000032 DiagnosticsEngine &Diags;
Stephen Hines651f13c2014-04-23 16:59:28 -070033 std::unique_ptr<const llvm::DataLayout> TD;
Chris Lattner8ee3c032008-02-06 02:01:47 +000034 ASTContext *Ctx;
Chandler Carruth2811ccf2009-11-12 17:24:48 +000035 const CodeGenOptions CodeGenOpts; // Intentionally copied in.
Chris Lattner8ee3c032008-02-06 02:01:47 +000036 protected:
Stephen Hines651f13c2014-04-23 16:59:28 -070037 std::unique_ptr<llvm::Module> M;
38 std::unique_ptr<CodeGen::CodeGenModule> Builder;
39
Chris Lattner8ee3c032008-02-06 02:01:47 +000040 public:
David Blaikied6471f72011-09-25 23:23:43 +000041 CodeGeneratorImpl(DiagnosticsEngine &diags, const std::string& ModuleName,
John McCall3abae092013-04-16 22:48:20 +000042 const CodeGenOptions &CGO, llvm::LLVMContext& C)
43 : Diags(diags), CodeGenOpts(CGO),
Bill Wendlinge1092df2013-02-14 08:09:20 +000044 M(new llvm::Module(ModuleName, C)) {}
Mike Stump1eb44332009-09-09 15:08:12 +000045
Ted Kremenek815c78f2008-08-05 18:50:11 +000046 virtual ~CodeGeneratorImpl() {}
Mike Stump1eb44332009-09-09 15:08:12 +000047
Stephen Hines651f13c2014-04-23 16:59:28 -070048 llvm::Module* GetModule() override {
Daniel Dunbard8c0ea12008-10-21 19:55:09 +000049 return M.get();
50 }
Mike Stump1eb44332009-09-09 15:08:12 +000051
Stephen Hinesef822542014-07-21 00:47:37 -070052 const Decl *GetDeclForMangledName(StringRef MangledName) override {
53 GlobalDecl Result;
54 if (!Builder->lookupRepresentativeDecl(MangledName, Result))
55 return nullptr;
56 const Decl *D = Result.getCanonicalDecl().getDecl();
57 if (auto FD = dyn_cast<FunctionDecl>(D)) {
58 if (FD->hasBody(FD))
59 return FD;
60 } else if (auto TD = dyn_cast<TagDecl>(D)) {
61 if (auto Def = TD->getDefinition())
62 return Def;
63 }
64 return D;
65 }
66
Stephen Hines651f13c2014-04-23 16:59:28 -070067 llvm::Module *ReleaseModule() override { return M.release(); }
Mike Stump1eb44332009-09-09 15:08:12 +000068
Stephen Hines651f13c2014-04-23 16:59:28 -070069 void Initialize(ASTContext &Context) override {
Chris Lattner8ee3c032008-02-06 02:01:47 +000070 Ctx = &Context;
Mike Stump1eb44332009-09-09 15:08:12 +000071
Douglas Gregorbcfd1f52011-09-02 00:18:52 +000072 M->setTargetTriple(Ctx->getTargetInfo().getTriple().getTriple());
73 M->setDataLayout(Ctx->getTargetInfo().getTargetDescription());
Micah Villmow25a6a842012-10-08 16:25:52 +000074 TD.reset(new llvm::DataLayout(Ctx->getTargetInfo().getTargetDescription()));
John McCall3abae092013-04-16 22:48:20 +000075 Builder.reset(new CodeGen::CodeGenModule(Context, CodeGenOpts, *M, *TD,
76 Diags));
Hans Wennborgb3574792013-08-08 00:17:41 +000077
78 for (size_t i = 0, e = CodeGenOpts.DependentLibraries.size(); i < e; ++i)
79 HandleDependentLibrary(CodeGenOpts.DependentLibraries[i]);
Chris Lattner8ee3c032008-02-06 02:01:47 +000080 }
Mike Stump1eb44332009-09-09 15:08:12 +000081
Stephen Hines651f13c2014-04-23 16:59:28 -070082 void HandleCXXStaticMemberVarInstantiation(VarDecl *VD) override {
David Blaikie0a1c8622013-08-19 21:02:26 +000083 if (Diags.hasErrorOccurred())
84 return;
85
Rafael Espindola02503932012-03-08 15:51:03 +000086 Builder->HandleCXXStaticMemberVarInstantiation(VD);
Rafael Espindola234fe652012-03-05 10:54:55 +000087 }
88
Stephen Hines651f13c2014-04-23 16:59:28 -070089 bool HandleTopLevelDecl(DeclGroupRef DG) override {
David Blaikie0a1c8622013-08-19 21:02:26 +000090 if (Diags.hasErrorOccurred())
91 return true;
92
Douglas Gregor4afa39d2009-01-20 01:17:11 +000093 // Make sure to emit all elements of a Decl.
Chris Lattner682bf922009-03-29 16:50:03 +000094 for (DeclGroupRef::iterator I = DG.begin(), E = DG.end(); I != E; ++I)
95 Builder->EmitTopLevelDecl(*I);
Stephen Hinesef822542014-07-21 00:47:37 -070096
97 // Emit any deferred inline method definitions.
98 for (CXXMethodDecl *MD : DeferredInlineMethodDefinitions)
99 Builder->EmitTopLevelDecl(MD);
100 DeferredInlineMethodDefinitions.clear();
101
Argyrios Kyrtzidis88c25962011-11-18 00:26:59 +0000102 return true;
Chris Lattner8ee3c032008-02-06 02:01:47 +0000103 }
Daniel Dunbar41071de2008-08-15 23:26:23 +0000104
Stephen Hines6bcf27b2014-05-29 04:14:42 -0700105 void HandleInlineMethodDefinition(CXXMethodDecl *D) override {
106 if (Diags.hasErrorOccurred())
107 return;
108
109 assert(D->doesThisDeclarationHaveABody());
110
Stephen Hinesef822542014-07-21 00:47:37 -0700111 // We may want to emit this definition. However, that decision might be
112 // based on computing the linkage, and we have to defer that in case we
113 // are inside of something that will change the method's final linkage,
114 // e.g.
115 // typedef struct {
116 // void bar();
117 // void foo() { bar(); }
118 // } A;
119 DeferredInlineMethodDefinitions.push_back(D);
Stephen Hines6bcf27b2014-05-29 04:14:42 -0700120 }
121
Chris Lattner9eee0f82008-02-06 04:51:19 +0000122 /// HandleTagDeclDefinition - This callback is invoked each time a TagDecl
Chris Lattner682bf922009-03-29 16:50:03 +0000123 /// to (e.g. struct, union, enum, class) is completed. This allows the
124 /// client hack on the type, which can occur at any point in the file
125 /// (because these can be defined in declspecs).
Stephen Hines651f13c2014-04-23 16:59:28 -0700126 void HandleTagDeclDefinition(TagDecl *D) override {
David Blaikie0a1c8622013-08-19 21:02:26 +0000127 if (Diags.hasErrorOccurred())
128 return;
129
Chris Lattnerc5b88062008-02-06 05:08:19 +0000130 Builder->UpdateCompletedType(D);
Chris Lattner9eee0f82008-02-06 04:51:19 +0000131 }
Ted Kremenek159346a2008-08-07 19:47:41 +0000132
Stephen Hines651f13c2014-04-23 16:59:28 -0700133 void HandleTagDeclRequiredDefinition(const TagDecl *D) override {
David Blaikie0a1c8622013-08-19 21:02:26 +0000134 if (Diags.hasErrorOccurred())
135 return;
136
David Blaikie658cd2c2013-07-13 21:08:14 +0000137 if (CodeGen::CGDebugInfo *DI = Builder->getModuleDebugInfo())
138 if (const RecordDecl *RD = dyn_cast<RecordDecl>(D))
David Blaikie27804892013-08-15 20:49:17 +0000139 DI->completeRequiredType(RD);
David Blaikie658cd2c2013-07-13 21:08:14 +0000140 }
141
Stephen Hines651f13c2014-04-23 16:59:28 -0700142 void HandleTranslationUnit(ASTContext &Ctx) override {
Ted Kremenek159346a2008-08-07 19:47:41 +0000143 if (Diags.hasErrorOccurred()) {
Stephen Hines651f13c2014-04-23 16:59:28 -0700144 if (Builder)
145 Builder->clear();
Ted Kremenek159346a2008-08-07 19:47:41 +0000146 M.reset();
147 return;
148 }
149
150 if (Builder)
151 Builder->Release();
Daniel Dunbar7177dee2009-12-19 17:50:07 +0000152 }
Douglas Gregorb6c8c8b2009-04-21 17:11:58 +0000153
Stephen Hines651f13c2014-04-23 16:59:28 -0700154 void CompleteTentativeDefinition(VarDecl *D) override {
Douglas Gregorb6c8c8b2009-04-21 17:11:58 +0000155 if (Diags.hasErrorOccurred())
156 return;
157
158 Builder->EmitTentativeDefinition(D);
159 }
Douglas Gregor6fb745b2010-05-13 16:44:06 +0000160
Stephen Hines651f13c2014-04-23 16:59:28 -0700161 void HandleVTable(CXXRecordDecl *RD, bool DefinitionRequired) override {
Douglas Gregor6fb745b2010-05-13 16:44:06 +0000162 if (Diags.hasErrorOccurred())
163 return;
164
165 Builder->EmitVTable(RD, DefinitionRequired);
166 }
Reid Kleckner3190ca92013-05-08 13:44:39 +0000167
Stephen Hines651f13c2014-04-23 16:59:28 -0700168 void HandleLinkerOptionPragma(llvm::StringRef Opts) override {
Reid Kleckner3190ca92013-05-08 13:44:39 +0000169 Builder->AppendLinkerOptions(Opts);
170 }
171
Stephen Hines651f13c2014-04-23 16:59:28 -0700172 void HandleDetectMismatch(llvm::StringRef Name,
173 llvm::StringRef Value) override {
Aaron Ballmana7ff62f2013-06-04 02:07:14 +0000174 Builder->AddDetectMismatch(Name, Value);
175 }
176
Stephen Hines651f13c2014-04-23 16:59:28 -0700177 void HandleDependentLibrary(llvm::StringRef Lib) override {
Reid Kleckner3190ca92013-05-08 13:44:39 +0000178 Builder->AddDependentLib(Lib);
179 }
Stephen Hinesef822542014-07-21 00:47:37 -0700180
181 private:
182 std::vector<CXXMethodDecl *> DeferredInlineMethodDefinitions;
Chris Lattner8ee3c032008-02-06 02:01:47 +0000183 };
Reid Spencer5f016e22007-07-11 17:01:13 +0000184}
185
David Blaikie99ba9e32011-12-20 02:48:34 +0000186void CodeGenerator::anchor() { }
187
David Blaikied6471f72011-09-25 23:23:43 +0000188CodeGenerator *clang::CreateLLVMCodeGen(DiagnosticsEngine &Diags,
Ted Kremenek815c78f2008-08-05 18:50:11 +0000189 const std::string& ModuleName,
Chandler Carruth2811ccf2009-11-12 17:24:48 +0000190 const CodeGenOptions &CGO,
John McCall3abae092013-04-16 22:48:20 +0000191 const TargetOptions &/*TO*/,
Owen Anderson8f1ca782009-07-01 23:14:14 +0000192 llvm::LLVMContext& C) {
John McCall3abae092013-04-16 22:48:20 +0000193 return new CodeGeneratorImpl(Diags, ModuleName, CGO, C);
Reid Spencer5f016e22007-07-11 17:01:13 +0000194}