| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709 |
- /*
- * Copyright (C) 2019 Intel Corporation. All rights reserved.
- * SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
- */
- #include "jit_emit_conversion.h"
- #include "jit_emit_exception.h"
- #include "jit_emit_function.h"
- #include "../jit_codegen.h"
- #include "../jit_frontend.h"
- #define F32_I32_S_MIN (-2147483904.0f)
- #define F32_I32_S_MAX (2147483648.0f)
- #define F32_I32_U_MIN (-1.0f)
- #define F32_I32_U_MAX (4294967296.0f)
- #define F32_I64_S_MIN (-9223373136366403584.0f)
- #define F32_I64_S_MAX (9223372036854775808.0f)
- #define F32_I64_U_MIN (-1.0f)
- #define F32_I64_U_MAX (18446744073709551616.0f)
- #define F64_I32_S_MIN (-2147483649.0)
- #define F64_I32_S_MAX (2147483648.0)
- #define F64_I32_U_MIN (-1.0)
- #define F64_I32_U_MAX (4294967296.0)
- #define F64_I64_S_MIN (-9223372036854777856.0)
- #define F64_I64_S_MAX (9223372036854775808.0)
- #define F64_I64_U_MIN (-1.0)
- #define F64_I64_U_MAX (18446744073709551616.0)
- #define FP_TO_INT(f_ty, i_ty, f_nm, i_nm) \
- static i_ty i_nm##_trunc_##f_nm(f_ty fp)
- #define INT_TO_FP(i_ty, f_ty, i_nm, f_nm) \
- static f_ty f_nm##_convert_##i_nm(i_ty i)
- #define FP_TO_INT_SAT(f_ty, i_ty, f_nm, i_nm) \
- static i_ty i_nm##_trunc_##f_nm##_sat(f_ty fp)
- static int
- local_isnan(double x)
- {
- return isnan(x);
- }
- static int
- local_isnanf(float x)
- {
- return isnan(x);
- }
- #define RETURN_IF_NANF(fp) \
- if (local_isnanf(fp)) { \
- return 0; \
- }
- #define RETURN_IF_NAN(fp) \
- if (local_isnan(fp)) { \
- return 0; \
- }
- #define RETURN_IF_INF(fp, i_min, i_max) \
- if (isinf(fp)) { \
- return fp < 0 ? i_min : i_max; \
- }
- #define RETURN_IF_MIN(fp, f_min, i_min) \
- if (fp <= f_min) { \
- return i_min; \
- }
- #define RETURN_IF_MAX(fp, f_max, i_max) \
- if (fp >= f_max) { \
- return i_max; \
- }
- FP_TO_INT_SAT(float, int32, f32, i32)
- {
- RETURN_IF_NANF(fp)
- RETURN_IF_INF(fp, INT32_MIN, INT32_MAX)
- RETURN_IF_MIN(fp, F32_I32_S_MIN, INT32_MIN)
- RETURN_IF_MAX(fp, F32_I32_S_MAX, INT32_MAX)
- return (int32)fp;
- }
- FP_TO_INT_SAT(float, uint32, f32, u32)
- {
- RETURN_IF_NANF(fp)
- RETURN_IF_INF(fp, 0, UINT32_MAX)
- RETURN_IF_MIN(fp, F32_I32_U_MIN, 0)
- RETURN_IF_MAX(fp, F32_I32_U_MAX, UINT32_MAX)
- return (uint32)fp;
- }
- FP_TO_INT_SAT(double, int32, f64, i32)
- {
- RETURN_IF_NAN(fp)
- RETURN_IF_INF(fp, INT32_MIN, INT32_MAX)
- RETURN_IF_MIN(fp, F64_I32_S_MIN, INT32_MIN)
- RETURN_IF_MAX(fp, F64_I32_S_MAX, INT32_MAX)
- return (int32)fp;
- }
- FP_TO_INT_SAT(double, uint32, f64, u32)
- {
- RETURN_IF_NAN(fp)
- RETURN_IF_INF(fp, 0, UINT32_MAX)
- RETURN_IF_MIN(fp, F64_I32_U_MIN, 0)
- RETURN_IF_MAX(fp, F64_I32_U_MAX, UINT32_MAX)
- return (uint32)fp;
- }
- FP_TO_INT_SAT(float, int64, f32, i64)
- {
- RETURN_IF_NANF(fp)
- RETURN_IF_INF(fp, INT64_MIN, INT64_MAX)
- RETURN_IF_MIN(fp, F32_I64_S_MIN, INT64_MIN)
- RETURN_IF_MAX(fp, F32_I64_S_MAX, INT64_MAX)
- return (int64)fp;
- }
- FP_TO_INT(float, uint64, f32, u64)
- {
- return (uint64)fp;
- }
- FP_TO_INT_SAT(float, uint64, f32, u64)
- {
- RETURN_IF_NANF(fp)
- RETURN_IF_INF(fp, 0, UINT64_MAX)
- RETURN_IF_MIN(fp, F32_I64_U_MIN, 0)
- RETURN_IF_MAX(fp, F32_I64_U_MAX, UINT64_MAX)
- return (uint64)fp;
- }
- FP_TO_INT_SAT(double, int64, f64, i64)
- {
- RETURN_IF_NANF(fp)
- RETURN_IF_INF(fp, INT64_MIN, INT64_MAX)
- RETURN_IF_MIN(fp, F64_I64_S_MIN, INT64_MIN)
- RETURN_IF_MAX(fp, F64_I64_S_MAX, INT64_MAX)
- return (int64)fp;
- }
- FP_TO_INT(double, uint64, f64, u64)
- {
- return (uint64)fp;
- }
- FP_TO_INT_SAT(double, uint64, f64, u64)
- {
- RETURN_IF_NANF(fp)
- RETURN_IF_INF(fp, 0, UINT64_MAX)
- RETURN_IF_MIN(fp, F64_I64_U_MIN, 0)
- RETURN_IF_MAX(fp, F64_I64_U_MAX, UINT64_MAX)
- return (uint64)fp;
- }
- INT_TO_FP(uint64, float, u64, f32)
- {
- return (float)i;
- }
- INT_TO_FP(uint64, double, u64, f64)
- {
- return (double)i;
- }
- bool
- jit_compile_op_i32_wrap_i64(JitCompContext *cc)
- {
- JitReg num, res;
- POP_I64(num);
- res = jit_cc_new_reg_I32(cc);
- GEN_INSN(I64TOI32, res, num);
- PUSH_I32(res);
- return true;
- fail:
- return false;
- }
- static bool
- jit_compile_check_value_range(JitCompContext *cc, JitReg value, JitReg min_fp,
- JitReg max_fp)
- {
- JitReg nan_ret = jit_cc_new_reg_I32(cc);
- JitRegKind kind = jit_reg_kind(value);
- bool emit_ret = false;
- bh_assert(JIT_REG_KIND_F32 == kind || JIT_REG_KIND_F64 == kind);
- if (JIT_REG_KIND_F32 == kind && jit_reg_is_const(value)) {
- /* value is an f32 const */
- float value_f32_const = jit_cc_get_const_F32(cc, value);
- float min_fp_f32_const = jit_cc_get_const_F32(cc, min_fp);
- float max_fp_f32_const = jit_cc_get_const_F32(cc, max_fp);
- if (isnan(value_f32_const)) {
- /* throw exception if value is nan */
- if (!jit_emit_exception(cc, EXCE_INVALID_CONVERSION_TO_INTEGER,
- JIT_OP_JMP, 0, NULL))
- goto fail;
- }
- if (value_f32_const <= min_fp_f32_const
- || value_f32_const >= max_fp_f32_const) {
- /* throw exception if value is out of range */
- if (!jit_emit_exception(cc, EXCE_INTEGER_OVERFLOW, JIT_OP_JMP, 0,
- NULL))
- goto fail;
- }
- /* value is in range, do nothing */
- return true;
- }
- else if (JIT_REG_KIND_F64 == kind && jit_reg_is_const(value)) {
- /* value is an f64 const */
- double value_f64_const = jit_cc_get_const_F64(cc, value);
- double min_fp_f64_const = jit_cc_get_const_F64(cc, min_fp);
- double max_fp_f64_const = jit_cc_get_const_F64(cc, max_fp);
- if (isnan(value_f64_const)) {
- /* throw exception if value is nan */
- if (!jit_emit_exception(cc, EXCE_INVALID_CONVERSION_TO_INTEGER,
- JIT_OP_JMP, 0, NULL))
- goto fail;
- }
- if (value_f64_const <= min_fp_f64_const
- || value_f64_const >= max_fp_f64_const) {
- /* throw exception if value is out of range */
- if (!jit_emit_exception(cc, EXCE_INTEGER_OVERFLOW, JIT_OP_JMP, 0,
- NULL))
- goto fail;
- }
- /* value is in range, do nothing */
- return true;
- }
- /* If value is NaN, throw exception */
- if (JIT_REG_KIND_F32 == kind)
- emit_ret = jit_emit_callnative(cc, local_isnanf, nan_ret, &value, 1);
- else
- emit_ret = jit_emit_callnative(cc, local_isnan, nan_ret, &value, 1);
- if (!emit_ret)
- goto fail;
- GEN_INSN(CMP, cc->cmp_reg, nan_ret, NEW_CONST(I32, 1));
- if (!jit_emit_exception(cc, EXCE_INVALID_CONVERSION_TO_INTEGER, JIT_OP_BEQ,
- cc->cmp_reg, NULL))
- goto fail;
- /* If value is out of integer range, throw exception */
- GEN_INSN(CMP, cc->cmp_reg, min_fp, value);
- if (!jit_emit_exception(cc, EXCE_INTEGER_OVERFLOW, JIT_OP_BGES, cc->cmp_reg,
- NULL))
- goto fail;
- GEN_INSN(CMP, cc->cmp_reg, value, max_fp);
- if (!jit_emit_exception(cc, EXCE_INTEGER_OVERFLOW, JIT_OP_BGES, cc->cmp_reg,
- NULL))
- goto fail;
- return true;
- fail:
- return false;
- }
- bool
- jit_compile_op_i32_trunc_f32(JitCompContext *cc, bool sign, bool sat)
- {
- JitReg value, res;
- POP_F32(value);
- res = jit_cc_new_reg_I32(cc);
- if (!sat) {
- JitReg min_fp = NEW_CONST(F32, sign ? F32_I32_S_MIN : F32_I32_U_MIN);
- JitReg max_fp = NEW_CONST(F32, sign ? F32_I32_S_MAX : F32_I32_U_MAX);
- if (!jit_compile_check_value_range(cc, value, min_fp, max_fp))
- goto fail;
- if (sign)
- GEN_INSN(F32TOI32, res, value);
- else
- GEN_INSN(F32TOU32, res, value);
- }
- else {
- if (!jit_emit_callnative(cc,
- sign ? (void *)i32_trunc_f32_sat
- : (void *)u32_trunc_f32_sat,
- res, &value, 1))
- goto fail;
- }
- PUSH_I32(res);
- return true;
- fail:
- return false;
- }
- bool
- jit_compile_op_i32_trunc_f64(JitCompContext *cc, bool sign, bool sat)
- {
- JitReg value, res;
- POP_F64(value);
- res = jit_cc_new_reg_I32(cc);
- if (!sat) {
- JitReg min_fp = NEW_CONST(F64, sign ? F64_I32_S_MIN : F64_I32_U_MIN);
- JitReg max_fp = NEW_CONST(F64, sign ? F64_I32_S_MAX : F64_I32_U_MAX);
- if (!jit_compile_check_value_range(cc, value, min_fp, max_fp))
- goto fail;
- if (sign)
- GEN_INSN(F64TOI32, res, value);
- else
- GEN_INSN(F64TOU32, res, value);
- }
- else {
- if (!jit_emit_callnative(cc,
- sign ? (void *)i32_trunc_f64_sat
- : (void *)u32_trunc_f64_sat,
- res, &value, 1))
- goto fail;
- }
- PUSH_I32(res);
- return true;
- fail:
- return false;
- }
- bool
- jit_compile_op_i64_extend_i32(JitCompContext *cc, bool sign)
- {
- JitReg num, res;
- POP_I32(num);
- res = jit_cc_new_reg_I64(cc);
- if (sign)
- GEN_INSN(I32TOI64, res, num);
- else
- GEN_INSN(U32TOI64, res, num);
- PUSH_I64(res);
- return true;
- fail:
- return false;
- }
- bool
- jit_compile_op_i64_extend_i64(JitCompContext *cc, int8 bitwidth)
- {
- JitReg value, tmp, res;
- POP_I64(value);
- tmp = jit_cc_new_reg_I32(cc);
- res = jit_cc_new_reg_I64(cc);
- switch (bitwidth) {
- case 8:
- {
- GEN_INSN(I64TOI8, tmp, value);
- GEN_INSN(I8TOI64, res, tmp);
- break;
- }
- case 16:
- {
- GEN_INSN(I64TOI16, tmp, value);
- GEN_INSN(I16TOI64, res, tmp);
- break;
- }
- case 32:
- {
- GEN_INSN(I64TOI32, tmp, value);
- GEN_INSN(I32TOI64, res, tmp);
- break;
- }
- default:
- {
- bh_assert(0);
- goto fail;
- }
- }
- PUSH_I64(res);
- return true;
- fail:
- return false;
- }
- bool
- jit_compile_op_i32_extend_i32(JitCompContext *cc, int8 bitwidth)
- {
- JitReg value, tmp, res;
- POP_I32(value);
- tmp = jit_cc_new_reg_I32(cc);
- res = jit_cc_new_reg_I32(cc);
- switch (bitwidth) {
- case 8:
- {
- GEN_INSN(I32TOI8, tmp, value);
- GEN_INSN(I8TOI32, res, tmp);
- break;
- }
- case 16:
- {
- GEN_INSN(I32TOI16, tmp, value);
- GEN_INSN(I16TOI32, res, tmp);
- break;
- }
- default:
- {
- bh_assert(0);
- goto fail;
- }
- }
- PUSH_I32(res);
- return true;
- fail:
- return false;
- }
- bool
- jit_compile_op_i64_trunc_f32(JitCompContext *cc, bool sign, bool sat)
- {
- JitReg value, res;
- POP_F32(value);
- res = jit_cc_new_reg_I64(cc);
- if (!sat) {
- JitReg min_fp = NEW_CONST(F32, sign ? F32_I64_S_MIN : F32_I64_U_MIN);
- JitReg max_fp = NEW_CONST(F32, sign ? F32_I64_S_MAX : F32_I64_U_MAX);
- if (!jit_compile_check_value_range(cc, value, min_fp, max_fp))
- goto fail;
- if (sign) {
- GEN_INSN(F32TOI64, res, value);
- }
- else {
- if (!jit_emit_callnative(cc, u64_trunc_f32, res, &value, 1))
- goto fail;
- }
- }
- else {
- if (!jit_emit_callnative(cc,
- sign ? (void *)i64_trunc_f32_sat
- : (void *)u64_trunc_f32_sat,
- res, &value, 1))
- goto fail;
- }
- PUSH_I64(res);
- return true;
- fail:
- return false;
- }
- bool
- jit_compile_op_i64_trunc_f64(JitCompContext *cc, bool sign, bool sat)
- {
- JitReg value, res;
- POP_F64(value);
- res = jit_cc_new_reg_I64(cc);
- if (!sat) {
- JitReg min_fp = NEW_CONST(F64, sign ? F64_I64_S_MIN : F64_I64_U_MIN);
- JitReg max_fp = NEW_CONST(F64, sign ? F64_I64_S_MAX : F64_I64_U_MAX);
- if (!jit_compile_check_value_range(cc, value, min_fp, max_fp))
- goto fail;
- if (sign) {
- GEN_INSN(F64TOI64, res, value);
- }
- else {
- if (!jit_emit_callnative(cc, u64_trunc_f64, res, &value, 1))
- goto fail;
- }
- }
- else {
- if (!jit_emit_callnative(cc,
- sign ? (void *)i64_trunc_f64_sat
- : (void *)u64_trunc_f64_sat,
- res, &value, 1))
- goto fail;
- }
- PUSH_I64(res);
- return true;
- fail:
- return false;
- }
- bool
- jit_compile_op_f32_convert_i32(JitCompContext *cc, bool sign)
- {
- JitReg value, res;
- POP_I32(value);
- res = jit_cc_new_reg_F32(cc);
- if (sign) {
- GEN_INSN(I32TOF32, res, value);
- }
- else {
- GEN_INSN(U32TOF32, res, value);
- }
- PUSH_F32(res);
- return true;
- fail:
- return false;
- }
- bool
- jit_compile_op_f32_convert_i64(JitCompContext *cc, bool sign)
- {
- JitReg value, res;
- POP_I64(value);
- res = jit_cc_new_reg_F32(cc);
- if (sign) {
- GEN_INSN(I64TOF32, res, value);
- }
- else {
- if (!jit_emit_callnative(cc, f32_convert_u64, res, &value, 1)) {
- goto fail;
- }
- }
- PUSH_F32(res);
- return true;
- fail:
- return false;
- }
- bool
- jit_compile_op_f32_demote_f64(JitCompContext *cc)
- {
- JitReg value, res;
- POP_F64(value);
- res = jit_cc_new_reg_F32(cc);
- GEN_INSN(F64TOF32, res, value);
- PUSH_F32(res);
- return true;
- fail:
- return false;
- }
- bool
- jit_compile_op_f64_convert_i32(JitCompContext *cc, bool sign)
- {
- JitReg value, res;
- POP_I32(value);
- res = jit_cc_new_reg_F64(cc);
- if (sign)
- GEN_INSN(I32TOF64, res, value);
- else
- GEN_INSN(U32TOF64, res, value);
- PUSH_F64(res);
- return true;
- fail:
- return false;
- }
- bool
- jit_compile_op_f64_convert_i64(JitCompContext *cc, bool sign)
- {
- JitReg value, res;
- POP_I64(value);
- res = jit_cc_new_reg_F64(cc);
- if (sign) {
- GEN_INSN(I64TOF64, res, value);
- }
- else {
- if (!jit_emit_callnative(cc, f64_convert_u64, res, &value, 1)) {
- goto fail;
- }
- }
- PUSH_F64(res);
- return true;
- fail:
- return false;
- }
- bool
- jit_compile_op_f64_promote_f32(JitCompContext *cc)
- {
- JitReg value, res;
- POP_F32(value);
- res = jit_cc_new_reg_F64(cc);
- GEN_INSN(F32TOF64, res, value);
- PUSH_F64(res);
- return true;
- fail:
- return false;
- }
- bool
- jit_compile_op_i64_reinterpret_f64(JitCompContext *cc)
- {
- JitReg value, res;
- POP_F64(value);
- res = jit_cc_new_reg_I64(cc);
- GEN_INSN(F64CASTI64, res, value);
- PUSH_I64(res);
- return true;
- fail:
- return false;
- }
- bool
- jit_compile_op_i32_reinterpret_f32(JitCompContext *cc)
- {
- JitReg value, res;
- POP_F32(value);
- res = jit_cc_new_reg_I32(cc);
- GEN_INSN(F32CASTI32, res, value);
- PUSH_I32(res);
- return true;
- fail:
- return false;
- }
- bool
- jit_compile_op_f64_reinterpret_i64(JitCompContext *cc)
- {
- JitReg value, res;
- POP_I64(value);
- res = jit_cc_new_reg_F64(cc);
- GEN_INSN(I64CASTF64, res, value);
- PUSH_F64(res);
- return true;
- fail:
- return false;
- }
- bool
- jit_compile_op_f32_reinterpret_i32(JitCompContext *cc)
- {
- JitReg value, res;
- POP_I32(value);
- res = jit_cc_new_reg_F32(cc);
- GEN_INSN(I32CASTF32, res, value);
- PUSH_F32(res);
- return true;
- fail:
- return false;
- }
|