mirror of
https://github.com/bytecodealliance/wasm-micro-runtime.git
synced 2025-02-11 09:25:20 +00:00
Implement float and integer conversion intrinsics (#718)
This commit is contained in:
parent
bb0e4503cb
commit
cef7d35be1
|
@ -58,6 +58,19 @@ static const aot_intrinsic g_intrinsic_mapping[] = {
|
|||
{ "llvm.cttz.i64", "aot_intrinsic_ctz_i64", AOT_INTRINSIC_FLAG_I64_CTZ },
|
||||
{ "llvm.ctpop.i32", "aot_intrinsic_popcnt_i32", AOT_INTRINSIC_FLAG_I32_POPCNT },
|
||||
{ "llvm.ctpop.i64", "aot_intrinsic_popcnt_i64", AOT_INTRINSIC_FLAG_I64_POPCNT },
|
||||
{ "f64_convert_i32_s", "aot_intrinsic_i32_to_f64", AOT_INTRINSIC_FLAG_I32_TO_F64},
|
||||
{ "f64_convert_i32_u", "aot_intrinsic_u32_to_f64", AOT_INTRINSIC_FLAG_U32_TO_F64},
|
||||
{ "f32_convert_i32_s", "aot_intrinsic_i32_to_f32", AOT_INTRINSIC_FLAG_I32_TO_F32},
|
||||
{ "f32_convert_i32_u", "aot_intrinsic_u32_to_f32", AOT_INTRINSIC_FLAG_U32_TO_F32},
|
||||
{ "f64_convert_i64_s", "aot_intrinsic_i64_to_f64", AOT_INTRINSIC_FLAG_I32_TO_F64},
|
||||
{ "f64_convert_i64_u", "aot_intrinsic_u64_to_f64", AOT_INTRINSIC_FLAG_U64_TO_F64},
|
||||
{ "f32_convert_i64_s", "aot_intrinsic_i64_to_f32", AOT_INTRINSIC_FLAG_I64_TO_F32},
|
||||
{ "f32_convert_i64_u", "aot_intrinsic_u64_to_f32", AOT_INTRINSIC_FLAG_U64_TO_F32},
|
||||
{ "i32_trunc_f64_u", "aot_intrinsic_f64_to_u32", AOT_INTRINSIC_FLAG_I32_TO_F64},
|
||||
{ "f32_demote_f64", "aot_intrinsic_f64_to_f32", AOT_INTRINSIC_FLAG_F64_TO_F32},
|
||||
{ "f64_promote_f32", "aot_intrinsic_f32_to_f64", AOT_INTRINSIC_FLAG_F32_TO_F64},
|
||||
{ "f32_cmp", "aot_intrinsic_f32_cmp", AOT_INTRINSIC_FLAG_F32_CMP},
|
||||
{ "f64_cmp", "aot_intrinsic_f64_cmp", AOT_INTRINSIC_FLAG_F64_CMP},
|
||||
};
|
||||
|
||||
static const uint32 g_intrinsic_count =
|
||||
|
@ -309,6 +322,170 @@ aot_intrinsic_popcnt_i64(uint64 u)
|
|||
return ret;
|
||||
}
|
||||
|
||||
float32
|
||||
aot_intrinsic_i32_to_f32(int32 i)
|
||||
{
|
||||
return (float32)i;
|
||||
}
|
||||
|
||||
float32
|
||||
aot_intrinsic_u32_to_f32(uint32 u)
|
||||
{
|
||||
return (float32)u;
|
||||
}
|
||||
|
||||
float64
|
||||
aot_intrinsic_i32_to_f64(int32 i)
|
||||
{
|
||||
return (float64)i;
|
||||
}
|
||||
|
||||
float64
|
||||
aot_intrinsic_u32_to_f64(uint32 u)
|
||||
{
|
||||
return (float64)u;
|
||||
}
|
||||
|
||||
float32
|
||||
aot_intrinsic_i64_to_f32(int64 i)
|
||||
{
|
||||
return (float32)i;
|
||||
}
|
||||
|
||||
float32
|
||||
aot_intrinsic_u64_to_f32(uint64 u)
|
||||
{
|
||||
return (float32)u;
|
||||
}
|
||||
|
||||
float64
|
||||
aot_intrinsic_i64_to_f64(int64 i)
|
||||
{
|
||||
return (float64)i;
|
||||
}
|
||||
|
||||
float64
|
||||
aot_intrinsic_u64_to_f64(uint64 u)
|
||||
{
|
||||
return (float64)u;
|
||||
}
|
||||
|
||||
int32
|
||||
aot_intrinsic_f32_to_i32(float32 f)
|
||||
{
|
||||
return (int32)f;
|
||||
}
|
||||
|
||||
uint32
|
||||
aot_intrinsic_f32_to_u32(float32 f)
|
||||
{
|
||||
return (uint32)f;
|
||||
}
|
||||
|
||||
int64
|
||||
aot_intrinsic_f32_to_i64(float32 f)
|
||||
{
|
||||
return (int64)f;
|
||||
}
|
||||
|
||||
uint64
|
||||
aot_intrinsic_f32_to_u64(float32 f)
|
||||
{
|
||||
return (uint64)f;
|
||||
}
|
||||
|
||||
int32
|
||||
aot_intrinsic_f64_to_i32(float64 f)
|
||||
{
|
||||
return (int32)f;
|
||||
}
|
||||
|
||||
uint32
|
||||
aot_intrinsic_f64_to_u32(float64 f)
|
||||
{
|
||||
return (uint32)f;
|
||||
}
|
||||
|
||||
int64
|
||||
aot_intrinsic_f64_to_i64(float64 f)
|
||||
{
|
||||
return (int64)f;
|
||||
}
|
||||
|
||||
uint64
|
||||
aot_intrinsic_f64_to_u64(float64 f)
|
||||
{
|
||||
return (uint64)f;
|
||||
}
|
||||
|
||||
float64
|
||||
aot_intrinsic_f32_to_f64(float32 f)
|
||||
{
|
||||
return (float64)f;
|
||||
}
|
||||
|
||||
float32
|
||||
aot_intrinsic_f64_to_f32(float64 f)
|
||||
{
|
||||
return (float32)f;
|
||||
}
|
||||
|
||||
int32
|
||||
aot_intrinsic_f32_cmp(AOTFloatCond cond, float32 lhs, float32 rhs)
|
||||
{
|
||||
switch (cond) {
|
||||
case FLOAT_LT:
|
||||
return lhs < rhs ? 1 : 0;
|
||||
|
||||
case FLOAT_GT:
|
||||
return lhs > rhs ? 1 : 0;
|
||||
|
||||
case FLOAT_LE:
|
||||
return lhs <= rhs ? 1 : 0;
|
||||
|
||||
case FLOAT_GE:
|
||||
return lhs >= rhs ? 1 : 0;
|
||||
|
||||
case FLOAT_NE:
|
||||
return (isnan(lhs) || isnan(rhs) || lhs != rhs) ? 1 : 0;
|
||||
|
||||
case FLOAT_UNO:
|
||||
return (isnan(lhs) || isnan(rhs)) ? 1 : 0;
|
||||
|
||||
default:
|
||||
break;
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
int32
|
||||
aot_intrinsic_f64_cmp(AOTFloatCond cond, float64 lhs, float64 rhs)
|
||||
{
|
||||
switch (cond) {
|
||||
case FLOAT_LT:
|
||||
return lhs < rhs ? 1 : 0;
|
||||
|
||||
case FLOAT_GT:
|
||||
return lhs > rhs ? 1 : 0;
|
||||
|
||||
case FLOAT_LE:
|
||||
return lhs <= rhs ? 1 : 0;
|
||||
|
||||
case FLOAT_GE:
|
||||
return lhs >= rhs ? 1 : 0;
|
||||
|
||||
case FLOAT_NE:
|
||||
return (isnan(lhs) || isnan(rhs) || lhs != rhs) ? 1 : 0;
|
||||
|
||||
case FLOAT_UNO:
|
||||
return (isnan(lhs) || isnan(rhs)) ? 1 : 0;
|
||||
|
||||
default:
|
||||
break;
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
const char *
|
||||
aot_intrinsic_get_symbol(const char *llvm_intrinsic)
|
||||
{
|
||||
|
@ -345,6 +522,7 @@ add_f32_common_intrinsics_for_thumb2_fpu(AOTCompContext *comp_ctx)
|
|||
add_intrinsic_capability(comp_ctx, AOT_INTRINSIC_FLAG_F32_FMUL);
|
||||
add_intrinsic_capability(comp_ctx, AOT_INTRINSIC_FLAG_F32_FDIV);
|
||||
add_intrinsic_capability(comp_ctx, AOT_INTRINSIC_FLAG_F32_SQRT);
|
||||
add_intrinsic_capability(comp_ctx, AOT_INTRINSIC_FLAG_F32_CMP);
|
||||
}
|
||||
|
||||
static void
|
||||
|
@ -356,6 +534,34 @@ add_f64_common_intrinsics_for_thumb2_fpu(AOTCompContext *comp_ctx)
|
|||
add_intrinsic_capability(comp_ctx, AOT_INTRINSIC_FLAG_F64_FMUL);
|
||||
add_intrinsic_capability(comp_ctx, AOT_INTRINSIC_FLAG_F64_FDIV);
|
||||
add_intrinsic_capability(comp_ctx, AOT_INTRINSIC_FLAG_F64_SQRT);
|
||||
add_intrinsic_capability(comp_ctx, AOT_INTRINSIC_FLAG_F64_CMP);
|
||||
}
|
||||
|
||||
static void
|
||||
add_common_float_integer_convertion(AOTCompContext *comp_ctx)
|
||||
{
|
||||
add_intrinsic_capability(comp_ctx, AOT_INTRINSIC_FLAG_I32_TO_F32);
|
||||
add_intrinsic_capability(comp_ctx, AOT_INTRINSIC_FLAG_U32_TO_F32);
|
||||
add_intrinsic_capability(comp_ctx, AOT_INTRINSIC_FLAG_I32_TO_F64);
|
||||
add_intrinsic_capability(comp_ctx, AOT_INTRINSIC_FLAG_U32_TO_F64);
|
||||
|
||||
add_intrinsic_capability(comp_ctx, AOT_INTRINSIC_FLAG_I64_TO_F32);
|
||||
add_intrinsic_capability(comp_ctx, AOT_INTRINSIC_FLAG_U64_TO_F32);
|
||||
add_intrinsic_capability(comp_ctx, AOT_INTRINSIC_FLAG_I64_TO_F64);
|
||||
add_intrinsic_capability(comp_ctx, AOT_INTRINSIC_FLAG_U64_TO_F64);
|
||||
|
||||
add_intrinsic_capability(comp_ctx, AOT_INTRINSIC_FLAG_F32_TO_I32);
|
||||
add_intrinsic_capability(comp_ctx, AOT_INTRINSIC_FLAG_F32_TO_U32);
|
||||
add_intrinsic_capability(comp_ctx, AOT_INTRINSIC_FLAG_F32_TO_I64);
|
||||
add_intrinsic_capability(comp_ctx, AOT_INTRINSIC_FLAG_F32_TO_U64);
|
||||
|
||||
add_intrinsic_capability(comp_ctx, AOT_INTRINSIC_FLAG_F64_TO_I32);
|
||||
add_intrinsic_capability(comp_ctx, AOT_INTRINSIC_FLAG_F64_TO_U32);
|
||||
add_intrinsic_capability(comp_ctx, AOT_INTRINSIC_FLAG_F64_TO_I64);
|
||||
add_intrinsic_capability(comp_ctx, AOT_INTRINSIC_FLAG_F64_TO_U64);
|
||||
|
||||
add_intrinsic_capability(comp_ctx, AOT_INTRINSIC_FLAG_F64_TO_F32);
|
||||
add_intrinsic_capability(comp_ctx, AOT_INTRINSIC_FLAG_F32_TO_F64);
|
||||
}
|
||||
|
||||
bool
|
||||
|
@ -401,6 +607,7 @@ aot_intrinsic_fill_capability_flags(AOTCompContext *comp_ctx)
|
|||
else {
|
||||
add_f32_common_intrinsics_for_thumb2_fpu(comp_ctx);
|
||||
add_f64_common_intrinsics_for_thumb2_fpu(comp_ctx);
|
||||
add_common_float_integer_convertion(comp_ctx);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
|
@ -47,6 +47,16 @@ extern "C" {
|
|||
#define AOT_INTRINSIC_FLAG_I32_CLZ AOT_INTRINSIC_FLAG(0, 13)
|
||||
#define AOT_INTRINSIC_FLAG_I32_CTZ AOT_INTRINSIC_FLAG(0, 14)
|
||||
#define AOT_INTRINSIC_FLAG_I32_POPCNT AOT_INTRINSIC_FLAG(0, 15)
|
||||
#define AOT_INTRINSIC_FLAG_I32_TO_F32 AOT_INTRINSIC_FLAG(0, 16)
|
||||
#define AOT_INTRINSIC_FLAG_U32_TO_F32 AOT_INTRINSIC_FLAG(0, 17)
|
||||
#define AOT_INTRINSIC_FLAG_I32_TO_F64 AOT_INTRINSIC_FLAG(0, 18)
|
||||
#define AOT_INTRINSIC_FLAG_U32_TO_F64 AOT_INTRINSIC_FLAG(0, 19)
|
||||
#define AOT_INTRINSIC_FLAG_F32_TO_I32 AOT_INTRINSIC_FLAG(0, 20)
|
||||
#define AOT_INTRINSIC_FLAG_F32_TO_U32 AOT_INTRINSIC_FLAG(0, 21)
|
||||
#define AOT_INTRINSIC_FLAG_F32_TO_I64 AOT_INTRINSIC_FLAG(0, 22)
|
||||
#define AOT_INTRINSIC_FLAG_F32_TO_U64 AOT_INTRINSIC_FLAG(0, 23)
|
||||
#define AOT_INTRINSIC_FLAG_F32_TO_F64 AOT_INTRINSIC_FLAG(0, 24)
|
||||
#define AOT_INTRINSIC_FLAG_F32_CMP AOT_INTRINSIC_FLAG(0, 25)
|
||||
|
||||
#define AOT_INTRINSIC_FLAG_F64_FADD AOT_INTRINSIC_FLAG(1, 0)
|
||||
#define AOT_INTRINSIC_FLAG_F64_FSUB AOT_INTRINSIC_FLAG(1, 1)
|
||||
|
@ -64,6 +74,16 @@ extern "C" {
|
|||
#define AOT_INTRINSIC_FLAG_I64_CLZ AOT_INTRINSIC_FLAG(1, 13)
|
||||
#define AOT_INTRINSIC_FLAG_I64_CTZ AOT_INTRINSIC_FLAG(1, 14)
|
||||
#define AOT_INTRINSIC_FLAG_I64_POPCNT AOT_INTRINSIC_FLAG(1, 15)
|
||||
#define AOT_INTRINSIC_FLAG_I64_TO_F32 AOT_INTRINSIC_FLAG(1, 16)
|
||||
#define AOT_INTRINSIC_FLAG_U64_TO_F32 AOT_INTRINSIC_FLAG(1, 17)
|
||||
#define AOT_INTRINSIC_FLAG_I64_TO_F64 AOT_INTRINSIC_FLAG(1, 18)
|
||||
#define AOT_INTRINSIC_FLAG_U64_TO_F64 AOT_INTRINSIC_FLAG(1, 19)
|
||||
#define AOT_INTRINSIC_FLAG_F64_TO_I32 AOT_INTRINSIC_FLAG(1, 20)
|
||||
#define AOT_INTRINSIC_FLAG_F64_TO_U32 AOT_INTRINSIC_FLAG(1, 21)
|
||||
#define AOT_INTRINSIC_FLAG_F64_TO_I64 AOT_INTRINSIC_FLAG(1, 22)
|
||||
#define AOT_INTRINSIC_FLAG_F64_TO_U64 AOT_INTRINSIC_FLAG(1, 23)
|
||||
#define AOT_INTRINSIC_FLAG_F64_TO_F32 AOT_INTRINSIC_FLAG(1, 24)
|
||||
#define AOT_INTRINSIC_FLAG_F64_CMP AOT_INTRINSIC_FLAG(1, 25)
|
||||
|
||||
float32
|
||||
aot_intrinsic_fadd_f32(float32 a, float32 b);
|
||||
|
@ -161,6 +181,66 @@ aot_intrinsic_popcnt_i32(uint32 u);
|
|||
uint32
|
||||
aot_intrinsic_popcnt_i64(uint64 u);
|
||||
|
||||
float32
|
||||
aot_intrinsic_i32_to_f32(int32 i);
|
||||
|
||||
float32
|
||||
aot_intrinsic_u32_to_f32(uint32 u);
|
||||
|
||||
float64
|
||||
aot_intrinsic_i32_to_f64(int32 i);
|
||||
|
||||
float64
|
||||
aot_intrinsic_u32_to_f64(uint32 u);
|
||||
|
||||
float32
|
||||
aot_intrinsic_i64_to_f32(int64 i);
|
||||
|
||||
float32
|
||||
aot_intrinsic_u64_to_f32(uint64 u);
|
||||
|
||||
float64
|
||||
aot_intrinsic_i64_to_f64(int64 i);
|
||||
|
||||
float64
|
||||
aot_intrinsic_u64_to_f64(uint64 u);
|
||||
|
||||
int32
|
||||
aot_intrinsic_f32_to_i32(float32 f);
|
||||
|
||||
uint32
|
||||
aot_intrinsic_f32_to_u32(float32 f);
|
||||
|
||||
int64
|
||||
aot_intrinsic_f32_to_i64(float32 f);
|
||||
|
||||
uint64
|
||||
aot_intrinsic_f32_to_u64(float32 f);
|
||||
|
||||
int32
|
||||
aot_intrinsic_f64_to_i32(float64 f);
|
||||
|
||||
uint32
|
||||
aot_intrinsic_f64_to_u32(float64 f);
|
||||
|
||||
int64
|
||||
aot_intrinsic_f64_to_i64(float64 f);
|
||||
|
||||
uint64
|
||||
aot_intrinsic_f64_to_u64(float64 f);
|
||||
|
||||
float64
|
||||
aot_intrinsic_f32_to_f64(float32 f);
|
||||
|
||||
float32
|
||||
aot_intrinsic_f64_to_f32(float64 f);
|
||||
|
||||
int32
|
||||
aot_intrinsic_f32_cmp(AOTFloatCond cond, float32 lhs, float32 rhs);
|
||||
|
||||
int32
|
||||
aot_intrinsic_f64_cmp(AOTFloatCond cond, float64 lhs, float64 rhs);
|
||||
|
||||
const char *
|
||||
aot_intrinsic_get_symbol(const char *llvm_intrinsic);
|
||||
|
||||
|
|
|
@ -82,6 +82,19 @@ typedef struct {
|
|||
REG_SYM(aot_intrinsic_ctz_i64), \
|
||||
REG_SYM(aot_intrinsic_popcnt_i32), \
|
||||
REG_SYM(aot_intrinsic_popcnt_i64), \
|
||||
REG_SYM(aot_intrinsic_i32_to_f32), \
|
||||
REG_SYM(aot_intrinsic_u32_to_f32), \
|
||||
REG_SYM(aot_intrinsic_i32_to_f64), \
|
||||
REG_SYM(aot_intrinsic_u32_to_f64), \
|
||||
REG_SYM(aot_intrinsic_i64_to_f32), \
|
||||
REG_SYM(aot_intrinsic_u64_to_f32), \
|
||||
REG_SYM(aot_intrinsic_i64_to_f64), \
|
||||
REG_SYM(aot_intrinsic_u64_to_f64), \
|
||||
REG_SYM(aot_intrinsic_f64_to_f32), \
|
||||
REG_SYM(aot_intrinsic_f64_to_u32), \
|
||||
REG_SYM(aot_intrinsic_f32_to_f64), \
|
||||
REG_SYM(aot_intrinsic_f32_cmp), \
|
||||
REG_SYM(aot_intrinsic_f64_cmp), \
|
||||
|
||||
#define REG_COMMON_SYMBOLS \
|
||||
REG_SYM(aot_set_exception_with_id), \
|
||||
|
|
|
@ -41,7 +41,8 @@ typedef enum AOTFloatCond {
|
|||
FLOAT_LT,
|
||||
FLOAT_GT,
|
||||
FLOAT_LE,
|
||||
FLOAT_GE
|
||||
FLOAT_GE,
|
||||
FLOAT_UNO
|
||||
} AOTFloatCond;
|
||||
|
||||
/**
|
||||
|
|
Loading…
Reference in New Issue
Block a user