blob: a9fa0df14f8c963b9e8dbae8ebbf00ef0d86e212 [file] [log] [blame]
Zonr Changc383a502010-10-12 01:52:08 +08001/*
2 * Copyright 2010, The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
zonr6315f762010-10-05 15:35:14 +080017#include "slang.h"
Shih-wei Liao462aefd2010-06-04 15:32:04 -070018
Shih-wei Liao9ef2f782010-10-01 12:31:37 -070019#include <stdlib.h>
20
Stephen Hinese639eb52010-11-08 19:27:20 -080021#include <string>
22#include <vector>
23
24#include "clang/AST/ASTConsumer.h"
25#include "clang/AST/ASTContext.h"
26
Loganbe274822011-02-16 22:02:54 +080027#include "clang/Basic/DiagnosticIDs.h"
Stephen Hinese639eb52010-11-08 19:27:20 -080028#include "clang/Basic/FileManager.h"
Shih-wei Liaodf5bcce2011-02-28 18:39:23 -080029#include "clang/Basic/FileSystemOptions.h"
Stephen Hinese639eb52010-11-08 19:27:20 -080030#include "clang/Basic/LangOptions.h"
31#include "clang/Basic/SourceManager.h"
32#include "clang/Basic/TargetInfo.h"
33#include "clang/Basic/TargetOptions.h"
34
35#include "clang/Frontend/CodeGenOptions.h"
36#include "clang/Frontend/DiagnosticOptions.h"
37#include "clang/Frontend/DependencyOutputOptions.h"
38#include "clang/Frontend/FrontendDiagnostic.h"
39#include "clang/Frontend/TextDiagnosticPrinter.h"
40#include "clang/Frontend/Utils.h"
41
42#include "clang/Lex/Preprocessor.h"
43#include "clang/Lex/HeaderSearch.h"
44
45#include "clang/Parse/ParseAST.h"
46
Loganbe274822011-02-16 22:02:54 +080047#include "llvm/ADT/IntrusiveRefCntPtr.h"
48
Stephen Hinese639eb52010-11-08 19:27:20 -080049#include "llvm/Bitcode/ReaderWriter.h"
50
51// More force linking
52#include "llvm/Linker.h"
53
Zonr Chang08df36e2010-10-07 18:50:42 +080054// Force linking all passes/vmcore stuffs to libslang.so
55#include "llvm/LinkAllPasses.h"
56#include "llvm/LinkAllVMCore.h"
57
Zonr Chang3a9ca1f2010-10-06 17:52:56 +080058#include "llvm/Support/raw_ostream.h"
zonr6315f762010-10-05 15:35:14 +080059#include "llvm/Support/MemoryBuffer.h"
60#include "llvm/Support/ErrorHandling.h"
61#include "llvm/Support/ManagedStatic.h"
Loganbe274822011-02-16 22:02:54 +080062#include "llvm/Support/Path.h"
Logan Chien9207a2e2011-10-21 15:39:28 +080063#include "llvm/Support/TargetSelect.h"
64#include "llvm/Support/ToolOutputFile.h"
Stephen Hinescc0efad2010-10-04 16:13:02 -070065
Stephen Hines6e6578a2011-02-07 18:05:48 -080066#include "slang_assert.h"
Zonr Chang3a9ca1f2010-10-06 17:52:56 +080067#include "slang_backend.h"
Stephen Hinese639eb52010-11-08 19:27:20 -080068#include "slang_utils.h"
Shih-wei Liaob81c6a42010-10-10 14:15:00 -070069
Zonr Chang08df36e2010-10-07 18:50:42 +080070namespace {
Zonr Chang08df36e2010-10-07 18:50:42 +080071
Stephen Hinese639eb52010-11-08 19:27:20 -080072struct ForceSlangLinking {
73 ForceSlangLinking() {
74 // We must reference the functions in such a way that compilers will not
75 // delete it all as dead code, even with whole program optimization,
76 // yet is effectively a NO-OP. As the compiler isn't smart enough
77 // to know that getenv() never returns -1, this will do the job.
78 if (std::getenv("bar") != reinterpret_cast<char*>(-1))
79 return;
Shih-wei Liaob81c6a42010-10-10 14:15:00 -070080
Stephen Hinese639eb52010-11-08 19:27:20 -080081 // llvm-rs-link needs following functions existing in libslang.
82 llvm::ParseBitcodeFile(NULL, llvm::getGlobalContext(), NULL);
Logan Chien9207a2e2011-10-21 15:39:28 +080083 llvm::Linker::LinkModules(NULL, NULL, 0, NULL);
Zonr Chang08df36e2010-10-07 18:50:42 +080084
Stephen Hinese639eb52010-11-08 19:27:20 -080085 // llvm-rs-cc need this.
86 new clang::TextDiagnosticPrinter(llvm::errs(),
87 clang::DiagnosticOptions());
88 }
89} ForceSlangLinking;
90
91} // namespace
92
93namespace slang {
Shih-wei Liao9ef2f782010-10-01 12:31:37 -070094
Shih-wei Liao462aefd2010-06-04 15:32:04 -070095#if defined(__arm__)
96# define DEFAULT_TARGET_TRIPLE_STRING "armv7-none-linux-gnueabi"
97#elif defined(__x86_64__)
98# define DEFAULT_TARGET_TRIPLE_STRING "x86_64-unknown-linux"
99#else
Shih-wei Liao9ef2f782010-10-01 12:31:37 -0700100// let's use x86 as default target
101# define DEFAULT_TARGET_TRIPLE_STRING "i686-unknown-linux"
Shih-wei Liao462aefd2010-06-04 15:32:04 -0700102#endif
103
Shih-wei Liao462aefd2010-06-04 15:32:04 -0700104bool Slang::GlobalInitialized = false;
105
Shih-wei Liao9ef2f782010-10-01 12:31:37 -0700106// Language option (define the language feature for compiler such as C99)
107clang::LangOptions Slang::LangOpts;
Shih-wei Liao462aefd2010-06-04 15:32:04 -0700108
zonr6315f762010-10-05 15:35:14 +0800109// Code generation option for the compiler
Shih-wei Liao9ef2f782010-10-01 12:31:37 -0700110clang::CodeGenOptions Slang::CodeGenOpts;
Shih-wei Liao462aefd2010-06-04 15:32:04 -0700111
Shih-wei Liao9ef2f782010-10-01 12:31:37 -0700112// The named of metadata node that pragma resides (should be synced with
113// bcc.cpp)
Shih-wei Liaof52a6202010-09-10 17:40:53 -0700114const llvm::StringRef Slang::PragmaMetadataName = "#pragma";
Shih-wei Liao462aefd2010-06-04 15:32:04 -0700115
Logan Chien9207a2e2011-10-21 15:39:28 +0800116static inline llvm::tool_output_file *
117OpenOutputFile(const char *OutputFile,
118 unsigned Flags,
119 std::string* Error,
Stephen Hinese67239d2012-02-24 15:08:36 -0800120 clang::DiagnosticsEngine *DiagEngine) {
Logan Chien9207a2e2011-10-21 15:39:28 +0800121 slangAssert((OutputFile != NULL) && (Error != NULL) &&
122 (DiagEngine != NULL) && "Invalid parameter!");
Zonr Change8c263a2010-10-12 00:35:29 +0800123
Logan Chien6f4e0a92011-03-10 01:10:25 +0800124 if (SlangUtils::CreateDirectoryWithParents(
125 llvm::sys::path::parent_path(OutputFile), Error)) {
Zonr Change8c263a2010-10-12 00:35:29 +0800126 llvm::tool_output_file *F =
127 new llvm::tool_output_file(OutputFile, *Error, Flags);
128 if (F != NULL)
129 return F;
130 }
131
132 // Report error here.
Logan Chien9207a2e2011-10-21 15:39:28 +0800133 DiagEngine->Report(clang::diag::err_fe_error_opening)
134 << OutputFile << *Error;
Zonr Change8c263a2010-10-12 00:35:29 +0800135
136 return NULL;
137}
138
Shih-wei Liao462aefd2010-06-04 15:32:04 -0700139void Slang::GlobalInitialization() {
Shih-wei Liao9ef2f782010-10-01 12:31:37 -0700140 if (!GlobalInitialized) {
141 // We only support x86, x64 and ARM target
Shih-wei Liao462aefd2010-06-04 15:32:04 -0700142
Shih-wei Liao9ef2f782010-10-01 12:31:37 -0700143 // For ARM
144 LLVMInitializeARMTargetInfo();
145 LLVMInitializeARMTarget();
146 LLVMInitializeARMAsmPrinter();
Shih-wei Liao462aefd2010-06-04 15:32:04 -0700147
Shih-wei Liao9ef2f782010-10-01 12:31:37 -0700148 // For x86 and x64
149 LLVMInitializeX86TargetInfo();
150 LLVMInitializeX86Target();
151 LLVMInitializeX86AsmPrinter();
Shih-wei Liao462aefd2010-06-04 15:32:04 -0700152
Zonr Chang3a9ca1f2010-10-06 17:52:56 +0800153 // Please refer to include/clang/Basic/LangOptions.h to setup
Shih-wei Liao9ef2f782010-10-01 12:31:37 -0700154 // the options.
155 LangOpts.RTTI = 0; // Turn off the RTTI information support
156 LangOpts.NeXTRuntime = 0; // Turn off the NeXT runtime uses
Shih-wei Liao245eba12011-01-19 02:14:07 -0800157 LangOpts.C99 = 1;
Stephen Hines52d132c2012-08-01 18:48:06 -0700158 LangOpts.Renderscript = 1;
Stephen Hines194d4032012-02-27 18:01:09 -0800159 LangOpts.CharIsSigned = 1; // Signed char is our default.
Shih-wei Liao462aefd2010-06-04 15:32:04 -0700160
mkopec1c460b372012-01-09 11:21:50 -0500161 CodeGenOpts.OptimizationLevel = 3;
Shih-wei Liao462aefd2010-06-04 15:32:04 -0700162
Shih-wei Liao9ef2f782010-10-01 12:31:37 -0700163 GlobalInitialized = true;
164 }
Shih-wei Liao462aefd2010-06-04 15:32:04 -0700165}
166
167void Slang::LLVMErrorHandler(void *UserData, const std::string &Message) {
Logan Chien9207a2e2011-10-21 15:39:28 +0800168 clang::DiagnosticsEngine* DiagEngine =
169 static_cast<clang::DiagnosticsEngine *>(UserData);
170
171 DiagEngine->Report(clang::diag::err_fe_error_backend) << Message;
Shih-wei Liao9ef2f782010-10-01 12:31:37 -0700172 exit(1);
Shih-wei Liao462aefd2010-06-04 15:32:04 -0700173}
174
Shih-wei Liaob81c6a42010-10-10 14:15:00 -0700175void Slang::createTarget(const std::string &Triple, const std::string &CPU,
176 const std::vector<std::string> &Features) {
177 if (!Triple.empty())
Shih-wei Liao9ef2f782010-10-01 12:31:37 -0700178 mTargetOpts.Triple = Triple;
179 else
180 mTargetOpts.Triple = DEFAULT_TARGET_TRIPLE_STRING;
Shih-wei Liao462aefd2010-06-04 15:32:04 -0700181
Shih-wei Liaob81c6a42010-10-10 14:15:00 -0700182 if (!CPU.empty())
Shih-wei Liao9ef2f782010-10-01 12:31:37 -0700183 mTargetOpts.CPU = CPU;
Shih-wei Liao462aefd2010-06-04 15:32:04 -0700184
Shih-wei Liaob81c6a42010-10-10 14:15:00 -0700185 if (!Features.empty())
186 mTargetOpts.Features = Features;
187
Logan Chien9207a2e2011-10-21 15:39:28 +0800188 mTarget.reset(clang::TargetInfo::CreateTargetInfo(*mDiagEngine,
Shih-wei Liao9ef2f782010-10-01 12:31:37 -0700189 mTargetOpts));
Shih-wei Liao462aefd2010-06-04 15:32:04 -0700190}
191
Zonr Chang3a9ca1f2010-10-06 17:52:56 +0800192void Slang::createFileManager() {
Shih-wei Liaodf5bcce2011-02-28 18:39:23 -0800193 mFileSysOpt.reset(new clang::FileSystemOptions());
Loganbe274822011-02-16 22:02:54 +0800194 mFileMgr.reset(new clang::FileManager(*mFileSysOpt));
Zonr Chang3a9ca1f2010-10-06 17:52:56 +0800195}
196
197void Slang::createSourceManager() {
Logan Chien9207a2e2011-10-21 15:39:28 +0800198 mSourceMgr.reset(new clang::SourceManager(*mDiagEngine, *mFileMgr));
Zonr Chang3a9ca1f2010-10-06 17:52:56 +0800199}
200
Shih-wei Liao68e8e9f2010-07-18 18:46:49 -0700201void Slang::createPreprocessor() {
Shih-wei Liao9ef2f782010-10-01 12:31:37 -0700202 // Default only search header file in current dir
Stephen Hines4c622e02011-11-10 18:57:34 -0800203 clang::HeaderSearch *HeaderInfo = new clang::HeaderSearch(*mFileMgr,
Stephen Hines0444de02012-03-02 23:19:06 -0800204 *mDiagEngine,
205 LangOpts,
206 mTarget.get());
Shih-wei Liao68e8e9f2010-07-18 18:46:49 -0700207
Logan Chien9207a2e2011-10-21 15:39:28 +0800208 mPP.reset(new clang::Preprocessor(*mDiagEngine,
Shih-wei Liao9ef2f782010-10-01 12:31:37 -0700209 LangOpts,
Logan Chien9207a2e2011-10-21 15:39:28 +0800210 mTarget.get(),
Shih-wei Liao9ef2f782010-10-01 12:31:37 -0700211 *mSourceMgr,
Logan Chien9207a2e2011-10-21 15:39:28 +0800212 *HeaderInfo,
213 *this,
Shih-wei Liao9ef2f782010-10-01 12:31:37 -0700214 NULL,
Zonr Chang3a9ca1f2010-10-06 17:52:56 +0800215 /* OwnsHeaderSearch = */true));
Stephen Hines3fd0a942011-01-18 12:27:39 -0800216 // Initialize the preprocessor
Shih-wei Liao68e8e9f2010-07-18 18:46:49 -0700217 mPragmas.clear();
Stephen Hines3fd0a942011-01-18 12:27:39 -0800218 mPP->AddPragmaHandler(new PragmaRecorder(&mPragmas));
Shih-wei Liao68e8e9f2010-07-18 18:46:49 -0700219
Shih-wei Liao9ef2f782010-10-01 12:31:37 -0700220 std::vector<clang::DirectoryLookup> SearchList;
Zonr Chang3a9ca1f2010-10-06 17:52:56 +0800221 for (unsigned i = 0, e = mIncludePaths.size(); i != e; i++) {
Shih-wei Liao9ef2f782010-10-01 12:31:37 -0700222 if (const clang::DirectoryEntry *DE =
Loganbe274822011-02-16 22:02:54 +0800223 mFileMgr->getDirectory(mIncludePaths[i])) {
Shih-wei Liao9ef2f782010-10-01 12:31:37 -0700224 SearchList.push_back(clang::DirectoryLookup(DE,
225 clang::SrcMgr::C_System,
226 false,
227 false));
228 }
Shih-wei Liao90898282010-07-19 18:38:57 -0700229 }
230
Logan Chien9207a2e2011-10-21 15:39:28 +0800231 HeaderInfo->SetSearchPaths(SearchList,
232 /* angledDirIdx = */1,
233 /* systemDixIdx = */1,
234 /* noCurDirSearch = */false);
Shih-wei Liao90898282010-07-19 18:38:57 -0700235
Zonr Chang3a9ca1f2010-10-06 17:52:56 +0800236 initPreprocessor();
Shih-wei Liao68e8e9f2010-07-18 18:46:49 -0700237}
238
Zonr Chang3a9ca1f2010-10-06 17:52:56 +0800239void Slang::createASTContext() {
240 mASTContext.reset(new clang::ASTContext(LangOpts,
241 *mSourceMgr,
Logan Chien9207a2e2011-10-21 15:39:28 +0800242 mTarget.get(),
Zonr Chang3a9ca1f2010-10-06 17:52:56 +0800243 mPP->getIdentifierTable(),
244 mPP->getSelectorTable(),
245 mPP->getBuiltinInfo(),
246 /* size_reserve = */0));
247 initASTContext();
Zonr Chang3a9ca1f2010-10-06 17:52:56 +0800248}
249
Logan Chien9207a2e2011-10-21 15:39:28 +0800250clang::ASTConsumer *
251Slang::createBackend(const clang::CodeGenOptions& CodeGenOpts,
252 llvm::raw_ostream *OS, OutputType OT) {
Stephen Hines8f4d9722011-12-05 14:14:42 -0800253 return new Backend(mDiagEngine, CodeGenOpts, mTargetOpts,
Logan Chien9207a2e2011-10-21 15:39:28 +0800254 &mPragmas, OS, OT);
Zonr Chang3a9ca1f2010-10-06 17:52:56 +0800255}
256
Zonr Chang641558f2010-10-12 21:07:06 +0800257Slang::Slang() : mInitialized(false), mDiagClient(NULL), mOT(OT_Default) {
Shih-wei Liao9ef2f782010-10-01 12:31:37 -0700258 GlobalInitialization();
Zonr Chang641558f2010-10-12 21:07:06 +0800259}
260
261void Slang::init(const std::string &Triple, const std::string &CPU,
Stephen Hines8f4d9722011-12-05 14:14:42 -0800262 const std::vector<std::string> &Features,
263 clang::DiagnosticsEngine *DiagEngine,
264 DiagnosticBuffer *DiagClient) {
Zonr Chang641558f2010-10-12 21:07:06 +0800265 if (mInitialized)
266 return;
Shih-wei Liao462aefd2010-06-04 15:32:04 -0700267
Stephen Hines8f4d9722011-12-05 14:14:42 -0800268 mDiagEngine = DiagEngine;
269 mDiagClient = DiagClient;
270 mDiag.reset(new clang::Diagnostic(mDiagEngine));
271 initDiagnostic();
272 llvm::install_fatal_error_handler(LLVMErrorHandler, mDiagEngine);
Shih-wei Liao462aefd2010-06-04 15:32:04 -0700273
Shih-wei Liao9ef2f782010-10-01 12:31:37 -0700274 createTarget(Triple, CPU, Features);
275 createFileManager();
276 createSourceManager();
Shih-wei Liao462aefd2010-06-04 15:32:04 -0700277
Zonr Chang641558f2010-10-12 21:07:06 +0800278 mInitialized = true;
Logan Chien9207a2e2011-10-21 15:39:28 +0800279}
Zonr Chang641558f2010-10-12 21:07:06 +0800280
Logan Chienac4e1852011-12-16 13:37:10 +0800281clang::Module *Slang::loadModule(clang::SourceLocation ImportLoc,
282 clang::ModuleIdPath Path,
283 clang::Module::NameVisibilityKind Visibility,
284 bool IsInclusionDirective) {
Logan Chien9207a2e2011-10-21 15:39:28 +0800285 slangAssert(0 && "Not implemented");
286 return NULL;
Shih-wei Liao462aefd2010-06-04 15:32:04 -0700287}
288
Zonr Chang3a9ca1f2010-10-06 17:52:56 +0800289bool Slang::setInputSource(llvm::StringRef InputFile,
290 const char *Text,
291 size_t TextLength) {
292 mInputFileName = InputFile.str();
Shih-wei Liao462aefd2010-06-04 15:32:04 -0700293
Shih-wei Liao9ef2f782010-10-01 12:31:37 -0700294 // Reset the ID tables if we are reusing the SourceManager
295 mSourceMgr->clearIDTables();
Shih-wei Liao462aefd2010-06-04 15:32:04 -0700296
Shih-wei Liao9ef2f782010-10-01 12:31:37 -0700297 // Load the source
zonr6315f762010-10-05 15:35:14 +0800298 llvm::MemoryBuffer *SB =
Zonr Chang3a9ca1f2010-10-06 17:52:56 +0800299 llvm::MemoryBuffer::getMemBuffer(Text, Text + TextLength);
Shih-wei Liao9ef2f782010-10-01 12:31:37 -0700300 mSourceMgr->createMainFileIDForMemBuffer(SB);
Shih-wei Liao462aefd2010-06-04 15:32:04 -0700301
Shih-wei Liao9ef2f782010-10-01 12:31:37 -0700302 if (mSourceMgr->getMainFileID().isInvalid()) {
Logan Chien9207a2e2011-10-21 15:39:28 +0800303 mDiagEngine->Report(clang::diag::err_fe_error_reading) << InputFile;
Shih-wei Liao9ef2f782010-10-01 12:31:37 -0700304 return false;
305 }
306 return true;
Shih-wei Liao462aefd2010-06-04 15:32:04 -0700307}
308
Zonr Chang3a9ca1f2010-10-06 17:52:56 +0800309bool Slang::setInputSource(llvm::StringRef InputFile) {
310 mInputFileName = InputFile.str();
Shih-wei Liao462aefd2010-06-04 15:32:04 -0700311
Shih-wei Liao9ef2f782010-10-01 12:31:37 -0700312 mSourceMgr->clearIDTables();
Shih-wei Liao462aefd2010-06-04 15:32:04 -0700313
Loganbe274822011-02-16 22:02:54 +0800314 const clang::FileEntry *File = mFileMgr->getFile(InputFile);
Shih-wei Liao9ef2f782010-10-01 12:31:37 -0700315 if (File)
316 mSourceMgr->createMainFileID(File);
Shih-wei Liao462aefd2010-06-04 15:32:04 -0700317
Shih-wei Liao9ef2f782010-10-01 12:31:37 -0700318 if (mSourceMgr->getMainFileID().isInvalid()) {
Logan Chien9207a2e2011-10-21 15:39:28 +0800319 mDiagEngine->Report(clang::diag::err_fe_error_reading) << InputFile;
Shih-wei Liao9ef2f782010-10-01 12:31:37 -0700320 return false;
321 }
Shih-wei Liao462aefd2010-06-04 15:32:04 -0700322
Shih-wei Liao9ef2f782010-10-01 12:31:37 -0700323 return true;
Shih-wei Liao462aefd2010-06-04 15:32:04 -0700324}
325
Zonr Chang3a9ca1f2010-10-06 17:52:56 +0800326bool Slang::setOutput(const char *OutputFile) {
Zonr Chang8c6d9b22010-10-07 18:01:19 +0800327 llvm::sys::Path OutputFilePath(OutputFile);
Shih-wei Liao9ef2f782010-10-01 12:31:37 -0700328 std::string Error;
Zonr Change8c263a2010-10-12 00:35:29 +0800329 llvm::tool_output_file *OS = NULL;
Shih-wei Liao462aefd2010-06-04 15:32:04 -0700330
Zonr Chang3a9ca1f2010-10-06 17:52:56 +0800331 switch (mOT) {
Stephen Hinescc0efad2010-10-04 16:13:02 -0700332 case OT_Dependency:
Zonr Chang3a9ca1f2010-10-06 17:52:56 +0800333 case OT_Assembly:
334 case OT_LLVMAssembly: {
Stephen Hines8f4d9722011-12-05 14:14:42 -0800335 OS = OpenOutputFile(OutputFile, 0, &Error, mDiagEngine);
Shih-wei Liao9ef2f782010-10-01 12:31:37 -0700336 break;
Shih-wei Liao462aefd2010-06-04 15:32:04 -0700337 }
Zonr Chang3a9ca1f2010-10-06 17:52:56 +0800338 case OT_Nothing: {
Shih-wei Liao9ef2f782010-10-01 12:31:37 -0700339 break;
Shih-wei Liao462aefd2010-06-04 15:32:04 -0700340 }
Zonr Chang3a9ca1f2010-10-06 17:52:56 +0800341 case OT_Object:
Stephen Hinescc0efad2010-10-04 16:13:02 -0700342 case OT_Bitcode: {
Logan Chien9207a2e2011-10-21 15:39:28 +0800343 OS = OpenOutputFile(OutputFile, llvm::raw_fd_ostream::F_Binary,
Stephen Hines8f4d9722011-12-05 14:14:42 -0800344 &Error, mDiagEngine);
Shih-wei Liao9ef2f782010-10-01 12:31:37 -0700345 break;
346 }
Zonr Chang8c6d9b22010-10-07 18:01:19 +0800347 default: {
Stephen Hinescc0efad2010-10-04 16:13:02 -0700348 llvm_unreachable("Unknown compiler output type");
Zonr Chang8c6d9b22010-10-07 18:01:19 +0800349 }
Shih-wei Liao9ef2f782010-10-01 12:31:37 -0700350 }
Shih-wei Liao462aefd2010-06-04 15:32:04 -0700351
Zonr Change8c263a2010-10-12 00:35:29 +0800352 if (!Error.empty())
Shih-wei Liao9ef2f782010-10-01 12:31:37 -0700353 return false;
Zonr Change8c263a2010-10-12 00:35:29 +0800354
355 mOS.reset(OS);
Shih-wei Liao462aefd2010-06-04 15:32:04 -0700356
Zonr Chang3a9ca1f2010-10-06 17:52:56 +0800357 mOutputFileName = OutputFile;
Shih-wei Liao9ef2f782010-10-01 12:31:37 -0700358
359 return true;
Shih-wei Liao462aefd2010-06-04 15:32:04 -0700360}
361
Stephen Hines0b7ef1a2010-10-07 16:36:59 -0700362bool Slang::setDepOutput(const char *OutputFile) {
Zonr Chang8c6d9b22010-10-07 18:01:19 +0800363 llvm::sys::Path OutputFilePath(OutputFile);
Stephen Hines0b7ef1a2010-10-07 16:36:59 -0700364 std::string Error;
Zonr Chang8c6d9b22010-10-07 18:01:19 +0800365
Stephen Hines8f4d9722011-12-05 14:14:42 -0800366 mDOS.reset(OpenOutputFile(OutputFile, 0, &Error, mDiagEngine));
Zonr Change8c263a2010-10-12 00:35:29 +0800367 if (!Error.empty() || (mDOS.get() == NULL))
Stephen Hines0b7ef1a2010-10-07 16:36:59 -0700368 return false;
Stephen Hines0b7ef1a2010-10-07 16:36:59 -0700369
370 mDepOutputFileName = OutputFile;
371
372 return true;
373}
374
Stephen Hinescc0efad2010-10-04 16:13:02 -0700375int Slang::generateDepFile() {
Logan Chien9207a2e2011-10-21 15:39:28 +0800376 if (mDiagEngine->hasErrorOccurred())
Loganbe274822011-02-16 22:02:54 +0800377 return 1;
Stephen Hines0b7ef1a2010-10-07 16:36:59 -0700378 if (mDOS.get() == NULL)
Stephen Hinesf7de8522010-10-06 11:46:18 -0700379 return 1;
Stephen Hinescc0efad2010-10-04 16:13:02 -0700380
Zonr Change8c263a2010-10-12 00:35:29 +0800381 // Initialize options for generating dependency file
Stephen Hinescc0efad2010-10-04 16:13:02 -0700382 clang::DependencyOutputOptions DepOpts;
383 DepOpts.IncludeSystemHeaders = 1;
Stephen Hines0b7ef1a2010-10-07 16:36:59 -0700384 DepOpts.OutputFile = mDepOutputFileName;
Shih-wei Liaob81c6a42010-10-10 14:15:00 -0700385 DepOpts.Targets = mAdditionalDepTargets;
Stephen Hinescc0efad2010-10-04 16:13:02 -0700386 DepOpts.Targets.push_back(mDepTargetBCFileName);
Stephen Hines4cc67fc2011-01-31 16:48:57 -0800387 for (std::vector<std::string>::const_iterator
388 I = mGeneratedFileNames.begin(), E = mGeneratedFileNames.end();
389 I != E;
390 I++) {
391 DepOpts.Targets.push_back(*I);
392 }
393 mGeneratedFileNames.clear();
Stephen Hinescc0efad2010-10-04 16:13:02 -0700394
Zonr Change8c263a2010-10-12 00:35:29 +0800395 // Per-compilation needed initialization
Stephen Hinescc0efad2010-10-04 16:13:02 -0700396 createPreprocessor();
397 AttachDependencyFileGen(*mPP.get(), DepOpts);
398
Zonr Change8c263a2010-10-12 00:35:29 +0800399 // Inform the diagnostic client we are processing a source file
Stephen Hinescc0efad2010-10-04 16:13:02 -0700400 mDiagClient->BeginSourceFile(LangOpts, mPP.get());
401
Zonr Change8c263a2010-10-12 00:35:29 +0800402 // Go through the source file (no operations necessary)
Stephen Hinescc0efad2010-10-04 16:13:02 -0700403 clang::Token Tok;
404 mPP->EnterMainSourceFile();
405 do {
406 mPP->Lex(Tok);
407 } while (Tok.isNot(clang::tok::eof));
408
409 mPP->EndSourceFile();
410
Zonr Change8c263a2010-10-12 00:35:29 +0800411 // Declare success if no error
Logan Chien9207a2e2011-10-21 15:39:28 +0800412 if (!mDiagEngine->hasErrorOccurred())
Zonr Change8c263a2010-10-12 00:35:29 +0800413 mDOS->keep();
414
415 // Clean up after compilation
Stephen Hinescc0efad2010-10-04 16:13:02 -0700416 mPP.reset();
Zonr Change8c263a2010-10-12 00:35:29 +0800417 mDOS.reset();
Stephen Hinescc0efad2010-10-04 16:13:02 -0700418
Logan Chien9207a2e2011-10-21 15:39:28 +0800419 return mDiagEngine->hasErrorOccurred() ? 1 : 0;
Stephen Hinescc0efad2010-10-04 16:13:02 -0700420}
421
Shih-wei Liao462aefd2010-06-04 15:32:04 -0700422int Slang::compile() {
Logan Chien9207a2e2011-10-21 15:39:28 +0800423 if (mDiagEngine->hasErrorOccurred())
Loganbe274822011-02-16 22:02:54 +0800424 return 1;
Stephen Hinesf7de8522010-10-06 11:46:18 -0700425 if (mOS.get() == NULL)
426 return 1;
Shih-wei Liao9ef2f782010-10-01 12:31:37 -0700427
428 // Here is per-compilation needed initialization
429 createPreprocessor();
430 createASTContext();
Zonr Chang3a9ca1f2010-10-06 17:52:56 +0800431
Shih-wei Liaodf5bcce2011-02-28 18:39:23 -0800432 mBackend.reset(createBackend(CodeGenOpts, &mOS->os(), mOT));
Shih-wei Liao9ef2f782010-10-01 12:31:37 -0700433
434 // Inform the diagnostic client we are processing a source file
435 mDiagClient->BeginSourceFile(LangOpts, mPP.get());
436
437 // The core of the slang compiler
438 ParseAST(*mPP, mBackend.get(), *mASTContext);
439
Zonr Change8c263a2010-10-12 00:35:29 +0800440 // Inform the diagnostic client we are done with previous source file
441 mDiagClient->EndSourceFile();
442
443 // Declare success if no error
Logan Chien9207a2e2011-10-21 15:39:28 +0800444 if (!mDiagEngine->hasErrorOccurred())
Zonr Change8c263a2010-10-12 00:35:29 +0800445 mOS->keep();
446
Zonr Chang3a9ca1f2010-10-06 17:52:56 +0800447 // The compilation ended, clear
Shih-wei Liao9ef2f782010-10-01 12:31:37 -0700448 mBackend.reset();
Shih-wei Liao9ef2f782010-10-01 12:31:37 -0700449 mASTContext.reset();
450 mPP.reset();
Zonr Change8c263a2010-10-12 00:35:29 +0800451 mOS.reset();
Shih-wei Liao9ef2f782010-10-01 12:31:37 -0700452
Logan Chien9207a2e2011-10-21 15:39:28 +0800453 return mDiagEngine->hasErrorOccurred() ? 1 : 0;
Shih-wei Liao462aefd2010-06-04 15:32:04 -0700454}
455
mkopec1c460b372012-01-09 11:21:50 -0500456void Slang::setDebugMetadataEmission(bool EmitDebug) {
457 CodeGenOpts.DebugInfo = EmitDebug;
458}
459
460void Slang::setOptimizationLevel(llvm::CodeGenOpt::Level OptimizationLevel) {
461 CodeGenOpts.OptimizationLevel = OptimizationLevel;
462}
463
Zonr Chang3a9ca1f2010-10-06 17:52:56 +0800464void Slang::reset() {
Stephen Hinesc632be22011-09-23 15:53:16 -0700465 llvm::errs() << mDiagClient->str();
Logan Chien9207a2e2011-10-21 15:39:28 +0800466 mDiagEngine->Reset();
Zonr Chang3a9ca1f2010-10-06 17:52:56 +0800467 mDiagClient->reset();
Shih-wei Liao462aefd2010-06-04 15:32:04 -0700468}
469
470Slang::~Slang() {
Shih-wei Liao9ef2f782010-10-01 12:31:37 -0700471 llvm::llvm_shutdown();
Shih-wei Liao462aefd2010-06-04 15:32:04 -0700472}
Stephen Hinese639eb52010-11-08 19:27:20 -0800473
474} // namespace slang