| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390 |
- /*
- * Copyright (C) 2019 Intel Corporation. All rights reserved.
- * SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
- */
- #ifndef _AOT_COMPILER_H_
- #define _AOT_COMPILER_H_
- #include "aot.h"
- #include "aot_llvm.h"
- #ifdef __cplusplus
- extern "C" {
- #endif
- typedef enum IntCond {
- INT_EQZ = 0,
- INT_EQ,
- INT_NE,
- INT_LT_S,
- INT_LT_U,
- INT_GT_S,
- INT_GT_U,
- INT_LE_S,
- INT_LE_U,
- INT_GE_S,
- INT_GE_U
- } IntCond;
- typedef enum FloatCond {
- FLOAT_EQ = 0,
- FLOAT_NE,
- FLOAT_LT,
- FLOAT_GT,
- FLOAT_LE,
- FLOAT_GE
- } FloatCond;
- typedef enum IntArithmetic {
- INT_ADD = 0,
- INT_SUB,
- INT_MUL,
- INT_DIV_S,
- INT_DIV_U,
- INT_REM_S,
- INT_REM_U
- } IntArithmetic;
- typedef enum V128Arithmetic {
- V128_ADD = 0,
- V128_ADD_SATURATE_S,
- V128_ADD_SATURATE_U,
- V128_SUB,
- V128_SUB_SATURATE_S,
- V128_SUB_SATURATE_U,
- V128_MUL,
- V128_DIV,
- V128_NEG,
- V128_MIN,
- V128_MAX,
- } V128Arithmetic;
- typedef enum IntBitwise {
- INT_AND = 0,
- INT_OR,
- INT_XOR,
- } IntBitwise;
- typedef enum V128Bitwise {
- V128_NOT,
- V128_AND,
- V128_ANDNOT,
- V128_OR,
- V128_XOR,
- V128_BITSELECT
- } V128Bitwise;
- typedef enum IntShift {
- INT_SHL = 0,
- INT_SHR_S,
- INT_SHR_U,
- INT_ROTL,
- INT_ROTR
- } IntShift;
- typedef enum FloatMath {
- FLOAT_ABS = 0,
- FLOAT_NEG,
- FLOAT_CEIL,
- FLOAT_FLOOR,
- FLOAT_TRUNC,
- FLOAT_NEAREST,
- FLOAT_SQRT
- } FloatMath;
- typedef enum FloatArithmetic {
- FLOAT_ADD = 0,
- FLOAT_SUB,
- FLOAT_MUL,
- FLOAT_DIV,
- FLOAT_MIN,
- FLOAT_MAX
- } FloatArithmetic;
- static inline bool
- check_type_compatible(uint8 src_type, uint8 dst_type)
- {
- if (src_type == dst_type) {
- return true;
- }
- /* ext i1 to i32 */
- if (src_type == VALUE_TYPE_I1 && dst_type == VALUE_TYPE_I32) {
- return true;
- }
- /* i32 <==> func.ref, i32 <==> extern.ref */
- if (src_type == VALUE_TYPE_I32
- && (dst_type == VALUE_TYPE_EXTERNREF
- || dst_type == VALUE_TYPE_FUNCREF)) {
- return true;
- }
- if (dst_type == VALUE_TYPE_I32
- && (src_type == VALUE_TYPE_FUNCREF
- || src_type == VALUE_TYPE_EXTERNREF)) {
- return true;
- }
- return false;
- }
- #define CHECK_STACK() do { \
- if (!func_ctx->block_stack.block_list_end) { \
- aot_set_last_error("WASM block stack underflow."); \
- goto fail; \
- } \
- if (!func_ctx->block_stack.block_list_end-> \
- value_stack.value_list_end) { \
- aot_set_last_error("WASM data stack underflow."); \
- goto fail; \
- } \
- } while (0)
- #define POP(llvm_value, value_type) do { \
- AOTValue *aot_value; \
- CHECK_STACK(); \
- aot_value = aot_value_stack_pop \
- (&func_ctx->block_stack.block_list_end->value_stack); \
- if (!check_type_compatible(aot_value->type, \
- value_type)) { \
- aot_set_last_error("invalid WASM stack data type."); \
- wasm_runtime_free(aot_value); \
- goto fail; \
- } \
- if (aot_value->type == value_type) \
- llvm_value = aot_value->value; \
- else { \
- if (aot_value->type == VALUE_TYPE_I1) { \
- if (!(llvm_value = LLVMBuildZExt(comp_ctx->builder, \
- aot_value->value, I32_TYPE, "i1toi32"))) { \
- aot_set_last_error("invalid WASM stack " \
- "data type."); \
- wasm_runtime_free(aot_value); \
- goto fail; \
- } \
- } \
- else { \
- bh_assert(aot_value->type == VALUE_TYPE_I32 \
- || aot_value->type == VALUE_TYPE_FUNCREF \
- || aot_value->type == VALUE_TYPE_EXTERNREF); \
- bh_assert(value_type == VALUE_TYPE_I32 \
- || value_type == VALUE_TYPE_FUNCREF \
- || value_type == VALUE_TYPE_EXTERNREF); \
- llvm_value = aot_value->value; \
- } \
- } \
- wasm_runtime_free(aot_value); \
- } while (0)
- #define POP_I32(v) POP(v, VALUE_TYPE_I32)
- #define POP_I64(v) POP(v, VALUE_TYPE_I64)
- #define POP_F32(v) POP(v, VALUE_TYPE_F32)
- #define POP_F64(v) POP(v, VALUE_TYPE_F64)
- #define POP_V128(v) POP(v, VALUE_TYPE_V128)
- #define POP_FUNCREF(v) POP(v, VALUE_TYPE_FUNCREF)
- #define POP_EXTERNREF(v) POP(v, VALUE_TYPE_EXTERNREF)
- #define POP_COND(llvm_value) do { \
- AOTValue *aot_value; \
- CHECK_STACK(); \
- aot_value = aot_value_stack_pop \
- (&func_ctx->block_stack.block_list_end->value_stack); \
- if (aot_value->type != VALUE_TYPE_I1 \
- && aot_value->type != VALUE_TYPE_I32) { \
- aot_set_last_error("invalid WASM stack data type."); \
- wasm_runtime_free(aot_value); \
- goto fail; \
- } \
- if (aot_value->type == VALUE_TYPE_I1) \
- llvm_value = aot_value->value; \
- else { \
- if (!(llvm_value = LLVMBuildICmp(comp_ctx->builder, \
- LLVMIntNE, aot_value->value, I32_ZERO, \
- "i1_cond"))){ \
- aot_set_last_error("llvm build trunc failed."); \
- wasm_runtime_free(aot_value); \
- goto fail; \
- } \
- } \
- wasm_runtime_free(aot_value); \
- } while (0)
- #define PUSH(llvm_value, value_type) do { \
- AOTValue *aot_value; \
- if (!func_ctx->block_stack.block_list_end) { \
- aot_set_last_error("WASM block stack underflow."); \
- goto fail; \
- } \
- aot_value = wasm_runtime_malloc(sizeof(AOTValue)); \
- if (!aot_value) { \
- aot_set_last_error("allocate memory failed."); \
- goto fail; \
- } \
- memset(aot_value, 0, sizeof(AOTValue)); \
- aot_value->type = value_type; \
- aot_value->value = llvm_value; \
- aot_value_stack_push \
- (&func_ctx->block_stack.block_list_end->value_stack,\
- aot_value); \
- } while (0)
- #define PUSH_I32(v) PUSH(v, VALUE_TYPE_I32)
- #define PUSH_I64(v) PUSH(v, VALUE_TYPE_I64)
- #define PUSH_F32(v) PUSH(v, VALUE_TYPE_F32)
- #define PUSH_F64(v) PUSH(v, VALUE_TYPE_F64)
- #define PUSH_V128(v) PUSH(v, VALUE_TYPE_V128)
- #define PUSH_COND(v) PUSH(v, VALUE_TYPE_I1)
- #define PUSH_FUNCREF(v) PUSH(v, VALUE_TYPE_FUNCREF)
- #define PUSH_EXTERNREF(v) PUSH(v, VALUE_TYPE_EXTERNREF)
- #define TO_LLVM_TYPE(wasm_type) \
- wasm_type_to_llvm_type(&comp_ctx->basic_types, wasm_type)
- #define I32_TYPE comp_ctx->basic_types.int32_type
- #define I64_TYPE comp_ctx->basic_types.int64_type
- #define F32_TYPE comp_ctx->basic_types.float32_type
- #define F64_TYPE comp_ctx->basic_types.float64_type
- #define VOID_TYPE comp_ctx->basic_types.void_type
- #define INT1_TYPE comp_ctx->basic_types.int1_type
- #define INT8_TYPE comp_ctx->basic_types.int8_type
- #define INT16_TYPE comp_ctx->basic_types.int16_type
- #define MD_TYPE comp_ctx->basic_types.meta_data_type
- #define INT8_PTR_TYPE comp_ctx->basic_types.int8_ptr_type
- #define INT16_PTR_TYPE comp_ctx->basic_types.int16_ptr_type
- #define INT32_PTR_TYPE comp_ctx->basic_types.int32_ptr_type
- #define INT64_PTR_TYPE comp_ctx->basic_types.int64_ptr_type
- #define F32_PTR_TYPE comp_ctx->basic_types.float32_ptr_type
- #define F64_PTR_TYPE comp_ctx->basic_types.float64_ptr_type
- #define FUNC_REF_TYPE comp_ctx->basic_types.funcref_type
- #define EXTERN_REF_TYPE comp_ctx->basic_types.externref_type
- #define I32_CONST(v) LLVMConstInt(I32_TYPE, v, true)
- #define I64_CONST(v) LLVMConstInt(I64_TYPE, v, true)
- #define F32_CONST(v) LLVMConstReal(F32_TYPE, v)
- #define F64_CONST(v) LLVMConstReal(F64_TYPE, v)
- #define I8_CONST(v) LLVMConstInt(INT8_TYPE, v, true)
- #define I8_ZERO (comp_ctx->llvm_consts.i8_zero)
- #define I32_ZERO (comp_ctx->llvm_consts.i32_zero)
- #define I64_ZERO (comp_ctx->llvm_consts.i64_zero)
- #define F32_ZERO (comp_ctx->llvm_consts.f32_zero)
- #define F64_ZERO (comp_ctx->llvm_consts.f64_zero)
- #define I32_ONE (comp_ctx->llvm_consts.i32_one)
- #define I32_TWO (comp_ctx->llvm_consts.i32_two)
- #define I32_THREE (comp_ctx->llvm_consts.i32_three)
- #define I32_FOUR (comp_ctx->llvm_consts.i32_four)
- #define I32_FIVE (comp_ctx->llvm_consts.i32_five)
- #define I32_SIX (comp_ctx->llvm_consts.i32_six)
- #define I32_SEVEN (comp_ctx->llvm_consts.i32_seven)
- #define I32_EIGHT (comp_ctx->llvm_consts.i32_eight)
- #define I32_NEG_ONE (comp_ctx->llvm_consts.i32_neg_one)
- #define I64_NEG_ONE (comp_ctx->llvm_consts.i64_neg_one)
- #define I32_MIN (comp_ctx->llvm_consts.i32_min)
- #define I64_MIN (comp_ctx->llvm_consts.i64_min)
- #define I32_31 (comp_ctx->llvm_consts.i32_31)
- #define I32_32 (comp_ctx->llvm_consts.i32_32)
- #define I64_63 (comp_ctx->llvm_consts.i64_63)
- #define I64_64 (comp_ctx->llvm_consts.i64_64)
- #define V128_TYPE comp_ctx->basic_types.v128_type
- #define V128_PTR_TYPE comp_ctx->basic_types.v128_ptr_type
- #define V128_i8x16_TYPE comp_ctx->basic_types.i8x16_vec_type
- #define V128_i16x8_TYPE comp_ctx->basic_types.i16x8_vec_type
- #define V128_i32x4_TYPE comp_ctx->basic_types.i32x4_vec_type
- #define V128_i64x2_TYPE comp_ctx->basic_types.i64x2_vec_type
- #define V128_f32x4_TYPE comp_ctx->basic_types.f32x4_vec_type
- #define V128_f64x2_TYPE comp_ctx->basic_types.f64x2_vec_type
- #define V128_ZERO (comp_ctx->llvm_consts.v128_zero)
- #define V128_i8x16_ZERO (comp_ctx->llvm_consts.i8x16_vec_zero)
- #define V128_i16x8_ZERO (comp_ctx->llvm_consts.i16x8_vec_zero)
- #define V128_i32x4_ZERO (comp_ctx->llvm_consts.i32x4_vec_zero)
- #define V128_i64x2_ZERO (comp_ctx->llvm_consts.i64x2_vec_zero)
- #define V128_f32x4_ZERO (comp_ctx->llvm_consts.f32x4_vec_zero)
- #define V128_f64x2_ZERO (comp_ctx->llvm_consts.f64x2_vec_zero)
- #define REF_NULL (comp_ctx->llvm_consts.i32_neg_one)
- #define TO_V128_i8x16(v) LLVMBuildBitCast(comp_ctx->builder, v, \
- V128_i8x16_TYPE, "i8x16_val")
- #define TO_V128_i16x8(v) LLVMBuildBitCast(comp_ctx->builder, v, \
- V128_i16x8_TYPE, "i16x8_val")
- #define TO_V128_i32x4(v) LLVMBuildBitCast(comp_ctx->builder, v, \
- V128_i32x4_TYPE, "i32x4_val")
- #define TO_V128_i64x2(v) LLVMBuildBitCast(comp_ctx->builder, v, \
- V128_i64x2_TYPE, "i64x2_val")
- #define TO_V128_f32x4(v) LLVMBuildBitCast(comp_ctx->builder, v, \
- V128_f32x4_TYPE, "f32x4_val")
- #define TO_V128_f64x2(v) LLVMBuildBitCast(comp_ctx->builder, v, \
- V128_f64x2_TYPE, "f64x2_val")
- #define CHECK_LLVM_CONST(v) do { \
- if (!v) { \
- aot_set_last_error("create llvm const failed."); \
- goto fail; \
- } \
- } while (0)
- #define GET_AOT_FUNCTION(name, argc) do { \
- if (!(func_type = LLVMFunctionType(ret_type, param_types, \
- argc, false))) { \
- aot_set_last_error("llvm add function type failed."); \
- return false; \
- } \
- if (comp_ctx->is_jit_mode) { \
- /* JIT mode, call the function directly */ \
- if (!(func_ptr_type = LLVMPointerType(func_type, 0))) { \
- aot_set_last_error("llvm add pointer type failed."); \
- return false; \
- } \
- if (!(value = I64_CONST((uint64)(uintptr_t)name)) \
- || !(func = LLVMConstIntToPtr(value, func_ptr_type))) { \
- aot_set_last_error("create LLVM value failed."); \
- return false; \
- } \
- } \
- else { \
- char *func_name = #name; \
- /* AOT mode, delcare the function */ \
- if (!(func = LLVMGetNamedFunction(comp_ctx->module, func_name)) \
- && !(func = LLVMAddFunction(comp_ctx->module, \
- func_name, func_type))) { \
- aot_set_last_error("llvm add function failed."); \
- return false; \
- } \
- } \
- } while (0)
- bool
- aot_compile_wasm(AOTCompContext *comp_ctx);
- bool
- aot_emit_llvm_file(AOTCompContext *comp_ctx, const char *file_name);
- bool
- aot_emit_aot_file(AOTCompContext *comp_ctx,
- AOTCompData *comp_data,
- const char *file_name);
- uint8*
- aot_emit_aot_file_buf(AOTCompContext *comp_ctx,
- AOTCompData *comp_data,
- uint32 *p_aot_file_size);
- bool
- aot_emit_object_file(AOTCompContext *comp_ctx, char *file_name);
- uint8*
- aot_compile_wasm_file(const uint8 *wasm_file_buf, uint32 wasm_file_size,
- uint32 opt_level, uint32 size_level,
- char *error_buf, uint32 error_buf_size,
- uint32 *p_aot_file_size);
- #ifdef __cplusplus
- } /* end of extern "C" */
- #endif
- #endif /* end of _AOT_COMPILER_H_ */
|