Commit dd89297388

Vexu <git@vexu.eu>
2020-07-21 20:43:40
stage2: actually implement float casting
1 parent c29c79b
Changed files (2)
src-self-hosted
src-self-hosted/Module.zig
@@ -3437,7 +3437,16 @@ fn coerce(self: *Module, scope: *Scope, dest_type: Type, inst: *Inst) !*Inst {
             }
         } else if (dst_zig_tag == .ComptimeFloat or dst_zig_tag == .Float) {
             if (src_zig_tag == .Float or src_zig_tag == .ComptimeFloat) {
-                return self.fail(scope, inst.src, "TODO float cast", .{});
+                const res = val.floatCast(scope.arena(), dest_type, self.target()) catch |err| switch (err) {
+                    error.Overflow => return self.fail(
+                        scope,
+                        inst.src,
+                        "cast of value {} to type '{}' loses information",
+                        .{ val, dest_type },
+                    ),
+                    error.OutOfMemory => return error.OutOfMemory,
+                };
+                return self.constInst(scope, inst.src, .{ .ty = dest_type, .val = res });
             } else if (src_zig_tag == .Int or src_zig_tag == .ComptimeInt) {
                 return self.fail(scope, inst.src, "TODO int to float", .{});
             }
src-self-hosted/value.zig
@@ -563,15 +563,15 @@ pub const Value = extern union {
     }
 
     /// Asserts that the value is a float or an integer.
-    pub fn toF128(self: Value) f128 {
+    pub fn toFloat(self: Value, comptime T: type) T {
         return switch (self.tag()) {
             .float_16 => @panic("TODO soft float"),
-            .float_32 => self.cast(Payload.Float_32).?.val,
-            .float_64 => self.cast(Payload.Float_64).?.val,
-            .float_128 => self.cast(Payload.Float_128).?.val,
+            .float_32 => @floatCast(T, self.cast(Payload.Float_32).?.val),
+            .float_64 => @floatCast(T, self.cast(Payload.Float_64).?.val),
+            .float_128 => @floatCast(T, self.cast(Payload.Float_128).?.val),
 
             .zero, .the_one_possible_value => 0,
-            .int_u64 => @intToFloat(f128, self.cast(Payload.Int_u64).?.int),
+            .int_u64 => @intToFloat(T, self.cast(Payload.Int_u64).?.int),
             // .int_i64 => @intToFloat(f128, self.cast(Payload.Int_i64).?.int),
             .int_i64 => @panic("TODO lld: error: undefined symbol: __floatditf"),
 
@@ -773,6 +773,50 @@ pub const Value = extern union {
         }
     }
 
+    /// Converts an integer or a float to a float.
+    /// Returns `error.Overflow` if the value does not fit in the new type.
+    pub fn floatCast(self: Value, allocator: *Allocator, ty: Type, target: Target) !Value {
+        const dest_bit_count = switch (ty.tag()) {
+            .comptime_float => 128,
+            else => ty.floatBits(target),
+        };
+        switch (dest_bit_count) {
+            16, 32, 64, 128 => {},
+            else => std.debug.panic("TODO float cast bit count {}\n", .{dest_bit_count}),
+        }
+        if (ty.isInt()) {
+            @panic("TODO int to float");
+        }
+
+        switch (dest_bit_count) {
+            16 => {
+                @panic("TODO soft float");
+                // var res_payload = Value.Payload.Float_16{.val = self.toFloat(f16)};
+                // if (!self.eql(Value.initPayload(&res_payload.base)))
+                //     return error.Overflow;
+                // return Value.initPayload(&res_payload.base).copy(allocator);
+            },
+            32 => {
+                var res_payload = Value.Payload.Float_32{.val = self.toFloat(f32)};
+                if (!self.eql(Value.initPayload(&res_payload.base)))
+                    return error.Overflow;
+                return Value.initPayload(&res_payload.base).copy(allocator);
+            },
+            64 => {
+                var res_payload = Value.Payload.Float_64{.val = self.toFloat(f64)};
+                if (!self.eql(Value.initPayload(&res_payload.base)))
+                    return error.Overflow;
+                return Value.initPayload(&res_payload.base).copy(allocator);
+            },
+            128 => {
+                const float_payload = try allocator.create(Value.Payload.Float_128);
+                float_payload.* = .{ .val = self.toFloat(f128) };
+                return Value.initPayload(&float_payload.base);
+            },
+            else => unreachable,
+        }
+    }
+
     /// Asserts the value is a float
     pub fn floatHasFraction(self: Value) bool {
         return switch (self.tag()) {
@@ -919,7 +963,7 @@ pub const Value = extern union {
     /// Asserts the value is comparable.
     pub fn order(lhs: Value, rhs: Value) std.math.Order {
         const lhs_tag = lhs.tag();
-        const rhs_tag = lhs.tag();
+        const rhs_tag = rhs.tag();
         const lhs_is_zero = lhs_tag == .zero or lhs_tag == .the_one_possible_value;
         const rhs_is_zero = rhs_tag == .zero or rhs_tag == .the_one_possible_value;
         if (lhs_is_zero) return rhs.orderAgainstZero().invert();
@@ -939,8 +983,8 @@ pub const Value = extern union {
             }
         }
         if (lhs_float or rhs_float) {
-            const lhs_f128 = lhs.toF128();
-            const rhs_f128 = rhs.toF128();
+            const lhs_f128 = lhs.toFloat(f128);
+            const rhs_f128 = rhs.toFloat(f128);
             return std.math.order(lhs_f128, rhs_f128);
         }