aot_orc_extra.cpp 8.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291
  1. /*
  2. * Copyright (C) 2019 Intel Corporation. All rights reserved.
  3. * SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
  4. */
  5. #include "llvm-c/LLJIT.h"
  6. #include "llvm-c/Orc.h"
  7. #include "llvm-c/OrcEE.h"
  8. #include "llvm-c/TargetMachine.h"
  9. #include "llvm/ADT/None.h"
  10. #include "llvm/ADT/Optional.h"
  11. #include "llvm/ExecutionEngine/Orc/JITTargetMachineBuilder.h"
  12. #include "llvm/ExecutionEngine/Orc/LLJIT.h"
  13. #include "llvm/ExecutionEngine/Orc/ObjectTransformLayer.h"
  14. #include "llvm/ExecutionEngine/Orc/RTDyldObjectLinkingLayer.h"
  15. #include "llvm/ExecutionEngine/Orc/CompileOnDemandLayer.h"
  16. #include "llvm/ExecutionEngine/SectionMemoryManager.h"
  17. #include "llvm/Support/CBindingWrapping.h"
  18. #include "aot_orc_extra.h"
  19. #include "aot.h"
  20. using namespace llvm;
  21. using namespace llvm::orc;
  22. using GlobalValueSet = std::set<const GlobalValue *>;
  23. namespace llvm {
  24. namespace orc {
  25. class InProgressLookupState;
  26. class OrcV2CAPIHelper
  27. {
  28. public:
  29. using PoolEntry = SymbolStringPtr::PoolEntry;
  30. using PoolEntryPtr = SymbolStringPtr::PoolEntryPtr;
  31. // Move from SymbolStringPtr to PoolEntryPtr (no change in ref count).
  32. static PoolEntryPtr moveFromSymbolStringPtr(SymbolStringPtr S)
  33. {
  34. PoolEntryPtr Result = nullptr;
  35. std::swap(Result, S.S);
  36. return Result;
  37. }
  38. // Move from a PoolEntryPtr to a SymbolStringPtr (no change in ref count).
  39. static SymbolStringPtr moveToSymbolStringPtr(PoolEntryPtr P)
  40. {
  41. SymbolStringPtr S;
  42. S.S = P;
  43. return S;
  44. }
  45. // Copy a pool entry to a SymbolStringPtr (increments ref count).
  46. static SymbolStringPtr copyToSymbolStringPtr(PoolEntryPtr P)
  47. {
  48. return SymbolStringPtr(P);
  49. }
  50. static PoolEntryPtr getRawPoolEntryPtr(const SymbolStringPtr &S)
  51. {
  52. return S.S;
  53. }
  54. static void retainPoolEntry(PoolEntryPtr P)
  55. {
  56. SymbolStringPtr S(P);
  57. S.S = nullptr;
  58. }
  59. static void releasePoolEntry(PoolEntryPtr P)
  60. {
  61. SymbolStringPtr S;
  62. S.S = P;
  63. }
  64. static InProgressLookupState *extractLookupState(LookupState &LS)
  65. {
  66. return LS.IPLS.release();
  67. }
  68. static void resetLookupState(LookupState &LS, InProgressLookupState *IPLS)
  69. {
  70. return LS.reset(IPLS);
  71. }
  72. };
  73. } // namespace orc
  74. } // namespace llvm
  75. // ORC.h
  76. DEFINE_SIMPLE_CONVERSION_FUNCTIONS(ExecutionSession, LLVMOrcExecutionSessionRef)
  77. DEFINE_SIMPLE_CONVERSION_FUNCTIONS(IRTransformLayer, LLVMOrcIRTransformLayerRef)
  78. DEFINE_SIMPLE_CONVERSION_FUNCTIONS(JITDylib, LLVMOrcJITDylibRef)
  79. DEFINE_SIMPLE_CONVERSION_FUNCTIONS(JITTargetMachineBuilder,
  80. LLVMOrcJITTargetMachineBuilderRef)
  81. DEFINE_SIMPLE_CONVERSION_FUNCTIONS(ObjectTransformLayer,
  82. LLVMOrcObjectTransformLayerRef)
  83. DEFINE_SIMPLE_CONVERSION_FUNCTIONS(OrcV2CAPIHelper::PoolEntry,
  84. LLVMOrcSymbolStringPoolEntryRef)
  85. DEFINE_SIMPLE_CONVERSION_FUNCTIONS(SymbolStringPool, LLVMOrcSymbolStringPoolRef)
  86. DEFINE_SIMPLE_CONVERSION_FUNCTIONS(ThreadSafeModule, LLVMOrcThreadSafeModuleRef)
  87. // LLJIT.h
  88. DEFINE_SIMPLE_CONVERSION_FUNCTIONS(LLJITBuilder, LLVMOrcLLJITBuilderRef)
  89. DEFINE_SIMPLE_CONVERSION_FUNCTIONS(LLLazyJITBuilder, LLVMOrcLLLazyJITBuilderRef)
  90. DEFINE_SIMPLE_CONVERSION_FUNCTIONS(LLLazyJIT, LLVMOrcLLLazyJITRef)
  91. void
  92. LLVMOrcLLJITBuilderSetNumCompileThreads(LLVMOrcLLJITBuilderRef Builder,
  93. unsigned NumCompileThreads)
  94. {
  95. unwrap(Builder)->setNumCompileThreads(NumCompileThreads);
  96. }
  97. LLVMOrcLLLazyJITBuilderRef
  98. LLVMOrcCreateLLLazyJITBuilder(void)
  99. {
  100. return wrap(new LLLazyJITBuilder());
  101. }
  102. void
  103. LLVMOrcDisposeLLLazyJITBuilder(LLVMOrcLLLazyJITBuilderRef Builder)
  104. {
  105. delete unwrap(Builder);
  106. }
  107. void
  108. LLVMOrcLLLazyJITBuilderSetNumCompileThreads(LLVMOrcLLLazyJITBuilderRef Builder,
  109. unsigned NumCompileThreads)
  110. {
  111. unwrap(Builder)->setNumCompileThreads(NumCompileThreads);
  112. }
  113. void
  114. LLVMOrcLLLazyJITBuilderSetJITTargetMachineBuilder(
  115. LLVMOrcLLLazyJITBuilderRef Builder, LLVMOrcJITTargetMachineBuilderRef JTMP)
  116. {
  117. unwrap(Builder)->setJITTargetMachineBuilder(*unwrap(JTMP));
  118. /* Destroy the JTMP, similar to
  119. LLVMOrcLLJITBuilderSetJITTargetMachineBuilder */
  120. LLVMOrcDisposeJITTargetMachineBuilder(JTMP);
  121. }
  122. static Optional<CompileOnDemandLayer::GlobalValueSet>
  123. PartitionFunction(GlobalValueSet Requested)
  124. {
  125. std::vector<const GlobalValue *> GVsToAdd;
  126. for (auto *GV : Requested) {
  127. if (isa<Function>(GV) && GV->hasName()) {
  128. auto &F = cast<Function>(*GV); /* get LLVM function */
  129. const Module *M = F.getParent(); /* get LLVM module */
  130. auto GVName = GV->getName(); /* get the function name */
  131. const char *gvname = GVName.begin(); /* C function name */
  132. const char *wrapper;
  133. uint32 prefix_len = strlen(AOT_FUNC_PREFIX);
  134. /* Convert "aot_func#n_wrapper" to "aot_func#n" */
  135. if (strstr(gvname, AOT_FUNC_PREFIX)
  136. && (wrapper = strstr(gvname + prefix_len, "_wrapper"))) {
  137. char buf[16] = { 0 };
  138. char func_name[64];
  139. int group_stride, i, j;
  140. bh_assert(wrapper - (gvname + prefix_len) > 0);
  141. /* Get AOT function index */
  142. bh_memcpy_s(buf, (uint32)sizeof(buf), gvname + prefix_len,
  143. (uint32)(wrapper - (gvname + prefix_len)));
  144. i = atoi(buf);
  145. group_stride = WASM_ORC_JIT_BACKEND_THREAD_NUM;
  146. /* Compile some functions each time */
  147. for (j = 0; j < WASM_ORC_JIT_COMPILE_THREAD_NUM; j++) {
  148. snprintf(func_name, sizeof(func_name), "%s%d",
  149. AOT_FUNC_PREFIX, i + j * group_stride);
  150. Function *F1 = M->getFunction(func_name);
  151. if (F1) {
  152. LOG_DEBUG("compile func %s", func_name);
  153. GVsToAdd.push_back(cast<GlobalValue>(F1));
  154. }
  155. }
  156. }
  157. }
  158. }
  159. for (auto *GV : GVsToAdd) {
  160. Requested.insert(GV);
  161. }
  162. return Requested;
  163. }
  164. LLVMErrorRef
  165. LLVMOrcCreateLLLazyJIT(LLVMOrcLLLazyJITRef *Result,
  166. LLVMOrcLLLazyJITBuilderRef Builder)
  167. {
  168. assert(Result && "Result can not be null");
  169. if (!Builder)
  170. Builder = LLVMOrcCreateLLLazyJITBuilder();
  171. auto J = unwrap(Builder)->create();
  172. LLVMOrcDisposeLLLazyJITBuilder(Builder);
  173. if (!J) {
  174. Result = nullptr;
  175. return 0;
  176. }
  177. LLLazyJIT *lazy_jit = J->release();
  178. lazy_jit->setPartitionFunction(PartitionFunction);
  179. *Result = wrap(lazy_jit);
  180. return LLVMErrorSuccess;
  181. }
  182. LLVMErrorRef
  183. LLVMOrcDisposeLLLazyJIT(LLVMOrcLLLazyJITRef J)
  184. {
  185. delete unwrap(J);
  186. return LLVMErrorSuccess;
  187. }
  188. LLVMErrorRef
  189. LLVMOrcLLLazyJITAddLLVMIRModule(LLVMOrcLLLazyJITRef J, LLVMOrcJITDylibRef JD,
  190. LLVMOrcThreadSafeModuleRef TSM)
  191. {
  192. std::unique_ptr<ThreadSafeModule> TmpTSM(unwrap(TSM));
  193. return wrap(unwrap(J)->addLazyIRModule(*unwrap(JD), std::move(*TmpTSM)));
  194. }
  195. LLVMErrorRef
  196. LLVMOrcLLLazyJITLookup(LLVMOrcLLLazyJITRef J, LLVMOrcExecutorAddress *Result,
  197. const char *Name)
  198. {
  199. assert(Result && "Result can not be null");
  200. auto Sym = unwrap(J)->lookup(Name);
  201. if (!Sym) {
  202. *Result = 0;
  203. return wrap(Sym.takeError());
  204. }
  205. #if LLVM_VERSION_MAJOR < 15
  206. *Result = Sym->getAddress();
  207. #else
  208. *Result = Sym->getValue();
  209. #endif
  210. return LLVMErrorSuccess;
  211. }
  212. LLVMOrcSymbolStringPoolEntryRef
  213. LLVMOrcLLLazyJITMangleAndIntern(LLVMOrcLLLazyJITRef J,
  214. const char *UnmangledName)
  215. {
  216. return wrap(OrcV2CAPIHelper::moveFromSymbolStringPtr(
  217. unwrap(J)->mangleAndIntern(UnmangledName)));
  218. }
  219. LLVMOrcJITDylibRef
  220. LLVMOrcLLLazyJITGetMainJITDylib(LLVMOrcLLLazyJITRef J)
  221. {
  222. return wrap(&unwrap(J)->getMainJITDylib());
  223. }
  224. const char *
  225. LLVMOrcLLLazyJITGetTripleString(LLVMOrcLLLazyJITRef J)
  226. {
  227. return unwrap(J)->getTargetTriple().str().c_str();
  228. }
  229. LLVMOrcExecutionSessionRef
  230. LLVMOrcLLLazyJITGetExecutionSession(LLVMOrcLLLazyJITRef J)
  231. {
  232. return wrap(&unwrap(J)->getExecutionSession());
  233. }
  234. LLVMOrcIRTransformLayerRef
  235. LLVMOrcLLLazyJITGetIRTransformLayer(LLVMOrcLLLazyJITRef J)
  236. {
  237. return wrap(&unwrap(J)->getIRTransformLayer());
  238. }
  239. LLVMOrcObjectTransformLayerRef
  240. LLVMOrcLLLazyJITGetObjTransformLayer(LLVMOrcLLLazyJITRef J)
  241. {
  242. return wrap(&unwrap(J)->getObjTransformLayer());
  243. }