Skip to content

Commit 789d168

Browse files
authored
Rollup merge of #91008 - Urgau:float-minimum-maximum, r=scottmcm
Adds IEEE 754-2019 minimun and maximum functions for f32/f64 IEEE 754-2019 removed the `minNum` (`min` in Rust) and `maxNum` (`max` in Rust) operations in favor of the newly created `minimum` and `maximum` operations due to their [non-associativity](https://grouper.ieee.org/groups/msc/ANSI_IEEE-Std-754-2019/background/minNum_maxNum_Removal_Demotion_v3.pdf) that cannot be fix in a backwards compatible manner. This PR adds `fN::{minimun,maximum}` functions following the new rules. ### IEEE 754-2019 Rules > **minimum(x, y)** is x if x < y, y if y < x, and a quiet NaN if either operand is a NaN, according to 6.2. For this operation, −0 compares less than +0. Otherwise (i.e., when x = y and signs are the same) it is either x or y. > **maximum(x, y)** is x if x > y, y if y > x, and a quiet NaN if either operand is a NaN, according to 6.2. For this operation, +0 compares greater than −0. Otherwise (i.e., when x = y and signs are the same) it is either x or y. "IEEE Standard for Floating-Point Arithmetic," in IEEE Std 754-2019 (Revision of IEEE 754-2008) , vol., no., pp.1-84, 22 July 2019, doi: 10.1109/IEEESTD.2019.8766229. ### Implementation This implementation is inspired by the one in [`glibc` ](https://github.com/bminor/glibc/blob/90f0ac10a74b2d43b5a65aab4be40565e359be43/math/s_fminimum_template.c) (it self derived from the C2X draft) expect that: - it doesn't use `copysign` because it's not available in `core` and also because `copysign` is unnecessary (we only want to check the sign, no need to create a new float) - it also prefer `other > self` instead of `self < other` like IEEE 754-2019 does I originally tried to implement them [using intrinsics](Urgau@1d8aa13) but LLVM [error out](https://godbolt.org/z/7sMrxW49a) when trying to lower them to machine intructions, GCC doesn't yet have built-ins for them, only cranelift support them nativelly (as it doesn't support the nativelly the old sementics). Helps with #83984
2 parents 02913c0 + e2ec3b1 commit 789d168

File tree

6 files changed

+211
-0
lines changed

6 files changed

+211
-0
lines changed

library/core/src/num/f32.rs

+68
Original file line numberDiff line numberDiff line change
@@ -673,6 +673,9 @@ impl f32 {
673673

674674
/// Returns the maximum of the two numbers.
675675
///
676+
/// Follows the IEEE-754 2008 semantics for maxNum, except for handling of signaling NaNs.
677+
/// This matches the behavior of libm’s fmin.
678+
///
676679
/// ```
677680
/// let x = 1.0f32;
678681
/// let y = 2.0f32;
@@ -689,6 +692,9 @@ impl f32 {
689692

690693
/// Returns the minimum of the two numbers.
691694
///
695+
/// Follows the IEEE-754 2008 semantics for minNum, except for handling of signaling NaNs.
696+
/// This matches the behavior of libm’s fmin.
697+
///
692698
/// ```
693699
/// let x = 1.0f32;
694700
/// let y = 2.0f32;
@@ -703,6 +709,68 @@ impl f32 {
703709
intrinsics::minnumf32(self, other)
704710
}
705711

712+
/// Returns the maximum of the two numbers, propagating NaNs.
713+
///
714+
/// This returns NaN when *either* argument is NaN, as opposed to
715+
/// [`f32::max`] which only returns NaN when *both* arguments are NaN.
716+
///
717+
/// ```
718+
/// #![feature(float_minimum_maximum)]
719+
/// let x = 1.0f32;
720+
/// let y = 2.0f32;
721+
///
722+
/// assert_eq!(x.maximum(y), y);
723+
/// assert!(x.maximum(f32::NAN).is_nan());
724+
/// ```
725+
///
726+
/// If one of the arguments is NaN, then NaN is returned. Otherwise this returns the greater
727+
/// of the two numbers. For this operation, -0.0 is considered to be less than +0.0.
728+
/// Note that this follows the semantics specified in IEEE 754-2019.
729+
#[unstable(feature = "float_minimum_maximum", issue = "91079")]
730+
#[inline]
731+
pub fn maximum(self, other: f32) -> f32 {
732+
if self > other {
733+
self
734+
} else if other > self {
735+
other
736+
} else if self == other {
737+
if self.is_sign_positive() && other.is_sign_negative() { self } else { other }
738+
} else {
739+
self + other
740+
}
741+
}
742+
743+
/// Returns the minimum of the two numbers, propagating NaNs.
744+
///
745+
/// This returns NaN when *either* argument is NaN, as opposed to
746+
/// [`f32::min`] which only returns NaN when *both* arguments are NaN.
747+
///
748+
/// ```
749+
/// #![feature(float_minimum_maximum)]
750+
/// let x = 1.0f32;
751+
/// let y = 2.0f32;
752+
///
753+
/// assert_eq!(x.minimum(y), x);
754+
/// assert!(x.minimum(f32::NAN).is_nan());
755+
/// ```
756+
///
757+
/// If one of the arguments is NaN, then NaN is returned. Otherwise this returns the lesser
758+
/// of the two numbers. For this operation, -0.0 is considered to be less than +0.0.
759+
/// Note that this follows the semantics specified in IEEE 754-2019.
760+
#[unstable(feature = "float_minimum_maximum", issue = "91079")]
761+
#[inline]
762+
pub fn minimum(self, other: f32) -> f32 {
763+
if self < other {
764+
self
765+
} else if other < self {
766+
other
767+
} else if self == other {
768+
if self.is_sign_negative() && other.is_sign_positive() { self } else { other }
769+
} else {
770+
self + other
771+
}
772+
}
773+
706774
/// Rounds toward zero and converts to any primitive integer type,
707775
/// assuming that the value is finite and fits in that type.
708776
///

library/core/src/num/f64.rs

+68
Original file line numberDiff line numberDiff line change
@@ -689,6 +689,9 @@ impl f64 {
689689

690690
/// Returns the maximum of the two numbers.
691691
///
692+
/// Follows the IEEE-754 2008 semantics for maxNum, except for handling of signaling NaNs.
693+
/// This matches the behavior of libm’s fmin.
694+
///
692695
/// ```
693696
/// let x = 1.0_f64;
694697
/// let y = 2.0_f64;
@@ -705,6 +708,9 @@ impl f64 {
705708

706709
/// Returns the minimum of the two numbers.
707710
///
711+
/// Follows the IEEE-754 2008 semantics for minNum, except for handling of signaling NaNs.
712+
/// This matches the behavior of libm’s fmin.
713+
///
708714
/// ```
709715
/// let x = 1.0_f64;
710716
/// let y = 2.0_f64;
@@ -719,6 +725,68 @@ impl f64 {
719725
intrinsics::minnumf64(self, other)
720726
}
721727

728+
/// Returns the maximum of the two numbers, propagating NaNs.
729+
///
730+
/// This returns NaN when *either* argument is NaN, as opposed to
731+
/// [`f64::max`] which only returns NaN when *both* arguments are NaN.
732+
///
733+
/// ```
734+
/// #![feature(float_minimum_maximum)]
735+
/// let x = 1.0_f64;
736+
/// let y = 2.0_f64;
737+
///
738+
/// assert_eq!(x.maximum(y), y);
739+
/// assert!(x.maximum(f64::NAN).is_nan());
740+
/// ```
741+
///
742+
/// If one of the arguments is NaN, then NaN is returned. Otherwise this returns the greater
743+
/// of the two numbers. For this operation, -0.0 is considered to be less than +0.0.
744+
/// Note that this follows the semantics specified in IEEE 754-2019.
745+
#[unstable(feature = "float_minimum_maximum", issue = "91079")]
746+
#[inline]
747+
pub fn maximum(self, other: f64) -> f64 {
748+
if self > other {
749+
self
750+
} else if other > self {
751+
other
752+
} else if self == other {
753+
if self.is_sign_positive() && other.is_sign_negative() { self } else { other }
754+
} else {
755+
self + other
756+
}
757+
}
758+
759+
/// Returns the minimum of the two numbers, propagating NaNs.
760+
///
761+
/// This returns NaN when *either* argument is NaN, as opposed to
762+
/// [`f64::min`] which only returns NaN when *both* arguments are NaN.
763+
///
764+
/// ```
765+
/// #![feature(float_minimum_maximum)]
766+
/// let x = 1.0_f64;
767+
/// let y = 2.0_f64;
768+
///
769+
/// assert_eq!(x.minimum(y), x);
770+
/// assert!(x.minimum(f64::NAN).is_nan());
771+
/// ```
772+
///
773+
/// If one of the arguments is NaN, then NaN is returned. Otherwise this returns the lesser
774+
/// of the two numbers. For this operation, -0.0 is considered to be less than +0.0.
775+
/// Note that this follows the semantics specified in IEEE 754-2019.
776+
#[unstable(feature = "float_minimum_maximum", issue = "91079")]
777+
#[inline]
778+
pub fn minimum(self, other: f64) -> f64 {
779+
if self < other {
780+
self
781+
} else if other < self {
782+
other
783+
} else if self == other {
784+
if self.is_sign_negative() && other.is_sign_positive() { self } else { other }
785+
} else {
786+
self + other
787+
}
788+
}
789+
722790
/// Rounds toward zero and converts to any primitive integer type,
723791
/// assuming that the value is finite and fits in that type.
724792
///

library/core/tests/lib.rs

+1
Original file line numberDiff line numberDiff line change
@@ -27,6 +27,7 @@
2727
#![feature(extern_types)]
2828
#![feature(flt2dec)]
2929
#![feature(fmt_internals)]
30+
#![feature(float_minimum_maximum)]
3031
#![feature(array_from_fn)]
3132
#![feature(hashmap_internals)]
3233
#![feature(try_find)]

library/core/tests/num/mod.rs

+61
Original file line numberDiff line numberDiff line change
@@ -715,6 +715,67 @@ macro_rules! test_float {
715715
assert!(($nan as $fty).max($nan).is_nan());
716716
}
717717
#[test]
718+
fn minimum() {
719+
assert_eq!((0.0 as $fty).minimum(0.0), 0.0);
720+
assert!((0.0 as $fty).minimum(0.0).is_sign_positive());
721+
assert_eq!((-0.0 as $fty).minimum(0.0), -0.0);
722+
assert!((-0.0 as $fty).minimum(0.0).is_sign_negative());
723+
assert_eq!((-0.0 as $fty).minimum(-0.0), -0.0);
724+
assert!((-0.0 as $fty).minimum(-0.0).is_sign_negative());
725+
assert_eq!((9.0 as $fty).minimum(9.0), 9.0);
726+
assert_eq!((-9.0 as $fty).minimum(0.0), -9.0);
727+
assert_eq!((0.0 as $fty).minimum(9.0), 0.0);
728+
assert!((0.0 as $fty).minimum(9.0).is_sign_positive());
729+
assert_eq!((-0.0 as $fty).minimum(9.0), -0.0);
730+
assert!((-0.0 as $fty).minimum(9.0).is_sign_negative());
731+
assert_eq!((-0.0 as $fty).minimum(-9.0), -9.0);
732+
assert_eq!(($inf as $fty).minimum(9.0), 9.0);
733+
assert_eq!((9.0 as $fty).minimum($inf), 9.0);
734+
assert_eq!(($inf as $fty).minimum(-9.0), -9.0);
735+
assert_eq!((-9.0 as $fty).minimum($inf), -9.0);
736+
assert_eq!(($neginf as $fty).minimum(9.0), $neginf);
737+
assert_eq!((9.0 as $fty).minimum($neginf), $neginf);
738+
assert_eq!(($neginf as $fty).minimum(-9.0), $neginf);
739+
assert_eq!((-9.0 as $fty).minimum($neginf), $neginf);
740+
assert!(($nan as $fty).minimum(9.0).is_nan());
741+
assert!(($nan as $fty).minimum(-9.0).is_nan());
742+
assert!((9.0 as $fty).minimum($nan).is_nan());
743+
assert!((-9.0 as $fty).minimum($nan).is_nan());
744+
assert!(($nan as $fty).minimum($nan).is_nan());
745+
}
746+
#[test]
747+
fn maximum() {
748+
assert_eq!((0.0 as $fty).maximum(0.0), 0.0);
749+
assert!((0.0 as $fty).maximum(0.0).is_sign_positive());
750+
assert_eq!((-0.0 as $fty).maximum(0.0), 0.0);
751+
assert!((-0.0 as $fty).maximum(0.0).is_sign_positive());
752+
assert_eq!((-0.0 as $fty).maximum(-0.0), -0.0);
753+
assert!((-0.0 as $fty).maximum(-0.0).is_sign_negative());
754+
assert_eq!((9.0 as $fty).maximum(9.0), 9.0);
755+
assert_eq!((-9.0 as $fty).maximum(0.0), 0.0);
756+
assert!((-9.0 as $fty).maximum(0.0).is_sign_positive());
757+
assert_eq!((-9.0 as $fty).maximum(-0.0), -0.0);
758+
assert!((-9.0 as $fty).maximum(-0.0).is_sign_negative());
759+
assert_eq!((0.0 as $fty).maximum(9.0), 9.0);
760+
assert_eq!((0.0 as $fty).maximum(-9.0), 0.0);
761+
assert!((0.0 as $fty).maximum(-9.0).is_sign_positive());
762+
assert_eq!((-0.0 as $fty).maximum(-9.0), -0.0);
763+
assert!((-0.0 as $fty).maximum(-9.0).is_sign_negative());
764+
assert_eq!(($inf as $fty).maximum(9.0), $inf);
765+
assert_eq!((9.0 as $fty).maximum($inf), $inf);
766+
assert_eq!(($inf as $fty).maximum(-9.0), $inf);
767+
assert_eq!((-9.0 as $fty).maximum($inf), $inf);
768+
assert_eq!(($neginf as $fty).maximum(9.0), 9.0);
769+
assert_eq!((9.0 as $fty).maximum($neginf), 9.0);
770+
assert_eq!(($neginf as $fty).maximum(-9.0), -9.0);
771+
assert_eq!((-9.0 as $fty).maximum($neginf), -9.0);
772+
assert!(($nan as $fty).maximum(9.0).is_nan());
773+
assert!(($nan as $fty).maximum(-9.0).is_nan());
774+
assert!((9.0 as $fty).maximum($nan).is_nan());
775+
assert!((-9.0 as $fty).maximum($nan).is_nan());
776+
assert!(($nan as $fty).maximum($nan).is_nan());
777+
}
778+
#[test]
718779
fn rem_euclid() {
719780
let a: $fty = 42.0;
720781
assert!($inf.rem_euclid(a).is_nan());

library/std/src/f32/tests.rs

+12
Original file line numberDiff line numberDiff line change
@@ -19,6 +19,18 @@ fn test_max_nan() {
1919
assert_eq!(2.0f32.max(f32::NAN), 2.0);
2020
}
2121

22+
#[test]
23+
fn test_minimum() {
24+
assert!(f32::NAN.minimum(2.0).is_nan());
25+
assert!(2.0f32.minimum(f32::NAN).is_nan());
26+
}
27+
28+
#[test]
29+
fn test_maximum() {
30+
assert!(f32::NAN.maximum(2.0).is_nan());
31+
assert!(2.0f32.maximum(f32::NAN).is_nan());
32+
}
33+
2234
#[test]
2335
fn test_nan() {
2436
let nan: f32 = f32::NAN;

library/std/src/lib.rs

+1
Original file line numberDiff line numberDiff line change
@@ -287,6 +287,7 @@
287287
#![feature(exhaustive_patterns)]
288288
#![feature(extend_one)]
289289
#![feature(fn_traits)]
290+
#![feature(float_minimum_maximum)]
290291
#![feature(format_args_nl)]
291292
#![feature(gen_future)]
292293
#![feature(generator_trait)]

0 commit comments

Comments
 (0)