• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright 2010-2012, 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 BCC_COMPILER_H
18 #define BCC_COMPILER_H
19 
20 namespace llvm {
21 
22 class raw_ostream;
23 class PassManager;
24 class DataLayout;
25 class TargetMachine;
26 
27 } // end namespace llvm
28 
29 namespace bcc {
30 
31 class CompilerConfig;
32 class OutputFile;
33 class Script;
34 
35 //===----------------------------------------------------------------------===//
36 // Design of Compiler
37 //===----------------------------------------------------------------------===//
38 // 1. A compiler instance can be constructed provided an "initial config."
39 // 2. A compiler can later be re-configured using config().
40 // 3. Once config() is invoked, it'll re-create TargetMachine instance (i.e.,
41 //    mTarget) according to the configuration supplied. TargetMachine instance
42 //    is *shared* across the different calls to compile() before the next call
43 //    to config().
44 // 4. Once a compiler instance is created, you can use the compile() service
45 //    to compile the file over and over again. Each call uses TargetMachine
46 //    instance to construct the compilation passes.
47 class Compiler {
48 public:
49   enum ErrorCode {
50     kSuccess,
51 
52     kInvalidConfigNoTarget,
53     kErrCreateTargetMachine,
54     kErrSwitchTargetMachine,
55     kErrNoTargetMachine,
56     kErrDataLayoutNoMemory,
57     kErrMaterialization,
58     kErrInvalidOutputFileState,
59     kErrPrepareOutput,
60     kPrepareCodeGenPass,
61 
62     kErrHookBeforeAddLTOPasses,
63     kErrHookAfterAddLTOPasses,
64     kErrHookAfterExecuteLTOPasses,
65 
66     kErrHookBeforeAddCodeGenPasses,
67     kErrHookAfterAddCodeGenPasses,
68     kErrHookBeforeExecuteCodeGenPasses,
69     kErrHookAfterExecuteCodeGenPasses,
70 
71     kErrInvalidSource
72   };
73 
74   static const char *GetErrorString(enum ErrorCode pErrCode);
75 
76 private:
77   llvm::TargetMachine *mTarget;
78   // LTO is enabled by default.
79   bool mEnableLTO;
80 
81   enum ErrorCode runLTO(Script &pScript);
82   enum ErrorCode runCodeGen(Script &pScript, llvm::raw_ostream &pResult);
83 
84 public:
85   Compiler();
86   Compiler(const CompilerConfig &pConfig);
87 
88   enum ErrorCode config(const CompilerConfig &pConfig);
89 
90   // Compile a script and output the result to a LLVM stream.
91   //
92   // @param IRStream If not NULL, the LLVM-IR that is fed to code generation
93   //                 will be written to IRStream.
94   enum ErrorCode compile(Script &pScript, llvm::raw_ostream &pResult,
95                          llvm::raw_ostream *IRStream);
96 
97   // Compile a script and output the result to a file.
98   enum ErrorCode compile(Script &pScript, OutputFile &pResult,
99                          llvm::raw_ostream *IRStream = 0);
100 
getTargetMachine()101   const llvm::TargetMachine& getTargetMachine() const
102   { return *mTarget; }
103 
104   void enableLTO(bool pEnable = true)
105   { mEnableLTO = pEnable; }
106 
107   virtual ~Compiler();
108 
109 protected:
110   //===--------------------------------------------------------------------===//
111   // Plugin callbacks for sub-class.
112   //===--------------------------------------------------------------------===//
113   // Called before adding first pass to code-generation passes.
beforeAddLTOPasses(Script & pScript,llvm::PassManager & pPM)114   virtual bool beforeAddLTOPasses(Script &pScript, llvm::PassManager &pPM)
115   { return true; }
116 
117   // Called after adding last pass to code-generation passes.
afterAddLTOPasses(Script & pScript,llvm::PassManager & pPM)118   virtual bool afterAddLTOPasses(Script &pScript, llvm::PassManager &pPM)
119   { return true; }
120 
121   // Called before executing code-generation passes.
beforeExecuteLTOPasses(Script & pScript,llvm::PassManager & pPM)122   virtual bool beforeExecuteLTOPasses(Script &pScript,
123                                           llvm::PassManager &pPM)
124   { return true; }
125 
126   // Called after executing code-generation passes.
afterExecuteLTOPasses(Script & pScript)127   virtual bool afterExecuteLTOPasses(Script &pScript)
128   { return true; }
129 
130   // Called before adding first pass to code-generation passes.
beforeAddCodeGenPasses(Script & pScript,llvm::PassManager & pPM)131   virtual bool beforeAddCodeGenPasses(Script &pScript, llvm::PassManager &pPM)
132   { return true; }
133 
134   // Called after adding last pass to code-generation passes.
afterAddCodeGenPasses(Script & pScript,llvm::PassManager & pPM)135   virtual bool afterAddCodeGenPasses(Script &pScript, llvm::PassManager &pPM)
136   { return true; }
137 
138   // Called before executing code-generation passes.
beforeExecuteCodeGenPasses(Script & pScript,llvm::PassManager & pPM)139   virtual bool beforeExecuteCodeGenPasses(Script &pScript,
140                                           llvm::PassManager &pPM)
141   { return true; }
142 
143   // Called after executing code-generation passes.
afterExecuteCodeGenPasses(Script & pScript)144   virtual bool afterExecuteCodeGenPasses(Script &pScript)
145   { return true; }
146 };
147 
148 } // end namespace bcc
149 
150 #endif // BCC_COMPILER_H
151