Skip to content

Commit 73c80c6

Browse files
committed
Add basic library support for f16 and f128
Implement basic operation traits that get lowered to intrinsics. This does not yet create a new module for these types. Additionally, add codegn tests for the implemented operations.
1 parent 3cdcdaf commit 73c80c6

File tree

15 files changed

+342
-10
lines changed

15 files changed

+342
-10
lines changed

library/core/src/clone.rs

+3
Original file line numberDiff line numberDiff line change
@@ -231,6 +231,9 @@ mod impls {
231231
bool char
232232
}
233233

234+
#[cfg(not(bootstrap))]
235+
impl_clone! { f16 f128 }
236+
234237
#[unstable(feature = "never_type", issue = "35121")]
235238
impl Clone for ! {
236239
#[inline]

library/core/src/cmp.rs

+6
Original file line numberDiff line numberDiff line change
@@ -1489,6 +1489,9 @@ mod impls {
14891489
bool char usize u8 u16 u32 u64 u128 isize i8 i16 i32 i64 i128 f32 f64
14901490
}
14911491

1492+
#[cfg(not(bootstrap))]
1493+
partial_eq_impl! { f16 f128 }
1494+
14921495
macro_rules! eq_impl {
14931496
($($t:ty)*) => ($(
14941497
#[stable(feature = "rust1", since = "1.0.0")]
@@ -1541,6 +1544,9 @@ mod impls {
15411544

15421545
partial_ord_impl! { f32 f64 }
15431546

1547+
#[cfg(not(bootstrap))]
1548+
partial_ord_impl! { f16 f128 }
1549+
15441550
macro_rules! ord_impl {
15451551
($($t:ty)*) => ($(
15461552
#[stable(feature = "rust1", since = "1.0.0")]

library/core/src/convert/num.rs

+15
Original file line numberDiff line numberDiff line change
@@ -35,8 +35,12 @@ macro_rules! impl_float_to_int {
3535
}
3636
}
3737

38+
#[cfg(not(bootstrap))]
39+
impl_float_to_int!(f16 => u8, u16, u32, u64, u128, usize, i8, i16, i32, i64, i128, isize);
3840
impl_float_to_int!(f32 => u8, u16, u32, u64, u128, usize, i8, i16, i32, i64, i128, isize);
3941
impl_float_to_int!(f64 => u8, u16, u32, u64, u128, usize, i8, i16, i32, i64, i128, isize);
42+
#[cfg(not(bootstrap))]
43+
impl_float_to_int!(f128 => u8, u16, u32, u64, u128, usize, i8, i16, i32, i64, i128, isize);
4044

4145
// Conversion traits for primitive integer and float types
4246
// Conversions T -> T are covered by a blanket impl and therefore excluded
@@ -164,7 +168,18 @@ impl_from!(u16 => f64, #[stable(feature = "lossless_float_conv", since = "1.6.0"
164168
impl_from!(u32 => f64, #[stable(feature = "lossless_float_conv", since = "1.6.0")]);
165169

166170
// float -> float
171+
// todo: these are insta-stable regardless of the gate, right?
172+
#[cfg(not(bootstrap))]
173+
impl_from!(f16 => f32, #[stable(feature = "lossless_float_conv", since = "1.6.0")]);
174+
#[cfg(not(bootstrap))]
175+
impl_from!(f16 => f64, #[stable(feature = "lossless_float_conv", since = "1.6.0")]);
176+
#[cfg(not(bootstrap))]
177+
impl_from!(f16 => f128, #[stable(feature = "lossless_float_conv", since = "1.6.0")]);
167178
impl_from!(f32 => f64, #[stable(feature = "lossless_float_conv", since = "1.6.0")]);
179+
#[cfg(not(bootstrap))]
180+
impl_from!(f32 => f128, #[stable(feature = "lossless_float_conv", since = "1.6.0")]);
181+
#[cfg(not(bootstrap))]
182+
impl_from!(f64 => f128, #[stable(feature = "lossless_float_conv", since = "1.6.0")]);
168183

169184
macro_rules! impl_float_from_bool {
170185
($float:ty) => {

library/core/src/default.rs

+4
Original file line numberDiff line numberDiff line change
@@ -176,5 +176,9 @@ default_impl! { i32, 0, "Returns the default value of `0`" }
176176
default_impl! { i64, 0, "Returns the default value of `0`" }
177177
default_impl! { i128, 0, "Returns the default value of `0`" }
178178

179+
#[cfg(not(bootstrap))]
180+
default_impl! { f16, 0.0f16, "Returns the default value of `0.0`" }
179181
default_impl! { f32, 0.0f32, "Returns the default value of `0.0`" }
180182
default_impl! { f64, 0.0f64, "Returns the default value of `0.0`" }
183+
#[cfg(not(bootstrap))]
184+
default_impl! { f128, 0.0f128, "Returns the default value of `0.0`" }

library/core/src/fmt/nofloat.rs

+4
Original file line numberDiff line numberDiff line change
@@ -11,5 +11,9 @@ macro_rules! floating {
1111
};
1212
}
1313

14+
#[cfg(not(bootstrap))]
15+
floating! { f16 }
1416
floating! { f32 }
1517
floating! { f64 }
18+
#[cfg(not(bootstrap))]
19+
floating! { f128 }

library/core/src/lib.rs

+2
Original file line numberDiff line numberDiff line change
@@ -206,6 +206,8 @@
206206
#![cfg_attr(bootstrap, feature(diagnostic_namespace))]
207207
#![cfg_attr(bootstrap, feature(exhaustive_patterns))]
208208
#![cfg_attr(bootstrap, feature(platform_intrinsics))]
209+
#![cfg_attr(not(bootstrap), feature(f128))]
210+
#![cfg_attr(not(bootstrap), feature(f16))]
209211
#![cfg_attr(not(bootstrap), feature(freeze_impls))]
210212
#![cfg_attr(not(bootstrap), feature(min_exhaustive_patterns))]
211213
#![feature(abi_unadjusted)]

library/core/src/marker.rs

+7
Original file line numberDiff line numberDiff line change
@@ -429,6 +429,13 @@ marker_impls! {
429429

430430
}
431431

432+
// todo: which feature gate should we use?
433+
#[cfg(not(bootstrap))]
434+
marker_impls! {
435+
#[stable(feature = "rust1", since = "1.0.0")]
436+
Copy for f16, f128
437+
}
438+
432439
#[unstable(feature = "never_type", issue = "35121")]
433440
impl Copy for ! {}
434441

library/core/src/ops/arith.rs

+33
Original file line numberDiff line numberDiff line change
@@ -111,6 +111,9 @@ macro_rules! add_impl {
111111

112112
add_impl! { usize u8 u16 u32 u64 u128 isize i8 i16 i32 i64 i128 f32 f64 }
113113

114+
#[cfg(not(bootstrap))]
115+
add_impl! { f16 f128 }
116+
114117
/// The subtraction operator `-`.
115118
///
116119
/// Note that `Rhs` is `Self` by default, but this is not mandatory. For
@@ -220,6 +223,9 @@ macro_rules! sub_impl {
220223

221224
sub_impl! { usize u8 u16 u32 u64 u128 isize i8 i16 i32 i64 i128 f32 f64 }
222225

226+
#[cfg(not(bootstrap))]
227+
sub_impl! { f16 f128 }
228+
223229
/// The multiplication operator `*`.
224230
///
225231
/// Note that `Rhs` is `Self` by default, but this is not mandatory.
@@ -350,6 +356,9 @@ macro_rules! mul_impl {
350356

351357
mul_impl! { usize u8 u16 u32 u64 u128 isize i8 i16 i32 i64 i128 f32 f64 }
352358

359+
#[cfg(not(bootstrap))]
360+
mul_impl! { f16 f128 }
361+
353362
/// The division operator `/`.
354363
///
355364
/// Note that `Rhs` is `Self` by default, but this is not mandatory.
@@ -508,6 +517,9 @@ macro_rules! div_impl_float {
508517

509518
div_impl_float! { f32 f64 }
510519

520+
#[cfg(not(bootstrap))]
521+
div_impl_float! { f16 f128 }
522+
511523
/// The remainder operator `%`.
512524
///
513525
/// Note that `Rhs` is `Self` by default, but this is not mandatory.
@@ -625,6 +637,9 @@ macro_rules! rem_impl_float {
625637

626638
rem_impl_float! { f32 f64 }
627639

640+
#[cfg(not(bootstrap))]
641+
rem_impl_float! { f16 f128 }
642+
628643
/// The unary negation operator `-`.
629644
///
630645
/// # Examples
@@ -700,6 +715,9 @@ macro_rules! neg_impl {
700715

701716
neg_impl! { isize i8 i16 i32 i64 i128 f32 f64 }
702717

718+
#[cfg(not(bootstrap))]
719+
neg_impl! { f16 f128 }
720+
703721
/// The addition assignment operator `+=`.
704722
///
705723
/// # Examples
@@ -767,6 +785,9 @@ macro_rules! add_assign_impl {
767785

768786
add_assign_impl! { usize u8 u16 u32 u64 u128 isize i8 i16 i32 i64 i128 f32 f64 }
769787

788+
#[cfg(not(bootstrap))]
789+
add_assign_impl! { f16 f128 }
790+
770791
/// The subtraction assignment operator `-=`.
771792
///
772793
/// # Examples
@@ -834,6 +855,9 @@ macro_rules! sub_assign_impl {
834855

835856
sub_assign_impl! { usize u8 u16 u32 u64 u128 isize i8 i16 i32 i64 i128 f32 f64 }
836857

858+
#[cfg(not(bootstrap))]
859+
sub_assign_impl! { f16 f128 }
860+
837861
/// The multiplication assignment operator `*=`.
838862
///
839863
/// # Examples
@@ -892,6 +916,9 @@ macro_rules! mul_assign_impl {
892916

893917
mul_assign_impl! { usize u8 u16 u32 u64 u128 isize i8 i16 i32 i64 i128 f32 f64 }
894918

919+
#[cfg(not(bootstrap))]
920+
mul_assign_impl! { f16 f128 }
921+
895922
/// The division assignment operator `/=`.
896923
///
897924
/// # Examples
@@ -949,6 +976,9 @@ macro_rules! div_assign_impl {
949976

950977
div_assign_impl! { usize u8 u16 u32 u64 u128 isize i8 i16 i32 i64 i128 f32 f64 }
951978

979+
#[cfg(not(bootstrap))]
980+
div_assign_impl! { f16 f128 }
981+
952982
/// The remainder assignment operator `%=`.
953983
///
954984
/// # Examples
@@ -1009,3 +1039,6 @@ macro_rules! rem_assign_impl {
10091039
}
10101040

10111041
rem_assign_impl! { usize u8 u16 u32 u64 u128 isize i8 i16 i32 i64 i128 f32 f64 }
1042+
1043+
#[cfg(not(bootstrap))]
1044+
rem_assign_impl! { f16 f128 }

tests/codegen/float/f128.rs

+129
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,129 @@
1+
// Verify that our intrinsics generate the correct LLVM calls for f128
2+
3+
#![crate_type = "lib"]
4+
#![feature(f128)]
5+
#![feature(core_intrinsics)]
6+
7+
// CHECK-LABEL: i1 @f128_eq(
8+
#[no_mangle]
9+
pub fn f128_eq(a: f128, b: f128) -> bool {
10+
// CHECK: fcmp oeq fp128 %{{.+}}, %{{.+}}
11+
a == b
12+
}
13+
14+
// CHECK-LABEL: i1 @f128_ne(
15+
#[no_mangle]
16+
pub fn f128_ne(a: f128, b: f128) -> bool {
17+
// CHECK: fcmp une fp128 %{{.+}}, %{{.+}}
18+
a != b
19+
}
20+
21+
// CHECK-LABEL: i1 @f128_gt(
22+
#[no_mangle]
23+
pub fn f128_gt(a: f128, b: f128) -> bool {
24+
// CHECK: fcmp ogt fp128 %{{.+}}, %{{.+}}
25+
a > b
26+
}
27+
28+
// CHECK-LABEL: i1 @f128_ge(
29+
#[no_mangle]
30+
pub fn f128_ge(a: f128, b: f128) -> bool {
31+
// CHECK: fcmp oge fp128 %{{.+}}, %{{.+}}
32+
a >= b
33+
}
34+
35+
// CHECK-LABEL: i1 @f128_lt(
36+
#[no_mangle]
37+
pub fn f128_lt(a: f128, b: f128) -> bool {
38+
// CHECK: fcmp olt fp128 %{{.+}}, %{{.+}}
39+
a < b
40+
}
41+
42+
// CHECK-LABEL: i1 @f128_le(
43+
#[no_mangle]
44+
pub fn f128_le(a: f128, b: f128) -> bool {
45+
// CHECK: fcmp ole fp128 %{{.+}}, %{{.+}}
46+
a <= b
47+
}
48+
49+
// CHECK-LABEL: fp128 @f128_neg(
50+
#[no_mangle]
51+
pub fn f128_neg(a: f128) -> f128 {
52+
// CHECK: fneg fp128
53+
-a
54+
}
55+
56+
// CHECK-LABEL: fp128 @f128_add(
57+
#[no_mangle]
58+
pub fn f128_add(a: f128, b: f128) -> f128 {
59+
// CHECK: fadd fp128 %{{.+}}, %{{.+}}
60+
a + b
61+
}
62+
63+
// CHECK-LABEL: fp128 @f128_sub(
64+
#[no_mangle]
65+
pub fn f128_sub(a: f128, b: f128) -> f128 {
66+
// CHECK: fsub fp128 %{{.+}}, %{{.+}}
67+
a - b
68+
}
69+
70+
// CHECK-LABEL: fp128 @f128_mul(
71+
#[no_mangle]
72+
pub fn f128_mul(a: f128, b: f128) -> f128 {
73+
// CHECK: fmul fp128 %{{.+}}, %{{.+}}
74+
a * b
75+
}
76+
77+
// CHECK-LABEL: fp128 @f128_div(
78+
#[no_mangle]
79+
pub fn f128_div(a: f128, b: f128) -> f128 {
80+
// CHECK: fdiv fp128 %{{.+}}, %{{.+}}
81+
a / b
82+
}
83+
84+
// CHECK-LABEL: fp128 @f128_rem(
85+
#[no_mangle]
86+
pub fn f128_rem(a: f128, b: f128) -> f128 {
87+
// CHECK: frem fp128 %{{.+}}, %{{.+}}
88+
a % b
89+
}
90+
91+
// CHECK-LABEL: void @f128_add_assign(
92+
#[no_mangle]
93+
pub fn f128_add_assign(a: &mut f128, b: f128) {
94+
// CHECK: fadd fp128 %{{.+}}, %{{.+}}
95+
// CHECK-NEXT: store fp128 %{{.+}}, ptr %{{.+}}
96+
*a += b;
97+
}
98+
99+
// CHECK-LABEL: void @f128_sub_assign(
100+
#[no_mangle]
101+
pub fn f128_sub_assign(a: &mut f128, b: f128) {
102+
// CHECK: fsub fp128 %{{.+}}, %{{.+}}
103+
// CHECK-NEXT: store fp128 %{{.+}}, ptr %{{.+}}
104+
*a -= b;
105+
}
106+
107+
// CHECK-LABEL: void @f128_mul_assign(
108+
#[no_mangle]
109+
pub fn f128_mul_assign(a: &mut f128, b: f128) {
110+
// CHECK: fmul fp128 %{{.+}}, %{{.+}}
111+
// CHECK-NEXT: store fp128 %{{.+}}, ptr %{{.+}}
112+
*a *= b
113+
}
114+
115+
// CHECK-LABEL: void @f128_div_assign(
116+
#[no_mangle]
117+
pub fn f128_div_assign(a: &mut f128, b: f128) {
118+
// CHECK: fdiv fp128 %{{.+}}, %{{.+}}
119+
// CHECK-NEXT: store fp128 %{{.+}}, ptr %{{.+}}
120+
*a /= b
121+
}
122+
123+
// CHECK-LABEL: void @f128_rem_assign(
124+
#[no_mangle]
125+
pub fn f128_rem_assign(a: &mut f128, b: f128) {
126+
// CHECK: frem fp128 %{{.+}}, %{{.+}}
127+
// CHECK-NEXT: store fp128 %{{.+}}, ptr %{{.+}}
128+
*a %= b
129+
}

0 commit comments

Comments
 (0)