/* * Copyright 2010-2012, 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 BCC_COMPILER_H #define BCC_COMPILER_H namespace llvm { class raw_ostream; class raw_pwrite_stream; class DataLayout; class TargetMachine; namespace legacy { class PassManager; } // end namespace legacy } // end namespace llvm namespace bcc { class CompilerConfig; class Script; //===----------------------------------------------------------------------===// // Design of Compiler //===----------------------------------------------------------------------===// // 1. A compiler instance can be constructed provided an "initial config." // 2. A compiler can later be re-configured using config(). // 3. Once config() is invoked, it'll re-create TargetMachine instance (i.e., // mTarget) according to the configuration supplied. TargetMachine instance // is *shared* across the different calls to compile() before the next call // to config(). // 4. Once a compiler instance is created, you can use the compile() service // to compile the file over and over again. Each call uses TargetMachine // instance to construct the compilation passes. class Compiler { public: enum ErrorCode { kSuccess, kInvalidConfigNoTarget, kErrCreateTargetMachine, kErrSwitchTargetMachine, kErrNoTargetMachine, kErrMaterialization, kErrInvalidOutputFileState, kErrPrepareOutput, kPrepareCodeGenPass, kErrCustomPasses, kErrInvalidSource, kIllegalGlobalFunction, kErrInvalidTargetMachine, kErrInvalidLayout }; static const char *GetErrorString(enum ErrorCode pErrCode); private: llvm::TargetMachine *mTarget; // Optimization is enabled by default. bool mEnableOpt; enum ErrorCode runPasses(Script &pScript, llvm::raw_pwrite_stream &pResult); bool addInternalizeSymbolsPass(Script &pScript, llvm::legacy::PassManager &pPM); void addExpandKernelPass(llvm::legacy::PassManager &pPM); void addDebugInfoPass(Script &pScript, llvm::legacy::PassManager &pPM); void addGlobalInfoPass(Script &pScript, llvm::legacy::PassManager &pPM); void addInvariantPass(llvm::legacy::PassManager &pPM); void addInvokeHelperPass(llvm::legacy::PassManager &pPM); public: Compiler(); explicit Compiler(const CompilerConfig &pConfig); enum ErrorCode config(const CompilerConfig &pConfig); // Compile a script and output the result to a LLVM stream. // // @param IRStream If not NULL, the LLVM-IR that is fed to code generation // will be written to IRStream. enum ErrorCode compile(Script &pScript, llvm::raw_pwrite_stream &pResult, llvm::raw_ostream *IRStream); const llvm::TargetMachine& getTargetMachine() const { return *mTarget; } void enableOpt(bool pEnable = true) { mEnableOpt = pEnable; } ~Compiler(); // Compare undefined external functions in pScript against a 'whitelist' of // all RenderScript functions. Returns error if any external function that is // not in this whitelist is callable from the script. enum ErrorCode screenGlobalFunctions(Script &pScript); void translateGEPs(Script &pScript); }; } // end namespace bcc #endif // BCC_COMPILER_H