1 /*
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 
17 #ifndef _FRAMEWORKS_COMPILE_SLANG_SLANG_H_  // NOLINT
18 #define _FRAMEWORKS_COMPILE_SLANG_SLANG_H_
19 
20 #include <cstdio>
21 #include <string>
22 #include <vector>
23 
24 // Terrible workaround for TargetOptions.h not using llvm::RefCountedBase!
25 #include "llvm/ADT/IntrusiveRefCntPtr.h"
26 using llvm::RefCountedBase;
27 
28 #include "clang/Basic/TargetOptions.h"
29 #include "clang/Lex/ModuleLoader.h"
30 
31 #include "llvm/ADT/StringRef.h"
32 
33 #include "llvm/Target/TargetMachine.h"
34 
35 #include "slang_diagnostic_buffer.h"
36 #include "slang_pragma_recorder.h"
37 
38 namespace llvm {
39   class tool_output_file;
40 }
41 
42 namespace clang {
43   class ASTConsumer;
44   class ASTContext;
45   class Backend;
46   class CodeGenOptions;
47   class Diagnostic;
48   class DiagnosticsEngine;
49   class FileManager;
50   class FileSystemOptions;
51   class LangOptions;
52   class Preprocessor;
53   class SourceManager;
54   class TargetInfo;
55 }  // namespace clang
56 
57 namespace slang {
58 
59 class Slang : public clang::ModuleLoader {
60   static clang::LangOptions LangOpts;
61   static clang::CodeGenOptions CodeGenOpts;
62 
63   static bool GlobalInitialized;
64 
65   static void LLVMErrorHandler(void *UserData, const std::string &Message,
66                                bool GenCrashDialog);
67 
68  public:
69   enum OutputType {
70     OT_Dependency,
71     OT_Assembly,
72     OT_LLVMAssembly,
73     OT_Bitcode,
74     OT_Nothing,
75     OT_Object,
76 
77     OT_Default = OT_Bitcode
78   };
79 
80  private:
81   bool mInitialized;
82 
83   // Diagnostics Mediator (An interface for both Producer and Consumer)
84   std::unique_ptr<clang::Diagnostic> mDiag;
85 
86   // Diagnostics Engine (Producer and Diagnostics Reporter)
87   clang::DiagnosticsEngine *mDiagEngine;
88 
89   // Diagnostics Consumer
90   // NOTE: The ownership is taken by mDiagEngine after creation.
91   DiagnosticBuffer *mDiagClient;
92 
93   // The target being compiled for
94   std::shared_ptr<clang::TargetOptions> mTargetOpts;
95   std::unique_ptr<clang::TargetInfo> mTarget;
96   void createTarget(uint32_t BitWidth);
97 
98 
99   // File manager (for prepocessor doing the job such as header file search)
100   std::unique_ptr<clang::FileManager> mFileMgr;
101   std::unique_ptr<clang::FileSystemOptions> mFileSysOpt;
102   void createFileManager();
103 
104 
105   // Source manager (responsible for the source code handling)
106   std::unique_ptr<clang::SourceManager> mSourceMgr;
107   void createSourceManager();
108 
109 
110   // Preprocessor (source code preprocessor)
111   std::unique_ptr<clang::Preprocessor> mPP;
112   void createPreprocessor();
113 
114 
115   // AST context (the context to hold long-lived AST nodes)
116   std::unique_ptr<clang::ASTContext> mASTContext;
117   void createASTContext();
118 
119 
120   // AST consumer, responsible for code generation
121   std::unique_ptr<clang::ASTConsumer> mBackend;
122 
123 
124   // File names
125   std::string mInputFileName;
126   std::string mOutputFileName;
127   std::string mOutput32FileName;
128 
129   std::string mDepOutputFileName;
130   std::string mDepTargetBCFileName;
131   std::vector<std::string> mAdditionalDepTargets;
132   std::vector<std::string> mGeneratedFileNames;
133 
134   OutputType mOT;
135 
136   // Output stream
137   std::unique_ptr<llvm::tool_output_file> mOS;
138 
139   // Dependency output stream
140   std::unique_ptr<llvm::tool_output_file> mDOS;
141 
142   std::vector<std::string> mIncludePaths;
143 
144  protected:
145   PragmaList mPragmas;
146 
getDiagnostics()147   clang::DiagnosticsEngine &getDiagnostics() { return *mDiagEngine; }
getTargetInfo()148   clang::TargetInfo const &getTargetInfo() const { return *mTarget; }
getFileManager()149   clang::FileManager &getFileManager() { return *mFileMgr; }
getSourceManager()150   clang::SourceManager &getSourceManager() { return *mSourceMgr; }
getPreprocessor()151   clang::Preprocessor &getPreprocessor() { return *mPP; }
getASTContext()152   clang::ASTContext &getASTContext() { return *mASTContext; }
153 
getTargetOptions()154   inline clang::TargetOptions const &getTargetOptions() const
155     { return *mTargetOpts.get(); }
156 
initDiagnostic()157   virtual void initDiagnostic() {}
initPreprocessor()158   virtual void initPreprocessor() {}
initASTContext()159   virtual void initASTContext() {}
160 
161   virtual clang::ASTConsumer *
162     createBackend(const clang::CodeGenOptions& CodeGenOpts,
163                   llvm::raw_ostream *OS,
164                   OutputType OT);
165 
166  public:
167   static const llvm::StringRef PragmaMetadataName;
168 
169   static void GlobalInitialization();
170 
171   Slang();
172 
173   void init(uint32_t BitWidth, clang::DiagnosticsEngine *DiagEngine,
174             DiagnosticBuffer *DiagClient);
175 
176   virtual clang::ModuleLoadResult loadModule(
177       clang::SourceLocation ImportLoc,
178       clang::ModuleIdPath Path,
179       clang::Module::NameVisibilityKind VK,
180       bool IsInclusionDirective);
181 
182   bool setInputSource(llvm::StringRef InputFile, const char *Text,
183                       size_t TextLength);
184 
185   bool setInputSource(llvm::StringRef InputFile);
186 
getInputFileName()187   std::string const &getInputFileName() const { return mInputFileName; }
188 
setIncludePaths(const std::vector<std::string> & IncludePaths)189   void setIncludePaths(const std::vector<std::string> &IncludePaths) {
190     mIncludePaths = IncludePaths;
191   }
192 
setOutputType(OutputType OT)193   void setOutputType(OutputType OT) { mOT = OT; }
194 
195   bool setOutput(const char *OutputFile);
196 
197   // For use with 64-bit compilation/reflection. This only sets the filename of
198   // the 32-bit bitcode file, and doesn't actually verify it already exists.
setOutput32(const char * OutputFile)199   void setOutput32(const char *OutputFile) {
200     mOutput32FileName = OutputFile;
201   }
202 
getOutputFileName()203   std::string const &getOutputFileName() const {
204     return mOutputFileName;
205   }
206 
getOutput32FileName()207   std::string const &getOutput32FileName() const {
208     return mOutput32FileName;
209   }
210 
211   bool setDepOutput(const char *OutputFile);
212 
setDepTargetBC(const char * TargetBCFile)213   void setDepTargetBC(const char *TargetBCFile) {
214     mDepTargetBCFileName = TargetBCFile;
215   }
216 
setAdditionalDepTargets(std::vector<std::string> const & AdditionalDepTargets)217   void setAdditionalDepTargets(
218       std::vector<std::string> const &AdditionalDepTargets) {
219     mAdditionalDepTargets = AdditionalDepTargets;
220   }
221 
appendGeneratedFileName(std::string const & GeneratedFileName)222   void appendGeneratedFileName(std::string const &GeneratedFileName) {
223     mGeneratedFileNames.push_back(GeneratedFileName);
224   }
225 
226   int generateDepFile();
227 
228   int compile();
229 
getErrorMessage()230   char const *getErrorMessage() { return mDiagClient->str().c_str(); }
231 
232   void setDebugMetadataEmission(bool EmitDebug);
233 
234   void setOptimizationLevel(llvm::CodeGenOpt::Level OptimizationLevel);
235 
236   // Reset the slang compiler state such that it can be reused to compile
237   // another file
238   virtual void reset(bool SuppressWarnings = false);
239 
240   virtual ~Slang();
241 };
242 
243 }  // namespace slang
244 
245 #endif  // _FRAMEWORKS_COMPILE_SLANG_SLANG_H_  NOLINT
246