diff options
| author | Andrew Kelley <andrew@ziglang.org> | 2020-01-29 17:21:29 -0500 |
|---|---|---|
| committer | Andrew Kelley <andrew@ziglang.org> | 2020-01-29 17:21:29 -0500 |
| commit | 7ebc624a15c5a01d6bee8eaf9c7487b30ed1904c (patch) | |
| tree | e4db4340a18e9a116f7e39e48ca9dc990f811575 /src/ir.cpp | |
| parent | d448c3d38af9f8f70daaa2817a5834e30da4b8ca (diff) | |
| parent | f97b398b654aecdb679a054d9078b92cb70ba3b5 (diff) | |
| download | zig-7ebc624a15c5a01d6bee8eaf9c7487b30ed1904c.tar.gz zig-7ebc624a15c5a01d6bee8eaf9c7487b30ed1904c.zip | |
Merge branch 'LemonBoy-fix-4259'
Diffstat (limited to 'src/ir.cpp')
| -rw-r--r-- | src/ir.cpp | 83 |
1 files changed, 63 insertions, 20 deletions
diff --git a/src/ir.cpp b/src/ir.cpp index 149c8fbdc9..ef3426d111 100644 --- a/src/ir.cpp +++ b/src/ir.cpp @@ -11161,6 +11161,38 @@ void float_read_ieee597(ZigValue *val, uint8_t *buf, bool is_big_endian) { } } +static void value_to_bigfloat(BigFloat *out, ZigValue *val) { + switch (val->type->id) { + case ZigTypeIdInt: + case ZigTypeIdComptimeInt: + bigfloat_init_bigint(out, &val->data.x_bigint); + return; + case ZigTypeIdComptimeFloat: + *out = val->data.x_bigfloat; + return; + case ZigTypeIdFloat: switch (val->type->data.floating.bit_count) { + case 16: + bigfloat_init_16(out, val->data.x_f16); + return; + case 32: + bigfloat_init_32(out, val->data.x_f32); + return; + case 64: + bigfloat_init_64(out, val->data.x_f64); + return; + case 80: + zig_panic("TODO"); + case 128: + bigfloat_init_128(out, val->data.x_f128); + return; + default: + zig_unreachable(); + } + default: + zig_unreachable(); + } +} + static bool ir_num_lit_fits_in_other_type(IrAnalyze *ira, IrInstGen *instruction, ZigType *other_type, bool explicit_cast) { @@ -15815,33 +15847,44 @@ never_mind_just_calculate_it_normally: bool op1_is_int = op1_val->type->id == ZigTypeIdInt || op1_val->type->id == ZigTypeIdComptimeInt; bool op2_is_int = op2_val->type->id == ZigTypeIdInt || op2_val->type->id == ZigTypeIdComptimeInt; - BigInt *op1_bigint; - BigInt *op2_bigint; - bool need_to_free_op1_bigint = false; - bool need_to_free_op2_bigint = false; - if (op1_is_float) { - op1_bigint = allocate<BigInt>(1, "BigInt"); - need_to_free_op1_bigint = true; - float_init_bigint(op1_bigint, op1_val); - } else { - assert(op1_is_int); - op1_bigint = &op1_val->data.x_bigint; + if (op1_is_int && op2_is_int) { + Cmp cmp_result = bigint_cmp(&op1_val->data.x_bigint, &op2_val->data.x_bigint); + out_val->special = ConstValSpecialStatic; + out_val->data.x_bool = resolve_cmp_op_id(op_id, cmp_result); + + return nullptr; } - if (op2_is_float) { - op2_bigint = allocate<BigInt>(1, "BigInt"); - need_to_free_op2_bigint = true; - float_init_bigint(op2_bigint, op2_val); + + // Handle the case where one of the two operands is a fp value and the other + // is an integer value + ZigValue *float_val; + if (op1_is_int && op2_is_float) { + float_val = op2_val; + } else if (op1_is_float && op2_is_int) { + float_val = op1_val; } else { - assert(op2_is_int); - op2_bigint = &op2_val->data.x_bigint; + zig_unreachable(); } - Cmp cmp_result = bigint_cmp(op1_bigint, op2_bigint); + // They can never be equal if the fp value has a non-zero decimal part + if (op_id == IrBinOpCmpEq || op_id == IrBinOpCmpNotEq) { + if (float_has_fraction(float_val)) { + out_val->special = ConstValSpecialStatic; + out_val->data.x_bool = op_id == IrBinOpCmpNotEq; + return nullptr; + } + } + + // Cast the integer operand into a fp value to perform the comparison + BigFloat op1_bigfloat; + BigFloat op2_bigfloat; + value_to_bigfloat(&op1_bigfloat, op1_val); + value_to_bigfloat(&op2_bigfloat, op2_val); + + Cmp cmp_result = bigfloat_cmp(&op1_bigfloat, &op2_bigfloat); out_val->special = ConstValSpecialStatic; out_val->data.x_bool = resolve_cmp_op_id(op_id, cmp_result); - if (need_to_free_op1_bigint) destroy(op1_bigint, "BigInt"); - if (need_to_free_op2_bigint) destroy(op2_bigint, "BigInt"); return nullptr; } |
