mirror of
				https://github.com/bytecodealliance/wasm-micro-runtime.git
				synced 2025-10-26 10:51:17 +00:00 
			
		
		
		
	 7d05dbc988
			
		
	
	
		7d05dbc988
		
			
		
	
	
	
	
		
			
			* implement extended const expr (#4318) * add a toggle to enable extended const on wamrc (#4412)
		
			
				
	
	
		
			4191 lines
		
	
	
		
			142 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			4191 lines
		
	
	
		
			142 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  * Copyright (C) 2019 Intel Corporation. All rights reserved.
 | |
|  * SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
 | |
|  */
 | |
| 
 | |
| #include "aot_llvm.h"
 | |
| #include "aot_llvm_extra2.h"
 | |
| #include "aot_compiler.h"
 | |
| #include "aot_emit_exception.h"
 | |
| #include "aot_emit_table.h"
 | |
| #include "../aot/aot_runtime.h"
 | |
| #include "../aot/aot_intrinsic.h"
 | |
| #include "../interpreter/wasm_runtime.h"
 | |
| 
 | |
| #if WASM_ENABLE_DEBUG_AOT != 0
 | |
| #include "debug/dwarf_extractor.h"
 | |
| #endif
 | |
| 
 | |
| static bool
 | |
| create_native_symbol(const AOTCompContext *comp_ctx, AOTFuncContext *func_ctx);
 | |
| static bool
 | |
| create_native_stack_bound(const AOTCompContext *comp_ctx,
 | |
|                           AOTFuncContext *func_ctx);
 | |
| static bool
 | |
| create_native_stack_top_min(const AOTCompContext *comp_ctx,
 | |
|                             AOTFuncContext *func_ctx);
 | |
| 
 | |
| LLVMTypeRef
 | |
| wasm_type_to_llvm_type(const AOTCompContext *comp_ctx,
 | |
|                        const AOTLLVMTypes *llvm_types, uint8 wasm_type)
 | |
| {
 | |
|     switch (wasm_type) {
 | |
|         case VALUE_TYPE_I32:
 | |
|             return llvm_types->int32_type;
 | |
|         case VALUE_TYPE_FUNCREF:
 | |
|         case VALUE_TYPE_EXTERNREF:
 | |
|             if (comp_ctx->enable_ref_types)
 | |
|                 return llvm_types->int32_type;
 | |
|             else {
 | |
|                 bh_assert(comp_ctx->enable_gc);
 | |
|                 return llvm_types->gc_ref_type;
 | |
|             }
 | |
|         case VALUE_TYPE_I64:
 | |
|             return llvm_types->int64_type;
 | |
|         case VALUE_TYPE_F32:
 | |
|             return llvm_types->float32_type;
 | |
|         case VALUE_TYPE_F64:
 | |
|             return llvm_types->float64_type;
 | |
|         case VALUE_TYPE_V128:
 | |
|             return llvm_types->i64x2_vec_type;
 | |
|         case VALUE_TYPE_VOID:
 | |
|             return llvm_types->void_type;
 | |
|         case REF_TYPE_NULLFUNCREF:
 | |
|         case REF_TYPE_NULLEXTERNREF:
 | |
|         case REF_TYPE_NULLREF:
 | |
|         /* case REF_TYPE_FUNCREF: */
 | |
|         /* case REF_TYPE_EXTERNREF: */
 | |
|         case REF_TYPE_ANYREF:
 | |
|         case REF_TYPE_EQREF:
 | |
|         case REF_TYPE_HT_NULLABLE:
 | |
|         case REF_TYPE_HT_NON_NULLABLE:
 | |
|         case REF_TYPE_I31REF:
 | |
|         case REF_TYPE_STRUCTREF:
 | |
|         case REF_TYPE_ARRAYREF:
 | |
| #if WASM_ENABLE_STRINGREF != 0
 | |
|         case REF_TYPE_STRINGREF:
 | |
|         case REF_TYPE_STRINGVIEWWTF8:
 | |
|         case REF_TYPE_STRINGVIEWWTF16:
 | |
|         case REF_TYPE_STRINGVIEWITER:
 | |
| #endif
 | |
|         case VALUE_TYPE_GC_REF:
 | |
|             bh_assert(comp_ctx->enable_gc);
 | |
|             return llvm_types->gc_ref_type;
 | |
|         default:
 | |
|             break;
 | |
|     }
 | |
|     bh_assert(0);
 | |
|     return NULL;
 | |
| }
 | |
| 
 | |
| static LLVMValueRef
 | |
| aot_add_llvm_func1(const AOTCompContext *comp_ctx, LLVMModuleRef module,
 | |
|                    uint32 func_index, uint32 param_count, LLVMTypeRef func_type,
 | |
|                    const char *prefix)
 | |
| {
 | |
|     char func_name[48] = { 0 };
 | |
|     LLVMValueRef func;
 | |
|     LLVMValueRef local_value;
 | |
|     uint32 i, j;
 | |
| 
 | |
|     /* Add LLVM function */
 | |
|     snprintf(func_name, sizeof(func_name), "%s%d", prefix, func_index);
 | |
|     if (!(func = LLVMAddFunction(module, func_name, func_type))) {
 | |
|         aot_set_last_error("add LLVM function failed.");
 | |
|         return NULL;
 | |
|     }
 | |
| 
 | |
|     j = 0;
 | |
|     local_value = LLVMGetParam(func, j++);
 | |
|     LLVMSetValueName(local_value, "exec_env");
 | |
| 
 | |
|     /* Set parameter names */
 | |
|     for (i = 0; i < param_count; i++) {
 | |
|         local_value = LLVMGetParam(func, j++);
 | |
|         LLVMSetValueName(local_value, "");
 | |
|     }
 | |
| 
 | |
|     return func;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * create a basic func_ctx enough to call aot_emit_exception.
 | |
|  *
 | |
|  * that is:
 | |
|  * - exec_env
 | |
|  * - aot_inst
 | |
|  * - native_symbol (if is_indirect_mode)
 | |
|  */
 | |
| static bool
 | |
| create_basic_func_context(const AOTCompContext *comp_ctx,
 | |
|                           AOTFuncContext *func_ctx)
 | |
| {
 | |
|     LLVMValueRef aot_inst_offset = I32_TWO, aot_inst_addr;
 | |
| 
 | |
|     /* Save the parameters for fast access */
 | |
|     func_ctx->exec_env = LLVMGetParam(func_ctx->func, 0);
 | |
| 
 | |
|     /* Get aot inst address, the layout of exec_env is:
 | |
|        exec_env->next, exec_env->prev, exec_env->module_inst, and argv_buf */
 | |
|     if (!(aot_inst_addr = LLVMBuildInBoundsGEP2(
 | |
|               comp_ctx->builder, OPQ_PTR_TYPE, func_ctx->exec_env,
 | |
|               &aot_inst_offset, 1, "aot_inst_addr"))) {
 | |
|         aot_set_last_error("llvm build in bounds gep failed");
 | |
|         goto fail;
 | |
|     }
 | |
| 
 | |
|     /* Load aot inst */
 | |
|     if (!(func_ctx->aot_inst = LLVMBuildLoad2(comp_ctx->builder, OPQ_PTR_TYPE,
 | |
|                                               aot_inst_addr, "aot_inst"))) {
 | |
|         aot_set_last_error("llvm build load failed");
 | |
|         goto fail;
 | |
|     }
 | |
| 
 | |
|     if (comp_ctx->is_indirect_mode
 | |
|         && !create_native_symbol(comp_ctx, func_ctx)) {
 | |
|         goto fail;
 | |
|     }
 | |
| 
 | |
|     return true;
 | |
| fail:
 | |
|     return false;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * return if the "precheck" wrapper function can use tail call optimization
 | |
|  */
 | |
| bool
 | |
| aot_target_precheck_can_use_musttail(const AOTCompContext *comp_ctx)
 | |
| {
 | |
|     if (!strcmp(comp_ctx->target_arch, "xtensa")) {
 | |
|         /*
 | |
|          * xtensa windowed ABI doesn't have tail call optimization.
 | |
|          *
 | |
|          * Note: as of writing this, the xtensa version of LLVM
 | |
|          * simply ignores the musttail attribute.
 | |
|          * https://github.com/espressif/llvm-project/pull/73
 | |
|          */
 | |
|         return false;
 | |
|     }
 | |
|     if (!strcmp(comp_ctx->target_arch, "riscv32")
 | |
|         || !strcmp(comp_ctx->target_arch, "riscv64")) {
 | |
|         /*
 | |
|          * REVISIT: actually, riscv can use tail call optimization
 | |
|          * in some cases. I (yamamoto) don't know the exact conditions
 | |
|          * though.
 | |
|          */
 | |
|         return false;
 | |
|     }
 | |
|     if (!strcmp(comp_ctx->target_arch, "mips")) {
 | |
|         /*
 | |
|          * cf.
 | |
|          * https://github.com/bytecodealliance/wasm-micro-runtime/issues/2412
 | |
|          */
 | |
|         return false;
 | |
|     }
 | |
|     if (strstr(comp_ctx->target_arch, "thumb")) {
 | |
|         /*
 | |
|          * cf.
 | |
|          * https://github.com/bytecodealliance/wasm-micro-runtime/issues/2412
 | |
|          */
 | |
|         return false;
 | |
|     }
 | |
|     /*
 | |
|      * x86-64/i386: true
 | |
|      *
 | |
|      * others: assume true for now
 | |
|      */
 | |
|     return true;
 | |
| }
 | |
| 
 | |
| unsigned int
 | |
| aot_estimate_stack_usage_for_function_call(const AOTCompContext *comp_ctx,
 | |
|                                            const AOTFuncType *callee_func_type)
 | |
| {
 | |
|     /*
 | |
|      * Estimate how much stack is necessary to make a function call.
 | |
|      * This does not include the stack consumption of the callee function.
 | |
|      *
 | |
|      * For precise estimation, ideally this function needs to be
 | |
|      * target-specific.
 | |
|      * However, this implementation aims to be target-independent,
 | |
|      * allowing a small overstimation, which is probably ok for our purpose.
 | |
|      * (overflow detection and memory profiling)
 | |
|      * On the other hand, an underestimation should be avoided as it
 | |
|      * can cause more serious problems like silent data corruptions.
 | |
|      *
 | |
|      * Assumptions:
 | |
|      *
 | |
|      * - the first result is returned via a register.
 | |
|      *
 | |
|      * - all parameters, including exec_env and pointers to non-first
 | |
|      *   results, are passed via stack.
 | |
|      *   (this is a bit pessimistic than many of real calling conventions,
 | |
|      *   where some of parameters are passed via register.)
 | |
|      *
 | |
|      * - N-byte value needs N-byte alignment on stack.
 | |
|      *
 | |
|      * - a value smaller than a pointer is extended.
 | |
|      *   (eg. 4 byte values are extended to 8 byte on x86-64.)
 | |
|      */
 | |
| 
 | |
|     const unsigned int param_count = callee_func_type->param_count;
 | |
|     const unsigned int result_count = callee_func_type->result_count;
 | |
|     unsigned int size = 0;
 | |
|     unsigned int i;
 | |
|     unsigned int nb;
 | |
| 
 | |
|     if (!strcmp(comp_ctx->target_arch, "xtensa")) {
 | |
|         /*
 | |
|          * In the xtensa windowed ABI, outgoing arguments are already
 | |
|          * included in the callee's stack frame size, which equals to
 | |
|          * the operand of the ENTRY instruction and what LLVM
 | |
|          * MFI->getStackSize returns.
 | |
|          */
 | |
|         return 0;
 | |
|     }
 | |
| 
 | |
|     /* exec_env */
 | |
|     size = comp_ctx->pointer_size;
 | |
| 
 | |
|     /* parameters */
 | |
|     for (i = 0; i < param_count; i++) {
 | |
|         nb = wasm_value_type_cell_num(callee_func_type->types[i]) * 4;
 | |
|         if (nb < comp_ctx->pointer_size) {
 | |
|             nb = comp_ctx->pointer_size;
 | |
|         }
 | |
|         size = align_uint(size, nb) + nb;
 | |
|     }
 | |
| 
 | |
|     /* pointers to results */
 | |
|     nb = comp_ctx->pointer_size;
 | |
|     for (i = 1; i < result_count; i++) {
 | |
|         size = align_uint(size, nb) + nb;
 | |
|     }
 | |
| 
 | |
|     /* return address */
 | |
|     nb = comp_ctx->pointer_size;
 | |
|     size = align_uint(size, nb) + nb;
 | |
| 
 | |
|     /*
 | |
|      * some extra for possible arch-dependent things like
 | |
|      * 16-byte alignment for x86_64.
 | |
|      */
 | |
|     size += 16;
 | |
|     return size;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * a "precheck" function performs a few things before calling wrapped_func.
 | |
|  *
 | |
|  * - update native_stack_top_min if necessary
 | |
|  * - stack overflow check (if it does, trap)
 | |
|  */
 | |
| static bool
 | |
| aot_build_precheck_function(AOTCompContext *comp_ctx, LLVMModuleRef module,
 | |
|                             LLVMValueRef precheck_func, uint32 func_index,
 | |
|                             LLVMTypeRef func_type, LLVMValueRef wrapped_func)
 | |
| {
 | |
|     LLVMBasicBlockRef begin = NULL;
 | |
|     LLVMBasicBlockRef check_top_block = NULL;
 | |
|     LLVMBasicBlockRef update_top_block = NULL;
 | |
|     LLVMBasicBlockRef stack_bound_check_block = NULL;
 | |
|     LLVMBasicBlockRef call_wrapped_func_block = NULL;
 | |
|     LLVMValueRef *params = NULL;
 | |
| 
 | |
|     begin = LLVMAppendBasicBlockInContext(comp_ctx->context, precheck_func,
 | |
|                                           "begin");
 | |
|     check_top_block = LLVMAppendBasicBlockInContext(
 | |
|         comp_ctx->context, precheck_func, "check_top_block");
 | |
|     if (comp_ctx->enable_stack_estimation) {
 | |
|         update_top_block = LLVMAppendBasicBlockInContext(
 | |
|             comp_ctx->context, precheck_func, "update_top_block");
 | |
|         if (!update_top_block) {
 | |
|             goto fail;
 | |
|         }
 | |
|     }
 | |
|     stack_bound_check_block = LLVMAppendBasicBlockInContext(
 | |
|         comp_ctx->context, precheck_func, "stack_bound_check_block");
 | |
|     call_wrapped_func_block = LLVMAppendBasicBlockInContext(
 | |
|         comp_ctx->context, precheck_func, "call_wrapped_func");
 | |
|     if (!begin || !check_top_block || !stack_bound_check_block
 | |
|         || !call_wrapped_func_block) {
 | |
|         goto fail;
 | |
|     }
 | |
|     LLVMBuilderRef b = comp_ctx->builder;
 | |
|     LLVMPositionBuilderAtEnd(b, begin);
 | |
| 
 | |
|     /* create a temporary minimum func_ctx */
 | |
|     AOTFuncContext tmp;
 | |
|     AOTFuncContext *func_ctx = &tmp;
 | |
|     memset(func_ctx, 0, sizeof(*func_ctx));
 | |
|     func_ctx->func = precheck_func;
 | |
|     func_ctx->module = module;
 | |
|     func_ctx->aot_func = comp_ctx->comp_data->funcs[func_index];
 | |
| #if WASM_ENABLE_DEBUG_AOT != 0
 | |
|     func_ctx->debug_func = NULL;
 | |
| #endif
 | |
|     if (!create_basic_func_context(comp_ctx, func_ctx))
 | |
|         goto fail;
 | |
|     if (comp_ctx->enable_stack_bound_check
 | |
|         && !create_native_stack_bound(comp_ctx, func_ctx))
 | |
|         goto fail;
 | |
|     if (comp_ctx->enable_stack_estimation
 | |
|         && !create_native_stack_top_min(comp_ctx, func_ctx)) {
 | |
|         goto fail;
 | |
|     }
 | |
| 
 | |
|     uint32 param_count = LLVMCountParams(precheck_func);
 | |
|     uint32 sz = param_count * (uint32)sizeof(LLVMValueRef);
 | |
|     params = wasm_runtime_malloc(sz);
 | |
|     if (params == NULL) {
 | |
|         goto fail;
 | |
|     }
 | |
|     LLVMGetParams(precheck_func, params);
 | |
| 
 | |
|     const bool is_64bit = comp_ctx->pointer_size == sizeof(uint64);
 | |
|     LLVMTypeRef uintptr_type;
 | |
|     if (is_64bit)
 | |
|         uintptr_type = I64_TYPE;
 | |
|     else
 | |
|         uintptr_type = I32_TYPE;
 | |
| 
 | |
|     /*
 | |
|      * load the stack pointer
 | |
|      */
 | |
|     LLVMValueRef sp_ptr = LLVMBuildAlloca(b, I32_TYPE, "sp_ptr");
 | |
|     if (!sp_ptr) {
 | |
|         goto fail;
 | |
|     }
 | |
|     LLVMValueRef sp = LLVMBuildPtrToInt(b, sp_ptr, uintptr_type, "sp");
 | |
|     if (!sp) {
 | |
|         goto fail;
 | |
|     }
 | |
| 
 | |
|     /*
 | |
|      * load the value for this wrapped function from the stack_sizes array
 | |
|      */
 | |
|     LLVMValueRef stack_sizes;
 | |
|     if (comp_ctx->is_indirect_mode) {
 | |
|         uint32 offset_u32;
 | |
|         LLVMValueRef offset;
 | |
|         LLVMValueRef stack_sizes_p;
 | |
| 
 | |
|         offset_u32 = get_module_inst_extra_offset(comp_ctx);
 | |
|         offset_u32 += offsetof(AOTModuleInstanceExtra, stack_sizes);
 | |
|         offset = I32_CONST(offset_u32);
 | |
|         if (!offset) {
 | |
|             goto fail;
 | |
|         }
 | |
|         stack_sizes_p =
 | |
|             LLVMBuildInBoundsGEP2(b, INT8_TYPE, func_ctx->aot_inst, &offset, 1,
 | |
|                                   "aot_inst_stack_sizes_p");
 | |
|         if (!stack_sizes_p) {
 | |
|             goto fail;
 | |
|         }
 | |
|         stack_sizes =
 | |
|             LLVMBuildLoad2(b, INT32_PTR_TYPE, stack_sizes_p, "stack_sizes");
 | |
|         if (!stack_sizes) {
 | |
|             goto fail;
 | |
|         }
 | |
|     }
 | |
|     else {
 | |
|         stack_sizes = comp_ctx->stack_sizes;
 | |
|     }
 | |
|     LLVMValueRef func_index_const = I32_CONST(func_index);
 | |
|     LLVMValueRef sizes =
 | |
|         LLVMBuildBitCast(b, stack_sizes, INT32_PTR_TYPE, "sizes");
 | |
|     if (!sizes) {
 | |
|         goto fail;
 | |
|     }
 | |
|     LLVMValueRef sizep = LLVMBuildInBoundsGEP2(b, I32_TYPE, sizes,
 | |
|                                                &func_index_const, 1, "sizep");
 | |
|     if (!sizep) {
 | |
|         goto fail;
 | |
|     }
 | |
|     LLVMValueRef size32 = LLVMBuildLoad2(b, I32_TYPE, sizep, "size32");
 | |
|     if (!size32) {
 | |
|         goto fail;
 | |
|     }
 | |
|     LLVMValueRef size;
 | |
|     if (is_64bit) {
 | |
|         size = LLVMBuildZExt(b, size32, uintptr_type, "size");
 | |
|         if (!size) {
 | |
|             goto fail;
 | |
|         }
 | |
|     }
 | |
|     else {
 | |
|         size = size32;
 | |
|     }
 | |
|     /*
 | |
|      * calculate new sp
 | |
|      */
 | |
|     LLVMValueRef underflow =
 | |
|         LLVMBuildICmp(b, LLVMIntULT, sp, size, "underflow");
 | |
|     if (!underflow) {
 | |
|         goto fail;
 | |
|     }
 | |
|     LLVMValueRef new_sp = LLVMBuildSub(b, sp, size, "new_sp");
 | |
|     if (!new_sp) {
 | |
|         goto fail;
 | |
|     }
 | |
|     if (!LLVMBuildBr(b, check_top_block)) {
 | |
|         goto fail;
 | |
|     }
 | |
| 
 | |
|     LLVMPositionBuilderAtEnd(b, check_top_block);
 | |
|     if (comp_ctx->enable_stack_estimation) {
 | |
|         /*
 | |
|          * load native_stack_top_min from the exec_env
 | |
|          */
 | |
|         LLVMValueRef top_min =
 | |
|             LLVMBuildLoad2(b, OPQ_PTR_TYPE, func_ctx->native_stack_top_min_addr,
 | |
|                            "native_stack_top_min");
 | |
|         if (!top_min) {
 | |
|             goto fail;
 | |
|         }
 | |
|         LLVMValueRef top_min_int = LLVMBuildPtrToInt(
 | |
|             b, top_min, uintptr_type, "native_stack_top_min_int");
 | |
|         if (!top_min_int) {
 | |
|             goto fail;
 | |
|         }
 | |
| 
 | |
|         bh_assert(update_top_block);
 | |
| 
 | |
|         /*
 | |
|          * update native_stack_top_min if
 | |
|          * new_sp = sp - size < native_stack_top_min
 | |
|          *
 | |
|          * Note: unless the stack has already overflown in this exec_env,
 | |
|          * native_stack_bound <= native_stack_top_min
 | |
|          */
 | |
|         LLVMValueRef cmp_top =
 | |
|             LLVMBuildICmp(b, LLVMIntULT, new_sp, top_min_int, "cmp_top");
 | |
|         if (!cmp_top) {
 | |
|             goto fail;
 | |
|         }
 | |
|         cmp_top = LLVMBuildOr(b, underflow, cmp_top, "cmp_top2");
 | |
|         if (!cmp_top) {
 | |
|             goto fail;
 | |
|         }
 | |
|         if (!LLVMBuildCondBr(b, cmp_top, update_top_block,
 | |
|                              call_wrapped_func_block)) {
 | |
|             aot_set_last_error("llvm build cond br failed.");
 | |
|             goto fail;
 | |
|         }
 | |
| 
 | |
|         /*
 | |
|          * update native_stack_top_min
 | |
|          */
 | |
|         LLVMPositionBuilderAtEnd(b, update_top_block);
 | |
|         LLVMValueRef new_sp_ptr =
 | |
|             LLVMBuildIntToPtr(b, new_sp, INT8_PTR_TYPE, "new_sp_ptr");
 | |
|         if (!new_sp_ptr) {
 | |
|             goto fail;
 | |
|         }
 | |
|         if (!LLVMBuildStore(b, new_sp_ptr,
 | |
|                             func_ctx->native_stack_top_min_addr)) {
 | |
|             goto fail;
 | |
|         }
 | |
|         if (!LLVMBuildBr(b, stack_bound_check_block)) {
 | |
|             goto fail;
 | |
|         }
 | |
|     }
 | |
|     else {
 | |
|         if (!LLVMBuildBr(b, stack_bound_check_block)) {
 | |
|             goto fail;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     LLVMPositionBuilderAtEnd(b, stack_bound_check_block);
 | |
|     if (comp_ctx->enable_stack_bound_check) {
 | |
|         /*
 | |
|          * trap if new_sp < native_stack_bound
 | |
|          */
 | |
|         LLVMValueRef bound_int = LLVMBuildPtrToInt(
 | |
|             b, func_ctx->native_stack_bound, uintptr_type, "bound_base_int");
 | |
|         if (!bound_int) {
 | |
|             goto fail;
 | |
|         }
 | |
|         LLVMValueRef cmp =
 | |
|             LLVMBuildICmp(b, LLVMIntULT, new_sp, bound_int, "cmp");
 | |
|         if (!cmp) {
 | |
|             goto fail;
 | |
|         }
 | |
|         cmp = LLVMBuildOr(b, underflow, cmp, "cmp2");
 | |
|         if (!cmp) {
 | |
|             goto fail;
 | |
|         }
 | |
|         /* todo: @llvm.expect.i1(i1 %cmp, i1 0) */
 | |
|         if (!aot_emit_exception(comp_ctx, func_ctx, EXCE_NATIVE_STACK_OVERFLOW,
 | |
|                                 true, cmp, call_wrapped_func_block))
 | |
|             goto fail;
 | |
|     }
 | |
|     else {
 | |
|         if (!LLVMBuildBr(b, call_wrapped_func_block)) {
 | |
|             goto fail;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /*
 | |
|      * call the wrapped function
 | |
|      * use a tail-call if possible
 | |
|      */
 | |
|     LLVMPositionBuilderAtEnd(b, call_wrapped_func_block);
 | |
|     const char *name = "tail_call";
 | |
|     LLVMTypeRef ret_type = LLVMGetReturnType(func_type);
 | |
|     if (ret_type == VOID_TYPE) {
 | |
|         name = "";
 | |
|     }
 | |
|     LLVMValueRef retval =
 | |
|         LLVMBuildCall2(b, func_type, wrapped_func, params, param_count, name);
 | |
|     if (!retval) {
 | |
|         goto fail;
 | |
|     }
 | |
|     wasm_runtime_free(params);
 | |
|     params = NULL;
 | |
|     if (aot_target_precheck_can_use_musttail(comp_ctx)) {
 | |
|         LLVMSetTailCallKind(retval, LLVMTailCallKindMustTail);
 | |
|     }
 | |
|     else {
 | |
|         LLVMSetTailCallKind(retval, LLVMTailCallKindTail);
 | |
|     }
 | |
|     if (ret_type == VOID_TYPE) {
 | |
|         if (!LLVMBuildRetVoid(b)) {
 | |
|             goto fail;
 | |
|         }
 | |
|     }
 | |
|     else {
 | |
|         if (!LLVMBuildRet(b, retval)) {
 | |
|             goto fail;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     return true;
 | |
| fail:
 | |
|     if (params != NULL) {
 | |
|         wasm_runtime_free(params);
 | |
|     }
 | |
|     aot_set_last_error("failed to build precheck wrapper function.");
 | |
|     return false;
 | |
| }
 | |
| 
 | |
| static bool
 | |
| check_wasm_type(AOTCompContext *comp_ctx, uint8 type)
 | |
| {
 | |
|     if (type == VALUE_TYPE_FUNCREF || type == VALUE_TYPE_EXTERNREF) {
 | |
|         if (!comp_ctx->enable_ref_types && !comp_ctx->enable_gc) {
 | |
|             aot_set_last_error("funcref or externref type was found, "
 | |
|                                "try removing --disable-ref-types option "
 | |
|                                "or adding --enable-gc option.");
 | |
|             return false;
 | |
|         }
 | |
|         else
 | |
|             return true;
 | |
|     }
 | |
|     else if (aot_is_type_gc_reftype(type)) {
 | |
|         if (!comp_ctx->enable_gc) {
 | |
|             aot_set_last_error("GC reference type was found, "
 | |
|                                "try adding --enable-gc option.");
 | |
|             return false;
 | |
|         }
 | |
|         else
 | |
|             return true;
 | |
|     }
 | |
|     else if (type == VALUE_TYPE_V128) {
 | |
|         if (!comp_ctx->enable_simd) {
 | |
|             aot_set_last_error("SIMD type was found, try removing "
 | |
|                                " --disable-simd option.");
 | |
|             return false;
 | |
|         }
 | |
|         return true;
 | |
|     }
 | |
|     else if (type != VALUE_TYPE_I32 && type != VALUE_TYPE_I64
 | |
|              && type != VALUE_TYPE_F32 && type != VALUE_TYPE_F64) {
 | |
|         bh_assert(0);
 | |
|     }
 | |
| 
 | |
|     return true;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Add LLVM function
 | |
|  */
 | |
| static LLVMValueRef
 | |
| aot_add_llvm_func(AOTCompContext *comp_ctx, LLVMModuleRef module,
 | |
|                   const AOTFuncType *aot_func_type, uint32 func_index,
 | |
|                   LLVMTypeRef *p_func_type, LLVMValueRef *p_precheck_func)
 | |
| {
 | |
|     WASMFunction *aot_func =
 | |
|         comp_ctx->comp_data->wasm_module->functions[func_index];
 | |
|     LLVMValueRef func = NULL;
 | |
|     LLVMTypeRef *param_types, ret_type, func_type;
 | |
|     LLVMTypeRef func_type_wrapper;
 | |
|     LLVMValueRef func_wrapper;
 | |
|     LLVMBasicBlockRef func_begin;
 | |
|     char func_name[48];
 | |
|     uint64 size;
 | |
|     uint32 i, j = 0, param_count = (uint64)aot_func_type->param_count;
 | |
|     uint32 backend_thread_num, compile_thread_num;
 | |
| 
 | |
|     /* Check function parameter types and result types */
 | |
|     for (i = 0;
 | |
|          i < (uint32)(aot_func_type->param_count + aot_func_type->result_count);
 | |
|          i++) {
 | |
|         if (!check_wasm_type(comp_ctx, aot_func_type->types[i]))
 | |
|             return NULL;
 | |
|     }
 | |
|     /* Check function local types */
 | |
|     for (i = 0; i < aot_func->local_count; i++) {
 | |
|         if (!check_wasm_type(comp_ctx, aot_func->local_types[i]))
 | |
|             return NULL;
 | |
|     }
 | |
| 
 | |
|     /* exec env as first parameter */
 | |
|     param_count++;
 | |
| 
 | |
|     /* Extra wasm function results(except the first one)'s address are
 | |
|      * appended to aot function parameters. */
 | |
|     if (aot_func_type->result_count > 1)
 | |
|         param_count += aot_func_type->result_count - 1;
 | |
| 
 | |
|     /* Initialize parameter types of the LLVM function */
 | |
|     size = sizeof(LLVMTypeRef) * ((uint64)param_count);
 | |
|     if (size >= UINT32_MAX
 | |
|         || !(param_types = wasm_runtime_malloc((uint32)size))) {
 | |
|         aot_set_last_error("allocate memory failed.");
 | |
|         return NULL;
 | |
|     }
 | |
| 
 | |
|     /* exec env as first parameter */
 | |
|     param_types[j++] = comp_ctx->exec_env_type;
 | |
|     for (i = 0; i < aot_func_type->param_count; i++)
 | |
|         param_types[j++] = TO_LLVM_TYPE(aot_func_type->types[i]);
 | |
|     /* Extra results' address */
 | |
|     for (i = 1; i < aot_func_type->result_count; i++, j++) {
 | |
|         param_types[j] =
 | |
|             TO_LLVM_TYPE(aot_func_type->types[aot_func_type->param_count + i]);
 | |
|         if (!(param_types[j] = LLVMPointerType(param_types[j], 0))) {
 | |
|             aot_set_last_error("llvm get pointer type failed.");
 | |
|             goto fail;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /* Resolve return type of the LLVM function */
 | |
|     if (aot_func_type->result_count)
 | |
|         ret_type =
 | |
|             TO_LLVM_TYPE(aot_func_type->types[aot_func_type->param_count]);
 | |
|     else
 | |
|         ret_type = VOID_TYPE;
 | |
| 
 | |
|     /* Resolve function prototype */
 | |
|     if (!(func_type =
 | |
|               LLVMFunctionType(ret_type, param_types, param_count, false))) {
 | |
|         aot_set_last_error("create LLVM function type failed.");
 | |
|         goto fail;
 | |
|     }
 | |
| 
 | |
|     bh_assert(func_index < comp_ctx->func_ctx_count);
 | |
|     bh_assert(LLVMGetReturnType(func_type) == ret_type);
 | |
| 
 | |
|     const char *prefix = AOT_FUNC_PREFIX;
 | |
|     const bool need_precheck =
 | |
|         comp_ctx->enable_stack_bound_check || comp_ctx->enable_stack_estimation;
 | |
|     LLVMValueRef precheck_func = NULL;
 | |
| 
 | |
|     if (need_precheck) {
 | |
|         precheck_func = aot_add_llvm_func1(comp_ctx, module, func_index,
 | |
|                                            aot_func_type->param_count,
 | |
|                                            func_type, AOT_FUNC_PREFIX);
 | |
|         if (!precheck_func) {
 | |
|             goto fail;
 | |
|         }
 | |
|         /*
 | |
|          * REVISIT: probably this breaks windows hw bound check
 | |
|          * (the RtlAddFunctionTable stuff)
 | |
|          */
 | |
|         prefix = AOT_FUNC_INTERNAL_PREFIX;
 | |
|     }
 | |
|     if (!(func = aot_add_llvm_func1(comp_ctx, module, func_index,
 | |
|                                     aot_func_type->param_count, func_type,
 | |
|                                     prefix)))
 | |
|         goto fail;
 | |
| 
 | |
|     if (comp_ctx->disable_llvm_jump_tables) {
 | |
|         LLVMAttributeRef attr_no_jump_tables = LLVMCreateStringAttribute(
 | |
|             comp_ctx->context, "no-jump-tables",
 | |
|             (uint32)strlen("no-jump-tables"), "true", (uint32)strlen("true"));
 | |
|         LLVMAddAttributeAtIndex(func, LLVMAttributeFunctionIndex,
 | |
|                                 attr_no_jump_tables);
 | |
|     }
 | |
| 
 | |
|     /* spread fp.all to every function */
 | |
|     if (comp_ctx->emit_frame_pointer) {
 | |
|         const char *key = "frame-pointer";
 | |
|         const char *val = "all";
 | |
|         LLVMAttributeRef no_omit_fp = LLVMCreateStringAttribute(
 | |
|             comp_ctx->context, key, (unsigned)strlen(key), val,
 | |
|             (unsigned)strlen(val));
 | |
|         if (!no_omit_fp) {
 | |
|             aot_set_last_error("create LLVM attribute (frame-pointer) failed.");
 | |
|             goto fail;
 | |
|         }
 | |
|         LLVMAddAttributeAtIndex(func, LLVMAttributeFunctionIndex, no_omit_fp);
 | |
|     }
 | |
| 
 | |
|     if (need_precheck) {
 | |
|         if (!comp_ctx->is_jit_mode)
 | |
|             LLVMSetLinkage(func, LLVMInternalLinkage);
 | |
|         unsigned int kind =
 | |
|             LLVMGetEnumAttributeKindForName("noinline", strlen("noinline"));
 | |
|         LLVMAttributeRef attr_noinline =
 | |
|             LLVMCreateEnumAttribute(comp_ctx->context, kind, 0);
 | |
|         LLVMAddAttributeAtIndex(func, LLVMAttributeFunctionIndex,
 | |
|                                 attr_noinline);
 | |
|         if (!strcmp(comp_ctx->target_arch, "xtensa")) {
 | |
|             /* Because "func" is only called by "precheck_func", short-call
 | |
|              * should be ok. We prefer short-call because it's smaller
 | |
|              * and more importantly doesn't involve relocations.
 | |
|              */
 | |
|             LLVMAttributeRef attr_short_call = LLVMCreateStringAttribute(
 | |
|                 comp_ctx->context, "short-call", (unsigned)strlen("short-call"),
 | |
|                 "", 0);
 | |
|             LLVMAddAttributeAtIndex(func, LLVMAttributeFunctionIndex,
 | |
|                                     attr_short_call);
 | |
|         }
 | |
|         if (!aot_build_precheck_function(comp_ctx, module, precheck_func,
 | |
|                                          func_index, func_type, func))
 | |
|             goto fail;
 | |
|         LLVMAddAttributeAtIndex(precheck_func, LLVMAttributeFunctionIndex,
 | |
|                                 attr_noinline);
 | |
|         *p_precheck_func = precheck_func;
 | |
|     }
 | |
|     else {
 | |
|         *p_precheck_func = func;
 | |
|     }
 | |
| 
 | |
|     if (p_func_type)
 | |
|         *p_func_type = func_type;
 | |
| 
 | |
|     backend_thread_num = WASM_ORC_JIT_BACKEND_THREAD_NUM;
 | |
|     compile_thread_num = WASM_ORC_JIT_COMPILE_THREAD_NUM;
 | |
| 
 | |
|     /* Add the jit wrapper function with simple prototype, so that we
 | |
|        can easily call it to trigger its compilation and let LLVM JIT
 | |
|        compile the actual jit functions by adding them into the function
 | |
|        list in the PartitionFunction callback */
 | |
|     if (comp_ctx->is_jit_mode
 | |
|         && (func_index % (backend_thread_num * compile_thread_num)
 | |
|             < backend_thread_num)) {
 | |
|         func_type_wrapper = LLVMFunctionType(VOID_TYPE, NULL, 0, false);
 | |
|         if (!func_type_wrapper) {
 | |
|             aot_set_last_error("create LLVM function type failed.");
 | |
|             goto fail;
 | |
|         }
 | |
| 
 | |
|         snprintf(func_name, sizeof(func_name), "%s%d%s", AOT_FUNC_PREFIX,
 | |
|                  func_index, "_wrapper");
 | |
|         if (!(func_wrapper =
 | |
|                   LLVMAddFunction(module, func_name, func_type_wrapper))) {
 | |
|             aot_set_last_error("add LLVM function failed.");
 | |
|             goto fail;
 | |
|         }
 | |
| 
 | |
|         if (!(func_begin = LLVMAppendBasicBlockInContext(
 | |
|                   comp_ctx->context, func_wrapper, "func_begin"))) {
 | |
|             aot_set_last_error("add LLVM basic block failed.");
 | |
|             goto fail;
 | |
|         }
 | |
| 
 | |
|         LLVMPositionBuilderAtEnd(comp_ctx->builder, func_begin);
 | |
|         if (!LLVMBuildRetVoid(comp_ctx->builder)) {
 | |
|             aot_set_last_error("llvm build ret failed.");
 | |
|             goto fail;
 | |
|         }
 | |
|     }
 | |
| 
 | |
| fail:
 | |
|     wasm_runtime_free(param_types);
 | |
|     return func;
 | |
| }
 | |
| 
 | |
| static void
 | |
| free_block_memory(AOTBlock *block)
 | |
| {
 | |
|     if (block->param_types)
 | |
|         wasm_runtime_free(block->param_types);
 | |
|     if (block->result_types)
 | |
|         wasm_runtime_free(block->result_types);
 | |
|     wasm_runtime_free(block);
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Create first AOTBlock, or function block for the function
 | |
|  */
 | |
| static AOTBlock *
 | |
| aot_create_func_block(const AOTCompContext *comp_ctx,
 | |
|                       const AOTFuncContext *func_ctx, const AOTFunc *func,
 | |
|                       const AOTFuncType *aot_func_type)
 | |
| {
 | |
|     AOTBlock *aot_block;
 | |
|     uint32 param_count = aot_func_type->param_count,
 | |
|            result_count = aot_func_type->result_count;
 | |
| 
 | |
|     /* Allocate memory */
 | |
|     if (!(aot_block = wasm_runtime_malloc(sizeof(AOTBlock)))) {
 | |
|         aot_set_last_error("allocate memory failed.");
 | |
|         return NULL;
 | |
|     }
 | |
|     memset(aot_block, 0, sizeof(AOTBlock));
 | |
|     if (param_count
 | |
|         && !(aot_block->param_types = wasm_runtime_malloc(param_count))) {
 | |
|         aot_set_last_error("allocate memory failed.");
 | |
|         goto fail;
 | |
|     }
 | |
|     if (result_count) {
 | |
|         if (!(aot_block->result_types = wasm_runtime_malloc(result_count))) {
 | |
|             aot_set_last_error("allocate memory failed.");
 | |
|             goto fail;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /* Set block data */
 | |
|     aot_block->label_type = LABEL_TYPE_FUNCTION;
 | |
|     aot_block->param_count = param_count;
 | |
|     if (param_count) {
 | |
|         bh_memcpy_s(aot_block->param_types, param_count, aot_func_type->types,
 | |
|                     param_count);
 | |
|     }
 | |
|     aot_block->result_count = result_count;
 | |
|     if (result_count) {
 | |
|         bh_memcpy_s(aot_block->result_types, result_count,
 | |
|                     aot_func_type->types + param_count, result_count);
 | |
|     }
 | |
|     aot_block->wasm_code_end = func->code + func->code_size;
 | |
| 
 | |
|     /* Add function entry block */
 | |
|     if (!(aot_block->llvm_entry_block = LLVMAppendBasicBlockInContext(
 | |
|               comp_ctx->context, func_ctx->func, "func_begin"))) {
 | |
|         aot_set_last_error("add LLVM basic block failed.");
 | |
|         goto fail;
 | |
|     }
 | |
| 
 | |
|     return aot_block;
 | |
| 
 | |
| fail:
 | |
|     free_block_memory(aot_block);
 | |
|     return NULL;
 | |
| }
 | |
| 
 | |
| static bool
 | |
| create_argv_buf(AOTCompContext *comp_ctx, AOTFuncContext *func_ctx)
 | |
| {
 | |
|     LLVMValueRef argv_buf_offset = I32_THREE, argv_buf_addr;
 | |
|     LLVMTypeRef int32_ptr_type;
 | |
| 
 | |
|     /* Get argv buffer address */
 | |
|     if (!(argv_buf_addr = LLVMBuildInBoundsGEP2(
 | |
|               comp_ctx->builder, OPQ_PTR_TYPE, func_ctx->exec_env,
 | |
|               &argv_buf_offset, 1, "argv_buf_addr"))) {
 | |
|         aot_set_last_error("llvm build in bounds gep failed");
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     if (!(int32_ptr_type = LLVMPointerType(INT32_PTR_TYPE, 0))) {
 | |
|         aot_set_last_error("llvm add pointer type failed");
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     /* Convert to int32 pointer type */
 | |
|     if (!(argv_buf_addr = LLVMBuildBitCast(comp_ctx->builder, argv_buf_addr,
 | |
|                                            int32_ptr_type, "argv_buf_ptr"))) {
 | |
|         aot_set_last_error("llvm build load failed");
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     if (!(func_ctx->argv_buf = LLVMBuildLoad2(comp_ctx->builder, INT32_PTR_TYPE,
 | |
|                                               argv_buf_addr, "argv_buf"))) {
 | |
|         aot_set_last_error("llvm build load failed");
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     return true;
 | |
| }
 | |
| 
 | |
| static bool
 | |
| create_native_stack_bound(const AOTCompContext *comp_ctx,
 | |
|                           AOTFuncContext *func_ctx)
 | |
| {
 | |
|     LLVMValueRef stack_bound_offset = I32_FOUR, stack_bound_addr;
 | |
| 
 | |
|     if (!(stack_bound_addr = LLVMBuildInBoundsGEP2(
 | |
|               comp_ctx->builder, OPQ_PTR_TYPE, func_ctx->exec_env,
 | |
|               &stack_bound_offset, 1, "stack_bound_addr"))) {
 | |
|         aot_set_last_error("llvm build in bounds gep failed");
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     if (!(func_ctx->native_stack_bound =
 | |
|               LLVMBuildLoad2(comp_ctx->builder, OPQ_PTR_TYPE, stack_bound_addr,
 | |
|                              "native_stack_bound"))) {
 | |
|         aot_set_last_error("llvm build load failed");
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     return true;
 | |
| }
 | |
| 
 | |
| static bool
 | |
| create_native_stack_top_min(const AOTCompContext *comp_ctx,
 | |
|                             AOTFuncContext *func_ctx)
 | |
| {
 | |
|     LLVMValueRef offset = I32_NINE;
 | |
| 
 | |
|     if (!(func_ctx->native_stack_top_min_addr = LLVMBuildInBoundsGEP2(
 | |
|               comp_ctx->builder, OPQ_PTR_TYPE, func_ctx->exec_env, &offset, 1,
 | |
|               "native_stack_top_min_addr"))) {
 | |
|         aot_set_last_error("llvm build in bounds gep failed");
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     return true;
 | |
| }
 | |
| 
 | |
| static bool
 | |
| create_aux_stack_info(const AOTCompContext *comp_ctx, AOTFuncContext *func_ctx)
 | |
| {
 | |
|     LLVMValueRef aux_stack_bound_offset = I32_SIX, aux_stack_bound_addr;
 | |
|     LLVMValueRef aux_stack_bottom_offset = I32_SEVEN, aux_stack_bottom_addr;
 | |
| 
 | |
|     /* Get aux stack boundary address */
 | |
|     if (!(aux_stack_bound_addr = LLVMBuildInBoundsGEP2(
 | |
|               comp_ctx->builder, OPQ_PTR_TYPE, func_ctx->exec_env,
 | |
|               &aux_stack_bound_offset, 1, "aux_stack_bound_addr"))) {
 | |
|         aot_set_last_error("llvm build in bounds gep failed");
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     if (!(aux_stack_bound_addr =
 | |
|               LLVMBuildBitCast(comp_ctx->builder, aux_stack_bound_addr,
 | |
|                                INTPTR_T_PTR_TYPE, "aux_stack_bound_ptr"))) {
 | |
|         aot_set_last_error("llvm build bit cast failed");
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     if (!(func_ctx->aux_stack_bound =
 | |
|               LLVMBuildLoad2(comp_ctx->builder, INTPTR_T_TYPE,
 | |
|                              aux_stack_bound_addr, "aux_stack_bound_intptr"))) {
 | |
|         aot_set_last_error("llvm build load failed");
 | |
|         return false;
 | |
|     }
 | |
|     if (!(func_ctx->aux_stack_bound =
 | |
|               LLVMBuildZExt(comp_ctx->builder, func_ctx->aux_stack_bound,
 | |
|                             I64_TYPE, "aux_stack_bound_i64"))) {
 | |
|         aot_set_last_error("llvm build truncOrBitCast failed.");
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     /* Get aux stack bottom address */
 | |
|     if (!(aux_stack_bottom_addr = LLVMBuildInBoundsGEP2(
 | |
|               comp_ctx->builder, OPQ_PTR_TYPE, func_ctx->exec_env,
 | |
|               &aux_stack_bottom_offset, 1, "aux_stack_bottom_addr"))) {
 | |
|         aot_set_last_error("llvm build in bounds gep failed");
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     if (!(aux_stack_bottom_addr =
 | |
|               LLVMBuildBitCast(comp_ctx->builder, aux_stack_bottom_addr,
 | |
|                                INTPTR_T_PTR_TYPE, "aux_stack_bottom_ptr"))) {
 | |
|         aot_set_last_error("llvm build bit cast failed");
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     if (!(func_ctx->aux_stack_bottom =
 | |
|               LLVMBuildLoad2(comp_ctx->builder, INTPTR_T_TYPE,
 | |
|                              aux_stack_bottom_addr, "aux_stack_bottom"))) {
 | |
|         aot_set_last_error("llvm build load failed");
 | |
|         return false;
 | |
|     }
 | |
|     if (!(func_ctx->aux_stack_bottom =
 | |
|               LLVMBuildZExt(comp_ctx->builder, func_ctx->aux_stack_bottom,
 | |
|                             I64_TYPE, "aux_stack_bottom_i64"))) {
 | |
|         aot_set_last_error("llvm build truncOrBitCast failed.");
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     return true;
 | |
| }
 | |
| 
 | |
| static bool
 | |
| create_aux_stack_frame(const AOTCompContext *comp_ctx, AOTFuncContext *func_ctx)
 | |
| {
 | |
|     LLVMValueRef wasm_stack_top_bound_ptr, offset;
 | |
| 
 | |
|     offset = I32_ONE;
 | |
|     if (!(func_ctx->cur_frame_ptr = LLVMBuildInBoundsGEP2(
 | |
|               comp_ctx->builder, OPQ_PTR_TYPE, func_ctx->exec_env, &offset, 1,
 | |
|               "cur_frame_ptr"))) {
 | |
|         aot_set_last_error("llvm build in bounds gep failed");
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     if (!(func_ctx->cur_frame =
 | |
|               LLVMBuildLoad2(comp_ctx->builder, OPQ_PTR_TYPE,
 | |
|                              func_ctx->cur_frame_ptr, "cur_frame"))) {
 | |
|         aot_set_last_error("llvm build load failed");
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     /* Get exec_env->wasm_stack.top_boundary and its address */
 | |
|     offset = I32_TEN;
 | |
|     if (!(wasm_stack_top_bound_ptr = LLVMBuildInBoundsGEP2(
 | |
|               comp_ctx->builder, OPQ_PTR_TYPE, func_ctx->exec_env, &offset, 1,
 | |
|               "wasm_stack_top_bound_ptr"))
 | |
|         || !(func_ctx->wasm_stack_top_bound = LLVMBuildLoad2(
 | |
|                  comp_ctx->builder, INT8_PTR_TYPE, wasm_stack_top_bound_ptr,
 | |
|                  "wasm_stack_top_bound"))) {
 | |
|         aot_set_last_error("load wasm_stack.top_boundary failed");
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     offset = I32_ELEVEN;
 | |
|     if (!(func_ctx->wasm_stack_top_ptr = LLVMBuildInBoundsGEP2(
 | |
|               comp_ctx->builder, OPQ_PTR_TYPE, func_ctx->exec_env, &offset, 1,
 | |
|               "wasm_stack_top_ptr"))) {
 | |
|         aot_set_last_error("llvm build inbounds gep failed");
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     return true;
 | |
| }
 | |
| 
 | |
| static bool
 | |
| create_native_symbol(const AOTCompContext *comp_ctx, AOTFuncContext *func_ctx)
 | |
| {
 | |
|     LLVMValueRef native_symbol_offset = I32_EIGHT, native_symbol_addr;
 | |
| 
 | |
|     if (!(native_symbol_addr = LLVMBuildInBoundsGEP2(
 | |
|               comp_ctx->builder, OPQ_PTR_TYPE, func_ctx->exec_env,
 | |
|               &native_symbol_offset, 1, "native_symbol_addr"))) {
 | |
|         aot_set_last_error("llvm build in bounds gep failed");
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     if (!(func_ctx->native_symbol =
 | |
|               LLVMBuildLoad2(comp_ctx->builder, OPQ_PTR_TYPE,
 | |
|                              native_symbol_addr, "native_symbol_tmp"))) {
 | |
|         aot_set_last_error("llvm build bit cast failed");
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     if (!(func_ctx->native_symbol =
 | |
|               LLVMBuildBitCast(comp_ctx->builder, func_ctx->native_symbol,
 | |
|                                comp_ctx->exec_env_type, "native_symbol"))) {
 | |
|         aot_set_last_error("llvm build bit cast failed");
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     return true;
 | |
| }
 | |
| 
 | |
| static bool
 | |
| create_local_variables(const AOTCompData *comp_data,
 | |
|                        const AOTCompContext *comp_ctx, AOTFuncContext *func_ctx,
 | |
|                        const AOTFunc *func)
 | |
| {
 | |
|     AOTFuncType *aot_func_type =
 | |
|         (AOTFuncType *)comp_data->types[func->func_type_index];
 | |
|     char local_name[32];
 | |
|     uint32 i, j = 1;
 | |
| 
 | |
|     for (i = 0; i < aot_func_type->param_count; i++, j++) {
 | |
|         snprintf(local_name, sizeof(local_name), "l%d", i);
 | |
|         func_ctx->locals[i] =
 | |
|             LLVMBuildAlloca(comp_ctx->builder,
 | |
|                             TO_LLVM_TYPE(aot_func_type->types[i]), local_name);
 | |
|         if (!func_ctx->locals[i]) {
 | |
|             aot_set_last_error("llvm build alloca failed.");
 | |
|             return false;
 | |
|         }
 | |
|         if (!LLVMBuildStore(comp_ctx->builder, LLVMGetParam(func_ctx->func, j),
 | |
|                             func_ctx->locals[i])) {
 | |
|             aot_set_last_error("llvm build store failed.");
 | |
|             return false;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     for (i = 0; i < func->local_count; i++) {
 | |
|         LLVMTypeRef local_type;
 | |
|         LLVMValueRef local_value = NULL;
 | |
|         snprintf(local_name, sizeof(local_name), "l%d",
 | |
|                  aot_func_type->param_count + i);
 | |
|         local_type = TO_LLVM_TYPE(func->local_types_wp[i]);
 | |
|         func_ctx->locals[aot_func_type->param_count + i] =
 | |
|             LLVMBuildAlloca(comp_ctx->builder, local_type, local_name);
 | |
|         if (!func_ctx->locals[aot_func_type->param_count + i]) {
 | |
|             aot_set_last_error("llvm build alloca failed.");
 | |
|             return false;
 | |
|         }
 | |
|         switch (func->local_types_wp[i]) {
 | |
|             case VALUE_TYPE_I32:
 | |
|                 local_value = I32_ZERO;
 | |
|                 break;
 | |
|             case VALUE_TYPE_I64:
 | |
|                 local_value = I64_ZERO;
 | |
|                 break;
 | |
|             case VALUE_TYPE_F32:
 | |
|                 local_value = F32_ZERO;
 | |
|                 break;
 | |
|             case VALUE_TYPE_F64:
 | |
|                 local_value = F64_ZERO;
 | |
|                 break;
 | |
|             case VALUE_TYPE_V128:
 | |
|                 local_value = V128_i64x2_ZERO;
 | |
|                 break;
 | |
|             case VALUE_TYPE_FUNCREF:
 | |
|             case VALUE_TYPE_EXTERNREF:
 | |
|                 if (!comp_ctx->enable_gc)
 | |
|                     local_value = REF_NULL;
 | |
|                 else
 | |
|                     local_value = GC_REF_NULL;
 | |
|                 break;
 | |
| #if WASM_ENABLE_GC != 0
 | |
|             case REF_TYPE_NULLFUNCREF:
 | |
|             case REF_TYPE_NULLEXTERNREF:
 | |
|             case REF_TYPE_NULLREF:
 | |
|             /* case REF_TYPE_FUNCREF: */
 | |
|             /* case REF_TYPE_EXTERNREF: */
 | |
|             case REF_TYPE_ANYREF:
 | |
|             case REF_TYPE_EQREF:
 | |
|             case REF_TYPE_HT_NULLABLE:
 | |
|             case REF_TYPE_HT_NON_NULLABLE:
 | |
|             case REF_TYPE_I31REF:
 | |
|             case REF_TYPE_STRUCTREF:
 | |
|             case REF_TYPE_ARRAYREF:
 | |
| #if WASM_ENABLE_STRINGREF != 0
 | |
|             case REF_TYPE_STRINGREF:
 | |
|             case REF_TYPE_STRINGVIEWWTF8:
 | |
|             case REF_TYPE_STRINGVIEWWTF16:
 | |
|             case REF_TYPE_STRINGVIEWITER:
 | |
| #endif
 | |
|                 local_value = GC_REF_NULL;
 | |
|                 break;
 | |
| #endif
 | |
|             default:
 | |
|                 bh_assert(0);
 | |
|                 break;
 | |
|         }
 | |
|         if (!LLVMBuildStore(comp_ctx->builder, local_value,
 | |
|                             func_ctx->locals[aot_func_type->param_count + i])) {
 | |
|             aot_set_last_error("llvm build store failed.");
 | |
|             return false;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     return true;
 | |
| }
 | |
| 
 | |
| static bool
 | |
| create_memory_info(const AOTCompContext *comp_ctx, AOTFuncContext *func_ctx,
 | |
|                    LLVMTypeRef int8_ptr_type, uint32 func_index)
 | |
| {
 | |
|     LLVMValueRef offset, mem_info_base;
 | |
|     uint32 memory_count;
 | |
|     WASMModule *module = comp_ctx->comp_data->wasm_module;
 | |
|     WASMFunction *func = module->functions[func_index];
 | |
|     LLVMTypeRef bound_check_type;
 | |
|     bool mem_space_unchanged =
 | |
|         (!func->has_op_memory_grow && !func->has_op_func_call)
 | |
|         || (!module->possible_memory_grow);
 | |
| #if WASM_ENABLE_SHARED_MEMORY != 0
 | |
|     bool is_shared_memory;
 | |
| #endif
 | |
| 
 | |
|     func_ctx->mem_space_unchanged = mem_space_unchanged;
 | |
| 
 | |
|     memory_count = module->memory_count + module->import_memory_count;
 | |
|     /* If the module doesn't have memory, reserve
 | |
|         one mem_info space with empty content */
 | |
|     if (memory_count == 0)
 | |
|         memory_count = 1;
 | |
| 
 | |
|     if (!(func_ctx->mem_info =
 | |
|               wasm_runtime_malloc(sizeof(AOTMemInfo) * memory_count))) {
 | |
|         return false;
 | |
|     }
 | |
|     memset(func_ctx->mem_info, 0, sizeof(AOTMemInfo));
 | |
| 
 | |
|     /* Currently we only create memory info for memory 0 */
 | |
|     /* Load memory base address */
 | |
| #if WASM_ENABLE_SHARED_MEMORY != 0
 | |
|     is_shared_memory =
 | |
|         comp_ctx->comp_data->memories[0].flags & 0x02 ? true : false;
 | |
|     if (is_shared_memory) {
 | |
|         LLVMValueRef shared_mem_addr;
 | |
|         offset = I32_CONST(offsetof(AOTModuleInstance, memories));
 | |
|         if (!offset) {
 | |
|             aot_set_last_error("create llvm const failed.");
 | |
|             return false;
 | |
|         }
 | |
| 
 | |
|         /* aot_inst->memories */
 | |
|         if (!(shared_mem_addr = LLVMBuildInBoundsGEP2(
 | |
|                   comp_ctx->builder, INT8_TYPE, func_ctx->aot_inst, &offset, 1,
 | |
|                   "shared_mem_addr_offset"))) {
 | |
|             aot_set_last_error("llvm build in bounds gep failed");
 | |
|             return false;
 | |
|         }
 | |
|         if (!(shared_mem_addr =
 | |
|                   LLVMBuildBitCast(comp_ctx->builder, shared_mem_addr,
 | |
|                                    int8_ptr_type, "shared_mem_addr_ptr"))) {
 | |
|             aot_set_last_error("llvm build bit cast failed");
 | |
|             return false;
 | |
|         }
 | |
|         /* aot_inst->memories[0] */
 | |
|         if (!(shared_mem_addr =
 | |
|                   LLVMBuildLoad2(comp_ctx->builder, OPQ_PTR_TYPE,
 | |
|                                  shared_mem_addr, "shared_mem_addr"))) {
 | |
|             aot_set_last_error("llvm build load failed");
 | |
|             return false;
 | |
|         }
 | |
|         if (!(shared_mem_addr =
 | |
|                   LLVMBuildBitCast(comp_ctx->builder, shared_mem_addr,
 | |
|                                    int8_ptr_type, "shared_mem_addr_ptr"))) {
 | |
|             aot_set_last_error("llvm build bit cast failed");
 | |
|             return false;
 | |
|         }
 | |
|         if (!(shared_mem_addr =
 | |
|                   LLVMBuildLoad2(comp_ctx->builder, OPQ_PTR_TYPE,
 | |
|                                  shared_mem_addr, "shared_mem_addr"))) {
 | |
|             aot_set_last_error("llvm build load failed");
 | |
|             return false;
 | |
|         }
 | |
|         /* memories[0]->memory_data */
 | |
|         offset = I32_CONST(offsetof(AOTMemoryInstance, memory_data));
 | |
|         if (!(func_ctx->mem_info[0].mem_base_addr = LLVMBuildInBoundsGEP2(
 | |
|                   comp_ctx->builder, INT8_TYPE, shared_mem_addr, &offset, 1,
 | |
|                   "mem_base_addr_offset"))) {
 | |
|             aot_set_last_error("llvm build in bounds gep failed");
 | |
|             return false;
 | |
|         }
 | |
|         /* memories[0]->cur_page_count */
 | |
|         offset = I32_CONST(offsetof(AOTMemoryInstance, cur_page_count));
 | |
|         if (!(func_ctx->mem_info[0].mem_cur_page_count_addr =
 | |
|                   LLVMBuildInBoundsGEP2(comp_ctx->builder, INT8_TYPE,
 | |
|                                         shared_mem_addr, &offset, 1,
 | |
|                                         "mem_cur_page_offset"))) {
 | |
|             aot_set_last_error("llvm build in bounds gep failed");
 | |
|             return false;
 | |
|         }
 | |
|         /* memories[0]->memory_data_size */
 | |
|         offset = I32_CONST(offsetof(AOTMemoryInstance, memory_data_size));
 | |
|         if (!(func_ctx->mem_info[0].mem_data_size_addr = LLVMBuildInBoundsGEP2(
 | |
|                   comp_ctx->builder, INT8_TYPE, shared_mem_addr, &offset, 1,
 | |
|                   "mem_data_size_offset"))) {
 | |
|             aot_set_last_error("llvm build in bounds gep failed");
 | |
|             return false;
 | |
|         }
 | |
|     }
 | |
|     else
 | |
| #endif
 | |
|     {
 | |
|         uint32 offset_of_global_table_data;
 | |
| 
 | |
|         if (comp_ctx->is_jit_mode)
 | |
|             offset_of_global_table_data =
 | |
|                 offsetof(WASMModuleInstance, global_table_data);
 | |
|         else
 | |
|             offset_of_global_table_data =
 | |
|                 offsetof(AOTModuleInstance, global_table_data);
 | |
| 
 | |
|         offset = I32_CONST(offset_of_global_table_data
 | |
|                            + offsetof(AOTMemoryInstance, memory_data));
 | |
|         if (!(func_ctx->mem_info[0].mem_base_addr = LLVMBuildInBoundsGEP2(
 | |
|                   comp_ctx->builder, INT8_TYPE, func_ctx->aot_inst, &offset, 1,
 | |
|                   "mem_base_addr_offset"))) {
 | |
|             aot_set_last_error("llvm build in bounds gep failed");
 | |
|             return false;
 | |
|         }
 | |
|         offset = I32_CONST(offset_of_global_table_data
 | |
|                            + offsetof(AOTMemoryInstance, cur_page_count));
 | |
|         if (!(func_ctx->mem_info[0].mem_cur_page_count_addr =
 | |
|                   LLVMBuildInBoundsGEP2(comp_ctx->builder, INT8_TYPE,
 | |
|                                         func_ctx->aot_inst, &offset, 1,
 | |
|                                         "mem_cur_page_offset"))) {
 | |
|             aot_set_last_error("llvm build in bounds gep failed");
 | |
|             return false;
 | |
|         }
 | |
|         offset = I32_CONST(offset_of_global_table_data
 | |
|                            + offsetof(AOTMemoryInstance, memory_data_size));
 | |
|         if (!(func_ctx->mem_info[0].mem_data_size_addr = LLVMBuildInBoundsGEP2(
 | |
|                   comp_ctx->builder, INT8_TYPE, func_ctx->aot_inst, &offset, 1,
 | |
|                   "mem_data_size_offset"))) {
 | |
|             aot_set_last_error("llvm build in bounds gep failed");
 | |
|             return false;
 | |
|         }
 | |
|     }
 | |
|     /* Store mem info base address before cast */
 | |
|     mem_info_base = func_ctx->mem_info[0].mem_base_addr;
 | |
| 
 | |
|     if (!(func_ctx->mem_info[0].mem_base_addr = LLVMBuildBitCast(
 | |
|               comp_ctx->builder, func_ctx->mem_info[0].mem_base_addr,
 | |
|               int8_ptr_type, "mem_base_addr_ptr"))) {
 | |
|         aot_set_last_error("llvm build bit cast failed");
 | |
|         return false;
 | |
|     }
 | |
|     if (!(func_ctx->mem_info[0].mem_cur_page_count_addr = LLVMBuildBitCast(
 | |
|               comp_ctx->builder, func_ctx->mem_info[0].mem_cur_page_count_addr,
 | |
|               INT32_PTR_TYPE, "mem_cur_page_ptr"))) {
 | |
|         aot_set_last_error("llvm build bit cast failed");
 | |
|         return false;
 | |
|     }
 | |
|     if (!(func_ctx->mem_info[0].mem_data_size_addr = LLVMBuildBitCast(
 | |
|               comp_ctx->builder, func_ctx->mem_info[0].mem_data_size_addr,
 | |
|               INT64_PTR_TYPE, "mem_data_size_ptr"))) {
 | |
|         aot_set_last_error("llvm build bit cast failed");
 | |
|         return false;
 | |
|     }
 | |
|     if (mem_space_unchanged) {
 | |
|         if (!(func_ctx->mem_info[0].mem_base_addr = LLVMBuildLoad2(
 | |
|                   comp_ctx->builder, OPQ_PTR_TYPE,
 | |
|                   func_ctx->mem_info[0].mem_base_addr, "mem_base_addr"))) {
 | |
|             aot_set_last_error("llvm build load failed");
 | |
|             return false;
 | |
|         }
 | |
|         if (!(func_ctx->mem_info[0].mem_cur_page_count_addr =
 | |
|                   LLVMBuildLoad2(comp_ctx->builder, I32_TYPE,
 | |
|                                  func_ctx->mem_info[0].mem_cur_page_count_addr,
 | |
|                                  "mem_cur_page_count"))) {
 | |
|             aot_set_last_error("llvm build load failed");
 | |
|             return false;
 | |
|         }
 | |
|         if (!(func_ctx->mem_info[0].mem_data_size_addr = LLVMBuildLoad2(
 | |
|                   comp_ctx->builder, I64_TYPE,
 | |
|                   func_ctx->mem_info[0].mem_data_size_addr, "mem_data_size"))) {
 | |
|             aot_set_last_error("llvm build load failed");
 | |
|             return false;
 | |
|         }
 | |
|     }
 | |
| #if WASM_ENABLE_SHARED_MEMORY != 0
 | |
|     else if (is_shared_memory) {
 | |
|         /* The base address for shared memory will never changed,
 | |
|             we can load the value here */
 | |
|         if (!(func_ctx->mem_info[0].mem_base_addr = LLVMBuildLoad2(
 | |
|                   comp_ctx->builder, OPQ_PTR_TYPE,
 | |
|                   func_ctx->mem_info[0].mem_base_addr, "mem_base_addr"))) {
 | |
|             aot_set_last_error("llvm build load failed");
 | |
|             return false;
 | |
|         }
 | |
|     }
 | |
| #endif
 | |
| 
 | |
|     bound_check_type = (comp_ctx->pointer_size == sizeof(uint64))
 | |
|                            ? INT64_PTR_TYPE
 | |
|                            : INT32_PTR_TYPE;
 | |
| 
 | |
|     /* Load memory bound check constants */
 | |
|     offset = I32_CONST(offsetof(AOTMemoryInstance, mem_bound_check_1byte)
 | |
|                        - offsetof(AOTMemoryInstance, memory_data));
 | |
|     if (!(func_ctx->mem_info[0].mem_bound_check_1byte =
 | |
|               LLVMBuildInBoundsGEP2(comp_ctx->builder, INT8_TYPE, mem_info_base,
 | |
|                                     &offset, 1, "bound_check_1byte_offset"))) {
 | |
|         aot_set_last_error("llvm build in bounds gep failed");
 | |
|         return false;
 | |
|     }
 | |
|     if (!(func_ctx->mem_info[0].mem_bound_check_1byte = LLVMBuildBitCast(
 | |
|               comp_ctx->builder, func_ctx->mem_info[0].mem_bound_check_1byte,
 | |
|               bound_check_type, "bound_check_1byte_ptr"))) {
 | |
|         aot_set_last_error("llvm build bit cast failed");
 | |
|         return false;
 | |
|     }
 | |
|     if (mem_space_unchanged) {
 | |
|         if (!(func_ctx->mem_info[0].mem_bound_check_1byte = LLVMBuildLoad2(
 | |
|                   comp_ctx->builder,
 | |
|                   (comp_ctx->pointer_size == sizeof(uint64)) ? I64_TYPE
 | |
|                                                              : I32_TYPE,
 | |
|                   func_ctx->mem_info[0].mem_bound_check_1byte,
 | |
|                   "bound_check_1byte"))) {
 | |
|             aot_set_last_error("llvm build load failed");
 | |
|             return false;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     offset = I32_CONST(offsetof(AOTMemoryInstance, mem_bound_check_2bytes)
 | |
|                        - offsetof(AOTMemoryInstance, memory_data));
 | |
|     if (!(func_ctx->mem_info[0].mem_bound_check_2bytes =
 | |
|               LLVMBuildInBoundsGEP2(comp_ctx->builder, INT8_TYPE, mem_info_base,
 | |
|                                     &offset, 1, "bound_check_2bytes_offset"))) {
 | |
|         aot_set_last_error("llvm build in bounds gep failed");
 | |
|         return false;
 | |
|     }
 | |
|     if (!(func_ctx->mem_info[0].mem_bound_check_2bytes = LLVMBuildBitCast(
 | |
|               comp_ctx->builder, func_ctx->mem_info[0].mem_bound_check_2bytes,
 | |
|               bound_check_type, "bound_check_2bytes_ptr"))) {
 | |
|         aot_set_last_error("llvm build bit cast failed");
 | |
|         return false;
 | |
|     }
 | |
|     if (mem_space_unchanged) {
 | |
|         if (!(func_ctx->mem_info[0].mem_bound_check_2bytes = LLVMBuildLoad2(
 | |
|                   comp_ctx->builder,
 | |
|                   (comp_ctx->pointer_size == sizeof(uint64)) ? I64_TYPE
 | |
|                                                              : I32_TYPE,
 | |
|                   func_ctx->mem_info[0].mem_bound_check_2bytes,
 | |
|                   "bound_check_2bytes"))) {
 | |
|             aot_set_last_error("llvm build load failed");
 | |
|             return false;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     offset = I32_CONST(offsetof(AOTMemoryInstance, mem_bound_check_4bytes)
 | |
|                        - offsetof(AOTMemoryInstance, memory_data));
 | |
|     if (!(func_ctx->mem_info[0].mem_bound_check_4bytes =
 | |
|               LLVMBuildInBoundsGEP2(comp_ctx->builder, INT8_TYPE, mem_info_base,
 | |
|                                     &offset, 1, "bound_check_4bytes_offset"))) {
 | |
|         aot_set_last_error("llvm build in bounds gep failed");
 | |
|         return false;
 | |
|     }
 | |
|     if (!(func_ctx->mem_info[0].mem_bound_check_4bytes = LLVMBuildBitCast(
 | |
|               comp_ctx->builder, func_ctx->mem_info[0].mem_bound_check_4bytes,
 | |
|               bound_check_type, "bound_check_4bytes_ptr"))) {
 | |
|         aot_set_last_error("llvm build bit cast failed");
 | |
|         return false;
 | |
|     }
 | |
|     if (mem_space_unchanged) {
 | |
|         if (!(func_ctx->mem_info[0].mem_bound_check_4bytes = LLVMBuildLoad2(
 | |
|                   comp_ctx->builder,
 | |
|                   (comp_ctx->pointer_size == sizeof(uint64)) ? I64_TYPE
 | |
|                                                              : I32_TYPE,
 | |
|                   func_ctx->mem_info[0].mem_bound_check_4bytes,
 | |
|                   "bound_check_4bytes"))) {
 | |
|             aot_set_last_error("llvm build load failed");
 | |
|             return false;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     offset = I32_CONST(offsetof(AOTMemoryInstance, mem_bound_check_8bytes)
 | |
|                        - offsetof(AOTMemoryInstance, memory_data));
 | |
|     if (!(func_ctx->mem_info[0].mem_bound_check_8bytes =
 | |
|               LLVMBuildInBoundsGEP2(comp_ctx->builder, INT8_TYPE, mem_info_base,
 | |
|                                     &offset, 1, "bound_check_8bytes_offset"))) {
 | |
|         aot_set_last_error("llvm build in bounds gep failed");
 | |
|         return false;
 | |
|     }
 | |
|     if (!(func_ctx->mem_info[0].mem_bound_check_8bytes = LLVMBuildBitCast(
 | |
|               comp_ctx->builder, func_ctx->mem_info[0].mem_bound_check_8bytes,
 | |
|               bound_check_type, "bound_check_8bytes_ptr"))) {
 | |
|         aot_set_last_error("llvm build bit cast failed");
 | |
|         return false;
 | |
|     }
 | |
|     if (mem_space_unchanged) {
 | |
|         if (!(func_ctx->mem_info[0].mem_bound_check_8bytes = LLVMBuildLoad2(
 | |
|                   comp_ctx->builder,
 | |
|                   (comp_ctx->pointer_size == sizeof(uint64)) ? I64_TYPE
 | |
|                                                              : I32_TYPE,
 | |
|                   func_ctx->mem_info[0].mem_bound_check_8bytes,
 | |
|                   "bound_check_8bytes"))) {
 | |
|             aot_set_last_error("llvm build load failed");
 | |
|             return false;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     offset = I32_CONST(offsetof(AOTMemoryInstance, mem_bound_check_16bytes)
 | |
|                        - offsetof(AOTMemoryInstance, memory_data));
 | |
|     if (!(func_ctx->mem_info[0].mem_bound_check_16bytes = LLVMBuildInBoundsGEP2(
 | |
|               comp_ctx->builder, INT8_TYPE, mem_info_base, &offset, 1,
 | |
|               "bound_check_16bytes_offset"))) {
 | |
|         aot_set_last_error("llvm build in bounds gep failed");
 | |
|         return false;
 | |
|     }
 | |
|     if (!(func_ctx->mem_info[0].mem_bound_check_16bytes = LLVMBuildBitCast(
 | |
|               comp_ctx->builder, func_ctx->mem_info[0].mem_bound_check_16bytes,
 | |
|               bound_check_type, "bound_check_16bytes_ptr"))) {
 | |
|         aot_set_last_error("llvm build bit cast failed");
 | |
|         return false;
 | |
|     }
 | |
|     if (mem_space_unchanged) {
 | |
|         if (!(func_ctx->mem_info[0].mem_bound_check_16bytes = LLVMBuildLoad2(
 | |
|                   comp_ctx->builder,
 | |
|                   (comp_ctx->pointer_size == sizeof(uint64)) ? I64_TYPE
 | |
|                                                              : I32_TYPE,
 | |
|                   func_ctx->mem_info[0].mem_bound_check_16bytes,
 | |
|                   "bound_check_16bytes"))) {
 | |
|             aot_set_last_error("llvm build load failed");
 | |
|             return false;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     return true;
 | |
| }
 | |
| 
 | |
| #define BUILD_IS_NOT_NULL(value, res, name)                                \
 | |
|     do {                                                                   \
 | |
|         if (!(res = LLVMBuildIsNotNull(comp_ctx->builder, value, name))) { \
 | |
|             aot_set_last_error("llvm build is not null failed.");          \
 | |
|             goto fail;                                                     \
 | |
|         }                                                                  \
 | |
|     } while (0)
 | |
| 
 | |
| #define get_module_extra_field_offset(field)                        \
 | |
|     do {                                                            \
 | |
|         offset_u32 = get_module_inst_extra_offset(comp_ctx);        \
 | |
|         if (comp_ctx->is_jit_mode)                                  \
 | |
|             offset_u32 += offsetof(WASMModuleInstanceExtra, field); \
 | |
|         else                                                        \
 | |
|             offset_u32 += offsetof(AOTModuleInstanceExtra, field);  \
 | |
|     } while (0)
 | |
| 
 | |
| #define LOAD_MODULE_EXTRA_FIELD_AND_ALLOCA(field, type)                        \
 | |
|     do {                                                                       \
 | |
|         get_module_extra_field_offset(field);                                  \
 | |
|         offset = I32_CONST(offset_u32);                                        \
 | |
|         CHECK_LLVM_CONST(offset);                                              \
 | |
|         if (!(field_p = LLVMBuildInBoundsGEP2(comp_ctx->builder, INT8_TYPE,    \
 | |
|                                               func_ctx->aot_inst, &offset, 1,  \
 | |
|                                               #field "_p"))) {                 \
 | |
|             aot_set_last_error("llvm build inbounds gep failed");              \
 | |
|             goto fail;                                                         \
 | |
|         }                                                                      \
 | |
|         if (!(load_val =                                                       \
 | |
|                   LLVMBuildLoad2(comp_ctx->builder, type, field_p, #field))) { \
 | |
|             aot_set_last_error("llvm build load failed");                      \
 | |
|             goto fail;                                                         \
 | |
|         }                                                                      \
 | |
|         if (!(func_ctx->field =                                                \
 | |
|                   LLVMBuildAlloca(comp_ctx->builder, type, #field))) {         \
 | |
|             aot_set_last_error("llvm build alloca failed");                    \
 | |
|             goto fail;                                                         \
 | |
|         }                                                                      \
 | |
|         if (!LLVMBuildStore(comp_ctx->builder, load_val, func_ctx->field)) {   \
 | |
|             aot_set_last_error("llvm build store failed");                     \
 | |
|             goto fail;                                                         \
 | |
|         }                                                                      \
 | |
|     } while (0)
 | |
| 
 | |
| static bool
 | |
| create_shared_heap_info(AOTCompContext *comp_ctx, AOTFuncContext *func_ctx)
 | |
| {
 | |
| #if WASM_ENABLE_SHARED_HEAP != 0
 | |
|     LLVMValueRef offset, field_p, load_val, shared_heap_head_p,
 | |
|         shared_heap_head, cmp, field_p_or_default, shared_heap_head_start_off,
 | |
|         shared_heap_head_start_off_minus_one;
 | |
|     LLVMTypeRef shared_heap_offset_type;
 | |
|     uint32 offset_u32;
 | |
| #if WASM_ENABLE_MEMORY64 == 0
 | |
|     bool is_memory64 = false;
 | |
| #else
 | |
|     bool is_memory64 = IS_MEMORY64;
 | |
| #endif
 | |
| 
 | |
|     shared_heap_offset_type =
 | |
|         comp_ctx->pointer_size == sizeof(uint64) ? I64_TYPE : I32_TYPE;
 | |
| 
 | |
|     /* shared_heap_base_addr_adj, shared_heap_start_off, and
 | |
|      * shared_heap_end_off can be updated later, use local variable to
 | |
|      * represent them */
 | |
|     LOAD_MODULE_EXTRA_FIELD_AND_ALLOCA(shared_heap_base_addr_adj,
 | |
|                                        INT8_PTR_TYPE);
 | |
|     LOAD_MODULE_EXTRA_FIELD_AND_ALLOCA(shared_heap_start_off,
 | |
|                                        shared_heap_offset_type);
 | |
|     LOAD_MODULE_EXTRA_FIELD_AND_ALLOCA(shared_heap_end_off,
 | |
|                                        shared_heap_offset_type);
 | |
| 
 | |
|     /* Shared Heap head start off won't be updated, no need to alloca */
 | |
|     get_module_extra_field_offset(shared_heap);
 | |
|     offset = I32_CONST(offset_u32);
 | |
|     CHECK_LLVM_CONST(offset);
 | |
|     if (!(shared_heap_head_p = LLVMBuildInBoundsGEP2(
 | |
|               comp_ctx->builder, INT8_TYPE, func_ctx->aot_inst, &offset, 1,
 | |
|               "shared_heap_head_p"))) {
 | |
|         aot_set_last_error("llvm build inbounds gep failed");
 | |
|         goto fail;
 | |
|     }
 | |
|     if (!(shared_heap_head =
 | |
|               LLVMBuildLoad2(comp_ctx->builder, INT8_PTR_TYPE,
 | |
|                              shared_heap_head_p, "shared_heap_head"))) {
 | |
|         aot_set_last_error("llvm build load failed");
 | |
|         goto fail;
 | |
|     }
 | |
|     BUILD_IS_NOT_NULL(shared_heap_head, cmp, "has_shared_heap");
 | |
| 
 | |
|     if (is_memory64) {
 | |
|         offset_u32 = offsetof(WASMSharedHeap, start_off_mem64);
 | |
|     }
 | |
|     else {
 | |
|         offset_u32 = offsetof(WASMSharedHeap, start_off_mem32);
 | |
|     }
 | |
|     offset = I32_CONST(offset_u32);
 | |
|     CHECK_LLVM_CONST(offset);
 | |
|     if (!(field_p = LLVMBuildInBoundsGEP2(comp_ctx->builder, INT8_TYPE,
 | |
|                                           shared_heap_head, &offset, 1,
 | |
|                                           "head_start_off_p"))) {
 | |
|         aot_set_last_error("llvm build inbounds gep failed");
 | |
|         goto fail;
 | |
|     }
 | |
| 
 | |
|     /* Select a valid shared heap head ptr or safe alloca ptr stores
 | |
|      * shared_heap_start_off(UINT32_MAX/UINT64_MAX) */
 | |
|     if (!(field_p_or_default = LLVMBuildSelect(comp_ctx->builder, cmp, field_p,
 | |
|                                                func_ctx->shared_heap_start_off,
 | |
|                                                "ptr_or_default"))) {
 | |
|         aot_set_last_error("llvm build select failed");
 | |
|         goto fail;
 | |
|     }
 | |
| 
 | |
|     if (!(shared_heap_head_start_off = LLVMBuildLoad2(
 | |
|               comp_ctx->builder, shared_heap_offset_type, field_p_or_default,
 | |
|               "shared_heap_head_start_off"))) {
 | |
|         aot_set_last_error("llvm build load failed");
 | |
|         goto fail;
 | |
|     }
 | |
|     if (!(shared_heap_head_start_off_minus_one = LLVMBuildAdd(
 | |
|               comp_ctx->builder, shared_heap_head_start_off,
 | |
|               comp_ctx->pointer_size == sizeof(uint64) ? I64_NEG_ONE
 | |
|                                                        : I32_NEG_ONE,
 | |
|               "head_start_off_minus_one"))) {
 | |
|         aot_set_last_error("llvm build load failed");
 | |
|         goto fail;
 | |
|     }
 | |
| 
 | |
|     /* if there is attached shared heap(s), the value will be valid start_off-1,
 | |
|      * otherwise it will be UINT32_MAX/UINT64_MAX, so during the bounds checks,
 | |
|      * when has attached shared heap:
 | |
|      *   offset > start_off - 1 => offset >= start_off
 | |
|      * when no attached shared heap:
 | |
|      *   offset > UINT32_MAX/UINT64_MAX is always false
 | |
|      * */
 | |
|     if (!(func_ctx->shared_heap_head_start_off = LLVMBuildSelect(
 | |
|               comp_ctx->builder, cmp, shared_heap_head_start_off_minus_one,
 | |
|               shared_heap_head_start_off, "head_start_off"))) {
 | |
|         aot_set_last_error("llvm build select failed");
 | |
|         goto fail;
 | |
|     }
 | |
|     return true;
 | |
| fail:
 | |
|     return false;
 | |
| #else  /* else of WASM_ENABLE_SHARED_HEAP != 0 */
 | |
|     return true;
 | |
| #endif /* end of WASM_ENABLE_SHARED_HEAP != 0 */
 | |
| }
 | |
| 
 | |
| static bool
 | |
| create_cur_exception(const AOTCompContext *comp_ctx, AOTFuncContext *func_ctx)
 | |
| {
 | |
|     LLVMValueRef offset;
 | |
| 
 | |
|     offset = I32_CONST(offsetof(AOTModuleInstance, cur_exception));
 | |
|     func_ctx->cur_exception =
 | |
|         LLVMBuildInBoundsGEP2(comp_ctx->builder, INT8_TYPE, func_ctx->aot_inst,
 | |
|                               &offset, 1, "cur_exception");
 | |
|     if (!func_ctx->cur_exception) {
 | |
|         aot_set_last_error("llvm build in bounds gep failed.");
 | |
|         return false;
 | |
|     }
 | |
|     return true;
 | |
| }
 | |
| 
 | |
| static bool
 | |
| create_func_type_indexes(const AOTCompContext *comp_ctx,
 | |
|                          AOTFuncContext *func_ctx)
 | |
| {
 | |
|     LLVMValueRef offset, func_type_indexes_ptr;
 | |
|     LLVMTypeRef int32_ptr_type;
 | |
| 
 | |
|     offset = I32_CONST(offsetof(AOTModuleInstance, func_type_indexes));
 | |
|     func_type_indexes_ptr =
 | |
|         LLVMBuildInBoundsGEP2(comp_ctx->builder, INT8_TYPE, func_ctx->aot_inst,
 | |
|                               &offset, 1, "func_type_indexes_ptr");
 | |
|     if (!func_type_indexes_ptr) {
 | |
|         aot_set_last_error("llvm build add failed.");
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     if (!(int32_ptr_type = LLVMPointerType(INT32_PTR_TYPE, 0))) {
 | |
|         aot_set_last_error("llvm get pointer type failed.");
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     func_ctx->func_type_indexes =
 | |
|         LLVMBuildBitCast(comp_ctx->builder, func_type_indexes_ptr,
 | |
|                          int32_ptr_type, "func_type_indexes_tmp");
 | |
|     if (!func_ctx->func_type_indexes) {
 | |
|         aot_set_last_error("llvm build bit cast failed.");
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     func_ctx->func_type_indexes =
 | |
|         LLVMBuildLoad2(comp_ctx->builder, INT32_PTR_TYPE,
 | |
|                        func_ctx->func_type_indexes, "func_type_indexes");
 | |
|     if (!func_ctx->func_type_indexes) {
 | |
|         aot_set_last_error("llvm build load failed.");
 | |
|         return false;
 | |
|     }
 | |
|     return true;
 | |
| }
 | |
| 
 | |
| static bool
 | |
| create_func_ptrs(const AOTCompContext *comp_ctx, AOTFuncContext *func_ctx)
 | |
| {
 | |
|     LLVMValueRef offset;
 | |
| 
 | |
|     offset = I32_CONST(offsetof(AOTModuleInstance, func_ptrs));
 | |
|     func_ctx->func_ptrs =
 | |
|         LLVMBuildInBoundsGEP2(comp_ctx->builder, INT8_TYPE, func_ctx->aot_inst,
 | |
|                               &offset, 1, "func_ptrs_offset");
 | |
|     if (!func_ctx->func_ptrs) {
 | |
|         aot_set_last_error("llvm build in bounds gep failed.");
 | |
|         return false;
 | |
|     }
 | |
|     func_ctx->func_ptrs =
 | |
|         LLVMBuildBitCast(comp_ctx->builder, func_ctx->func_ptrs,
 | |
|                          comp_ctx->exec_env_type, "func_ptrs_tmp");
 | |
|     if (!func_ctx->func_ptrs) {
 | |
|         aot_set_last_error("llvm build bit cast failed.");
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     func_ctx->func_ptrs = LLVMBuildLoad2(comp_ctx->builder, OPQ_PTR_TYPE,
 | |
|                                          func_ctx->func_ptrs, "func_ptrs_ptr");
 | |
|     if (!func_ctx->func_ptrs) {
 | |
|         aot_set_last_error("llvm build load failed.");
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     func_ctx->func_ptrs =
 | |
|         LLVMBuildBitCast(comp_ctx->builder, func_ctx->func_ptrs,
 | |
|                          comp_ctx->exec_env_type, "func_ptrs");
 | |
|     if (!func_ctx->func_ptrs) {
 | |
|         aot_set_last_error("llvm build bit cast failed.");
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     return true;
 | |
| }
 | |
| 
 | |
| const char *aot_stack_sizes_name = AOT_STACK_SIZES_NAME;
 | |
| const char *aot_stack_sizes_alias_name = AOT_STACK_SIZES_ALIAS_NAME;
 | |
| const char *aot_stack_sizes_section_name = AOT_STACK_SIZES_SECTION_NAME;
 | |
| 
 | |
| static bool
 | |
| aot_create_stack_sizes(const AOTCompData *comp_data, AOTCompContext *comp_ctx)
 | |
| {
 | |
|     LLVMValueRef stack_sizes, *values, array, alias;
 | |
|     LLVMTypeRef stack_sizes_type;
 | |
| #if LLVM_VERSION_MAJOR <= 13
 | |
|     LLVMTypeRef alias_type;
 | |
| #endif
 | |
|     uint64 size;
 | |
|     uint32 i;
 | |
| 
 | |
|     stack_sizes_type = LLVMArrayType(I32_TYPE, comp_data->func_count);
 | |
|     if (!stack_sizes_type) {
 | |
|         aot_set_last_error("failed to create stack_sizes type.");
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     stack_sizes =
 | |
|         LLVMAddGlobal(comp_ctx->module, stack_sizes_type, aot_stack_sizes_name);
 | |
|     if (!stack_sizes) {
 | |
|         aot_set_last_error("failed to create stack_sizes global.");
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     size = sizeof(LLVMValueRef) * comp_data->func_count;
 | |
|     if (size >= UINT32_MAX || !(values = wasm_runtime_malloc((uint32)size))) {
 | |
|         aot_set_last_error("allocate memory failed.");
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     for (i = 0; i < comp_data->func_count; i++) {
 | |
|         /*
 | |
|          * This value is a placeholder, which will be replaced
 | |
|          * after the corresponding functions are compiled.
 | |
|          *
 | |
|          * Don't use zeros because LLVM can optimize them to
 | |
|          * zeroinitializer.
 | |
|          */
 | |
|         values[i] = I32_NEG_ONE;
 | |
|     }
 | |
| 
 | |
|     array = LLVMConstArray(I32_TYPE, values, comp_data->func_count);
 | |
|     wasm_runtime_free(values);
 | |
|     if (!array) {
 | |
|         aot_set_last_error("failed to create stack_sizes initializer.");
 | |
|         return false;
 | |
|     }
 | |
|     LLVMSetInitializer(stack_sizes, array);
 | |
| 
 | |
|     /*
 | |
|      * create an alias so that aot_resolve_stack_sizes can find it.
 | |
|      */
 | |
| #if LLVM_VERSION_MAJOR > 13
 | |
|     alias = LLVMAddAlias2(comp_ctx->module, stack_sizes_type, 0, stack_sizes,
 | |
|                           aot_stack_sizes_alias_name);
 | |
| #else
 | |
|     alias_type = LLVMPointerType(stack_sizes_type, 0);
 | |
|     if (!alias_type) {
 | |
|         aot_set_last_error("failed to create alias type.");
 | |
|         return false;
 | |
|     }
 | |
|     alias = LLVMAddAlias(comp_ctx->module, alias_type, stack_sizes,
 | |
|                          aot_stack_sizes_alias_name);
 | |
| #endif
 | |
|     if (!alias) {
 | |
|         aot_set_last_error("failed to create stack_sizes alias.");
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     /*
 | |
|      * make the original symbol internal. we mainly use this version to
 | |
|      * avoid creating extra relocations in the precheck functions.
 | |
|      */
 | |
|     LLVMSetLinkage(stack_sizes, LLVMInternalLinkage);
 | |
|     /*
 | |
|      * for AOT, place it into a dedicated section for the convenience
 | |
|      * of the AOT file generation and symbol resolutions.
 | |
|      *
 | |
|      * for JIT, it doesn't matter.
 | |
|      */
 | |
|     if (!comp_ctx->is_jit_mode) {
 | |
|         LLVMSetSection(stack_sizes, aot_stack_sizes_section_name);
 | |
|     }
 | |
|     comp_ctx->stack_sizes_type = stack_sizes_type;
 | |
|     comp_ctx->stack_sizes = stack_sizes;
 | |
|     return true;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Create function compiler context
 | |
|  */
 | |
| static AOTFuncContext *
 | |
| aot_create_func_context(const AOTCompData *comp_data, AOTCompContext *comp_ctx,
 | |
|                         AOTFunc *func, uint32 func_index)
 | |
| {
 | |
|     AOTFuncContext *func_ctx;
 | |
|     AOTFuncType *aot_func_type =
 | |
|         (AOTFuncType *)comp_data->types[func->func_type_index];
 | |
|     WASMModule *module = comp_ctx->comp_data->wasm_module;
 | |
|     WASMFunction *wasm_func = module->functions[func_index];
 | |
|     AOTBlock *aot_block;
 | |
|     LLVMTypeRef int8_ptr_type;
 | |
|     uint64 size;
 | |
| 
 | |
|     /* Allocate memory for the function context */
 | |
|     size = offsetof(AOTFuncContext, locals)
 | |
|            + sizeof(LLVMValueRef)
 | |
|                  * ((uint64)aot_func_type->param_count + func->local_count);
 | |
|     if (size >= UINT32_MAX || !(func_ctx = wasm_runtime_malloc((uint32)size))) {
 | |
|         aot_set_last_error("allocate memory failed.");
 | |
|         return NULL;
 | |
|     }
 | |
| 
 | |
|     memset(func_ctx, 0, (uint32)size);
 | |
|     func_ctx->aot_func = func;
 | |
| 
 | |
|     func_ctx->module = comp_ctx->module;
 | |
| 
 | |
|     /* Add LLVM function */
 | |
|     if (!(func_ctx->func = aot_add_llvm_func(
 | |
|               comp_ctx, func_ctx->module, aot_func_type, func_index,
 | |
|               &func_ctx->func_type, &func_ctx->precheck_func))) {
 | |
|         goto fail;
 | |
|     }
 | |
| 
 | |
|     /* Create function's first AOTBlock */
 | |
|     if (!(aot_block =
 | |
|               aot_create_func_block(comp_ctx, func_ctx, func, aot_func_type))) {
 | |
|         goto fail;
 | |
|     }
 | |
| 
 | |
| #if WASM_ENABLE_DEBUG_AOT != 0
 | |
|     func_ctx->debug_func = dwarf_gen_func_info(comp_ctx, func_ctx);
 | |
| #endif
 | |
| 
 | |
|     aot_block_stack_push(&func_ctx->block_stack, aot_block);
 | |
| 
 | |
|     /* Add local variables */
 | |
|     LLVMPositionBuilderAtEnd(comp_ctx->builder, aot_block->llvm_entry_block);
 | |
| 
 | |
|     if (!create_basic_func_context(comp_ctx, func_ctx)) {
 | |
|         goto fail;
 | |
|     }
 | |
| 
 | |
|     /* Get argv buffer address */
 | |
|     if (wasm_func->has_op_func_call && !create_argv_buf(comp_ctx, func_ctx)) {
 | |
|         goto fail;
 | |
|     }
 | |
| 
 | |
|     /* Get auxiliary stack info */
 | |
|     if (wasm_func->has_op_set_global_aux_stack
 | |
|         && !create_aux_stack_info(comp_ctx, func_ctx)) {
 | |
|         goto fail;
 | |
|     }
 | |
| 
 | |
|     if (comp_ctx->aux_stack_frame_type
 | |
|         && !create_aux_stack_frame(comp_ctx, func_ctx)) {
 | |
|         goto fail;
 | |
|     }
 | |
| 
 | |
|     /* Create local variables */
 | |
|     if (!create_local_variables(comp_data, comp_ctx, func_ctx, func)) {
 | |
|         goto fail;
 | |
|     }
 | |
| 
 | |
|     if (!(int8_ptr_type = LLVMPointerType(INT8_PTR_TYPE, 0))) {
 | |
|         aot_set_last_error("llvm add pointer type failed.");
 | |
|         goto fail;
 | |
|     }
 | |
| 
 | |
|     /* Create base addr, end addr, data size of mem, heap */
 | |
|     if (wasm_func->has_memory_operations
 | |
|         && !create_memory_info(comp_ctx, func_ctx, int8_ptr_type, func_index)) {
 | |
|         goto fail;
 | |
|     }
 | |
| 
 | |
|     /* Load current exception */
 | |
|     if (!create_cur_exception(comp_ctx, func_ctx)) {
 | |
|         goto fail;
 | |
|     }
 | |
| 
 | |
|     /* Load function type indexes */
 | |
|     if (wasm_func->has_op_call_indirect
 | |
|         && !create_func_type_indexes(comp_ctx, func_ctx)) {
 | |
|         goto fail;
 | |
|     }
 | |
| 
 | |
|     /* Load function pointers */
 | |
|     if (!create_func_ptrs(comp_ctx, func_ctx)) {
 | |
|         goto fail;
 | |
|     }
 | |
| 
 | |
|     /* Load shared heap, shared heap start off mem32 or mem64 */
 | |
|     if ((comp_ctx->enable_shared_heap || comp_ctx->enable_shared_chain)
 | |
|         && !create_shared_heap_info(comp_ctx, func_ctx)) {
 | |
|         goto fail;
 | |
|     }
 | |
| 
 | |
|     return func_ctx;
 | |
| 
 | |
| fail:
 | |
|     if (func_ctx->mem_info)
 | |
|         wasm_runtime_free(func_ctx->mem_info);
 | |
|     aot_block_stack_destroy(comp_ctx, &func_ctx->block_stack);
 | |
|     wasm_runtime_free(func_ctx);
 | |
|     return NULL;
 | |
| }
 | |
| 
 | |
| static void
 | |
| aot_destroy_func_contexts(AOTCompContext *comp_ctx, AOTFuncContext **func_ctxes,
 | |
|                           uint32 count)
 | |
| {
 | |
|     uint32 i;
 | |
| 
 | |
|     for (i = 0; i < count; i++)
 | |
|         if (func_ctxes[i]) {
 | |
|             if (func_ctxes[i]->mem_info)
 | |
|                 wasm_runtime_free(func_ctxes[i]->mem_info);
 | |
|             aot_block_stack_destroy(comp_ctx, &func_ctxes[i]->block_stack);
 | |
|             aot_checked_addr_list_destroy(func_ctxes[i]);
 | |
|             wasm_runtime_free(func_ctxes[i]);
 | |
|         }
 | |
|     wasm_runtime_free(func_ctxes);
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Create function compiler contexts
 | |
|  */
 | |
| static AOTFuncContext **
 | |
| aot_create_func_contexts(const AOTCompData *comp_data, AOTCompContext *comp_ctx)
 | |
| {
 | |
|     AOTFuncContext **func_ctxes;
 | |
|     uint64 size;
 | |
|     uint32 i;
 | |
| 
 | |
|     if ((comp_ctx->enable_stack_bound_check
 | |
|          || comp_ctx->enable_stack_estimation)
 | |
|         && !aot_create_stack_sizes(comp_data, comp_ctx))
 | |
|         return NULL;
 | |
| 
 | |
|     /* Allocate memory */
 | |
|     size = sizeof(AOTFuncContext *) * (uint64)comp_data->func_count;
 | |
|     if (size >= UINT32_MAX
 | |
|         || !(func_ctxes = wasm_runtime_malloc((uint32)size))) {
 | |
|         aot_set_last_error("allocate memory failed.");
 | |
|         return NULL;
 | |
|     }
 | |
| 
 | |
|     memset(func_ctxes, 0, size);
 | |
| 
 | |
|     /* Create each function context */
 | |
|     for (i = 0; i < comp_data->func_count; i++) {
 | |
|         AOTFunc *func = comp_data->funcs[i];
 | |
|         if (!(func_ctxes[i] =
 | |
|                   aot_create_func_context(comp_data, comp_ctx, func, i))) {
 | |
|             aot_destroy_func_contexts(comp_ctx, func_ctxes,
 | |
|                                       comp_data->func_count);
 | |
|             return NULL;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     return func_ctxes;
 | |
| }
 | |
| 
 | |
| static bool
 | |
| aot_set_llvm_basic_types(AOTLLVMTypes *basic_types, LLVMContextRef context,
 | |
|                          int pointer_size)
 | |
| {
 | |
|     basic_types->int1_type = LLVMInt1TypeInContext(context);
 | |
|     basic_types->int8_type = LLVMInt8TypeInContext(context);
 | |
|     basic_types->int16_type = LLVMInt16TypeInContext(context);
 | |
|     basic_types->int32_type = LLVMInt32TypeInContext(context);
 | |
|     basic_types->int64_type = LLVMInt64TypeInContext(context);
 | |
|     basic_types->float32_type = LLVMFloatTypeInContext(context);
 | |
|     basic_types->float64_type = LLVMDoubleTypeInContext(context);
 | |
|     basic_types->void_type = LLVMVoidTypeInContext(context);
 | |
| 
 | |
|     basic_types->meta_data_type = LLVMMetadataTypeInContext(context);
 | |
| 
 | |
|     basic_types->int8_ptr_type = LLVMPointerType(basic_types->int8_type, 0);
 | |
| 
 | |
|     if (basic_types->int8_ptr_type) {
 | |
|         basic_types->int8_pptr_type =
 | |
|             LLVMPointerType(basic_types->int8_ptr_type, 0);
 | |
|     }
 | |
| 
 | |
|     basic_types->int16_ptr_type = LLVMPointerType(basic_types->int16_type, 0);
 | |
|     basic_types->int32_ptr_type = LLVMPointerType(basic_types->int32_type, 0);
 | |
|     basic_types->int64_ptr_type = LLVMPointerType(basic_types->int64_type, 0);
 | |
|     basic_types->float32_ptr_type =
 | |
|         LLVMPointerType(basic_types->float32_type, 0);
 | |
|     basic_types->float64_ptr_type =
 | |
|         LLVMPointerType(basic_types->float64_type, 0);
 | |
| 
 | |
|     basic_types->i8x16_vec_type = LLVMVectorType(basic_types->int8_type, 16);
 | |
|     basic_types->i16x8_vec_type = LLVMVectorType(basic_types->int16_type, 8);
 | |
|     basic_types->i32x4_vec_type = LLVMVectorType(basic_types->int32_type, 4);
 | |
|     basic_types->i64x2_vec_type = LLVMVectorType(basic_types->int64_type, 2);
 | |
|     basic_types->f32x4_vec_type = LLVMVectorType(basic_types->float32_type, 4);
 | |
|     basic_types->f64x2_vec_type = LLVMVectorType(basic_types->float64_type, 2);
 | |
| 
 | |
|     basic_types->v128_type = basic_types->i64x2_vec_type;
 | |
|     basic_types->v128_ptr_type = LLVMPointerType(basic_types->v128_type, 0);
 | |
| 
 | |
|     basic_types->int8_ptr_type_gs =
 | |
|         LLVMPointerType(basic_types->int8_type, 256);
 | |
|     basic_types->int16_ptr_type_gs =
 | |
|         LLVMPointerType(basic_types->int16_type, 256);
 | |
|     basic_types->int32_ptr_type_gs =
 | |
|         LLVMPointerType(basic_types->int32_type, 256);
 | |
|     basic_types->int64_ptr_type_gs =
 | |
|         LLVMPointerType(basic_types->int64_type, 256);
 | |
|     basic_types->float32_ptr_type_gs =
 | |
|         LLVMPointerType(basic_types->float32_type, 256);
 | |
|     basic_types->float64_ptr_type_gs =
 | |
|         LLVMPointerType(basic_types->float64_type, 256);
 | |
|     basic_types->v128_ptr_type_gs =
 | |
|         LLVMPointerType(basic_types->v128_type, 256);
 | |
|     if (!basic_types->int8_ptr_type_gs || !basic_types->int16_ptr_type_gs
 | |
|         || !basic_types->int32_ptr_type_gs || !basic_types->int64_ptr_type_gs
 | |
|         || !basic_types->float32_ptr_type_gs
 | |
|         || !basic_types->float64_ptr_type_gs
 | |
|         || !basic_types->v128_ptr_type_gs) {
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     basic_types->i1x2_vec_type = LLVMVectorType(basic_types->int1_type, 2);
 | |
| 
 | |
|     basic_types->funcref_type = LLVMInt32TypeInContext(context);
 | |
|     basic_types->externref_type = LLVMInt32TypeInContext(context);
 | |
| 
 | |
|     if (pointer_size == 4) {
 | |
|         basic_types->intptr_t_type = basic_types->int32_type;
 | |
|         basic_types->intptr_t_ptr_type = basic_types->int32_ptr_type;
 | |
|         basic_types->size_t_type = basic_types->int32_type;
 | |
|     }
 | |
|     else {
 | |
|         basic_types->intptr_t_type = basic_types->int64_type;
 | |
|         basic_types->intptr_t_ptr_type = basic_types->int64_ptr_type;
 | |
|         basic_types->size_t_type = basic_types->int64_type;
 | |
|     }
 | |
| 
 | |
|     basic_types->gc_ref_type = basic_types->int8_ptr_type;
 | |
|     basic_types->gc_ref_ptr_type = basic_types->int8_pptr_type;
 | |
| 
 | |
|     return (basic_types->int8_ptr_type && basic_types->int8_pptr_type
 | |
|             && basic_types->int16_ptr_type && basic_types->int32_ptr_type
 | |
|             && basic_types->int64_ptr_type && basic_types->intptr_t_type
 | |
|             && basic_types->intptr_t_ptr_type && basic_types->float32_ptr_type
 | |
|             && basic_types->float64_ptr_type && basic_types->i8x16_vec_type
 | |
|             && basic_types->i16x8_vec_type && basic_types->i32x4_vec_type
 | |
|             && basic_types->i64x2_vec_type && basic_types->f32x4_vec_type
 | |
|             && basic_types->f64x2_vec_type && basic_types->i1x2_vec_type
 | |
|             && basic_types->meta_data_type && basic_types->funcref_type
 | |
|             && basic_types->externref_type && basic_types->gc_ref_type
 | |
|             && basic_types->gc_ref_ptr_type)
 | |
|                ? true
 | |
|                : false;
 | |
| }
 | |
| 
 | |
| static bool
 | |
| aot_create_llvm_consts(AOTLLVMConsts *consts, AOTCompContext *comp_ctx)
 | |
| {
 | |
| #define CREATE_I1_CONST(name, value)                                       \
 | |
|     if (!(consts->i1_##name =                                              \
 | |
|               LLVMConstInt(comp_ctx->basic_types.int1_type, value, true))) \
 | |
|         return false;
 | |
| 
 | |
|     CREATE_I1_CONST(zero, 0)
 | |
|     CREATE_I1_CONST(one, 1)
 | |
| #undef CREATE_I1_CONST
 | |
| 
 | |
|     if (!(consts->i8_zero = I8_CONST(0)))
 | |
|         return false;
 | |
| 
 | |
|     if (!(consts->i8_one = I8_CONST(1)))
 | |
|         return false;
 | |
| 
 | |
|     if (!(consts->f32_zero = F32_CONST(0)))
 | |
|         return false;
 | |
| 
 | |
|     if (!(consts->f64_zero = F64_CONST(0)))
 | |
|         return false;
 | |
| 
 | |
| #define CREATE_I32_CONST(name, value)                                \
 | |
|     if (!(consts->i32_##name = LLVMConstInt(I32_TYPE, value, true))) \
 | |
|         return false;
 | |
| 
 | |
|     CREATE_I32_CONST(min, (uint32)INT32_MIN)
 | |
|     CREATE_I32_CONST(neg_one, (uint32)-1)
 | |
|     CREATE_I32_CONST(zero, 0)
 | |
|     CREATE_I32_CONST(one, 1)
 | |
|     CREATE_I32_CONST(two, 2)
 | |
|     CREATE_I32_CONST(three, 3)
 | |
|     CREATE_I32_CONST(four, 4)
 | |
|     CREATE_I32_CONST(five, 5)
 | |
|     CREATE_I32_CONST(six, 6)
 | |
|     CREATE_I32_CONST(seven, 7)
 | |
|     CREATE_I32_CONST(eight, 8)
 | |
|     CREATE_I32_CONST(nine, 9)
 | |
|     CREATE_I32_CONST(ten, 10)
 | |
|     CREATE_I32_CONST(eleven, 11)
 | |
|     CREATE_I32_CONST(twelve, 12)
 | |
|     CREATE_I32_CONST(thirteen, 13)
 | |
|     CREATE_I32_CONST(fourteen, 14)
 | |
|     CREATE_I32_CONST(fifteen, 15)
 | |
|     CREATE_I32_CONST(31, 31)
 | |
|     CREATE_I32_CONST(32, 32)
 | |
| #undef CREATE_I32_CONST
 | |
| 
 | |
| #define CREATE_I64_CONST(name, value)                                \
 | |
|     if (!(consts->i64_##name = LLVMConstInt(I64_TYPE, value, true))) \
 | |
|         return false;
 | |
| 
 | |
|     CREATE_I64_CONST(min, (uint64)INT64_MIN)
 | |
|     CREATE_I64_CONST(neg_one, (uint64)-1)
 | |
|     CREATE_I64_CONST(zero, 0)
 | |
|     CREATE_I64_CONST(63, 63)
 | |
|     CREATE_I64_CONST(64, 64)
 | |
| #undef CREATE_I64_CONST
 | |
| 
 | |
| #define CREATE_V128_CONST(name, type)                     \
 | |
|     if (!(consts->name##_vec_zero = LLVMConstNull(type))) \
 | |
|         return false;                                     \
 | |
|     if (!(consts->name##_undef = LLVMGetUndef(type)))     \
 | |
|         return false;
 | |
| 
 | |
|     CREATE_V128_CONST(i8x16, V128_i8x16_TYPE)
 | |
|     CREATE_V128_CONST(i16x8, V128_i16x8_TYPE)
 | |
|     CREATE_V128_CONST(i32x4, V128_i32x4_TYPE)
 | |
|     CREATE_V128_CONST(i64x2, V128_i64x2_TYPE)
 | |
|     CREATE_V128_CONST(f32x4, V128_f32x4_TYPE)
 | |
|     CREATE_V128_CONST(f64x2, V128_f64x2_TYPE)
 | |
| #undef CREATE_V128_CONST
 | |
| 
 | |
| #define CREATE_VEC_ZERO_MASK(slot)                                       \
 | |
|     {                                                                    \
 | |
|         LLVMTypeRef type = LLVMVectorType(I32_TYPE, slot);               \
 | |
|         if (!type || !(consts->i32x##slot##_zero = LLVMConstNull(type))) \
 | |
|             return false;                                                \
 | |
|     }
 | |
| 
 | |
|     CREATE_VEC_ZERO_MASK(16)
 | |
|     CREATE_VEC_ZERO_MASK(8)
 | |
|     CREATE_VEC_ZERO_MASK(4)
 | |
|     CREATE_VEC_ZERO_MASK(2)
 | |
| #undef CREATE_VEC_ZERO_MASK
 | |
| 
 | |
|     if (!(consts->gc_ref_null =
 | |
|               LLVMConstNull(comp_ctx->basic_types.gc_ref_type)))
 | |
|         return false;
 | |
|     if (!(consts->i8_ptr_null =
 | |
|               LLVMConstNull(comp_ctx->basic_types.int8_ptr_type)))
 | |
|         return false;
 | |
| 
 | |
|     return true;
 | |
| }
 | |
| 
 | |
| typedef struct ArchItem {
 | |
|     char *arch;
 | |
|     bool support_eb;
 | |
| } ArchItem;
 | |
| 
 | |
| /* clang-format off */
 | |
| static ArchItem valid_archs[] = {
 | |
|     { "x86_64", false },
 | |
|     { "i386", false },
 | |
|     { "xtensa", false },
 | |
|     { "mips", true },
 | |
|     { "mipsel", false },
 | |
|     { "aarch64v8", false },
 | |
|     { "aarch64v8.1", false },
 | |
|     { "aarch64v8.2", false },
 | |
|     { "aarch64v8.3", false },
 | |
|     { "aarch64v8.4", false },
 | |
|     { "aarch64v8.5", false },
 | |
|     { "aarch64_bev8", false }, /* big endian */
 | |
|     { "aarch64_bev8.1", false },
 | |
|     { "aarch64_bev8.2", false },
 | |
|     { "aarch64_bev8.3", false },
 | |
|     { "aarch64_bev8.4", false },
 | |
|     { "aarch64_bev8.5", false },
 | |
|     { "armv4", true },
 | |
|     { "armv4t", true },
 | |
|     { "armv5t", true },
 | |
|     { "armv5te", true },
 | |
|     { "armv5tej", true },
 | |
|     { "armv6", true },
 | |
|     { "armv6kz", true },
 | |
|     { "armv6t2", true },
 | |
|     { "armv6k", true },
 | |
|     { "armv7", true },
 | |
|     { "armv6m", true },
 | |
|     { "armv6sm", true },
 | |
|     { "armv7em", true },
 | |
|     { "armv8a", true },
 | |
|     { "armv8r", true },
 | |
|     { "armv8m.base", true },
 | |
|     { "armv8m.main", true },
 | |
|     { "armv8.1m.main", true },
 | |
|     { "thumbv4", true },
 | |
|     { "thumbv4t", true },
 | |
|     { "thumbv5t", true },
 | |
|     { "thumbv5te", true },
 | |
|     { "thumbv5tej", true },
 | |
|     { "thumbv6", true },
 | |
|     { "thumbv6kz", true },
 | |
|     { "thumbv6t2", true },
 | |
|     { "thumbv6k", true },
 | |
|     { "thumbv7", true },
 | |
|     { "thumbv6m", true },
 | |
|     { "thumbv6sm", true },
 | |
|     { "thumbv7em", true },
 | |
|     { "thumbv8a", true },
 | |
|     { "thumbv8r", true },
 | |
|     { "thumbv8m.base", true },
 | |
|     { "thumbv8m.main", true },
 | |
|     { "thumbv8.1m.main", true },
 | |
|     { "riscv32", true },
 | |
|     { "riscv64", true },
 | |
|     { "arc", true }
 | |
| };
 | |
| 
 | |
| static const char *valid_abis[] = {
 | |
|     "gnu",
 | |
|     "eabi",
 | |
|     "eabihf",
 | |
|     "gnueabihf",
 | |
|     "msvc",
 | |
|     "ilp32",
 | |
|     "ilp32f",
 | |
|     "ilp32d",
 | |
|     "lp64",
 | |
|     "lp64f",
 | |
|     "lp64d"
 | |
| };
 | |
| /* clang-format on */
 | |
| 
 | |
| static void
 | |
| print_supported_targets()
 | |
| {
 | |
|     uint32 i;
 | |
|     const char *target_name;
 | |
| 
 | |
|     os_printf("Supported targets:\n");
 | |
|     /* over the list of all available targets */
 | |
|     for (LLVMTargetRef target = LLVMGetFirstTarget(); target != NULL;
 | |
|          target = LLVMGetNextTarget(target)) {
 | |
|         target_name = LLVMGetTargetName(target);
 | |
|         /* Skip mipsel, aarch64_be since prefix mips, aarch64 will cover them */
 | |
|         if (strcmp(target_name, "mipsel") == 0)
 | |
|             continue;
 | |
|         else if (strcmp(target_name, "aarch64_be") == 0)
 | |
|             continue;
 | |
| 
 | |
|         if (strcmp(target_name, "x86-64") == 0)
 | |
|             os_printf("  x86_64\n");
 | |
|         else if (strcmp(target_name, "x86") == 0)
 | |
|             os_printf("  i386\n");
 | |
|         else {
 | |
|             for (i = 0; i < sizeof(valid_archs) / sizeof(ArchItem); i++) {
 | |
|                 /* If target_name is prefix for valid_archs[i].arch */
 | |
|                 if ((strncmp(target_name, valid_archs[i].arch,
 | |
|                              strlen(target_name))
 | |
|                      == 0))
 | |
|                     os_printf("  %s\n", valid_archs[i].arch);
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| static void
 | |
| print_supported_abis()
 | |
| {
 | |
|     uint32 i;
 | |
|     os_printf("Supported ABI: ");
 | |
|     for (i = 0; i < sizeof(valid_abis) / sizeof(const char *); i++)
 | |
|         os_printf("%s ", valid_abis[i]);
 | |
|     os_printf("\n");
 | |
| }
 | |
| 
 | |
| static bool
 | |
| check_target_arch(const char *target_arch)
 | |
| {
 | |
|     uint32 i;
 | |
|     char *arch;
 | |
|     bool support_eb;
 | |
| 
 | |
|     for (i = 0; i < sizeof(valid_archs) / sizeof(ArchItem); i++) {
 | |
|         arch = valid_archs[i].arch;
 | |
|         support_eb = valid_archs[i].support_eb;
 | |
| 
 | |
|         if (!strncmp(target_arch, arch, strlen(arch))
 | |
|             && ((support_eb
 | |
|                  && (!strcmp(target_arch + strlen(arch), "eb")
 | |
|                      || !strcmp(target_arch + strlen(arch), "")))
 | |
|                 || (!support_eb && !strcmp(target_arch + strlen(arch), "")))) {
 | |
|             return true;
 | |
|         }
 | |
|     }
 | |
|     return false;
 | |
| }
 | |
| 
 | |
| static bool
 | |
| check_target_abi(const char *target_abi)
 | |
| {
 | |
|     uint32 i;
 | |
|     for (i = 0; i < sizeof(valid_abis) / sizeof(char *); i++) {
 | |
|         if (!strcmp(target_abi, valid_abis[i]))
 | |
|             return true;
 | |
|     }
 | |
|     return false;
 | |
| }
 | |
| 
 | |
| static void
 | |
| get_target_arch_from_triple(const char *triple, char *arch_buf, uint32 buf_size)
 | |
| {
 | |
|     uint32 i = 0;
 | |
|     while (*triple != '-' && *triple != '\0' && i < buf_size - 1)
 | |
|         arch_buf[i++] = *triple++;
 | |
|     /* Make sure buffer is long enough */
 | |
|     bh_assert(*triple == '-' || *triple == '\0');
 | |
| }
 | |
| 
 | |
| static bool
 | |
| is_baremetal_target(const char *target, const char *cpu, const char *abi)
 | |
| {
 | |
|     /* TODO: support more baremetal targets */
 | |
|     if (target) {
 | |
|         /* If target is thumbxxx, then it is baremetal target */
 | |
|         if (!strncmp(target, "thumb", strlen("thumb")))
 | |
|             return true;
 | |
|     }
 | |
|     return false;
 | |
| }
 | |
| 
 | |
| void
 | |
| aot_handle_llvm_errmsg(const char *string, LLVMErrorRef err)
 | |
| {
 | |
|     char *err_msg = LLVMGetErrorMessage(err);
 | |
|     aot_set_last_error_v("%s: %s", string, err_msg);
 | |
|     LLVMDisposeErrorMessage(err_msg);
 | |
| }
 | |
| 
 | |
| static bool
 | |
| create_target_machine_detect_host(AOTCompContext *comp_ctx)
 | |
| {
 | |
|     char *triple = NULL;
 | |
|     LLVMTargetRef target = NULL;
 | |
|     char *err_msg = NULL;
 | |
|     char *cpu = NULL;
 | |
|     char *features = NULL;
 | |
|     LLVMTargetMachineRef target_machine = NULL;
 | |
|     bool ret = false;
 | |
| 
 | |
|     triple = LLVMGetDefaultTargetTriple();
 | |
|     if (triple == NULL) {
 | |
|         aot_set_last_error("failed to get default target triple.");
 | |
|         goto fail;
 | |
|     }
 | |
| 
 | |
|     if (LLVMGetTargetFromTriple(triple, &target, &err_msg) != 0) {
 | |
|         aot_set_last_error_v("failed to get llvm target from triple %s.",
 | |
|                              err_msg);
 | |
|         LLVMDisposeMessage(err_msg);
 | |
|         goto fail;
 | |
|     }
 | |
| 
 | |
|     if (!LLVMTargetHasJIT(target)) {
 | |
|         aot_set_last_error("unsupported JIT on this platform.");
 | |
|         goto fail;
 | |
|     }
 | |
| 
 | |
|     cpu = LLVMGetHostCPUName();
 | |
|     if (cpu == NULL) {
 | |
|         aot_set_last_error("failed to get host cpu information.");
 | |
|         goto fail;
 | |
|     }
 | |
| 
 | |
|     features = LLVMGetHostCPUFeatures();
 | |
|     if (features == NULL) {
 | |
|         aot_set_last_error("failed to get host cpu features.");
 | |
|         goto fail;
 | |
|     }
 | |
| 
 | |
|     LOG_VERBOSE("LLVM ORCJIT detected CPU \"%s\", with features \"%s\"\n", cpu,
 | |
|                 features);
 | |
| 
 | |
|     /* create TargetMachine */
 | |
|     target_machine = LLVMCreateTargetMachine(
 | |
|         target, triple, cpu, features, LLVMCodeGenLevelDefault,
 | |
|         LLVMRelocDefault, LLVMCodeModelJITDefault);
 | |
|     if (!target_machine) {
 | |
|         aot_set_last_error("failed to create target machine.");
 | |
|         goto fail;
 | |
|     }
 | |
|     comp_ctx->target_machine = target_machine;
 | |
| 
 | |
|     /* Save target arch */
 | |
|     get_target_arch_from_triple(triple, comp_ctx->target_arch,
 | |
|                                 sizeof(comp_ctx->target_arch));
 | |
|     ret = true;
 | |
| 
 | |
| fail:
 | |
|     if (triple)
 | |
|         LLVMDisposeMessage(triple);
 | |
|     if (features)
 | |
|         LLVMDisposeMessage(features);
 | |
|     if (cpu)
 | |
|         LLVMDisposeMessage(cpu);
 | |
| 
 | |
|     return ret;
 | |
| }
 | |
| 
 | |
| static void
 | |
| jit_stack_size_callback(void *user_data, const char *name, size_t namelen,
 | |
|                         size_t stack_size)
 | |
| {
 | |
|     AOTCompContext *comp_ctx = user_data;
 | |
|     /*
 | |
|      * Note: the longest name we care is
 | |
|      * something like "aot_func_internal#4294967295".
 | |
|      */
 | |
|     char buf[64];
 | |
|     uint32 func_idx;
 | |
|     const AOTFuncContext *func_ctx;
 | |
|     bool musttail;
 | |
|     unsigned int stack_consumption_to_call_wrapped_func;
 | |
|     unsigned int call_size;
 | |
|     int ret;
 | |
| 
 | |
|     bh_assert(comp_ctx != NULL);
 | |
|     bh_assert(comp_ctx->jit_stack_sizes != NULL);
 | |
| 
 | |
|     if (namelen >= sizeof(buf)) {
 | |
|         LOG_DEBUG("too long name: %.*s", (int)namelen, name);
 | |
|         return;
 | |
|     }
 | |
|     /* ensure NUL termination */
 | |
|     bh_memcpy_s(buf, (uint32)sizeof(buf), name, (uint32)namelen);
 | |
|     buf[namelen] = 0;
 | |
| 
 | |
|     ret = sscanf(buf, AOT_FUNC_INTERNAL_PREFIX "%" SCNu32, &func_idx);
 | |
|     if (ret != 1) {
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     bh_assert(func_idx < comp_ctx->func_ctx_count);
 | |
|     func_ctx = comp_ctx->func_ctxes[func_idx];
 | |
|     call_size = func_ctx->stack_consumption_for_func_call;
 | |
|     musttail = aot_target_precheck_can_use_musttail(comp_ctx);
 | |
|     stack_consumption_to_call_wrapped_func =
 | |
|         musttail ? 0
 | |
|                  : aot_estimate_stack_usage_for_function_call(
 | |
|                      comp_ctx, func_ctx->aot_func->func_type);
 | |
|     LOG_VERBOSE("func %.*s stack %u + %zu + %u", (int)namelen, name,
 | |
|                 stack_consumption_to_call_wrapped_func, stack_size, call_size);
 | |
| 
 | |
|     /* Note: -1 == AOT_NEG_ONE from aot_create_stack_sizes */
 | |
|     bh_assert(comp_ctx->jit_stack_sizes[func_idx] == (uint32)-1);
 | |
|     comp_ctx->jit_stack_sizes[func_idx] = (uint32)stack_size + call_size;
 | |
| }
 | |
| 
 | |
| static bool
 | |
| orc_jit_create(AOTCompContext *comp_ctx)
 | |
| {
 | |
|     LLVMErrorRef err;
 | |
|     LLVMOrcLLLazyJITRef orc_jit = NULL;
 | |
|     LLVMOrcLLLazyJITBuilderRef builder = NULL;
 | |
|     LLVMOrcJITTargetMachineBuilderRef jtmb = NULL;
 | |
|     bool ret = false;
 | |
| 
 | |
|     builder = LLVMOrcCreateLLLazyJITBuilder();
 | |
|     if (builder == NULL) {
 | |
|         aot_set_last_error("failed to create jit builder.");
 | |
|         goto fail;
 | |
|     }
 | |
| 
 | |
|     if (comp_ctx->enable_stack_bound_check || comp_ctx->enable_stack_estimation)
 | |
|         LLVMOrcLLJITBuilderSetCompileFunctionCreatorWithStackSizesCallback(
 | |
|             builder, jit_stack_size_callback, comp_ctx);
 | |
| 
 | |
|     err = LLVMOrcJITTargetMachineBuilderDetectHost(&jtmb);
 | |
|     if (err != LLVMErrorSuccess) {
 | |
|         aot_handle_llvm_errmsg(
 | |
|             "quited to create LLVMOrcJITTargetMachineBuilderRef", err);
 | |
|         goto fail;
 | |
|     }
 | |
| 
 | |
|     LLVMOrcLLLazyJITBuilderSetNumCompileThreads(
 | |
|         builder, WASM_ORC_JIT_COMPILE_THREAD_NUM);
 | |
| 
 | |
|     /* Ownership transfer:
 | |
|        LLVMOrcJITTargetMachineBuilderRef -> LLVMOrcLLJITBuilderRef */
 | |
|     LLVMOrcLLLazyJITBuilderSetJITTargetMachineBuilder(builder, jtmb);
 | |
|     err = LLVMOrcCreateLLLazyJIT(&orc_jit, builder);
 | |
|     if (err != LLVMErrorSuccess) {
 | |
|         aot_handle_llvm_errmsg("quited to create llvm lazy orcjit instance",
 | |
|                                err);
 | |
|         goto fail;
 | |
|     }
 | |
|     /* Ownership transfer: LLVMOrcLLJITBuilderRef -> LLVMOrcLLJITRef */
 | |
|     builder = NULL;
 | |
| 
 | |
| #if WASM_ENABLE_LINUX_PERF != 0
 | |
|     if (wasm_runtime_get_linux_perf()) {
 | |
|         LOG_DEBUG("Enable linux perf support in JIT");
 | |
|         LLVMOrcObjectLayerRef obj_linking_layer =
 | |
|             (LLVMOrcObjectLayerRef)LLVMOrcLLLazyJITGetObjLinkingLayer(orc_jit);
 | |
|         LLVMOrcRTDyldObjectLinkingLayerRegisterJITEventListener(
 | |
|             obj_linking_layer, LLVMCreatePerfJITEventListener());
 | |
|     }
 | |
| #endif
 | |
| 
 | |
|     /* Ownership transfer: local -> AOTCompContext */
 | |
|     comp_ctx->orc_jit = orc_jit;
 | |
|     orc_jit = NULL;
 | |
|     ret = true;
 | |
| 
 | |
| fail:
 | |
|     if (builder)
 | |
|         LLVMOrcDisposeLLLazyJITBuilder(builder);
 | |
| 
 | |
|     if (orc_jit)
 | |
|         LLVMOrcDisposeLLLazyJIT(orc_jit);
 | |
|     return ret;
 | |
| }
 | |
| 
 | |
| bool
 | |
| aot_compiler_init(void)
 | |
| {
 | |
|     /* Initialize LLVM environment */
 | |
| #if LLVM_VERSION_MAJOR < 17
 | |
|     LLVMInitializeCore(LLVMGetGlobalPassRegistry());
 | |
| #endif
 | |
| 
 | |
| /* fuzzing only use host targets for simple */
 | |
| #if WASM_ENABLE_WAMR_COMPILER != 0 && WASM_ENABLE_FUZZ_TEST == 0
 | |
|     /* Init environment of all targets for AOT compiler */
 | |
|     LLVMInitializeAllTargetInfos();
 | |
|     LLVMInitializeAllTargets();
 | |
|     LLVMInitializeAllTargetMCs();
 | |
|     LLVMInitializeAllAsmPrinters();
 | |
| #else
 | |
|     /* Init environment of native for JIT compiler */
 | |
|     LLVMInitializeNativeTarget();
 | |
|     LLVMInitializeNativeTarget();
 | |
|     LLVMInitializeNativeAsmPrinter();
 | |
| #endif
 | |
| 
 | |
|     return true;
 | |
| }
 | |
| 
 | |
| void
 | |
| aot_compiler_destroy(void)
 | |
| {
 | |
|     LLVMShutdown();
 | |
| }
 | |
| 
 | |
| AOTCompContext *
 | |
| aot_create_comp_context(const AOTCompData *comp_data, aot_comp_option_t option)
 | |
| {
 | |
|     AOTCompContext *comp_ctx, *ret = NULL;
 | |
|     LLVMTargetRef target;
 | |
|     char *triple = NULL, *triple_norm, *arch, *abi;
 | |
|     char *cpu = NULL, *features, buf[128];
 | |
|     char *triple_norm_new = NULL, *cpu_new = NULL;
 | |
|     char *err = NULL, *fp_round = "round.tonearest",
 | |
|          *fp_exce = "fpexcept.strict";
 | |
|     char triple_buf[128] = { 0 }, features_buf[128] = { 0 };
 | |
|     uint32 opt_level, size_level, i;
 | |
|     LLVMCodeModel code_model;
 | |
|     LLVMTargetDataRef target_data_ref;
 | |
| 
 | |
|     /* Allocate memory */
 | |
|     if (!(comp_ctx = wasm_runtime_malloc(sizeof(AOTCompContext)))) {
 | |
|         aot_set_last_error("allocate memory failed.");
 | |
|         return NULL;
 | |
|     }
 | |
| 
 | |
|     memset(comp_ctx, 0, sizeof(AOTCompContext));
 | |
|     comp_ctx->comp_data = comp_data;
 | |
| 
 | |
|     /* Create LLVM context, module and builder */
 | |
|     comp_ctx->orc_thread_safe_context = LLVMOrcCreateNewThreadSafeContext();
 | |
|     if (!comp_ctx->orc_thread_safe_context) {
 | |
|         aot_set_last_error("create LLVM ThreadSafeContext failed.");
 | |
|         goto fail;
 | |
|     }
 | |
| 
 | |
|     /* Get a reference to the underlying LLVMContext, note:
 | |
|          different from non LAZY JIT mode, no need to dispose this context,
 | |
|          if will be disposed when the thread safe context is disposed */
 | |
|     if (!(comp_ctx->context = LLVMOrcThreadSafeContextGetContext(
 | |
|               comp_ctx->orc_thread_safe_context))) {
 | |
|         aot_set_last_error("get context from LLVM ThreadSafeContext failed.");
 | |
|         goto fail;
 | |
|     }
 | |
| 
 | |
|     if (!(comp_ctx->builder = LLVMCreateBuilderInContext(comp_ctx->context))) {
 | |
|         aot_set_last_error("create LLVM builder failed.");
 | |
|         goto fail;
 | |
|     }
 | |
| 
 | |
|     /* Create LLVM module for each jit function, note:
 | |
|        different from non ORC JIT mode, no need to dispose it,
 | |
|        it will be disposed when the thread safe context is disposed */
 | |
|     if (!(comp_ctx->module = LLVMModuleCreateWithNameInContext(
 | |
|               "WASM Module", comp_ctx->context))) {
 | |
|         aot_set_last_error("create LLVM module failed.");
 | |
|         goto fail;
 | |
|     }
 | |
| #if LLVM_VERSION_MAJOR >= 19
 | |
|     LLVMSetIsNewDbgInfoFormat(comp_ctx->module, true);
 | |
| #endif
 | |
| 
 | |
| #if WASM_ENABLE_LINUX_PERF != 0
 | |
|     if (wasm_runtime_get_linux_perf()) {
 | |
|         /* FramePointerKind.All */
 | |
|         LLVMMetadataRef val =
 | |
|             LLVMValueAsMetadata(LLVMConstInt(LLVMInt32Type(), 2, false));
 | |
|         const char *key = "frame-pointer";
 | |
|         LLVMAddModuleFlag(comp_ctx->module, LLVMModuleFlagBehaviorWarning, key,
 | |
|                           strlen(key), val);
 | |
| 
 | |
|         comp_ctx->emit_frame_pointer = true;
 | |
|     }
 | |
| #endif
 | |
| 
 | |
|     if (BH_LIST_ERROR == bh_list_init(&comp_ctx->native_symbols)) {
 | |
|         goto fail;
 | |
|     }
 | |
| 
 | |
| #if WASM_ENABLE_DEBUG_AOT != 0
 | |
|     if (!(comp_ctx->debug_builder = LLVMCreateDIBuilder(comp_ctx->module))) {
 | |
|         aot_set_last_error("create LLVM Debug Infor builder failed.");
 | |
|         goto fail;
 | |
|     }
 | |
| 
 | |
|     LLVMAddModuleFlag(
 | |
|         comp_ctx->module, LLVMModuleFlagBehaviorWarning, "Debug Info Version",
 | |
|         strlen("Debug Info Version"),
 | |
|         LLVMValueAsMetadata(LLVMConstInt(LLVMInt32Type(), 3, false)));
 | |
| 
 | |
|     comp_ctx->debug_file = dwarf_gen_file_info(comp_ctx);
 | |
|     if (!comp_ctx->debug_file) {
 | |
|         aot_set_last_error("dwarf generate file info failed");
 | |
|         goto fail;
 | |
|     }
 | |
|     comp_ctx->debug_comp_unit = dwarf_gen_comp_unit_info(comp_ctx);
 | |
|     if (!comp_ctx->debug_comp_unit) {
 | |
|         aot_set_last_error("dwarf generate compile unit info failed");
 | |
|         goto fail;
 | |
|     }
 | |
| #endif
 | |
| 
 | |
|     if (option->enable_bulk_memory)
 | |
|         comp_ctx->enable_bulk_memory = true;
 | |
| 
 | |
|     if (option->enable_thread_mgr)
 | |
|         comp_ctx->enable_thread_mgr = true;
 | |
| 
 | |
|     if (option->enable_tail_call)
 | |
|         comp_ctx->enable_tail_call = true;
 | |
| 
 | |
|     if (option->enable_ref_types)
 | |
|         comp_ctx->enable_ref_types = true;
 | |
| 
 | |
|     comp_ctx->aux_stack_frame_type = option->aux_stack_frame_type;
 | |
|     comp_ctx->call_stack_features = option->call_stack_features;
 | |
| 
 | |
|     if (option->enable_perf_profiling)
 | |
|         comp_ctx->enable_perf_profiling = true;
 | |
| 
 | |
|     if (option->enable_memory_profiling)
 | |
|         comp_ctx->enable_memory_profiling = true;
 | |
| 
 | |
|     if (option->enable_aux_stack_check)
 | |
|         comp_ctx->enable_aux_stack_check = true;
 | |
| 
 | |
|     if (option->is_indirect_mode) {
 | |
|         comp_ctx->is_indirect_mode = true;
 | |
|         /* avoid LUT relocations ("switch-table") */
 | |
|         comp_ctx->disable_llvm_jump_tables = true;
 | |
|     }
 | |
| 
 | |
|     if (option->disable_llvm_intrinsics)
 | |
|         comp_ctx->disable_llvm_intrinsics = true;
 | |
| 
 | |
|     if (option->disable_llvm_jump_tables)
 | |
|         comp_ctx->disable_llvm_jump_tables = true;
 | |
| 
 | |
|     if (option->disable_llvm_lto)
 | |
|         comp_ctx->disable_llvm_lto = true;
 | |
| 
 | |
|     if (option->enable_llvm_pgo)
 | |
|         comp_ctx->enable_llvm_pgo = true;
 | |
| 
 | |
|     if (option->use_prof_file)
 | |
|         comp_ctx->use_prof_file = option->use_prof_file;
 | |
| 
 | |
|     if (option->enable_stack_estimation)
 | |
|         comp_ctx->enable_stack_estimation = true;
 | |
| 
 | |
|     if (option->quick_invoke_c_api_import)
 | |
|         comp_ctx->quick_invoke_c_api_import = true;
 | |
| 
 | |
|     if (option->llvm_passes)
 | |
|         comp_ctx->llvm_passes = option->llvm_passes;
 | |
| 
 | |
|     if (option->builtin_intrinsics)
 | |
|         comp_ctx->builtin_intrinsics = option->builtin_intrinsics;
 | |
| 
 | |
|     if (option->enable_gc)
 | |
|         comp_ctx->enable_gc = true;
 | |
| 
 | |
|     if (option->enable_shared_heap)
 | |
|         comp_ctx->enable_shared_heap = true;
 | |
| 
 | |
|     if (option->enable_shared_chain)
 | |
|         comp_ctx->enable_shared_chain = true;
 | |
| 
 | |
|     if (option->enable_extended_const)
 | |
|         comp_ctx->enable_extended_const = true;
 | |
| 
 | |
|     comp_ctx->opt_level = option->opt_level;
 | |
|     comp_ctx->size_level = option->size_level;
 | |
| 
 | |
|     comp_ctx->custom_sections_wp = option->custom_sections;
 | |
|     comp_ctx->custom_sections_count = option->custom_sections_count;
 | |
| 
 | |
|     if (option->is_jit_mode) {
 | |
|         comp_ctx->is_jit_mode = true;
 | |
| 
 | |
| #ifndef OS_ENABLE_HW_BOUND_CHECK
 | |
|         comp_ctx->enable_bound_check = true;
 | |
|         /* Always enable stack boundary check if `bounds-checks`
 | |
|            is enabled */
 | |
|         comp_ctx->enable_stack_bound_check = true;
 | |
| #else
 | |
|         comp_ctx->enable_bound_check = false;
 | |
|         /* When `bounds-checks` is disabled, we set stack boundary
 | |
|            check status according to the compilation option */
 | |
| #if WASM_DISABLE_STACK_HW_BOUND_CHECK != 0
 | |
|         /* Native stack overflow check with hardware trap is disabled,
 | |
|            we need to enable the check by LLVM JITed/AOTed code */
 | |
|         comp_ctx->enable_stack_bound_check = true;
 | |
| #else
 | |
|         /* Native stack overflow check with hardware trap is enabled,
 | |
|            no need to enable the check by LLVM JITed/AOTed code */
 | |
|         comp_ctx->enable_stack_bound_check = false;
 | |
| #endif
 | |
| #endif
 | |
| 
 | |
|         /* Create TargetMachine */
 | |
|         if (!create_target_machine_detect_host(comp_ctx))
 | |
|             goto fail;
 | |
| 
 | |
|         /* Create LLJIT Instance */
 | |
|         if (!orc_jit_create(comp_ctx))
 | |
|             goto fail;
 | |
|     }
 | |
|     else {
 | |
|         /* Create LLVM target machine */
 | |
|         if (!option->target_arch || !strstr(option->target_arch, "-")) {
 | |
|             /* Retrieve the target triple based on user input */
 | |
|             triple = NULL;
 | |
|             arch = option->target_arch;
 | |
|             abi = option->target_abi;
 | |
|             cpu = option->target_cpu;
 | |
|             features = option->cpu_features;
 | |
|         }
 | |
|         else {
 | |
|             /* Form a target triple */
 | |
|             triple = option->target_arch;
 | |
|             arch = NULL;
 | |
|             abi = NULL;
 | |
|             cpu = NULL;
 | |
|             features = NULL;
 | |
|         }
 | |
| 
 | |
|         opt_level = option->opt_level;
 | |
|         size_level = option->size_level;
 | |
| 
 | |
|         /* verify external llc compiler */
 | |
|         comp_ctx->external_llc_compiler = getenv("WAMRC_LLC_COMPILER");
 | |
|         if (comp_ctx->external_llc_compiler) {
 | |
|             if (access(comp_ctx->external_llc_compiler, X_OK) != 0) {
 | |
|                 LOG_WARNING("WAMRC_LLC_COMPILER [%s] not found, fallback to "
 | |
|                             "default pipeline",
 | |
|                             comp_ctx->external_llc_compiler);
 | |
|                 comp_ctx->external_llc_compiler = NULL;
 | |
|             }
 | |
|             else {
 | |
|                 comp_ctx->llc_compiler_flags = getenv("WAMRC_LLC_FLAGS");
 | |
|                 LOG_VERBOSE("Using external LLC compiler [%s]",
 | |
|                             comp_ctx->external_llc_compiler);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         /* verify external asm compiler */
 | |
|         if (!comp_ctx->external_llc_compiler) {
 | |
|             comp_ctx->external_asm_compiler = getenv("WAMRC_ASM_COMPILER");
 | |
|             if (comp_ctx->external_asm_compiler) {
 | |
|                 if (access(comp_ctx->external_asm_compiler, X_OK) != 0) {
 | |
|                     LOG_WARNING(
 | |
|                         "WAMRC_ASM_COMPILER [%s] not found, fallback to "
 | |
|                         "default pipeline",
 | |
|                         comp_ctx->external_asm_compiler);
 | |
|                     comp_ctx->external_asm_compiler = NULL;
 | |
|                 }
 | |
|                 else {
 | |
|                     comp_ctx->asm_compiler_flags = getenv("WAMRC_ASM_FLAGS");
 | |
|                     LOG_VERBOSE("Using external ASM compiler [%s]",
 | |
|                                 comp_ctx->external_asm_compiler);
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         if (arch) {
 | |
|             /* Add default sub-arch if not specified */
 | |
|             if (!strcmp(arch, "arm"))
 | |
|                 arch = "armv4";
 | |
|             else if (!strcmp(arch, "armeb"))
 | |
|                 arch = "armv4eb";
 | |
|             else if (!strcmp(arch, "thumb"))
 | |
|                 arch = "thumbv4t";
 | |
|             else if (!strcmp(arch, "thumbeb"))
 | |
|                 arch = "thumbv4teb";
 | |
|             else if (!strcmp(arch, "aarch64"))
 | |
|                 arch = "aarch64v8";
 | |
|             else if (!strcmp(arch, "aarch64_be"))
 | |
|                 arch = "aarch64_bev8";
 | |
|         }
 | |
| 
 | |
|         /* Check target arch */
 | |
|         if (arch && !check_target_arch(arch)) {
 | |
|             if (!strcmp(arch, "help"))
 | |
|                 print_supported_targets();
 | |
|             else
 | |
|                 aot_set_last_error(
 | |
|                     "Invalid target. "
 | |
|                     "Use --target=help to list all supported targets");
 | |
|             goto fail;
 | |
|         }
 | |
| 
 | |
|         /* Check target ABI */
 | |
|         if (abi && !check_target_abi(abi)) {
 | |
|             if (!strcmp(abi, "help"))
 | |
|                 print_supported_abis();
 | |
|             else
 | |
|                 aot_set_last_error(
 | |
|                     "Invalid target ABI. "
 | |
|                     "Use --target-abi=help to list all supported ABI");
 | |
|             goto fail;
 | |
|         }
 | |
| 
 | |
|         /* Set default abi for riscv target */
 | |
|         if (arch && !strncmp(arch, "riscv", 5) && !abi) {
 | |
|             if (!strcmp(arch, "riscv64"))
 | |
|                 abi = "lp64d";
 | |
|             else
 | |
|                 abi = "ilp32d";
 | |
|         }
 | |
| 
 | |
| #if defined(__APPLE__) || defined(__MACH__)
 | |
|         if (!abi) {
 | |
|             /* On MacOS platform, set abi to "gnu" to avoid generating
 | |
|                object file of Mach-O binary format which is unsupported */
 | |
|             abi = "gnu";
 | |
|             if (!arch && !cpu && !features) {
 | |
|                 /* Get CPU name of the host machine to avoid checking
 | |
|                    SIMD capability failed */
 | |
|                 if (!(cpu = cpu_new = LLVMGetHostCPUName())) {
 | |
|                     aot_set_last_error("llvm get host cpu name failed.");
 | |
|                     goto fail;
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| #endif
 | |
| 
 | |
|         if (abi) {
 | |
|             /* Construct target triple: <arch>-<vendor>-<sys>-<abi> */
 | |
|             const char *vendor_sys;
 | |
|             char *arch1 = arch, default_arch[32] = { 0 };
 | |
| 
 | |
|             if (!arch1) {
 | |
|                 char *default_triple = LLVMGetDefaultTargetTriple();
 | |
| 
 | |
|                 if (!default_triple) {
 | |
|                     aot_set_last_error(
 | |
|                         "llvm get default target triple failed.");
 | |
|                     goto fail;
 | |
|                 }
 | |
| 
 | |
|                 vendor_sys = strstr(default_triple, "-");
 | |
|                 bh_assert(vendor_sys);
 | |
|                 bh_memcpy_s(default_arch, sizeof(default_arch), default_triple,
 | |
|                             (uint32)(vendor_sys - default_triple));
 | |
|                 /**
 | |
|                  * On Mac M[1-9]+ LLVM will report arm64 as the
 | |
|                  * architecture, for the purposes of wamr this is the
 | |
|                  * same as aarch64v8 so we'll normalize it here.
 | |
|                  */
 | |
|                 if (!strcmp(default_arch, "arm64")) {
 | |
|                     bh_strcpy_s(default_arch, sizeof(default_arch),
 | |
|                                 "aarch64v8");
 | |
|                 }
 | |
|                 arch1 = default_arch;
 | |
| 
 | |
|                 LLVMDisposeMessage(default_triple);
 | |
|             }
 | |
| 
 | |
|             /**
 | |
|              * Set <vendor>-<sys> according to abi to generate the object file
 | |
|              * with the correct file format which might be different from the
 | |
|              * default object file format of the host, e.g., generating AOT file
 | |
|              * for Windows/MacOS under Linux host, or generating AOT file for
 | |
|              * Linux/MacOS under Windows host.
 | |
|              */
 | |
| 
 | |
|             if (!strcmp(abi, "msvc")) {
 | |
|                 if (!strcmp(arch1, "i386"))
 | |
|                     vendor_sys = "-pc-win32-";
 | |
|                 else
 | |
|                     vendor_sys = "-pc-windows-";
 | |
|             }
 | |
|             else {
 | |
|                 if (is_baremetal_target(arch, cpu, abi))
 | |
|                     vendor_sys = "-unknown-none-";
 | |
|                 else
 | |
|                     vendor_sys = "-pc-linux-";
 | |
|             }
 | |
| 
 | |
|             bh_assert(strlen(arch1) + strlen(vendor_sys) + strlen(abi)
 | |
|                       < sizeof(triple_buf));
 | |
|             bh_memcpy_s(triple_buf, (uint32)sizeof(triple_buf), arch1,
 | |
|                         (uint32)strlen(arch1));
 | |
|             bh_memcpy_s(triple_buf + strlen(arch1),
 | |
|                         (uint32)(sizeof(triple_buf) - strlen(arch1)),
 | |
|                         vendor_sys, (uint32)strlen(vendor_sys));
 | |
|             bh_memcpy_s(triple_buf + strlen(arch1) + strlen(vendor_sys),
 | |
|                         (uint32)(sizeof(triple_buf) - strlen(arch1)
 | |
|                                  - strlen(vendor_sys)),
 | |
|                         abi, (uint32)strlen(abi));
 | |
|             triple = triple_buf;
 | |
|         }
 | |
|         else if (arch) {
 | |
|             /* Construct target triple: <arch>-<vendor>-<sys>-<abi> */
 | |
|             const char *vendor_sys;
 | |
|             char *default_triple = LLVMGetDefaultTargetTriple();
 | |
| 
 | |
|             if (!default_triple) {
 | |
|                 aot_set_last_error("llvm get default target triple failed.");
 | |
|                 goto fail;
 | |
|             }
 | |
| 
 | |
|             if (strstr(default_triple, "windows")) {
 | |
|                 vendor_sys = "-pc-windows-";
 | |
|                 if (!abi)
 | |
|                     abi = "msvc";
 | |
|             }
 | |
|             else if (strstr(default_triple, "win32")) {
 | |
|                 vendor_sys = "-pc-win32-";
 | |
|                 if (!abi)
 | |
|                     abi = "msvc";
 | |
|             }
 | |
|             else if (is_baremetal_target(arch, cpu, abi)) {
 | |
|                 vendor_sys = "-unknown-none-";
 | |
|                 if (!abi)
 | |
|                     abi = "gnu";
 | |
|             }
 | |
|             else {
 | |
|                 vendor_sys = "-pc-linux-";
 | |
|                 if (!abi)
 | |
|                     abi = "gnu";
 | |
|             }
 | |
| 
 | |
|             LLVMDisposeMessage(default_triple);
 | |
| 
 | |
|             bh_assert(strlen(arch) + strlen(vendor_sys) + strlen(abi)
 | |
|                       < sizeof(triple_buf));
 | |
|             bh_memcpy_s(triple_buf, (uint32)sizeof(triple_buf), arch,
 | |
|                         (uint32)strlen(arch));
 | |
|             bh_memcpy_s(triple_buf + strlen(arch),
 | |
|                         (uint32)(sizeof(triple_buf) - strlen(arch)), vendor_sys,
 | |
|                         (uint32)strlen(vendor_sys));
 | |
|             bh_memcpy_s(triple_buf + strlen(arch) + strlen(vendor_sys),
 | |
|                         (uint32)(sizeof(triple_buf) - strlen(arch)
 | |
|                                  - strlen(vendor_sys)),
 | |
|                         abi, (uint32)strlen(abi));
 | |
|             triple = triple_buf;
 | |
|         }
 | |
| 
 | |
|         if (!cpu && features) {
 | |
|             aot_set_last_error("cpu isn't specified for cpu features.");
 | |
|             goto fail;
 | |
|         }
 | |
| 
 | |
|         if (!triple && !cpu) {
 | |
|             /* Get a triple for the host machine */
 | |
|             if (!(triple_norm = triple_norm_new =
 | |
|                       LLVMGetDefaultTargetTriple())) {
 | |
|                 aot_set_last_error("llvm get default target triple failed.");
 | |
|                 goto fail;
 | |
|             }
 | |
|             /* Get CPU name of the host machine */
 | |
|             if (!(cpu = cpu_new = LLVMGetHostCPUName())) {
 | |
|                 aot_set_last_error("llvm get host cpu name failed.");
 | |
|                 goto fail;
 | |
|             }
 | |
|         }
 | |
|         else if (triple) {
 | |
|             /* Normalize a target triple */
 | |
|             if (!(triple_norm = triple_norm_new =
 | |
|                       LLVMNormalizeTargetTriple(triple))) {
 | |
|                 snprintf(buf, sizeof(buf),
 | |
|                          "llvm normlalize target triple (%s) failed.", triple);
 | |
|                 aot_set_last_error(buf);
 | |
|                 goto fail;
 | |
|             }
 | |
|             LOG_VERBOSE("triple: %s => normailized: %s", triple, triple_norm);
 | |
|             if (!cpu)
 | |
|                 cpu = "";
 | |
|         }
 | |
|         else {
 | |
|             /* triple is NULL, cpu isn't NULL */
 | |
|             snprintf(buf, sizeof(buf), "target isn't specified for cpu %s.",
 | |
|                      cpu);
 | |
|             aot_set_last_error(buf);
 | |
|             goto fail;
 | |
|         }
 | |
| 
 | |
|         /* Add module flag and cpu feature for riscv target */
 | |
|         if (arch && !strncmp(arch, "riscv", 5)) {
 | |
|             LLVMMetadataRef meta_target_abi;
 | |
| 
 | |
|             if (!(meta_target_abi = LLVMMDStringInContext2(comp_ctx->context,
 | |
|                                                            abi, strlen(abi)))) {
 | |
|                 aot_set_last_error("create metadata string failed.");
 | |
|                 goto fail;
 | |
|             }
 | |
|             LLVMAddModuleFlag(comp_ctx->module, LLVMModuleFlagBehaviorError,
 | |
|                               "target-abi", strlen("target-abi"),
 | |
|                               meta_target_abi);
 | |
| 
 | |
|             if (!strcmp(abi, "lp64d") || !strcmp(abi, "ilp32d")) {
 | |
|                 if (features && !strstr(features, "+d")) {
 | |
|                     snprintf(features_buf, sizeof(features_buf), "%s%s",
 | |
|                              features, ",+d");
 | |
|                     features = features_buf;
 | |
|                 }
 | |
|                 else if (!features) {
 | |
|                     features = "+d";
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         if (!features)
 | |
|             features = "";
 | |
| 
 | |
|         /* Get target with triple, note that LLVMGetTargetFromTriple()
 | |
|            return 0 when success, but not true. */
 | |
|         if (LLVMGetTargetFromTriple(triple_norm, &target, &err) != 0) {
 | |
|             if (err) {
 | |
|                 LLVMDisposeMessage(err);
 | |
|                 err = NULL;
 | |
|             }
 | |
|             snprintf(buf, sizeof(buf),
 | |
|                      "llvm get target from triple (%s) failed", triple_norm);
 | |
|             aot_set_last_error(buf);
 | |
|             goto fail;
 | |
|         }
 | |
| 
 | |
|         /* Save target arch */
 | |
|         get_target_arch_from_triple(triple_norm, comp_ctx->target_arch,
 | |
|                                     sizeof(comp_ctx->target_arch));
 | |
| 
 | |
|         if (option->bounds_checks == 1 || option->bounds_checks == 0) {
 | |
|             /* Set by the user */
 | |
|             comp_ctx->enable_bound_check =
 | |
|                 (option->bounds_checks == 1) ? true : false;
 | |
|         }
 | |
|         else {
 | |
|             /* Unset by the user, use the default value */
 | |
|             if (strstr(comp_ctx->target_arch, "64")
 | |
|                 && !option->is_sgx_platform) {
 | |
|                 comp_ctx->enable_bound_check = false;
 | |
|             }
 | |
|             else {
 | |
|                 comp_ctx->enable_bound_check = true;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         if (option->stack_bounds_checks == 1
 | |
|             || option->stack_bounds_checks == 0) {
 | |
|             /* Set by the user */
 | |
|             comp_ctx->enable_stack_bound_check =
 | |
|                 (option->stack_bounds_checks == 1) ? true : false;
 | |
|         }
 | |
|         else {
 | |
|             /* Unset by the user, use the default value, it will be the same
 | |
|              * value as the bound check */
 | |
|             comp_ctx->enable_stack_bound_check = comp_ctx->enable_bound_check;
 | |
|         }
 | |
| 
 | |
|         if ((comp_ctx->enable_stack_bound_check
 | |
|              || comp_ctx->enable_stack_estimation)
 | |
|             && option->stack_usage_file == NULL) {
 | |
|             if (!aot_generate_tempfile_name(
 | |
|                     "wamrc-su", "su", comp_ctx->stack_usage_temp_file,
 | |
|                     sizeof(comp_ctx->stack_usage_temp_file)))
 | |
|                 goto fail;
 | |
|             comp_ctx->stack_usage_file = comp_ctx->stack_usage_temp_file;
 | |
|         }
 | |
|         else {
 | |
|             comp_ctx->stack_usage_file = option->stack_usage_file;
 | |
|         }
 | |
| 
 | |
|         os_printf("Create AoT compiler with:\n");
 | |
|         os_printf("  target:        %s\n", comp_ctx->target_arch);
 | |
|         os_printf("  target cpu:    %s\n", cpu);
 | |
|         os_printf("  target triple: %s\n", triple_norm);
 | |
|         os_printf("  cpu features:  %s\n", features);
 | |
|         os_printf("  opt level:     %d\n", opt_level);
 | |
|         os_printf("  size level:    %d\n", size_level);
 | |
|         switch (option->output_format) {
 | |
|             case AOT_LLVMIR_UNOPT_FILE:
 | |
|                 os_printf("  output format: unoptimized LLVM IR\n");
 | |
|                 break;
 | |
|             case AOT_LLVMIR_OPT_FILE:
 | |
|                 os_printf("  output format: optimized LLVM IR\n");
 | |
|                 break;
 | |
|             case AOT_FORMAT_FILE:
 | |
|                 os_printf("  output format: AoT file\n");
 | |
|                 break;
 | |
|             case AOT_OBJECT_FILE:
 | |
|                 os_printf("  output format: native object file\n");
 | |
|                 break;
 | |
|         }
 | |
| 
 | |
|         LLVMSetTarget(comp_ctx->module, triple_norm);
 | |
| 
 | |
|         if (!LLVMTargetHasTargetMachine(target)) {
 | |
|             snprintf(buf, sizeof(buf),
 | |
|                      "no target machine for this target (%s).", triple_norm);
 | |
|             aot_set_last_error(buf);
 | |
|             goto fail;
 | |
|         }
 | |
| 
 | |
|         /* Report error if target isn't arc and hasn't asm backend.
 | |
|            For arc target, as it cannot emit to memory buffer of elf file
 | |
|            currently, we let it emit to assembly file instead, and then call
 | |
|            arc-gcc to compile
 | |
|            asm file to elf file, and read elf file to memory buffer. */
 | |
|         if (strncmp(comp_ctx->target_arch, "arc", 3)
 | |
|             && !LLVMTargetHasAsmBackend(target)) {
 | |
|             snprintf(buf, sizeof(buf), "no asm backend for this target (%s).",
 | |
|                      LLVMGetTargetName(target));
 | |
|             aot_set_last_error(buf);
 | |
|             goto fail;
 | |
|         }
 | |
| 
 | |
|         /* Set code model */
 | |
|         if (size_level == 0)
 | |
|             code_model = LLVMCodeModelLarge;
 | |
|         else if (size_level == 1)
 | |
|             code_model = LLVMCodeModelMedium;
 | |
|         else if (size_level == 2)
 | |
|             code_model = LLVMCodeModelKernel;
 | |
|         else
 | |
|             code_model = LLVMCodeModelSmall;
 | |
| 
 | |
|         /* Create the target machine */
 | |
|         if (!(comp_ctx->target_machine = LLVMCreateTargetMachineWithOpts(
 | |
|                   target, triple_norm, cpu, features, opt_level,
 | |
|                   LLVMRelocStatic, code_model, false,
 | |
|                   comp_ctx->stack_usage_file))) {
 | |
|             aot_set_last_error("create LLVM target machine failed.");
 | |
|             goto fail;
 | |
|         }
 | |
| 
 | |
|         /* If only to create target machine for querying information, early stop
 | |
|          */
 | |
|         if ((arch && !strcmp(arch, "help")) || (abi && !strcmp(abi, "help"))
 | |
|             || (cpu && !strcmp(cpu, "help"))
 | |
|             || (features && !strcmp(features, "+help"))) {
 | |
|             LOG_DEBUG(
 | |
|                 "create LLVM target machine only for printing help info.");
 | |
|             goto fail;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     triple = LLVMGetTargetMachineTriple(comp_ctx->target_machine);
 | |
|     if (!triple) {
 | |
|         aot_set_last_error("get target machine triple failed.");
 | |
|         goto fail;
 | |
|     }
 | |
|     if (strstr(triple, "linux") && !strcmp(comp_ctx->target_arch, "x86_64")) {
 | |
|         if (option->segue_flags) {
 | |
|             if (option->segue_flags & (1 << 0))
 | |
|                 comp_ctx->enable_segue_i32_load = true;
 | |
|             if (option->segue_flags & (1 << 1))
 | |
|                 comp_ctx->enable_segue_i64_load = true;
 | |
|             if (option->segue_flags & (1 << 2))
 | |
|                 comp_ctx->enable_segue_f32_load = true;
 | |
|             if (option->segue_flags & (1 << 3))
 | |
|                 comp_ctx->enable_segue_f64_load = true;
 | |
|             if (option->segue_flags & (1 << 4))
 | |
|                 comp_ctx->enable_segue_v128_load = true;
 | |
|             if (option->segue_flags & (1 << 8))
 | |
|                 comp_ctx->enable_segue_i32_store = true;
 | |
|             if (option->segue_flags & (1 << 9))
 | |
|                 comp_ctx->enable_segue_i64_store = true;
 | |
|             if (option->segue_flags & (1 << 10))
 | |
|                 comp_ctx->enable_segue_f32_store = true;
 | |
|             if (option->segue_flags & (1 << 11))
 | |
|                 comp_ctx->enable_segue_f64_store = true;
 | |
|             if (option->segue_flags & (1 << 12))
 | |
|                 comp_ctx->enable_segue_v128_store = true;
 | |
|         }
 | |
|     }
 | |
|     LLVMDisposeMessage(triple);
 | |
| 
 | |
| #if WASM_ENABLE_WAMR_COMPILER != 0
 | |
|     WASMModule *wasm_module = (WASMModule *)comp_data->wasm_module;
 | |
|     bool is_memory64 = false;
 | |
| 
 | |
|     /* TODO: multi-memories for now assuming the memory64 flag of a memory is
 | |
|      * consistent across multi-memories */
 | |
|     if (wasm_module->import_memory_count > 0)
 | |
|         is_memory64 = !!(wasm_module->import_memories[0].u.memory.mem_type.flags
 | |
|                          & MEMORY64_FLAG);
 | |
|     else if (wasm_module->memory_count > 0)
 | |
|         is_memory64 = !!(wasm_module->memories[0].flags & MEMORY64_FLAG);
 | |
| 
 | |
|     if (!(option->bounds_checks == 1 || option->bounds_checks == 0)
 | |
|         && is_memory64) {
 | |
|         /* For memory64, the boundary check default value is true */
 | |
|         comp_ctx->enable_bound_check = true;
 | |
|     }
 | |
| 
 | |
|     /* Return error if SIMD is disabled by command line but SIMD instructions
 | |
|      * are used */
 | |
|     if (!option->enable_simd && wasm_module->is_simd_used) {
 | |
|         aot_set_last_error("SIMD is disabled by --disable-simd but SIMD "
 | |
|                            "instructions are used in this module");
 | |
|         goto fail;
 | |
|     }
 | |
| 
 | |
|     /* Return error if ref-types and GC are disabled by command line but
 | |
|        ref-types instructions are used */
 | |
|     if (!option->enable_ref_types && !option->enable_gc
 | |
|         && wasm_module->is_ref_types_used) {
 | |
|         aot_set_last_error("ref-types instruction was found, "
 | |
|                            "try removing --disable-ref-types option "
 | |
|                            "or adding --enable-gc option.");
 | |
|         goto fail;
 | |
|     }
 | |
| 
 | |
|     /* Disable features when they are not actually used */
 | |
|     if (!wasm_module->is_simd_used) {
 | |
|         option->enable_simd = comp_ctx->enable_simd = false;
 | |
|     }
 | |
|     if (!wasm_module->is_ref_types_used) {
 | |
|         option->enable_ref_types = comp_ctx->enable_ref_types = false;
 | |
|     }
 | |
|     if (!wasm_module->is_bulk_memory_used) {
 | |
|         option->enable_bulk_memory = comp_ctx->enable_bulk_memory = false;
 | |
|     }
 | |
| #endif
 | |
| 
 | |
|     if (option->enable_simd && strcmp(comp_ctx->target_arch, "x86_64") != 0
 | |
|         && strncmp(comp_ctx->target_arch, "aarch64", 7) != 0
 | |
|         && strcmp(comp_ctx->target_arch, "arc") != 0) {
 | |
|         /* Disable simd if it isn't supported by target arch */
 | |
|         option->enable_simd = false;
 | |
|     }
 | |
| 
 | |
|     if (option->enable_simd) {
 | |
|         char *tmp;
 | |
|         bool check_simd_ret;
 | |
| 
 | |
|         comp_ctx->enable_simd = true;
 | |
| 
 | |
|         if (!(tmp = LLVMGetTargetMachineCPU(comp_ctx->target_machine))) {
 | |
|             aot_set_last_error("get CPU from Target Machine fail");
 | |
|             goto fail;
 | |
|         }
 | |
| 
 | |
|         check_simd_ret =
 | |
|             aot_check_simd_compatibility(comp_ctx->target_arch, tmp);
 | |
|         LLVMDisposeMessage(tmp);
 | |
|         if (!check_simd_ret) {
 | |
|             aot_set_last_error("SIMD compatibility check failed, "
 | |
|                                "try adding --cpu=<cpu> to specify a cpu "
 | |
|                                "or adding --disable-simd to disable SIMD");
 | |
|             goto fail;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     if (!(target_data_ref =
 | |
|               LLVMCreateTargetDataLayout(comp_ctx->target_machine))) {
 | |
|         aot_set_last_error("create LLVM target data layout failed.");
 | |
|         goto fail;
 | |
|     }
 | |
|     LLVMSetModuleDataLayout(comp_ctx->module, target_data_ref);
 | |
|     comp_ctx->pointer_size = LLVMPointerSize(target_data_ref);
 | |
|     LLVMDisposeTargetData(target_data_ref);
 | |
| 
 | |
|     comp_ctx->optimize = true;
 | |
|     if (option->output_format == AOT_LLVMIR_UNOPT_FILE)
 | |
|         comp_ctx->optimize = false;
 | |
| 
 | |
|     /* Create metadata for llvm float experimental constrained intrinsics */
 | |
|     if (!(comp_ctx->fp_rounding_mode = LLVMMDStringInContext(
 | |
|               comp_ctx->context, fp_round, (uint32)strlen(fp_round)))
 | |
|         || !(comp_ctx->fp_exception_behavior = LLVMMDStringInContext(
 | |
|                  comp_ctx->context, fp_exce, (uint32)strlen(fp_exce)))) {
 | |
|         aot_set_last_error("create float llvm metadata failed.");
 | |
|         goto fail;
 | |
|     }
 | |
| 
 | |
|     if (!aot_set_llvm_basic_types(&comp_ctx->basic_types, comp_ctx->context,
 | |
|                                   comp_ctx->pointer_size)) {
 | |
|         aot_set_last_error("create LLVM basic types failed.");
 | |
|         goto fail;
 | |
|     }
 | |
| 
 | |
|     if (!aot_create_llvm_consts(&comp_ctx->llvm_consts, comp_ctx)) {
 | |
|         aot_set_last_error("create LLVM const values failed.");
 | |
|         goto fail;
 | |
|     }
 | |
| 
 | |
|     /* set exec_env data type to int8** */
 | |
|     comp_ctx->exec_env_type = comp_ctx->basic_types.int8_pptr_type;
 | |
| 
 | |
|     /* set aot_inst data type to int8* */
 | |
|     comp_ctx->aot_inst_type = INT8_PTR_TYPE;
 | |
| 
 | |
|     /* Create function context for each function */
 | |
|     comp_ctx->func_ctx_count = comp_data->func_count;
 | |
|     if (comp_data->func_count > 0
 | |
|         && !(comp_ctx->func_ctxes =
 | |
|                  aot_create_func_contexts(comp_data, comp_ctx)))
 | |
|         goto fail;
 | |
| 
 | |
|     if (cpu) {
 | |
|         uint32 len = (uint32)strlen(cpu) + 1;
 | |
|         if (!(comp_ctx->target_cpu = wasm_runtime_malloc(len))) {
 | |
|             aot_set_last_error("allocate memory failed");
 | |
|             goto fail;
 | |
|         }
 | |
|         bh_memcpy_s(comp_ctx->target_cpu, len, cpu, len);
 | |
|     }
 | |
| 
 | |
|     if (comp_ctx->disable_llvm_intrinsics)
 | |
|         aot_intrinsic_fill_capability_flags(comp_ctx);
 | |
| 
 | |
|     ret = comp_ctx;
 | |
| 
 | |
| fail:
 | |
|     if (triple_norm_new)
 | |
|         LLVMDisposeMessage(triple_norm_new);
 | |
| 
 | |
|     if (cpu_new)
 | |
|         LLVMDisposeMessage(cpu_new);
 | |
| 
 | |
|     if (!ret)
 | |
|         aot_destroy_comp_context(comp_ctx);
 | |
| 
 | |
|     (void)i;
 | |
|     return ret;
 | |
| }
 | |
| 
 | |
| void
 | |
| aot_destroy_comp_context(AOTCompContext *comp_ctx)
 | |
| {
 | |
|     if (!comp_ctx)
 | |
|         return;
 | |
| 
 | |
|     if (comp_ctx->stack_usage_file == comp_ctx->stack_usage_temp_file) {
 | |
|         (void)unlink(comp_ctx->stack_usage_temp_file);
 | |
|     }
 | |
| 
 | |
|     if (comp_ctx->target_machine)
 | |
|         LLVMDisposeTargetMachine(comp_ctx->target_machine);
 | |
| 
 | |
|     if (comp_ctx->builder)
 | |
|         LLVMDisposeBuilder(comp_ctx->builder);
 | |
| 
 | |
| #if WASM_ENABLE_DEBUG_AOT != 0
 | |
|     if (comp_ctx->debug_builder)
 | |
|         LLVMDisposeDIBuilder(comp_ctx->debug_builder);
 | |
| #endif
 | |
| 
 | |
|     if (comp_ctx->orc_thread_safe_context)
 | |
|         LLVMOrcDisposeThreadSafeContext(comp_ctx->orc_thread_safe_context);
 | |
| 
 | |
|     /* Note: don't dispose comp_ctx->context and comp_ctx->module as
 | |
|        they are disposed when disposing the thread safe context */
 | |
| 
 | |
|     /* Has to be the last one */
 | |
|     if (comp_ctx->orc_jit)
 | |
|         LLVMOrcDisposeLLLazyJIT(comp_ctx->orc_jit);
 | |
| 
 | |
|     if (comp_ctx->func_ctxes)
 | |
|         aot_destroy_func_contexts(comp_ctx, comp_ctx->func_ctxes,
 | |
|                                   comp_ctx->func_ctx_count);
 | |
| 
 | |
|     if (bh_list_length(&comp_ctx->native_symbols) > 0) {
 | |
|         AOTNativeSymbol *sym = bh_list_first_elem(&comp_ctx->native_symbols);
 | |
|         while (sym) {
 | |
|             AOTNativeSymbol *t = bh_list_elem_next(sym);
 | |
|             bh_list_remove(&comp_ctx->native_symbols, sym);
 | |
|             wasm_runtime_free(sym);
 | |
|             sym = t;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     if (comp_ctx->target_cpu) {
 | |
|         wasm_runtime_free(comp_ctx->target_cpu);
 | |
|     }
 | |
| 
 | |
|     if (comp_ctx->aot_frame) {
 | |
|         wasm_runtime_free(comp_ctx->aot_frame);
 | |
|     }
 | |
| 
 | |
|     wasm_runtime_free(comp_ctx);
 | |
| }
 | |
| 
 | |
| static bool
 | |
| insert_native_symbol(AOTCompContext *comp_ctx, const char *symbol, int32 idx)
 | |
| {
 | |
|     AOTNativeSymbol *sym = wasm_runtime_malloc(sizeof(AOTNativeSymbol));
 | |
|     int ret;
 | |
| 
 | |
|     if (!sym) {
 | |
|         aot_set_last_error("alloc native symbol failed.");
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     memset(sym, 0, sizeof(AOTNativeSymbol));
 | |
|     bh_assert(strlen(symbol) <= sizeof(sym->symbol));
 | |
|     ret = snprintf(sym->symbol, sizeof(sym->symbol), "%s", symbol);
 | |
|     if (ret < 0 || ret + 1 > (int)sizeof(sym->symbol)) {
 | |
|         wasm_runtime_free(sym);
 | |
|         aot_set_last_error_v("symbol name too long: %s", symbol);
 | |
|         return false;
 | |
|     }
 | |
|     sym->index = idx;
 | |
| 
 | |
|     if (BH_LIST_ERROR == bh_list_insert(&comp_ctx->native_symbols, sym)) {
 | |
|         wasm_runtime_free(sym);
 | |
|         aot_set_last_error("insert native symbol to list failed.");
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     return true;
 | |
| }
 | |
| 
 | |
| int32
 | |
| aot_get_native_symbol_index(AOTCompContext *comp_ctx, const char *symbol)
 | |
| {
 | |
|     int32 idx = -1;
 | |
|     AOTNativeSymbol *sym = NULL;
 | |
| 
 | |
|     sym = bh_list_first_elem(&comp_ctx->native_symbols);
 | |
| 
 | |
|     /* Lookup an existing symbol record */
 | |
| 
 | |
|     while (sym) {
 | |
|         if (strcmp(sym->symbol, symbol) == 0) {
 | |
|             idx = sym->index;
 | |
|             break;
 | |
|         }
 | |
|         sym = bh_list_elem_next(sym);
 | |
|     }
 | |
| 
 | |
|     /* Given symbol is not exist in list, then we alloc a new index for it */
 | |
| 
 | |
|     if (idx < 0) {
 | |
|         if (comp_ctx->pointer_size == sizeof(uint32)
 | |
|             && (!strncmp(symbol, "f64#", 4) || !strncmp(symbol, "i64#", 4))) {
 | |
|             idx = bh_list_length(&comp_ctx->native_symbols);
 | |
|             /* Add 4 bytes padding on 32-bit target to make sure that
 | |
|                the f64 const is stored on 8-byte aligned address */
 | |
|             if (idx & 1) {
 | |
|                 if (!insert_native_symbol(comp_ctx, "__ignore", idx)) {
 | |
|                     return -1;
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         idx = bh_list_length(&comp_ctx->native_symbols);
 | |
|         if (!insert_native_symbol(comp_ctx, symbol, idx)) {
 | |
|             return -1;
 | |
|         }
 | |
| 
 | |
|         if (comp_ctx->pointer_size == sizeof(uint32)
 | |
|             && (!strncmp(symbol, "f64#", 4) || !strncmp(symbol, "i64#", 4))) {
 | |
|             /* f64 const occupies 2 pointer slots on 32-bit target */
 | |
|             if (!insert_native_symbol(comp_ctx, "__ignore", idx + 1)) {
 | |
|                 return -1;
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     return idx;
 | |
| }
 | |
| 
 | |
| void
 | |
| aot_value_stack_push(const AOTCompContext *comp_ctx, AOTValueStack *stack,
 | |
|                      AOTValue *value)
 | |
| {
 | |
|     if (!stack->value_list_head)
 | |
|         stack->value_list_head = stack->value_list_end = value;
 | |
|     else {
 | |
|         stack->value_list_end->next = value;
 | |
|         value->prev = stack->value_list_end;
 | |
|         stack->value_list_end = value;
 | |
|     }
 | |
| 
 | |
|     if (comp_ctx->aot_frame) {
 | |
|         switch (value->type) {
 | |
|             case VALUE_TYPE_I32:
 | |
|             case VALUE_TYPE_I1:
 | |
|                 push_i32(comp_ctx->aot_frame, value);
 | |
|                 break;
 | |
|             case VALUE_TYPE_I64:
 | |
|                 push_i64(comp_ctx->aot_frame, value);
 | |
|                 break;
 | |
|             case VALUE_TYPE_F32:
 | |
|                 push_f32(comp_ctx->aot_frame, value);
 | |
|                 break;
 | |
|             case VALUE_TYPE_F64:
 | |
|                 push_f64(comp_ctx->aot_frame, value);
 | |
|                 break;
 | |
|             case VALUE_TYPE_V128:
 | |
|                 push_v128(comp_ctx->aot_frame, value);
 | |
|                 break;
 | |
|             case VALUE_TYPE_FUNCREF:
 | |
|             case VALUE_TYPE_EXTERNREF:
 | |
|                 push_ref(comp_ctx->aot_frame, value);
 | |
|                 break;
 | |
| #if WASM_ENABLE_GC != 0
 | |
|             case VALUE_TYPE_GC_REF:
 | |
|                 bh_assert(comp_ctx->enable_gc);
 | |
|                 push_gc_ref(comp_ctx->aot_frame, value);
 | |
|                 break;
 | |
| #endif
 | |
|             default:
 | |
|                 bh_assert(0);
 | |
|                 break;
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| AOTValue *
 | |
| aot_value_stack_pop(const AOTCompContext *comp_ctx, AOTValueStack *stack)
 | |
| {
 | |
|     AOTValue *value = stack->value_list_end;
 | |
| 
 | |
|     bh_assert(stack->value_list_end);
 | |
| 
 | |
|     if (stack->value_list_head == stack->value_list_end)
 | |
|         stack->value_list_head = stack->value_list_end = NULL;
 | |
|     else {
 | |
|         stack->value_list_end = stack->value_list_end->prev;
 | |
|         stack->value_list_end->next = NULL;
 | |
|         value->prev = NULL;
 | |
|     }
 | |
| 
 | |
|     if (comp_ctx->aot_frame) {
 | |
|         bh_assert(value);
 | |
|         bh_assert(value->value == (comp_ctx->aot_frame->sp - 1)->value);
 | |
|         bh_assert(value->type == (comp_ctx->aot_frame->sp - 1)->type);
 | |
| 
 | |
|         switch (value->type) {
 | |
|             case VALUE_TYPE_I32:
 | |
|             case VALUE_TYPE_I1:
 | |
|                 pop_i32(comp_ctx->aot_frame);
 | |
|                 break;
 | |
|             case VALUE_TYPE_I64:
 | |
|                 pop_i64(comp_ctx->aot_frame);
 | |
|                 break;
 | |
|             case VALUE_TYPE_F32:
 | |
|                 pop_f32(comp_ctx->aot_frame);
 | |
|                 break;
 | |
|             case VALUE_TYPE_F64:
 | |
|                 pop_f64(comp_ctx->aot_frame);
 | |
|                 break;
 | |
|             case VALUE_TYPE_V128:
 | |
|                 pop_v128(comp_ctx->aot_frame);
 | |
|                 break;
 | |
|             case VALUE_TYPE_FUNCREF:
 | |
|             case VALUE_TYPE_EXTERNREF:
 | |
|                 pop_ref(comp_ctx->aot_frame);
 | |
|                 break;
 | |
| #if WASM_ENABLE_GC != 0
 | |
|             case VALUE_TYPE_GC_REF:
 | |
|                 bh_assert(comp_ctx->enable_gc);
 | |
|                 pop_gc_ref(comp_ctx->aot_frame);
 | |
|                 break;
 | |
| #endif
 | |
|             default:
 | |
|                 bh_assert(0);
 | |
|                 break;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     return value;
 | |
| }
 | |
| 
 | |
| void
 | |
| aot_value_stack_destroy(AOTCompContext *comp_ctx, AOTValueStack *stack)
 | |
| {
 | |
|     AOTValue *value = stack->value_list_head, *p;
 | |
| 
 | |
|     while (value) {
 | |
|         p = value->next;
 | |
|         wasm_runtime_free(value);
 | |
|         value = p;
 | |
|     }
 | |
| 
 | |
|     stack->value_list_head = NULL;
 | |
|     stack->value_list_end = NULL;
 | |
| }
 | |
| 
 | |
| void
 | |
| aot_block_stack_push(AOTBlockStack *stack, AOTBlock *block)
 | |
| {
 | |
|     if (!stack->block_list_head)
 | |
|         stack->block_list_head = stack->block_list_end = block;
 | |
|     else {
 | |
|         stack->block_list_end->next = block;
 | |
|         block->prev = stack->block_list_end;
 | |
|         stack->block_list_end = block;
 | |
|     }
 | |
| }
 | |
| 
 | |
| AOTBlock *
 | |
| aot_block_stack_pop(AOTBlockStack *stack)
 | |
| {
 | |
|     AOTBlock *block = stack->block_list_end;
 | |
| 
 | |
|     bh_assert(stack->block_list_end);
 | |
| 
 | |
|     if (stack->block_list_head == stack->block_list_end)
 | |
|         stack->block_list_head = stack->block_list_end = NULL;
 | |
|     else {
 | |
|         stack->block_list_end = stack->block_list_end->prev;
 | |
|         stack->block_list_end->next = NULL;
 | |
|         block->prev = NULL;
 | |
|     }
 | |
| 
 | |
|     return block;
 | |
| }
 | |
| 
 | |
| void
 | |
| aot_block_stack_destroy(AOTCompContext *comp_ctx, AOTBlockStack *stack)
 | |
| {
 | |
|     AOTBlock *block = stack->block_list_head, *p;
 | |
| 
 | |
|     while (block) {
 | |
|         p = block->next;
 | |
|         aot_value_stack_destroy(comp_ctx, &block->value_stack);
 | |
|         aot_block_destroy(comp_ctx, block);
 | |
|         block = p;
 | |
|     }
 | |
| 
 | |
|     stack->block_list_head = NULL;
 | |
|     stack->block_list_end = NULL;
 | |
| }
 | |
| 
 | |
| void
 | |
| aot_block_destroy(AOTCompContext *comp_ctx, AOTBlock *block)
 | |
| {
 | |
|     aot_value_stack_destroy(comp_ctx, &block->value_stack);
 | |
|     if (block->param_types)
 | |
|         wasm_runtime_free(block->param_types);
 | |
|     if (block->param_phis)
 | |
|         wasm_runtime_free(block->param_phis);
 | |
|     if (block->else_param_phis)
 | |
|         wasm_runtime_free(block->else_param_phis);
 | |
|     if (block->result_types)
 | |
|         wasm_runtime_free(block->result_types);
 | |
|     if (block->result_phis)
 | |
|         wasm_runtime_free(block->result_phis);
 | |
|     wasm_runtime_free(block);
 | |
| }
 | |
| 
 | |
| bool
 | |
| aot_checked_addr_list_add(AOTFuncContext *func_ctx, uint32 local_idx,
 | |
|                           uint64 offset, uint32 bytes)
 | |
| {
 | |
|     AOTCheckedAddr *node = func_ctx->checked_addr_list;
 | |
| 
 | |
|     if (!(node = wasm_runtime_malloc(sizeof(AOTCheckedAddr)))) {
 | |
|         aot_set_last_error("allocate memory failed.");
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     node->local_idx = local_idx;
 | |
|     node->offset = offset;
 | |
|     node->bytes = bytes;
 | |
| 
 | |
|     node->next = func_ctx->checked_addr_list;
 | |
|     func_ctx->checked_addr_list = node;
 | |
|     return true;
 | |
| }
 | |
| 
 | |
| void
 | |
| aot_checked_addr_list_del(AOTFuncContext *func_ctx, uint32 local_idx)
 | |
| {
 | |
|     AOTCheckedAddr *node = func_ctx->checked_addr_list;
 | |
|     AOTCheckedAddr *node_prev = NULL, *node_next;
 | |
| 
 | |
|     while (node) {
 | |
|         node_next = node->next;
 | |
| 
 | |
|         if (node->local_idx == local_idx) {
 | |
|             if (!node_prev)
 | |
|                 func_ctx->checked_addr_list = node_next;
 | |
|             else
 | |
|                 node_prev->next = node_next;
 | |
|             wasm_runtime_free(node);
 | |
|         }
 | |
|         else {
 | |
|             node_prev = node;
 | |
|         }
 | |
| 
 | |
|         node = node_next;
 | |
|     }
 | |
| }
 | |
| 
 | |
| bool
 | |
| aot_checked_addr_list_find(AOTFuncContext *func_ctx, uint32 local_idx,
 | |
|                            uint64 offset, uint32 bytes)
 | |
| {
 | |
|     AOTCheckedAddr *node = func_ctx->checked_addr_list;
 | |
| 
 | |
|     while (node) {
 | |
|         if (node->local_idx == local_idx && node->offset == offset
 | |
|             && node->bytes >= bytes) {
 | |
|             return true;
 | |
|         }
 | |
|         node = node->next;
 | |
|     }
 | |
| 
 | |
|     return false;
 | |
| }
 | |
| 
 | |
| void
 | |
| aot_checked_addr_list_destroy(AOTFuncContext *func_ctx)
 | |
| {
 | |
|     AOTCheckedAddr *node = func_ctx->checked_addr_list, *node_next;
 | |
| 
 | |
|     while (node) {
 | |
|         node_next = node->next;
 | |
|         wasm_runtime_free(node);
 | |
|         node = node_next;
 | |
|     }
 | |
| 
 | |
|     func_ctx->checked_addr_list = NULL;
 | |
| }
 | |
| 
 | |
| bool
 | |
| aot_build_zero_function_ret(const AOTCompContext *comp_ctx,
 | |
|                             AOTFuncContext *func_ctx, AOTFuncType *func_type)
 | |
| {
 | |
|     LLVMValueRef ret = NULL;
 | |
| 
 | |
|     if (func_type->result_count) {
 | |
|         switch (func_type->types[func_type->param_count]) {
 | |
|             case VALUE_TYPE_I32:
 | |
|                 ret = LLVMBuildRet(comp_ctx->builder, I32_ZERO);
 | |
|                 break;
 | |
|             case VALUE_TYPE_I64:
 | |
|                 ret = LLVMBuildRet(comp_ctx->builder, I64_ZERO);
 | |
|                 break;
 | |
|             case VALUE_TYPE_F32:
 | |
|                 ret = LLVMBuildRet(comp_ctx->builder, F32_ZERO);
 | |
|                 break;
 | |
|             case VALUE_TYPE_F64:
 | |
|                 ret = LLVMBuildRet(comp_ctx->builder, F64_ZERO);
 | |
|                 break;
 | |
|             case VALUE_TYPE_V128:
 | |
|                 ret =
 | |
|                     LLVMBuildRet(comp_ctx->builder, LLVM_CONST(i64x2_vec_zero));
 | |
|                 break;
 | |
|             case VALUE_TYPE_FUNCREF:
 | |
|             case VALUE_TYPE_EXTERNREF:
 | |
|                 if (comp_ctx->enable_ref_types)
 | |
|                     ret = LLVMBuildRet(comp_ctx->builder, REF_NULL);
 | |
| #if WASM_ENABLE_GC != 0
 | |
|                 else if (comp_ctx->enable_gc)
 | |
|                     ret = LLVMBuildRet(comp_ctx->builder, GC_REF_NULL);
 | |
| #endif
 | |
|                 else
 | |
|                     bh_assert(0);
 | |
|                 break;
 | |
| #if WASM_ENABLE_GC != 0
 | |
|             case REF_TYPE_NULLFUNCREF:
 | |
|             case REF_TYPE_NULLEXTERNREF:
 | |
|             case REF_TYPE_NULLREF:
 | |
|             /* case REF_TYPE_FUNCREF: */
 | |
|             /* case REF_TYPE_EXTERNREF: */
 | |
|             case REF_TYPE_ANYREF:
 | |
|             case REF_TYPE_EQREF:
 | |
|             case REF_TYPE_HT_NULLABLE:
 | |
|             case REF_TYPE_HT_NON_NULLABLE:
 | |
|             case REF_TYPE_I31REF:
 | |
|             case REF_TYPE_STRUCTREF:
 | |
|             case REF_TYPE_ARRAYREF:
 | |
| #if WASM_ENABLE_STRINGREF != 0
 | |
|             case REF_TYPE_STRINGREF:
 | |
|             case REF_TYPE_STRINGVIEWWTF8:
 | |
|             case REF_TYPE_STRINGVIEWWTF16:
 | |
|             case REF_TYPE_STRINGVIEWITER:
 | |
| #endif
 | |
|                 bh_assert(comp_ctx->enable_gc);
 | |
|                 ret = LLVMBuildRet(comp_ctx->builder, GC_REF_NULL);
 | |
|                 break;
 | |
| #endif
 | |
|             default:
 | |
|                 bh_assert(0);
 | |
|         }
 | |
|     }
 | |
|     else {
 | |
|         ret = LLVMBuildRetVoid(comp_ctx->builder);
 | |
|     }
 | |
| 
 | |
|     if (!ret) {
 | |
|         aot_set_last_error("llvm build ret failed.");
 | |
|         return false;
 | |
|     }
 | |
| #if WASM_ENABLE_DEBUG_AOT != 0
 | |
|     /* debug_func is NULL for precheck function */
 | |
|     if (func_ctx->debug_func != NULL) {
 | |
|         LLVMMetadataRef return_location =
 | |
|             dwarf_gen_func_ret_location(comp_ctx, func_ctx);
 | |
|         LLVMInstructionSetDebugLoc(ret, return_location);
 | |
|     }
 | |
| #endif
 | |
|     return true;
 | |
| }
 | |
| 
 | |
| static LLVMValueRef
 | |
| __call_llvm_intrinsic(const AOTCompContext *comp_ctx,
 | |
|                       const AOTFuncContext *func_ctx, const char *name,
 | |
|                       LLVMTypeRef ret_type, LLVMTypeRef *param_types,
 | |
|                       int param_count, LLVMValueRef *param_values)
 | |
| {
 | |
|     LLVMValueRef func, ret;
 | |
|     LLVMTypeRef func_type;
 | |
|     const char *symname;
 | |
|     int32 func_idx;
 | |
| 
 | |
|     if (comp_ctx->disable_llvm_intrinsics
 | |
|         && aot_intrinsic_check_capability(comp_ctx, name)) {
 | |
|         if (func_ctx == NULL) {
 | |
|             aot_set_last_error_v("invalid func_ctx for intrinsic: %s", name);
 | |
|             return NULL;
 | |
|         }
 | |
| 
 | |
|         if (!(func_type = LLVMFunctionType(ret_type, param_types,
 | |
|                                            (uint32)param_count, false))) {
 | |
|             aot_set_last_error("create LLVM intrinsic function type failed.");
 | |
|             return NULL;
 | |
|         }
 | |
|         if (!(func_type = LLVMPointerType(func_type, 0))) {
 | |
|             aot_set_last_error(
 | |
|                 "create LLVM intrinsic function pointer type failed.");
 | |
|             return NULL;
 | |
|         }
 | |
| 
 | |
|         if (!(symname = aot_intrinsic_get_symbol(name))) {
 | |
|             aot_set_last_error_v("runtime intrinsic not implemented: %s\n",
 | |
|                                  name);
 | |
|             return NULL;
 | |
|         }
 | |
| 
 | |
|         func_idx =
 | |
|             aot_get_native_symbol_index((AOTCompContext *)comp_ctx, symname);
 | |
|         if (func_idx < 0) {
 | |
|             aot_set_last_error_v("get runtime intrinsc index failed: %s\n",
 | |
|                                  name);
 | |
|             return NULL;
 | |
|         }
 | |
| 
 | |
|         if (!(func = aot_get_func_from_table(comp_ctx, func_ctx->native_symbol,
 | |
|                                              func_type, func_idx))) {
 | |
|             aot_set_last_error_v("get runtime intrinsc failed: %s\n", name);
 | |
|             return NULL;
 | |
|         }
 | |
|     }
 | |
|     else {
 | |
|         /* Declare llvm intrinsic function if necessary */
 | |
|         if (!(func = LLVMGetNamedFunction(func_ctx->module, name))) {
 | |
|             if (!(func_type = LLVMFunctionType(ret_type, param_types,
 | |
|                                                (uint32)param_count, false))) {
 | |
|                 aot_set_last_error(
 | |
|                     "create LLVM intrinsic function type failed.");
 | |
|                 return NULL;
 | |
|             }
 | |
| 
 | |
|             if (!(func = LLVMAddFunction(func_ctx->module, name, func_type))) {
 | |
|                 aot_set_last_error("add LLVM intrinsic function failed.");
 | |
|                 return NULL;
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
| #if LLVM_VERSION_MAJOR >= 14
 | |
|     func_type =
 | |
|         LLVMFunctionType(ret_type, param_types, (uint32)param_count, false);
 | |
| #endif
 | |
| 
 | |
|     /* Call the LLVM intrinsic function */
 | |
|     if (!(ret = LLVMBuildCall2(comp_ctx->builder, func_type, func, param_values,
 | |
|                                (uint32)param_count, "call"))) {
 | |
|         aot_set_last_error("llvm build intrinsic call failed.");
 | |
|         return NULL;
 | |
|     }
 | |
| 
 | |
|     return ret;
 | |
| }
 | |
| 
 | |
| LLVMValueRef
 | |
| aot_call_llvm_intrinsic(const AOTCompContext *comp_ctx,
 | |
|                         const AOTFuncContext *func_ctx, const char *intrinsic,
 | |
|                         LLVMTypeRef ret_type, LLVMTypeRef *param_types,
 | |
|                         int param_count, ...)
 | |
| {
 | |
|     LLVMValueRef *param_values, ret;
 | |
|     va_list argptr;
 | |
|     uint64 total_size;
 | |
|     int i = 0;
 | |
| 
 | |
|     /* Create param values */
 | |
|     total_size = sizeof(LLVMValueRef) * (uint64)param_count;
 | |
|     if (total_size >= UINT32_MAX
 | |
|         || !(param_values = wasm_runtime_malloc((uint32)total_size))) {
 | |
|         aot_set_last_error("allocate memory for param values failed.");
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     /* Load each param value */
 | |
|     va_start(argptr, param_count);
 | |
|     while (i < param_count)
 | |
|         param_values[i++] = va_arg(argptr, LLVMValueRef);
 | |
|     va_end(argptr);
 | |
| 
 | |
|     ret = __call_llvm_intrinsic(comp_ctx, func_ctx, intrinsic, ret_type,
 | |
|                                 param_types, param_count, param_values);
 | |
| 
 | |
|     wasm_runtime_free(param_values);
 | |
| 
 | |
|     return ret;
 | |
| }
 | |
| 
 | |
| LLVMValueRef
 | |
| aot_call_llvm_intrinsic_v(const AOTCompContext *comp_ctx,
 | |
|                           const AOTFuncContext *func_ctx, const char *intrinsic,
 | |
|                           LLVMTypeRef ret_type, LLVMTypeRef *param_types,
 | |
|                           int param_count, va_list param_value_list)
 | |
| {
 | |
|     LLVMValueRef *param_values, ret;
 | |
|     uint64 total_size;
 | |
|     int i = 0;
 | |
| 
 | |
|     /* Create param values */
 | |
|     total_size = sizeof(LLVMValueRef) * (uint64)param_count;
 | |
|     if (total_size >= UINT32_MAX
 | |
|         || !(param_values = wasm_runtime_malloc((uint32)total_size))) {
 | |
|         aot_set_last_error("allocate memory for param values failed.");
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     /* Load each param value */
 | |
|     while (i < param_count)
 | |
|         param_values[i++] = va_arg(param_value_list, LLVMValueRef);
 | |
| 
 | |
|     ret = __call_llvm_intrinsic(comp_ctx, func_ctx, intrinsic, ret_type,
 | |
|                                 param_types, param_count, param_values);
 | |
| 
 | |
|     wasm_runtime_free(param_values);
 | |
| 
 | |
|     return ret;
 | |
| }
 | |
| 
 | |
| LLVMValueRef
 | |
| aot_get_func_from_table(const AOTCompContext *comp_ctx, LLVMValueRef base,
 | |
|                         LLVMTypeRef func_type, int32 index)
 | |
| {
 | |
|     LLVMValueRef func;
 | |
|     LLVMValueRef func_addr;
 | |
| 
 | |
|     if (!(func_addr = I32_CONST(index))) {
 | |
|         aot_set_last_error("construct function index failed.");
 | |
|         goto fail;
 | |
|     }
 | |
| 
 | |
|     if (!(func_addr =
 | |
|               LLVMBuildInBoundsGEP2(comp_ctx->builder, OPQ_PTR_TYPE, base,
 | |
|                                     &func_addr, 1, "func_addr"))) {
 | |
|         aot_set_last_error("get function addr by index failed.");
 | |
|         goto fail;
 | |
|     }
 | |
| 
 | |
|     func =
 | |
|         LLVMBuildLoad2(comp_ctx->builder, OPQ_PTR_TYPE, func_addr, "func_tmp");
 | |
| 
 | |
|     if (func == NULL) {
 | |
|         aot_set_last_error("get function pointer failed.");
 | |
|         goto fail;
 | |
|     }
 | |
| 
 | |
|     if (!(func =
 | |
|               LLVMBuildBitCast(comp_ctx->builder, func, func_type, "func"))) {
 | |
|         aot_set_last_error("cast function failed.");
 | |
|         goto fail;
 | |
|     }
 | |
| 
 | |
|     return func;
 | |
| fail:
 | |
|     return NULL;
 | |
| }
 | |
| 
 | |
| LLVMValueRef
 | |
| aot_load_const_from_table(AOTCompContext *comp_ctx, LLVMValueRef base,
 | |
|                           const WASMValue *value, uint8 value_type)
 | |
| {
 | |
|     LLVMValueRef const_index, const_addr, const_value;
 | |
|     LLVMTypeRef const_ptr_type, const_type;
 | |
|     char buf[128] = { 0 };
 | |
|     int32 index;
 | |
| 
 | |
|     switch (value_type) {
 | |
|         case VALUE_TYPE_I32:
 | |
|             /* Store the raw int bits of i32 const as a hex string */
 | |
|             snprintf(buf, sizeof(buf), "i32#%08" PRIX32, value->i32);
 | |
|             const_ptr_type = INT32_PTR_TYPE;
 | |
|             const_type = I32_TYPE;
 | |
|             break;
 | |
|         case VALUE_TYPE_I64:
 | |
|             /* Store the raw int bits of i64 const as a hex string */
 | |
|             snprintf(buf, sizeof(buf), "i64#%016" PRIX64, value->i64);
 | |
|             const_ptr_type = INT64_PTR_TYPE;
 | |
|             const_type = I64_TYPE;
 | |
|             break;
 | |
|         case VALUE_TYPE_F32:
 | |
|             /* Store the raw int bits of f32 const as a hex string */
 | |
|             snprintf(buf, sizeof(buf), "f32#%08" PRIX32, value->i32);
 | |
|             const_ptr_type = F32_PTR_TYPE;
 | |
|             const_type = F32_TYPE;
 | |
|             break;
 | |
|         case VALUE_TYPE_F64:
 | |
|             /* Store the raw int bits of f64 const as a hex string */
 | |
|             snprintf(buf, sizeof(buf), "f64#%016" PRIX64, value->i64);
 | |
|             const_ptr_type = F64_PTR_TYPE;
 | |
|             const_type = F64_TYPE;
 | |
|             break;
 | |
|         default:
 | |
|             bh_assert(0);
 | |
|             return NULL;
 | |
|     }
 | |
| 
 | |
|     /* Load f32/f64 const from exec_env->native_symbol[index] */
 | |
| 
 | |
|     index = aot_get_native_symbol_index(comp_ctx, buf);
 | |
|     if (index < 0) {
 | |
|         return NULL;
 | |
|     }
 | |
| 
 | |
|     if (!(const_index = I32_CONST(index))) {
 | |
|         aot_set_last_error("construct const index failed.");
 | |
|         return NULL;
 | |
|     }
 | |
| 
 | |
|     if (!(const_addr =
 | |
|               LLVMBuildInBoundsGEP2(comp_ctx->builder, OPQ_PTR_TYPE, base,
 | |
|                                     &const_index, 1, "const_addr_tmp"))) {
 | |
|         aot_set_last_error("get const addr by index failed.");
 | |
|         return NULL;
 | |
|     }
 | |
| 
 | |
|     if (!(const_addr = LLVMBuildBitCast(comp_ctx->builder, const_addr,
 | |
|                                         const_ptr_type, "const_addr"))) {
 | |
|         aot_set_last_error("cast const failed.");
 | |
|         return NULL;
 | |
|     }
 | |
| 
 | |
|     if (!(const_value = LLVMBuildLoad2(comp_ctx->builder, const_type,
 | |
|                                        const_addr, "const_value"))) {
 | |
|         aot_set_last_error("load const failed.");
 | |
|         return NULL;
 | |
|     }
 | |
| 
 | |
|     (void)const_type;
 | |
|     return const_value;
 | |
| }
 | |
| 
 | |
| bool
 | |
| aot_set_cond_br_weights(AOTCompContext *comp_ctx, LLVMValueRef cond_br,
 | |
|                         int32 weights_true, int32 weights_false)
 | |
| {
 | |
|     LLVMMetadataRef md_nodes[3], meta_data;
 | |
|     LLVMValueRef meta_data_as_value;
 | |
| 
 | |
|     md_nodes[0] = LLVMMDStringInContext2(comp_ctx->context, "branch_weights",
 | |
|                                          strlen("branch_weights"));
 | |
|     md_nodes[1] = LLVMValueAsMetadata(I32_CONST(weights_true));
 | |
|     md_nodes[2] = LLVMValueAsMetadata(I32_CONST(weights_false));
 | |
| 
 | |
|     meta_data = LLVMMDNodeInContext2(comp_ctx->context, md_nodes, 3);
 | |
|     meta_data_as_value = LLVMMetadataAsValue(comp_ctx->context, meta_data);
 | |
| 
 | |
|     LLVMSetMetadata(cond_br, 2, meta_data_as_value);
 | |
| 
 | |
|     return true;
 | |
| }
 |