aot_llvm.h 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648
  1. /*
  2. * Copyright (C) 2019 Intel Corporation. All rights reserved.
  3. * SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
  4. */
  5. #ifndef _AOT_LLVM_H_
  6. #define _AOT_LLVM_H_
  7. #include "aot.h"
  8. #include "llvm/Config/llvm-config.h"
  9. #include "llvm-c/Types.h"
  10. #include "llvm-c/Target.h"
  11. #include "llvm-c/Core.h"
  12. #include "llvm-c/Object.h"
  13. #include "llvm-c/OrcEE.h"
  14. #include "llvm-c/ExecutionEngine.h"
  15. #include "llvm-c/Analysis.h"
  16. #include "llvm-c/BitWriter.h"
  17. #if LLVM_VERSION_MAJOR < 17
  18. #include "llvm-c/Transforms/Utils.h"
  19. #include "llvm-c/Transforms/Scalar.h"
  20. #include "llvm-c/Transforms/Vectorize.h"
  21. #include "llvm-c/Transforms/PassManagerBuilder.h"
  22. #include "llvm-c/Initialization.h"
  23. #endif
  24. #include "llvm-c/Orc.h"
  25. #include "llvm-c/Error.h"
  26. #include "llvm-c/Support.h"
  27. #include "llvm-c/TargetMachine.h"
  28. #include "llvm-c/LLJIT.h"
  29. #if WASM_ENABLE_DEBUG_AOT != 0
  30. #include "llvm-c/DebugInfo.h"
  31. #endif
  32. #include "aot_orc_extra.h"
  33. #include "aot_comp_option.h"
  34. #ifdef __cplusplus
  35. extern "C" {
  36. #endif
  37. #if LLVM_VERSION_MAJOR < 14
  38. #define LLVMBuildLoad2(builder, type, value, name) \
  39. LLVMBuildLoad(builder, value, name)
  40. #define LLVMBuildCall2(builder, type, func, args, num_args, name) \
  41. LLVMBuildCall(builder, func, args, num_args, name)
  42. #define LLVMBuildInBoundsGEP2(builder, type, ptr, indices, num_indices, name) \
  43. LLVMBuildInBoundsGEP(builder, ptr, indices, num_indices, name)
  44. #else
  45. /* Opaque pointer type */
  46. #define OPQ_PTR_TYPE INT8_PTR_TYPE
  47. #endif
  48. #ifndef NDEBUG
  49. #undef DEBUG_PASS
  50. #undef DUMP_MODULE
  51. // #define DEBUG_PASS
  52. // #define DUMP_MODULE
  53. #else
  54. #undef DEBUG_PASS
  55. #undef DUMP_MODULE
  56. #endif
  57. struct AOTValueSlot;
  58. /**
  59. * Value in the WASM operation stack, each stack element
  60. * is an LLVM value
  61. */
  62. typedef struct AOTValue {
  63. struct AOTValue *next;
  64. struct AOTValue *prev;
  65. LLVMValueRef value;
  66. uint64 const_value; /* valid if is_const is true */
  67. uint32 local_idx;
  68. /* VALUE_TYPE_I32/I64/F32/F64/VOID */
  69. uint8 type;
  70. bool is_local;
  71. bool is_const;
  72. } AOTValue;
  73. /**
  74. * Value stack, represents stack elements in a WASM block
  75. */
  76. typedef struct AOTValueStack {
  77. AOTValue *value_list_head;
  78. AOTValue *value_list_end;
  79. } AOTValueStack;
  80. /* Record information of a value slot of local variable or stack
  81. during translation */
  82. typedef struct AOTValueSlot {
  83. /* The LLVM value of this slot */
  84. LLVMValueRef value;
  85. /* The value type of this slot */
  86. uint8 type;
  87. /* The dirty bit of the value slot. It's set if the value in
  88. register is newer than the value in memory. */
  89. uint32 dirty : 1;
  90. /* Whether the new value in register is a reference, which is valid
  91. only when the dirty bit is set. */
  92. uint32 ref : 1;
  93. /* Committed reference flag:
  94. 0: uncommitted, 1: not-reference, 2: reference */
  95. uint32 committed_ref : 2;
  96. } AOTValueSlot;
  97. /* Frame information for translation */
  98. typedef struct AOTCompFrame {
  99. /* The current compilation context */
  100. struct AOTCompContext *comp_ctx;
  101. /* The current function context */
  102. struct AOTFuncContext *func_ctx;
  103. /* The current instruction pointer which is being compiled */
  104. const uint8 *frame_ip;
  105. /* Max local slot number */
  106. uint32 max_local_cell_num;
  107. /* Max operand stack slot number */
  108. uint32 max_stack_cell_num;
  109. /* Size of current AOTFrame/WASMInterpFrame */
  110. uint32 cur_frame_size;
  111. /* Stack top pointer */
  112. AOTValueSlot *sp;
  113. /* Local variables + stack operands */
  114. AOTValueSlot lp[1];
  115. } AOTCompFrame;
  116. typedef struct AOTBlock {
  117. struct AOTBlock *next;
  118. struct AOTBlock *prev;
  119. /* Block index */
  120. uint32 block_index;
  121. /* LABEL_TYPE_BLOCK/LOOP/IF/FUNCTION */
  122. uint32 label_type;
  123. /* Whether it is reachable */
  124. bool is_reachable;
  125. /* Whether skip translation of wasm else branch */
  126. bool skip_wasm_code_else;
  127. /* code of else opcode of this block, if it is a IF block */
  128. uint8 *wasm_code_else;
  129. /* code end of this block */
  130. uint8 *wasm_code_end;
  131. /* LLVM label points to code begin */
  132. LLVMBasicBlockRef llvm_entry_block;
  133. /* LLVM label points to code else */
  134. LLVMBasicBlockRef llvm_else_block;
  135. /* LLVM label points to code end */
  136. LLVMBasicBlockRef llvm_end_block;
  137. /* WASM operation stack */
  138. AOTValueStack value_stack;
  139. /* Param count/types/PHIs of this block */
  140. uint32 param_count;
  141. uint8 *param_types;
  142. LLVMValueRef *param_phis;
  143. LLVMValueRef *else_param_phis;
  144. /* Result count/types/PHIs of this block */
  145. uint32 result_count;
  146. uint8 *result_types;
  147. LLVMValueRef *result_phis;
  148. /* The begin frame stack pointer of this block */
  149. AOTValueSlot *frame_sp_begin;
  150. /* The max frame stack pointer that br/br_if/br_table/br_on_xxx
  151. opcodes ever reached when they jumped to the end this block */
  152. AOTValueSlot *frame_sp_max_reached;
  153. } AOTBlock;
  154. /**
  155. * Block stack, represents WASM block stack elements
  156. */
  157. typedef struct AOTBlockStack {
  158. AOTBlock *block_list_head;
  159. AOTBlock *block_list_end;
  160. /* Current block index of each block type */
  161. uint32 block_index[3];
  162. } AOTBlockStack;
  163. typedef struct AOTCheckedAddr {
  164. struct AOTCheckedAddr *next;
  165. uint32 local_idx;
  166. uint64 offset;
  167. uint32 bytes;
  168. } AOTCheckedAddr, *AOTCheckedAddrList;
  169. typedef struct AOTMemInfo {
  170. LLVMValueRef mem_base_addr;
  171. LLVMValueRef mem_data_size_addr;
  172. LLVMValueRef mem_cur_page_count_addr;
  173. LLVMValueRef mem_bound_check_1byte;
  174. LLVMValueRef mem_bound_check_2bytes;
  175. LLVMValueRef mem_bound_check_4bytes;
  176. LLVMValueRef mem_bound_check_8bytes;
  177. LLVMValueRef mem_bound_check_16bytes;
  178. } AOTMemInfo;
  179. typedef struct AOTFuncContext {
  180. AOTFunc *aot_func;
  181. LLVMValueRef func;
  182. LLVMValueRef precheck_func;
  183. LLVMTypeRef func_type;
  184. LLVMModuleRef module;
  185. AOTBlockStack block_stack;
  186. LLVMValueRef exec_env;
  187. LLVMValueRef aot_inst;
  188. LLVMValueRef argv_buf;
  189. LLVMValueRef native_stack_bound;
  190. LLVMValueRef native_stack_top_min_addr;
  191. LLVMValueRef aux_stack_bound;
  192. LLVMValueRef aux_stack_bottom;
  193. LLVMValueRef native_symbol;
  194. LLVMValueRef func_ptrs;
  195. AOTMemInfo *mem_info;
  196. LLVMValueRef cur_exception;
  197. LLVMValueRef cur_frame;
  198. LLVMValueRef cur_frame_ptr;
  199. LLVMValueRef wasm_stack_top_bound;
  200. LLVMValueRef wasm_stack_top_ptr;
  201. bool mem_space_unchanged;
  202. AOTCheckedAddrList checked_addr_list;
  203. LLVMValueRef shared_heap_base_addr_adj;
  204. LLVMValueRef shared_heap_start_off;
  205. LLVMBasicBlockRef got_exception_block;
  206. LLVMBasicBlockRef func_return_block;
  207. LLVMValueRef exception_id_phi;
  208. /* current ip when exception is thrown */
  209. LLVMValueRef exception_ip_phi;
  210. LLVMValueRef func_type_indexes;
  211. #if WASM_ENABLE_DEBUG_AOT != 0
  212. LLVMMetadataRef debug_func;
  213. #endif
  214. unsigned int stack_consumption_for_func_call;
  215. LLVMValueRef locals[1];
  216. } AOTFuncContext;
  217. typedef struct AOTLLVMTypes {
  218. LLVMTypeRef int1_type;
  219. LLVMTypeRef int8_type;
  220. LLVMTypeRef int16_type;
  221. LLVMTypeRef int32_type;
  222. LLVMTypeRef int64_type;
  223. LLVMTypeRef intptr_t_type;
  224. LLVMTypeRef size_t_type;
  225. LLVMTypeRef float32_type;
  226. LLVMTypeRef float64_type;
  227. LLVMTypeRef void_type;
  228. LLVMTypeRef int8_ptr_type;
  229. LLVMTypeRef int8_pptr_type;
  230. LLVMTypeRef int16_ptr_type;
  231. LLVMTypeRef int32_ptr_type;
  232. LLVMTypeRef int64_ptr_type;
  233. LLVMTypeRef intptr_t_ptr_type;
  234. LLVMTypeRef float32_ptr_type;
  235. LLVMTypeRef float64_ptr_type;
  236. LLVMTypeRef v128_type;
  237. LLVMTypeRef v128_ptr_type;
  238. LLVMTypeRef i8x16_vec_type;
  239. LLVMTypeRef i16x8_vec_type;
  240. LLVMTypeRef i32x4_vec_type;
  241. LLVMTypeRef i64x2_vec_type;
  242. LLVMTypeRef f32x4_vec_type;
  243. LLVMTypeRef f64x2_vec_type;
  244. LLVMTypeRef int8_ptr_type_gs;
  245. LLVMTypeRef int16_ptr_type_gs;
  246. LLVMTypeRef int32_ptr_type_gs;
  247. LLVMTypeRef int64_ptr_type_gs;
  248. LLVMTypeRef float32_ptr_type_gs;
  249. LLVMTypeRef float64_ptr_type_gs;
  250. LLVMTypeRef v128_ptr_type_gs;
  251. LLVMTypeRef i1x2_vec_type;
  252. LLVMTypeRef meta_data_type;
  253. LLVMTypeRef funcref_type;
  254. LLVMTypeRef externref_type;
  255. LLVMTypeRef gc_ref_type;
  256. LLVMTypeRef gc_ref_ptr_type;
  257. } AOTLLVMTypes;
  258. typedef struct AOTLLVMConsts {
  259. LLVMValueRef i1_zero;
  260. LLVMValueRef i1_one;
  261. LLVMValueRef i8_zero;
  262. LLVMValueRef i8_one;
  263. LLVMValueRef i32_zero;
  264. LLVMValueRef i64_zero;
  265. LLVMValueRef f32_zero;
  266. LLVMValueRef f64_zero;
  267. LLVMValueRef i32_one;
  268. LLVMValueRef i32_two;
  269. LLVMValueRef i32_three;
  270. LLVMValueRef i32_four;
  271. LLVMValueRef i32_five;
  272. LLVMValueRef i32_six;
  273. LLVMValueRef i32_seven;
  274. LLVMValueRef i32_eight;
  275. LLVMValueRef i32_nine;
  276. LLVMValueRef i32_ten;
  277. LLVMValueRef i32_eleven;
  278. LLVMValueRef i32_twelve;
  279. LLVMValueRef i32_thirteen;
  280. LLVMValueRef i32_fourteen;
  281. LLVMValueRef i32_fifteen;
  282. LLVMValueRef i32_neg_one;
  283. LLVMValueRef i64_neg_one;
  284. LLVMValueRef i32_min;
  285. LLVMValueRef i64_min;
  286. LLVMValueRef i32_31;
  287. LLVMValueRef i32_32;
  288. LLVMValueRef i64_63;
  289. LLVMValueRef i64_64;
  290. LLVMValueRef i8x16_vec_zero;
  291. LLVMValueRef i16x8_vec_zero;
  292. LLVMValueRef i32x4_vec_zero;
  293. LLVMValueRef i64x2_vec_zero;
  294. LLVMValueRef f32x4_vec_zero;
  295. LLVMValueRef f64x2_vec_zero;
  296. LLVMValueRef i8x16_undef;
  297. LLVMValueRef i16x8_undef;
  298. LLVMValueRef i32x4_undef;
  299. LLVMValueRef i64x2_undef;
  300. LLVMValueRef f32x4_undef;
  301. LLVMValueRef f64x2_undef;
  302. LLVMValueRef i32x16_zero;
  303. LLVMValueRef i32x8_zero;
  304. LLVMValueRef i32x4_zero;
  305. LLVMValueRef i32x2_zero;
  306. LLVMValueRef gc_ref_null;
  307. LLVMValueRef i8_ptr_null;
  308. } AOTLLVMConsts;
  309. /**
  310. * Compiler context
  311. */
  312. typedef struct AOTCompContext {
  313. const AOTCompData *comp_data;
  314. /* LLVM variables required to emit LLVM IR */
  315. LLVMContextRef context;
  316. LLVMBuilderRef builder;
  317. #if WASM_ENABLE_DEBUG_AOT
  318. LLVMDIBuilderRef debug_builder;
  319. LLVMMetadataRef debug_file;
  320. LLVMMetadataRef debug_comp_unit;
  321. #endif
  322. LLVMTargetMachineRef target_machine;
  323. char *target_cpu;
  324. char target_arch[16];
  325. unsigned pointer_size;
  326. /* Hardware intrinsic compatibility flags */
  327. uint64 flags[8];
  328. /* required by JIT */
  329. LLVMOrcLLLazyJITRef orc_jit;
  330. LLVMOrcThreadSafeContextRef orc_thread_safe_context;
  331. LLVMModuleRef module;
  332. bool is_jit_mode;
  333. /* AOT indirect mode flag & symbol list */
  334. bool is_indirect_mode;
  335. bh_list native_symbols;
  336. /* Bulk memory feature */
  337. bool enable_bulk_memory;
  338. /* Boundary Check */
  339. bool enable_bound_check;
  340. /* Native stack boundary Check */
  341. bool enable_stack_bound_check;
  342. /* Native stack usage estimation */
  343. bool enable_stack_estimation;
  344. /* 128-bit SIMD */
  345. bool enable_simd;
  346. /* Auxiliary stack overflow/underflow check */
  347. bool enable_aux_stack_check;
  348. /* Generate auxiliary stack frame */
  349. AOTStackFrameType aux_stack_frame_type;
  350. /* Auxiliary call stack features */
  351. AOTCallStackFeatures call_stack_features;
  352. /* Function performance profiling */
  353. bool enable_perf_profiling;
  354. /* Memory usage profiling */
  355. bool enable_memory_profiling;
  356. /* Thread Manager */
  357. bool enable_thread_mgr;
  358. /* Tail Call */
  359. bool enable_tail_call;
  360. /* Reference Types */
  361. bool enable_ref_types;
  362. /* Disable LLVM built-in intrinsics */
  363. bool disable_llvm_intrinsics;
  364. /* Disable LLVM link time optimization */
  365. bool disable_llvm_lto;
  366. /* Enable LLVM PGO (Profile-Guided Optimization) */
  367. bool enable_llvm_pgo;
  368. /* Treat unknown import function as wasm-c-api import function
  369. and allow to directly invoke it from AOT/JIT code */
  370. bool quick_invoke_c_api_import;
  371. /* Use profile file collected by LLVM PGO */
  372. char *use_prof_file;
  373. /* Enable to use segment register as the base addr
  374. of linear memory for load/store operations */
  375. bool enable_segue_i32_load;
  376. bool enable_segue_i64_load;
  377. bool enable_segue_f32_load;
  378. bool enable_segue_f64_load;
  379. bool enable_segue_v128_load;
  380. bool enable_segue_i32_store;
  381. bool enable_segue_i64_store;
  382. bool enable_segue_f32_store;
  383. bool enable_segue_f64_store;
  384. bool enable_segue_v128_store;
  385. /* Whether optimize the JITed code */
  386. bool optimize;
  387. bool emit_frame_pointer;
  388. /* Enable GC */
  389. bool enable_gc;
  390. bool enable_shared_heap;
  391. uint32 opt_level;
  392. uint32 size_level;
  393. /* LLVM floating-point rounding mode metadata */
  394. LLVMValueRef fp_rounding_mode;
  395. /* LLVM floating-point exception behavior metadata */
  396. LLVMValueRef fp_exception_behavior;
  397. /* a global array to store stack sizes */
  398. LLVMTypeRef stack_sizes_type;
  399. LLVMValueRef stack_sizes;
  400. uint32 *jit_stack_sizes; /* for JIT */
  401. /* LLVM data types */
  402. AOTLLVMTypes basic_types;
  403. LLVMTypeRef exec_env_type;
  404. LLVMTypeRef aot_inst_type;
  405. /* LLVM const values */
  406. AOTLLVMConsts llvm_consts;
  407. /* Function contexts */
  408. AOTFuncContext **func_ctxes;
  409. uint32 func_ctx_count;
  410. char **custom_sections_wp;
  411. uint32 custom_sections_count;
  412. /* 3rd-party toolchains */
  413. /* External llc compiler, if specified, wamrc will emit the llvm-ir file and
  414. * invoke the llc compiler to generate object file.
  415. * This can be used when we want to benefit from the optimization of other
  416. * LLVM based toolchains */
  417. const char *external_llc_compiler;
  418. const char *llc_compiler_flags;
  419. /* External asm compiler, if specified, wamrc will emit the text-based
  420. * assembly file (.s) and invoke the llc compiler to generate object file.
  421. * This will be useful when the upstream LLVM doesn't support to emit object
  422. * file for some architecture (such as arc) */
  423. const char *external_asm_compiler;
  424. const char *asm_compiler_flags;
  425. const char *stack_usage_file;
  426. char stack_usage_temp_file[64];
  427. const char *llvm_passes;
  428. const char *builtin_intrinsics;
  429. /* Current frame information for translation */
  430. AOTCompFrame *aot_frame;
  431. } AOTCompContext;
  432. enum {
  433. AOT_FORMAT_FILE,
  434. AOT_OBJECT_FILE,
  435. AOT_LLVMIR_UNOPT_FILE,
  436. AOT_LLVMIR_OPT_FILE,
  437. };
  438. bool
  439. aot_compiler_init(void);
  440. void
  441. aot_compiler_destroy(void);
  442. AOTCompContext *
  443. aot_create_comp_context(const AOTCompData *comp_data, aot_comp_option_t option);
  444. void
  445. aot_destroy_comp_context(AOTCompContext *comp_ctx);
  446. int32
  447. aot_get_native_symbol_index(AOTCompContext *comp_ctx, const char *symbol);
  448. bool
  449. aot_compile_wasm(AOTCompContext *comp_ctx);
  450. uint8 *
  451. aot_emit_elf_file(AOTCompContext *comp_ctx, uint32 *p_elf_file_size);
  452. void
  453. aot_destroy_elf_file(uint8 *elf_file);
  454. void
  455. aot_value_stack_push(const AOTCompContext *comp_ctx, AOTValueStack *stack,
  456. AOTValue *value);
  457. AOTValue *
  458. aot_value_stack_pop(const AOTCompContext *comp_ctx, AOTValueStack *stack);
  459. void
  460. aot_value_stack_destroy(AOTCompContext *comp_ctx, AOTValueStack *stack);
  461. void
  462. aot_block_stack_push(AOTBlockStack *stack, AOTBlock *block);
  463. AOTBlock *
  464. aot_block_stack_pop(AOTBlockStack *stack);
  465. void
  466. aot_block_stack_destroy(AOTCompContext *comp_ctx, AOTBlockStack *stack);
  467. void
  468. aot_block_destroy(AOTCompContext *comp_ctx, AOTBlock *block);
  469. LLVMTypeRef
  470. wasm_type_to_llvm_type(const AOTCompContext *comp_ctx,
  471. const AOTLLVMTypes *llvm_types, uint8 wasm_type);
  472. bool
  473. aot_checked_addr_list_add(AOTFuncContext *func_ctx, uint32 local_idx,
  474. uint64 offset, uint32 bytes);
  475. void
  476. aot_checked_addr_list_del(AOTFuncContext *func_ctx, uint32 local_idx);
  477. bool
  478. aot_checked_addr_list_find(AOTFuncContext *func_ctx, uint32 local_idx,
  479. uint64 offset, uint32 bytes);
  480. void
  481. aot_checked_addr_list_destroy(AOTFuncContext *func_ctx);
  482. bool
  483. aot_build_zero_function_ret(const AOTCompContext *comp_ctx,
  484. AOTFuncContext *func_ctx, AOTFuncType *func_type);
  485. LLVMValueRef
  486. aot_call_llvm_intrinsic(const AOTCompContext *comp_ctx,
  487. const AOTFuncContext *func_ctx, const char *intrinsic,
  488. LLVMTypeRef ret_type, LLVMTypeRef *param_types,
  489. int param_count, ...);
  490. LLVMValueRef
  491. aot_call_llvm_intrinsic_v(const AOTCompContext *comp_ctx,
  492. const AOTFuncContext *func_ctx, const char *intrinsic,
  493. LLVMTypeRef ret_type, LLVMTypeRef *param_types,
  494. int param_count, va_list param_value_list);
  495. LLVMValueRef
  496. aot_get_func_from_table(const AOTCompContext *comp_ctx, LLVMValueRef base,
  497. LLVMTypeRef func_type, int32 index);
  498. LLVMValueRef
  499. aot_load_const_from_table(AOTCompContext *comp_ctx, LLVMValueRef base,
  500. const WASMValue *value, uint8 value_type);
  501. bool
  502. aot_check_simd_compatibility(const char *arch_c_str, const char *cpu_c_str);
  503. void
  504. aot_apply_llvm_new_pass_manager(AOTCompContext *comp_ctx, LLVMModuleRef module);
  505. void
  506. aot_handle_llvm_errmsg(const char *string, LLVMErrorRef err);
  507. char *
  508. aot_compress_aot_func_names(AOTCompContext *comp_ctx, uint32 *p_size);
  509. bool
  510. aot_set_cond_br_weights(AOTCompContext *comp_ctx, LLVMValueRef cond_br,
  511. int32 weights_true, int32 weights_false);
  512. bool
  513. aot_target_precheck_can_use_musttail(const AOTCompContext *comp_ctx);
  514. unsigned int
  515. aot_estimate_stack_usage_for_function_call(const AOTCompContext *comp_ctx,
  516. const AOTFuncType *callee_func_type);
  517. #ifdef __cplusplus
  518. } /* end of extern "C" */
  519. #endif
  520. #endif /* end of _AOT_LLVM_H_ */