/* * Copyright 2010, The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #ifndef _FRAMEWORKS_COMPILE_SLANG_SLANG_H_ // NOLINT #define _FRAMEWORKS_COMPILE_SLANG_SLANG_H_ #include #include #include // Terrible workaround for TargetOptions.h not using llvm::RefCountedBase! #include "llvm/ADT/IntrusiveRefCntPtr.h" using llvm::RefCountedBase; #include "clang/Basic/TargetOptions.h" #include "clang/Lex/ModuleLoader.h" #include "llvm/ADT/StringRef.h" #include "llvm/Target/TargetMachine.h" #include "slang_diagnostic_buffer.h" #include "slang_pragma_recorder.h" namespace llvm { class tool_output_file; } namespace clang { class ASTConsumer; class ASTContext; class Backend; class CodeGenOptions; class Diagnostic; class DiagnosticsEngine; class FileManager; class FileSystemOptions; class LangOptions; class Preprocessor; class SourceManager; class TargetInfo; } // namespace clang namespace slang { class Slang : public clang::ModuleLoader { static clang::LangOptions LangOpts; static clang::CodeGenOptions CodeGenOpts; static bool GlobalInitialized; static void LLVMErrorHandler(void *UserData, const std::string &Message, bool GenCrashDialog); public: enum OutputType { OT_Dependency, OT_Assembly, OT_LLVMAssembly, OT_Bitcode, OT_Nothing, OT_Object, OT_Default = OT_Bitcode }; private: bool mInitialized; // Diagnostics Mediator (An interface for both Producer and Consumer) std::unique_ptr mDiag; // Diagnostics Engine (Producer and Diagnostics Reporter) clang::DiagnosticsEngine *mDiagEngine; // Diagnostics Consumer // NOTE: The ownership is taken by mDiagEngine after creation. DiagnosticBuffer *mDiagClient; // The target being compiled for std::shared_ptr mTargetOpts; std::unique_ptr mTarget; void createTarget(uint32_t BitWidth); // File manager (for prepocessor doing the job such as header file search) std::unique_ptr mFileMgr; std::unique_ptr mFileSysOpt; void createFileManager(); // Source manager (responsible for the source code handling) std::unique_ptr mSourceMgr; void createSourceManager(); // Preprocessor (source code preprocessor) std::unique_ptr mPP; void createPreprocessor(); // AST context (the context to hold long-lived AST nodes) std::unique_ptr mASTContext; void createASTContext(); // AST consumer, responsible for code generation std::unique_ptr mBackend; // File names std::string mInputFileName; std::string mOutputFileName; std::string mOutput32FileName; std::string mDepOutputFileName; std::string mDepTargetBCFileName; std::vector mAdditionalDepTargets; std::vector mGeneratedFileNames; OutputType mOT; // Output stream std::unique_ptr mOS; // Dependency output stream std::unique_ptr mDOS; std::vector mIncludePaths; protected: PragmaList mPragmas; clang::DiagnosticsEngine &getDiagnostics() { return *mDiagEngine; } clang::TargetInfo const &getTargetInfo() const { return *mTarget; } clang::FileManager &getFileManager() { return *mFileMgr; } clang::SourceManager &getSourceManager() { return *mSourceMgr; } clang::Preprocessor &getPreprocessor() { return *mPP; } clang::ASTContext &getASTContext() { return *mASTContext; } inline clang::TargetOptions const &getTargetOptions() const { return *mTargetOpts.get(); } virtual void initDiagnostic() {} virtual void initPreprocessor() {} virtual void initASTContext() {} virtual clang::ASTConsumer * createBackend(const clang::CodeGenOptions& CodeGenOpts, llvm::raw_ostream *OS, OutputType OT); public: static const llvm::StringRef PragmaMetadataName; static void GlobalInitialization(); Slang(); void init(uint32_t BitWidth, clang::DiagnosticsEngine *DiagEngine, DiagnosticBuffer *DiagClient); virtual clang::ModuleLoadResult loadModule( clang::SourceLocation ImportLoc, clang::ModuleIdPath Path, clang::Module::NameVisibilityKind VK, bool IsInclusionDirective); bool setInputSource(llvm::StringRef InputFile, const char *Text, size_t TextLength); bool setInputSource(llvm::StringRef InputFile); std::string const &getInputFileName() const { return mInputFileName; } void setIncludePaths(const std::vector &IncludePaths) { mIncludePaths = IncludePaths; } void setOutputType(OutputType OT) { mOT = OT; } bool setOutput(const char *OutputFile); // For use with 64-bit compilation/reflection. This only sets the filename of // the 32-bit bitcode file, and doesn't actually verify it already exists. void setOutput32(const char *OutputFile) { mOutput32FileName = OutputFile; } std::string const &getOutputFileName() const { return mOutputFileName; } std::string const &getOutput32FileName() const { return mOutput32FileName; } bool setDepOutput(const char *OutputFile); void setDepTargetBC(const char *TargetBCFile) { mDepTargetBCFileName = TargetBCFile; } void setAdditionalDepTargets( std::vector const &AdditionalDepTargets) { mAdditionalDepTargets = AdditionalDepTargets; } void appendGeneratedFileName(std::string const &GeneratedFileName) { mGeneratedFileNames.push_back(GeneratedFileName); } int generateDepFile(); int compile(); char const *getErrorMessage() { return mDiagClient->str().c_str(); } void setDebugMetadataEmission(bool EmitDebug); void setOptimizationLevel(llvm::CodeGenOpt::Level OptimizationLevel); // Reset the slang compiler state such that it can be reused to compile // another file virtual void reset(bool SuppressWarnings = false); virtual ~Slang(); }; } // namespace slang #endif // _FRAMEWORKS_COMPILE_SLANG_SLANG_H_ NOLINT