blob: f3e933d8c8142c6a0ec644cfc97e5e0ca5c4c439 [file] [log] [blame]
Logan3f3d31f2010-11-27 13:52:03 +08001/*
Stephen Hinescc366e52012-02-21 17:22:04 -08002 * Copyright 2010-2012, The Android Open Source Project
Logan3f3d31f2010-11-27 13:52:03 +08003 *
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_SCRIPT_H
18#define BCC_SCRIPT_H
19
20#include <bcc/bcc.h>
Loganf340bf72011-01-14 17:51:40 +080021#include "bcc_internal.h"
Logan3f3d31f2010-11-27 13:52:03 +080022
Loganecf4cbd2011-01-06 05:34:11 +080023#include "Compiler.h"
24
Shih-wei Liao8afed382012-01-10 15:57:24 +080025#include <llvm/Support/CodeGen.h>
26
Logan Chien7890d432011-08-03 14:55:17 +080027#include <vector>
28#include <string>
29
Loganbe79ada2011-01-13 01:33:45 +080030#include <stddef.h>
31
Loganeaa0cc32010-12-29 01:04:20 +080032namespace llvm {
33 class Module;
Daniel Malea094881f2011-12-14 17:39:16 -050034 class GDBJITRegistrar;
Loganeaa0cc32010-12-29 01:04:20 +080035}
36
Logan3f3d31f2010-11-27 13:52:03 +080037namespace bcc {
Logancf3e5212010-12-29 01:44:55 +080038 class ScriptCompiled;
39 class ScriptCached;
Logan474cbd22011-01-31 01:47:44 +080040 class SourceInfo;
Zonr Chang2fcbd022012-01-06 21:04:31 +080041 struct CompilerOption;
Logancf3e5212010-12-29 01:44:55 +080042
43 namespace ScriptStatus {
44 enum StatusType {
45 Unknown,
46 Compiled,
Logan35849002011-01-15 07:30:43 +080047#if USE_CACHE
Loganf7f0ac52011-01-07 03:53:43 +080048 Cached,
Logan35849002011-01-15 07:30:43 +080049#endif
Logancf3e5212010-12-29 01:44:55 +080050 };
51 }
Logan3f3d31f2010-11-27 13:52:03 +080052
Zonr Chang4ea08862012-01-17 17:26:49 +080053 namespace ScriptObject {
54 enum ObjectType {
55 Unknown,
56 Relocatable,
Shih-wei Liaoa471ebb2012-02-05 00:49:58 -080057 SharedObject,
Zonr Chang4ea08862012-01-17 17:26:49 +080058 Executable,
59 };
60 }
61
Logan0647e9e2010-12-29 00:21:31 +080062 class Script {
Logan39736412010-12-29 00:24:04 +080063 private:
Loganf340bf72011-01-14 17:51:40 +080064 int mErrorCode;
Logan39736412010-12-29 00:24:04 +080065
Logancf3e5212010-12-29 01:44:55 +080066 ScriptStatus::StatusType mStatus;
Shih-wei Liao8454a3a2012-03-03 01:50:08 -080067 // The type of the object behind this script after compilation. For
68 // example, after returning from a successful call to prepareRelocatable(),
69 // the value of mObjectType will be ScriptObject::Relocatable.
Zonr Chang4ea08862012-01-17 17:26:49 +080070 ScriptObject::ObjectType mObjectType;
Logancf3e5212010-12-29 01:44:55 +080071
72 union {
73 ScriptCompiled *mCompiled;
Logan35849002011-01-15 07:30:43 +080074#if USE_CACHE
Logancf3e5212010-12-29 01:44:55 +080075 ScriptCached *mCached;
Logan35849002011-01-15 07:30:43 +080076#endif
Logancf3e5212010-12-29 01:44:55 +080077 };
78
Logan Chien311c26f2011-07-11 14:30:34 +080079#if USE_CACHE
80 std::string mCacheDir;
81 std::string mCacheName;
Zonr Chang4ea08862012-01-17 17:26:49 +080082
83 inline std::string getCachedObjectPath() const {
Shih-wei Liao8454a3a2012-03-03 01:50:08 -080084 return std::string(mCacheDir + mCacheName + ".o");
Zonr Chang4ea08862012-01-17 17:26:49 +080085 }
86
87 inline std::string getCacheInfoPath() const {
Zonr Chang6692ab52012-01-17 17:29:13 +080088 return getCachedObjectPath().append(".info");
Zonr Chang4ea08862012-01-17 17:26:49 +080089 }
Logan Chien311c26f2011-07-11 14:30:34 +080090#endif
Loganecf4cbd2011-01-06 05:34:11 +080091
Logan42598052011-01-26 22:41:13 +080092 bool mIsContextSlotNotAvail;
93
Logan474cbd22011-01-31 01:47:44 +080094 // Source List
95 SourceInfo *mSourceList[2];
96 // Note: mSourceList[0] (main source)
97 // Note: mSourceList[1] (library source)
98 // TODO(logan): Generalize this, use vector or SmallVector instead!
Logan3133c412011-01-06 06:15:40 +080099
Logan Chien7890d432011-08-03 14:55:17 +0800100 // External Function List
101 std::vector<char const *> mUserDefinedExternalSymbols;
102
Loganecf4cbd2011-01-06 05:34:11 +0800103 // Register Symbol Lookup Function
Logancf3e5212010-12-29 01:44:55 +0800104 BCCSymbolLookupFn mpExtSymbolLookupFn;
Loganf340bf72011-01-14 17:51:40 +0800105 void *mpExtSymbolLookupFnContext;
Loganeaa0cc32010-12-29 01:04:20 +0800106
Logan3f3d31f2010-11-27 13:52:03 +0800107 public:
Logan033f46e2011-01-06 05:51:24 +0800108 Script() : mErrorCode(BCC_NO_ERROR), mStatus(ScriptStatus::Unknown),
Zonr Chang4ea08862012-01-17 17:26:49 +0800109 mObjectType(ScriptObject::Unknown), mIsContextSlotNotAvail(false),
Logan65719812011-01-07 11:17:14 +0800110 mpExtSymbolLookupFn(NULL), mpExtSymbolLookupFnContext(NULL) {
Loganecf4cbd2011-01-06 05:34:11 +0800111 Compiler::GlobalInitialization();
Logan474cbd22011-01-31 01:47:44 +0800112
113 mSourceList[0] = NULL;
114 mSourceList[1] = NULL;
Logan3f3d31f2010-11-27 13:52:03 +0800115 }
116
Logancf3e5212010-12-29 01:44:55 +0800117 ~Script();
Loganeaa0cc32010-12-29 01:04:20 +0800118
Logan474cbd22011-01-31 01:47:44 +0800119 int addSourceBC(size_t idx,
120 char const *resName,
121 const char *bitcode,
122 size_t bitcodeSize,
123 unsigned long flags);
Loganeaa0cc32010-12-29 01:04:20 +0800124
Logan474cbd22011-01-31 01:47:44 +0800125 int addSourceModule(size_t idx,
126 llvm::Module *module,
127 unsigned long flags);
Loganecf4cbd2011-01-06 05:34:11 +0800128
Logan474cbd22011-01-31 01:47:44 +0800129 int addSourceFile(size_t idx,
130 char const *path,
131 unsigned long flags);
Loganeaa0cc32010-12-29 01:04:20 +0800132
Logan Chien7890d432011-08-03 14:55:17 +0800133 void markExternalSymbol(char const *name) {
134 mUserDefinedExternalSymbols.push_back(name);
135 }
136
137 std::vector<char const *> const &getUserDefinedExternalSymbols() const {
138 return mUserDefinedExternalSymbols;
139 }
140
Logan Chien311c26f2011-07-11 14:30:34 +0800141 int prepareExecutable(char const *cacheDir,
142 char const *cacheName,
143 unsigned long flags);
Shih-wei Liaoa0ed34e2012-03-03 01:33:30 -0800144 int writeCache();
Loganeaa0cc32010-12-29 01:04:20 +0800145
Shih-wei Liaoa471ebb2012-02-05 00:49:58 -0800146 /*
147 * Link the given bitcodes in mSourceList to shared object (.so).
148 *
149 * Currently, it requires one to provide the relocatable object files with
150 * given bitcodes to output a shared object.
151 *
152 * The usage of this function is flexible. You can have a relocatable object
153 * compiled before and pass it in objPath to generate shared object. If the
154 * objPath is NULL, we'll invoke prepareRelocatable() to get .o first (if
155 * you haven't done that yet) and then link the output relocatable object
Shih-wei Liao69341742012-03-03 01:45:36 -0800156 * file to .so in dsoPath.
Shih-wei Liaoa471ebb2012-02-05 00:49:58 -0800157 *
158 * TODO: Currently, we only support to link the bitcodes in mSourceList[0].
159 *
160 */
Shih-wei Liao69341742012-03-03 01:45:36 -0800161 int prepareSharedObject(char const *objPath,
Shih-wei Liaoa471ebb2012-02-05 00:49:58 -0800162 char const *dsoPath,
163 unsigned long flags);
164
Shih-wei Liaod8ed6a92012-03-03 01:44:29 -0800165 int prepareRelocatable(char const *objPath,
Shih-wei Liao6a60f4e2012-01-17 02:58:40 -0800166 llvm::Reloc::Model RelocModel,
167 unsigned long flags);
Joseph Wen34c600a2011-07-25 17:59:17 -0700168
Logancf3e5212010-12-29 01:44:55 +0800169 char const *getCompilerErrorMessage();
Loganeaa0cc32010-12-29 01:04:20 +0800170
Logancf3e5212010-12-29 01:44:55 +0800171 void *lookup(const char *name);
Loganeaa0cc32010-12-29 01:04:20 +0800172
Loganeaa0cc32010-12-29 01:04:20 +0800173
Loganbe79ada2011-01-13 01:33:45 +0800174 size_t getExportVarCount() const;
Loganeaa0cc32010-12-29 01:04:20 +0800175
Loganbe79ada2011-01-13 01:33:45 +0800176 size_t getExportFuncCount() const;
Loganeaa0cc32010-12-29 01:04:20 +0800177
Stephen Hinescc366e52012-02-21 17:22:04 -0800178 size_t getExportForEachCount() const;
179
Loganbe79ada2011-01-13 01:33:45 +0800180 size_t getPragmaCount() const;
Loganeaa0cc32010-12-29 01:04:20 +0800181
Loganbe79ada2011-01-13 01:33:45 +0800182 size_t getFuncCount() const;
Loganeaa0cc32010-12-29 01:04:20 +0800183
Stephen Hines071288a2011-01-27 14:38:26 -0800184 size_t getObjectSlotCount() const;
Loganbe79ada2011-01-13 01:33:45 +0800185
186 void getExportVarList(size_t size, void **list);
187
188 void getExportFuncList(size_t size, void **list);
189
Stephen Hinescc366e52012-02-21 17:22:04 -0800190 void getExportForEachList(size_t size, void **list);
191
Joseph Wenf36637f2011-07-06 18:27:12 -0700192 void getExportVarNameList(std::vector<std::string> &list);
193
194 void getExportFuncNameList(std::vector<std::string> &list);
195
Stephen Hinescc366e52012-02-21 17:22:04 -0800196 void getExportForEachNameList(std::vector<std::string> &list);
197
Loganbe79ada2011-01-13 01:33:45 +0800198 void getPragmaList(size_t size,
199 char const **keyList,
200 char const **valueList);
201
Loganf340bf72011-01-14 17:51:40 +0800202 void getFuncInfoList(size_t size, FuncInfo *list);
Loganbe79ada2011-01-13 01:33:45 +0800203
Stephen Hines071288a2011-01-27 14:38:26 -0800204 void getObjectSlotList(size_t size, uint32_t *list);
205
Shih-wei Liao5e3e0ce2011-06-17 13:59:46 -0700206 size_t getELFSize() const;
207
208 const char *getELF() const;
Logan02286cb2011-01-07 00:30:47 +0800209
Shih-wei Liaoce82d492011-01-20 12:34:03 -0800210 int registerSymbolCallback(BCCSymbolLookupFn pFn, void *pContext);
Loganeaa0cc32010-12-29 01:04:20 +0800211
Shih-wei Liaoabc7f512012-01-18 00:34:07 -0800212 bool isCacheable() const;
Loganeaa0cc32010-12-29 01:04:20 +0800213
Loganf340bf72011-01-14 17:51:40 +0800214 void setError(int error) {
Logan39736412010-12-29 00:24:04 +0800215 if (mErrorCode == BCC_NO_ERROR && error != BCC_NO_ERROR) {
216 mErrorCode = error;
Logan3f3d31f2010-11-27 13:52:03 +0800217 }
218 }
219
Loganf340bf72011-01-14 17:51:40 +0800220 int getError() {
221 int result = mErrorCode;
Logan39736412010-12-29 00:24:04 +0800222 mErrorCode = BCC_NO_ERROR;
Logan3f3d31f2010-11-27 13:52:03 +0800223 return result;
224 }
Logan033f46e2011-01-06 05:51:24 +0800225
226 private:
Logan35849002011-01-15 07:30:43 +0800227#if USE_CACHE
Zonr Chang743dd712012-01-19 10:13:52 +0800228 //
229 // It returns 0 if there's a cache hit.
230 //
Shih-wei Liao8454a3a2012-03-03 01:50:08 -0800231 // Side effect: it will set mCacheDir, mCacheName.
Zonr Chang743dd712012-01-19 10:13:52 +0800232 int internalLoadCache(char const *cacheDir, char const *cacheName,
Shih-wei Liao8454a3a2012-03-03 01:50:08 -0800233 bool checkOnly);
Logan35849002011-01-15 07:30:43 +0800234#endif
Shih-wei Liao9e81e372012-01-17 16:38:40 -0800235 int internalCompile(const CompilerOption&);
Logan3f3d31f2010-11-27 13:52:03 +0800236 };
237
238} // namespace bcc
239
240#endif // BCC_SCRIPT_H