identities.rs 2.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123
  1. use std::ops::{Add, Mul};
  2. use std::num::Wrapping;
  3. /// Defines an additive identity element for `Self`.
  4. pub trait Zero: Sized + Add<Self, Output = Self> {
  5. /// Returns the additive identity element of `Self`, `0`.
  6. ///
  7. /// # Laws
  8. ///
  9. /// ```{.text}
  10. /// a + 0 = a ∀ a ∈ Self
  11. /// 0 + a = a ∀ a ∈ Self
  12. /// ```
  13. ///
  14. /// # Purity
  15. ///
  16. /// This function should return the same result at all times regardless of
  17. /// external mutable state, for example values stored in TLS or in
  18. /// `static mut`s.
  19. // FIXME (#5527): This should be an associated constant
  20. fn zero() -> Self;
  21. /// Returns `true` if `self` is equal to the additive identity.
  22. #[inline]
  23. fn is_zero(&self) -> bool;
  24. }
  25. macro_rules! zero_impl {
  26. ($t:ty, $v:expr) => {
  27. impl Zero for $t {
  28. #[inline]
  29. fn zero() -> $t { $v }
  30. #[inline]
  31. fn is_zero(&self) -> bool { *self == $v }
  32. }
  33. }
  34. }
  35. zero_impl!(usize, 0usize);
  36. zero_impl!(u8, 0u8);
  37. zero_impl!(u16, 0u16);
  38. zero_impl!(u32, 0u32);
  39. zero_impl!(u64, 0u64);
  40. zero_impl!(isize, 0isize);
  41. zero_impl!(i8, 0i8);
  42. zero_impl!(i16, 0i16);
  43. zero_impl!(i32, 0i32);
  44. zero_impl!(i64, 0i64);
  45. zero_impl!(f32, 0.0f32);
  46. zero_impl!(f64, 0.0f64);
  47. impl<T: Zero> Zero for Wrapping<T> where Wrapping<T>: Add<Output=Wrapping<T>> {
  48. #[inline]
  49. fn is_zero(&self) -> bool {
  50. self.0.is_zero()
  51. }
  52. #[inline]
  53. fn zero() -> Self {
  54. Wrapping(T::zero())
  55. }
  56. }
  57. /// Defines a multiplicative identity element for `Self`.
  58. pub trait One: Sized + Mul<Self, Output = Self> {
  59. /// Returns the multiplicative identity element of `Self`, `1`.
  60. ///
  61. /// # Laws
  62. ///
  63. /// ```{.text}
  64. /// a * 1 = a ∀ a ∈ Self
  65. /// 1 * a = a ∀ a ∈ Self
  66. /// ```
  67. ///
  68. /// # Purity
  69. ///
  70. /// This function should return the same result at all times regardless of
  71. /// external mutable state, for example values stored in TLS or in
  72. /// `static mut`s.
  73. // FIXME (#5527): This should be an associated constant
  74. fn one() -> Self;
  75. }
  76. macro_rules! one_impl {
  77. ($t:ty, $v:expr) => {
  78. impl One for $t {
  79. #[inline]
  80. fn one() -> $t { $v }
  81. }
  82. }
  83. }
  84. one_impl!(usize, 1usize);
  85. one_impl!(u8, 1u8);
  86. one_impl!(u16, 1u16);
  87. one_impl!(u32, 1u32);
  88. one_impl!(u64, 1u64);
  89. one_impl!(isize, 1isize);
  90. one_impl!(i8, 1i8);
  91. one_impl!(i16, 1i16);
  92. one_impl!(i32, 1i32);
  93. one_impl!(i64, 1i64);
  94. one_impl!(f32, 1.0f32);
  95. one_impl!(f64, 1.0f64);
  96. impl<T: One> One for Wrapping<T> where Wrapping<T>: Mul<Output=Wrapping<T>> {
  97. #[inline]
  98. fn one() -> Self {
  99. Wrapping(T::one())
  100. }
  101. }
  102. // Some helper functions provided for backwards compatibility.
  103. /// Returns the additive identity, `0`.
  104. #[inline(always)] pub fn zero<T: Zero>() -> T { Zero::zero() }
  105. /// Returns the multiplicative identity, `1`.
  106. #[inline(always)] pub fn one<T: One>() -> T { One::one() }